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
|
# Copyright 2009 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
# pylint: disable-msg=E0211,E0213
"""Interfaces for objects that can be linked to bugs."""
__metaclass__ = type
__all__ = [
'IBugLink',
'IBugLinkForm',
'IBugLinkTarget',
'IUnlinkBugsForm',
]
from lazr.restful.declarations import (
export_as_webservice_entry,
exported,
)
from lazr.restful.fields import (
CollectionField,
Reference,
)
from zope.interface import (
Attribute,
implements,
Interface,
)
from zope.schema import (
Choice,
List,
Object,
Set,
)
from zope.schema.interfaces import IContextSourceBinder
from zope.schema.vocabulary import (
SimpleTerm,
SimpleVocabulary,
)
from zope.security.interfaces import Unauthorized
from lp import _
from lp.bugs.interfaces.hasbug import IHasBug
from lp.bugs.interfaces.bug import IBug
from lp.services.fields import BugField
class IBugLink(IHasBug):
"""An entity representing a link between a bug and its target."""
bug = BugField(title=_("The bug that is linked to."),
required=True, readonly=True)
bugID = Attribute("Database id of the bug.")
target = Object(title=_("The object to which the bug is linked."),
required=True, readonly=True, schema=Interface)
class IBugLinkTarget(Interface):
"""An entity which can be linked to bugs.
Examples include an ISpecification.
"""
export_as_webservice_entry(as_of="beta")
bugs = exported(
CollectionField(title=_("Bugs related to this object."),
value_type=Reference(schema=IBug), readonly=True),
as_of="devel")
bug_links = List(title=_("The links between bugs and this object."),
value_type=Object(schema=IBugLink), readonly=True)
def linkBug(bug):
"""Link the object with this bug. If the object is already linked,
return the old linker, otherwise return a new IBugLink object.
If a new IBugLink is created by this method, a ObjectCreatedEvent
should be sent.
"""
def unlinkBug(bug):
"""Remove any link between this object and the bug. If the bug wasn't
linked to the target, returns None otherwise returns the IBugLink
object which was removed.
If an IBugLink is removed by this method, a ObjectDeletedEvent
should be sent.
"""
# These schemas are only used by browser/buglinktarget.py and should really
# live there. See Bug #66950.
class IBugLinkForm(Interface):
"""Schema for the unlink bugs form."""
bug = BugField(
title=_('Bug ID'), required=True)
# XXX flacoste 2006-08-29: To remain consistent with the existing source
# code layout policy, this should really be in vocabularies.buglinks but this
# is not possible because of dependencies on interfaces in some vocabularies
# modules.
class BugLinksVocabularyFactory:
"""IContextSourceBinder that creates a vocabulary of the linked bugs on
the IBugLinkTarget.
"""
implements(IContextSourceBinder)
def __call__(self, context):
"""See IContextSourceBinder."""
terms = []
for bug in context.bugs:
try:
title = _(
'#${bugid}: ${title}',
mapping={'bugid': bug.id, 'title': bug.title})
terms.append(SimpleTerm(bug, bug.id, title))
except Unauthorized:
pass
return SimpleVocabulary(terms)
class IUnlinkBugsForm(Interface):
"""Schema for the unlink bugs form."""
bugs = Set(title=_('Bug Links'), required=True,
value_type=Choice(source=BugLinksVocabularyFactory()),
description=_('Select the bug links that you want to remove.'))
|