~bzr-pqm/bzr/bzr.dev

1685.1.60 by Martin Pool
[broken] NotBranchError should unescape the url if possible
1
# Copyright (C) 2005, 2006 by Canonical Ltd
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.
1685.1.60 by Martin Pool
[broken] NotBranchError should unescape the url if possible
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.
1685.1.60 by Martin Pool
[broken] NotBranchError should unescape the url if possible
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
34
    elif pl != None:
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()
1624.3.11 by Olaf Conradi
Test cases exposed a bug in missing revisions count of working tree. It
176
    tree_last_id = working.last_revision()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
177
1624.3.11 by Olaf Conradi
Test cases exposed a bug in missing revisions count of working tree. It
178
    if len(history) and tree_last_id != history[-1]:
179
        tree_last_revno = branch.revision_id_to_revno(tree_last_id)
180
        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
181
        print
1534.5.1 by Robert Collins
Give info some reasonable output and tests.
182
        print 'Working tree is out of date: missing %d revision%s.' % (
183
            missing_count, plural(missing_count))
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
184
185
186
def _show_working_stats(working):
187
    """Show statistics about a working tree."""
188
    basis = working.basis_tree()
189
    work_inv = working.inventory
1852.10.3 by Robert Collins
Remove all uses of compare_trees and replace with Tree.changes_from throughout bzrlib.
190
    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
191
192
    print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
193
    print 'In the working tree:'
463 by Martin Pool
- compare_trees() also reports unchanged files
194
    print '  %8s unchanged' % len(delta.unchanged)
462 by Martin Pool
- New form 'file_id in tree' to check if the file is present
195
    print '  %8d modified' % len(delta.modified)
196
    print '  %8d added' % len(delta.added)
197
    print '  %8d removed' % len(delta.removed)
198
    print '  %8d renamed' % len(delta.renamed)
199
200
    ignore_cnt = unknown_cnt = 0
201
    for path in working.extras():
202
        if working.is_ignored(path):
203
            ignore_cnt += 1
204
        else:
205
            unknown_cnt += 1
206
    print '  %8d unknown' % unknown_cnt
207
    print '  %8d ignored' % ignore_cnt
208
209
    dir_cnt = 0
210
    for file_id in work_inv:
211
        if work_inv.get_file_kind(file_id) == 'directory':
212
            dir_cnt += 1
213
    print '  %8d versioned %s' \
214
          % (dir_cnt,
215
             plural(dir_cnt, 'subdirectory', 'subdirectories'))
77 by mbp at sourcefrog
- split info command out into separate file
216
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
217
218
def _show_branch_stats(branch, verbose):
219
    """Show statistics about a branch."""
220
    repository = branch.repository
221
    history = branch.revision_history()
222
77 by mbp at sourcefrog
- split info command out into separate file
223
    print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
224
    print 'Branch history:'
77 by mbp at sourcefrog
- split info command out into separate file
225
    revno = len(history)
111 by mbp at sourcefrog
Make fields wider in 'bzr info' output to accomodate big trees
226
    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
227
    if verbose:
228
        committers = {}
229
        for rev in history:
230
            committers[repository.get_revision(rev).committer] = True
231
        print '  %8d committer%s' % (len(committers), plural(len(committers)))
77 by mbp at sourcefrog
- split info command out into separate file
232
    if revno > 0:
1624.3.17 by Olaf Conradi
Remove indirection in branch.repository as it is available as repository
233
        firstrev = repository.get_revision(history[0])
77 by mbp at sourcefrog
- split info command out into separate file
234
        age = int((time.time() - firstrev.timestamp) / 3600 / 24)
111 by mbp at sourcefrog
Make fields wider in 'bzr info' output to accomodate big trees
235
        print '  %8d day%s old' % (age, plural(age))
1694.2.6 by Martin Pool
[merge] bzr.dev
236
        print '   first revision: %s' % osutils.format_date(firstrev.timestamp,
237
                                                            firstrev.timezone)
77 by mbp at sourcefrog
- split info command out into separate file
238
1624.3.17 by Olaf Conradi
Remove indirection in branch.repository as it is available as repository
239
        lastrev = repository.get_revision(history[-1])
1694.2.6 by Martin Pool
[merge] bzr.dev
240
        print '  latest revision: %s' % osutils.format_date(lastrev.timestamp,
241
                                                            lastrev.timezone)
80 by mbp at sourcefrog
show_info: Show number of entries in the branch stores
242
1286 by Martin Pool
- stub out display of store size in info command
243
#     print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
244
#     print 'Text store:'
1624.3.2 by Olaf Conradi
Implemented table of constructs from BzrInfo specification.
245
#     c, t = branch.text_store.total_size()
1286 by Martin Pool
- stub out display of store size in info command
246
#     print '  %8d file texts' % c
1624.3.14 by Olaf Conradi
Move to using kibi for binary prefix as per standard IEEE 1541.
247
#     print '  %8d KiB' % (t/1024)
80 by mbp at sourcefrog
show_info: Show number of entries in the branch stores
248
1286 by Martin Pool
- stub out display of store size in info command
249
#     print
1624.3.8 by Olaf Conradi
Adjust formatting. Start sections with a capital letter.
250
#     print 'Inventory store:'
1624.3.2 by Olaf Conradi
Implemented table of constructs from BzrInfo specification.
251
#     c, t = branch.inventory_store.total_size()
1286 by Martin Pool
- stub out display of store size in info command
252
#     print '  %8d inventories' % c
1624.3.14 by Olaf Conradi
Move to using kibi for binary prefix as per standard IEEE 1541.
253
#     print '  %8d KiB' % (t/1024)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
254
255
256
def _show_repository_info(repository):
257
    """Show settings of a repository."""
258
    if repository.make_working_trees():
259
        print
260
        print 'Create working tree for new branches inside the repository.'
261
262
263
def _show_repository_stats(repository):
264
    """Show statistics about a repository."""
265
    if repository.bzrdir.root_transport.listable():
266
        print
267
        print 'Revision store:'
268
        c, t = repository._revision_store.total_size(repository.get_transaction())
269
        print '  %8d revision%s' % (c, plural(c))
270
        print '  %8d KiB' % (t/1024)
271
272
273
@deprecated_function(zero_eight)
274
def show_info(b):
275
    """Please see show_bzrdir_info."""
276
    return show_bzrdir_info(b.bzrdir)
277
278
279
def show_bzrdir_info(a_bzrdir, verbose=False):
280
    """Output to stdout the 'info' for a_bzrdir."""
281
    try:
282
        working = a_bzrdir.open_workingtree()
1624.3.35 by Olaf Conradi
Implemented locking status for formats using LockDir.
283
        working.lock_read()
284
        try:
285
            show_tree_info(working, verbose)
286
        finally:
287
            working.unlock()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
288
        return
289
    except (NoWorkingTree, NotLocalUrl):
290
        pass
291
292
    try:
293
        branch = a_bzrdir.open_branch()
1624.3.35 by Olaf Conradi
Implemented locking status for formats using LockDir.
294
        branch.lock_read()
295
        try:
296
            show_branch_info(branch, verbose)
297
        finally:
298
            branch.unlock()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
299
        return
300
    except NotBranchError:
301
        pass
302
303
    try:
304
        repository = a_bzrdir.open_repository()
1624.3.35 by Olaf Conradi
Implemented locking status for formats using LockDir.
305
        repository.lock_read()
306
        try:
307
            show_repository_info(repository, verbose)
308
        finally:
309
            repository.unlock()
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
310
        return
311
    except NoRepositoryPresent:
312
        pass
313
1694.2.6 by Martin Pool
[merge] bzr.dev
314
    # 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
315
    # could be opened.
316
317
318
def show_tree_info(working, verbose):
319
    """Output to stdout the 'info' for working."""
320
    branch = working.branch
321
    repository = branch.repository
322
    control = working.bzrdir
323
324
    _show_location_info(repository, branch, working)
1694.2.6 by Martin Pool
[merge] bzr.dev
325
    _show_related_info(branch)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
326
    _show_format_info(control, repository, branch, working)
1694.2.6 by Martin Pool
[merge] bzr.dev
327
    _show_locking_info(repository, branch, working)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
328
    _show_missing_revisions_branch(branch)
329
    _show_missing_revisions_working(working)
330
    _show_working_stats(working)
331
    _show_branch_stats(branch, verbose)
332
    _show_repository_stats(repository)
333
334
335
def show_branch_info(branch, verbose):
336
    """Output to stdout the 'info' for branch."""
337
    repository = branch.repository
338
    control = branch.bzrdir
339
340
    _show_location_info(repository, branch)
1694.2.6 by Martin Pool
[merge] bzr.dev
341
    _show_related_info(branch)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
342
    _show_format_info(control, repository, branch)
1694.2.6 by Martin Pool
[merge] bzr.dev
343
    _show_locking_info(repository, branch)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
344
    _show_missing_revisions_branch(branch)
345
    _show_branch_stats(branch, verbose)
346
    _show_repository_stats(repository)
347
348
349
def show_repository_info(repository, verbose):
1694.2.6 by Martin Pool
[merge] bzr.dev
350
    """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
351
    control = repository.bzrdir
352
353
    _show_location_info(repository)
354
    _show_format_info(control, repository)
1694.2.6 by Martin Pool
[merge] bzr.dev
355
    _show_locking_info(repository)
1624.3.21 by Olaf Conradi
Make bzr info command work on both local and remote locations. Support
356
    _show_repository_info(repository)
357
    _show_repository_stats(repository)