~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/info.py

  • Committer: Martin Pool
  • Date: 2006-05-10 03:49:42 UTC
  • mfrom: (1624.3.47 bzr.olaf.info)
  • mto: This revision was merged to the branch mainline in revision 1707.
  • Revision ID: mbp@sourcefrog.net-20060510034942-d317207825dff3c1
(merge olaf) info test improvements

Show diffs side-by-side

added added

removed removed

Lines of Context:
16
16
# along with this program; if not, write to the Free Software
17
17
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18
18
 
19
 
from sets import Set
 
19
__all__ = ['show_bzrdir_info']
 
20
 
20
21
import time
21
22
 
22
 
import bzrlib
23
 
from osutils import format_date
24
 
 
25
 
def show_info(b):
26
 
    print 'branch format:', b.controlfile('branch-format', 'r').readline().rstrip('\n')
27
 
 
28
 
    def plural(n, base='', pl=None):
29
 
        if n == 1:
30
 
            return base
31
 
        elif pl != None:
32
 
            return pl
33
 
        else:
34
 
            return 's'
35
 
 
36
 
    count_version_dirs = 0
37
 
 
38
 
    count_status = {'A': 0, 'D': 0, 'M': 0, 'R': 0, '?': 0, 'I': 0, '.': 0}
39
 
    for st_tup in bzrlib.diff_trees(b.basis_tree(), b.working_tree()):
40
 
        fs = st_tup[0]
41
 
        count_status[fs] += 1
42
 
        if fs not in ['I', '?'] and st_tup[4] == 'directory':
43
 
            count_version_dirs += 1
44
 
 
45
 
    print
46
 
    print 'in the working tree:'
47
 
    for name, fs in (('unchanged', '.'),
48
 
                     ('modified', 'M'), ('added', 'A'), ('removed', 'D'),
49
 
                     ('renamed', 'R'), ('unknown', '?'), ('ignored', 'I'),
50
 
                     ):
51
 
        print '  %8d %s' % (count_status[fs], name)
52
 
    print '  %8d versioned subdirector%s' % (count_version_dirs,
53
 
                                             plural(count_version_dirs, 'y', 'ies'))
54
 
 
55
 
    print
56
 
    print 'branch history:'
57
 
    history = b.revision_history()
 
23
 
 
24
import bzrlib.diff as diff
 
25
from bzrlib.errors import (NoWorkingTree, NotBranchError,
 
26
                           NoRepositoryPresent, NotLocalUrl)
 
27
from bzrlib.missing import find_unmerged
 
28
import bzrlib.osutils as osutils
 
29
from bzrlib.symbol_versioning import *
 
30
 
 
31
 
 
32
def plural(n, base='', pl=None):
 
33
    if n == 1:
 
34
        return base
 
35
    elif pl != None:
 
36
        return pl
 
37
    else:
 
38
        return 's'
 
39
 
 
40
 
 
41
def _repo_relpath(repo_path, path):
 
42
    """Return path with common prefix of repository path removed.
 
43
 
 
44
    If path is not part of the repository, the original path is returned.
 
45
    If path is equal to the repository, the current directory marker '.' is
 
46
    returned.
 
47
    """
 
48
    path = osutils.normalizepath(path)
 
49
    repo_path = osutils.normalizepath(repo_path)
 
50
    if path == repo_path:
 
51
        return '.'
 
52
    if osutils.is_inside(repo_path, path):
 
53
        return osutils.relpath(repo_path, path)
 
54
    return path
 
55
 
 
56
 
 
57
def _show_location_info(repository, branch=None, working=None):
 
58
    """Show known locations for working, branch and repository."""
 
59
    repository_path = repository.bzrdir.root_transport.base
 
60
    print 'Location:'
 
61
    if working and branch:
 
62
        working_path = working.bzrdir.root_transport.base
 
63
        branch_path = branch.bzrdir.root_transport.base
 
64
        if working_path != branch_path:
 
65
            # lightweight checkout
 
66
            print '  light checkout root: %s' % working_path
 
67
            if repository.is_shared():
 
68
                # lightweight checkout of branch in shared repository
 
69
                print '    shared repository: %s' % repository_path
 
70
                print '    repository branch: %s' % (
 
71
                    _repo_relpath(repository_path, branch_path))
 
72
            else:
 
73
                # lightweight checkout of standalone branch
 
74
                print '   checkout of branch: %s' % branch_path
 
75
        elif repository.is_shared():
 
76
            # branch with tree inside shared repository
 
77
            print '    shared repository: %s' % repository_path
 
78
            print '  repository checkout: %s' % (
 
79
                _repo_relpath(repository_path, branch_path))
 
80
        elif branch.get_bound_location():
 
81
            # normal checkout
 
82
            print '       checkout root: %s' % working_path
 
83
            print '  checkout of branch: %s' % branch.get_bound_location()
 
84
        else:
 
85
            # standalone
 
86
            print '  branch root: %s' % working_path
 
87
    elif branch:
 
88
        # branch is part of shared repository
 
89
        assert repository.is_shared()
 
90
        branch_path = branch.bzrdir.root_transport.base
 
91
        print '  shared repository: %s' % repository_path
 
92
        print '  repository branch: %s' % (
 
93
            _repo_relpath(repository_path, branch_path))
 
94
    else:
 
95
        # shared repository
 
96
        assert repository.is_shared()
 
97
        print '  shared repository: %s' % repository_path
 
98
 
 
99
 
 
100
def _show_related_info(branch):
 
101
    """Show parent and push location of branch."""
 
102
    if branch.get_parent() or branch.get_push_location():
 
103
        print
 
104
        print 'Related branches:'
 
105
        if branch.get_parent():
 
106
            if branch.get_push_location():
 
107
                print '      parent branch: %s' % branch.get_parent()
 
108
            else:
 
109
                print '  parent branch: %s' % branch.get_parent()
 
110
        if branch.get_push_location():
 
111
            print '  publish to branch: %s' % branch.get_push_location()
 
112
 
 
113
 
 
114
def _show_format_info(control=None, repository=None, branch=None, working=None):
 
115
    """Show known formats for control, working, branch and repository."""
 
116
    print
 
117
    print 'Format:'
 
118
    if control:
 
119
        print '       control: %s' % control._format.get_format_description()
 
120
    if working:
 
121
        print '  working tree: %s' % working._format.get_format_description()
 
122
    if branch:
 
123
        print '        branch: %s' % branch._format.get_format_description()
 
124
    if repository:
 
125
        print '    repository: %s' % repository._format.get_format_description()
 
126
 
 
127
 
 
128
def _show_locking_info(repository, branch=None, working=None):
 
129
    """Show locking status of working, branch and repository."""
 
130
    if (repository.get_physical_lock_status() or
 
131
        (branch and branch.get_physical_lock_status()) or
 
132
        (working and working.get_physical_lock_status())):
 
133
        print
 
134
        print 'Lock status:'
 
135
        if working:
 
136
            if working.get_physical_lock_status():
 
137
                status = 'locked'
 
138
            else:
 
139
                status = 'unlocked'
 
140
            print '  working tree: %s' % status
 
141
        if branch:
 
142
            if branch.get_physical_lock_status():
 
143
                status = 'locked'
 
144
            else:
 
145
                status = 'unlocked'
 
146
            print '        branch: %s' % status
 
147
        if repository:
 
148
            if repository.get_physical_lock_status():
 
149
                status = 'locked'
 
150
            else:
 
151
                status = 'unlocked'
 
152
            print '    repository: %s' % status
 
153
 
 
154
 
 
155
def _show_missing_revisions_branch(branch):
 
156
    """Show missing master revisions in branch."""
 
157
    # Try with inaccessible branch ?
 
158
    master = branch.get_master_branch()
 
159
    if master:
 
160
        local_extra, remote_extra = find_unmerged(branch, master)
 
161
        if remote_extra:
 
162
            print
 
163
            print 'Branch is out of date: missing %d revision%s.' % (
 
164
                len(remote_extra), plural(len(remote_extra)))
 
165
 
 
166
 
 
167
def _show_missing_revisions_working(working):
 
168
    """Show missing revisions in working tree."""
 
169
    branch = working.branch
 
170
    basis = working.basis_tree()
 
171
    work_inv = working.inventory
 
172
    delta = diff.compare_trees(basis, working, want_unchanged=True)
 
173
    history = branch.revision_history()
 
174
    tree_last_id = working.last_revision()
 
175
 
 
176
    if len(history) and tree_last_id != history[-1]:
 
177
        tree_last_revno = branch.revision_id_to_revno(tree_last_id)
 
178
        missing_count = len(history) - tree_last_revno
 
179
        print
 
180
        print 'Working tree is out of date: missing %d revision%s.' % (
 
181
            missing_count, plural(missing_count))
 
182
 
 
183
 
 
184
def _show_working_stats(working):
 
185
    """Show statistics about a working tree."""
 
186
    basis = working.basis_tree()
 
187
    work_inv = working.inventory
 
188
    delta = diff.compare_trees(basis, working, want_unchanged=True)
 
189
 
 
190
    print
 
191
    print 'In the working tree:'
 
192
    print '  %8s unchanged' % len(delta.unchanged)
 
193
    print '  %8d modified' % len(delta.modified)
 
194
    print '  %8d added' % len(delta.added)
 
195
    print '  %8d removed' % len(delta.removed)
 
196
    print '  %8d renamed' % len(delta.renamed)
 
197
 
 
198
    ignore_cnt = unknown_cnt = 0
 
199
    for path in working.extras():
 
200
        if working.is_ignored(path):
 
201
            ignore_cnt += 1
 
202
        else:
 
203
            unknown_cnt += 1
 
204
    print '  %8d unknown' % unknown_cnt
 
205
    print '  %8d ignored' % ignore_cnt
 
206
 
 
207
    dir_cnt = 0
 
208
    for file_id in work_inv:
 
209
        if work_inv.get_file_kind(file_id) == 'directory':
 
210
            dir_cnt += 1
 
211
    print '  %8d versioned %s' \
 
212
          % (dir_cnt,
 
213
             plural(dir_cnt, 'subdirectory', 'subdirectories'))
 
214
 
 
215
 
 
216
def _show_branch_stats(branch, verbose):
 
217
    """Show statistics about a branch."""
 
218
    repository = branch.repository
 
219
    history = branch.revision_history()
 
220
 
 
221
    print
 
222
    print 'Branch history:'
58
223
    revno = len(history)
59
224
    print '  %8d revision%s' % (revno, plural(revno))
60
 
    committers = Set()
61
 
    for rev in history:
62
 
        committers.add(b.get_revision(rev).committer)
63
 
    print '  %8d committer%s' % (len(committers), plural(len(committers)))
 
225
    if verbose:
 
226
        committers = {}
 
227
        for rev in history:
 
228
            committers[repository.get_revision(rev).committer] = True
 
229
        print '  %8d committer%s' % (len(committers), plural(len(committers)))
64
230
    if revno > 0:
65
 
        firstrev = b.get_revision(history[0])
 
231
        firstrev = repository.get_revision(history[0])
66
232
        age = int((time.time() - firstrev.timestamp) / 3600 / 24)
67
233
        print '  %8d day%s old' % (age, plural(age))
68
 
        print '   first revision: %s' % format_date(firstrev.timestamp,
69
 
                                                    firstrev.timezone)
70
 
 
71
 
        lastrev = b.get_revision(history[-1])
72
 
        print '  latest revision: %s' % format_date(lastrev.timestamp,
73
 
                                                    lastrev.timezone)
74
 
 
75
 
    print
76
 
    print 'text store:'
77
 
    c, t = b.text_store.total_size()
78
 
    print '  %8d file texts' % c
79
 
    print '  %8d kB' % (t/1024)
80
 
 
81
 
    print
82
 
    print 'revision store:'
83
 
    c, t = b.revision_store.total_size()
84
 
    print '  %8d revisions' % c
85
 
    print '  %8d kB' % (t/1024)
86
 
 
87
 
 
88
 
    print
89
 
    print 'inventory store:'
90
 
    c, t = b.inventory_store.total_size()
91
 
    print '  %8d inventories' % c
92
 
    print '  %8d kB' % (t/1024)
93
 
 
 
234
        print '   first revision: %s' % osutils.format_date(firstrev.timestamp,
 
235
                                                            firstrev.timezone)
 
236
 
 
237
        lastrev = repository.get_revision(history[-1])
 
238
        print '  latest revision: %s' % osutils.format_date(lastrev.timestamp,
 
239
                                                            lastrev.timezone)
 
240
 
 
241
#     print
 
242
#     print 'Text store:'
 
243
#     c, t = branch.text_store.total_size()
 
244
#     print '  %8d file texts' % c
 
245
#     print '  %8d KiB' % (t/1024)
 
246
 
 
247
#     print
 
248
#     print 'Inventory store:'
 
249
#     c, t = branch.inventory_store.total_size()
 
250
#     print '  %8d inventories' % c
 
251
#     print '  %8d KiB' % (t/1024)
 
252
 
 
253
 
 
254
def _show_repository_info(repository):
 
255
    """Show settings of a repository."""
 
256
    if repository.make_working_trees():
 
257
        print
 
258
        print 'Create working tree for new branches inside the repository.'
 
259
 
 
260
 
 
261
def _show_repository_stats(repository):
 
262
    """Show statistics about a repository."""
 
263
    if repository.bzrdir.root_transport.listable():
 
264
        print
 
265
        print 'Revision store:'
 
266
        c, t = repository._revision_store.total_size(repository.get_transaction())
 
267
        print '  %8d revision%s' % (c, plural(c))
 
268
        print '  %8d KiB' % (t/1024)
 
269
 
 
270
 
 
271
@deprecated_function(zero_eight)
 
272
def show_info(b):
 
273
    """Please see show_bzrdir_info."""
 
274
    return show_bzrdir_info(b.bzrdir)
 
275
 
 
276
 
 
277
def show_bzrdir_info(a_bzrdir, verbose=False):
 
278
    """Output to stdout the 'info' for a_bzrdir."""
 
279
    try:
 
280
        working = a_bzrdir.open_workingtree()
 
281
        working.lock_read()
 
282
        try:
 
283
            show_tree_info(working, verbose)
 
284
        finally:
 
285
            working.unlock()
 
286
        return
 
287
    except (NoWorkingTree, NotLocalUrl):
 
288
        pass
 
289
 
 
290
    try:
 
291
        branch = a_bzrdir.open_branch()
 
292
        branch.lock_read()
 
293
        try:
 
294
            show_branch_info(branch, verbose)
 
295
        finally:
 
296
            branch.unlock()
 
297
        return
 
298
    except NotBranchError:
 
299
        pass
 
300
 
 
301
    try:
 
302
        repository = a_bzrdir.open_repository()
 
303
        repository.lock_read()
 
304
        try:
 
305
            show_repository_info(repository, verbose)
 
306
        finally:
 
307
            repository.unlock()
 
308
        return
 
309
    except NoRepositoryPresent:
 
310
        pass
 
311
 
 
312
    # Return silently, cmd_info already returned NotBranchError if no bzrdir
 
313
    # could be opened.
 
314
 
 
315
 
 
316
def show_tree_info(working, verbose):
 
317
    """Output to stdout the 'info' for working."""
 
318
    branch = working.branch
 
319
    repository = branch.repository
 
320
    control = working.bzrdir
 
321
 
 
322
    _show_location_info(repository, branch, working)
 
323
    _show_related_info(branch)
 
324
    _show_format_info(control, repository, branch, working)
 
325
    _show_locking_info(repository, branch, working)
 
326
    _show_missing_revisions_branch(branch)
 
327
    _show_missing_revisions_working(working)
 
328
    _show_working_stats(working)
 
329
    _show_branch_stats(branch, verbose)
 
330
    _show_repository_stats(repository)
 
331
 
 
332
 
 
333
def show_branch_info(branch, verbose):
 
334
    """Output to stdout the 'info' for branch."""
 
335
    repository = branch.repository
 
336
    control = branch.bzrdir
 
337
 
 
338
    _show_location_info(repository, branch)
 
339
    _show_related_info(branch)
 
340
    _show_format_info(control, repository, branch)
 
341
    _show_locking_info(repository, branch)
 
342
    _show_missing_revisions_branch(branch)
 
343
    _show_branch_stats(branch, verbose)
 
344
    _show_repository_stats(repository)
 
345
 
 
346
 
 
347
def show_repository_info(repository, verbose):
 
348
    """Output to stdout the 'info' for repository."""
 
349
    control = repository.bzrdir
 
350
 
 
351
    _show_location_info(repository)
 
352
    _show_format_info(control, repository)
 
353
    _show_locking_info(repository)
 
354
    _show_repository_info(repository)
 
355
    _show_repository_stats(repository)