/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
0.4.29 by Martin Pool
(register-branch) override xmlrpc user-agent; move Transport construction
131
3193.5.1 by Tim Penhey
Mostly working, just need to update the tests for lp://dev
132
    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.
133
        """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
134
        self._lp_instance = lp_instance
0.4.29 by Martin Pool
(register-branch) override xmlrpc user-agent; move Transport construction
135
        if transport is None:
6973.12.11 by Jelmer Vernooij
Fix some more tests.
136
            uri_type = urlutils.parse_url(self.service_url)[0]
4776.2.6 by Vincent Ladeuil
Fixed as per review comments.
137
            transport = XMLRPCTransport(uri_type)
6973.12.9 by Jelmer Vernooij
More fixes.
138
            transport.user_agent = 'Breezy/%s (xmlrpc/%s)' \
139
                    % (_breezy_version, xmlrpc_version)
0.4.29 by Martin Pool
(register-branch) override xmlrpc user-agent; move Transport construction
140
        self.transport = transport
141
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.
142
    @property
143
    def service_url(self):
144
        """Return the http or https url for the xmlrpc server.
145
146
        This does not include the username/password credentials.
147
        """
6622.1.28 by Jelmer Vernooij
More renames; commands in output, environment variables.
148
        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.
149
        if key in os.environ:
150
            return os.environ[key]
3193.5.1 by Tim Penhey
Mostly working, just need to update the tests for lp://dev
151
        elif self._lp_instance is not None:
3193.5.2 by Tim Penhey
Updated the tests to handle unknown launchpad instances.
152
            try:
153
                return self.LAUNCHPAD_INSTANCE[self._lp_instance]
154
            except KeyError:
155
                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.
156
        else:
157
            return self.DEFAULT_SERVICE_URL
158
4505.6.2 by Jonathan Lange
Extract a method that gets a service from a URL.
159
    @classmethod
160
    def for_url(cls, url, **kwargs):
161
        """Return the Launchpad service corresponding to the given URL."""
162
        result = urlsplit(url)
163
        lp_instance = result[1]
164
        if lp_instance == '':
165
            lp_instance = None
166
        elif lp_instance not in cls.LAUNCHPAD_INSTANCE:
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
167
            raise InvalidURL(url)
4505.6.2 by Jonathan Lange
Extract a method that gets a service from a URL.
168
        return cls(lp_instance=lp_instance, **kwargs)
169
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
170
    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.
171
        """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.
172
        url = self.service_url
6973.12.9 by Jelmer Vernooij
More fixes.
173
        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.
174
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
175
    def send_request(self, method_name, method_params):
176
        proxy = self.get_proxy()
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
177
        method = getattr(proxy, method_name)
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
178
        try:
179
            result = method(*method_params)
6973.12.9 by Jelmer Vernooij
More fixes.
180
        except ProtocolError as e:
1668.1.9 by Martin Pool
(launchpad plugin) Better reporting of errors from xmlrpc
181
            if e.errcode == 301:
182
                # TODO: This can give a ProtocolError representing a 301 error, whose
183
                # e.headers['location'] tells where to go and e.errcode==301; should
184
                # probably log something and retry on the new url.
185
                raise NotImplementedError("should resend request to %s, but this isn't implemented"
186
                        % e.headers.get('Location', 'NO-LOCATION-PRESENT'))
187
            else:
188
                # we don't want to print the original message because its
189
                # str representation includes the plaintext password.
190
                # TODO: print more headers to help in tracking down failures
191
                raise errors.BzrError("xmlrpc protocol error connecting to %s: %s %s"
192
                        % (self.service_url, e.errcode, e.errmsg))
6619.3.2 by Jelmer Vernooij
Apply 2to3 except fix.
193
        except socket.gaierror as e:
4505.4.1 by Jonathan Lange
Trap gaierror and reraise appropriate ConnectionError.
194
            raise errors.ConnectionError(
195
                "Could not resolve '%s'" % self.domain,
196
                orig_error=e)
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
197
        return result
198
3955.3.1 by Jonathan Lange
Start doing URL stuff, extracting the domain bit out of LaunchpadService,
199
    @property
200
    def domain(self):
201
        if self._lp_instance is None:
202
            instance = self.DEFAULT_INSTANCE
203
        else:
204
            instance = self._lp_instance
205
        return self.LAUNCHPAD_DOMAINS[instance]
206
4505.6.5 by Jonathan Lange
Factor out some code that guesses a branch's URL.
207
    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,
208
        scheme, hostinfo, path = urlsplit(branch_url)[:3]
3955.3.8 by Jonathan Lange
Support lp URL shortcuts.
209
        if _request_factory is None:
210
            _request_factory = ResolveLaunchpadPathRequest
211
        if scheme == 'lp':
212
            resolve = _request_factory(path)
3955.3.9 by Jonathan Lange
Catch errors.
213
            try:
214
                result = resolve.submit(self)
6973.12.9 by Jelmer Vernooij
More fixes.
215
            except Fault as fault:
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
216
                raise InvalidURL(branch_url, str(fault))
3955.3.8 by Jonathan Lange
Support lp URL shortcuts.
217
            branch_url = result['urls'][0]
218
            path = urlsplit(branch_url)[2]
219
        else:
220
            domains = (
221
                'bazaar.%s' % domain
6656.1.1 by Martin
Apply 2to3 dict fixer and clean up resulting mess using view helpers
222
                for domain in self.LAUNCHPAD_DOMAINS.values())
3955.3.5 by Jonathan Lange
Add an untested plugin, make the error handling a little nicer.
223
            if hostinfo not in domains:
224
                raise NotLaunchpadBranch(branch_url)
4505.6.5 by Jonathan Lange
Factor out some code that guesses a branch's URL.
225
        return path.lstrip('/')
226
227
    def get_web_url_from_branch_url(self, branch_url, _request_factory=None):
228
        """Get the Launchpad web URL for the given branch URL.
229
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
230
        :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.
231
            Launchpad branch URL.
232
        :return: The URL of the branch on Launchpad.
233
        """
234
        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.
235
        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,
236
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
237
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.
238
class BaseRequest(object):
239
    """Base request for talking to a XMLRPC server."""
240
241
    # Set this to the XMLRPC method name.
242
    _methodname = None
243
244
    def _request_params(self):
245
        """Return the arguments to pass to the method"""
246
        raise NotImplementedError(self._request_params)
247
248
    def submit(self, service):
0.4.21 by Martin Pool
Refactor BaseRequest.submit so details of submission are in the LaunchpadService
249
        """Submit request to Launchpad XMLRPC server.
250
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
251
        :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.
252
        """
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
253
        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.
254
255
1668.1.12 by Martin Pool
(launchpad plugin) Improved --dry-run that uses a dummy xmlrpc service.
256
class DryRunLaunchpadService(LaunchpadService):
6622.7.1 by Colin Watson
Remove `bzr register-branch`, since it has not worked for a long time.
257
    """Service that just absorbs requests without sending to server."""
258
259
    def send_request(self, method_name, method_params):
260
        pass
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
261
262
2898.4.3 by James Henstridge
Make launchpad_transport_indirect() use XMLRPC to resolve the lp: URL.
263
class ResolveLaunchpadPathRequest(BaseRequest):
264
    """Request to resolve the path component of an lp: URL."""
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
265
2898.4.3 by James Henstridge
Make launchpad_transport_indirect() use XMLRPC to resolve the lp: URL.
266
    _methodname = 'resolve_lp_path'
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
267
268
    def __init__(self, path):
3246.4.1 by Daniel Watkins
Replaced problematic assertion with exception call.
269
        if not path:
6729.6.1 by Jelmer Vernooij
Move urlutils errors.
270
            raise InvalidURL(url=path, extra="You must specify a project.")
2898.4.2 by James Henstridge
Add ResolveLaunchpadURLRequest() class to handle lp: URL resolution.
271
        self.path = path
272
273
    def _request_params(self):
274
        """Return xmlrpc request parameters"""
275
        return (self.path,)