Coverage for distro_tracker/mail/dispatch.py: 96%

201 statements  

« prev     ^ index     » next       coverage.py v6.5.0, created at 2025-01-12 09:15 +0000

1# Copyright 2013-2016 The Distro Tracker Developers 

2# See the COPYRIGHT file at the top-level directory of this distribution and 

3# at https://deb.li/DTAuthors 

4# 

5# This file is part of Distro Tracker. It is subject to the license terms 

6# in the LICENSE file found in the top-level directory of this 

7# distribution and at https://deb.li/DTLicense. No part of Distro Tracker, 

8# including this file, may be copied, modified, propagated, or distributed 

9# except according to the terms contained in the LICENSE file. 

10""" 

11Implements the processing of received package messages in order to dispatch 

12them to subscribers. 

13""" 

14import logging 

15import re 

16from copy import deepcopy 

17from datetime import datetime 

18 

19from django.conf import settings 

20from django.core.mail import EmailMessage, get_connection 

21from django.utils import timezone 

22 

23from distro_tracker import vendor 

24from distro_tracker.core.models import Keyword, PackageName, Team 

25from distro_tracker.core.utils import ( 

26 distro_tracker_render_to_string, 

27 extract_email_address_from_header, 

28 get_decoded_message_payload, 

29 get_or_none, 

30 verp 

31) 

32from distro_tracker.core.utils.email_messages import ( 

33 CustomEmailMessage, 

34 patch_message_for_django_compat 

35) 

36from distro_tracker.mail.models import UserEmailBounceStats 

37 

38DISTRO_TRACKER_CONTROL_EMAIL = settings.DISTRO_TRACKER_CONTROL_EMAIL 

39DISTRO_TRACKER_FQDN = settings.DISTRO_TRACKER_FQDN 

40 

41logger = logging.getLogger(__name__) 

42 

43 

44class SkipMessage(Exception): 

45 """This exception can be raised by the vendor provided classify_message() 

46 to tell the dispatch code to skip processing this message being processed. 

47 The mail is then silently dropped.""" 

48 

49 

50def _get_logdata(msg, package, keyword, team): 

51 return { 

52 'from': extract_email_address_from_header(msg.get('From', '')), 

53 'msgid': msg.get('Message-ID', 'no-msgid-present@localhost'), 

54 'package': package or '<unknown>', 

55 'keyword': keyword or '<unknown>', 

56 'team': team or '<unknown>', 

57 } 

58 

59 

60def _must_discard(msg, logdata): 

61 # Check loop 

62 dispatch_email = 'dispatch@{}'.format(DISTRO_TRACKER_FQDN) 

63 if dispatch_email in msg.get_all('X-Loop', ()): 

64 # Bad X-Loop, discard the message 

65 logger.info('dispatch :: discarded %(msgid)s due to X-Loop', logdata) 

66 return True 

67 return False 

68 

69 

70def process(msg, package=None, keyword=None): 

71 """ 

72 Dispatches received messages by identifying where they should 

73 be sent and then by forwarding them. 

74 

75 :param msg: The received message 

76 :type msg: :py:class:`email.message.Message` 

77 

78 :param str package: The package to which the message was sent. 

79 

80 :param str keyword: The keyword under which the message must be dispatched. 

81 """ 

82 logdata = _get_logdata(msg, package, keyword, None) 

83 logger.info("dispatch :: received from %(from)s :: %(msgid)s", 

84 logdata) 

85 try: 

86 package, keyword = classify_message(msg, package, keyword) 

87 except SkipMessage: 

88 logger.info('dispatch :: skipping %(msgid)s', logdata) 

89 return 

90 

91 if package is None: 

92 logger.warning('dispatch :: no package identified for %(msgid)s', 

93 logdata) 

94 return 

95 

96 if _must_discard(msg, logdata): 

97 return 

98 

99 if isinstance(package, (list, set)): 

100 for pkg in package: 

101 forward(msg, pkg, keyword) 

102 else: 

103 forward(msg, package, keyword) 

104 

105 

106def forward(msg, package, keyword): 

107 """ 

108 Forwards a received message to the various subscribers of the 

109 given package/keyword combination. 

110 

111 :param msg: The received message 

112 :type msg: :py:class:`email.message.Message` 

113 

114 :param str package: The package name. 

115 

116 :param str keyword: The keyword under which the message must be forwarded. 

117 """ 

118 logdata = _get_logdata(msg, package, keyword, None) 

119 

120 logger.info("dispatch :: forward to %(package)s %(keyword)s :: %(msgid)s", 

121 logdata) 

122 

123 # Default keywords require special approvement 

124 if keyword == 'default' and not approved_default(msg): 

125 logger.info('dispatch :: discarded non-approved message %(msgid)s', 

126 logdata) 

127 return 

128 

129 # Now send the message to subscribers 

130 add_new_headers(msg, package_name=package, keyword=keyword) 

131 send_to_subscribers(msg, package, keyword) 

132 send_to_teams(msg, package, keyword) 

133 

134 

135def process_for_team(msg, team_slug): 

136 """Dispatch a message sent to a team.""" 

137 logdata = _get_logdata(msg, None, None, team_slug) 

138 logger.info("dispatch :: received for team %(team)s " 

139 "from %(from)s :: %(msgid)s", logdata) 

140 

141 if _must_discard(msg, logdata): 

142 return 

143 

144 try: 

145 team = Team.objects.get(slug=team_slug) 

146 except Team.DoesNotExist: 

147 logger.info("dispatch :: discarded %(msgid)s for team %(team)s " 

148 "since team doesn't exist", logdata) 

149 return 

150 

151 package, keyword = classify_message(msg) 

152 if package: 

153 logger.info("dispatch :: discarded %(msgid)s for team %(team)s " 

154 "as an automatic mail", logdata) 

155 return 

156 

157 forward_to_team(msg, team) 

158 

159 

160def forward_to_team(msg, team): 

161 """Forward a message to a team, adding headers as required.""" 

162 logdata = _get_logdata(msg, None, None, team.slug) 

163 logger.info("dispatch :: forward to team %(team)s :: %(msgid)s", 

164 logdata) 

165 

166 add_new_headers(msg, keyword="contact", team=team.slug) 

167 send_to_team(msg, team, keyword="contact") 

168 

169 

170def classify_message(msg, package=None, keyword=None): 

171 """ 

172 Analyzes a message to identify what package it is about and 

173 what keyword is appropriate. 

174 

175 :param msg: The received message 

176 :type msg: :py:class:`email.message.Message` 

177 

178 :param str package: The suggested package name. 

179 

180 :param str keyword: The suggested keyword under which the message can be 

181 forwarded. 

182 

183 """ 

184 if package is None: 

185 package = msg.get('X-Distro-Tracker-Package') 

186 if keyword is None: 

187 keyword = msg.get('X-Distro-Tracker-Keyword') 

188 

189 result, implemented = vendor.call('classify_message', msg, 

190 package=package, keyword=keyword) 

191 if implemented: 

192 package, keyword = result 

193 if package and keyword is None: 

194 keyword = 'default' 

195 return (package, keyword) 

196 

197 

198def approved_default(msg): 

199 """ 

200 The function checks whether a message tagged with the default keyword should 

201 be approved, meaning that it gets forwarded to subscribers. 

202 

203 :param msg: The received package message 

204 :type msg: :py:class:`email.message.Message` or an equivalent interface 

205 object 

206 """ 

207 if 'X-Distro-Tracker-Approved' in msg: 

208 return True 

209 

210 approved, implemented = vendor.call('approve_default_message', msg) 

211 if implemented: 211 ↛ 214line 211 didn't jump to line 214, because the condition on line 211 was never false

212 return approved 

213 else: 

214 return False 

215 

216 

217def add_new_headers(received_message, package_name=None, keyword=None, 

218 team=None): 

219 """ 

220 The function adds new distro-tracker specific headers to the received 

221 message. This is used before forwarding the message to subscribers. 

222 

223 The headers added by this function are used regardless whether the 

224 message is forwarded due to direct package subscriptions or a team 

225 subscription. 

226 

227 :param received_message: The received package message 

228 :type received_message: :py:class:`email.message.Message` or an equivalent 

229 interface object 

230 

231 :param package_name: The name of the package for which this message was 

232 intended. 

233 :type package_name: string 

234 

235 :param keyword: The keyword with which the message should be tagged 

236 :type keyword: string 

237 """ 

238 new_headers = [ 

239 ('X-Loop', 'dispatch@{}'.format(DISTRO_TRACKER_FQDN)), 

240 ] 

241 if keyword: 241 ↛ 243line 241 didn't jump to line 243, because the condition on line 241 was never false

242 new_headers.append(('X-Distro-Tracker-Keyword', keyword)) 

243 if package_name: 

244 new_headers.extend([ 

245 ('X-Distro-Tracker-Package', package_name), 

246 ('List-Id', '<{}.{}>'.format(package_name, DISTRO_TRACKER_FQDN)), 

247 ]) 

248 if team: 

249 new_headers.append(('X-Distro-Tracker-Team', team)) 

250 

251 extra_vendor_headers, implemented = vendor.call( 

252 'add_new_headers', received_message, package_name, keyword, team) 

253 if implemented: 

254 new_headers.extend(extra_vendor_headers) 

255 

256 for header_name, header_value in new_headers: 

257 received_message[header_name] = header_value 

258 

259 

260def add_direct_subscription_headers(received_message, package_name, keyword): 

261 """ 

262 The function adds headers to the received message which are specific for 

263 messages to be sent to users that are directly subscribed to the package. 

264 """ 

265 new_headers = [ 

266 ('Precedence', 'list'), 

267 ('List-Unsubscribe', 

268 '<mailto:{control_email}?body=unsubscribe%20{package}>'.format( 

269 control_email=DISTRO_TRACKER_CONTROL_EMAIL, 

270 package=package_name)), 

271 ] 

272 for header_name, header_value in new_headers: 

273 received_message[header_name] = header_value 

274 

275 

276def add_team_membership_headers(received_message, keyword, team): 

277 """ 

278 The function adds headers to the received message which are specific for 

279 messages to be sent to users that are members of a team. 

280 """ 

281 received_message['X-Distro-Tracker-Team'] = team.slug 

282 

283 

284def send_to_teams(received_message, package_name, keyword): 

285 """ 

286 Sends the given email message to all members of each team that has the 

287 given package. 

288 

289 The message is only sent to those users who have not muted the team 

290 and have the given keyword in teir set of keywords for the team 

291 membership. 

292 

293 :param received_message: The modified received package message to be sent 

294 to the subscribers. 

295 :type received_message: :py:class:`email.message.Message` or an equivalent 

296 interface object 

297 

298 :param package_name: The name of the package for which this message was 

299 intended. 

300 :type package_name: string 

301 

302 :param keyword: The keyword with which the message should be tagged 

303 :type keyword: string 

304 """ 

305 keyword = get_or_none(Keyword, name=keyword) 

306 package = get_or_none(PackageName, name=package_name) 

307 if not keyword or not package: 

308 return 

309 # Get all teams that have the given package 

310 teams = Team.objects.filter(packages=package) 

311 teams = teams.prefetch_related('team_membership_set') 

312 

313 for team in teams: 

314 send_to_team(received_message, team, keyword, package.name) 

315 

316 

317def send_to_team(received_message, team, keyword, package_name=None): 

318 """Send a message to a team.""" 

319 keyword = get_or_none(Keyword, name=keyword) 

320 package = get_or_none(PackageName, name=package_name) 

321 date = timezone.now().date() 

322 messages_to_send = [] 

323 logger.info('dispatch :: sending to team %s', team.slug) 

324 team_message = deepcopy(received_message) 

325 add_team_membership_headers(team_message, keyword.name, team) 

326 

327 # Send the message to each member of the team 

328 for membership in team.team_membership_set.all(): 

329 # Do not send messages to muted memberships 

330 if membership.is_muted(package): 

331 continue 

332 # Do not send the message if the user has disabled the keyword 

333 if keyword not in membership.get_keywords(package): 

334 continue 

335 

336 messages_to_send.append(prepare_message( 

337 team_message, membership.user_email.email, date)) 

338 

339 send_messages(messages_to_send, date) 

340 

341 

342def send_to_subscribers(received_message, package_name, keyword): 

343 """ 

344 Sends the given email message to all subscribers of the package with the 

345 given name and those that accept messages tagged with the given keyword. 

346 

347 :param received_message: The modified received package message to be sent 

348 to the subscribers. 

349 :type received_message: :py:class:`email.message.Message` or an equivalent 

350 interface object 

351 

352 :param package_name: The name of the package for which this message was 

353 intended. 

354 :type package_name: string 

355 

356 :param keyword: The keyword with which the message should be tagged 

357 :type keyword: string 

358 """ 

359 # Make a copy of the message to be sent and add any headers which are 

360 # specific for users that are directly subscribed to the package. 

361 received_message = deepcopy(received_message) 

362 add_direct_subscription_headers(received_message, package_name, keyword) 

363 package = get_or_none(PackageName, name=package_name) 

364 if not package: 

365 return 

366 # Build a list of all messages to be sent 

367 date = timezone.now().date() 

368 messages_to_send = [ 

369 prepare_message(received_message, 

370 subscription.email_settings.user_email.email, 

371 date) 

372 for subscription in package.subscription_set.all_active(keyword) 

373 ] 

374 send_messages(messages_to_send, date) 

375 

376 

377def send_messages(messages_to_send, date): 

378 """ 

379 Sends all the given email messages over a single SMTP connection. 

380 """ 

381 connection = get_connection() 

382 connection.send_messages(messages_to_send) 

383 

384 for message in messages_to_send: 

385 logger.info("dispatch => %s", message.to[0]) 

386 UserEmailBounceStats.objects.add_sent_for_user(email=message.to[0], 

387 date=date) 

388 

389 

390def prepare_message(received_message, to_email, date): 

391 """ 

392 Converts a message which is to be sent to a subscriber to a 

393 :py:class:`CustomEmailMessage 

394 <distro_tracker.core.utils.email_messages.CustomEmailMessage>` 

395 so that it can be sent out using Django's API. 

396 It also sets the required evelope-to value in order to track the bounce for 

397 the message. 

398 

399 :param received_message: The modified received package message to be sent 

400 to the subscribers. 

401 :type received_message: :py:class:`email.message.Message` or an equivalent 

402 interface object 

403 

404 :param to_email: The email of the subscriber to whom the message is to be 

405 sent 

406 :type to_email: string 

407 

408 :param date: The date which should be used as the message's sent date. 

409 :type date: :py:class:`datetime.datetime` 

410 """ 

411 bounce_address = 'bounces+{date}@{distro_tracker_fqdn}'.format( 

412 date=date.strftime('%Y%m%d'), 

413 distro_tracker_fqdn=DISTRO_TRACKER_FQDN) 

414 message = CustomEmailMessage( 

415 msg=patch_message_for_django_compat(received_message), 

416 from_email=verp.encode(bounce_address, to_email), 

417 to=[to_email]) 

418 return message 

419 

420 

421def bounce_is_for_spam(message): 

422 """Return True if the bounce has been generated by spam, False otherwise.""" 

423 spam_bounce_re = [ 

424 # Google blocks executables files 

425 # 552-5.7.0 This message was blocked because its content presents a[...] 

426 # 552-5.7.0 security issue. Please visit 

427 # 552-5.7.0 https://support.google.com/mail/?p=BlockedMessage to [...] 

428 # 552 5.7.0 message content and attachment content guidelines. [...] 

429 r"552-5.7.0 This message was blocked", 

430 # host ...: 550 High probability of spam 

431 # host ...: 554 5.7.1 Message rejected because it contains malware 

432 # 550 Executable files are not allowed in compressed files. 

433 # 554 5.7.1 Spam message rejected 

434 r"55[0-9][- ].*(?:[Ss]pam|malware|virus|[Ee]xecutable files)", 

435 

436 ] 

437 # XXX: Handle delivery report properly 

438 for part in message.walk(): 

439 if not part or part.is_multipart(): 439 ↛ 440line 439 didn't jump to line 440, because the condition on line 439 was never true

440 continue 

441 text = get_decoded_message_payload(part) 

442 if text is None: 

443 continue 

444 for line in text.splitlines()[0:15]: 444 ↛ 438line 444 didn't jump to line 438, because the loop on line 444 didn't complete

445 for rule in spam_bounce_re: 

446 if re.search(rule, line): 

447 return True 

448 

449 return False 

450 

451 

452def handle_bounces(sent_to_address, message): 

453 """ 

454 Handles a received bounce message. 

455 

456 :param sent_to_address: The envelope-to (return path) address to which the 

457 bounced email was returned. 

458 :type sent_to_address: string 

459 """ 

460 try: 

461 bounce_email, user_email = verp.decode(sent_to_address) 

462 except ValueError: 

463 logger.warning('bounces :: no VERP data to extract from %s', 

464 sent_to_address) 

465 return 

466 match = re.match(r'^bounces\+(\d{8})@' + DISTRO_TRACKER_FQDN, bounce_email) 

467 if not match: 467 ↛ 468line 467 didn't jump to line 468, because the condition on line 467 was never true

468 logger.warning('bounces :: invalid address %s', bounce_email) 

469 return 

470 try: 

471 date = datetime.strptime(match.group(1), '%Y%m%d') 

472 except ValueError: 

473 logger.warning('bounces :: invalid date in address %s', bounce_email) 

474 return 

475 

476 logger.info('bounces :: received one for %s/%s', user_email, date) 

477 try: 

478 user = UserEmailBounceStats.objects.get(email__iexact=user_email) 

479 except UserEmailBounceStats.DoesNotExist: 

480 logger.warning('bounces :: unknown user email %s', user_email) 

481 return 

482 

483 if bounce_is_for_spam(message): 

484 logger.info('bounces :: discarded spam bounce for %s/%s', 

485 user_email, date) 

486 return 

487 

488 UserEmailBounceStats.objects.add_bounce_for_user(email=user_email, 

489 date=date) 

490 

491 if user.has_too_many_bounces(): 

492 logger.info('bounces => %s has too many bounces', user_email) 

493 

494 packages = list(user.emailsettings.packagename_set.all()) 

495 teams = [m.team for m in user.membership_set.all()] 

496 email_body = distro_tracker_render_to_string( 

497 'dispatch/unsubscribed-due-to-bounces-email.txt', { 

498 'email': user_email, 

499 'packages': packages, 

500 'teams': teams, 

501 }) 

502 EmailMessage( 

503 subject='All your package subscriptions have been cancelled', 

504 from_email=settings.DISTRO_TRACKER_BOUNCES_LIKELY_SPAM_EMAIL, 

505 to=[user_email], 

506 cc=[settings.DISTRO_TRACKER_CONTACT_EMAIL], 

507 body=email_body, 

508 headers={ 

509 'From': settings.DISTRO_TRACKER_CONTACT_EMAIL, 

510 }, 

511 ).send() 

512 

513 user.emailsettings.unsubscribe_all() 

514 for package in packages: 

515 logger.info('bounces :: removed %s from %s', user_email, 

516 package.name) 

517 user.membership_set.all().update(muted=True) 

518 for team in teams: 

519 logger.info('bounces :: muted membership of %s in team %s', 

520 user_email, team.slug)