13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18
"""Directory lookup that uses Launchpad."""
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18
"""Transport indirection that uses Launchpad as a directory lookup.
20
When the transport is opened, it immediately redirects to a url
21
on Launchpad, which can then either serve the branch itself or redirect
20
25
from urlparse import urlsplit
23
28
from bzrlib import (
28
31
from bzrlib.transport import (
30
register_urlparse_netloc_protocol,
33
36
from bzrlib.plugins.launchpad.lp_registration import (
34
37
LaunchpadService, ResolveLaunchpadPathRequest)
35
from bzrlib.plugins.launchpad.account import get_lp_login
38
# As bzrlib.transport.remote may not be loaded yet, make sure bzr+ssh
39
# is counted as a netloc protocol.
40
register_urlparse_netloc_protocol('bzr+ssh')
41
register_urlparse_netloc_protocol('lp')
44
class LaunchpadDirectory(object):
46
def _requires_launchpad_login(self, scheme, netloc, path, query,
48
"""Does the URL require a Launchpad login in order to be reached?
50
The URL is specified by its parsed components, as returned from
53
return (scheme in ('bzr+ssh', 'sftp')
54
and (netloc.endswith('launchpad.net')
55
or netloc.endswith('launchpad.dev')))
57
def look_up(self, name, url):
58
"""See DirectoryService.look_up"""
59
return self._resolve(url)
61
def _resolve(self, url,
62
_request_factory=ResolveLaunchpadPathRequest,
64
"""Resolve the base URL for this transport."""
65
service = LaunchpadService.for_url(url)
66
result = urlsplit(url)
67
resolve = _request_factory(result[2].strip('/'))
69
result = resolve.submit(service)
70
except xmlrpclib.Fault, fault:
71
raise errors.InvalidURL(
72
path=url, extra=fault.faultString)
74
if 'launchpad' in debug.debug_flags:
75
trace.mutter("resolve_lp_path(%r) == %r", url, result)
78
_lp_login = get_lp_login()
80
for url in result['urls']:
81
scheme, netloc, path, query, fragment = urlsplit(url)
82
if self._requires_launchpad_login(scheme, netloc, path, query,
84
# Only accept launchpad.net bzr+ssh URLs if we know
85
# the user's Launchpad login:
86
if _lp_login is not None:
91
'You have not informed bzr of your Launchpad ID, and you must do this to\n'
92
'write to Launchpad or access private data. See "bzr help launchpad-login".')
95
# Use the URL if we can create a transport for it.
98
except (errors.PathError, errors.TransportError):
103
raise errors.InvalidURL(path=url, extra='no supported schemes')
40
def launchpad_transport_indirect(
41
base_url, request_factory=ResolveLaunchpadPathRequest):
42
"""Uses Launchpad.net as a directory of open source software"""
43
if base_url.startswith('lp:///'):
45
elif base_url.startswith('lp:') and base_url[3] != '/':
48
raise errors.InvalidURL(path=base_url)
50
# Perform an XMLRPC request to resolve the path
51
resolve = request_factory(path)
52
service = LaunchpadService()
54
result = resolve.submit(service)
55
except xmlrpclib.Fault, fault:
56
raise errors.InvalidURL(
57
path=base_url, extra=fault.faultString)
59
for url in result['urls']:
60
scheme = urlsplit(url)[0]
61
# Once we have a way to know if the user can do bzr+ssh, we
62
# should allow it through here too.
63
if scheme in ['bzr+http', 'http']:
66
raise errors.InvalidURL(path=base_url,
67
extra='no supported schemes')
69
return get_transport(url)
107
72
def get_test_permutations():