14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
from __future__ import absolute_import
17
19
__all__ = ['show_bzrdir_info']
19
from cStringIO import StringIO
26
28
hooks as _mod_hooks,
30
from bzrlib.errors import (NoWorkingTree, NotBranchError,
32
from .errors import (NoWorkingTree, NotBranchError,
31
33
NoRepositoryPresent, NotLocalUrl)
32
from bzrlib.missing import find_unmerged
34
from .missing import find_unmerged
35
40
def plural(n, base='', pl=None):
76
81
return [" %*s: %s\n" % (max_len, l, u) for l, u in self.locs ]
79
def gather_location_info(repository, branch=None, working=None):
84
def gather_location_info(repository=None, branch=None, working=None,
81
repository_path = repository.user_url
82
87
if branch is not None:
83
88
branch_path = branch.user_url
84
89
master_path = branch.get_bound_location()
105
115
locs['branch root'] = branch_path
107
117
working_path = None
108
if repository.is_shared():
118
if repository is not None and repository.is_shared():
109
119
# lightweight checkout of branch in shared repository
110
120
if branch_path is not None:
111
121
locs['repository branch'] = branch_path
112
122
elif branch_path is not None:
114
124
locs['branch root'] = branch_path
115
if master_path != branch_path:
116
locs['bound to branch'] = master_path
125
elif repository is not None:
126
locs['repository'] = repository.user_url
127
elif control is not None:
128
locs['control directory'] = control.user_url
118
locs['repository'] = repository_path
119
if repository.is_shared():
130
# Really, at least a control directory should be
131
# passed in for this method to be useful.
133
if master_path != branch_path:
134
locs['bound to branch'] = master_path
135
if repository is not None and repository.is_shared():
120
136
# lightweight checkout of branch in shared repository
121
locs['shared repository'] = repository_path
122
order = ['light checkout root', 'repository checkout root',
123
'checkout root', 'checkout of branch', 'shared repository',
137
locs['shared repository'] = repository.user_url
138
order = ['control directory', 'light checkout root',
139
'repository checkout root', 'checkout root',
140
'checkout of branch', 'shared repository',
124
141
'repository', 'repository branch', 'branch root',
125
142
'bound to branch']
126
143
return [(n, locs[n]) for n in order if n in locs]
158
175
outfile.writelines(locs.get_lines())
178
def _show_control_dir_info(control, outfile):
179
"""Show control dir information."""
180
if control._format.colocated_branches:
182
outfile.write('Control directory:\n')
183
outfile.write(' %d branches\n' % len(control.list_branches()))
161
186
def _show_format_info(control=None, repository=None, branch=None,
162
187
working=None, outfile=None):
163
188
"""Show known formats for control, working, branch and repository."""
177
202
repository._format.get_format_description())
180
def _show_locking_info(repository, branch=None, working=None, outfile=None):
205
def _show_locking_info(repository=None, branch=None, working=None,
181
207
"""Show locking status of working, branch and repository."""
182
if (repository.get_physical_lock_status() or
208
if (repository and repository.get_physical_lock_status() or
183
209
(branch and branch.get_physical_lock_status()) or
184
210
(working and working.get_physical_lock_status())):
185
211
outfile.write('\n')
221
247
"""Show missing revisions in working tree."""
222
248
branch = working.branch
223
249
basis = working.basis_tree()
224
work_inv = working.inventory
225
branch_revno, branch_last_revision = branch.last_revision_info()
251
branch_revno, branch_last_revision = branch.last_revision_info()
252
except errors.UnsupportedOperation:
227
255
tree_last_id = working.get_parent_ids()[0]
228
256
except IndexError:
239
267
def _show_working_stats(working, outfile):
240
268
"""Show statistics about a working tree."""
241
269
basis = working.basis_tree()
242
work_inv = working.inventory
243
270
delta = working.changes_from(basis, want_unchanged=True)
245
272
outfile.write('\n')
260
287
outfile.write(' %8d ignored\n' % ignore_cnt)
263
for file_id in work_inv:
264
if (work_inv.get_file_kind(file_id) == 'directory' and
265
not work_inv.is_root(file_id)):
290
root_id = working.get_root_id()
291
for path, entry in working.iter_entries_by_dir():
292
if entry.kind == 'directory' and entry.file_id != root_id:
267
294
outfile.write(' %8d versioned %s\n' % (dir_cnt,
268
295
plural(dir_cnt, 'subdirectory', 'subdirectories')))
271
298
def _show_branch_stats(branch, verbose, outfile):
272
299
"""Show statistics about a branch."""
273
revno, head = branch.last_revision_info()
301
revno, head = branch.last_revision_info()
302
except errors.UnsupportedOperation:
274
304
outfile.write('\n')
275
305
outfile.write('Branch history:\n')
276
306
outfile.write(' %8d revision%s\n' % (revno, plural(revno)))
302
332
def _show_repository_stats(repository, stats, outfile):
303
333
"""Show statistics about a repository."""
305
335
if 'revisions' in stats:
306
336
revisions = stats['revisions']
307
337
f.write(' %8d revision%s\n' % (revisions, plural(revisions)))
323
353
tree = a_bzrdir.open_workingtree(
324
354
recommend_upgrade=False)
325
except (NoWorkingTree, NotLocalUrl):
355
except (NoWorkingTree, NotLocalUrl, NotBranchError):
328
branch = a_bzrdir.open_branch()
358
branch = a_bzrdir.open_branch(name="")
329
359
except NotBranchError:
332
362
repository = a_bzrdir.open_repository()
333
363
except NoRepositoryPresent:
334
# Return silently; cmd_info already returned NotBranchError
335
# if no bzrdir could be opened.
338
367
lockable = repository
344
373
repository = branch.repository
376
if lockable is not None:
349
379
show_component_info(a_bzrdir, repository, branch, tree, verbose,
382
if lockable is not None:
355
386
def show_component_info(control, repository, branch=None, working=None,
362
393
if verbose is True:
364
layout = describe_layout(repository, branch, working)
395
layout = describe_layout(repository, branch, working, control)
365
396
format = describe_format(control, repository, branch, working)
366
397
outfile.write("%s (format: %s)\n" % (layout, format))
367
_show_location_info(gather_location_info(repository, branch, working),
399
gather_location_info(control=control, repository=repository,
400
branch=branch, working=working),
369
402
if branch is not None:
370
403
_show_related_info(branch, outfile)
373
406
_show_format_info(control, repository, branch, working, outfile)
374
407
_show_locking_info(repository, branch, working, outfile)
408
_show_control_dir_info(control, outfile)
375
409
if branch is not None:
376
410
_show_missing_revisions_branch(branch, outfile)
377
411
if working is not None:
382
416
if branch is not None:
383
417
show_committers = verbose >= 2
384
418
stats = _show_branch_stats(branch, show_committers, outfile)
419
elif repository is not None:
386
420
stats = repository.gather_stats()
387
if branch is None and working is None:
421
if branch is None and working is None and repository is not None:
388
422
_show_repository_info(repository, outfile)
389
_show_repository_stats(repository, stats, outfile)
392
def describe_layout(repository=None, branch=None, tree=None):
423
if repository is not None:
424
_show_repository_stats(repository, stats, outfile)
427
def describe_layout(repository=None, branch=None, tree=None, control=None):
393
428
"""Convert a control directory layout into a user-understandable term
395
430
Common outputs include "Standalone tree", "Repository branch" and
396
431
"Checkout". Uncommon outputs include "Unshared repository with trees"
397
432
and "Empty control directory"
434
if branch is None and control is not None:
436
branch_reference = control.get_branch_reference()
437
except NotBranchError:
440
if branch_reference is not None:
441
return "Dangling branch reference"
399
442
if repository is None:
400
443
return 'Empty control directory'
401
444
if branch is None and tree is None:
403
446
phrase = 'Shared repository'
405
448
phrase = 'Unshared repository'
406
450
if repository.make_working_trees():
407
phrase += ' with trees'
451
extra.append('trees')
452
if len(control.get_branches()) > 0:
453
extra.append('colocated branches')
455
phrase += ' with ' + " and ".join(extra)
410
458
if repository.is_shared():
447
495
branch.user_url != tree.user_url):
449
497
repository = None
450
non_aliases = set(bzrdir.format_registry.keys())
451
non_aliases.difference_update(bzrdir.format_registry.aliases())
498
non_aliases = set(controldir.format_registry.keys())
499
non_aliases.difference_update(controldir.format_registry.aliases())
452
500
for key in non_aliases:
453
format = bzrdir.format_registry.make_bzrdir(key)
501
format = controldir.format_registry.make_bzrdir(key)
454
502
if isinstance(format, bzrdir.BzrDirMetaFormat1):
455
503
if (tree and format.workingtree_format !=
469
517
candidates.sort()
470
518
new_candidates = [c for c in candidates if not
471
bzrdir.format_registry.get_info(c).hidden]
519
controldir.format_registry.get_info(c).hidden]
472
520
if len(new_candidates) > 0:
473
521
# If there are any non-hidden formats that match, only return those to
474
522
# avoid listing hidden formats except when only a hidden format will
481
529
"""Hooks for the info command."""
483
531
def __init__(self):
484
super(InfoHooks, self).__init__()
485
self.create_hook(_mod_hooks.HookPoint('repository',
532
super(InfoHooks, self).__init__("breezy.info", "hooks")
533
self.add_hook('repository',
486
534
"Invoked when displaying the statistics for a repository. "
487
535
"repository is called with a statistics dictionary as returned "
488
"by the repository and a file-like object to write to.", (1, 15),
536
"by the repository and a file-like object to write to.", (1, 15))
492
539
hooks = InfoHooks()