47
56
# merges from, then it should still be reported as newly added
48
57
# relative to the basis revision.
50
# TODO: Change the parameter 'rev_id' to 'revision_id' to be consistent with
51
# the rest of the code; add a deprecation of the old name.
60
from .branch import Branch
61
from .cleanup import OperationWithCleanups
63
from .errors import (BzrError,
67
from .osutils import (get_user_encoding,
69
minimum_path_selection,
72
from .trace import mutter, note, is_quiet
73
from .urlutils import unescape_for_display
74
from .i18n import gettext
77
class PointlessCommit(BzrError):
79
_fmt = "No changes to commit"
82
class CannotCommitSelectedFileMerge(BzrError):
84
_fmt = 'Selected-file commit of merges is not supported yet:'\
85
' files %(files_str)s'
87
def __init__(self, files):
88
files_str = ', '.join(files)
89
BzrError.__init__(self, files=files, files_str=files_str)
92
def filter_excluded(iter_changes, exclude):
93
"""Filter exclude filenames.
95
:param iter_changes: iter_changes function
96
:param exclude: List of paths to exclude
97
:return: iter_changes function
59
# TODO: Do checks that the tree can be committed *before* running the
60
# editor; this should include checks for a pointless commit and for
61
# unknown or missing files.
63
# TODO: If commit fails, leave the message in a file somewhere.
72
from binascii import hexlify
73
from cStringIO import StringIO
75
from bzrlib.atomicfile import AtomicFile
76
from bzrlib.osutils import (local_time_offset,
77
rand_bytes, compact_date,
78
kind_marker, is_inside_any, quotefn,
79
sha_file, isdir, isfile,
82
import bzrlib.errors as errors
83
from bzrlib.errors import (BzrError, PointlessCommit,
88
from bzrlib.revision import Revision
89
from bzrlib.testament import Testament
90
from bzrlib.trace import mutter, note, warning
91
from bzrlib.xml5 import serializer_v5
92
from bzrlib.inventory import Inventory, InventoryEntry
93
from bzrlib.symbol_versioning import *
94
from bzrlib.workingtree import WorkingTree
97
@deprecated_function(zero_seven)
98
def commit(*args, **kwargs):
99
"""Commit a new revision to a branch.
101
Function-style interface for convenience of old callers.
103
New code should use the Commit class instead.
99
for change in iter_changes:
100
old_path = change[1][0]
101
new_path = change[1][1]
103
new_excluded = (new_path is not None and
104
is_inside_any(exclude, new_path))
106
old_excluded = (old_path is not None and
107
is_inside_any(exclude, old_path))
109
if old_excluded and new_excluded:
112
if old_excluded or new_excluded:
113
# TODO(jelmer): Perhaps raise an error here instead?
105
## XXX: Remove this in favor of Branch.commit?
106
Commit().commit(*args, **kwargs)
119
109
class NullCommitReporter(object):
120
110
"""I report on progress of a commit."""
122
def started(self, revno, revid, location):
125
112
def snapshot_change(self, change, path):
128
115
def completed(self, revno, rev_id):
131
def deleted(self, path):
118
def deleted(self, file_id):
121
def escaped(self, escape_count, message):
134
124
def missing(self, path):
242
194
working_tree=None,
246
message_callback=None,
249
possible_master_transports=None,
251
198
"""Commit working copy as a new revision.
253
:param message: the commit message (it or message_callback is required)
254
:param message_callback: A callback: message = message_callback(cmt_obj)
256
:param timestamp: if not None, seconds-since-epoch for a
257
postdated/predated commit.
259
:param specific_files: If not None, commit only those files. An empty
260
list means 'commit no files'.
262
:param rev_id: If set, use this as the new revision id.
200
branch -- the deprecated branch to commit to. New callers should pass in
203
message -- the commit message, a mandatory parameter
205
timestamp -- if not None, seconds-since-epoch for a
206
postdated/predated commit.
208
specific_files -- If true, commit only those files.
210
rev_id -- If set, use this as the new revision id.
263
211
Useful for test or import commands that need to tightly
264
212
control what revisions are assigned. If you duplicate
265
213
a revision id that exists elsewhere it is your own fault.
266
214
If null (default), a time/random revision id is generated.
268
:param allow_pointless: If true (default), commit even if nothing
216
allow_pointless -- If true (default), commit even if nothing
269
217
has changed and no merges are recorded.
271
:param strict: If true, don't allow a commit if the working tree
219
strict -- If true, don't allow a commit if the working tree
272
220
contains unknown files.
274
:param revprops: Properties for new revision
222
revprops -- Properties for new revision
275
223
:param local: Perform a local only commit.
276
:param reporter: the reporter to use or None for the default
277
:param verbose: if True and the reporter is not None, report everything
278
:param recursive: If set to 'down', commit in any subtrees that have
279
pending changes of any sort during this commit.
280
:param exclude: None or a list of relative paths to exclude from the
281
commit. Pending changes to excluded files will be ignored by the
283
:param lossy: When committing to a foreign VCS, ignore any
284
data that can not be natively represented.
286
operation = OperationWithCleanups(self._commit)
287
self.revprops = revprops or {}
288
# XXX: Can be set on __init__ or passed in - this is a bit ugly.
289
self.config_stack = config or self.config_stack
290
return operation.run(
295
specific_files=specific_files,
297
allow_pointless=allow_pointless,
300
working_tree=working_tree,
303
message_callback=message_callback,
306
possible_master_transports=possible_master_transports,
309
def _commit(self, operation, message, timestamp, timezone, committer,
310
specific_files, rev_id, allow_pointless, strict, verbose,
311
working_tree, local, reporter, message_callback, recursive,
312
exclude, possible_master_transports, lossy):
313
225
mutter('preparing to commit')
315
if working_tree is None:
316
raise BzrError("working_tree must be passed into commit().")
227
if deprecated_passed(branch):
228
warn("Commit.commit (branch, ...): The branch parameter is "
229
"deprecated as of bzr 0.8. Please use working_tree= instead.",
230
DeprecationWarning, stacklevel=2)
232
self.work_tree = self.branch.bzrdir.open_workingtree()
233
elif working_tree is None:
234
raise BzrError("One of branch and working_tree must be passed into commit().")
318
236
self.work_tree = working_tree
319
237
self.branch = self.work_tree.branch
320
if getattr(self.work_tree, 'requires_rich_root', lambda: False)():
321
if not self.branch.repository.supports_rich_root():
322
raise errors.RootNotRich()
323
if message_callback is None:
324
if message is not None:
325
if isinstance(message, bytes):
326
message = message.decode(get_user_encoding())
327
message_callback = lambda x: message
329
raise BzrError("The message or message_callback keyword"
330
" parameter is required for commit().")
239
raise BzrError("The message keyword parameter is required for commit().")
241
self.weave_store = self.branch.repository.weave_store
332
242
self.bound_branch = None
333
self.any_entries_deleted = False
334
if exclude is not None:
335
self.exclude = sorted(
336
minimum_path_selection(exclude))
339
243
self.local = local
340
244
self.master_branch = None
341
self.recursive = recursive
343
# self.specific_files is None to indicate no filter, or any iterable to
344
# indicate a filter - [] means no files at all, as per iter_changes.
345
if specific_files is not None:
346
self.specific_files = sorted(
347
minimum_path_selection(specific_files))
349
self.specific_files = None
245
self.master_locked = False
247
self.specific_files = specific_files
351
248
self.allow_pointless = allow_pointless
352
self.message_callback = message_callback
353
self.timestamp = timestamp
354
self.timezone = timezone
355
self.committer = committer
357
self.verbose = verbose
250
if revprops is not None:
251
self.revprops.update(revprops)
253
if reporter is None and self.reporter is None:
254
self.reporter = NullCommitReporter()
255
elif reporter is not None:
256
self.reporter = reporter
359
258
self.work_tree.lock_write()
360
operation.add_cleanup(self.work_tree.unlock)
361
self.parents = self.work_tree.get_parent_ids()
362
self.pb = ui.ui_factory.nested_progress_bar()
363
operation.add_cleanup(self.pb.finished)
364
self.basis_revid = self.work_tree.last_revision()
365
self.basis_tree = self.work_tree.basis_tree()
366
self.basis_tree.lock_read()
367
operation.add_cleanup(self.basis_tree.unlock)
368
# Cannot commit with conflicts present.
369
if len(self.work_tree.conflicts()) > 0:
370
raise ConflictsInTree
372
# Setup the bound branch variables as needed.
373
self._check_bound_branch(operation, possible_master_transports)
375
# Check that the working tree is up to date
376
old_revno, old_revid, new_revno = self._check_out_of_date_tree()
378
# Complete configuration setup
379
if reporter is not None:
380
self.reporter = reporter
381
elif self.reporter is None:
382
self.reporter = self._select_reporter()
383
if self.config_stack is None:
384
self.config_stack = self.work_tree.get_config_stack()
386
# Setup the progress bar. As the number of files that need to be
387
# committed in unknown, progress is reported as stages.
388
# We keep track of entries separately though and include that
389
# information in the progress bar during the relevant stages.
390
self.pb_stage_name = ""
391
self.pb_stage_count = 0
392
self.pb_stage_total = 5
393
if self.bound_branch:
394
# 2 extra stages: "Uploading data to master branch" and "Merging
395
# tags to master branch"
396
self.pb_stage_total += 2
397
self.pb.show_pct = False
398
self.pb.show_spinner = False
399
self.pb.show_eta = False
400
self.pb.show_count = True
401
self.pb.show_bar = True
403
# After a merge, a selected file commit is not supported.
404
# See 'bzr help merge' for an explanation as to why.
405
if len(self.parents) > 1 and self.specific_files is not None:
406
raise CannotCommitSelectedFileMerge(self.specific_files)
407
# Excludes are a form of selected file commit.
408
if len(self.parents) > 1 and self.exclude:
409
raise CannotCommitSelectedFileMerge(self.exclude)
411
# Collect the changes
412
self._set_progress_stage("Collecting changes", counter=True)
414
self.builder = self.branch.get_commit_builder(self.parents,
415
self.config_stack, timestamp, timezone, committer, self.revprops,
418
if self.builder.updates_branch and self.bound_branch:
420
raise AssertionError(
421
"bound branches not supported for commit builders "
422
"that update the branch")
259
self.pb = bzrlib.ui.ui_factory.nested_progress_bar()
425
# find the location being committed to
426
if self.bound_branch:
427
master_location = self.master_branch.base
429
master_location = self.branch.base
431
# report the start of the commit
432
self.reporter.started(new_revno, self.rev_id, master_location)
434
self._update_builder_with_changes()
435
self._check_pointless()
437
# TODO: Now the new inventory is known, check for conflicts.
438
# ADHB 2006-08-08: If this is done, populate_new_inv should not add
439
# weave lines, because nothing should be recorded until it is known
440
# that commit will succeed.
441
self._set_progress_stage("Saving data locally")
442
self.builder.finish_inventory()
444
# Prompt the user for a commit message if none provided
445
message = message_callback(self)
261
# Cannot commit with conflicts present.
262
if len(self.work_tree.conflicts())>0:
263
raise ConflictsInTree
265
# setup the bound branch variables as needed.
266
self._check_bound_branch()
268
# check for out of date working trees
269
# if we are bound, then self.branch is the master branch and this
270
# test is thus all we need.
271
if self.work_tree.last_revision() != self.master_branch.last_revision():
272
raise errors.OutOfDateTree(self.work_tree)
275
# raise an exception as soon as we find a single unknown.
276
for unknown in self.work_tree.unknowns():
277
raise StrictCommitFailed()
279
if timestamp is None:
280
self.timestamp = time.time()
282
self.timestamp = long(timestamp)
284
if self.config is None:
285
self.config = bzrlib.config.BranchConfig(self.branch)
288
self.rev_id = _gen_revision_id(self.config, self.timestamp)
292
if committer is None:
293
self.committer = self.config.username()
295
assert isinstance(committer, basestring), type(committer)
296
self.committer = committer
299
self.timezone = local_time_offset()
301
self.timezone = int(timezone)
303
if isinstance(message, str):
304
message = message.decode(bzrlib.user_encoding)
305
assert isinstance(message, unicode), type(message)
446
306
self.message = message
448
# Add revision data to the local branch
449
self.rev_id = self.builder.commit(self.message)
451
except Exception as e:
452
mutter("aborting commit write group because of exception:")
453
trace.log_exception_quietly()
457
self._update_branches(old_revno, old_revid, new_revno)
459
# Make the working tree be up to date with the branch. This
460
# includes automatic changes scheduled to be made to the tree, such
461
# as updating its basis and unversioning paths that were missing.
462
self.work_tree.unversion(self.deleted_paths)
463
self._set_progress_stage("Updating the working tree")
464
self.work_tree.update_basis_by_delta(self.rev_id,
465
self.builder.get_basis_delta())
466
self.reporter.completed(new_revno, self.rev_id)
467
self._process_post_hooks(old_revno, new_revno)
470
def _update_branches(self, old_revno, old_revid, new_revno):
471
"""Update the master and local branch to the new revision.
473
This will try to make sure that the master branch is updated
474
before the local branch.
476
:param old_revno: Revision number of master branch before the
478
:param old_revid: Tip of master branch before the commit
479
:param new_revno: Revision number of the new commit
481
if not self.builder.updates_branch:
482
self._process_pre_hooks(old_revno, new_revno)
484
# Upload revision data to the master.
485
# this will propagate merged revisions too if needed.
307
self._escape_commit_message()
309
self.work_inv = self.work_tree.inventory
310
self.basis_tree = self.work_tree.basis_tree()
311
self.basis_inv = self.basis_tree.inventory
312
# one to finish, one for rev and inventory, and one for each
313
# inventory entry, and the same for the new inventory.
314
# note that this estimate is too long when we do a partial tree
315
# commit which excludes some new files from being considered.
316
# The estimate is corrected when we populate the new inv.
317
self.pb_total = len(self.basis_inv) + len(self.work_inv) + 3 - 1
320
self._gather_parents()
321
if len(self.parents) > 1 and self.specific_files:
322
raise NotImplementedError('selected-file commit of merges is not supported yet: files %r',
324
self._check_parents_present()
326
self._remove_deleted()
327
self._populate_new_inv()
328
self._store_snapshot()
329
self._report_deletes()
331
if not (self.allow_pointless
332
or len(self.parents) > 1
333
or (self.new_inv != self.basis_inv and
334
(len(self.basis_inv) != 0 or len(self.new_inv) > 1))):
335
raise PointlessCommit()
337
self._emit_progress_update()
338
self.inv_sha1 = self.branch.repository.add_inventory(
343
self._emit_progress_update()
344
self._make_revision()
345
# revision data is in the local branch now.
347
# upload revision data to the master.
348
# this will propogate merged revisions too if needed.
486
349
if self.bound_branch:
487
self._set_progress_stage("Uploading data to master branch")
488
# 'commit' to the master first so a timeout here causes the
489
# local branch to be out of date
490
(new_revno, self.rev_id) = self.master_branch.import_last_revision_info_and_tags(
491
self.branch, new_revno, self.rev_id, lossy=self._lossy)
493
self.branch.fetch(self.master_branch, self.rev_id)
350
self.master_branch.repository.fetch(self.branch.repository,
351
revision_id=self.rev_id)
352
# now the master has the revision data
353
# 'commit' to the master first so a timeout here causes the local
354
# branch to be out of date
355
self.master_branch.append_revision(self.rev_id)
495
357
# and now do the commit locally.
496
self.branch.set_last_revision_info(new_revno, self.rev_id)
499
self._process_pre_hooks(old_revno, new_revno)
501
# The commit builder will already have updated the branch,
503
self.branch.set_last_revision_info(old_revno, old_revid)
506
# Merge local tags to remote
507
if self.bound_branch:
508
self._set_progress_stage("Merging tags to master branch")
509
tag_updates, tag_conflicts = self.branch.tags.merge_to(
510
self.master_branch.tags)
512
warning_lines = [' ' + name for name, _, _ in tag_conflicts]
513
note( gettext("Conflicting tags in bound branch:\n{0}".format(
514
"\n".join(warning_lines))) )
516
def _select_reporter(self):
517
"""Select the CommitReporter to use."""
519
return NullCommitReporter()
520
return ReportCommitToLog()
522
def _check_pointless(self):
523
if self.allow_pointless:
525
# A merge with no effect on files
526
if len(self.parents) > 1:
528
if self.builder.any_changes():
530
raise PointlessCommit()
532
def _check_bound_branch(self, operation, possible_master_transports=None):
358
self.branch.append_revision(self.rev_id)
360
self.work_tree.set_pending_merges([])
361
self.work_tree.set_last_revision(self.rev_id)
362
# now the work tree is up to date with the branch
364
self.reporter.completed(self.branch.revno(), self.rev_id)
365
if self.config.post_commit() is not None:
366
hooks = self.config.post_commit().split(' ')
367
# this would be nicer with twisted.python.reflect.namedAny
369
result = eval(hook + '(branch, rev_id)',
370
{'branch':self.branch,
372
'rev_id':self.rev_id})
373
self._emit_progress_update()
377
def _check_bound_branch(self):
533
378
"""Check to see if the local branch is bound.
535
380
If it is bound, then most of the commit will actually be
570
414
# so grab the lock
571
415
self.bound_branch = self.branch
572
416
self.master_branch.lock_write()
573
operation.add_cleanup(self.master_branch.unlock)
575
def _check_out_of_date_tree(self):
576
"""Check that the working tree is up to date.
578
:return: old_revision_number, old_revision_id, new_revision_number
417
self.master_locked = True
419
#### # Check to see if we have any pending merges. If we do
420
#### # those need to be pushed into the master branch
421
#### pending_merges = self.work_tree.pending_merges()
422
#### if pending_merges:
423
#### for revision_id in pending_merges:
424
#### self.master_branch.repository.fetch(self.bound_branch.repository,
425
#### revision_id=revision_id)
428
"""Cleanup any open locks, progress bars etc."""
429
cleanups = [self._cleanup_bound_branch,
430
self.work_tree.unlock,
432
found_exception = None
433
for cleanup in cleanups:
436
# we want every cleanup to run no matter what.
437
# so we have a catchall here, but we will raise the
438
# last encountered exception up the stack: and
439
# typically this will be useful enough.
442
if found_exception is not None:
443
# dont do a plan raise, because the last exception may have been
444
# trashed, e is our sure-to-work exception even though it loses the
445
# full traceback. XXX: RBC 20060421 perhaps we could check the
446
# exc_info and if its the same one do a plain raise otherwise
447
# 'raise e' as we do now.
450
def _cleanup_bound_branch(self):
451
"""Executed at the end of a try/finally to cleanup a bound branch.
453
If the branch wasn't bound, this is a no-op.
454
If it was, it resents self.branch to the local branch, instead
582
first_tree_parent = self.work_tree.get_parent_ids()[0]
584
# if there are no parents, treat our parent as 'None'
585
# this is so that we still consider the master branch
586
# - in a checkout scenario the tree may have no
587
# parents but the branch may do.
588
first_tree_parent = breezy.revision.NULL_REVISION
589
old_revno, master_last = self.master_branch.last_revision_info()
590
if master_last != first_tree_parent:
591
if master_last != breezy.revision.NULL_REVISION:
592
raise errors.OutOfDateTree(self.work_tree)
593
if self.branch.repository.has_revision(first_tree_parent):
594
new_revno = old_revno + 1
596
# ghost parents never appear in revision history.
598
return old_revno, master_last, new_revno
600
def _process_pre_hooks(self, old_revno, new_revno):
601
"""Process any registered pre commit hooks."""
602
self._set_progress_stage("Running pre_commit hooks")
603
self._process_hooks("pre_commit", old_revno, new_revno)
605
def _process_post_hooks(self, old_revno, new_revno):
606
"""Process any registered post commit hooks."""
607
# Process the post commit hooks, if any
608
self._set_progress_stage("Running post_commit hooks")
609
# old style commit hooks - should be deprecated ? (obsoleted in
610
# 0.15^H^H^H^H 2.5.0)
611
post_commit = self.config_stack.get('post_commit')
612
if post_commit is not None:
613
hooks = post_commit.split(' ')
614
# this would be nicer with twisted.python.reflect.namedAny
616
result = eval(hook + '(branch, rev_id)',
617
{'branch':self.branch,
619
'rev_id':self.rev_id})
620
# process new style post commit hooks
621
self._process_hooks("post_commit", old_revno, new_revno)
623
def _process_hooks(self, hook_name, old_revno, new_revno):
624
if not Branch.hooks[hook_name]:
627
# new style commit hooks:
628
457
if not self.bound_branch:
629
hook_master = self.branch
632
hook_master = self.master_branch
633
hook_local = self.branch
634
# With bound branches, when the master is behind the local branch,
635
# the 'old_revno' and old_revid values here are incorrect.
636
# XXX: FIXME ^. RBC 20060206
638
old_revid = self.parents[0]
640
old_revid = breezy.revision.NULL_REVISION
642
if hook_name == "pre_commit":
643
future_tree = self.builder.revision_tree()
644
tree_delta = future_tree.changes_from(self.basis_tree,
647
for hook in Branch.hooks[hook_name]:
648
# show the running hook in the progress bar. As hooks may
649
# end up doing nothing (e.g. because they are not configured by
650
# the user) this is still showing progress, not showing overall
651
# actions - its up to each plugin to show a UI if it want's to
652
# (such as 'Emailing diff to foo@example.com').
653
self.pb_stage_name = "Running %s hooks [%s]" % \
654
(hook_name, Branch.hooks.get_hook_name(hook))
655
self._emit_progress()
656
if 'hooks' in debug.debug_flags:
657
mutter("Invoking commit hook: %r", hook)
658
if hook_name == "post_commit":
659
hook(hook_local, hook_master, old_revno, old_revid, new_revno,
661
elif hook_name == "pre_commit":
662
hook(hook_local, hook_master,
663
old_revno, old_revid, new_revno, self.rev_id,
664
tree_delta, future_tree)
666
def _update_builder_with_changes(self):
667
"""Update the commit builder with the data about what has changed.
669
specific_files = self.specific_files
670
mutter("Selecting files for commit with filter %r", specific_files)
673
iter_changes = self.work_tree.iter_changes(self.basis_tree,
674
specific_files=specific_files)
676
iter_changes = filter_excluded(iter_changes, self.exclude)
677
iter_changes = self._filter_iter_changes(iter_changes)
678
for file_id, path, fs_hash in self.builder.record_iter_changes(
679
self.work_tree, self.basis_revid, iter_changes):
680
self.work_tree._observed_sha1(file_id, path, fs_hash)
682
def _filter_iter_changes(self, iter_changes):
683
"""Process iter_changes.
685
This method reports on the changes in iter_changes to the user, and
686
converts 'missing' entries in the iter_changes iterator to 'deleted'
687
entries. 'missing' entries have their
689
:param iter_changes: An iter_changes to process.
690
:return: A generator of changes.
692
reporter = self.reporter
693
report_changes = reporter.is_verbose()
695
for change in iter_changes:
697
old_path = change[1][0]
698
new_path = change[1][1]
699
versioned = change[3][1]
701
versioned = change[3][1]
702
if kind is None and versioned:
705
reporter.missing(new_path)
706
deleted_paths.append(change[1][1])
707
# Reset the new path (None) and new versioned flag (False)
708
change = (change[0], (change[1][0], None), change[2],
709
(change[3][0], False)) + change[4:]
710
new_path = change[1][1]
712
elif kind == 'tree-reference':
713
if self.recursive == 'down':
714
self._commit_nested_tree(change[0], change[1][1])
715
if change[3][0] or change[3][1]:
719
reporter.deleted(old_path)
720
elif old_path is None:
721
reporter.snapshot_change(gettext('added'), new_path)
722
elif old_path != new_path:
723
reporter.renamed(gettext('renamed'), old_path, new_path)
726
self.work_tree.branch.repository._format.rich_root_data):
727
# Don't report on changes to '' in non rich root
729
reporter.snapshot_change(gettext('modified'), new_path)
730
self._next_progress_entry()
731
# Unversion files that were found to be deleted
732
self.deleted_paths = deleted_paths
734
def _check_strict(self):
735
# XXX: when we use iter_changes this would likely be faster if
736
# iter_changes would check for us (even in the presence of
739
# raise an exception as soon as we find a single unknown.
740
for unknown in self.work_tree.unknowns():
741
raise StrictCommitFailed()
743
def _commit_nested_tree(self, file_id, path):
744
"Commit a nested tree."
745
sub_tree = self.work_tree.get_nested_tree(path, file_id)
746
# FIXME: be more comprehensive here:
747
# this works when both trees are in --trees repository,
748
# but when both are bound to a different repository,
749
# it fails; a better way of approaching this is to
750
# finally implement the explicit-caches approach design
751
# a while back - RBC 20070306.
752
if sub_tree.branch.repository.has_same_location(
753
self.work_tree.branch.repository):
754
sub_tree.branch.repository = \
755
self.work_tree.branch.repository
757
return sub_tree.commit(message=None, revprops=self.revprops,
758
recursive=self.recursive,
759
message_callback=self.message_callback,
760
timestamp=self.timestamp, timezone=self.timezone,
761
committer=self.committer,
762
allow_pointless=self.allow_pointless,
763
strict=self.strict, verbose=self.verbose,
764
local=self.local, reporter=self.reporter)
765
except PointlessCommit:
766
return self.work_tree.get_reference_revision(path, file_id)
768
def _set_progress_stage(self, name, counter=False):
769
"""Set the progress stage and emit an update to the progress bar."""
770
self.pb_stage_name = name
771
self.pb_stage_count += 1
773
self.pb_entries_count = 0
775
self.pb_entries_count = None
776
self._emit_progress()
778
def _next_progress_entry(self):
779
"""Emit an update to the progress bar and increment the entry count."""
780
self.pb_entries_count += 1
781
self._emit_progress()
783
def _emit_progress(self):
784
if self.pb_entries_count is not None:
785
text = gettext("{0} [{1}] - Stage").format(self.pb_stage_name,
786
self.pb_entries_count)
788
text = gettext("%s - Stage") % (self.pb_stage_name, )
789
self.pb.update(text, self.pb_stage_count, self.pb_stage_total)
459
if self.master_locked:
460
self.master_branch.unlock()
462
def _escape_commit_message(self):
463
"""Replace xml-incompatible control characters."""
464
# FIXME: RBC 20060419 this should be done by the revision
465
# serialiser not by commit. Then we can also add an unescaper
466
# in the deserializer and start roundtripping revision messages
467
# precisely. See repository_implementations/test_repository.py
469
# Python strings can include characters that can't be
470
# represented in well-formed XML; escape characters that
471
# aren't listed in the XML specification
472
# (http://www.w3.org/TR/REC-xml/#NT-Char).
473
self.message, escape_count = re.subn(
474
u'[^\x09\x0A\x0D\u0020-\uD7FF\uE000-\uFFFD]+',
475
lambda match: match.group(0).encode('unicode_escape'),
478
self.reporter.escaped(escape_count, self.message)
480
def _gather_parents(self):
481
"""Record the parents of a merge for merge detection."""
482
pending_merges = self.work_tree.pending_merges()
484
self.parent_invs = []
485
self.present_parents = []
486
precursor_id = self.branch.last_revision()
488
self.parents.append(precursor_id)
489
self.parents += pending_merges
490
for revision in self.parents:
491
if self.branch.repository.has_revision(revision):
492
inventory = self.branch.repository.get_inventory(revision)
493
self.parent_invs.append(inventory)
494
self.present_parents.append(revision)
496
def _check_parents_present(self):
497
for parent_id in self.parents:
498
mutter('commit parent revision {%s}', parent_id)
499
if not self.branch.repository.has_revision(parent_id):
500
if parent_id == self.branch.last_revision():
501
warning("parent is missing %r", parent_id)
502
raise HistoryMissing(self.branch, 'revision', parent_id)
504
mutter("commit will ghost revision %r", parent_id)
506
def _make_revision(self):
507
"""Record a new revision object for this commit."""
508
rev = Revision(timestamp=self.timestamp,
509
timezone=self.timezone,
510
committer=self.committer,
511
message=self.message,
512
inventory_sha1=self.inv_sha1,
513
revision_id=self.rev_id,
514
properties=self.revprops)
515
rev.parent_ids = self.parents
516
self.branch.repository.add_revision(self.rev_id, rev, self.new_inv, self.config)
518
def _remove_deleted(self):
519
"""Remove deleted files from the working inventories.
521
This is done prior to taking the working inventory as the
522
basis for the new committed inventory.
524
This returns true if any files
525
*that existed in the basis inventory* were deleted.
526
Files that were added and deleted
527
in the working copy don't matter.
529
specific = self.specific_files
531
for path, ie in self.work_inv.iter_entries():
532
if specific and not is_inside_any(specific, path):
534
if not self.work_tree.has_filename(path):
535
self.reporter.missing(path)
536
deleted_ids.append((path, ie.file_id))
538
deleted_ids.sort(reverse=True)
539
for path, file_id in deleted_ids:
540
del self.work_inv[file_id]
541
self.work_tree._write_inventory(self.work_inv)
543
def _store_snapshot(self):
544
"""Pass over inventory and record a snapshot.
546
Entries get a new revision when they are modified in
547
any way, which includes a merge with a new set of
548
parents that have the same entry.
550
# XXX: Need to think more here about when the user has
551
# made a specific decision on a particular value -- c.f.
554
# iter_entries does not visit the root node so we need to call
555
# self._emit_progress_update once by hand.
556
self._emit_progress_update()
557
for path, ie in self.new_inv.iter_entries():
558
self._emit_progress_update()
559
previous_entries = ie.find_previous_heads(
562
self.branch.repository.get_transaction())
563
if ie.revision is None:
564
# we are creating a new revision for ie in the history store
566
ie.snapshot(self.rev_id, path, previous_entries,
567
self.work_tree, self.weave_store,
568
self.branch.repository.get_transaction())
569
# describe the nature of the change that has occured relative to
570
# the basis inventory.
571
if (self.basis_inv.has_id(ie.file_id)):
572
basis_ie = self.basis_inv[ie.file_id]
575
change = ie.describe_change(basis_ie, ie)
576
if change in (InventoryEntry.RENAMED,
577
InventoryEntry.MODIFIED_AND_RENAMED):
578
old_path = self.basis_inv.id2path(ie.file_id)
579
self.reporter.renamed(change, old_path, path)
581
self.reporter.snapshot_change(change, path)
583
def _populate_new_inv(self):
584
"""Build revision inventory.
586
This creates a new empty inventory. Depending on
587
which files are selected for commit, and what is present in the
588
current tree, the new inventory is populated. inventory entries
589
which are candidates for modification have their revision set to
590
None; inventory entries that are carried over untouched have their
591
revision set to their prior value.
593
mutter("Selecting files for commit with filter %s", self.specific_files)
594
self.new_inv = Inventory(root_id=self.work_inv.root.file_id,
595
revision_id=self.rev_id)
596
# iter_entries does not visit the root node so we need to call
597
# self._emit_progress_update once by hand.
598
self._emit_progress_update()
599
for path, new_ie in self.work_inv.iter_entries():
600
self._emit_progress_update()
601
file_id = new_ie.file_id
602
mutter('check %s {%s}', path, new_ie.file_id)
603
if self.specific_files:
604
if not is_inside_any(self.specific_files, path):
605
mutter('%s not selected for commit', path)
606
self._carry_entry(file_id)
609
# this is selected, ensure its parents are too.
610
parent_id = new_ie.parent_id
611
while parent_id != self.work_inv.root.file_id:
612
if not self.new_inv.has_id(parent_id):
613
ie = self._select_entry(self.work_inv[parent_id])
614
mutter('%s selected for commit because of %s',
615
self.new_inv.id2path(parent_id), path)
617
ie = self.new_inv[parent_id]
618
if ie.revision is not None:
620
mutter('%s selected for commit because of %s',
621
self.new_inv.id2path(parent_id), path)
622
parent_id = ie.parent_id
623
mutter('%s selected for commit', path)
624
self._select_entry(new_ie)
626
def _emit_progress_update(self):
627
"""Emit an update to the progress bar."""
628
self.pb.update("Committing", self.pb_count, self.pb_total)
631
def _select_entry(self, new_ie):
632
"""Make new_ie be considered for committing."""
638
def _carry_entry(self, file_id):
639
"""Carry the file unchanged from the basis revision."""
640
if self.basis_inv.has_id(file_id):
641
self.new_inv.add(self.basis_inv[file_id].copy())
643
# this entry is new and not being committed
646
def _report_deletes(self):
647
for path, ie in self.basis_inv.iter_entries():
648
if ie.file_id not in self.new_inv:
649
self.reporter.deleted(path)
651
def _gen_revision_id(config, when):
652
"""Return new revision-id."""
653
s = '%s-%s-' % (config.user_email(), compact_date(when))
654
s += hexlify(rand_bytes(8))