1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
|
Upload processing queue
=======================
The upload processing queue (PackageUpload and friends) is where
uploads go after they have been checked by process-upload.py and
before they get published by publish-distro.py.
Upload system stores syntactically corrects (see
nascentupload.txt) uploads driven by a submitted changesfile.
One PackageUpload ~ One upload ~ One changesfile.
An upload can contain a combination of content types, organized by:
* Source (PackageUploadSource):
a SourcePackageRelease ([tar.gz +] diff)
* Build (PackageUploadBuild):
a Build result, one or more BinaryPackageReleases resulted from a
successfully build (deb).
* Custom (PackageUploadCustom):
a special file which will be processed in a specific way to publish
its contents in the archive (tar.gz). Currently we support
translations, installer and upgrader (see
distroseriesqueue-{translation, dist-upgrader}).
Each of those instances points back to a PackageUpload entry
(parent) and to its type target:
* PackageUploadSource -> SourcePackageRelease (SPR),
* PackageUploadBuild -> Build,
* PackageUploadCustom -> LibraryFileAlias (LFA).
The combination is assured by the upload policy used (see
nascentupload.txt), some of them allow source + binaries, other only
binaries + custom, other only source.
First up, we need to actually process an upload to get it into the
queue. To do this we prepare an OpenPGP key, and then run the upload handler.
>>> from lp.testing.gpgkeys import import_public_test_keys
>>> import_public_test_keys()
We need some setup for the upload handler.
>>> from lp.archiveuploader.nascentupload import NascentUpload
>>> from lp.archiveuploader.tests import datadir, getPolicy
Since we landed correct security adapters for PackageUpload,
we need to perform further actions logged in as an admins, which have
launchpad.Edit on the records:
>>> from lp.testing import login
>>> login("foo.bar@canonical.com")
>>> from lp.registry.interfaces.distribution import IDistributionSet
>>> from lp.soyuz.interfaces.component import IComponentSet
>>> from lp.soyuz.model.component import ComponentSelection
>>> ubuntu = getUtility(IDistributionSet)['ubuntu']
>>> hoary = ubuntu['hoary']
>>> universe = getUtility(IComponentSet)['universe']
>>> trash = ComponentSelection(distroseries=hoary, component=universe)
Construct an upload.
>>> anything_policy = getPolicy(
... name='absolutely-anything', distro='ubuntu', distroseries='hoary')
>>> from lp.services.log.logger import DevNullLogger
>>> ed_upload = NascentUpload.from_changesfile_path(
... datadir("ed_0.2-20_i386.changes.source-only-unsigned"),
... anything_policy, DevNullLogger())
>>> ed_upload.process()
>>> success = ed_upload.do_accept()
>>> success
True
Now the upload is in the queue, it'll likely be there as NEW because that's
what we expect the ed upload to produce. Let's find the queue item and
convert it to an ACCEPTED item.
>>> from lp.soyuz.enums import PackageUploadStatus
>>> from lp.soyuz.interfaces.queue import QueueInconsistentStateError
>>> new_queue = hoary.getPackageUploads(PackageUploadStatus.NEW)
Use state-machine method provided by PackageUpload to ACCEPT an
upload. If some designed check according the request state do not
pass, the state-machine methods will raise an exception indicating the
upload can not have that state.
XXX cprov 20051209: need to build a broken upload to test it properly
>>> for item in new_queue:
... try:
... item.setAccepted()
... item.syncUpdate()
... except QueueInconsistentStateError, info:
... print info
>>> accepted_queue = hoary.getPackageUploads(PackageUploadStatus.ACCEPTED)
>>> from lp.services.log.logger import FakeLogger
>>> for item in accepted_queue:
... for source in item.sources:
... print source.sourcepackagerelease.name
... pub_records = item.realiseUpload(FakeLogger())
ed
DEBUG Publishing source ed/0.2-20 to ubuntu/hoary in the primary archive
Confirm we can now find ed published in hoary.
>>> from lp.soyuz.enums import PackagePublishingStatus
>>> from lp.soyuz.model.publishing import SourcePackagePublishingHistory
>>> for release in SourcePackagePublishingHistory.selectBy(
... distroseries=hoary, status=PackagePublishingStatus.PENDING):
... if release.sourcepackagerelease.sourcepackagename.name == "ed":
... print release.sourcepackagerelease.version
0.2-20
Check IPackageUploadSet behaviour:
>>> from lp.services.webapp.testing import verifyObject
>>> from lp.soyuz.interfaces.queue import IPackageUploadSet
Grab an utility:
>>> qset = getUtility(IPackageUploadSet)
Check if it implements its interface completely:
>>> verifyObject(IPackageUploadSet, qset)
True
Iterating over IPackageUploads via iPackageUploadSet:
>>> len([item for item in qset])
16
Retrieving an IPackageUpload by its id:
>>> qset[1].id
1
>>> qset.get(1).id
1
Counter, optionally by status (informally named "queue") and or distroseries:
>>> qset.count()
16
>>> qset.count(PackageUploadStatus.DONE)
5
>>> qset.count(PackageUploadStatus.REJECTED)
0
Retrieve some data from DB to play more with counter.
>>> from lp.registry.interfaces.distribution import IDistributionSet
>>> distro = getUtility(IDistributionSet).getByName('ubuntu')
>>> breezy_autotest = distro['breezy-autotest']
>>> qset.count(distroseries=breezy_autotest)
13
>>> qset.count(status=PackageUploadStatus.ACCEPTED,
... distroseries=breezy_autotest)
0
>>> qset.count(status=PackageUploadStatus.DONE,
... distroseries=hoary)
1
Check the behaviour of @cachedproperty attributes:
>>> qitem = qset.get(1)
>>> qitem.date_created
datetime.datetime(...)
>>> print qitem.changesfile.filename
mozilla-firefox_0.9_i386.changes
>>> print qitem.sourcepackagerelease.name
mozilla-firefox
>>> print qitem.displayname
mozilla-firefox
>>> print qitem.displayversion
0.9
>>> print qitem.displayarchs
i386
>>> qitem.sourcepackagerelease
<SourcePackageRelease at ...>
Let's check the behaviour of @cachedproperty attributes in a custom upload:
>>> custom_item = qset.get(5)
>>> custom_item.date_created
datetime.datetime(...)
>>> print custom_item.changesfile.filename
netapplet-1.0.0.tar.gz
>>> print custom_item.displayname
netapplet-1.0.0.tar.gz
>>> print custom_item.displayversion
-
>>> print custom_item.displayarchs
raw-translations
>>> print custom_item.sourcepackagerelease
None
The method getBuildByBuildIDs() will return all the PackageUploadBuild
records that match the supplied build IDs.
>>> ids = (18,19)
>>> for package_upload_build in qset.getBuildByBuildIDs(ids):
... print package_upload_build.packageupload.displayname
mozilla-firefox
pmount
If the supplied IDs is empty or None, an empty list is returned:
>>> qset.getBuildByBuildIDs([])
[]
>>> qset.getBuildByBuildIDs(None)
[]
Upload Signing Key
------------------
IPackageUpload.signing_key should store the IGPGKey reference to
the key used to sign the changesfile when it applies (insecure policy
uploads).
It's mainly used to identify sponsored uploads, when someone with
rights to upload to ubuntu (mostly MOTU) signed over package changes
done by someone else.
Let's process a new upload:
>>> insecure_policy = getPolicy(
... name='insecure', distro='ubuntu', distroseries='hoary')
>>> bar_ok = NascentUpload.from_changesfile_path(
... datadir('suite/bar_1.0-1/bar_1.0-1_source.changes'),
... insecure_policy, DevNullLogger())
>>> bar_ok.process()
>>> success = bar_ok.do_accept()
>>> success
True
>>> signed_queue = bar_ok.queue_root
>>> from lp.registry.interfaces.gpg import IGPGKey
>>> from lp.soyuz.interfaces.queue import IPackageUpload
>>> verifyObject(IPackageUpload, signed_queue)
True
>>> verifyObject(IGPGKey, signed_queue.signing_key)
True
Let's check the IPerson entities related to this source upload:
>>> signed_src = signed_queue.sources[0].sourcepackagerelease
>>> signed_src.creator.displayname
u'Daniel Silverstone'
>>> signed_src.maintainer.displayname
u'Launchpad team'
>>> signed_queue.signing_key.owner.displayname
u'Foo Bar'
Based on this information we can conclude that source 'bar' is
maintained by 'Launchpad Team', was modified by 'Daniel Silverstone'
and sponsored by 'Foo Bar'.
IHasQueueItems
--------------
Check State Machine over PackageUploadBuilds:
Performing full acceptance:
>>> items = breezy_autotest.getPackageUploads(PackageUploadStatus.NEW)
>>> for item in items:
... try:
... item.setAccepted()
... except QueueInconsistentStateError, e:
... print item.displayname, e
... else:
... print item.displayname, item.status.name
netapplet-1.0.0.tar.gz ACCEPTED
netapplet-1.0.0.tar.gz ACCEPTED
alsa-utils ACCEPTED
netapplet ACCEPTED
pmount ACCEPTED
mozilla-firefox ACCEPTED
Move the ACCEPTED items back to NEW.
>>> items = breezy_autotest.getPackageUploads(
... PackageUploadStatus.ACCEPTED)
>>> for item in items:
... item.setNew()
... print item.displayname, item.status.name
netapplet-1.0.0.tar.gz NEW
netapplet-1.0.0.tar.gz NEW
alsa-utils NEW
netapplet NEW
pmount NEW
mozilla-firefox NEW
Check if the state machine methods raise when trying to set the same state:
>>> items = breezy_autotest.getPackageUploads(PackageUploadStatus.NEW)
>>> for item in items:
... try:
... item.setNew()
... except QueueInconsistentStateError, e:
... print item.displayname, e
... else:
... print item.displayname, 'BOGUS'
netapplet-1.0.0.tar.gz Queue item already new
netapplet-1.0.0.tar.gz Queue item already new
alsa-utils Queue item already new
netapplet Queue item already new
pmount Queue item already new
mozilla-firefox Queue item already new
Check several available state machine methods on a NEW queue item
(except setAccepted, it's already covered by other tests, check if they
don't raise any exception):
>>> test_qitem = getUtility(IPackageUploadSet)[1]
>>> test_qitem.setUnapproved()
>>> test_qitem.setRejected()
>>> test_qitem.setDone()
>>> test_qitem.setNew()
Check forbidden approval of not selected Section:
>>> from zope.security.proxy import removeSecurityProxy
>>> from lp.soyuz.interfaces.component import IComponentSet
>>> from lp.soyuz.interfaces.section import ISectionSet
Retrieve mozilla-firefox Upload:
>>> item = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'mozilla')[0]
Override the mozilla-firefox component to fresh created 'hell' component.
XXX cprov 20060118: remove proxy magic is required for BPR instances.
>>> naked_bin = removeSecurityProxy(
... item.builds[0].build.binarypackages[0])
>>> naked_bin.component = getUtility(IComponentSet).new('hell')
>>> try:
... item.setAccepted()
... except QueueInconsistentStateError, e:
... print item.displayname, e
... else:
... print item.displayname, 'ACCEPTED'
mozilla-firefox Component "hell" is not allowed in breezy-autotest
Check how we treat source upload duplications in UNAPPROVED queue (NEW
has a similar behaviour):
>>> dups = breezy_autotest.getPackageUploads(
... PackageUploadStatus.UNAPPROVED, name=u'cnews')
>>> dups.count()
2
>>> dup_one, dup_two = list(dups)
>>> dup_one.displayname, dup_one.displayversion
(u'cnews', u'1.0')
>>> dup_two.displayname, dup_two.displayversion
(u'cnews', u'1.0')
The upload admin can not accept both since we check unique
(name, version) accross distribution:
>>> dup_one.setAccepted()
>>> dup_one.status == PackageUploadStatus.ACCEPTED
True
The database modification needs to be realised in the DB, otherwise
the look up code won't be able to identify any duplications:
>>> dup_one.syncUpdate()
As expected the second item acceptance will fail and the item will
remain in the original queue
>>> dup_two.setAccepted()
Traceback (most recent call last):
...
QueueInconsistentStateError: The source cnews - 1.0 is already
accepted in ubuntu/breezy-autotest and you cannot upload the same
version within the same distribution. You have to modify the source
version and re-upload.
>>> dup_two.status.name
'UNAPPROVED'
The only available action will be rejection:
>>> dup_two.setRejected()
>>> dup_one.syncUpdate()
>>> dup_two.status.name
'REJECTED'
Move the second item back to its original queue to perform the same
test after the former accepted item was published (DONE queue)
>>> dup_two.setUnapproved()
>>> dup_two.syncUpdate()
>>> dup_two.status.name
'UNAPPROVED'
>>> dup_one.setDone()
>>> dup_one.status == PackageUploadStatus.DONE
True
>>> dup_one.syncUpdate()
The protection code should also identify dups with items in DONE queue
>>> dup_two.setAccepted()
Traceback (most recent call last):
...
QueueInconsistentStateError: The source cnews - 1.0 is already
accepted in ubuntu/breezy-autotest and you cannot upload the same
version within the same distribution. You have to modify the source
version and re-upload.
The ubuntu policy allows unofficial sections to live sometime in the
repository, until someone find time to override them. It's better than
dropping binary packages that might have consumed a lot of resources
for such a unimportant issue.
Retrieve the 'pmount' NEW queue entry and override it with a
just-created, thus unofficial, section named 'boing'.
>>> item = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'pmount')[0]
>>> pmount_binary = item.builds[0].build.binarypackages[0]
>>> removeSecurityProxy(
... pmount_binary).section = getUtility(ISectionSet).new('boing')
The 'pmount' entry for the unofficial section 'boing', can be
normally accepted.
>>> item.setAccepted()
>>> print item.status.name
ACCEPTED
We will reject the just-accepted entry so it does not mess with the
other tests.
>>> item.setRejected()
Check how the security adapters work for PackageUpload:
>>> pub_qitem = getUtility(IPackageUploadSet)[1]
UNAPPROVED queue items are expected to be private:
>>> priv_qitem = getUtility(IPackageUploadSet)[5]
Anonymous can view all queue items, public and private ones.
>>> login(ANONYMOUS)
>>> pub_qitem.displayname
u'mozilla-firefox'
>>> priv_qitem.displayname
u'netapplet-1.0.0.tar.gz'
A normal user can't act on both of them:
>>> login('no-priv@canonical.com')
>>> pub_qitem.setDone()
Traceback (most recent call last):
...
Unauthorized: (<PackageUpload ..., 'setDone', 'launchpad.Edit')
>>> priv_qitem.setDone()
Traceback (most recent call last):
...
Unauthorized: (<PackageUpload ..., 'setDone', 'launchpad.Edit')
Only the members of distroseries drivers in question have
launchpad.Edit on a packageupload record.
>>> login('foo.bar@canonical.com')
>>> pub_qitem.displayname
u'mozilla-firefox'
>>> priv_qitem.displayname
u'netapplet-1.0.0.tar.gz'
>>> pub_qitem.setDone()
>>> priv_qitem.setDone()
Roll back modified data:
>>> transaction.abort()
Clear existing mail stack:
>>> from lp.testing.mail_helpers import pop_notifications
>>> rubbish = pop_notifications()
As mentioned above, values returned by getPackageUploads matching a given
'name' and 'version' may contain different types of uploads.
Sampledata contains only a i386 binary exactly matching 'pmount 0.1-1'.
>>> from operator import attrgetter
>>> def print_queue_items(queue_items):
... for queue_item in queue_items:
... print "%s %s %s" % (
... queue_item.displayname, queue_item.displayversion,
... queue_item.displayarchs)
>>> queue_items = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'pmount', version=u'0.1-1',
... exact_match=True)
>>> print_queue_items(queue_items)
pmount 0.1-1 i386
We will set up a very peculiar environment to test this aspect of
`getPackageUploads`. First we will include another a non-matching source
version of pmount (0.1-2) into the binary pmount upload we already
have in the sampledata.
>>> [binary_queue] = queue_items
>>> pmount = ubuntu.getSourcePackage('pmount')
>>> non_matching_pmount = pmount.getVersion('0.1-2')
>>> unused = binary_queue.addSource(
... non_matching_pmount.sourcepackagerelease)
'pmount 0.1-1' binary upload continues to be returned when we query
the queue for 'pmount 0.1-1', via the existing binary path.
>>> queue_items = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'pmount', version=u'0.1-1',
... exact_match=True)
>>> print_queue_items(queue_items)
pmount 0.1-1 i386
Also, when we can create a source 'pmount 0.1-1' upload in the
breezy-autotest context. It also becomes part of the lookup results.
>>> from lp.registry.interfaces.pocket import PackagePublishingPocket
>>> candidate_queue = breezy_autotest.createQueueEntry(
... PackagePublishingPocket.RELEASE,
... breezy_autotest.main_archive,
... 'pmount_0.1-1_source.changes', 'some content')
>>> matching_pmount = pmount.getVersion('0.1-1')
>>> unused = candidate_queue.addSource(
... matching_pmount.sourcepackagerelease)
>>> queue_items = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'pmount', version=u'0.1-1',
... exact_match=True)
>>> print_queue_items(queue_items)
pmount 0.1-1 source
pmount 0.1-1 i386
It means that call sites querying the upload queue should be aware of
this aspect and filter the results appropriately.
One special call site is `PackageUploadSource.verifyBeforeAccepted`.
It should allow the acceptance of 'pmount 0.1-1' source, even if there
is a 'pmount 0.1-1' binary upload already accepted in its context.
(see bug #280700 for more information about this policy decision)
# XXX StuartBishop 20100311 bug=537335: Need to order results here.
>>> queue_items = sorted(list(queue_items),
... key=attrgetter('displayarchs'))
>>> [binary_item, source_item] = queue_items
>>> binary_item.setAccepted()
>>> queue_items = breezy_autotest.getPackageUploads(
... PackageUploadStatus.ACCEPTED, name=u'pmount',
... version=u'0.1-1', exact_match=True)
>>> print_queue_items(queue_items)
pmount 0.1-1 i386
Binary accepted, let's accept the source.
>>> source_item.setAccepted()
Both uploads are waiting to be published.
>>> queue_items = breezy_autotest.getPackageUploads(
... PackageUploadStatus.ACCEPTED, name=u'pmount',
... version='0.1-1', exact_match=True)
>>> print_queue_items(queue_items)
pmount 0.1-1 source
pmount 0.1-1 i386
Let's publish them.
>>> binary_item.setDone()
>>> source_item.setDone()
Roll back modified data:
>>> transaction.abort()
Overriding uploads
------------------
Sources and binaries for the upload may be overridden via the methods
overrideSource() and overrideBinaries(). The former allows overriding
of component and section and the latter both those plus the section.
In addition to these parameters, you must also supply
"allowed_components", which is a sequence of IComponent. Any overrides
must have the existing and new component in this sequence otherwise
QueueInconsistentStateError is raised.
The alsa-utils source is already in the queue with component "main"
and section "base".
>>> [item] = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'alsa-utils')
>>> [source] = item.sources
>>> print "%s/%s" % (
... source.sourcepackagerelease.component.name,
... source.sourcepackagerelease.section.name)
main/base
Overriding to a component not in the allowed_components list results in
an error:
>>> restricted = getUtility(IComponentSet)['restricted']
>>> universe = getUtility(IComponentSet)['universe']
>>> main = getUtility(IComponentSet)['main']
>>> web = getUtility(ISectionSet)['web']
>>> print item.overrideSource(
... new_component=restricted, new_section=web,
... allowed_components=(universe,))
Traceback (most recent call last):
...
QueueInconsistentStateError: No rights to override to restricted
Allowing "restricted" still won't work because the original component
is "main":
>>> print item.overrideSource(
... new_component=restricted, new_section=web,
... allowed_components=(restricted,))
Traceback (most recent call last):
...
QueueInconsistentStateError: No rights to override from main
Specifying both main and restricted allows the override to restricted/web.
overrideSource() returns True if it completed the task.
>>> print item.overrideSource(
... new_component=restricted, new_section=web,
... allowed_components=(main,restricted))
True
>>> print "%s/%s" % (
... source.sourcepackagerelease.component.name,
... source.sourcepackagerelease.section.name)
restricted/web
Similarly for binaries:
>>> [item] = breezy_autotest.getPackageUploads(
... PackageUploadStatus.NEW, name=u'pmount')
>>> [build] = item.builds
>>> [binary_package] = build.build.binarypackages
>>> print "%s/%s/%s" % (
... binary_package.component.name,
... binary_package.section.name,
... binary_package.priority.title)
main/base/Important
>>> from lp.soyuz.enums import PackagePublishingPriority
>>> print item.overrideBinaries(
... new_component=restricted,
... new_section=web,
... new_priority=PackagePublishingPriority.EXTRA,
... allowed_components=(universe,))
Traceback (most recent call last):
...
QueueInconsistentStateError: No rights to override to restricted
>>> print item.overrideBinaries(
... new_component=restricted,
... new_section=web,
... new_priority=PackagePublishingPriority.EXTRA,
... allowed_components=(restricted,))
Traceback (most recent call last):
...
QueueInconsistentStateError: No rights to override from main
>>> print item.overrideBinaries(
... new_component=restricted,
... new_section=web,
... new_priority=PackagePublishingPriority.EXTRA,
... allowed_components=(main,restricted))
True
>>> print "%s/%s/%s" % (
... binary_package.component.name,
... binary_package.section.name,
... binary_package.priority.title)
restricted/web/Extra
Queue items retrieval
---------------------
IPackageUploadSet.getPackageUploads() returns an optionally filtered list of
PackageUpload records for the supplied distroseries.
>>> warty = distro['warty']
>>> warty.getPackageUploads().count()
1
Filtering by status:
>>> warty.getPackageUploads(
... status=PackageUploadStatus.DONE).count()
1
Filtering by archive:
>>> from lp.soyuz.enums import ArchivePurpose
>>> from lp.soyuz.interfaces.archive import IArchiveSet
>>> partner_archive = getUtility(IArchiveSet).getByDistroPurpose(
... warty.distribution, ArchivePurpose.PARTNER)
>>> warty.getPackageUploads(archive=partner_archive).count()
0
Filtering by pocket:
>>> warty.getPackageUploads(
... pocket=PackagePublishingPocket.RELEASE).count()
1
Filtering by custom_type. We need to add some custom uploads to show this.
>>> from lp.soyuz.enums import PackageUploadCustomFormat
>>> static_xlat = PackageUploadCustomFormat.STATIC_TRANSLATIONS
>>> def add_static_xlat_upload():
... upload = warty.createQueueEntry(
... pocket=PackagePublishingPocket.RELEASE,
... changesfilename="test", changesfilecontent="test",
... archive=warty.main_archive)
... arbitrary_file = factory.makeLibraryFileAlias()
... upload.addCustom(arbitrary_file, static_xlat)
>>> add_static_xlat_upload()
>>> print warty.getPackageUploads(
... custom_type=static_xlat).count()
1
There is also a created_since_date filter that will only return packages
uploaded since the timestamp specified. This is most useful when retrieving
static translation files. Static translation files are Gnome help files that
are stripped from built packages and uploaded with the binary as a custom
file.
Providing the timestamp of the last one returned in the previous call is a
convenient way to continue from where the caller left off.
Add another custom upload.
>>> add_static_xlat_upload()
>>> uploads = warty.getPackageUploads(custom_type=static_xlat)
>>> print uploads.count()
2
Commit a transaction to ensure new DB objects get a later timestamp.
>>> import transaction
>>> transaction.commit()
>>> last_custom_time = uploads[1].date_created
>>> add_static_xlat_upload()
>>> uploads = warty.getPackageUploads(
... created_since_date=last_custom_time, custom_type=static_xlat)
Only the just-created file is returned:
>>> uploads.count()
1
>>> uploads[0].date_created > last_custom_time
True
Queue Manipulation
------------------
Two convenience methods exist, acceptFromQueue and rejectFromQueue that will
accept or reject the item and send an email respectively.
Let's accept something in the queue. (We need to populate the librarian
with fake changes files first so that emails can be generated.)
>>> from lp.archiveuploader.tests import insertFakeChangesFile
>>> insertFakeChangesFile(items[1].changesfile.id)
>>> insertFakeChangesFile(items[3].changesfile.id)
>>> try:
... items[1].acceptFromQueue()
... except QueueInconsistentStateError, e:
... print 'FAILED'
... else:
... print 'DONE'
DONE
Two emails are generated. We won't look what is inside them here, that is
well shown in nascentupload-announcements.txt.
>>> [notification, announcement] = pop_notifications()
When accepting single sources we also immediately create its
corresponding build records. It means that the source will be ready to
build once it was accepted, using 'queue-tool' or via the Web UI.
>>> queue_source = items[1].sources[0]
>>> [build] = queue_source.sourcepackagerelease.builds
>>> print build.title
i386 build of alsa-utils 1.0.9a-4ubuntu1 in ubuntu hoary RELEASE
>>> print build.status.name
NEEDSBUILD
>>> print build.buildqueue_record.lastscore
10
Let's reject something in the queue:
>>> try:
... items[3].rejectFromQueue()
... except QueueInconsistentStateError, e:
... print 'FAILED'
... else:
... print 'REJECTED'
REJECTED
One email is generated (see nascentupload-announcements.txt)
>>> [notification] = pop_notifications()
Clean up the librarian files:
>>> from lp.testing.layers import LibrarianLayer
>>> LibrarianLayer.librarian_fixture.clear()
Delayed copies
--------------
IPackageUploadSet allows call sites to create 'delayed-copy'
PackageUpload records.
>>> from lp.registry.interfaces.gpg import IGPGKeySet
>>> testing_key = getUtility(IGPGKeySet).get(1)
>>> from lp.soyuz.tests.test_publishing import SoyuzTestPublisher
>>> test_publisher = SoyuzTestPublisher()
>>> test_publisher.addFakeChroots(hoary)
>>> ignore = test_publisher.setUpDefaultDistroSeries(hoary)
>>> from lp.registry.interfaces.person import IPersonSet
>>> cprov = getUtility(IPersonSet).getByName('cprov')
A 'delayed-copy' is a PackageUpload record.
>>> delayed_copy = getUtility(IPackageUploadSet).createDelayedCopy(
... ubuntu.main_archive, hoary,
... PackagePublishingPocket.RELEASE, testing_key)
Delayed copies are manipulated exactly as normal uploads
are. Contents can be attached to it.
>>> a_source_package = test_publisher.createSource(
... cprov.archive, 'foocomm', '1.0-1', new_version='2.0-1')
>>> transaction.commit()
>>> unused = delayed_copy.addSource(a_source_package.sourcepackagerelease)
>>> print delayed_copy.displayname
foocomm (delayed)
>>> verifyObject(IPackageUpload, delayed_copy)
True
The distinction between normal uploads and delayed-copies can be made
by checking IPackageUpload.is_delayed_copy.
>>> delayed_copy.is_delayed_copy
True
>>> normal_upload = breezy_autotest.createQueueEntry(
... PackagePublishingPocket.RELEASE,
... ubuntu.main_archive, 'foo.changes',
... 'something')
>>> normal_upload.is_delayed_copy
False
Delayed-copies are targeted to a publication context (archive,
distroseries, pocket).
>>> print delayed_copy.archive.displayname
Primary Archive for Ubuntu Linux
>>> print delayed_copy.distroseries.displayname
Hoary
>>> print delayed_copy.pocket.name
RELEASE
And are created in NEW state, so they have to pass through the upload
state-machine.
>>> print delayed_copy.status.name
NEW
However they have no 'changesfile', only a signing_key representing the
requester of the copy.
>>> print delayed_copy.changesfile
None
>>> print delayed_copy.signing_key.owner.displayname
Foo Bar
IPackageUpload.acceptFromCopy() simply checks and accepts a
delayed-copy record. Bugs mentioned in the changelog are closed by
`process-accepted` (transition to ACCEPTED to DONE) see
closing-bugs-from-changelogs.txt for more information.
>>> print delayed_copy.status.name
NEW
>>> delayed_copy.acceptFromCopy()
>>> print delayed_copy.status.name
ACCEPTED
Once accepted a delayed-copy is published as a normal upload, via
IPackageUpload.realiseUpload().
>>> publishing_records = delayed_copy.realiseUpload()
>>> for pub_record in publishing_records:
... print pub_record.displayname, pub_record.status.name
foocomm 2.0-1 in hoary PENDING
>>> print delayed_copy.status.name
DONE
As mentioned above, delayed copies wait in the ACCEPTED upload queue
until they get processed, resulting in the corresponding publishing
records.
For that period of time, between creation and processing, a
delayed-copy existence is not apparent in the publishing domain and
creating a duplicated delayed-copy is perfectly possible.
>>> dup_delayed_copy = getUtility(IPackageUploadSet).createDelayedCopy(
... ubuntu.main_archive, breezy_autotest,
... PackagePublishingPocket.RELEASE, testing_key)
>>> unused = dup_delayed_copy.addSource(
... a_source_package.sourcepackagerelease)
Although it cannot be accepted.
>>> dup_delayed_copy.acceptFromCopy()
Traceback (most recent call last):
...
QueueInconsistentStateError: The source foocomm - 2.0-1 is already
accepted in ubuntu/hoary and you cannot upload the same version within the
same distribution. You have to modify the source version and re-upload.
For detecting conflicting delayed copies we have to inspect the
ACCEPTED and DONE queue for uploads of sources matching a given
(name, version) pair in a publishing context, archive and
distribution.
>>> conflict = getUtility(IPackageUploadSet).findSourceUpload(
... 'foocomm', '2.0-1', ubuntu.main_archive, ubuntu)
>>> conflict.id == delayed_copy.id
True
|