/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
1
# Copyright (C) 2007-2011 Canonical Ltd
2245.8.3 by Martin Pool
Start adding indirection transport
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
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
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
2245.8.3 by Martin Pool
Start adding indirection transport
16
17
3251.4.3 by Aaron Bentley
More renames and cleanups
18
"""Directory lookup that uses Launchpad."""
2245.8.3 by Martin Pool
Start adding indirection transport
19
5121.2.1 by Jelmer Vernooij
Remove unused imports in lp_directory.
20
from urlparse import urlsplit
2898.4.4 by James Henstridge
Changes to account for modifications to the XMLRPC API.
21
import xmlrpclib
22
2245.8.4 by Martin Pool
lp:/// indirection works
23
from bzrlib import (
2898.4.14 by James Henstridge
* Use urlsplit() to process URLs.
24
    debug,
2245.8.4 by Martin Pool
lp:/// indirection works
25
    errors,
2898.4.14 by James Henstridge
* Use urlsplit() to process URLs.
26
    trace,
5609.9.1 by Martin
Blindly change all users of get_transport to address the function via the transport module
27
    transport,
2245.8.4 by Martin Pool
lp:/// indirection works
28
    )
6150.3.12 by Jonathan Riddell
add missing import
29
from bzrlib.i18n import gettext
2245.8.4 by Martin Pool
lp:/// indirection works
30
2898.4.7 by James Henstridge
Fix up tests.
31
from bzrlib.plugins.launchpad.lp_registration import (
32
    LaunchpadService, ResolveLaunchpadPathRequest)
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
33
from bzrlib.plugins.launchpad.account import get_lp_login
34
35
2898.4.9 by James Henstridge
Add some more tests for the bzr+ssh://bazaar.launchpad.net URL
36
# As bzrlib.transport.remote may not be loaded yet, make sure bzr+ssh
37
# is counted as a netloc protocol.
5609.9.1 by Martin
Blindly change all users of get_transport to address the function via the transport module
38
transport.register_urlparse_netloc_protocol('bzr+ssh')
39
transport.register_urlparse_netloc_protocol('lp')
2898.4.9 by James Henstridge
Add some more tests for the bzr+ssh://bazaar.launchpad.net URL
40
5462.4.5 by Barry Warsaw
We only need to include distroseries shortcuts.
41
_ubuntu_series_shortcuts = {
5462.4.3 by Barry Warsaw
* Add back Ubuntu distroseries shortcuts.
42
    'n': 'natty',
43
    'm': 'maverick',
44
    'l': 'lucid',
45
    'k': 'karmic',
46
    'j': 'jaunty',
47
    'h': 'hardy',
48
    'd': 'dapper',
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
49
    }
50
2898.4.9 by James Henstridge
Add some more tests for the bzr+ssh://bazaar.launchpad.net URL
51
3251.4.1 by Aaron Bentley
Convert LP transport into directory service
52
class LaunchpadDirectory(object):
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
53
3031.2.4 by jml at canonical
Only split the URL once.
54
    def _requires_launchpad_login(self, scheme, netloc, path, query,
55
                                  fragment):
56
        """Does the URL require a Launchpad login in order to be reached?
57
58
        The URL is specified by its parsed components, as returned from
59
        urlsplit.
60
        """
3031.2.3 by jml at canonical
Make the test pass -- don't include sftp URLs if there's no lp login.
61
        return (scheme in ('bzr+ssh', 'sftp')
62
                and (netloc.endswith('launchpad.net')
63
                     or netloc.endswith('launchpad.dev')))
3031.2.1 by jml at canonical
Factor out the method that determines if a URL is a LP url.
64
3251.4.1 by Aaron Bentley
Convert LP transport into directory service
65
    def look_up(self, name, url):
3251.4.5 by Aaron Bentley
Add docstring
66
        """See DirectoryService.look_up"""
3251.4.1 by Aaron Bentley
Convert LP transport into directory service
67
        return self._resolve(url)
68
5609.24.9 by John Arbash Meinel
A bunch more tests and bug fixes for the local resolution.
69
    def _resolve_locally(self, path, url, _request_factory):
70
        # This is the best I could work out about XMLRPC. If an lp: url
71
        # includes ~user, then it is specially validated. Otherwise, it is just
72
        # sent to +branch/$path.
73
        _, netloc, _, _, _ = urlsplit(url)
74
        if netloc == '':
75
            netloc = LaunchpadService.DEFAULT_INSTANCE
76
        base_url = LaunchpadService.LAUNCHPAD_DOMAINS[netloc]
77
        base = 'bzr+ssh://bazaar.%s/' % (base_url,)
78
        maybe_invalid = False
79
        if path.startswith('~'):
80
            # A ~user style path, validate it a bit.
81
            # If a path looks fishy, fall back to asking XMLRPC to
82
            # resolve it for us. That way we still get their nicer error
83
            # messages.
84
            parts = path.split('/')
85
            if (len(parts) < 3
86
                or (parts[1] in ('ubuntu', 'debian') and len(parts) < 5)):
87
                # This special case requires 5-parts to be valid.
88
                maybe_invalid = True
89
        else:
90
            base += '+branch/'
91
        if maybe_invalid:
92
            return self._resolve_via_xmlrpc(path, url, _request_factory)
93
        return {'urls': [base + path]}
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
94
95
    def _resolve_via_xmlrpc(self, path, url, _request_factory):
96
        service = LaunchpadService.for_url(url)
97
        resolve = _request_factory(path)
98
        try:
99
            result = resolve.submit(service)
100
        except xmlrpclib.Fault, fault:
101
            raise errors.InvalidURL(
102
                path=url, extra=fault.faultString)
103
        return result
104
105
    def _update_url_scheme(self, url):
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
106
        # Do ubuntu: and debianlp: expansions.
5462.4.10 by Vincent Ladeuil
Fix python2.4 compatibility.
107
        scheme, netloc, path, query, fragment = urlsplit(url)
108
        if scheme in ('ubuntu', 'debianlp'):
109
            if scheme == 'ubuntu':
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
110
                distro = 'ubuntu'
5462.4.5 by Barry Warsaw
We only need to include distroseries shortcuts.
111
                distro_series = _ubuntu_series_shortcuts
5462.4.10 by Vincent Ladeuil
Fix python2.4 compatibility.
112
            elif scheme == 'debianlp':
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
113
                distro = 'debian'
5462.4.5 by Barry Warsaw
We only need to include distroseries shortcuts.
114
                # No shortcuts for Debian distroseries.
115
                distro_series = {}
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
116
            else:
117
                raise AssertionError('scheme should be ubuntu: or debianlp:')
5462.4.3 by Barry Warsaw
* Add back Ubuntu distroseries shortcuts.
118
            # Split the path.  It's either going to be 'project' or
119
            # 'series/project', but recognize that it may be a series we don't
120
            # know about.
5462.4.10 by Vincent Ladeuil
Fix python2.4 compatibility.
121
            path_parts = path.split('/')
5462.4.3 by Barry Warsaw
* Add back Ubuntu distroseries shortcuts.
122
            if len(path_parts) == 1:
123
                # It's just a project name.
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
124
                lp_url_template = 'lp:%(distro)s/%(project)s'
125
                project = path_parts[0]
5462.4.3 by Barry Warsaw
* Add back Ubuntu distroseries shortcuts.
126
                series = None
127
            elif len(path_parts) == 2:
128
                # It's a series and project.
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
129
                lp_url_template = 'lp:%(distro)s/%(series)s/%(project)s'
5462.4.3 by Barry Warsaw
* Add back Ubuntu distroseries shortcuts.
130
                series, project = path_parts
131
            else:
132
                # There are either 0 or > 2 path parts, neither of which is
133
                # supported for these schemes.
6191.2.1 by Martin Pool
Avoid NameError when given an invalid ubuntu: launchpad url
134
                raise errors.InvalidURL('Bad path: %s' % url)
5462.4.3 by Barry Warsaw
* Add back Ubuntu distroseries shortcuts.
135
            # Expand any series shortcuts, but keep unknown series.
136
            series = distro_series.get(series, series)
5462.4.1 by Barry Warsaw
Added support for ubuntu: and debianlp: schemes, accessing the relevant
137
            # Hack the url and let the following do the final resolution.
138
            url = lp_url_template % dict(
139
                distro=distro,
140
                series=series,
141
                project=project)
5462.4.10 by Vincent Ladeuil
Fix python2.4 compatibility.
142
            scheme, netloc, path, query, fragment = urlsplit(url)
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
143
        return url, path
144
5609.24.8 by John Arbash Meinel
Fix the launchpad-login tests. And fix a bad commit typo.
145
    def _expand_user(self, path, url, lp_login):
5361.1.1 by John Arbash Meinel
Handle a simple ~ in lp: urls.
146
        if path.startswith('~/'):
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
147
            if lp_login is None:
5361.1.1 by John Arbash Meinel
Handle a simple ~ in lp: urls.
148
                raise errors.InvalidURL(path=url,
149
                    extra='Cannot resolve "~" to your username.'
150
                          ' See "bzr help launchpad-login"')
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
151
            path = '~' + lp_login + path[1:]
152
        return path
153
154
    def _resolve(self, url,
155
                 _request_factory=ResolveLaunchpadPathRequest,
156
                 _lp_login=None):
157
        """Resolve the base URL for this transport."""
158
        url, path = self._update_url_scheme(url)
159
        if _lp_login is None:
160
            _lp_login = get_lp_login()
161
        path = path.strip('/')
5609.24.8 by John Arbash Meinel
Fix the launchpad-login tests. And fix a bad commit typo.
162
        path = self._expand_user(path, url, _lp_login)
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
163
        if _lp_login is not None:
5609.24.9 by John Arbash Meinel
A bunch more tests and bug fixes for the local resolution.
164
            result = self._resolve_locally(path, url, _request_factory)
165
            if 'launchpad' in debug.debug_flags:
5609.24.11 by John Arbash Meinel
If someone uses -Dlaunchpad, use the 'official' URL instead of the local one.
166
                local_res = result
167
                result = self._resolve_via_xmlrpc(path, url, _request_factory)
6150.3.4 by Jonathan Riddell
more launchpad plugin gettext()ing
168
                trace.note(gettext(
169
                    'resolution for {0}\n  local: {1}\n remote: {2}').format(
6150.3.11 by Jonathan Riddell
syntax fixes
170
                           url, local_res['urls'], result['urls']))
5609.24.7 by John Arbash Meinel
Fix bug #397739. Avoid doing an XMLRPC lookup if a user
171
        else:
172
            result = self._resolve_via_xmlrpc(path, url, _request_factory)
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
173
2898.4.14 by James Henstridge
* Use urlsplit() to process URLs.
174
        if 'launchpad' in debug.debug_flags:
4505.6.2 by Jonathan Lange
Extract a method that gets a service from a URL.
175
            trace.mutter("resolve_lp_path(%r) == %r", url, result)
2898.4.14 by James Henstridge
* Use urlsplit() to process URLs.
176
3270.4.1 by James Westby
Warn the user when resolving lp: URLs if they haven't set their login.
177
        _warned_login = False
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
178
        for url in result['urls']:
179
            scheme, netloc, path, query, fragment = urlsplit(url)
3031.2.4 by jml at canonical
Only split the URL once.
180
            if self._requires_launchpad_login(scheme, netloc, path, query,
181
                                              fragment):
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
182
                # Only accept launchpad.net bzr+ssh URLs if we know
183
                # the user's Launchpad login:
3777.1.21 by Aaron Bentley
Stop including usernames in resolved lp: urls
184
                if _lp_login is not None:
185
                    break
2898.4.9 by James Henstridge
Add some more tests for the bzr+ssh://bazaar.launchpad.net URL
186
                if _lp_login is None:
3270.4.1 by James Westby
Warn the user when resolving lp: URLs if they haven't set their login.
187
                    if not _warned_login:
3834.1.2 by Martin Pool
Better message when launchpad-login is needed
188
                        trace.warning(
3834.1.3 by Martin Pool
Shorter message about launchpad-login
189
'You have not informed bzr of your Launchpad ID, and you must do this to\n'
190
'write to Launchpad or access private data.  See "bzr help launchpad-login".')
3270.4.1 by James Westby
Warn the user when resolving lp: URLs if they haven't set their login.
191
                        _warned_login = True
2898.4.15 by James Henstridge
Use get_transport() to decide whether Bazaar supports a given URL.
192
            else:
193
                # Use the URL if we can create a transport for it.
194
                try:
5609.9.1 by Martin
Blindly change all users of get_transport to address the function via the transport module
195
                    transport.get_transport(url)
2898.4.15 by James Henstridge
Use get_transport() to decide whether Bazaar supports a given URL.
196
                except (errors.PathError, errors.TransportError):
197
                    pass
198
                else:
199
                    break
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
200
        else:
3251.4.3 by Aaron Bentley
More renames and cleanups
201
            raise errors.InvalidURL(path=url, extra='no supported schemes')
2898.4.8 by James Henstridge
Switch lp: over to a pass-through transport, so that the XMLRPC gets
202
        return url
203
2245.8.3 by Martin Pool
Start adding indirection transport
204
205
def get_test_permutations():
206
    # Since this transport doesn't do anything once opened, it's not subjected
207
    # to the usual transport tests.
208
    return []