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
27
hooks as _mod_hooks,
30
from bzrlib.errors import (NoWorkingTree, NotBranchError,
34
from .errors import (NoWorkingTree, NotBranchError,
31
35
NoRepositoryPresent, NotLocalUrl)
32
from bzrlib.missing import find_unmerged
36
from .missing import find_unmerged
35
42
def plural(n, base='', pl=None):
76
83
return [" %*s: %s\n" % (max_len, l, u) for l, u in self.locs ]
79
def gather_location_info(repository, branch=None, working=None):
86
def gather_location_info(repository=None, branch=None, working=None,
81
repository_path = repository.user_url
82
89
if branch is not None:
83
90
branch_path = branch.user_url
84
91
master_path = branch.get_bound_location()
105
117
locs['branch root'] = branch_path
107
119
working_path = None
108
if repository.is_shared():
120
if repository is not None and repository.is_shared():
109
121
# lightweight checkout of branch in shared repository
110
122
if branch_path is not None:
111
123
locs['repository branch'] = branch_path
112
124
elif branch_path is not None:
114
126
locs['branch root'] = branch_path
115
if master_path != branch_path:
116
locs['bound to branch'] = master_path
127
elif repository is not None:
128
locs['repository'] = repository.user_url
129
elif control is not None:
130
locs['control directory'] = control.user_url
118
locs['repository'] = repository_path
119
if repository.is_shared():
132
# Really, at least a control directory should be
133
# passed in for this method to be useful.
135
if master_path != branch_path:
136
locs['bound to branch'] = master_path
137
if repository is not None and repository.is_shared():
120
138
# 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',
139
locs['shared repository'] = repository.user_url
140
order = ['control directory', 'light checkout root',
141
'repository checkout root', 'checkout root',
142
'checkout of branch', 'shared repository',
124
143
'repository', 'repository branch', 'branch root',
125
144
'bound to branch']
126
145
return [(n, locs[n]) for n in order if n in locs]
158
177
outfile.writelines(locs.get_lines())
180
def _show_control_dir_info(control, outfile):
181
"""Show control dir information."""
182
if control._format.colocated_branches:
184
outfile.write('Control directory:\n')
185
outfile.write(' %d branches\n' % len(control.list_branches()))
161
188
def _show_format_info(control=None, repository=None, branch=None,
162
189
working=None, outfile=None):
163
190
"""Show known formats for control, working, branch and repository."""
177
204
repository._format.get_format_description())
180
def _show_locking_info(repository, branch=None, working=None, outfile=None):
207
def _show_locking_info(repository=None, branch=None, working=None,
181
209
"""Show locking status of working, branch and repository."""
182
if (repository.get_physical_lock_status() or
210
if (repository and repository.get_physical_lock_status() or
183
211
(branch and branch.get_physical_lock_status()) or
184
212
(working and working.get_physical_lock_status())):
185
213
outfile.write('\n')
221
249
"""Show missing revisions in working tree."""
222
250
branch = working.branch
223
251
basis = working.basis_tree()
224
work_inv = working.inventory
225
branch_revno, branch_last_revision = branch.last_revision_info()
253
branch_revno, branch_last_revision = branch.last_revision_info()
254
except errors.UnsupportedOperation:
227
257
tree_last_id = working.get_parent_ids()[0]
228
258
except IndexError:
239
269
def _show_working_stats(working, outfile):
240
270
"""Show statistics about a working tree."""
241
271
basis = working.basis_tree()
242
work_inv = working.inventory
243
272
delta = working.changes_from(basis, want_unchanged=True)
245
274
outfile.write('\n')
260
289
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)):
292
root_id = working.get_root_id()
293
for path, entry in working.iter_entries_by_dir():
294
if entry.kind == 'directory' and entry.file_id != root_id:
267
296
outfile.write(' %8d versioned %s\n' % (dir_cnt,
268
297
plural(dir_cnt, 'subdirectory', 'subdirectories')))
271
300
def _show_branch_stats(branch, verbose, outfile):
272
301
"""Show statistics about a branch."""
273
revno, head = branch.last_revision_info()
303
revno, head = branch.last_revision_info()
304
except errors.UnsupportedOperation:
274
306
outfile.write('\n')
275
307
outfile.write('Branch history:\n')
276
308
outfile.write(' %8d revision%s\n' % (revno, plural(revno)))
302
334
def _show_repository_stats(repository, stats, outfile):
303
335
"""Show statistics about a repository."""
305
337
if 'revisions' in stats:
306
338
revisions = stats['revisions']
307
339
f.write(' %8d revision%s\n' % (revisions, plural(revisions)))
323
355
tree = a_bzrdir.open_workingtree(
324
356
recommend_upgrade=False)
325
except (NoWorkingTree, NotLocalUrl):
357
except (NoWorkingTree, NotLocalUrl, NotBranchError):
328
branch = a_bzrdir.open_branch()
360
branch = a_bzrdir.open_branch(name="")
329
361
except NotBranchError:
332
364
repository = a_bzrdir.open_repository()
333
365
except NoRepositoryPresent:
334
# Return silently; cmd_info already returned NotBranchError
335
# if no bzrdir could be opened.
338
369
lockable = repository
344
375
repository = branch.repository
378
if lockable is not None:
349
381
show_component_info(a_bzrdir, repository, branch, tree, verbose,
384
if lockable is not None:
355
388
def show_component_info(control, repository, branch=None, working=None,
362
395
if verbose is True:
364
layout = describe_layout(repository, branch, working)
397
layout = describe_layout(repository, branch, working, control)
365
398
format = describe_format(control, repository, branch, working)
366
399
outfile.write("%s (format: %s)\n" % (layout, format))
367
_show_location_info(gather_location_info(repository, branch, working),
401
gather_location_info(control=control, repository=repository,
402
branch=branch, working=working),
369
404
if branch is not None:
370
405
_show_related_info(branch, outfile)
373
408
_show_format_info(control, repository, branch, working, outfile)
374
409
_show_locking_info(repository, branch, working, outfile)
410
_show_control_dir_info(control, outfile)
375
411
if branch is not None:
376
412
_show_missing_revisions_branch(branch, outfile)
377
413
if working is not None:
382
418
if branch is not None:
383
419
show_committers = verbose >= 2
384
420
stats = _show_branch_stats(branch, show_committers, outfile)
421
elif repository is not None:
386
422
stats = repository.gather_stats()
387
if branch is None and working is None:
423
if branch is None and working is None and repository is not None:
388
424
_show_repository_info(repository, outfile)
389
_show_repository_stats(repository, stats, outfile)
392
def describe_layout(repository=None, branch=None, tree=None):
425
if repository is not None:
426
_show_repository_stats(repository, stats, outfile)
429
def describe_layout(repository=None, branch=None, tree=None, control=None):
393
430
"""Convert a control directory layout into a user-understandable term
395
432
Common outputs include "Standalone tree", "Repository branch" and
396
433
"Checkout". Uncommon outputs include "Unshared repository with trees"
397
434
and "Empty control directory"
436
if branch is None and control is not None:
438
branch_reference = control.get_branch_reference()
439
except NotBranchError:
442
if branch_reference is not None:
443
return "Dangling branch reference"
399
444
if repository is None:
400
445
return 'Empty control directory'
401
446
if branch is None and tree is None:
403
448
phrase = 'Shared repository'
405
450
phrase = 'Unshared repository'
406
452
if repository.make_working_trees():
407
phrase += ' with trees'
453
extra.append('trees')
454
if len(control.get_branches()) > 0:
455
extra.append('colocated branches')
457
phrase += ' with ' + " and ".join(extra)
410
460
if repository.is_shared():
447
497
branch.user_url != tree.user_url):
449
499
repository = None
450
non_aliases = set(bzrdir.format_registry.keys())
451
non_aliases.difference_update(bzrdir.format_registry.aliases())
500
non_aliases = set(controldir.format_registry.keys())
501
non_aliases.difference_update(controldir.format_registry.aliases())
452
502
for key in non_aliases:
453
format = bzrdir.format_registry.make_bzrdir(key)
503
format = controldir.format_registry.make_bzrdir(key)
454
504
if isinstance(format, bzrdir.BzrDirMetaFormat1):
455
505
if (tree and format.workingtree_format !=
469
519
candidates.sort()
470
520
new_candidates = [c for c in candidates if not
471
bzrdir.format_registry.get_info(c).hidden]
521
controldir.format_registry.get_info(c).hidden]
472
522
if len(new_candidates) > 0:
473
523
# If there are any non-hidden formats that match, only return those to
474
524
# avoid listing hidden formats except when only a hidden format will
481
531
"""Hooks for the info command."""
483
533
def __init__(self):
484
super(InfoHooks, self).__init__()
485
self.create_hook(_mod_hooks.HookPoint('repository',
534
super(InfoHooks, self).__init__("breezy.info", "hooks")
535
self.add_hook('repository',
486
536
"Invoked when displaying the statistics for a repository. "
487
537
"repository is called with a statistics dictionary as returned "
488
"by the repository and a file-like object to write to.", (1, 15),
538
"by the repository and a file-like object to write to.", (1, 15))
492
541
hooks = InfoHooks()