~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/info.py

  • Committer: Patch Queue Manager
  • Date: 2016-04-21 04:10:52 UTC
  • mfrom: (6616.1.1 fix-en-user-guide)
  • Revision ID: pqm@pqm.ubuntu.com-20160421041052-clcye7ns1qcl2n7w
(richard-wilbur) Ensure build of English use guide always uses English text
 even when user's locale specifies a different language. (Jelmer Vernooij)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005, 2006, 2007 Canonical Ltd
2
 
 
1
# Copyright (C) 2005-2010 Canonical Ltd
 
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
5
5
# the Free Software Foundation; either version 2 of the License, or
12
12
#
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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
16
 
 
17
from __future__ import absolute_import
16
18
 
17
19
__all__ = ['show_bzrdir_info']
18
20
 
19
 
import os
 
21
from cStringIO import StringIO
20
22
import time
21
23
import sys
22
24
 
23
25
from bzrlib import (
24
26
    bzrdir,
25
 
    diff,
 
27
    controldir,
26
28
    errors,
 
29
    hooks as _mod_hooks,
27
30
    osutils,
28
31
    urlutils,
29
32
    )
30
33
from bzrlib.errors import (NoWorkingTree, NotBranchError,
31
34
                           NoRepositoryPresent, NotLocalUrl)
32
35
from bzrlib.missing import find_unmerged
33
 
from bzrlib.symbol_versioning import (deprecated_function,
34
 
        zero_eighteen)
35
36
 
36
37
 
37
38
def plural(n, base='', pl=None):
78
79
        return ["  %*s: %s\n" % (max_len, l, u) for l, u in self.locs ]
79
80
 
80
81
 
81
 
def gather_location_info(repository, branch=None, working=None):
 
82
def gather_location_info(repository=None, branch=None, working=None,
 
83
        control=None):
82
84
    locs = {}
83
 
    repository_path = repository.bzrdir.root_transport.base
84
85
    if branch is not None:
85
 
        branch_path = branch.bzrdir.root_transport.base
 
86
        branch_path = branch.user_url
86
87
        master_path = branch.get_bound_location()
87
88
        if master_path is None:
88
89
            master_path = branch_path
89
90
    else:
90
91
        branch_path = None
91
92
        master_path = None
 
93
        try:
 
94
            if control is not None and control.get_branch_reference():
 
95
                locs['checkout of branch'] = control.get_branch_reference()
 
96
        except NotBranchError:
 
97
            pass
92
98
    if working:
93
 
        working_path = working.bzrdir.root_transport.base
 
99
        working_path = working.user_url
94
100
        if working_path != branch_path:
95
101
            locs['light checkout root'] = working_path
96
102
        if master_path != branch_path:
107
113
            locs['branch root'] = branch_path
108
114
    else:
109
115
        working_path = None
110
 
        if repository.is_shared():
 
116
        if repository is not None and repository.is_shared():
111
117
            # lightweight checkout of branch in shared repository
112
118
            if branch_path is not None:
113
119
                locs['repository branch'] = branch_path
114
120
        elif branch_path is not None:
115
121
            # standalone
116
122
            locs['branch root'] = branch_path
117
 
            if master_path != branch_path:
118
 
                locs['bound to branch'] = master_path
 
123
        elif repository is not None:
 
124
            locs['repository'] = repository.user_url
 
125
        elif control is not None:
 
126
            locs['control directory'] = control.user_url
119
127
        else:
120
 
            locs['repository'] = repository_path
121
 
    if repository.is_shared():
 
128
            # Really, at least a control directory should be
 
129
            # passed in for this method to be useful.
 
130
            pass
 
131
        if master_path != branch_path:
 
132
            locs['bound to branch'] = master_path
 
133
    if repository is not None and repository.is_shared():
122
134
        # lightweight checkout of branch in shared repository
123
 
        locs['shared repository'] = repository_path
124
 
    order = ['light checkout root', 'repository checkout root',
125
 
             'checkout root', 'checkout of branch', 'shared repository',
 
135
        locs['shared repository'] = repository.user_url
 
136
    order = ['control directory', 'light checkout root',
 
137
             'repository checkout root', 'checkout root',
 
138
             'checkout of branch', 'shared repository',
126
139
             'repository', 'repository branch', 'branch root',
127
140
             'bound to branch']
128
141
    return [(n, locs[n]) for n in order if n in locs]
143
156
    locs.add_url('push branch', branch.get_push_location())
144
157
    locs.add_url('parent branch', branch.get_parent())
145
158
    locs.add_url('submit branch', branch.get_submit_branch())
 
159
    try:
 
160
        locs.add_url('stacked on', branch.get_stacked_on_url())
 
161
    except (errors.UnstackableBranchFormat, errors.UnstackableRepositoryFormat,
 
162
        errors.NotStacked):
 
163
        pass
146
164
    return locs
147
165
 
148
166
 
155
173
        outfile.writelines(locs.get_lines())
156
174
 
157
175
 
 
176
def _show_control_dir_info(control, outfile):
 
177
    """Show control dir information."""
 
178
    if control._format.colocated_branches:
 
179
        outfile.write('\n')
 
180
        outfile.write('Control directory:\n')
 
181
        outfile.write('         %d branches\n' % len(control.list_branches()))
 
182
 
 
183
 
158
184
def _show_format_info(control=None, repository=None, branch=None,
159
185
                      working=None, outfile=None):
160
186
    """Show known formats for control, working, branch and repository."""
174
200
            repository._format.get_format_description())
175
201
 
176
202
 
177
 
def _show_locking_info(repository, branch=None, working=None, outfile=None):
 
203
def _show_locking_info(repository=None, branch=None, working=None,
 
204
        outfile=None):
178
205
    """Show locking status of working, branch and repository."""
179
 
    if (repository.get_physical_lock_status() or
 
206
    if (repository and repository.get_physical_lock_status() or
180
207
        (branch and branch.get_physical_lock_status()) or
181
208
        (working and working.get_physical_lock_status())):
182
209
        outfile.write('\n')
218
245
    """Show missing revisions in working tree."""
219
246
    branch = working.branch
220
247
    basis = working.basis_tree()
221
 
    work_inv = working.inventory
222
 
    branch_revno, branch_last_revision = branch.last_revision_info()
 
248
    try:
 
249
        branch_revno, branch_last_revision = branch.last_revision_info()
 
250
    except errors.UnsupportedOperation:
 
251
        return
223
252
    try:
224
253
        tree_last_id = working.get_parent_ids()[0]
225
254
    except IndexError:
236
265
def _show_working_stats(working, outfile):
237
266
    """Show statistics about a working tree."""
238
267
    basis = working.basis_tree()
239
 
    work_inv = working.inventory
240
268
    delta = working.changes_from(basis, want_unchanged=True)
241
269
 
242
270
    outfile.write('\n')
257
285
    outfile.write('  %8d ignored\n' % ignore_cnt)
258
286
 
259
287
    dir_cnt = 0
260
 
    for file_id in work_inv:
261
 
        if (work_inv.get_file_kind(file_id) == 'directory' and 
262
 
            not work_inv.is_root(file_id)):
 
288
    root_id = working.get_root_id()
 
289
    for path, entry in working.iter_entries_by_dir():
 
290
        if entry.kind == 'directory' and entry.file_id != root_id:
263
291
            dir_cnt += 1
264
292
    outfile.write('  %8d versioned %s\n' % (dir_cnt,
265
293
        plural(dir_cnt, 'subdirectory', 'subdirectories')))
267
295
 
268
296
def _show_branch_stats(branch, verbose, outfile):
269
297
    """Show statistics about a branch."""
270
 
    revno, head = branch.last_revision_info()
 
298
    try:
 
299
        revno, head = branch.last_revision_info()
 
300
    except errors.UnsupportedOperation:
 
301
        return {}
271
302
    outfile.write('\n')
272
303
    outfile.write('Branch history:\n')
273
304
    outfile.write('  %8d revision%s\n' % (revno, plural(revno)))
296
327
            'the repository.\n')
297
328
 
298
329
 
299
 
def _show_repository_stats(stats, outfile):
 
330
def _show_repository_stats(repository, stats, outfile):
300
331
    """Show statistics about a repository."""
301
 
    if 'revisions' in stats or 'size' in stats:
302
 
        outfile.write('\n')
303
 
        outfile.write('Repository:\n')
 
332
    f = StringIO()
304
333
    if 'revisions' in stats:
305
334
        revisions = stats['revisions']
306
 
        outfile.write('  %8d revision%s\n' % (revisions, plural(revisions)))
 
335
        f.write('  %8d revision%s\n' % (revisions, plural(revisions)))
307
336
    if 'size' in stats:
308
 
        outfile.write('  %8d KiB\n' % (stats['size']/1024))
 
337
        f.write('  %8d KiB\n' % (stats['size']/1024))
 
338
    for hook in hooks['repository']:
 
339
        hook(repository, stats, f)
 
340
    if f.getvalue() != "":
 
341
        outfile.write('\n')
 
342
        outfile.write('Repository:\n')
 
343
        outfile.write(f.getvalue())
309
344
 
310
345
 
311
346
def show_bzrdir_info(a_bzrdir, verbose=False, outfile=None):
315
350
    try:
316
351
        tree = a_bzrdir.open_workingtree(
317
352
            recommend_upgrade=False)
318
 
    except (NoWorkingTree, NotLocalUrl):
 
353
    except (NoWorkingTree, NotLocalUrl, NotBranchError):
319
354
        tree = None
320
355
        try:
321
 
            branch = a_bzrdir.open_branch()
 
356
            branch = a_bzrdir.open_branch(name="")
322
357
        except NotBranchError:
323
358
            branch = None
324
359
            try:
325
360
                repository = a_bzrdir.open_repository()
326
361
            except NoRepositoryPresent:
327
 
                # Return silently; cmd_info already returned NotBranchError
328
 
                # if no bzrdir could be opened.
329
 
                return
 
362
                lockable = None
 
363
                repository = None
330
364
            else:
331
365
                lockable = repository
332
366
        else:
337
371
        repository = branch.repository
338
372
        lockable = tree
339
373
 
340
 
    lockable.lock_read()
 
374
    if lockable is not None:
 
375
        lockable.lock_read()
341
376
    try:
342
377
        show_component_info(a_bzrdir, repository, branch, tree, verbose,
343
378
                            outfile)
344
379
    finally:
345
 
        lockable.unlock()
 
380
        if lockable is not None:
 
381
            lockable.unlock()
346
382
 
347
383
 
348
384
def show_component_info(control, repository, branch=None, working=None,
354
390
        verbose = 1
355
391
    if verbose is True:
356
392
        verbose = 2
357
 
    layout = describe_layout(repository, branch, working)
 
393
    layout = describe_layout(repository, branch, working, control)
358
394
    format = describe_format(control, repository, branch, working)
359
395
    outfile.write("%s (format: %s)\n" % (layout, format))
360
 
    _show_location_info(gather_location_info(repository, branch, working),
361
 
                        outfile)
 
396
    _show_location_info(
 
397
        gather_location_info(control=control, repository=repository,
 
398
            branch=branch, working=working),
 
399
        outfile)
362
400
    if branch is not None:
363
401
        _show_related_info(branch, outfile)
364
402
    if verbose == 0:
365
403
        return
366
404
    _show_format_info(control, repository, branch, working, outfile)
367
405
    _show_locking_info(repository, branch, working, outfile)
 
406
    _show_control_dir_info(control, outfile)
368
407
    if branch is not None:
369
408
        _show_missing_revisions_branch(branch, outfile)
370
409
    if working is not None:
373
412
    elif branch is not None:
374
413
        _show_missing_revisions_branch(branch, outfile)
375
414
    if branch is not None:
376
 
        stats = _show_branch_stats(branch, verbose==2, outfile)
377
 
    else:
 
415
        show_committers = verbose >= 2
 
416
        stats = _show_branch_stats(branch, show_committers, outfile)
 
417
    elif repository is not None:
378
418
        stats = repository.gather_stats()
379
 
    if branch is None and working is None:
 
419
    if branch is None and working is None and repository is not None:
380
420
        _show_repository_info(repository, outfile)
381
 
    _show_repository_stats(stats, outfile)
382
 
 
383
 
 
384
 
def describe_layout(repository=None, branch=None, tree=None):
 
421
    if repository is not None:
 
422
        _show_repository_stats(repository, stats, outfile)
 
423
 
 
424
 
 
425
def describe_layout(repository=None, branch=None, tree=None, control=None):
385
426
    """Convert a control directory layout into a user-understandable term
386
427
 
387
428
    Common outputs include "Standalone tree", "Repository branch" and
388
429
    "Checkout".  Uncommon outputs include "Unshared repository with trees"
389
430
    and "Empty control directory"
390
431
    """
 
432
    if branch is None and control is not None:
 
433
        try:
 
434
            branch_reference = control.get_branch_reference()
 
435
        except NotBranchError:
 
436
            pass
 
437
        else:
 
438
            if branch_reference is not None:
 
439
                return "Dangling branch reference"
391
440
    if repository is None:
392
441
        return 'Empty control directory'
393
442
    if branch is None and tree is None:
395
444
            phrase = 'Shared repository'
396
445
        else:
397
446
            phrase = 'Unshared repository'
 
447
        extra = []
398
448
        if repository.make_working_trees():
399
 
            phrase += ' with trees'
 
449
            extra.append('trees')
 
450
        if len(control.get_branches()) > 0:
 
451
            extra.append('colocated branches')
 
452
        if extra:
 
453
            phrase += ' with ' + " and ".join(extra)
400
454
        return phrase
401
455
    else:
402
456
        if repository.is_shared():
410
464
        if branch is None and tree is not None:
411
465
            phrase = "branchless tree"
412
466
        else:
413
 
            if (tree is not None and tree.bzrdir.root_transport.base !=
414
 
                branch.bzrdir.root_transport.base):
 
467
            if (tree is not None and tree.user_url !=
 
468
                branch.user_url):
415
469
                independence = ''
416
470
                phrase = "Lightweight checkout"
417
471
            elif branch.get_bound_location() is not None:
436
490
    """
437
491
    candidates  = []
438
492
    if (branch is not None and tree is not None and
439
 
        branch.bzrdir.root_transport.base !=
440
 
        tree.bzrdir.root_transport.base):
 
493
        branch.user_url != tree.user_url):
441
494
        branch = None
442
495
        repository = None
443
 
    for key in bzrdir.format_registry.keys():
444
 
        format = bzrdir.format_registry.make_bzrdir(key)
 
496
    non_aliases = set(controldir.format_registry.keys())
 
497
    non_aliases.difference_update(controldir.format_registry.aliases())
 
498
    for key in non_aliases:
 
499
        format = controldir.format_registry.make_bzrdir(key)
445
500
        if isinstance(format, bzrdir.BzrDirMetaFormat1):
446
501
            if (tree and format.workingtree_format !=
447
502
                tree._format):
457
512
        candidates.append(key)
458
513
    if len(candidates) == 0:
459
514
        return 'unnamed'
460
 
    new_candidates = [c for c in candidates if c != 'default']
461
 
    if len(new_candidates) > 0:
462
 
        candidates = new_candidates
 
515
    candidates.sort()
463
516
    new_candidates = [c for c in candidates if not
464
 
        bzrdir.format_registry.get_info(c).hidden]
 
517
        controldir.format_registry.get_info(c).hidden]
465
518
    if len(new_candidates) > 0:
 
519
        # If there are any non-hidden formats that match, only return those to
 
520
        # avoid listing hidden formats except when only a hidden format will
 
521
        # do.
466
522
        candidates = new_candidates
467
523
    return ' or '.join(candidates)
 
524
 
 
525
 
 
526
class InfoHooks(_mod_hooks.Hooks):
 
527
    """Hooks for the info command."""
 
528
 
 
529
    def __init__(self):
 
530
        super(InfoHooks, self).__init__("bzrlib.info", "hooks")
 
531
        self.add_hook('repository',
 
532
            "Invoked when displaying the statistics for a repository. "
 
533
            "repository is called with a statistics dictionary as returned "
 
534
            "by the repository and a file-like object to write to.", (1, 15))
 
535
 
 
536
 
 
537
hooks = InfoHooks()