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
|
# Copyright (C) 1998-2008 by the Free Software Foundation, Inc.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
# USA.
"""Mailman errors."""
# Base class for all exceptions raised in Mailman (XXX except legacy string
# exceptions).
class MailmanException(Exception):
pass
# Exceptions for problems related to opening a list
class MMListError(MailmanException): pass
class MMUnknownListError(MMListError):
def __init__(self, listname=None):
self._listname = listname
def __str__(self):
return self._listname
# Membership exceptions
class MMMemberError(MailmanException): pass
class MMBadUserError(MMMemberError): pass
class MMAlreadyAMember(MMMemberError): pass
# "New" style membership exceptions (new w/ MM2.1)
class MemberError(MailmanException): pass
class NotAMemberError(MemberError): pass
class AlreadyReceivingDigests(MemberError): pass
class AlreadyReceivingRegularDeliveries(MemberError): pass
class CantDigestError(MemberError): pass
class MustDigestError(MemberError): pass
class MembershipIsBanned(MemberError): pass
# Exception hierarchy for various authentication failures, can be
# raised from functions in SecurityManager.py
class MMAuthenticationError(MailmanException): pass
class MMCookieError(MMAuthenticationError): pass
class MMExpiredCookieError(MMCookieError): pass
class MMInvalidCookieError(MMCookieError): pass
# BAW: these still need to be converted to classes.
MMMustDigestError = "MMMustDigestError"
MMCantDigestError = "MMCantDigestError"
MMNeedApproval = "MMNeedApproval"
MMSubscribeNeedsConfirmation = "MMSubscribeNeedsConfirmation"
MMBadConfirmation = "MMBadConfirmation"
MMAlreadyDigested = "MMAlreadyDigested"
MMAlreadyUndigested = "MMAlreadyUndigested"
MODERATED_LIST_MSG = "Moderated list"
IMPLICIT_DEST_MSG = "Implicit destination"
SUSPICIOUS_HEADER_MSG = "Suspicious header"
FORBIDDEN_SENDER_MSG = "Forbidden sender"
# New style class based exceptions. All the above errors should eventually be
# converted.
class MailmanError(MailmanException):
"""Base class for all Mailman errors."""
pass
class BadDomainSpecificationError(MailmanError):
"""The specification of a virtual domain is invalid or duplicated."""
class MMLoopingPost(MailmanError):
"""Post already went through this list!"""
pass
# Exception hierarchy for bad email address errors that can be raised from
# Utils.ValidateEmail()
class EmailAddressError(MailmanError):
"""Base class for email address validation errors."""
class InvalidEmailAddress(EmailAddressError):
"""Email address is invalid."""
# Exceptions for admin request database
class LostHeldMessage(MailmanError):
"""Held message was lost."""
pass
def _(s):
return s
# Exceptions for the Handler subsystem
class HandlerError(MailmanError):
"""Base class for all handler errors."""
class HoldMessage(HandlerError):
"""Base class for all message-being-held short circuits."""
# funky spelling is necessary to break import loops
reason = _('For some unknown reason')
def reason_notice(self):
return self.reason
# funky spelling is necessary to break import loops
rejection = _('Your message was rejected')
def rejection_notice(self, mlist):
return self.rejection
class DiscardMessage(HandlerError):
"""The message can be discarded with no further action"""
class SomeRecipientsFailed(HandlerError):
"""Delivery to some or all recipients failed"""
def __init__(self, tempfailures, permfailures):
HandlerError.__init__(self)
self.tempfailures = tempfailures
self.permfailures = permfailures
# multiple inheritance for backwards compatibility
class LoopError(DiscardMessage, MMLoopingPost):
"""We've seen this message before"""
class RejectMessage(HandlerError):
"""The message will be bounced back to the sender"""
def __init__(self, notice=None):
if notice is None:
notice = _('Your message was rejected')
if notice.endswith('\n\n'):
pass
elif notice.endswith('\n'):
notice += '\n'
else:
notice += '\n\n'
self.notice = notice
# Subscription exceptions
class SubscriptionError(MailmanError):
"""Subscription errors base class."""
class HostileSubscriptionError(SubscriptionError):
"""A cross-subscription attempt was made.
This exception gets raised when an invitee attempts to use the
invitation to cross-subscribe to some other mailing list.
"""
class AlreadySubscribedError(SubscriptionError):
"""The member is already subscribed to the mailing list with this role."""
def __init__(self, fqdn_listname, address, role):
self._fqdn_listname = fqdn_listname
self._address = address
self._role = role
def __str__(self):
return '%s is already a %s of mailing list %s' % (
self._address, self._role, self._fqdn_listname)
class PasswordError(MailmanError):
"""A password related error."""
class MMBadPasswordError(PasswordError, MMAuthenticationError):
"""A bad password was given."""
class MMPasswordsMustMatch(PasswordError, MMAuthenticationError):
"""The given passwords don't match."""
class BadPasswordSchemeError(PasswordError):
"""A bad password scheme was given."""
def __init__(self, scheme_name='unknown'):
self.scheme_name = scheme_name
def __str__(self):
return 'A bad password scheme was given: %s' % self.scheme_name
|