~bzr-pqm/bzr/bzr.dev

2052.3.2 by John Arbash Meinel
Change Copyright .. by Canonical to Copyright ... Canonical
1
# Copyright (C) 2005, 2006 Canonical Ltd
1685.1.60 by Martin Pool
[broken] NotBranchError should unescape the url if possible
2
# 
77 by mbp at sourcefrog
- split info command out into separate file
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
2052.3.1 by John Arbash Meinel
Add tests to cleanup the copyright of all source files
7
#
77 by mbp at sourcefrog
- split info command out into separate file
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
2052.3.1 by John Arbash Meinel
Add tests to cleanup the copyright of all source files
12
#
77 by mbp at sourcefrog
- split info command out into separate file
13
# You should have received a copy of the GNU General Public License
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
16
1534.5.1 by Robert Collins
Give info some reasonable output and tests.
17
__all__ = ['show_bzrdir_info']
18
77 by mbp at sourcefrog
- split info command out into separate file
19
import time
20
1534.5.1 by Robert Collins
Give info some reasonable output and tests.
21
22
import bzrlib.diff as diff
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
23
from bzrlib.errors import (NoWorkingTree, NotBranchError,
24
                           NoRepositoryPresent, NotLocalUrl)
1587.1.14 by Robert Collins
Make bound branch creation happen via 'checkout'
25
from bzrlib.missing import find_unmerged
1694.2.6 by Martin Pool
[merge] bzr.dev
26
import bzrlib.osutils as osutils
1773.4.1 by Martin Pool
Add pyflakes makefile target; fix many warnings
27
from bzrlib.symbol_versioning import (deprecated_function, 
28
        zero_eight)
77 by mbp at sourcefrog
- split info command out into separate file
29
462 by Martin Pool
- New form 'file_id in tree' to check if the file is present
30
1563.2.28 by Robert Collins
Add total_size to the revision_store api.
31
def plural(n, base='', pl=None):
32
    if n == 1:
33
        return base
1963.2.6 by Robey Pointer
pychecker is on crack; go back to using 'is None'.
34
    elif pl is not None:
1563.2.28 by Robert Collins
Add total_size to the revision_store api.
35
        return pl
36
    else:
37
        return 's'
38
39
1694.2.6 by Martin Pool
[merge] bzr.dev
40
def _repo_relpath(repo_path, path):
41
    """Return path with common prefix of repository path removed.
42
43
    If path is not part of the repository, the original path is returned.
44
    If path is equal to the repository, the current directory marker '.' is
45
    returned.
46
    """
47
    path = osutils.normalizepath(path)
48
    repo_path = osutils.normalizepath(repo_path)
49
    if path == repo_path:
50
        return '.'
51
    if osutils.is_inside(repo_path, path):
52
        return osutils.relpath(repo_path, path)
53
    return path
54
55
56
def _show_location_info(repository, branch=None, working=None):
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
57
    """Show known locations for working, branch and repository."""
1694.2.6 by Martin Pool
[merge] bzr.dev
58
    repository_path = repository.bzrdir.root_transport.base
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
59
    print 'Location:'
1694.2.6 by Martin Pool
[merge] bzr.dev
60
    if working and branch:
61
        working_path = working.bzrdir.root_transport.base
62
        branch_path = branch.bzrdir.root_transport.base
63
        if working_path != branch_path:
64
            # lightweight checkout
1780.1.2 by Robert Collins
(robertc)Partial refactoring of info tests to be more robust to format changes.
65
            print ' light checkout root: %s' % working_path
1694.2.6 by Martin Pool
[merge] bzr.dev
66
            if repository.is_shared():
67
                # lightweight checkout of branch in shared repository
1780.1.2 by Robert Collins
(robertc)Partial refactoring of info tests to be more robust to format changes.
68
                print '   shared repository: %s' % repository_path
69
                print '   repository branch: %s' % (
1694.2.6 by Martin Pool
[merge] bzr.dev
70
                    _repo_relpath(repository_path, branch_path))
71
            else:
72
                # lightweight checkout of standalone branch
1780.1.2 by Robert Collins
(robertc)Partial refactoring of info tests to be more robust to format changes.
73
                print '  checkout of branch: %s' % branch_path
1694.2.6 by Martin Pool
[merge] bzr.dev
74
        elif repository.is_shared():
75
            # branch with tree inside shared repository
76
            print '    shared repository: %s' % repository_path
77
            print '  repository checkout: %s' % (
78
                _repo_relpath(repository_path, branch_path))
79
        elif branch.get_bound_location():
80
            # normal checkout
81
            print '       checkout root: %s' % working_path
82
            print '  checkout of branch: %s' % branch.get_bound_location()
83
        else:
84
            # standalone
85
            print '  branch root: %s' % working_path
1624.3.33 by Olaf Conradi
Simplified construct detection in location overview a lot.
86
    elif branch:
1694.2.6 by Martin Pool
[merge] bzr.dev
87
        branch_path = branch.bzrdir.root_transport.base
1624.3.48 by Olaf Conradi
Add info on standalone branches without a working tree.
88
        if repository.is_shared():
89
            # branch is part of shared repository
90
            print '  shared repository: %s' % repository_path
91
            print '  repository branch: %s' % (
92
                _repo_relpath(repository_path, branch_path))
93
        else:
94
            # standalone branch
95
            print '  branch root: %s' % branch_path
1694.2.6 by Martin Pool
[merge] bzr.dev
96
    else:
97
        # shared repository
98
        assert repository.is_shared()
99
        print '  shared repository: %s' % repository_path
100
101
102
def _show_related_info(branch):
103
    """Show parent and push location of branch."""
104
    if branch.get_parent() or branch.get_push_location():
105
        print
106
        print 'Related branches:'
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
107
        if branch.get_parent():
1694.2.6 by Martin Pool
[merge] bzr.dev
108
            if branch.get_push_location():
109
                print '      parent branch: %s' % branch.get_parent()
110
            else:
111
                print '  parent branch: %s' % branch.get_parent()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
112
        if branch.get_push_location():
1694.2.6 by Martin Pool
[merge] bzr.dev
113
            print '  publish to branch: %s' % branch.get_push_location()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
114
115
116
def _show_format_info(control=None, repository=None, branch=None, working=None):
117
    """Show known formats for control, working, branch and repository."""
1624.3.4 by Olaf Conradi
Simplify construct detection. Make terms in user interface consistent.
118
    print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
119
    print 'Format:'
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
120
    if control:
121
        print '       control: %s' % control._format.get_format_description()
122
    if working:
123
        print '  working tree: %s' % working._format.get_format_description()
124
    if branch:
125
        print '        branch: %s' % branch._format.get_format_description()
126
    if repository:
127
        print '    repository: %s' % repository._format.get_format_description()
128
129
1694.2.6 by Martin Pool
[merge] bzr.dev
130
def _show_locking_info(repository, branch=None, working=None):
131
    """Show locking status of working, branch and repository."""
132
    if (repository.get_physical_lock_status() or
133
        (branch and branch.get_physical_lock_status()) or
134
        (working and working.get_physical_lock_status())):
135
        print
136
        print 'Lock status:'
137
        if working:
138
            if working.get_physical_lock_status():
139
                status = 'locked'
140
            else:
141
                status = 'unlocked'
142
            print '  working tree: %s' % status
143
        if branch:
144
            if branch.get_physical_lock_status():
145
                status = 'locked'
146
            else:
147
                status = 'unlocked'
148
            print '        branch: %s' % status
149
        if repository:
150
            if repository.get_physical_lock_status():
151
                status = 'locked'
152
            else:
153
                status = 'unlocked'
154
            print '    repository: %s' % status
155
156
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
157
def _show_missing_revisions_branch(branch):
158
    """Show missing master revisions in branch."""
1587.1.14 by Robert Collins
Make bound branch creation happen via 'checkout'
159
    # Try with inaccessible branch ?
1624.3.2 by Olaf Conradi
Implemented table of constructs from BzrInfo specification.
160
    master = branch.get_master_branch()
1587.1.14 by Robert Collins
Make bound branch creation happen via 'checkout'
161
    if master:
1624.3.2 by Olaf Conradi
Implemented table of constructs from BzrInfo specification.
162
        local_extra, remote_extra = find_unmerged(branch, master)
1587.1.14 by Robert Collins
Make bound branch creation happen via 'checkout'
163
        if remote_extra:
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
164
            print
1587.1.14 by Robert Collins
Make bound branch creation happen via 'checkout'
165
            print 'Branch is out of date: missing %d revision%s.' % (
166
                len(remote_extra), plural(len(remote_extra)))
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
167
168
169
def _show_missing_revisions_working(working):
170
    """Show missing revisions in working tree."""
171
    branch = working.branch
172
    basis = working.basis_tree()
173
    work_inv = working.inventory
1852.10.3 by Robert Collins
Remove all uses of compare_trees and replace with Tree.changes_from throughout bzrlib.
174
    delta = working.changes_from(basis, want_unchanged=True)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
175
    history = branch.revision_history()
1908.7.6 by Robert Collins
Deprecate WorkingTree.last_revision.
176
    try:
177
        tree_last_id = working.get_parent_ids()[0]
178
    except IndexError:
179
        tree_last_id = None
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
180
1624.3.11 by Olaf Conradi
Test cases exposed a bug in missing revisions count of working tree. It
181
    if len(history) and tree_last_id != history[-1]:
182
        tree_last_revno = branch.revision_id_to_revno(tree_last_id)
183
        missing_count = len(history) - tree_last_revno
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
184
        print
1534.5.1 by Robert Collins
Give info some reasonable output and tests.
185
        print 'Working tree is out of date: missing %d revision%s.' % (
186
            missing_count, plural(missing_count))
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
187
188
189
def _show_working_stats(working):
190
    """Show statistics about a working tree."""
191
    basis = working.basis_tree()
192
    work_inv = working.inventory
1852.10.3 by Robert Collins
Remove all uses of compare_trees and replace with Tree.changes_from throughout bzrlib.
193
    delta = working.changes_from(basis, want_unchanged=True)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
194
195
    print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
196
    print 'In the working tree:'
463 by Martin Pool
- compare_trees() also reports unchanged files
197
    print '  %8s unchanged' % len(delta.unchanged)
462 by Martin Pool
- New form 'file_id in tree' to check if the file is present
198
    print '  %8d modified' % len(delta.modified)
199
    print '  %8d added' % len(delta.added)
200
    print '  %8d removed' % len(delta.removed)
201
    print '  %8d renamed' % len(delta.renamed)
202
203
    ignore_cnt = unknown_cnt = 0
204
    for path in working.extras():
205
        if working.is_ignored(path):
206
            ignore_cnt += 1
207
        else:
208
            unknown_cnt += 1
209
    print '  %8d unknown' % unknown_cnt
210
    print '  %8d ignored' % ignore_cnt
211
212
    dir_cnt = 0
1731.1.39 by Aaron Bentley
Reject removing is_root
213
    for file_id in work_inv:
214
        if (work_inv.get_file_kind(file_id) == 'directory' and 
215
            not work_inv.is_root(file_id)):
216
            dir_cnt += 1
462 by Martin Pool
- New form 'file_id in tree' to check if the file is present
217
    print '  %8d versioned %s' \
218
          % (dir_cnt,
219
             plural(dir_cnt, 'subdirectory', 'subdirectories'))
77 by mbp at sourcefrog
- split info command out into separate file
220
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
221
222
def _show_branch_stats(branch, verbose):
223
    """Show statistics about a branch."""
224
    repository = branch.repository
225
    history = branch.revision_history()
226
77 by mbp at sourcefrog
- split info command out into separate file
227
    print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
228
    print 'Branch history:'
77 by mbp at sourcefrog
- split info command out into separate file
229
    revno = len(history)
111 by mbp at sourcefrog
Make fields wider in 'bzr info' output to accomodate big trees
230
    print '  %8d revision%s' % (revno, plural(revno))
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
231
    if verbose:
232
        committers = {}
233
        for rev in history:
234
            committers[repository.get_revision(rev).committer] = True
235
        print '  %8d committer%s' % (len(committers), plural(len(committers)))
77 by mbp at sourcefrog
- split info command out into separate file
236
    if revno > 0:
1624.3.17 by Olaf Conradi
Remove indirection in branch.repository as it is available as repository
237
        firstrev = repository.get_revision(history[0])
77 by mbp at sourcefrog
- split info command out into separate file
238
        age = int((time.time() - firstrev.timestamp) / 3600 / 24)
111 by mbp at sourcefrog
Make fields wider in 'bzr info' output to accomodate big trees
239
        print '  %8d day%s old' % (age, plural(age))
1694.2.6 by Martin Pool
[merge] bzr.dev
240
        print '   first revision: %s' % osutils.format_date(firstrev.timestamp,
241
                                                            firstrev.timezone)
77 by mbp at sourcefrog
- split info command out into separate file
242
1624.3.17 by Olaf Conradi
Remove indirection in branch.repository as it is available as repository
243
        lastrev = repository.get_revision(history[-1])
1694.2.6 by Martin Pool
[merge] bzr.dev
244
        print '  latest revision: %s' % osutils.format_date(lastrev.timestamp,
245
                                                            lastrev.timezone)
80 by mbp at sourcefrog
show_info: Show number of entries in the branch stores
246
1286 by Martin Pool
- stub out display of store size in info command
247
#     print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
248
#     print 'Text store:'
1624.3.2 by Olaf Conradi
Implemented table of constructs from BzrInfo specification.
249
#     c, t = branch.text_store.total_size()
1286 by Martin Pool
- stub out display of store size in info command
250
#     print '  %8d file texts' % c
1624.3.14 by Olaf Conradi
Move to using kibi for binary prefix as per standard IEEE 1541.
251
#     print '  %8d KiB' % (t/1024)
80 by mbp at sourcefrog
show_info: Show number of entries in the branch stores
252
1286 by Martin Pool
- stub out display of store size in info command
253
#     print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
254
#     print 'Inventory store:'
1624.3.2 by Olaf Conradi
Implemented table of constructs from BzrInfo specification.
255
#     c, t = branch.inventory_store.total_size()
1286 by Martin Pool
- stub out display of store size in info command
256
#     print '  %8d inventories' % c
1624.3.14 by Olaf Conradi
Move to using kibi for binary prefix as per standard IEEE 1541.
257
#     print '  %8d KiB' % (t/1024)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
258
259
260
def _show_repository_info(repository):
261
    """Show settings of a repository."""
262
    if repository.make_working_trees():
263
        print
264
        print 'Create working tree for new branches inside the repository.'
265
266
267
def _show_repository_stats(repository):
268
    """Show statistics about a repository."""
269
    if repository.bzrdir.root_transport.listable():
270
        print
271
        print 'Revision store:'
272
        c, t = repository._revision_store.total_size(repository.get_transaction())
273
        print '  %8d revision%s' % (c, plural(c))
274
        print '  %8d KiB' % (t/1024)
275
276
277
@deprecated_function(zero_eight)
278
def show_info(b):
279
    """Please see show_bzrdir_info."""
280
    return show_bzrdir_info(b.bzrdir)
281
282
283
def show_bzrdir_info(a_bzrdir, verbose=False):
284
    """Output to stdout the 'info' for a_bzrdir."""
285
    try:
286
        working = a_bzrdir.open_workingtree()
1624.3.35 by Olaf Conradi
Implemented locking status for formats using LockDir.
287
        working.lock_read()
288
        try:
289
            show_tree_info(working, verbose)
290
        finally:
291
            working.unlock()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
292
        return
293
    except (NoWorkingTree, NotLocalUrl):
294
        pass
295
296
    try:
297
        branch = a_bzrdir.open_branch()
1624.3.35 by Olaf Conradi
Implemented locking status for formats using LockDir.
298
        branch.lock_read()
299
        try:
300
            show_branch_info(branch, verbose)
301
        finally:
302
            branch.unlock()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
303
        return
304
    except NotBranchError:
305
        pass
306
307
    try:
308
        repository = a_bzrdir.open_repository()
1624.3.35 by Olaf Conradi
Implemented locking status for formats using LockDir.
309
        repository.lock_read()
310
        try:
311
            show_repository_info(repository, verbose)
312
        finally:
313
            repository.unlock()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
314
        return
315
    except NoRepositoryPresent:
316
        pass
317
1694.2.6 by Martin Pool
[merge] bzr.dev
318
    # Return silently, cmd_info already returned NotBranchError if no bzrdir
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
319
    # could be opened.
320
321
322
def show_tree_info(working, verbose):
323
    """Output to stdout the 'info' for working."""
324
    branch = working.branch
325
    repository = branch.repository
326
    control = working.bzrdir
327
328
    _show_location_info(repository, branch, working)
1694.2.6 by Martin Pool
[merge] bzr.dev
329
    _show_related_info(branch)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
330
    _show_format_info(control, repository, branch, working)
1694.2.6 by Martin Pool
[merge] bzr.dev
331
    _show_locking_info(repository, branch, working)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
332
    _show_missing_revisions_branch(branch)
333
    _show_missing_revisions_working(working)
334
    _show_working_stats(working)
335
    _show_branch_stats(branch, verbose)
336
    _show_repository_stats(repository)
337
338
339
def show_branch_info(branch, verbose):
340
    """Output to stdout the 'info' for branch."""
341
    repository = branch.repository
342
    control = branch.bzrdir
343
344
    _show_location_info(repository, branch)
1694.2.6 by Martin Pool
[merge] bzr.dev
345
    _show_related_info(branch)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
346
    _show_format_info(control, repository, branch)
1694.2.6 by Martin Pool
[merge] bzr.dev
347
    _show_locking_info(repository, branch)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
348
    _show_missing_revisions_branch(branch)
349
    _show_branch_stats(branch, verbose)
350
    _show_repository_stats(repository)
351
352
353
def show_repository_info(repository, verbose):
1694.2.6 by Martin Pool
[merge] bzr.dev
354
    """Output to stdout the 'info' for repository."""
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
355
    control = repository.bzrdir
356
357
    _show_location_info(repository)
358
    _show_format_info(control, repository)
1694.2.6 by Martin Pool
[merge] bzr.dev
359
    _show_locking_info(repository)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
360
    _show_repository_info(repository)
361
    _show_repository_stats(repository)