Clean up ChunkedUpdateDrawingAreaProxy
[WebKit-https.git] / WebKitTools / Scripts / webkitpy / common / net / bugzilla / bug.py
1 # Copyright (c) 2009 Google Inc. All rights reserved.
2 # Copyright (c) 2009 Apple Inc. All rights reserved.
3 # Copyright (c) 2010 Research In Motion Limited. All rights reserved.
4 #
5 # Redistribution and use in source and binary forms, with or without
6 # modification, are permitted provided that the following conditions are
7 # met:
8 #
9 #     * Redistributions of source code must retain the above copyright
10 # notice, this list of conditions and the following disclaimer.
11 #     * Redistributions in binary form must reproduce the above
12 # copyright notice, this list of conditions and the following disclaimer
13 # in the documentation and/or other materials provided with the
14 # distribution.
15 #     * Neither the name of Google Inc. nor the names of its
16 # contributors may be used to endorse or promote products derived from
17 # this software without specific prior written permission.
18 #
19 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22 # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23 # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29 # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
31 from .attachment import Attachment
32
33
34 class Bug(object):
35     # FIXME: This class is kinda a hack for now.  It exists so we have one
36     # place to hold bug logic, even if much of the code deals with
37     # dictionaries still.
38
39     def __init__(self, bug_dictionary, bugzilla):
40         self.bug_dictionary = bug_dictionary
41         self._bugzilla = bugzilla
42
43     def id(self):
44         return self.bug_dictionary["id"]
45
46     def title(self):
47         return self.bug_dictionary["title"]
48
49     def reporter_email(self):
50         return self.bug_dictionary["reporter_email"]
51
52     def assigned_to_email(self):
53         return self.bug_dictionary["assigned_to_email"]
54
55     # FIXME: This information should be stored in some sort of webkit_config.py instead of here.
56     unassigned_emails = frozenset([
57         "webkit-unassigned@lists.webkit.org",
58         "webkit-qt-unassigned@trolltech.com",
59     ])
60
61     def is_unassigned(self):
62         return self.assigned_to_email() in self.unassigned_emails
63
64     def status(self):
65         return self.bug_dictionary["bug_status"]
66
67     # Bugzilla has many status states we don't really use in WebKit:
68     # https://bugs.webkit.org/page.cgi?id=fields.html#status
69     _open_states = ["UNCONFIRMED", "NEW", "ASSIGNED", "REOPENED"]
70     _closed_states = ["RESOLVED", "VERIFIED", "CLOSED"]
71
72     def is_open(self):
73         return self.status() in self._open_states
74
75     def is_closed(self):
76         return not self.is_open()
77
78     def duplicate_of(self):
79         return self.bug_dictionary.get('dup_id', None)
80
81     # Rarely do we actually want obsolete attachments
82     def attachments(self, include_obsolete=False):
83         attachments = self.bug_dictionary["attachments"]
84         if not include_obsolete:
85             attachments = filter(lambda attachment:
86                                  not attachment["is_obsolete"], attachments)
87         return [Attachment(attachment, self) for attachment in attachments]
88
89     def patches(self, include_obsolete=False):
90         return [patch for patch in self.attachments(include_obsolete)
91                                    if patch.is_patch()]
92
93     def unreviewed_patches(self):
94         return [patch for patch in self.patches() if patch.review() == "?"]
95
96     def reviewed_patches(self, include_invalid=False):
97         patches = [patch for patch in self.patches() if patch.review() == "+"]
98         if include_invalid:
99             return patches
100         # Checking reviewer() ensures that it was both reviewed and has a valid
101         # reviewer.
102         return filter(lambda patch: patch.reviewer(), patches)
103
104     def commit_queued_patches(self, include_invalid=False):
105         patches = [patch for patch in self.patches()
106                                       if patch.commit_queue() == "+"]
107         if include_invalid:
108             return patches
109         # Checking committer() ensures that it was both commit-queue+'d and has
110         # a valid committer.
111         return filter(lambda patch: patch.committer(), patches)