~launchpad-pqm/launchpad/devel

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
Distributions
=============

From the DerivationOverview spec
<https://launchpad.canonical.com/DerivationOverview>:

    A distribution of GNU/Linux comprises a set of packages, an
    installer, possibly a live-CD, some amount of metadata associated with
    the arrangement of those elements and also a lot of information on
    managing it.

In Launchpad, one distribution is mapped to one row in the Distribution
table.  To retrieve a distribution, use the IDistributionSet utility. If
you've already used IPersonSet to retrieve a Person, or IBugTaskSet to
retrieve a task, this syntax should look familiar.

The IDistributionSet utility is accessed in the usual fashion:

    >>> from lp.registry.interfaces.distribution import (
    ...     IDistribution, IDistributionSet)
    >>> from lp.translations.interfaces.hastranslationimports import (
    ...     IHasTranslationImports)
    >>> distroset = getUtility(IDistributionSet)

To retrieve a specific distribution, use IDistributionSet.get:

    >>> ubuntu = distroset.get(1)
    >>> print ubuntu.name
    ubuntu

The distribution has a useful string representation containing its display
name in quotes and its name in parentheses.

    >>> ubuntu
    <Distribution 'Ubuntu' (ubuntu)>

Or, to grab one by name, use either getByName() or __getitem__().  They both
can be used to look up distributions by their aliases too.

    >>> gentoo = distroset.getByName("gentoo")
    >>> gentoo.name
    u'gentoo'
    >>> distroset["gentoo"].name
    u'gentoo'

    # Need to login as an LP admin to set a project's aliases.
    >>> login('foo.bar@canonical.com')
    >>> gentoo.setAliases(['jackass'])
    >>> gentoo.aliases
    [u'jackass']
    >>> login(ANONYMOUS)
    >>> distroset['jackass'].name
    u'gentoo'
    >>> distroset.getByName('jackass').name
    u'gentoo'

Let's make sure a distribution object properly implements its interfaces.

    >>> IDistribution.providedBy(gentoo)
    True
    >>> verifyObject(IDistribution, gentoo)
    True
    >>> IHasTranslationImports.providedBy(gentoo)
    True
    >>> verifyObject(IHasTranslationImports, gentoo)
    True

Once you've got a distribution, you can retrieve a source package if you
have a SourcePackageName object for it.

    >>> from lp.registry.model.sourcepackagename import (
    ...                                          SourcePackageName)
    >>> from lp.registry.interfaces.distributionsourcepackage import (
    ...     IDistributionSourcePackage)
    >>> from lp.soyuz.interfaces.distributionsourcepackagerelease import (
    ...     IDistributionSourcePackageRelease)

    >>> evo = SourcePackageName.byName("evolution")
    >>> evo_ubuntu = ubuntu.getSourcePackage(evo)
    >>> print evo_ubuntu.name
    evolution

    >>> IDistributionSourcePackage.providedBy(evo_ubuntu)
    True

    >>> from lp.soyuz.model.sourcepackagerelease import (
    ...                                           SourcePackageRelease)
    >>> sourcepackagerelease = SourcePackageRelease.selectOneBy(
    ...     sourcepackagenameID=evo.id, version='1.0')
    >>> sourcepackagerelease.name
    u'evolution'

    >>> evo_ubuntu_rel = ubuntu.getSourcePackageRelease(
    ...                    sourcepackagerelease)
    >>> IDistributionSourcePackageRelease.providedBy(evo_ubuntu_rel)
    True

You can also get a release by name:

    >>> hoary = ubuntu.getSeries("hoary")
    >>> print hoary.name
    hoary

Or by version:

    >>> v504 = ubuntu.getSeries("5.04")
    >>> print v504.name
    hoary

You can list development distroseriess:

    >>> devdists = ubuntu.getDevelopmentSeries()
    >>> for devdist in devdists:
    ...     print devdist.name
    hoary

You can list the series for a distribution,

    >>> for series in ubuntu.series:
    ...     print series.name
    breezy-autotest
    grumpy
    hoary
    warty

as well as the distribution architecture series for a distribution:

    >>> for architecture in ubuntu.architectures:
    ...     print architecture.displayname
    Ubuntu Breezy Badger Autotest i386
    Ubuntu Hoary hppa
    Ubuntu Hoary i386
    Ubuntu Warty hppa
    Ubuntu Warty i386

You can use the has_published_binaries property to find out if the
distribution has any binaries on disk.  This is useful when searching for
packages and you need to tailor any user messages about what types of packages
are available.

    >>> ubuntu.has_published_binaries
    True

    >>> gentoo.has_published_binaries
    False

You can use the has_published_sources property to find out if the
distribution has any published sources.

    >>> ubuntu.has_published_sources
    True

    >>> gentoo.has_published_sources
    False


Distribution Sorting
--------------------

If you ask for all the distributions in the DistributionSet you should get
Ubuntu (and all flavours of it) first and the rest alphabetically:

    >>> for item in distroset.getDistros():
    ...     print item.name
    ubuntu
    kubuntu
    ubuntutest
    debian
    gentoo
    guadalinex
    redhat

DistributionSet also defines __iter__ as a shortcut to getDistros().

    >>> list(distroset) == distroset.getDistros()
    True


Searching for DistributionSourcePackages
........................................

The distribution also allows you to look for source packages that match
a certain string through the magic of fti. For instance:

    >>> packages = ubuntu.searchSourcePackageCaches("mozilla")
    >>> for distro_source_package_cache, source_name, rank in packages:
    ...     print "%-17s rank:%s" % (
    ...         distro_source_package_cache.name,
    ...         type(rank))
    mozilla-firefox   rank:<type 'float'>

The search also matches on exact package names which fti doesn't like,
and even on substrings:

    >>> packages = ubuntu.searchSourcePackageCaches("linux-source-2.6.15")
    >>> print packages.count()
    1
    >>> packages = ubuntu.searchSourcePackageCaches('a')
    >>> for distro_source_package_cache, source_name, rank in packages:
    ...     print "%s: %-17s rank:%s" % (
    ...         distro_source_package_cache.__class__.__name__,
    ...         distro_source_package_cache.name,
    ...         type(rank))
    DistributionSourcePackageCache: alsa-utils        rank:<type 'NoneType'>
    DistributionSourcePackageCache: commercialpackage rank:<type 'NoneType'>
    DistributionSourcePackageCache: foobar            rank:<type 'NoneType'>
    DistributionSourcePackageCache: mozilla-firefox   rank:<type 'NoneType'>
    DistributionSourcePackageCache: netapplet         rank:<type 'NoneType'>

The searchSourcePackages() method just returns a decorated version
of the results from searchSourcePackageCaches():

    >>> packages = ubuntu.searchSourcePackages('a')
    >>> for dsp in packages:
    ...     print "%s: %s" % (dsp.__class__.__name__, dsp.name)
    DistributionSourcePackage: alsa-utils
    DistributionSourcePackage: commercialpackage
    DistributionSourcePackage: foobar
    DistributionSourcePackage: mozilla-firefox
    DistributionSourcePackage: netapplet

searchSourcePackages() also has a has_packaging parameter that
it just passes on to searchSourcePackageCaches(), and it restricts
the results based on whether the source package has an entry
in the Packaging table linking it to an upstream project.

    >>> packages = ubuntu.searchSourcePackages('a', has_packaging=True)
    >>> for dsp in packages:
    ...     print "%s: %s" % (dsp.__class__.__name__, dsp.name)
    DistributionSourcePackage: alsa-utils
    DistributionSourcePackage: mozilla-firefox
    DistributionSourcePackage: netapplet
    >>> packages = ubuntu.searchSourcePackages('a', has_packaging=False)
    >>> for dsp in packages:
    ...     print "%s: %s" % (dsp.__class__.__name__, dsp.name)
    DistributionSourcePackage: commercialpackage
    DistributionSourcePackage: foobar

searchSourcePackages() also has a publishing_distroseries parameter that
it just passes on to searchSourcePackageCaches(), and it restricts the
results based on whether the source package has an entry in the
SourcePackagePublishingHistory table for the given distroseries.

    >>> packages = ubuntu.searchSourcePackages(
    ...     'a', publishing_distroseries=ubuntu.currentseries)
    >>> for dsp in packages:
    ...     print "%s: %s" % (dsp.__class__.__name__, dsp.name)
    DistributionSourcePackage: alsa-utils
    DistributionSourcePackage: netapplet


Searching for binary packages
.............................

searchBinaryPackages() does a name substring match to find binary
packages related to the distribution. It returns
DistributionSourcePackageCache objects, which makes it very easy to
associate the binary name with its source.

Searching for an exact match on a valid binary name returns the
expected results:

    >>> results = ubuntu.searchBinaryPackages(
    ...     "mozilla-firefox", exact_match=True)
    >>> print [result.name for result in results]
    [u'mozilla-firefox']

An exact match search with no matches on any package name returns
an empty result set:

    >>> results = ubuntu.searchBinaryPackages("mozilla", exact_match=True)
    >>> results.count()
    0

Loosening to substring matches gives another result:

    >>> results = ubuntu.searchBinaryPackages("mozilla", exact_match=False)
    >>> print results[0]
    <...DistributionSourcePackageCache instance ...

    >>> print [result.name for result in results]
    [u'mozilla-firefox']
    >>> print [result.binpkgnames for result in results]
    [u'mozilla-firefox mozilla-firefox-data']

The results of searchBinaryPackages() are simply ordered alphabetically
for the moment until we have a better FTI rank to order with.

    >>> results = ubuntu.searchBinaryPackages("m")
    >>> for result in results:
    ...     print result.name
    mozilla-firefox
    pmount


Finding distroseriess and pockets from distribution names
.........................................................

A distribution knows what distroseriess it has. Those distroseriess have
pockets which have suffixes used by the archive publisher. Because we
sometimes need to talk about distroseriess such as ubuntu/hoary-security
we need some way to decompose that into the distroseries and the pocket.
Distribution can do that for us.

If we ask for a totally unknown distroseries, we raise NotFoundError
    >>> ubuntu.getDistroSeriesAndPocket('unknown')
    Traceback (most recent call last):
    ...
    NotFoundError: 'unknown'

If we ask for a plain distroseries, it should come back with the RELEASE
pocket as the pocket.
    >>> dr, pocket = ubuntu.getDistroSeriesAndPocket('hoary')
    >>> print dr.name
    hoary
    >>> print pocket.name
    RELEASE

If we ask for a security pocket in a known distroseries it should come out
on the other side.
    >>> dr, pocket = ubuntu.getDistroSeriesAndPocket('hoary-security')
    >>> print dr.name
    hoary
    >>> print pocket.name
    SECURITY

Find the backports pocket, too:
    >>> dr, pocket = ubuntu.getDistroSeriesAndPocket('hoary-backports')
    >>> print dr.name
    hoary
    >>> print pocket.name
    BACKPORTS

If we ask for a valid distroseries which doesn't have a given pocket it should
raise NotFoundError for us
    >>> ubuntu.getDistroSeriesAndPocket('hoary-bullshit')
    Traceback (most recent call last):
    ...
    NotFoundError: 'hoary-bullshit'


Upload related stuff
....................

When uploading to a distribution we need to query its uploaders. Each
uploader record is in fact an ArchivePermission record that tells us
what component is uploadable to by what person or group of people.

    >>> from operator import attrgetter
    >>> for permission in sorted(
    ...     ubuntu.uploaders, key=attrgetter("id")):
    ...     assert not permission.archive.is_ppa
    ...     print permission.component.name
    ...     print permission.person.displayname
    universe
    Ubuntu Team
    restricted
    Ubuntu Team
    main
    Ubuntu Team
    partner
    Canonical Partner Developers


Launchpad Usage
...............

A distribution can specify if it uses Malone, Rosetta, or Answers
officially. Ubuntu uses all of them:

    >>> from lp.app.interfaces.launchpad import ILaunchpadCelebrities

    >>> ubuntu = getUtility(ILaunchpadCelebrities).ubuntu
    >>> ubuntu.official_malone
    True
    >>> print ubuntu.answers_usage.name
    LAUNCHPAD
    >>> print ubuntu.blueprints_usage.name
    LAUNCHPAD
    >>> print ubuntu.translations_usage.name
    LAUNCHPAD

The bug_tracking_usage property currently only tracks official_malone.

    >>> print ubuntu.bug_tracking_usage.name
    LAUNCHPAD

While the other attributes track the other official_ attributes.

    >>> print ubuntu.official_rosetta
    True
    >>> print ubuntu.translations_usage.name
    LAUNCHPAD
    >>> print ubuntu.official_answers
    True
    >>> print ubuntu.answers_usage.name
    LAUNCHPAD
    >>> print ubuntu.official_blueprints
    True
    >>> print ubuntu.blueprints_usage.name
    LAUNCHPAD

If the official_ attributes are False and the enum hasn't been set,
the usage enums don't know anything.

    >>> login_person(ubuntu.owner.teamowner)
    >>> ubuntu.official_rosetta = False
    >>> print ubuntu.translations_usage.name
    UNKNOWN

A distribution *cannot* specify that it uses codehosting. Currently there's
no way for a distribution to use codehosting.

    >>> from lp.app.enums import ServiceUsage
    >>> print ubuntu.codehosting_usage.name
    NOT_APPLICABLE
    >>> ubuntu.codehosting_usage = ServiceUsage.LAUNCHPAD
    Traceback (most recent call last):
    AttributeError: can't set attribute...

While Debian uses none:

    >>> debian = getUtility(ILaunchpadCelebrities).debian
    >>> print debian.bug_tracking_usage.name
    UNKNOWN
    >>> print debian.translations_usage.name
    UNKNOWN
    >>> print debian.answers_usage.name
    UNKNOWN
    >>> print debian.codehosting_usage.name
    NOT_APPLICABLE
    >>> print debian.blueprints_usage.name
    UNKNOWN

Gentoo only uses Malone

    >>> print gentoo.bug_tracking_usage.name
    LAUNCHPAD
    >>> print gentoo.translations_usage.name
    UNKNOWN
    >>> print gentoo.answers_usage.name
    UNKNOWN

Launchpad admins and the distro owner can set these fields.

    >>> from lp.app.enums import ServiceUsage
    >>> login('mark@example.com')
    >>> debian = getUtility(ILaunchpadCelebrities).debian
    >>> debian.blueprints_usage = ServiceUsage.LAUNCHPAD
    >>> print debian.blueprints_usage.name
    LAUNCHPAD
    >>> debian.official_malone = True
    >>> debian.official_malone
    True
    >>> debian.translations_usage = ServiceUsage.LAUNCHPAD
    >>> debian.translations_usage.name
    'LAUNCHPAD'

    >>> debian_owner = factory.makePerson()
    >>> debian.owner = debian_owner
    >>> login_person(debian_owner)
    >>> debian.blueprints_usage = ServiceUsage.NOT_APPLICABLE
    >>> print debian.blueprints_usage.name
    NOT_APPLICABLE

But others can't.

    >>> login('no-priv@canonical.com')
    >>> debian.blueprints_usage = ServiceUsage.LAUNCHPAD
    Traceback (most recent call last):
    Unauthorized: (..., 'blueprints_usage', 'launchpad.Edit')
    >>> debian.official_malone = True
    Traceback (most recent call last):
    Unauthorized: (..., 'official_malone', 'launchpad.Edit')
    >>> debian.translations_usage = ServiceUsage.LAUNCHPAD
    Traceback (most recent call last):
    Unauthorized: (..., 'translations_usage', 'launchpad.Edit')


Specification Listings
......................

We should be able to get lists of specifications in different states
related to a distro.

Basically, we can filter by completeness, and by whether or not the spec is
informational.

    >>> kubuntu = distroset.getByName("kubuntu")

    >>> from lp.blueprints.enums import SpecificationFilter

First, there should be one informational spec for kubuntu, but it is
complete so it will not show up unless we explicitly ask for complete specs:

    >>> filter = [SpecificationFilter.INFORMATIONAL]
    >>> kubuntu.specifications(filter=filter).count()
    0
    >>> filter = [SpecificationFilter.INFORMATIONAL,
    ...           SpecificationFilter.COMPLETE]
    >>> kubuntu.specifications(filter=filter).count()
    1


There are 2 completed specs for Kubuntu:

    >>> filter = [SpecificationFilter.COMPLETE]
    >>> for spec in kubuntu.specifications(filter=filter):
    ...    print spec.name, spec.is_complete
    thinclient-local-devices True
    usplash-on-hibernation True


And there are four incomplete specs:

    >>> filter = [SpecificationFilter.INCOMPLETE]
    >>> for spec in kubuntu.specifications(filter=filter):
    ...    print spec.name, spec.is_complete
    cluster-installation False
    revu False
    kde-desktopfile-langpacks False
    krunch-desktop-plan False


If we ask for all specs, we get them in the order of priority.

    >>> filter = [SpecificationFilter.ALL]
    >>> for spec in kubuntu.specifications(filter=filter):
    ...    print spec.priority.title, spec.name
    Essential cluster-installation
    High revu
    Medium thinclient-local-devices
    Low usplash-on-hibernation
    Undefined kde-desktopfile-langpacks
    Not krunch-desktop-plan


And if we ask just for specs, we get the incomplete ones.

    >>> for spec in kubuntu.specifications():
    ...     print spec.name, spec.is_complete
    cluster-installation False
    revu False
    kde-desktopfile-langpacks False
    krunch-desktop-plan False

We can filter for specifications that contain specific text:

    >>> for spec in kubuntu.specifications(filter=['package']):
    ...     print spec.name
    revu

We can get only valid specs (those that are not obsolete or superseded):

    >>> from lp.blueprints.enums import SpecificationDefinitionStatus
    >>> login('mark@example.com')
    >>> for spec in kubuntu.specifications():
    ...     # Do this here, otherwise, the change will be flush before
    ...     # updateLifecycleStatus() acts and an IntegrityError will be
    ...     # raised.
    ...     owner = spec.owner
    ...     if spec.name in ['cluster-installation', 'revu']:
    ...         spec.definition_status = (
    ...             SpecificationDefinitionStatus.OBSOLETE)
    ...     if spec.name in ['krunch-desktop-plan']:
    ...         spec.definition_status = (
    ...             SpecificationDefinitionStatus.SUPERSEDED)
    ...     shim = spec.updateLifecycleStatus(owner)
    >>> for spec in kubuntu.valid_specifications:
    ...     print spec.name
    kde-desktopfile-langpacks


Milestones
----------

We can use IDistribution.milestones to get all milestones associated with any
series of a distribution.

    >>> from datetime import datetime
    >>> [milestone.name for milestone in debian.milestones]
    [u'3.1', u'3.1-rc1']

    >>> woody = debian['woody']

Milestones for distros can only be created by distro owners or admins.

    >>> login('no-priv@canonical.com')
    >>> woody.newMilestone(
    ...     name='impossible', dateexpected=datetime(2028, 10, 1))
    Traceback (most recent call last):
    ...
    Unauthorized: (<DistroSeries u'woody'>, 'newMilestone', 'launchpad.Edit')
    >>> login('mark@example.com')
    >>> debian_milestone = woody.newMilestone(
    ...     name='woody-rc1', dateexpected=datetime(2028, 10, 1))

They're ordered by dateexpected.

    >>> [(milestone.name, milestone.dateexpected.strftime('%Y-%m-%d'))
    ...  for milestone in debian.milestones]
    [(u'3.1', '2056-05-16'), (u'3.1-rc1', '2056-02-16'),
     (u'woody-rc1', '2028-10-01')]

Only milestones which have visible=True are returned by the .milestones
property.

    >>> debian_milestone.active = False
    >>> [milestone.name for milestone in debian.milestones]
    [u'3.1', u'3.1-rc1']

To get all milestones of a given distro we have the .all_milestones property.

    >>> [milestone.name for milestone in debian.all_milestones]
    [u'3.1', u'3.1-rc1', u'woody-rc1']


Archives
--------

A distribution archive (primary, partner, debug or copy) can be retrieved
by name using IDistribution.getArchive.

    >>> def display_archive(archive):
    ...     print '%s %s %s' % (
    ...         archive.distribution.name, archive.owner.name, archive.name)
    >>> display_archive(ubuntu.getArchive('primary'))
    ubuntu ubuntu-team primary
    >>> display_archive(ubuntu.getArchive('partner'))
    ubuntu ubuntu-team partner
    >>> display_archive(debian.getArchive('primary'))
    debian mark primary
    >>> ubuntu.getArchive('ppa')
    >>> debian.getArchive('partner')