103
def _format_to_capabilities(self, repo_format):
104
rich_root = self._boolean_to_yes_no(repo_format.rich_root_data)
105
tree_ref = self._boolean_to_yes_no(
106
repo_format.supports_tree_reference)
107
external_lookup = self._boolean_to_yes_no(
108
repo_format.supports_external_lookups)
109
return rich_root, tree_ref, external_lookup
111
def _repo_relpath(self, current_transport, repository):
112
"""Get the relative path for repository from current_transport."""
113
# the relpath of the bzrdir in the found repository gives us the
114
# path segments to pop-out.
115
relpath = repository.user_transport.relpath(
116
current_transport.base)
118
segments = ['..'] * len(relpath.split('/'))
121
return '/'.join(segments)
124
class SmartServerBzrDirRequestCloningMetaDir(SmartServerRequestBzrDir):
126
def do_bzrdir_request(self, require_stacking):
127
"""Get the format that should be used when cloning from this dir.
131
:return: on success, a 3-tuple of network names for (control,
132
repository, branch) directories, where '' signifies "not present".
133
If this BzrDir contains a branch reference then this will fail with
134
BranchReference; clients should resolve branch references before
138
branch_ref = self._bzrdir.get_branch_reference()
139
except errors.NotBranchError:
141
if branch_ref is not None:
142
# The server shouldn't try to resolve references, and it quite
143
# possibly can't reach them anyway. The client needs to resolve
144
# the branch reference to determine the cloning_metadir.
145
return FailedSmartServerResponse(('BranchReference',))
146
if require_stacking == "True":
147
require_stacking = True
149
require_stacking = False
150
control_format = self._bzrdir.cloning_metadir(
151
require_stacking=require_stacking)
152
control_name = control_format.network_name()
153
# XXX: There should be a method that tells us that the format does/does
154
# not have subformats.
155
if isinstance(control_format, BzrDirMetaFormat1):
156
branch_name = ('branch',
157
control_format.get_branch_format().network_name())
158
repository_name = control_format.repository_format.network_name()
160
# Only MetaDir has delegated formats today.
161
branch_name = ('branch', '')
163
return SuccessfulSmartServerResponse((control_name, repository_name,
167
class SmartServerRequestCreateBranch(SmartServerRequestBzrDir):
169
def do(self, path, network_name):
170
"""Create a branch in the bzr dir at path.
172
This operates precisely like 'bzrdir.create_branch'.
174
If a bzrdir is not present, an exception is propogated
175
rather than 'no branch' because these are different conditions (and
176
this method should only be called after establishing that a bzr dir
179
This is the initial version of this method introduced to the smart
182
:param path: The path to the bzrdir.
183
:param network_name: The network name of the branch type to create.
184
:return: ('ok', branch_format, repo_path, rich_root, tree_ref,
185
external_lookup, repo_format)
187
bzrdir = BzrDir.open_from_transport(
188
self.transport_from_client_path(path))
189
format = branch.network_format_registry.get(network_name)
190
bzrdir.branch_format = format
191
result = format.initialize(bzrdir)
192
rich_root, tree_ref, external_lookup = self._format_to_capabilities(
193
result.repository._format)
194
branch_format = result._format.network_name()
195
repo_format = result.repository._format.network_name()
196
repo_path = self._repo_relpath(bzrdir.root_transport,
198
# branch format, repo relpath, rich_root, tree_ref, external_lookup,
200
return SuccessfulSmartServerResponse(('ok', branch_format, repo_path,
201
rich_root, tree_ref, external_lookup, repo_format))
204
class SmartServerRequestCreateRepository(SmartServerRequestBzrDir):
206
def do(self, path, network_name, shared):
207
"""Create a repository in the bzr dir at path.
209
This operates precisely like 'bzrdir.create_repository'.
211
If a bzrdir is not present, an exception is propagated
212
rather than 'no branch' because these are different conditions (and
213
this method should only be called after establishing that a bzr dir
216
This is the initial version of this method introduced to the smart
219
:param path: The path to the bzrdir.
220
:param network_name: The network name of the repository type to create.
221
:param shared: The value to pass create_repository for the shared
223
:return: (ok, rich_root, tree_ref, external_lookup, network_name)
225
bzrdir = BzrDir.open_from_transport(
226
self.transport_from_client_path(path))
227
shared = shared == 'True'
228
format = repository.network_format_registry.get(network_name)
229
bzrdir.repository_format = format
230
result = format.initialize(bzrdir, shared=shared)
231
rich_root, tree_ref, external_lookup = self._format_to_capabilities(
233
return SuccessfulSmartServerResponse(('ok', rich_root, tree_ref,
234
external_lookup, result._format.network_name()))
237
class SmartServerRequestFindRepository(SmartServerRequestBzrDir):
62
239
def _find(self, path):
63
240
"""try to find a repository from path upwards
65
242
This operates precisely like 'bzrdir.find_repository'.
67
:return: (relpath, rich_root, tree_ref, external_lookup) flags. All are
68
strings, relpath is a / prefixed path, and the other three are
244
:return: (relpath, rich_root, tree_ref, external_lookup, network_name).
245
All are strings, relpath is a / prefixed path, the next three are
246
either 'yes' or 'no', and the last is a repository format network
70
248
:raises errors.NoRepositoryPresent: When there is no repository
73
251
bzrdir = BzrDir.open_from_transport(
74
252
self.transport_from_client_path(path))
75
253
repository = bzrdir.find_repository()
76
# the relpath of the bzrdir in the found repository gives us the
77
# path segments to pop-out.
78
relpath = repository.bzrdir.root_transport.relpath(
79
bzrdir.root_transport.base)
81
segments = ['..'] * len(relpath.split('/'))
84
rich_root = self._boolean_to_yes_no(repository.supports_rich_root())
85
tree_ref = self._boolean_to_yes_no(
86
repository._format.supports_tree_reference)
87
external_lookup = self._boolean_to_yes_no(
88
repository._format.supports_external_lookups)
89
return '/'.join(segments), rich_root, tree_ref, external_lookup
254
path = self._repo_relpath(bzrdir.root_transport, repository)
255
rich_root, tree_ref, external_lookup = self._format_to_capabilities(
257
network_name = repository._format.network_name()
258
return path, rich_root, tree_ref, external_lookup, network_name
92
261
class SmartServerRequestFindRepositoryV1(SmartServerRequestFindRepository):
94
263
def do(self, path):
95
264
"""try to find a repository from path upwards
97
266
This operates precisely like 'bzrdir.find_repository'.
99
If a bzrdir is not present, an exception is propogated
268
If a bzrdir is not present, an exception is propagated
100
269
rather than 'no branch' because these are different conditions.
102
271
This is the initial version of this method introduced with the smart
117
286
def do(self, path):
118
287
"""try to find a repository from path upwards
120
289
This operates precisely like 'bzrdir.find_repository'.
122
If a bzrdir is not present, an exception is propogated
291
If a bzrdir is not present, an exception is propagated
123
292
rather than 'no branch' because these are different conditions.
125
294
This is the second edition of this method introduced in bzr 1.3, which
126
295
returns information about the supports_external_lookups format
129
:return: norepository or ok, relpath.
298
:return: norepository or ok, relpath, rich_root, tree_ref,
132
path, rich_root, tree_ref, external_lookup = self._find(path)
302
path, rich_root, tree_ref, external_lookup, name = self._find(path)
133
303
return SuccessfulSmartServerResponse(
134
304
('ok', path, rich_root, tree_ref, external_lookup))
135
305
except errors.NoRepositoryPresent:
136
306
return FailedSmartServerResponse(('norepository', ))
309
class SmartServerRequestFindRepositoryV3(SmartServerRequestFindRepository):
312
"""try to find a repository from path upwards
314
This operates precisely like 'bzrdir.find_repository'.
316
If a bzrdir is not present, an exception is propogated
317
rather than 'no branch' because these are different conditions.
319
This is the third edition of this method introduced in bzr 1.13, which
320
returns information about the network name of the repository format.
322
:return: norepository or ok, relpath, rich_root, tree_ref,
323
external_lookup, network_name.
326
path, rich_root, tree_ref, external_lookup, name = self._find(path)
327
return SuccessfulSmartServerResponse(
328
('ok', path, rich_root, tree_ref, external_lookup, name))
329
except errors.NoRepositoryPresent:
330
return FailedSmartServerResponse(('norepository', ))
333
class SmartServerBzrDirRequestConfigFile(SmartServerRequestBzrDir):
335
def do_bzrdir_request(self):
336
"""Get the configuration bytes for a config file in bzrdir.
338
The body is not utf8 decoded - it is the literal bytestream from disk.
340
config = self._bzrdir._get_config()
344
content = config._get_config_file().read()
345
return SuccessfulSmartServerResponse((), content)
139
348
class SmartServerRequestInitializeBzrDir(SmartServerRequest):
141
350
def do(self, path):
149
358
return SuccessfulSmartServerResponse(('ok', ))
152
class SmartServerRequestOpenBranch(SmartServerRequest):
155
"""try to open a branch at path and return ok/nobranch.
157
If a bzrdir is not present, an exception is propogated
158
rather than 'no branch' because these are different conditions.
361
class SmartServerRequestBzrDirInitializeEx(SmartServerRequestBzrDir):
363
def parse_NoneTrueFalse(self, arg):
370
raise AssertionError("invalid arg %r" % arg)
372
def parse_NoneString(self, arg):
375
def _serialize_NoneTrueFalse(self, arg):
382
def do(self, bzrdir_network_name, path, use_existing_dir, create_prefix,
383
force_new_repo, stacked_on, stack_on_pwd, repo_format_name,
384
make_working_trees, shared_repo):
385
"""Initialize a bzrdir at path as per
386
BzrDirFormat.initialize_on_transport_ex.
388
New in 1.16. (Replaces BzrDirFormat.initialize_ex verb from 1.15).
390
:return: return SuccessfulSmartServerResponse((repo_path, rich_root,
391
tree_ref, external_lookup, repo_network_name,
392
repo_bzrdir_network_name, bzrdir_format_network_name,
393
NoneTrueFalse(stacking), final_stack, final_stack_pwd,
160
bzrdir = BzrDir.open_from_transport(
161
self.transport_from_client_path(path))
396
target_transport = self.transport_from_client_path(path)
397
format = network_format_registry.get(bzrdir_network_name)
398
use_existing_dir = self.parse_NoneTrueFalse(use_existing_dir)
399
create_prefix = self.parse_NoneTrueFalse(create_prefix)
400
force_new_repo = self.parse_NoneTrueFalse(force_new_repo)
401
stacked_on = self.parse_NoneString(stacked_on)
402
stack_on_pwd = self.parse_NoneString(stack_on_pwd)
403
make_working_trees = self.parse_NoneTrueFalse(make_working_trees)
404
shared_repo = self.parse_NoneTrueFalse(shared_repo)
405
if stack_on_pwd == '.':
406
stack_on_pwd = target_transport.base
407
repo_format_name = self.parse_NoneString(repo_format_name)
408
repo, bzrdir, stacking, repository_policy = \
409
format.initialize_on_transport_ex(target_transport,
410
use_existing_dir=use_existing_dir, create_prefix=create_prefix,
411
force_new_repo=force_new_repo, stacked_on=stacked_on,
412
stack_on_pwd=stack_on_pwd, repo_format_name=repo_format_name,
413
make_working_trees=make_working_trees, shared_repo=shared_repo)
417
rich_root = tree_ref = external_lookup = ''
418
repo_bzrdir_name = ''
420
final_stack_pwd = None
423
repo_path = self._repo_relpath(bzrdir.root_transport, repo)
426
rich_root, tree_ref, external_lookup = self._format_to_capabilities(
428
repo_name = repo._format.network_name()
429
repo_bzrdir_name = repo.bzrdir._format.network_name()
430
final_stack = repository_policy._stack_on
431
final_stack_pwd = repository_policy._stack_on_pwd
432
# It is returned locked, but we need to do the lock to get the lock
435
repo_lock_token = repo.lock_write().repository_token or ''
437
repo.leave_lock_in_place()
439
final_stack = final_stack or ''
440
final_stack_pwd = final_stack_pwd or ''
442
# We want this to be relative to the bzrdir.
444
final_stack_pwd = urlutils.relative_url(
445
target_transport.base, final_stack_pwd)
447
# Can't meaningfully return a root path.
448
if final_stack.startswith('/'):
449
client_path = self._root_client_path + final_stack[1:]
450
final_stack = urlutils.relative_url(
451
self._root_client_path, client_path)
452
final_stack_pwd = '.'
454
return SuccessfulSmartServerResponse((repo_path, rich_root, tree_ref,
455
external_lookup, repo_name, repo_bzrdir_name,
456
bzrdir._format.network_name(),
457
self._serialize_NoneTrueFalse(stacking), final_stack,
458
final_stack_pwd, repo_lock_token))
461
class SmartServerRequestOpenBranch(SmartServerRequestBzrDir):
463
def do_bzrdir_request(self):
464
"""open a branch at path and return the branch reference or branch."""
163
reference_url = bzrdir.get_branch_reference()
466
reference_url = self._bzrdir.get_branch_reference()
164
467
if reference_url is None:
165
468
return SuccessfulSmartServerResponse(('ok', ''))
167
470
return SuccessfulSmartServerResponse(('ok', reference_url))
168
except errors.NotBranchError:
169
return FailedSmartServerResponse(('nobranch', ))
471
except errors.NotBranchError, e:
472
return FailedSmartServerResponse(('nobranch',))
475
class SmartServerRequestOpenBranchV2(SmartServerRequestBzrDir):
477
def do_bzrdir_request(self):
478
"""open a branch at path and return the reference or format."""
480
reference_url = self._bzrdir.get_branch_reference()
481
if reference_url is None:
482
br = self._bzrdir.open_branch(ignore_fallbacks=True)
483
format = br._format.network_name()
484
return SuccessfulSmartServerResponse(('branch', format))
486
return SuccessfulSmartServerResponse(('ref', reference_url))
487
except errors.NotBranchError, e:
488
return FailedSmartServerResponse(('nobranch',))
491
class SmartServerRequestOpenBranchV3(SmartServerRequestBzrDir):
493
def do_bzrdir_request(self):
494
"""Open a branch at path and return the reference or format.
496
This version introduced in 2.1.
498
Differences to SmartServerRequestOpenBranchV2:
499
* can return 2-element ('nobranch', extra), where 'extra' is a string
500
with an explanation like 'location is a repository'. Previously
501
a 'nobranch' response would never have more than one element.
504
reference_url = self._bzrdir.get_branch_reference()
505
if reference_url is None:
506
br = self._bzrdir.open_branch(ignore_fallbacks=True)
507
format = br._format.network_name()
508
return SuccessfulSmartServerResponse(('branch', format))
510
return SuccessfulSmartServerResponse(('ref', reference_url))
511
except errors.NotBranchError, e:
512
# Stringify the exception so that its .detail attribute will be
518
if detail.startswith(': '):
521
return FailedSmartServerResponse(resp)