2009-08-04 13:43:31 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# mail.py - methods used by requestsync when used in "mail" mode
|
|
|
|
#
|
2011-01-15 20:54:45 +02:00
|
|
|
# Copyright © 2009 Michael Bienia <geser@ubuntu.com>,
|
|
|
|
# 2011 Stefano Rivera <stefanor@ubuntu.com>
|
2009-08-04 13:43:31 +02:00
|
|
|
#
|
2009-08-05 23:10:05 +02:00
|
|
|
# This module may contain code written by other authors/contributors to
|
|
|
|
# the main requestsync script. See there for their names.
|
|
|
|
#
|
2009-08-04 13:43:31 +02:00
|
|
|
# This program is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License
|
2009-08-05 23:10:05 +02:00
|
|
|
# as published by the Free Software Foundation; version 2
|
2010-12-03 00:06:43 +01:00
|
|
|
#
|
2009-08-04 13:43:31 +02:00
|
|
|
# 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.
|
|
|
|
#
|
2009-08-05 23:10:05 +02:00
|
|
|
# Please see the /usr/share/common-licenses/GPL-2 file for the full text
|
2009-08-04 13:43:31 +02:00
|
|
|
# of the GNU General Public License license.
|
|
|
|
|
2009-08-07 12:53:33 +02:00
|
|
|
import os
|
2011-12-03 18:09:49 +01:00
|
|
|
import re
|
2009-08-07 12:53:33 +02:00
|
|
|
import sys
|
2009-08-22 17:12:55 +02:00
|
|
|
import smtplib
|
|
|
|
import socket
|
2011-06-25 17:53:44 +02:00
|
|
|
|
2011-11-13 22:50:34 +02:00
|
|
|
from debian.changelog import Changelog, Version
|
2011-09-09 20:03:47 +02:00
|
|
|
from devscripts.logger import Logger
|
2011-06-25 17:53:44 +02:00
|
|
|
from distro_info import DebianDistroInfo
|
|
|
|
|
2011-01-15 20:54:45 +02:00
|
|
|
from ubuntutools.archive import rmadison, FakeSPPH
|
2011-11-13 20:15:19 +02:00
|
|
|
from ubuntutools.question import confirmation_prompt, YesNoQuestion
|
2011-05-24 20:22:37 +02:00
|
|
|
from ubuntutools import subprocess
|
2010-02-04 00:37:02 +01:00
|
|
|
from ubuntutools.lp.udtexceptions import PackageNotFoundException
|
2009-08-04 13:43:31 +02:00
|
|
|
|
2009-08-22 17:39:38 +02:00
|
|
|
__all__ = [
|
2011-09-10 15:42:40 +02:00
|
|
|
'get_debian_srcpkg',
|
|
|
|
'get_ubuntu_srcpkg',
|
|
|
|
'need_sponsorship',
|
|
|
|
'check_existing_reports',
|
|
|
|
'get_ubuntu_delta_changelog',
|
|
|
|
'mail_bug',
|
2010-12-23 00:01:39 +02:00
|
|
|
]
|
2009-08-06 00:23:58 +02:00
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def _get_srcpkg(distro, name, release):
|
2011-03-06 00:48:24 +02:00
|
|
|
if distro == 'debian':
|
|
|
|
# Canonicalise release:
|
|
|
|
debian_info = DebianDistroInfo()
|
|
|
|
release = debian_info.codename(release, default=release)
|
|
|
|
|
2011-01-15 20:54:45 +02:00
|
|
|
lines = list(rmadison(distro, name, suite=release, arch='source'))
|
|
|
|
if not lines:
|
2010-12-23 00:01:39 +02:00
|
|
|
raise PackageNotFoundException("'%s' doesn't appear to exist "
|
|
|
|
"in %s '%s'"
|
|
|
|
% (name, distro.capitalize(), release))
|
2011-01-15 20:54:45 +02:00
|
|
|
pkg = max(lines, key=lambda x: Version(x['version']))
|
2009-08-06 00:23:58 +02:00
|
|
|
|
2011-11-13 22:50:34 +02:00
|
|
|
return FakeSPPH(pkg['source'], pkg['version'], pkg['component'], distro)
|
2009-08-06 00:23:58 +02:00
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def get_debian_srcpkg(name, release):
|
|
|
|
return _get_srcpkg('debian', name, release)
|
2009-08-06 00:23:58 +02:00
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def get_ubuntu_srcpkg(name, release):
|
|
|
|
return _get_srcpkg('ubuntu', name, release)
|
2009-08-07 12:53:33 +02:00
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def need_sponsorship(name, component, release):
|
2010-12-22 23:04:29 +02:00
|
|
|
'''
|
|
|
|
Ask the user if he has upload permissions for the package or the
|
|
|
|
component.
|
|
|
|
'''
|
|
|
|
|
2011-11-13 20:15:19 +02:00
|
|
|
val = YesNoQuestion().ask("Do you have upload permissions for the "
|
|
|
|
"'%s' component or the package '%s' in "
|
|
|
|
"Ubuntu %s?\n"
|
|
|
|
"If in doubt answer 'n'."
|
|
|
|
% (component, name, release), 'no')
|
|
|
|
return val == 'no'
|
2009-08-22 17:12:55 +02:00
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def check_existing_reports(srcpkg):
|
2010-12-22 23:04:29 +02:00
|
|
|
'''
|
|
|
|
Point the user to the URL to manually check for duplicate bug reports.
|
|
|
|
'''
|
2010-12-23 00:01:39 +02:00
|
|
|
print ('Please check on '
|
|
|
|
'https://bugs.launchpad.net/ubuntu/+source/%s/+bugs\n'
|
|
|
|
'for duplicate sync requests before continuing.' % srcpkg)
|
2011-11-13 20:15:19 +02:00
|
|
|
confirmation_prompt()
|
2009-08-22 17:39:38 +02:00
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def get_ubuntu_delta_changelog(srcpkg):
|
2011-09-10 10:28:41 +02:00
|
|
|
'''
|
|
|
|
Download the Ubuntu changelog and extract the entries since the last sync
|
|
|
|
from Debian.
|
|
|
|
'''
|
2011-11-13 22:50:34 +02:00
|
|
|
changelog = Changelog(srcpkg.getChangelog())
|
2011-09-10 10:28:41 +02:00
|
|
|
if changelog is None:
|
|
|
|
return u''
|
|
|
|
delta = []
|
|
|
|
debian_info = DebianDistroInfo()
|
|
|
|
for block in changelog:
|
|
|
|
distribution = block.distributions.split()[0].split('-')[0]
|
|
|
|
if debian_info.valid(distribution):
|
|
|
|
break
|
|
|
|
delta += [unicode(change) for change in block.changes()
|
|
|
|
if change.strip()]
|
|
|
|
|
|
|
|
return u'\n'.join(delta)
|
|
|
|
|
2011-09-10 15:42:40 +02:00
|
|
|
def mail_bug(srcpkg, subscribe, status, bugtitle, bugtext, bug_mail_domain,
|
|
|
|
keyid, myemailaddr, mailserver_host, mailserver_port,
|
|
|
|
mailserver_user, mailserver_pass):
|
2010-12-22 23:04:29 +02:00
|
|
|
'''
|
|
|
|
Submit the sync request per email.
|
|
|
|
'''
|
|
|
|
|
2011-02-01 10:10:31 +02:00
|
|
|
to = 'new@' + bug_mail_domain
|
2010-12-22 23:04:29 +02:00
|
|
|
|
|
|
|
# generate mailbody
|
|
|
|
if srcpkg:
|
|
|
|
mailbody = ' affects ubuntu/%s\n' % srcpkg
|
|
|
|
else:
|
|
|
|
mailbody = ' affects ubuntu\n'
|
|
|
|
mailbody += '''\
|
2009-08-22 17:12:55 +02:00
|
|
|
status %s
|
|
|
|
importance wishlist
|
|
|
|
subscribe %s
|
|
|
|
done
|
|
|
|
|
|
|
|
%s''' % (status, subscribe, bugtext)
|
2010-12-03 00:10:41 +01:00
|
|
|
|
2010-12-22 23:04:29 +02:00
|
|
|
# prepare sign command
|
|
|
|
gpg_command = None
|
2011-09-09 20:03:47 +02:00
|
|
|
for cmd in ('gnome-gpg', 'gpg2', 'gpg'):
|
2010-12-22 23:04:29 +02:00
|
|
|
if os.access('/usr/bin/%s' % cmd, os.X_OK):
|
|
|
|
gpg_command = [cmd]
|
2011-09-09 20:03:47 +02:00
|
|
|
break
|
|
|
|
|
|
|
|
if not gpg_command:
|
2011-09-10 00:50:48 +02:00
|
|
|
Logger.error("Cannot locate gpg, please install the 'gnupg' package!")
|
2011-09-09 20:03:47 +02:00
|
|
|
sys.exit(1)
|
2009-08-22 17:12:55 +02:00
|
|
|
|
2010-12-22 23:04:29 +02:00
|
|
|
gpg_command.append('--clearsign')
|
|
|
|
if keyid:
|
|
|
|
gpg_command.extend(('-u', keyid))
|
2009-08-22 17:12:55 +02:00
|
|
|
|
2010-12-22 23:04:29 +02:00
|
|
|
# sign the mail body
|
2010-12-23 00:01:39 +02:00
|
|
|
gpg = subprocess.Popen(gpg_command, stdin=subprocess.PIPE,
|
|
|
|
stdout=subprocess.PIPE)
|
2010-12-22 23:04:29 +02:00
|
|
|
signed_report = gpg.communicate(mailbody.encode('utf-8'))[0].decode('utf-8')
|
2011-09-09 20:03:47 +02:00
|
|
|
if gpg.returncode != 0:
|
2011-09-10 00:50:48 +02:00
|
|
|
Logger.error("%s failed.", gpg_command[0])
|
2011-09-09 20:03:47 +02:00
|
|
|
sys.exit(1)
|
2009-08-22 17:12:55 +02:00
|
|
|
|
2010-12-22 23:04:29 +02:00
|
|
|
# generate email
|
|
|
|
mail = u'''\
|
2009-08-22 17:12:55 +02:00
|
|
|
From: %s
|
|
|
|
To: %s
|
|
|
|
Subject: %s
|
|
|
|
Content-Type: text/plain; charset=UTF-8
|
|
|
|
|
|
|
|
%s''' % (myemailaddr, to, bugtitle, signed_report)
|
|
|
|
|
2010-12-22 23:04:29 +02:00
|
|
|
print 'The final report is:\n%s' % mail
|
2011-11-13 20:15:19 +02:00
|
|
|
confirmation_prompt()
|
2010-12-22 23:04:29 +02:00
|
|
|
|
2011-11-28 20:50:28 +01:00
|
|
|
# save mail in temporary file
|
2011-12-03 18:09:49 +01:00
|
|
|
f=open("/tmp/requestsync-" + re.sub("[^a-zA-Z0-9_\-]","",bugtitle.replace(" ","_")), "w")
|
2011-11-28 20:50:28 +01:00
|
|
|
f.write(mail)
|
|
|
|
f.close()
|
|
|
|
|
2011-11-29 20:20:25 +01:00
|
|
|
Logger.normal('The e-mail has been saved in %s and will be deleted '
|
|
|
|
'after succesful transmission', f.name)
|
2011-11-28 20:50:28 +01:00
|
|
|
|
2010-12-22 23:04:29 +02:00
|
|
|
# connect to the server
|
2011-11-27 22:14:05 +01:00
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
Logger.info('Connecting to %s:%s ...', mailserver_host, mailserver_port)
|
|
|
|
s = smtplib.SMTP(mailserver_host, mailserver_port)
|
|
|
|
break
|
|
|
|
except socket.error, s:
|
|
|
|
Logger.error('Could not connect to %s:%s: %s (%i)',
|
|
|
|
mailserver_host, mailserver_port, s[1], s[0])
|
|
|
|
return
|
|
|
|
except smtplib.SMTPConnectError, s:
|
|
|
|
Logger.error('Could not connect to %s:%s: %s (%i)',
|
|
|
|
mailserver_host, mailserver_port, s[1], s[0])
|
2011-11-28 19:53:23 +01:00
|
|
|
if s.smtp_code == 421:
|
2011-11-28 20:50:28 +01:00
|
|
|
confirmation_prompt(message='This is a temporary error, press '
|
|
|
|
'[Enter] to retry. Press [Ctrl-C] to abort now.')
|
2010-12-22 23:04:29 +02:00
|
|
|
|
|
|
|
if mailserver_user and mailserver_pass:
|
|
|
|
try:
|
|
|
|
s.login(mailserver_user, mailserver_pass)
|
|
|
|
except smtplib.SMTPAuthenticationError:
|
2011-09-09 20:03:47 +02:00
|
|
|
Logger.error('Error authenticating to the server: '
|
|
|
|
'invalid username and password.')
|
2010-12-22 23:04:29 +02:00
|
|
|
s.quit()
|
|
|
|
return
|
|
|
|
except:
|
2011-09-09 20:03:47 +02:00
|
|
|
Logger.error('Unknown SMTP error.')
|
2010-12-22 23:04:29 +02:00
|
|
|
s.quit()
|
|
|
|
return
|
|
|
|
|
2011-12-03 11:57:18 +01:00
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
s.sendmail(myemailaddr, to, mail.encode('utf-8'))
|
|
|
|
s.quit()
|
|
|
|
os.remove(f.name)
|
|
|
|
Logger.normal('Sync request mailed.')
|
|
|
|
break
|
|
|
|
except smtplib.SMTPRecipientsRefused, smtperror:
|
|
|
|
smtp_code, smtp_message = smtperror.recipients[to]
|
|
|
|
Logger.error('Error while sending: %i, %s', smtp_code, smtp_message)
|
|
|
|
if smtp_code == 450:
|
|
|
|
confirmation_prompt(message='This is a temporary error, press '
|
|
|
|
'[Enter] to retry. Press [Ctrl-C] to abort now.')
|
|
|
|
else:
|
|
|
|
return
|
2011-12-03 18:01:55 +01:00
|
|
|
except smtplib.SMTPServerDisconnected:
|
|
|
|
Logger.error('Server disconnected while sending the mail.')
|
|
|
|
return
|