/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
1
# Copyright (C) 2006-2017 Canonical Ltd
0.4.4 by Martin Pool
Start forming xmlrpc requests
2
#
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
7
#
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
2052.3.1 by John Arbash Meinel
Add tests to cleanup the copyright of all source files
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
0.4.4 by Martin Pool
Start forming xmlrpc requests
12
#
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
4183.7.1 by Sabin Iacob
update FSF mailing address
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
0.4.4 by Martin Pool
Start forming xmlrpc requests
16
6379.6.3 by Jelmer Vernooij
Use absolute_import.
17
from __future__ import absolute_import
18
0.4.4 by Martin Pool
Start forming xmlrpc requests
19
0.4.17 by Martin Pool
Allow xmlrpc service url to be overridden by $BZR_LP_XMLRPC_URL
20
import os
4505.4.1 by Jonathan Lange
Trap gaierror and reraise appropriate ConnectionError.
21
import socket
6791.2.3 by Jelmer Vernooij
Fix more imports.
22
try:
23
    from urllib.parse import urlsplit, urlunsplit
24
except ImportError:
7143.11.1 by Jelmer Vernooij
Remove some unused imports.
25
    from urlparse import urlsplit, urlunsplit  # noqa: F401
0.4.29 by Martin Pool
(register-branch) override xmlrpc user-agent; move Transport construction
26
import urllib
6973.12.9 by Jelmer Vernooij
More fixes.
27
try:
28
    from xmlrpc.client import (
29
        __version__ as xmlrpc_version,
30
        Fault,
31
        ProtocolError,
32
        ServerProxy,
33
        Transport,
34
        )
35
except ImportError:  # python < 3
36
    from xmlrpclib import (
37
        __version__ as xmlrpc_version,
38
        Fault,
39
        ProtocolError,
40
        Transport,
41
        ServerProxy,
42
        )
0.4.13 by Martin Pool
Update xmlrpc api to pass product name as a parameter.
43
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
44
from ... import (
2900.2.21 by Vincent Ladeuil
Make lp_registration aware of authentication config.
45
    errors,
4505.6.3 by Jonathan Lange
Don't bother lazy importing.
46
    urlutils,
6622.1.34 by Jelmer Vernooij
Rename brzlib => breezy.
47
    __version__ as _breezy_version,
2900.2.21 by Vincent Ladeuil
Make lp_registration aware of authentication config.
48
    )
6624 by Jelmer Vernooij
Merge Python3 porting work ('py3 pokes')
49
from ...transport.http import _urllib2_wrappers
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
50
4505.6.5 by Jonathan Lange
Factor out some code that guesses a branch's URL.
51
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
52
# for testing, do
53
'''
6622.1.28 by Jelmer Vernooij
More renames; commands in output, environment variables.
54
export BRZ_LP_XMLRPC_URL=http://xmlrpc.staging.launchpad.net/bazaar/
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
55
'''
0.4.13 by Martin Pool
Update xmlrpc api to pass product name as a parameter.
56
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
57
58
class InvalidURL(errors.PathError):
59
60
    _fmt = 'Invalid url supplied to transport: "%(path)s"%(extra)s'
61
62
3193.5.2 by Tim Penhey
Updated the tests to handle unknown launchpad instances.
63
class InvalidLaunchpadInstance(errors.BzrError):
64
65
    _fmt = "%(lp_instance)s is not a valid Launchpad instance."
66
67
    def __init__(self, lp_instance):
68
        errors.BzrError.__init__(self, lp_instance=lp_instance)
69
70
3955.3.5 by Jonathan Lange
Add an untested plugin, make the error handling a little nicer.
71
class NotLaunchpadBranch(errors.BzrError):
72
4031.2.8 by Jonathan Lange
Say "registered on", not "hosted on".
73
    _fmt = "%(url)s is not registered on Launchpad."
3955.3.5 by Jonathan Lange
Add an untested plugin, make the error handling a little nicer.
74
75
    def __init__(self, url):
76
        errors.BzrError.__init__(self, url=url)
77
78
6973.12.9 by Jelmer Vernooij
More fixes.
79
class XMLRPCTransport(Transport):
4776.2.2 by Vincent Ladeuil
Start testing the XMLRPC transport re-implemented on top of _urllib2_wrappers.
80
4776.3.1 by Vincent Ladeuil
Fix python2.4 compatibility with xmlrpclib.
81
    def __init__(self, scheme):
6973.12.9 by Jelmer Vernooij
More fixes.
82
        Transport.__init__(self)
4776.2.2 by Vincent Ladeuil
Start testing the XMLRPC transport re-implemented on top of _urllib2_wrappers.
83
        self._scheme = scheme
84
        self._opener = _urllib2_wrappers.Opener()
85
        self.verbose = 0
86
87
    def request(self, host, handler, request_body, verbose=0):
88
        self.verbose = verbose
89
        url = self._scheme + "://" + host + handler
90
        request = _urllib2_wrappers.Request("POST", url, request_body)
6883.17.1 by Jelmer Vernooij
Only add default HTTP headers if they're not present yet.
91
        request.add_header("User-Agent", self.user_agent)
4776.2.2 by Vincent Ladeuil
Start testing the XMLRPC transport re-implemented on top of _urllib2_wrappers.
92
        request.add_header("Content-Type", "text/xml")
93
94
        response = self._opener.open(request)
95
        if response.code != 200:
6973.12.9 by Jelmer Vernooij
More fixes.
96
            raise ProtocolError(host + handler, response.code,
97
                                response.msg, response.info())
4776.2.2 by Vincent Ladeuil
Start testing the XMLRPC transport re-implemented on top of _urllib2_wrappers.
98
        return self.parse_response(response)
99
100
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
101
class LaunchpadService(object):
0.4.27 by Martin Pool
doc
102
    """A service to talk to Launchpad via XMLRPC.
3193.5.2 by Tim Penhey
Updated the tests to handle unknown launchpad instances.
103
6622.1.33 by Jelmer Vernooij
Fix more tests (all?)
104
    See http://wiki.bazaar.canonical.com/Specs/LaunchpadRpc for the methods we
105
    can call.
0.4.27 by Martin Pool
doc
106
    """
0.4.6 by Martin Pool
Put the rest of the parameters into the registration request.
107
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
108
    LAUNCHPAD_DOMAINS = {
109
        'production': 'launchpad.net',
110
        'staging': 'staging.launchpad.net',
5615.2.1 by Jelmer Vernooij
Support the 'qastaging' instance of Launchpad.
111
        'qastaging': 'qastaging.launchpad.net',
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
112
        'demo': 'demo.launchpad.net',
113
        'dev': 'launchpad.dev',
114
        }
115
3211.1.1 by Ian Clatworthy
Extends the launchpad plugin's implementation of lp spec urls (Tim Penhey)
116
    # NB: these should always end in a slash to avoid xmlrpclib appending
0.4.7 by Martin Pool
Start making provision to test using a mock xmlrpc transport.
117
    # '/RPC2'
3955.3.2 by Jonathan Lange
Tighten up the code a little, changing the dev service to use https,
118
    LAUNCHPAD_INSTANCE = {}
6656.1.1 by Martin
Apply 2to3 dict fixer and clean up resulting mess using view helpers
119
    for instance, domain in LAUNCHPAD_DOMAINS.items():
3955.3.2 by Jonathan Lange
Tighten up the code a little, changing the dev service to use https,
120
        LAUNCHPAD_INSTANCE[instance] = 'https://xmlrpc.%s/bazaar/' % domain
121
4634.165.3 by Vincent Ladeuil
Delete all references to edge.launchpad.net in code and associated tests.
122
    # We use production as the default because edge has been deprecated circa
123
    # 2010-11 (see bug https://bugs.launchpad.net/bzr/+bug/583667)
5243.1.1 by Martin
Use the production server in the launchpad plugin rather than edge
124
    DEFAULT_INSTANCE = 'production'
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
125
    DEFAULT_SERVICE_URL = LAUNCHPAD_INSTANCE[DEFAULT_INSTANCE]
0.4.13 by Martin Pool
Update xmlrpc api to pass product name as a parameter.
126
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
127
    transport = None
128
    registrant_email = None
129
    registrant_password = None
130
3193.5.1 by Tim Penhey
Mostly working, just need to update the tests for lp://dev
131
    def __init__(self, transport=None, lp_instance=None):
0.4.23 by Martin Pool
(register-branch) fix ordering of parameters and restore transport-level test.
132
        """Construct a new service talking to the launchpad rpc server"""
3193.5.1 by Tim Penhey
Mostly working, just need to update the tests for lp://dev
133
        self._lp_instance = lp_instance
0.4.29 by Martin Pool
(register-branch) override xmlrpc user-agent; move Transport construction
134
        if transport is None:
6973.12.11 by Jelmer Vernooij
Fix some more tests.
135
            uri_type = urlutils.parse_url(self.service_url)[0]
4776.2.6 by Vincent Ladeuil
Fixed as per review comments.
136
            transport = XMLRPCTransport(uri_type)
6973.12.9 by Jelmer Vernooij
More fixes.
137
            transport.user_agent = 'Breezy/%s (xmlrpc/%s)' \
7143.15.2 by Jelmer Vernooij
Run autopep8.
138
                % (_breezy_version, xmlrpc_version)
0.4.29 by Martin Pool
(register-branch) override xmlrpc user-agent; move Transport construction
139
        self.transport = transport
140
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
141
    @property
142
    def service_url(self):
143
        """Return the http or https url for the xmlrpc server.
144
145
        This does not include the username/password credentials.
146
        """
6622.1.28 by Jelmer Vernooij
More renames; commands in output, environment variables.
147
        key = 'BRZ_LP_XMLRPC_URL'
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
148
        if key in os.environ:
149
            return os.environ[key]
3193.5.1 by Tim Penhey
Mostly working, just need to update the tests for lp://dev
150
        elif self._lp_instance is not None:
3193.5.2 by Tim Penhey
Updated the tests to handle unknown launchpad instances.
151
            try:
152
                return self.LAUNCHPAD_INSTANCE[self._lp_instance]
153
            except KeyError:
154
                raise InvalidLaunchpadInstance(self._lp_instance)
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
155
        else:
156
            return self.DEFAULT_SERVICE_URL
157
4505.6.2 by Jonathan Lange
Extract a method that gets a service from a URL.
158
    @classmethod
159
    def for_url(cls, url, **kwargs):
160
        """Return the Launchpad service corresponding to the given URL."""
161
        result = urlsplit(url)
162
        lp_instance = result[1]
163
        if lp_instance == '':
164
            lp_instance = None
165
        elif lp_instance not in cls.LAUNCHPAD_INSTANCE:
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
166
            raise InvalidURL(url)
4505.6.2 by Jonathan Lange
Extract a method that gets a service from a URL.
167
        return cls(lp_instance=lp_instance, **kwargs)
168
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
169
    def get_proxy(self):
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
170
        """Return the proxy for XMLRPC requests."""
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
171
        url = self.service_url
6973.12.9 by Jelmer Vernooij
More fixes.
172
        return ServerProxy(url, transport=self.transport)
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
173
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
174
    def send_request(self, method_name, method_params):
175
        proxy = self.get_proxy()
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
176
        method = getattr(proxy, method_name)
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
177
        try:
178
            result = method(*method_params)
6973.12.9 by Jelmer Vernooij
More fixes.
179
        except ProtocolError as e:
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
180
            if e.errcode == 301:
181
                # TODO: This can give a ProtocolError representing a 301 error, whose
182
                # e.headers['location'] tells where to go and e.errcode==301; should
183
                # probably log something and retry on the new url.
184
                raise NotImplementedError("should resend request to %s, but this isn't implemented"
7143.15.2 by Jelmer Vernooij
Run autopep8.
185
                                          % e.headers.get('Location', 'NO-LOCATION-PRESENT'))
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
186
            else:
187
                # we don't want to print the original message because its
188
                # str representation includes the plaintext password.
189
                # TODO: print more headers to help in tracking down failures
190
                raise errors.BzrError("xmlrpc protocol error connecting to %s: %s %s"
7143.15.2 by Jelmer Vernooij
Run autopep8.
191
                                      % (self.service_url, e.errcode, e.errmsg))
6619.3.2 by Jelmer Vernooij
Apply 2to3 except fix.
192
        except socket.gaierror as e:
4505.4.1 by Jonathan Lange
Trap gaierror and reraise appropriate ConnectionError.
193
            raise errors.ConnectionError(
194
                "Could not resolve '%s'" % self.domain,
195
                orig_error=e)
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
196
        return result
197
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
198
    @property
199
    def domain(self):
200
        if self._lp_instance is None:
201
            instance = self.DEFAULT_INSTANCE
202
        else:
203
            instance = self._lp_instance
204
        return self.LAUNCHPAD_DOMAINS[instance]
205
4505.6.5 by Jonathan Lange
Factor out some code that guesses a branch's URL.
206
    def _guess_branch_path(self, branch_url, _request_factory=None):
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
207
        scheme, hostinfo, path = urlsplit(branch_url)[:3]
3955.3.8 by Jonathan Lange
Support lp URL shortcuts.
208
        if _request_factory is None:
209
            _request_factory = ResolveLaunchpadPathRequest
210
        if scheme == 'lp':
211
            resolve = _request_factory(path)
3955.3.9 by Jonathan Lange
Catch errors.
212
            try:
213
                result = resolve.submit(self)
6973.12.9 by Jelmer Vernooij
More fixes.
214
            except Fault as fault:
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
215
                raise InvalidURL(branch_url, str(fault))
3955.3.8 by Jonathan Lange
Support lp URL shortcuts.
216
            branch_url = result['urls'][0]
217
            path = urlsplit(branch_url)[2]
218
        else:
219
            domains = (
220
                'bazaar.%s' % domain
6656.1.1 by Martin
Apply 2to3 dict fixer and clean up resulting mess using view helpers
221
                for domain in self.LAUNCHPAD_DOMAINS.values())
3955.3.5 by Jonathan Lange
Add an untested plugin, make the error handling a little nicer.
222
            if hostinfo not in domains:
223
                raise NotLaunchpadBranch(branch_url)
4505.6.5 by Jonathan Lange
Factor out some code that guesses a branch's URL.
224
        return path.lstrip('/')
225
226
    def get_web_url_from_branch_url(self, branch_url, _request_factory=None):
227
        """Get the Launchpad web URL for the given branch URL.
228
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
229
        :raise InvalidURL: if 'branch_url' cannot be identified as a
4505.6.5 by Jonathan Lange
Factor out some code that guesses a branch's URL.
230
            Launchpad branch URL.
231
        :return: The URL of the branch on Launchpad.
232
        """
233
        path = self._guess_branch_path(branch_url, _request_factory)
3955.3.7 by Jonathan Lange
Test the launchpad-open command. Fix up some minor bugs.
234
        return urlutils.join('https://code.%s' % self.domain, path)
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
235
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
236
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
237
class BaseRequest(object):
238
    """Base request for talking to a XMLRPC server."""
239
240
    # Set this to the XMLRPC method name.
241
    _methodname = None
242
243
    def _request_params(self):
244
        """Return the arguments to pass to the method"""
245
        raise NotImplementedError(self._request_params)
246
247
    def submit(self, service):
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
248
        """Submit request to Launchpad XMLRPC server.
249
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
250
        :param service: LaunchpadService indicating where to send the request.
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
251
        """
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
252
        return service.send_request(self._methodname, self._request_params())
0.4.19 by test at canonical
add possibility to link to a bug when registering a branch. factor out some common functionality from BranchRegistrationRequest.
253
254
1668.1.12 by Martin Pool
(launchpad plugin) Improved --dry-run that uses a dummy xmlrpc service.
255
class DryRunLaunchpadService(LaunchpadService):
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
256
    """Service that just absorbs requests without sending to server."""
257
258
    def send_request(self, method_name, method_params):
259
        pass
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
260
261
2898.4.3 by James Henstridge
Make launchpad_transport_indirect() use XMLRPC to resolve the lp: URL.
262
class ResolveLaunchpadPathRequest(BaseRequest):
263
    """Request to resolve the path component of an lp: URL."""
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
264
2898.4.3 by James Henstridge
Make launchpad_transport_indirect() use XMLRPC to resolve the lp: URL.
265
    _methodname = 'resolve_lp_path'
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
266
267
    def __init__(self, path):
3246.4.1 by Daniel Watkins
Replaced problematic assertion with exception call.
268
        if not path:
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
269
            raise InvalidURL(url=path, extra="You must specify a project.")
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
270
        self.path = path
271
272
    def _request_params(self):
273
        """Return xmlrpc request parameters"""
274
        return (self.path,)