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."""
20
from urlparse import urlsplit
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
25
from urlparse import urlsplit, urlunsplit
23
28
from bzrlib import (
28
34
from bzrlib.transport import (
30
36
register_urlparse_netloc_protocol,
33
40
from bzrlib.plugins.launchpad.lp_registration import (
41
48
register_urlparse_netloc_protocol('lp')
44
class LaunchpadDirectory(object):
51
class LaunchpadTransport(Transport):
52
"""lp:/// URL transport
54
This transport redirects requests to the real branch location
55
after resolving the URL via an XMLRPC request to Launchpad.
58
def __init__(self, base):
59
super(LaunchpadTransport, self).__init__(base)
60
# We only support URLs without a netloc
61
self.lp_instance = urlsplit(base)[1]
62
if self.lp_instance == '':
63
self.lp_instance = None
64
elif self.lp_instance not in LaunchpadService.LAUNCHPAD_INSTANCE:
65
raise errors.InvalidURL(path=base)
46
67
def _requires_launchpad_login(self, scheme, netloc, path, query,
54
75
and (netloc.endswith('launchpad.net')
55
76
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,
78
def _resolve(self, abspath,
62
79
_request_factory=ResolveLaunchpadPathRequest,
64
81
"""Resolve the base URL for this transport."""
65
service = LaunchpadService.for_url(url)
66
result = urlsplit(url)
67
resolve = _request_factory(result[2].strip('/'))
82
path = urlsplit(abspath)[2].lstrip('/')
83
# Perform an XMLRPC request to resolve the path
84
resolve = _request_factory(path)
85
service = LaunchpadService(lp_instance=self.lp_instance)
69
87
result = resolve.submit(service)
70
88
except xmlrpclib.Fault, fault:
71
89
raise errors.InvalidURL(
72
path=url, extra=fault.faultString)
90
path=abspath, extra=fault.faultString)
74
92
if 'launchpad' in debug.debug_flags:
75
trace.mutter("resolve_lp_path(%r) == %r", url, result)
93
trace.mutter("resolve_lp_path(%r) == %r", path, result)
77
95
if _lp_login is None:
78
96
_lp_login = get_lp_login()
80
97
for url in result['urls']:
81
98
scheme, netloc, path, query, fragment = urlsplit(url)
82
99
if self._requires_launchpad_login(scheme, netloc, path, query,
84
101
# Only accept launchpad.net bzr+ssh URLs if we know
85
102
# the user's Launchpad login:
86
if _lp_login is not None:
88
103
if _lp_login is 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".')
105
url = urlunsplit((scheme, '%s@%s' % (_lp_login, netloc),
106
path, query, fragment))
95
109
# Use the URL if we can create a transport for it.
103
raise errors.InvalidURL(path=url, extra='no supported schemes')
117
raise errors.InvalidURL(path=abspath,
118
extra='no supported schemes')
121
def _request_redirect(self, relpath):
122
source = urlutils.join(self.base, relpath)
123
# Split the source location into the branch location, and the
124
# extra path components.
125
pos = source.find('/.bzr/')
127
branchpath = source[:pos]
132
target = self._resolve(branchpath) + extra
133
raise errors.RedirectRequested(
137
def get(self, relpath):
138
"""See Transport.get()."""
139
self._request_redirect(relpath)
141
def mkdir(self, relpath, mode=None):
142
"""See Transport.mkdir()."""
143
self._request_redirect(relpath)
107
146
def get_test_permutations():
108
147
# Since this transport doesn't do anything once opened, it's not subjected