cmdutil.py
4234 lines
| 140.1 KiB
| text/x-python
|
PythonLexer
/ mercurial / cmdutil.py
Vadim Gelfer
|
r2957 | # cmdutil.py - help for command processing in mercurial | ||
Vadim Gelfer
|
r2874 | # | ||
Raphaël Gomès
|
r47575 | # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com> | ||
Vadim Gelfer
|
r2874 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Vadim Gelfer
|
r2874 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r28322 | |||
Matt Harbison
|
r42143 | import copy as copymod | ||
Gregory Szorc
|
r28322 | import errno | ||
r51098 | import functools | |||
Gregory Szorc
|
r28322 | import os | ||
import re | ||||
Matt Harbison
|
r52622 | import typing | ||
Gregory Szorc
|
r28322 | |||
Matt Harbison
|
r50987 | from typing import ( | ||
Any, | ||||
AnyStr, | ||||
Dict, | ||||
Iterable, | ||||
Optional, | ||||
r52178 | TYPE_CHECKING, | |||
Matt Harbison
|
r50987 | cast, | ||
) | ||||
Gregory Szorc
|
r28322 | from .i18n import _ | ||
from .node import ( | ||||
hex, | ||||
Joerg Sonnenberger
|
r47600 | nullrev, | ||
Gregory Szorc
|
r28322 | short, | ||
) | ||||
Gregory Szorc
|
r43357 | from .pycompat import ( | ||
open, | ||||
) | ||||
Rodrigo Damazio Bovendorp
|
r44293 | from .thirdparty import attr | ||
Gregory Szorc
|
r28322 | |||
Matt Harbison
|
r52622 | # Force pytype to use the non-vendored package | ||
if typing.TYPE_CHECKING: | ||||
# noinspection PyPackageRequirements | ||||
import attr | ||||
Gregory Szorc
|
r28322 | from . import ( | ||
bookmarks, | ||||
r52448 | bundle2, | |||
Gregory Szorc
|
r28322 | changelog, | ||
copies, | ||||
crecord as crecordmod, | ||||
encoding, | ||||
error, | ||||
r52448 | exchange, | |||
Gregory Szorc
|
r28322 | formatter, | ||
Yuya Nishihara
|
r35903 | logcmdutil, | ||
Gregory Szorc
|
r28322 | match as matchmod, | ||
Yuya Nishihara
|
r36027 | merge as mergemod, | ||
Augie Fackler
|
r45383 | mergestate as mergestatemod, | ||
Yuya Nishihara
|
r36927 | mergeutil, | ||
Gregory Szorc
|
r28322 | obsolete, | ||
patch, | ||||
pathutil, | ||||
Martin von Zweigbergk
|
r38442 | phases, | ||
Pulkit Goyal
|
r30519 | pycompat, | ||
Taapas Agrawal
|
r42768 | repair, | ||
Gregory Szorc
|
r28322 | revlog, | ||
Pulkit Goyal
|
r35763 | rewriteutil, | ||
Gregory Szorc
|
r28322 | scmutil, | ||
Taapas Agrawal
|
r42729 | state as statemod, | ||
r52448 | streamclone, | |||
Yuya Nishihara
|
r36026 | subrepoutil, | ||
Gregory Szorc
|
r28322 | templatekw, | ||
templater, | ||||
util, | ||||
Pierre-Yves David
|
r31237 | vfs as vfsmod, | ||
Gregory Szorc
|
r28322 | ) | ||
Yuya Nishihara
|
r37102 | |||
from .utils import ( | ||||
dateutil, | ||||
stringutil, | ||||
r52448 | urlutil, | |||
Yuya Nishihara
|
r37102 | ) | ||
r47838 | from .revlogutils import ( | |||
constants as revlog_constants, | ||||
) | ||||
r52178 | if TYPE_CHECKING: | |||
Matt Harbison
|
r50987 | from . import ( | ||
ui as uimod, | ||||
Augie Fackler
|
r44099 | ) | ||
timeless
|
r28861 | stringio = util.stringio | ||
Vadim Gelfer
|
r2874 | |||
Yuya Nishihara
|
r32375 | # templates of common command options | ||
dryrunopts = [ | ||||
Augie Fackler
|
r43347 | (b'n', b'dry-run', None, _(b'do not perform actions, just print output')), | ||
Yuya Nishihara
|
r32375 | ] | ||
Sushil khanchi
|
r38689 | confirmopts = [ | ||
Augie Fackler
|
r43347 | (b'', b'confirm', None, _(b'ask before applying actions')), | ||
Sushil khanchi
|
r38689 | ] | ||
Yuya Nishihara
|
r32375 | remoteopts = [ | ||
Augie Fackler
|
r43347 | (b'e', b'ssh', b'', _(b'specify ssh command to use'), _(b'CMD')), | ||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'remotecmd', | ||||
b'', | ||||
_(b'specify hg command to run on the remote side'), | ||||
_(b'CMD'), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'', | ||
b'insecure', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'do not verify server certificate (ignoring web.cacerts config)'), | ||
Augie Fackler
|
r43346 | ), | ||
Yuya Nishihara
|
r32375 | ] | ||
walkopts = [ | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'I', | ||
b'include', | ||||
Augie Fackler
|
r43346 | [], | ||
Augie Fackler
|
r43347 | _(b'include names matching the given patterns'), | ||
_(b'PATTERN'), | ||||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'X', | ||
b'exclude', | ||||
Augie Fackler
|
r43346 | [], | ||
Augie Fackler
|
r43347 | _(b'exclude names matching the given patterns'), | ||
_(b'PATTERN'), | ||||
Augie Fackler
|
r43346 | ), | ||
Yuya Nishihara
|
r32375 | ] | ||
commitopts = [ | ||||
Augie Fackler
|
r43347 | (b'm', b'message', b'', _(b'use text as commit message'), _(b'TEXT')), | ||
(b'l', b'logfile', b'', _(b'read commit message from file'), _(b'FILE')), | ||||
Yuya Nishihara
|
r32375 | ] | ||
commitopts2 = [ | ||||
Augie Fackler
|
r43347 | ( | ||
b'd', | ||||
b'date', | ||||
b'', | ||||
_(b'record the specified date as commit date'), | ||||
_(b'DATE'), | ||||
), | ||||
( | ||||
b'u', | ||||
b'user', | ||||
b'', | ||||
_(b'record the specified user as committer'), | ||||
_(b'USER'), | ||||
), | ||||
Yuya Nishihara
|
r32375 | ] | ||
Matt Harbison
|
r43173 | commitopts3 = [ | ||
Augie Fackler
|
r43346 | (b'D', b'currentdate', None, _(b'record the current date as commit date')), | ||
(b'U', b'currentuser', None, _(b'record the current user as committer')), | ||||
Matt Harbison
|
r43173 | ] | ||
Yuya Nishihara
|
r32375 | formatteropts = [ | ||
Augie Fackler
|
r43347 | (b'T', b'template', b'', _(b'display with template'), _(b'TEMPLATE')), | ||
Yuya Nishihara
|
r32375 | ] | ||
templateopts = [ | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'', | ||
b'style', | ||||
b'', | ||||
_(b'display using template map file (DEPRECATED)'), | ||||
_(b'STYLE'), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | (b'T', b'template', b'', _(b'display with template'), _(b'TEMPLATE')), | ||
Yuya Nishihara
|
r32375 | ] | ||
logopts = [ | ||||
Augie Fackler
|
r43347 | (b'p', b'patch', None, _(b'show patch')), | ||
(b'g', b'git', None, _(b'use git extended diff format')), | ||||
(b'l', b'limit', b'', _(b'limit number of changes displayed'), _(b'NUM')), | ||||
(b'M', b'no-merges', None, _(b'do not show merges')), | ||||
(b'', b'stat', None, _(b'output diffstat-style summary of changes')), | ||||
(b'G', b'graph', None, _(b"show the revision DAG")), | ||||
Yuya Nishihara
|
r32375 | ] + templateopts | ||
diffopts = [ | ||||
Augie Fackler
|
r43347 | (b'a', b'text', None, _(b'treat all files as text')), | ||
Augie Fackler
|
r44787 | ( | ||
b'g', | ||||
b'git', | ||||
None, | ||||
_(b'use git extended diff format (DEFAULT: diff.git)'), | ||||
), | ||||
Augie Fackler
|
r43347 | (b'', b'binary', None, _(b'generate binary diffs in git mode (default)')), | ||
(b'', b'nodates', None, _(b'omit dates from diff headers')), | ||||
Yuya Nishihara
|
r32375 | ] | ||
diffwsopts = [ | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'w', | ||
b'ignore-all-space', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'ignore white space when comparing lines'), | ||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'b', | ||
b'ignore-space-change', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'ignore changes in the amount of white space'), | ||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'B', | ||
b'ignore-blank-lines', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'ignore changes whose lines are all blank'), | ||
Augie Fackler
|
r43346 | ), | ||
( | ||||
Augie Fackler
|
r43347 | b'Z', | ||
b'ignore-space-at-eol', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r43347 | _(b'ignore changes in whitespace at EOL'), | ||
Augie Fackler
|
r43346 | ), | ||
Yuya Nishihara
|
r32375 | ] | ||
Augie Fackler
|
r43346 | diffopts2 = ( | ||
[ | ||||
Augie Fackler
|
r43347 | (b'', b'noprefix', None, _(b'omit a/ and b/ prefixes from filenames')), | ||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b'p', | ||
b'show-function', | ||||
Augie Fackler
|
r43346 | None, | ||
Augie Fackler
|
r44787 | _( | ||
b'show which function each change is in (DEFAULT: diff.showfunc)' | ||||
), | ||||
Augie Fackler
|
r43346 | ), | ||
Augie Fackler
|
r43347 | (b'', b'reverse', None, _(b'produce a diff that undoes the changes')), | ||
Augie Fackler
|
r43346 | ] | ||
+ diffwsopts | ||||
+ [ | ||||
Augie Fackler
|
r43347 | ( | ||
b'U', | ||||
b'unified', | ||||
b'', | ||||
_(b'number of lines of context to show'), | ||||
_(b'NUM'), | ||||
), | ||||
(b'', b'stat', None, _(b'output diffstat-style summary of changes')), | ||||
( | ||||
b'', | ||||
b'root', | ||||
b'', | ||||
_(b'produce diffs relative to subdirectory'), | ||||
_(b'DIR'), | ||||
), | ||||
Augie Fackler
|
r43346 | ] | ||
) | ||||
Yuya Nishihara
|
r32375 | |||
mergetoolopts = [ | ||||
Augie Fackler
|
r43347 | (b't', b'tool', b'', _(b'specify merge tool'), _(b'TOOL')), | ||
Yuya Nishihara
|
r32375 | ] | ||
similarityopts = [ | ||||
Augie Fackler
|
r43346 | ( | ||
Augie Fackler
|
r43347 | b's', | ||
b'similarity', | ||||
b'', | ||||
_(b'guess renamed files by similarity (0<=s<=100)'), | ||||
_(b'SIMILARITY'), | ||||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r32375 | ] | ||
Augie Fackler
|
r43347 | subrepoopts = [(b'S', b'subrepos', None, _(b'recurse into subrepositories'))] | ||
Yuya Nishihara
|
r32375 | |||
debugrevlogopts = [ | ||||
Augie Fackler
|
r43347 | (b'c', b'changelog', False, _(b'open changelog')), | ||
(b'm', b'manifest', False, _(b'open manifest')), | ||||
(b'', b'dir', b'', _(b'open directory manifest')), | ||||
Yuya Nishihara
|
r32375 | ] | ||
Sean Farley
|
r30703 | # special string such that everything below this line will be ingored in the | ||
# editor text | ||||
Augie Fackler
|
r43347 | _linebelow = b"^HG: ------------------------ >8 ------------------------$" | ||
Sean Farley
|
r30703 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50987 | def check_at_most_one_arg( | ||
opts: Dict[AnyStr, Any], | ||||
*args: AnyStr, | ||||
) -> Optional[AnyStr]: | ||||
Martin von Zweigbergk
|
r44351 | """abort if more than one of the arguments are in opts | ||
Returns the unique argument or None if none of them were specified. | ||||
""" | ||||
Martin von Zweigbergk
|
r44394 | |||
Matt Harbison
|
r50987 | def to_display(name: AnyStr) -> bytes: | ||
Martin von Zweigbergk
|
r44395 | return pycompat.sysbytes(name).replace(b'_', b'-') | ||
Martin von Zweigbergk
|
r44394 | |||
Martin von Zweigbergk
|
r44344 | previous = None | ||
for x in args: | ||||
if opts.get(x): | ||||
if previous: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
Martin von Zweigbergk
|
r44394 | _(b'cannot specify both --%s and --%s') | ||
% (to_display(previous), to_display(x)) | ||||
Martin von Zweigbergk
|
r44344 | ) | ||
previous = x | ||||
Martin von Zweigbergk
|
r44351 | return previous | ||
Martin von Zweigbergk
|
r44344 | |||
Matt Harbison
|
r50987 | def check_incompatible_arguments( | ||
opts: Dict[AnyStr, Any], | ||||
first: AnyStr, | ||||
others: Iterable[AnyStr], | ||||
) -> None: | ||||
Martin von Zweigbergk
|
r44350 | """abort if the first argument is given along with any of the others | ||
Unlike check_at_most_one_arg(), `others` are not mutually exclusive | ||||
Martin von Zweigbergk
|
r44655 | among themselves, and they're passed as a single collection. | ||
Martin von Zweigbergk
|
r44350 | """ | ||
for other in others: | ||||
check_at_most_one_arg(opts, first, other) | ||||
Matt Harbison
|
r50987 | def resolve_commit_options(ui: "uimod.ui", opts: Dict[str, Any]) -> bool: | ||
Matt Harbison
|
r43173 | """modify commit options dict to handle related options | ||
Matt Harbison
|
r43202 | |||
The return value indicates that ``rewrite.update-timestamp`` is the reason | ||||
the ``date`` option is set. | ||||
Matt Harbison
|
r43173 | """ | ||
Martin von Zweigbergk
|
r48225 | check_at_most_one_arg(opts, 'date', 'currentdate') | ||
check_at_most_one_arg(opts, 'user', 'currentuser') | ||||
Matt Harbison
|
r43191 | |||
Matt Harbison
|
r43202 | datemaydiffer = False # date-only change should be ignored? | ||
Martin von Zweigbergk
|
r48225 | if opts.get('currentdate'): | ||
opts['date'] = b'%d %d' % dateutil.makedate() | ||||
Augie Fackler
|
r43346 | elif ( | ||
Martin von Zweigbergk
|
r48225 | not opts.get('date') | ||
Augie Fackler
|
r43347 | and ui.configbool(b'rewrite', b'update-timestamp') | ||
Martin von Zweigbergk
|
r48225 | and opts.get('currentdate') is None | ||
Augie Fackler
|
r43346 | ): | ||
Martin von Zweigbergk
|
r48225 | opts['date'] = b'%d %d' % dateutil.makedate() | ||
Matt Harbison
|
r43202 | datemaydiffer = True | ||
Martin von Zweigbergk
|
r48225 | if opts.get('currentuser'): | ||
opts['user'] = ui.username() | ||||
Matt Harbison
|
r43173 | |||
Matt Harbison
|
r43202 | return datemaydiffer | ||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50987 | def check_note_size(opts: Dict[str, Any]) -> None: | ||
Kyle Lippincott
|
r47856 | """make sure note is of valid format""" | ||
Matt Harbison
|
r43203 | |||
Martin von Zweigbergk
|
r48221 | note = opts.get('note') | ||
Matt Harbison
|
r43203 | if not note: | ||
return | ||||
if len(note) > 255: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"cannot store a note of more than 255 bytes")) | ||
Matt Harbison
|
r43203 | if b'\n' in note: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"note cannot contain a newline")) | ||
Matt Harbison
|
r43203 | |||
Augie Fackler
|
r43346 | |||
Laurent Charignon
|
r25256 | def ishunk(x): | ||
hunkclasses = (crecordmod.uihunk, patch.recordhunk) | ||||
return isinstance(x, hunkclasses) | ||||
Augie Fackler
|
r43346 | |||
Daniel Ploch
|
r48365 | def isheader(x): | ||
headerclasses = (crecordmod.uiheader, patch.header) | ||||
return isinstance(x, headerclasses) | ||||
def newandmodified(chunks): | ||||
Laurent Charignon
|
r25257 | newlyaddedandmodifiedfiles = set() | ||
Kyle Lippincott
|
r43122 | alsorestore = set() | ||
Laurent Charignon
|
r25257 | for chunk in chunks: | ||
Daniel Ploch
|
r48365 | if isheader(chunk) and chunk.isnewfile(): | ||
newlyaddedandmodifiedfiles.add(chunk.filename()) | ||||
alsorestore.update(set(chunk.files()) - {chunk.filename()}) | ||||
Kyle Lippincott
|
r43122 | return newlyaddedandmodifiedfiles, alsorestore | ||
Laurent Charignon
|
r25257 | |||
Augie Fackler
|
r43346 | |||
Brendan Cully
|
r10401 | def parsealiases(cmd): | ||
r46858 | base_aliases = cmd.split(b"|") | |||
all_aliases = set(base_aliases) | ||||
extra_aliases = [] | ||||
for alias in base_aliases: | ||||
if b'-' in alias: | ||||
folded_alias = alias.replace(b'-', b'') | ||||
if folded_alias not in all_aliases: | ||||
all_aliases.add(folded_alias) | ||||
extra_aliases.append(folded_alias) | ||||
base_aliases.extend(extra_aliases) | ||||
return base_aliases | ||||
Brendan Cully
|
r10401 | |||
Augie Fackler
|
r43346 | |||
Laurent Charignon
|
r24356 | def setupwrapcolorwrite(ui): | ||
# wrap ui.write so diff output can be labeled/colorized | ||||
def wrapwrite(orig, *args, **kw): | ||||
Augie Fackler
|
r43906 | label = kw.pop('label', b'') | ||
Laurent Charignon
|
r24356 | for chunk, l in patch.difflabel(lambda: args): | ||
orig(chunk, label=label + l) | ||||
oldwrite = ui.write | ||||
Augie Fackler
|
r43346 | |||
Laurent Charignon
|
r24356 | def wrap(*args, **kwargs): | ||
return wrapwrite(oldwrite, *args, **kwargs) | ||||
Augie Fackler
|
r43346 | |||
Laurent Charignon
|
r24356 | setattr(ui, 'write', wrap) | ||
return oldwrite | ||||
Augie Fackler
|
r43346 | |||
def filterchunks(ui, originalhunks, usecurses, testfile, match, operation=None): | ||||
Kyle Lippincott
|
r38058 | try: | ||
if usecurses: | ||||
if testfile: | ||||
recordfn = crecordmod.testdecorator( | ||||
Augie Fackler
|
r43346 | testfile, crecordmod.testchunkselector | ||
) | ||||
Kyle Lippincott
|
r38058 | else: | ||
recordfn = crecordmod.chunkselector | ||||
Augie Fackler
|
r43346 | return crecordmod.filterpatch( | ||
ui, originalhunks, recordfn, operation | ||||
) | ||||
Kyle Lippincott
|
r38058 | except crecordmod.fallbackerror as e: | ||
Kyle Lippincott
|
r44557 | ui.warn(b'%s\n' % e) | ||
Augie Fackler
|
r43347 | ui.warn(_(b'falling back to text mode\n')) | ||
Kyle Lippincott
|
r38058 | |||
Denis Laxalde
|
r42238 | return patch.filterpatch(ui, originalhunks, match, operation) | ||
Augie Fackler
|
r43346 | |||
Denis Laxalde
|
r42238 | def recordfilter(ui, originalhunks, match, operation=None): | ||
Augie Fackler
|
r46554 | """Prompts the user to filter the originalhunks and return a list of | ||
Laurent Charignon
|
r25310 | selected hunks. | ||
Denis Laxalde
|
r29326 | *operation* is used for to build ui messages to indicate the user what | ||
kind of filtering they are doing: reverting, committing, shelving, etc. | ||||
(see patch.filterpatch). | ||||
Laurent Charignon
|
r25310 | """ | ||
Sean Farley
|
r27531 | usecurses = crecordmod.checkcurses(ui) | ||
Augie Fackler
|
r43347 | testfile = ui.config(b'experimental', b'crecordtest') | ||
Laurent Charignon
|
r24358 | oldwrite = setupwrapcolorwrite(ui) | ||
try: | ||||
Augie Fackler
|
r43346 | newchunks, newopts = filterchunks( | ||
ui, originalhunks, usecurses, testfile, match, operation | ||||
) | ||||
Laurent Charignon
|
r24358 | finally: | ||
ui.write = oldwrite | ||||
Laurent Charignon
|
r27155 | return newchunks, newopts | ||
Laurent Charignon
|
r24357 | |||
Augie Fackler
|
r43346 | |||
r51098 | def _record( | |||
ui, | ||||
repo, | ||||
message, | ||||
match, | ||||
opts, | ||||
commitfunc, | ||||
backupall, | ||||
filterfn, | ||||
pats, | ||||
): | ||||
"""This is generic record driver. | ||||
Its job is to interactively filter local changes, and | ||||
accordingly prepare working directory into a state in which the | ||||
job can be delegated to a non-interactive commit command such as | ||||
'commit' or 'qrefresh'. | ||||
After the actual job is done by non-interactive command, the | ||||
working directory is restored to its original state. | ||||
In the end we'll record interesting changes, and everything else | ||||
will be left in place, so the user can continue working. | ||||
""" | ||||
assert repo.currentwlock() is not None | ||||
if not opts.get(b'interactive-unshelve'): | ||||
checkunfinished(repo, commit=True) | ||||
wctx = repo[None] | ||||
merge = len(wctx.parents()) > 1 | ||||
if merge: | ||||
raise error.InputError( | ||||
_(b'cannot partially commit a merge ' b'(use "hg commit" instead)') | ||||
) | ||||
def fail(f, msg): | ||||
raise error.InputError(b'%s: %s' % (f, msg)) | ||||
force = opts.get(b'force') | ||||
if not force: | ||||
match = matchmod.badmatch(match, fail) | ||||
status = repo.status(match=match) | ||||
overrides = {(b'ui', b'commitsubrepos'): True} | ||||
with repo.ui.configoverride(overrides, b'record'): | ||||
# subrepoutil.precommit() modifies the status | ||||
tmpstatus = scmutil.status( | ||||
copymod.copy(status.modified), | ||||
copymod.copy(status.added), | ||||
copymod.copy(status.removed), | ||||
copymod.copy(status.deleted), | ||||
copymod.copy(status.unknown), | ||||
copymod.copy(status.ignored), | ||||
copymod.copy(status.clean), # pytype: disable=wrong-arg-count | ||||
) | ||||
# Force allows -X subrepo to skip the subrepo. | ||||
subs, commitsubs, newstate = subrepoutil.precommit( | ||||
repo.ui, wctx, tmpstatus, match, force=True | ||||
) | ||||
for s in subs: | ||||
if s in commitsubs: | ||||
dirtyreason = wctx.sub(s).dirtyreason(True) | ||||
raise error.Abort(dirtyreason) | ||||
if not force: | ||||
repo.checkcommitpatterns(wctx, match, status, fail) | ||||
diffopts = patch.difffeatureopts( | ||||
ui, | ||||
opts=opts, | ||||
whitespace=True, | ||||
section=b'commands', | ||||
configprefix=b'commit.interactive.', | ||||
) | ||||
diffopts.nodates = True | ||||
diffopts.git = True | ||||
diffopts.showfunc = True | ||||
originaldiff = patch.diff(repo, changes=status, opts=diffopts) | ||||
original_headers = patch.parsepatch(originaldiff) | ||||
match = scmutil.match(repo[None], pats) | ||||
# 1. filter patch, since we are intending to apply subset of it | ||||
try: | ||||
chunks, newopts = filterfn(ui, original_headers, match) | ||||
except error.PatchParseError as err: | ||||
raise error.InputError(_(b'error parsing patch: %s') % err) | ||||
except error.PatchApplicationError as err: | ||||
raise error.StateError(_(b'error applying patch: %s') % err) | ||||
opts.update(newopts) | ||||
# We need to keep a backup of files that have been newly added and | ||||
# modified during the recording process because there is a previous | ||||
# version without the edit in the workdir. We also will need to restore | ||||
# files that were the sources of renames so that the patch application | ||||
# works. | ||||
newlyaddedandmodifiedfiles, alsorestore = newandmodified(chunks) | ||||
contenders = set() | ||||
for h in chunks: | ||||
if isheader(h): | ||||
contenders.update(set(h.files())) | ||||
changed = status.modified + status.added + status.removed | ||||
newfiles = [f for f in changed if f in contenders] | ||||
if not newfiles: | ||||
ui.status(_(b'no changes to record\n')) | ||||
return 0 | ||||
modified = set(status.modified) | ||||
# 2. backup changed files, so we can restore them in the end | ||||
if backupall: | ||||
tobackup = changed | ||||
else: | ||||
tobackup = [ | ||||
f | ||||
for f in newfiles | ||||
if f in modified or f in newlyaddedandmodifiedfiles | ||||
] | ||||
backups = {} | ||||
if tobackup: | ||||
backupdir = repo.vfs.join(b'record-backups') | ||||
try: | ||||
os.mkdir(backupdir) | ||||
except FileExistsError: | ||||
pass | ||||
try: | ||||
# backup continues | ||||
for f in tobackup: | ||||
fd, tmpname = pycompat.mkstemp( | ||||
prefix=os.path.basename(f) + b'.', dir=backupdir | ||||
) | ||||
os.close(fd) | ||||
ui.debug(b'backup %r as %r\n' % (f, tmpname)) | ||||
util.copyfile(repo.wjoin(f), tmpname, copystat=True) | ||||
backups[f] = tmpname | ||||
fp = stringio() | ||||
for c in chunks: | ||||
fname = c.filename() | ||||
if fname in backups: | ||||
c.write(fp) | ||||
dopatch = fp.tell() | ||||
fp.seek(0) | ||||
# 2.5 optionally review / modify patch in text editor | ||||
if opts.get(b'review', False): | ||||
patchtext = ( | ||||
crecordmod.diffhelptext + crecordmod.patchhelptext + fp.read() | ||||
) | ||||
reviewedpatch = ui.edit( | ||||
patchtext, b"", action=b"diff", repopath=repo.path | ||||
) | ||||
fp.truncate(0) | ||||
fp.write(reviewedpatch) | ||||
fp.seek(0) | ||||
[os.unlink(repo.wjoin(c)) for c in newlyaddedandmodifiedfiles] | ||||
# 3a. apply filtered patch to clean repo (clean) | ||||
if backups: | ||||
m = scmutil.matchfiles(repo, set(backups.keys()) | alsorestore) | ||||
mergemod.revert_to(repo[b'.'], matcher=m) | ||||
# 3b. (apply) | ||||
if dopatch: | ||||
try: | ||||
ui.debug(b'applying patch\n') | ||||
ui.debug(fp.getvalue()) | ||||
patch.internalpatch(ui, repo, fp, 1, eolmode=None) | ||||
except error.PatchParseError as err: | ||||
raise error.InputError(pycompat.bytestr(err)) | ||||
except error.PatchApplicationError as err: | ||||
raise error.StateError(pycompat.bytestr(err)) | ||||
del fp | ||||
# 4. We prepared working directory according to filtered | ||||
# patch. Now is the time to delegate the job to | ||||
# commit/qrefresh or the like! | ||||
# Make all of the pathnames absolute. | ||||
newfiles = [repo.wjoin(nf) for nf in newfiles] | ||||
return commitfunc(ui, repo, *newfiles, **pycompat.strkwargs(opts)) | ||||
finally: | ||||
# 5. finally restore backed-up files | ||||
try: | ||||
dirstate = repo.dirstate | ||||
for realname, tmpname in backups.items(): | ||||
ui.debug(b'restoring %r to %r\n' % (tmpname, realname)) | ||||
if dirstate.get_entry(realname).maybe_clean: | ||||
# without normallookup, restoring timestamp | ||||
# may cause partially committed files | ||||
# to be treated as unmodified | ||||
# XXX-PENDINGCHANGE: We should clarify the context in | ||||
# which this function is called to make sure it | ||||
# already called within a `pendingchange`, However we | ||||
# are taking a shortcut here in order to be able to | ||||
# quickly deprecated the older API. | ||||
with dirstate.changing_parents(repo): | ||||
dirstate.update_file( | ||||
realname, | ||||
p1_tracked=True, | ||||
wc_tracked=True, | ||||
possibly_dirty=True, | ||||
) | ||||
# copystat=True here and above are a hack to trick any | ||||
# editors that have f open that we haven't modified them. | ||||
# | ||||
# Also note that this racy as an editor could notice the | ||||
# file's mtime before we've finished writing it. | ||||
util.copyfile(tmpname, repo.wjoin(realname), copystat=True) | ||||
os.unlink(tmpname) | ||||
if tobackup: | ||||
os.rmdir(backupdir) | ||||
except OSError: | ||||
pass | ||||
Augie Fackler
|
r43346 | def dorecord( | ||
ui, repo, commitfunc, cmdsuggest, backupall, filterfn, *pats, **opts | ||||
): | ||||
Pulkit Goyal
|
r32144 | opts = pycompat.byteskwargs(opts) | ||
Laurent Charignon
|
r24272 | if not ui.interactive(): | ||
FUJIWARA Katsunori
|
r25795 | if cmdsuggest: | ||
Augie Fackler
|
r43347 | msg = _(b'running non-interactively, use %s instead') % cmdsuggest | ||
FUJIWARA Katsunori
|
r25795 | else: | ||
Augie Fackler
|
r43347 | msg = _(b'running non-interactively') | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(msg) | ||
Laurent Charignon
|
r24272 | |||
# make sure username is set before going interactive | ||||
Augie Fackler
|
r43347 | if not opts.get(b'user'): | ||
Augie Fackler
|
r43346 | ui.username() # raise exception, username not provided | ||
Laurent Charignon
|
r24272 | |||
r51098 | func = functools.partial( | |||
_record, | ||||
commitfunc=commitfunc, | ||||
backupall=backupall, | ||||
filterfn=filterfn, | ||||
pats=pats, | ||||
) | ||||
return commit(ui, repo, func, pats, opts) | ||||
Laurent Charignon
|
r24272 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class dirnode: | ||
Pulkit Goyal
|
r33548 | """ | ||
Denis Laxalde
|
r34699 | Represent a directory in user working copy with information required for | ||
the purpose of tersing its status. | ||||
Pulkit Goyal
|
r34683 | |||
Matt Harbison
|
r38221 | path is the path to the directory, without a trailing '/' | ||
Pulkit Goyal
|
r34683 | |||
statuses is a set of statuses of all files in this directory (this includes | ||||
all the files in all the subdirectories too) | ||||
files is a list of files which are direct child of this directory | ||||
subdirs is a dictionary of sub-directory name as the key and it's own | ||||
dirnode object as the value | ||||
""" | ||||
def __init__(self, dirpath): | ||||
self.path = dirpath | ||||
Martin von Zweigbergk
|
r42224 | self.statuses = set() | ||
Pulkit Goyal
|
r34683 | self.files = [] | ||
self.subdirs = {} | ||||
def _addfileindir(self, filename, status): | ||||
Denis Laxalde
|
r34699 | """Add a file in this directory as a direct child.""" | ||
Pulkit Goyal
|
r34683 | self.files.append((filename, status)) | ||
def addfile(self, filename, status): | ||||
Pulkit Goyal
|
r33548 | """ | ||
Denis Laxalde
|
r34699 | Add a file to this directory or to its direct parent directory. | ||
If the file is not direct child of this directory, we traverse to the | ||||
directory of which this file is a direct child of and add the file | ||||
there. | ||||
Pulkit Goyal
|
r33548 | """ | ||
Pulkit Goyal
|
r34683 | |||
# the filename contains a path separator, it means it's not the direct | ||||
# child of this directory | ||||
Augie Fackler
|
r43347 | if b'/' in filename: | ||
subdir, filep = filename.split(b'/', 1) | ||||
Pulkit Goyal
|
r34683 | |||
# does the dirnode object for subdir exists | ||||
if subdir not in self.subdirs: | ||||
Matt Harbison
|
r38221 | subdirpath = pathutil.join(self.path, subdir) | ||
Pulkit Goyal
|
r34683 | self.subdirs[subdir] = dirnode(subdirpath) | ||
# try adding the file in subdir | ||||
self.subdirs[subdir].addfile(filep, status) | ||||
else: | ||||
self._addfileindir(filename, status) | ||||
if status not in self.statuses: | ||||
self.statuses.add(status) | ||||
Denis Laxalde
|
r34685 | def iterfilepaths(self): | ||
Denis Laxalde
|
r34699 | """Yield (status, path) for files directly under this directory.""" | ||
Denis Laxalde
|
r34684 | for f, st in self.files: | ||
Matt Harbison
|
r38221 | yield st, pathutil.join(self.path, f) | ||
Denis Laxalde
|
r34685 | |||
def tersewalk(self, terseargs): | ||||
Denis Laxalde
|
r34684 | """ | ||
Denis Laxalde
|
r34699 | Yield (status, path) obtained by processing the status of this | ||
dirnode. | ||||
Denis Laxalde
|
r34684 | |||
terseargs is the string of arguments passed by the user with `--terse` | ||||
flag. | ||||
Following are the cases which can happen: | ||||
1) All the files in the directory (including all the files in its | ||||
subdirectories) share the same status and the user has asked us to terse | ||||
Matt Harbison
|
r38221 | that status. -> yield (status, dirpath). dirpath will end in '/'. | ||
Denis Laxalde
|
r34684 | |||
Denis Laxalde
|
r34699 | 2) Otherwise, we do following: | ||
Denis Laxalde
|
r34684 | |||
Denis Laxalde
|
r34685 | a) Yield (status, filepath) for all the files which are in this | ||
directory (only the ones in this directory, not the subdirs) | ||||
Denis Laxalde
|
r34684 | |||
b) Recurse the function on all the subdirectories of this | ||||
directory | ||||
""" | ||||
if len(self.statuses) == 1: | ||||
onlyst = self.statuses.pop() | ||||
# Making sure we terse only when the status abbreviation is | ||||
# passed as terse argument | ||||
if onlyst in terseargs: | ||||
Augie Fackler
|
r43347 | yield onlyst, self.path + b'/' | ||
Denis Laxalde
|
r34684 | return | ||
# add the files to status list | ||||
Denis Laxalde
|
r34685 | for st, fpath in self.iterfilepaths(): | ||
yield st, fpath | ||||
Denis Laxalde
|
r34684 | |||
Augie Fackler
|
r43346 | # recurse on the subdirs | ||
Denis Laxalde
|
r34684 | for dirobj in self.subdirs.values(): | ||
Denis Laxalde
|
r34685 | for st, fpath in dirobj.tersewalk(terseargs): | ||
yield st, fpath | ||||
Pulkit Goyal
|
r34683 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r34683 | def tersedir(statuslist, terseargs): | ||
""" | ||||
Denis Laxalde
|
r34699 | Terse the status if all the files in a directory shares the same status. | ||
Pulkit Goyal
|
r34683 | |||
statuslist is scmutil.status() object which contains a list of files for | ||||
each status. | ||||
terseargs is string which is passed by the user as the argument to `--terse` | ||||
flag. | ||||
The function makes a tree of objects of dirnode class, and at each node it | ||||
stores the information required to know whether we can terse a certain | ||||
directory or not. | ||||
""" | ||||
# the order matters here as that is used to produce final list | ||||
Augie Fackler
|
r43347 | allst = (b'm', b'a', b'r', b'd', b'u', b'i', b'c') | ||
Pulkit Goyal
|
r34683 | |||
# checking the argument validity | ||||
Augie Fackler
|
r34894 | for s in pycompat.bytestr(terseargs): | ||
Pulkit Goyal
|
r34683 | if s not in allst: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"'%s' not recognized") % s) | ||
Pulkit Goyal
|
r34683 | |||
# creating a dirnode object for the root of the repo | ||||
Augie Fackler
|
r43347 | rootobj = dirnode(b'') | ||
Augie Fackler
|
r43346 | pstatus = ( | ||
r51803 | ('modified', b'm'), | |||
('added', b'a'), | ||||
('deleted', b'd'), | ||||
('clean', b'c'), | ||||
('unknown', b'u'), | ||||
('ignored', b'i'), | ||||
('removed', b'r'), | ||||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r34683 | |||
tersedict = {} | ||||
r51803 | for attrname, statuschar in pstatus: | |||
Pulkit Goyal
|
r34683 | for f in getattr(statuslist, attrname): | ||
Augie Fackler
|
r34894 | rootobj.addfile(f, statuschar) | ||
tersedict[statuschar] = [] | ||||
Pulkit Goyal
|
r34683 | |||
# we won't be tersing the root dir, so add files in it | ||||
Denis Laxalde
|
r34685 | for st, fpath in rootobj.iterfilepaths(): | ||
tersedict[st].append(fpath) | ||||
Pulkit Goyal
|
r34683 | |||
# process each sub-directory and build tersedict | ||||
for subdir in rootobj.subdirs.values(): | ||||
Denis Laxalde
|
r34685 | for st, f in subdir.tersewalk(terseargs): | ||
tersedict[st].append(f) | ||||
Pulkit Goyal
|
r34683 | |||
tersedlist = [] | ||||
for st in allst: | ||||
tersedict[st].sort() | ||||
tersedlist.append(tersedict[st]) | ||||
Augie Fackler
|
r44044 | return scmutil.status(*tersedlist) | ||
Pulkit Goyal
|
r33548 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r33766 | def _commentlines(raw): | ||
'''Surround lineswith a comment char and a new line''' | ||||
lines = raw.splitlines() | ||||
Augie Fackler
|
r43347 | commentedlines = [b'# %s' % line for line in lines] | ||
return b'\n'.join(commentedlines) + b'\n' | ||||
Pulkit Goyal
|
r33766 | |||
Augie Fackler
|
r43346 | |||
Rodrigo Damazio Bovendorp
|
r44293 | @attr.s(frozen=True) | ||
Gregory Szorc
|
r49801 | class morestatus: | ||
Martin von Zweigbergk
|
r50351 | repo = attr.ib() | ||
Rodrigo Damazio Bovendorp
|
r44293 | unfinishedop = attr.ib() | ||
unfinishedmsg = attr.ib() | ||||
Rodrigo Damazio Bovendorp
|
r44390 | activemerge = attr.ib() | ||
Rodrigo Damazio Bovendorp
|
r44293 | unresolvedpaths = attr.ib() | ||
Rodrigo Damazio Bovendorp
|
r44392 | _formattedpaths = attr.ib(init=False, default=set()) | ||
Rodrigo Damazio Bovendorp
|
r44293 | _label = b'status.morestatus' | ||
Rodrigo Damazio Bovendorp
|
r44294 | def formatfile(self, path, fm): | ||
Rodrigo Damazio Bovendorp
|
r44392 | self._formattedpaths.add(path) | ||
Rodrigo Damazio Bovendorp
|
r44390 | if self.activemerge and path in self.unresolvedpaths: | ||
Rodrigo Damazio Bovendorp
|
r44294 | fm.data(unresolved=True) | ||
Rodrigo Damazio Bovendorp
|
r44293 | def formatfooter(self, fm): | ||
Rodrigo Damazio Bovendorp
|
r44391 | if self.unfinishedop or self.unfinishedmsg: | ||
fm.startitem() | ||||
fm.data(itemtype=b'morestatus') | ||||
if self.unfinishedop: | ||||
fm.data(unfinished=self.unfinishedop) | ||||
statemsg = ( | ||||
_(b'The repository is in an unfinished *%s* state.') | ||||
% self.unfinishedop | ||||
) | ||||
fm.plain(b'%s\n' % _commentlines(statemsg), label=self._label) | ||||
if self.unfinishedmsg: | ||||
fm.data(unfinishedmsg=self.unfinishedmsg) | ||||
Rodrigo Damazio Bovendorp
|
r44293 | |||
Rodrigo Damazio Bovendorp
|
r44392 | # May also start new data items. | ||
Rodrigo Damazio Bovendorp
|
r44293 | self._formatconflicts(fm) | ||
Rodrigo Damazio Bovendorp
|
r44391 | |||
Rodrigo Damazio Bovendorp
|
r44293 | if self.unfinishedmsg: | ||
Martin von Zweigbergk
|
r44307 | fm.plain( | ||
b'%s\n' % _commentlines(self.unfinishedmsg), label=self._label | ||||
) | ||||
Rodrigo Damazio Bovendorp
|
r44293 | |||
def _formatconflicts(self, fm): | ||||
Rodrigo Damazio Bovendorp
|
r44390 | if not self.activemerge: | ||
Rodrigo Damazio Bovendorp
|
r44293 | return | ||
if self.unresolvedpaths: | ||||
mergeliststr = b'\n'.join( | ||||
[ | ||||
Martin von Zweigbergk
|
r44307 | b' %s' | ||
Martin von Zweigbergk
|
r50351 | % util.pathto(self.repo.root, encoding.getcwd(), path) | ||
Rodrigo Damazio Bovendorp
|
r44293 | for path in self.unresolvedpaths | ||
] | ||||
) | ||||
msg = ( | ||||
_( | ||||
Matt Harbison
|
r47514 | b'''Unresolved merge conflicts: | ||
Pulkit Goyal
|
r33766 | |||
%s | ||||
Augie Fackler
|
r43346 | To mark files as resolved: hg resolve --mark FILE''' | ||
Rodrigo Damazio Bovendorp
|
r44293 | ) | ||
% mergeliststr | ||||
Augie Fackler
|
r43346 | ) | ||
Rodrigo Damazio Bovendorp
|
r44392 | |||
# If any paths with unresolved conflicts were not previously | ||||
# formatted, output them now. | ||||
for f in self.unresolvedpaths: | ||||
if f in self._formattedpaths: | ||||
# Already output. | ||||
continue | ||||
fm.startitem() | ||||
Martin von Zweigbergk
|
r50351 | fm.context(repo=self.repo) | ||
Rodrigo Damazio Bovendorp
|
r44392 | # We can't claim to know the status of the file - it may just | ||
# have been in one of the states that were not requested for | ||||
# display, so it could be anything. | ||||
fm.data(itemtype=b'file', path=f, unresolved=True) | ||||
Rodrigo Damazio Bovendorp
|
r44293 | else: | ||
msg = _(b'No unresolved merge conflicts.') | ||||
fm.plain(b'%s\n' % _commentlines(msg), label=self._label) | ||||
def readmorestatus(repo): | ||||
"""Returns a morestatus object if the repo has unfinished state.""" | ||||
Taapas Agrawal
|
r42731 | statetuple = statemod.getrepostate(repo) | ||
Augie Fackler
|
r45383 | mergestate = mergestatemod.mergestate.read(repo) | ||
Rodrigo Damazio Bovendorp
|
r44390 | activemerge = mergestate.active() | ||
Rodrigo Damazio Bovendorp
|
r44391 | if not statetuple and not activemerge: | ||
Rodrigo Damazio Bovendorp
|
r44293 | return None | ||
Rodrigo Damazio Bovendorp
|
r44391 | unfinishedop = unfinishedmsg = unresolved = None | ||
if statetuple: | ||||
unfinishedop, unfinishedmsg = statetuple | ||||
Rodrigo Damazio Bovendorp
|
r44390 | if activemerge: | ||
Rodrigo Damazio Bovendorp
|
r44293 | unresolved = sorted(mergestate.unresolved()) | ||
Martin von Zweigbergk
|
r44307 | return morestatus( | ||
Martin von Zweigbergk
|
r50351 | repo, unfinishedop, unfinishedmsg, activemerge, unresolved | ||
Martin von Zweigbergk
|
r44307 | ) | ||
Pulkit Goyal
|
r33766 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r7213 | def findpossible(cmd, table, strict=False): | ||
Matt Mackall
|
r4549 | """ | ||
Return cmd -> (aliases, command table entry) | ||||
for each matching command. | ||||
Return debug commands (or their aliases) only if no normal command matches. | ||||
""" | ||||
choice = {} | ||||
debugchoice = {} | ||||
Matt Mackall
|
r15600 | |||
if cmd in table: | ||||
Rodrigo Damazio
|
r40331 | # short-circuit exact matches, "log" alias beats "log|history" | ||
Matt Mackall
|
r15600 | keys = [cmd] | ||
else: | ||||
keys = table.keys() | ||||
Augie Fackler
|
r24222 | allcmds = [] | ||
Matt Mackall
|
r15600 | for e in keys: | ||
Brendan Cully
|
r10401 | aliases = parsealiases(e) | ||
Augie Fackler
|
r24222 | allcmds.extend(aliases) | ||
Matt Mackall
|
r4549 | found = None | ||
if cmd in aliases: | ||||
found = cmd | ||||
Matt Mackall
|
r7213 | elif not strict: | ||
Matt Mackall
|
r4549 | for a in aliases: | ||
if a.startswith(cmd): | ||||
found = a | ||||
break | ||||
if found is not None: | ||||
Augie Fackler
|
r43347 | if aliases[0].startswith(b"debug") or found.startswith(b"debug"): | ||
Matt Mackall
|
r5178 | debugchoice[found] = (aliases, table[e]) | ||
Matt Mackall
|
r4549 | else: | ||
Matt Mackall
|
r5178 | choice[found] = (aliases, table[e]) | ||
Matt Mackall
|
r4549 | |||
if not choice and debugchoice: | ||||
choice = debugchoice | ||||
Augie Fackler
|
r24222 | return choice, allcmds | ||
Matt Mackall
|
r4549 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r7213 | def findcmd(cmd, table, strict=True): | ||
Matt Mackall
|
r4549 | """Return (aliases, command table entry) for command string.""" | ||
Augie Fackler
|
r24222 | choice, allcmds = findpossible(cmd, table, strict) | ||
Matt Mackall
|
r4549 | |||
Christian Ebert
|
r5915 | if cmd in choice: | ||
Matt Mackall
|
r4549 | return choice[cmd] | ||
if len(choice) > 1: | ||||
Augie Fackler
|
r32528 | clist = sorted(choice) | ||
Matt Mackall
|
r7643 | raise error.AmbiguousCommand(cmd, clist) | ||
Matt Mackall
|
r4549 | |||
if choice: | ||||
Pulkit Goyal
|
r32862 | return list(choice.values())[0] | ||
Matt Mackall
|
r4549 | |||
Augie Fackler
|
r24222 | raise error.UnknownCommand(cmd, allcmds) | ||
Matt Mackall
|
r4549 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r51721 | def changebranch(ui, repo, revs, label, **opts): | ||
Kyle Lippincott
|
r47856 | """Change the branch name of given revs to label""" | ||
Pulkit Goyal
|
r35763 | |||
Augie Fackler
|
r43347 | with repo.wlock(), repo.lock(), repo.transaction(b'branches'): | ||
Pulkit Goyal
|
r35763 | # abort in case of uncommitted merge or dirty wdir | ||
bailifchanged(repo) | ||||
Martin von Zweigbergk
|
r48928 | revs = logcmdutil.revrange(repo, revs) | ||
Pulkit Goyal
|
r35763 | if not revs: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(b"empty revision set") | ||
Augie Fackler
|
r43347 | roots = repo.revs(b'roots(%ld)', revs) | ||
Pulkit Goyal
|
r35763 | if len(roots) > 1: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
Augie Fackler
|
r43347 | _(b"cannot change branch of non-linear revisions") | ||
) | ||||
rewriteutil.precheck(repo, revs, b'change branch of') | ||||
Pulkit Goyal
|
r35764 | |||
root = repo[roots.first()] | ||||
r40702 | rpb = {parent.branch() for parent in root.parents()} | |||
Manuel Jacob
|
r44891 | if ( | ||
Matt Harbison
|
r51721 | not opts.get('force') | ||
Manuel Jacob
|
r44891 | and label not in rpb | ||
and label in repo.branchmap() | ||||
): | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
_(b"a branch of the same name already exists") | ||||
) | ||||
Augie Fackler
|
r43347 | |||
Pulkit Goyal
|
r35763 | # make sure only topological heads | ||
Augie Fackler
|
r43347 | if repo.revs(b'heads(%ld) - head()', revs): | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
_(b"cannot change branch in middle of a stack") | ||||
) | ||||
Pulkit Goyal
|
r35763 | |||
replacements = {} | ||||
# avoid import cycle mercurial.cmdutil -> mercurial.context -> | ||||
# mercurial.subrepo -> mercurial.cmdutil | ||||
from . import context | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35763 | for rev in revs: | ||
ctx = repo[rev] | ||||
oldbranch = ctx.branch() | ||||
# check if ctx has same branch | ||||
if oldbranch == label: | ||||
continue | ||||
def filectxfn(repo, newctx, path): | ||||
try: | ||||
return ctx[path] | ||||
except error.ManifestLookupError: | ||||
return None | ||||
Augie Fackler
|
r43346 | ui.debug( | ||
Augie Fackler
|
r43347 | b"changing branch of '%s' from '%s' to '%s'\n" | ||
Augie Fackler
|
r43346 | % (hex(ctx.node()), oldbranch, label) | ||
) | ||||
Pulkit Goyal
|
r35763 | extra = ctx.extra() | ||
Augie Fackler
|
r43347 | extra[b'branch_change'] = hex(ctx.node()) | ||
Pulkit Goyal
|
r35763 | # While changing branch of set of linear commits, make sure that | ||
# we base our commits on new parent rather than old parent which | ||||
# was obsoleted while changing the branch | ||||
p1 = ctx.p1().node() | ||||
p2 = ctx.p2().node() | ||||
if p1 in replacements: | ||||
p1 = replacements[p1][0] | ||||
if p2 in replacements: | ||||
p2 = replacements[p2][0] | ||||
Augie Fackler
|
r43346 | mc = context.memctx( | ||
repo, | ||||
(p1, p2), | ||||
ctx.description(), | ||||
ctx.files(), | ||||
filectxfn, | ||||
user=ctx.user(), | ||||
date=ctx.date(), | ||||
extra=extra, | ||||
branch=label, | ||||
) | ||||
Pulkit Goyal
|
r35763 | |||
Martin von Zweigbergk
|
r38442 | newnode = repo.commitctx(mc) | ||
Pulkit Goyal
|
r35763 | replacements[ctx.node()] = (newnode,) | ||
Augie Fackler
|
r43347 | ui.debug(b'new node id is %s\n' % hex(newnode)) | ||
Pulkit Goyal
|
r35763 | |||
# create obsmarkers and move bookmarks | ||||
Augie Fackler
|
r43347 | scmutil.cleanupnodes( | ||
repo, replacements, b'branch-change', fixphase=True | ||||
) | ||||
Pulkit Goyal
|
r35763 | |||
# move the working copy too | ||||
wctx = repo[None] | ||||
# in-progress merge is a bit too complex for now. | ||||
if len(wctx.parents()) == 1: | ||||
newid = replacements.get(wctx.p1().node()) | ||||
if newid is not None: | ||||
# avoid import cycle mercurial.cmdutil -> mercurial.hg -> | ||||
# mercurial.cmdutil | ||||
from . import hg | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35763 | hg.update(repo, newid[0], quietempty=True) | ||
Augie Fackler
|
r43347 | ui.status(_(b"changed branch on %d changesets\n") % len(replacements)) | ||
Pulkit Goyal
|
r35763 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r52608 | def findrepo(p: bytes) -> Optional[bytes]: | ||
Augie Fackler
|
r43347 | while not os.path.isdir(os.path.join(p, b".hg")): | ||
Brendan Cully
|
r10402 | oldp, p = p, os.path.dirname(p) | ||
if p == oldp: | ||||
return None | ||||
return p | ||||
Augie Fackler
|
r43346 | |||
Valters Vingolds
|
r30755 | def bailifchanged(repo, merge=True, hint=None): | ||
Augie Fackler
|
r46554 | """enforce the precondition that working directory must be clean. | ||
Valters Vingolds
|
r30755 | |||
'merge' can be set to false if a pending uncommitted merge should be | ||||
ignored (such as when 'update --check' runs). | ||||
'hint' is the usual hint given to Abort exception. | ||||
""" | ||||
Joerg Sonnenberger
|
r47771 | if merge and repo.dirstate.p2() != repo.nullid: | ||
Martin von Zweigbergk
|
r46444 | raise error.StateError(_(b'outstanding uncommitted merge'), hint=hint) | ||
Augie Fackler
|
r44043 | st = repo.status() | ||
if st.modified or st.added or st.removed or st.deleted: | ||||
Martin von Zweigbergk
|
r46444 | raise error.StateError(_(b'uncommitted changes'), hint=hint) | ||
Eric Roshan Eisner
|
r15231 | ctx = repo[None] | ||
Mads Kiilerich
|
r18364 | for s in sorted(ctx.substate): | ||
Valters Vingolds
|
r30755 | ctx.sub(s).bailifchanged(hint=hint) | ||
Matt Mackall
|
r4549 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50987 | def logmessage(ui: "uimod.ui", opts: Dict[bytes, Any]) -> Optional[bytes]: | ||
Kyle Lippincott
|
r47856 | """get the log message according to -m and -l option""" | ||
Martin von Zweigbergk
|
r44346 | |||
check_at_most_one_arg(opts, b'message', b'logfile') | ||||
Matt Harbison
|
r50987 | message = cast(Optional[bytes], opts.get(b'message')) | ||
Augie Fackler
|
r43347 | logfile = opts.get(b'logfile') | ||
Matt Mackall
|
r4549 | |||
if not message and logfile: | ||||
try: | ||||
Yuya Nishihara
|
r32618 | if isstdiofilename(logfile): | ||
Idan Kamara
|
r14635 | message = ui.fin.read() | ||
Matt Mackall
|
r4549 | else: | ||
Augie Fackler
|
r43347 | message = b'\n'.join(util.readfile(logfile).splitlines()) | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b"can't read commit message '%s': %s") | ||
Augie Fackler
|
r43346 | % (logfile, encoding.strtolocal(inst.strerror)) | ||
) | ||||
Matt Mackall
|
r4549 | return message | ||
Augie Fackler
|
r43346 | |||
Mads Kiilerich
|
r24180 | def mergeeditform(ctxorbool, baseformname): | ||
"""return appropriate editform name (referencing a committemplate) | ||||
'ctxorbool' is either a ctx to be committed, or a bool indicating whether | ||||
FUJIWARA Katsunori
|
r22248 | merging is committed. | ||
Mads Kiilerich
|
r24180 | This returns baseformname with '.merge' appended if it is a merge, | ||
otherwise '.normal' is appended. | ||||
FUJIWARA Katsunori
|
r22248 | """ | ||
if isinstance(ctxorbool, bool): | ||||
if ctxorbool: | ||||
Augie Fackler
|
r43347 | return baseformname + b".merge" | ||
Martin von Zweigbergk
|
r40065 | elif len(ctxorbool.parents()) > 1: | ||
Augie Fackler
|
r43347 | return baseformname + b".merge" | ||
return baseformname + b".normal" | ||||
FUJIWARA Katsunori
|
r22248 | |||
Augie Fackler
|
r43346 | |||
def getcommiteditor( | ||||
Augie Fackler
|
r43347 | edit=False, finishdesc=None, extramsg=None, editform=b'', **opts | ||
Augie Fackler
|
r43346 | ): | ||
FUJIWARA Katsunori
|
r21419 | """get appropriate commit message editor according to '--edit' option | ||
'finishdesc' is a function to be called with edited commit message | ||||
(= 'description' of the new changeset) just after editing, but | ||||
before checking empty-ness. It should return actual text to be | ||||
stored into history. This allows to change description before | ||||
storing. | ||||
'extramsg' is a extra message to be shown in the editor instead of | ||||
'Leave message empty to abort commit' line. 'HG: ' prefix and EOL | ||||
is automatically added. | ||||
FUJIWARA Katsunori
|
r21999 | 'editform' is a dot-separated list of names, to distinguish | ||
the purpose of commit text editing. | ||||
FUJIWARA Katsunori
|
r21419 | 'getcommiteditor' returns 'commitforceeditor' regardless of | ||
'edit', if one of 'finishdesc' or 'extramsg' is specified, because | ||||
they are specific for usage in MQ. | ||||
""" | ||||
if edit or finishdesc or extramsg: | ||||
Augie Fackler
|
r43346 | return lambda r, c, s: commitforceeditor( | ||
r, c, s, finishdesc=finishdesc, extramsg=extramsg, editform=editform | ||||
) | ||||
FUJIWARA Katsunori
|
r21999 | elif editform: | ||
return lambda r, c, s: commiteditor(r, c, s, editform=editform) | ||||
FUJIWARA Katsunori
|
r21405 | else: | ||
return commiteditor | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37792 | def _escapecommandtemplate(tmpl): | ||
parts = [] | ||||
for typ, start, end in templater.scantemplate(tmpl, raw=True): | ||||
if typ == b'string': | ||||
parts.append(stringutil.escapestr(tmpl[start:end])) | ||||
else: | ||||
parts.append(tmpl[start:end]) | ||||
return b''.join(parts) | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37792 | def rendercommandtemplate(ui, tmpl, props): | ||
r"""Expand a literal template 'tmpl' in a way suitable for command line | ||||
'\' in outermost string is not taken as an escape character because it | ||||
is a directory separator on Windows. | ||||
>>> from . import ui as uimod | ||||
>>> ui = uimod.ui() | ||||
>>> rendercommandtemplate(ui, b'c:\\{path}', {b'path': b'foo'}) | ||||
'c:\\foo' | ||||
>>> rendercommandtemplate(ui, b'{"c:\\{path}"}', {'path': b'foo'}) | ||||
'c:{path}' | ||||
""" | ||||
if not tmpl: | ||||
return tmpl | ||||
t = formatter.maketemplater(ui, _escapecommandtemplate(tmpl)) | ||||
return t.renderdefault(props) | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r36528 | def rendertemplate(ctx, tmpl, props=None): | ||
"""Expand a literal template 'tmpl' byte-string against one changeset | ||||
Each props item must be a stringify-able value or a callable returning | ||||
such value, i.e. no bare list nor dict should be passed. | ||||
""" | ||||
repo = ctx.repo() | ||||
tres = formatter.templateresources(repo.ui, repo) | ||||
Augie Fackler
|
r43346 | t = formatter.maketemplater( | ||
repo.ui, tmpl, defaults=templatekw.keywords, resources=tres | ||||
) | ||||
Augie Fackler
|
r43347 | mapping = {b'ctx': ctx} | ||
Yuya Nishihara
|
r36528 | if props: | ||
mapping.update(props) | ||||
Yuya Nishihara
|
r37003 | return t.renderdefault(mapping) | ||
Yuya Nishihara
|
r36528 | |||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r46355 | def format_changeset_summary(ui, ctx, command=None, default_spec=None): | ||
"""Format a changeset summary (one line).""" | ||||
spec = None | ||||
if command: | ||||
spec = ui.config( | ||||
b'command-templates', b'oneline-summary.%s' % command, None | ||||
) | ||||
if not spec: | ||||
spec = ui.config(b'command-templates', b'oneline-summary') | ||||
if not spec: | ||||
spec = default_spec | ||||
if not spec: | ||||
Martin von Zweigbergk
|
r46356 | spec = ( | ||
b'{separate(" ", ' | ||||
Martin von Zweigbergk
|
r46386 | b'label("oneline-summary.changeset", "{rev}:{node|short}")' | ||
Martin von Zweigbergk
|
r46356 | b', ' | ||
Martin von Zweigbergk
|
r46386 | b'join(filter(namespaces % "{ifeq(namespace, "branches", "", join(names % "{label("oneline-summary.{namespace}", name)}", " "))}"), " ")' | ||
Martin von Zweigbergk
|
r46356 | b')} ' | ||
Martin von Zweigbergk
|
r46386 | b'"{label("oneline-summary.desc", desc|firstline)}"' | ||
Martin von Zweigbergk
|
r46356 | ) | ||
Martin von Zweigbergk
|
r46355 | text = rendertemplate(ctx, spec) | ||
return text.split(b'\n')[0] | ||||
Yuya Nishihara
|
r36528 | def _buildfntemplate(pat, total=None, seqno=None, revwidth=None, pathname=None): | ||
r"""Convert old-style filename format string to template string | ||||
>>> _buildfntemplate(b'foo-%b-%n.patch', seqno=0) | ||||
'foo-{reporoot|basename}-{seqno}.patch' | ||||
>>> _buildfntemplate(b'%R{tags % "{tag}"}%H') | ||||
'{rev}{tags % "{tag}"}{node}' | ||||
'\' in outermost strings has to be escaped because it is a directory | ||||
separator on Windows: | ||||
>>> _buildfntemplate(b'c:\\tmp\\%R\\%n.patch', seqno=0) | ||||
'c:\\\\tmp\\\\{rev}\\\\{seqno}.patch' | ||||
>>> _buildfntemplate(b'\\\\foo\\bar.patch') | ||||
'\\\\\\\\foo\\\\bar.patch' | ||||
>>> _buildfntemplate(b'\\{tags % "{tag}"}') | ||||
'\\\\{tags % "{tag}"}' | ||||
but inner strings follow the template rules (i.e. '\' is taken as an | ||||
escape character): | ||||
>>> _buildfntemplate(br'{"c:\tmp"}', seqno=0) | ||||
'{"c:\\tmp"}' | ||||
""" | ||||
Vadim Gelfer
|
r2874 | expander = { | ||
Yuya Nishihara
|
r36528 | b'H': b'{node}', | ||
b'R': b'{rev}', | ||||
b'h': b'{node|short}', | ||||
b'm': br'{sub(r"[^\w]", "_", desc|firstline)}', | ||||
b'r': b'{if(revwidth, pad(rev, revwidth, "0", left=True), rev)}', | ||||
b'%': b'%', | ||||
b'b': b'{reporoot|basename}', | ||||
} | ||||
Yuya Nishihara
|
r36257 | if total is not None: | ||
Yuya Nishihara
|
r36528 | expander[b'N'] = b'{total}' | ||
Yuya Nishihara
|
r36257 | if seqno is not None: | ||
Yuya Nishihara
|
r36528 | expander[b'n'] = b'{seqno}' | ||
Yuya Nishihara
|
r36257 | if total is not None and seqno is not None: | ||
Yuya Nishihara
|
r36528 | expander[b'n'] = b'{pad(seqno, total|stringify|count, "0", left=True)}' | ||
Yuya Nishihara
|
r36257 | if pathname is not None: | ||
Yuya Nishihara
|
r36528 | expander[b's'] = b'{pathname|basename}' | ||
expander[b'd'] = b'{if(pathname|dirname, pathname|dirname, ".")}' | ||||
expander[b'p'] = b'{pathname}' | ||||
Yuya Nishihara
|
r36257 | |||
newname = [] | ||||
Yuya Nishihara
|
r36528 | for typ, start, end in templater.scantemplate(pat, raw=True): | ||
if typ != b'string': | ||||
newname.append(pat[start:end]) | ||||
continue | ||||
i = start | ||||
while i < end: | ||||
n = pat.find(b'%', i, end) | ||||
if n < 0: | ||||
Yuya Nishihara
|
r37102 | newname.append(stringutil.escapestr(pat[i:end])) | ||
Yuya Nishihara
|
r36528 | break | ||
Yuya Nishihara
|
r37102 | newname.append(stringutil.escapestr(pat[i:n])) | ||
Yuya Nishihara
|
r36528 | if n + 2 > end: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Martin von Zweigbergk
|
r43387 | _(b"incomplete format spec in output filename") | ||
Augie Fackler
|
r43346 | ) | ||
c = pat[n + 1 : n + 2] | ||||
Yuya Nishihara
|
r36528 | i = n + 2 | ||
Yuya Nishihara
|
r36257 | try: | ||
Yuya Nishihara
|
r36528 | newname.append(expander[c]) | ||
Yuya Nishihara
|
r36257 | except KeyError: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Martin von Zweigbergk
|
r43387 | _(b"invalid format spec '%%%s' in output filename") % c | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | return b''.join(newname) | ||
Vadim Gelfer
|
r2874 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r36528 | def makefilename(ctx, pat, **props): | ||
if not pat: | ||||
return pat | ||||
tmpl = _buildfntemplate(pat, **props) | ||||
# BUG: alias expansion shouldn't be made against template fragments | ||||
# rewritten from %-format strings, but we have no easy way to partially | ||||
# disable the expansion. | ||||
return rendertemplate(ctx, tmpl, pycompat.byteskwargs(props)) | ||||
Vadim Gelfer
|
r2874 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r32539 | def isstdiofilename(pat): | ||
"""True if the given pat looks like a filename denoting stdin/stdout""" | ||||
Augie Fackler
|
r43347 | return not pat or pat == b'-' | ||
Yuya Nishihara
|
r32539 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class _unclosablefile: | ||
Yuya Nishihara
|
r27418 | def __init__(self, fp): | ||
self._fp = fp | ||||
def close(self): | ||||
pass | ||||
def __iter__(self): | ||||
return iter(self._fp) | ||||
def __getattr__(self, attr): | ||||
return getattr(self._fp, attr) | ||||
Mads Kiilerich
|
r30142 | def __enter__(self): | ||
return self | ||||
def __exit__(self, exc_type, exc_value, exc_tb): | ||||
pass | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | def makefileobj(ctx, pat, mode=b'wb', **props): | ||
writable = mode not in (b'r', b'rb') | ||||
Ronny Pfannschmidt
|
r7319 | |||
Yuya Nishihara
|
r32539 | if isstdiofilename(pat): | ||
Yuya Nishihara
|
r36223 | repo = ctx.repo() | ||
Jordi Gutiérrez Hermoso
|
r24306 | if writable: | ||
fp = repo.ui.fout | ||||
else: | ||||
fp = repo.ui.fin | ||||
Yuya Nishihara
|
r27419 | return _unclosablefile(fp) | ||
Yuya Nishihara
|
r36526 | fn = makefilename(ctx, pat, **props) | ||
Augie Fackler
|
r18613 | return open(fn, mode) | ||
Vadim Gelfer
|
r2882 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r39314 | def openstorage(repo, cmd, file_, opts, returnrevlog=False): | ||
Sune Foldager
|
r14323 | """opens the changelog, manifest, a filelog or a given revlog""" | ||
Augie Fackler
|
r43347 | cl = opts[b'changelog'] | ||
mf = opts[b'manifest'] | ||||
dir = opts[b'dir'] | ||||
Sune Foldager
|
r14323 | msg = None | ||
if cl and mf: | ||||
Augie Fackler
|
r43347 | msg = _(b'cannot specify --changelog and --manifest at the same time') | ||
Martin von Zweigbergk
|
r25119 | elif cl and dir: | ||
Augie Fackler
|
r43347 | msg = _(b'cannot specify --changelog and --dir at the same time') | ||
Martin von Zweigbergk
|
r29427 | elif cl or mf or dir: | ||
Sune Foldager
|
r14323 | if file_: | ||
Augie Fackler
|
r43347 | msg = _(b'cannot specify filename with --changelog or --manifest') | ||
Sune Foldager
|
r14323 | elif not repo: | ||
Augie Fackler
|
r43346 | msg = _( | ||
Augie Fackler
|
r43347 | b'cannot specify --changelog or --manifest or --dir ' | ||
b'without a repository' | ||||
Augie Fackler
|
r43346 | ) | ||
Sune Foldager
|
r14323 | if msg: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(msg) | ||
Sune Foldager
|
r14323 | |||
r = None | ||||
if repo: | ||||
if cl: | ||||
Matt Mackall
|
r21033 | r = repo.unfiltered().changelog | ||
Martin von Zweigbergk
|
r25119 | elif dir: | ||
Pulkit Goyal
|
r46129 | if not scmutil.istreemanifest(repo): | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b"--dir can only be used on repos with " | ||
b"treemanifest enabled" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Augie Fackler
|
r43347 | if not dir.endswith(b'/'): | ||
dir = dir + b'/' | ||||
Gregory Szorc
|
r39280 | dirlog = repo.manifestlog.getstorage(dir) | ||
Martin von Zweigbergk
|
r25119 | if len(dirlog): | ||
r = dirlog | ||||
Sune Foldager
|
r14323 | elif mf: | ||
Gregory Szorc
|
r39280 | r = repo.manifestlog.getstorage(b'') | ||
Sune Foldager
|
r14323 | elif file_: | ||
filelog = repo.file(file_) | ||||
if len(filelog): | ||||
r = filelog | ||||
Gregory Szorc
|
r39314 | |||
# Not all storage may be revlogs. If requested, try to return an actual | ||||
# revlog instance. | ||||
if returnrevlog: | ||||
if isinstance(r, revlog.revlog): | ||||
pass | ||||
r51821 | elif hasattr(r, '_revlog'): | |||
Augie Fackler
|
r43792 | r = r._revlog # pytype: disable=attribute-error | ||
Gregory Szorc
|
r39314 | elif r is not None: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
_(b'%r does not appear to be a revlog') % r | ||||
) | ||||
Gregory Szorc
|
r39314 | |||
Sune Foldager
|
r14323 | if not r: | ||
Gregory Szorc
|
r39314 | if not returnrevlog: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'cannot give path to non-revlog')) | ||
Gregory Szorc
|
r39314 | |||
Sune Foldager
|
r14323 | if not file_: | ||
Augie Fackler
|
r43347 | raise error.CommandError(cmd, _(b'invalid arguments')) | ||
Sune Foldager
|
r14323 | if not os.path.isfile(file_): | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"revlog '%s' not found") % file_) | ||
r47838 | ||||
target = (revlog_constants.KIND_OTHER, b'free-form:%s' % file_) | ||||
Augie Fackler
|
r43346 | r = revlog.revlog( | ||
r47838 | vfsmod.vfs(encoding.getcwd(), audit=False), | |||
target=target, | ||||
r47921 | radix=file_[:-2], | |||
Augie Fackler
|
r43346 | ) | ||
Sune Foldager
|
r14323 | return r | ||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r39314 | def openrevlog(repo, cmd, file_, opts): | ||
"""Obtain a revlog backing storage of an item. | ||||
This is similar to ``openstorage()`` except it always returns a revlog. | ||||
In most cases, a caller cares about the main storage object - not the | ||||
revlog backing it. Therefore, this function should only be used by code | ||||
that needs to examine low-level revlog implementation details. e.g. debug | ||||
commands. | ||||
""" | ||||
return openstorage(repo, cmd, file_, opts, returnrevlog=True) | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50987 | def copy(ui, repo, pats, opts: Dict[bytes, Any], rename=False): | ||
Martin von Zweigbergk
|
r44844 | check_incompatible_arguments(opts, b'forget', [b'dry_run']) | ||
Matt Mackall
|
r5589 | # called with the repo lock held | ||
# | ||||
# hgsep => pathname that uses "/" to separate directories | ||||
# ossep => pathname that uses os.sep to separate directories | ||||
cwd = repo.getcwd() | ||||
targets = {} | ||||
Martin von Zweigbergk
|
r44844 | forget = opts.get(b"forget") | ||
Augie Fackler
|
r43347 | after = opts.get(b"after") | ||
dryrun = opts.get(b"dry_run") | ||||
Martin von Zweigbergk
|
r44847 | rev = opts.get(b'at_rev') | ||
if rev: | ||||
if not forget and not after: | ||||
# TODO: Remove this restriction and make it also create the copy | ||||
# targets (and remove the rename source if rename==True). | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'--at-rev requires --after')) | ||
Martin von Zweigbergk
|
r48930 | ctx = logcmdutil.revsingle(repo, rev) | ||
Martin von Zweigbergk
|
r44847 | if len(ctx.parents()) > 1: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
_(b'cannot mark/unmark copy in merge commit') | ||||
) | ||||
Martin von Zweigbergk
|
r44847 | else: | ||
ctx = repo[None] | ||||
Martin von Zweigbergk
|
r44840 | pctx = ctx.p1() | ||
Matt Mackall
|
r5589 | |||
Martin von Zweigbergk
|
r41806 | uipathfn = scmutil.getuipathfn(repo, legacyrelativevalue=True) | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r44844 | if forget: | ||
Martin von Zweigbergk
|
r44845 | if ctx.rev() is None: | ||
new_ctx = ctx | ||||
else: | ||||
if len(ctx.parents()) > 1: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'cannot unmark copy in merge commit')) | ||
Martin von Zweigbergk
|
r44845 | # avoid cycle context -> subrepo -> cmdutil | ||
from . import context | ||||
rewriteutil.precheck(repo, [ctx.rev()], b'uncopy') | ||||
new_ctx = context.overlayworkingctx(repo) | ||||
new_ctx.setbase(ctx.p1()) | ||||
mergemod.graft(repo, ctx, wctx=new_ctx) | ||||
match = scmutil.match(ctx, pats, opts) | ||||
current_copies = ctx.p1copies() | ||||
current_copies.update(ctx.p2copies()) | ||||
uipathfn = scmutil.getuipathfn(repo) | ||||
for f in ctx.walk(match): | ||||
Martin von Zweigbergk
|
r44844 | if f in current_copies: | ||
Martin von Zweigbergk
|
r44845 | new_ctx[f].markcopied(None) | ||
Martin von Zweigbergk
|
r44844 | elif match.exact(f): | ||
ui.warn( | ||||
_( | ||||
b'%s: not unmarking as copy - file is not marked as copied\n' | ||||
) | ||||
% uipathfn(f) | ||||
) | ||||
Martin von Zweigbergk
|
r44845 | |||
if ctx.rev() is not None: | ||||
with repo.lock(): | ||||
mem_ctx = new_ctx.tomemctx_for_amend(ctx) | ||||
new_node = mem_ctx.commit() | ||||
if repo.dirstate.p1() == ctx.node(): | ||||
r50855 | with repo.dirstate.changing_parents(repo): | |||
Martin von Zweigbergk
|
r44845 | scmutil.movedirstate(repo, repo[new_node]) | ||
replacements = {ctx.node(): [new_node]} | ||||
scmutil.cleanupnodes( | ||||
repo, replacements, b'uncopy', fixphase=True | ||||
) | ||||
Martin von Zweigbergk
|
r44844 | return | ||
Martin von Zweigbergk
|
r44846 | pats = scmutil.expandpats(pats) | ||
if not pats: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'no source or destination specified')) | ||
Martin von Zweigbergk
|
r44846 | if len(pats) == 1: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'no destination specified')) | ||
Martin von Zweigbergk
|
r44846 | dest = pats.pop() | ||
Matt Mackall
|
r5605 | def walkpat(pat): | ||
srcs = [] | ||||
Martin von Zweigbergk
|
r45314 | # TODO: Inline and simplify the non-working-copy version of this code | ||
# since it shares very little with the working-copy version of it. | ||||
ctx_to_walk = ctx if ctx.rev() is None else pctx | ||||
m = scmutil.match(ctx_to_walk, [pat], opts, globbed=True) | ||||
for abs in ctx_to_walk.walk(m): | ||||
Martin von Zweigbergk
|
r41806 | rel = uipathfn(abs) | ||
Matt Mackall
|
r6584 | exact = m.exact(abs) | ||
Martin von Zweigbergk
|
r44840 | if abs not in ctx: | ||
Martin von Zweigbergk
|
r44839 | if abs in pctx: | ||
if not after: | ||||
if exact: | ||||
ui.warn( | ||||
_( | ||||
b'%s: not copying - file has been marked ' | ||||
b'for remove\n' | ||||
) | ||||
% rel | ||||
) | ||||
continue | ||||
else: | ||||
if exact: | ||||
ui.warn( | ||||
_(b'%s: not copying - file is not managed\n') % rel | ||||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r44839 | continue | ||
Matt Mackall
|
r5605 | # abs: hgsep | ||
# rel: ossep | ||||
srcs.append((abs, rel, exact)) | ||||
return srcs | ||||
Matt Mackall
|
r5589 | |||
Martin von Zweigbergk
|
r44847 | if ctx.rev() is not None: | ||
rewriteutil.precheck(repo, [ctx.rev()], b'uncopy') | ||||
absdest = pathutil.canonpath(repo.root, cwd, dest) | ||||
if ctx.hasdir(absdest): | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
Martin von Zweigbergk
|
r44847 | _(b'%s: --at-rev does not support a directory as destination') | ||
% uipathfn(absdest) | ||||
) | ||||
if absdest not in ctx: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
Martin von Zweigbergk
|
r44847 | _(b'%s: copy destination does not exist in %s') | ||
% (uipathfn(absdest), ctx) | ||||
) | ||||
# avoid cycle context -> subrepo -> cmdutil | ||||
from . import context | ||||
copylist = [] | ||||
for pat in pats: | ||||
srcs = walkpat(pat) | ||||
if not srcs: | ||||
continue | ||||
for abs, rel, exact in srcs: | ||||
copylist.append(abs) | ||||
Martin von Zweigbergk
|
r45316 | if not copylist: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'no files to copy')) | ||
Martin von Zweigbergk
|
r44847 | # TODO: Add support for `hg cp --at-rev . foo bar dir` and | ||
# `hg cp --at-rev . dir1 dir2`, preferably unifying the code with the | ||||
# existing functions below. | ||||
if len(copylist) != 1: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'--at-rev requires a single source')) | ||
Martin von Zweigbergk
|
r44847 | |||
new_ctx = context.overlayworkingctx(repo) | ||||
new_ctx.setbase(ctx.p1()) | ||||
mergemod.graft(repo, ctx, wctx=new_ctx) | ||||
new_ctx.markcopied(absdest, copylist[0]) | ||||
with repo.lock(): | ||||
mem_ctx = new_ctx.tomemctx_for_amend(ctx) | ||||
new_node = mem_ctx.commit() | ||||
if repo.dirstate.p1() == ctx.node(): | ||||
r50855 | with repo.dirstate.changing_parents(repo): | |||
Martin von Zweigbergk
|
r44847 | scmutil.movedirstate(repo, repo[new_node]) | ||
replacements = {ctx.node(): [new_node]} | ||||
scmutil.cleanupnodes(repo, replacements, b'copy', fixphase=True) | ||||
return | ||||
Matt Mackall
|
r5589 | # abssrc: hgsep | ||
# relsrc: ossep | ||||
# otarget: ossep | ||||
Matt Mackall
|
r5605 | def copyfile(abssrc, relsrc, otarget, exact): | ||
Augie Fackler
|
r20033 | abstarget = pathutil.canonpath(repo.root, cwd, otarget) | ||
Augie Fackler
|
r43347 | if b'/' in abstarget: | ||
Patrick Mezard
|
r16542 | # We cannot normalize abstarget itself, this would prevent | ||
# case only renames, like a => A. | ||||
Augie Fackler
|
r43347 | abspath, absname = abstarget.rsplit(b'/', 1) | ||
abstarget = repo.dirstate.normalize(abspath) + b'/' + absname | ||||
Matt Mackall
|
r5589 | reltarget = repo.pathto(abstarget, cwd) | ||
Matt Mackall
|
r5607 | target = repo.wjoin(abstarget) | ||
Matt Mackall
|
r5589 | src = repo.wjoin(abssrc) | ||
r48914 | entry = repo.dirstate.get_entry(abstarget) | |||
already_commited = entry.tracked and not entry.added | ||||
Matt Mackall
|
r5607 | |||
Adrian Buehlmann
|
r13962 | scmutil.checkportable(ui, abstarget) | ||
Adrian Buehlmann
|
r13945 | |||
Matt Mackall
|
r5607 | # check for collisions | ||
prevsrc = targets.get(abstarget) | ||||
Matt Mackall
|
r5589 | if prevsrc is not None: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'%s: not overwriting - %s collides with %s\n') | ||
Augie Fackler
|
r43346 | % ( | ||
reltarget, | ||||
repo.pathto(abssrc, cwd), | ||||
repo.pathto(prevsrc, cwd), | ||||
) | ||||
) | ||||
return True # report a failure | ||||
Matt Mackall
|
r5607 | |||
# check for overwrites | ||||
Patrick Mezard
|
r12342 | exists = os.path.lexists(target) | ||
Matt Mackall
|
r16283 | samefile = False | ||
if exists and abssrc != abstarget: | ||||
Augie Fackler
|
r43346 | if repo.dirstate.normalize(abssrc) == repo.dirstate.normalize( | ||
abstarget | ||||
): | ||||
Matt Mackall
|
r16283 | if not rename: | ||
Augie Fackler
|
r43347 | ui.warn(_(b"%s: can't copy - same file\n") % reltarget) | ||
Augie Fackler
|
r43346 | return True # report a failure | ||
Matt Mackall
|
r16283 | exists = False | ||
samefile = True | ||||
r48914 | if not after and exists or after and already_commited: | |||
Augie Fackler
|
r43347 | if not opts[b'force']: | ||
r48914 | if already_commited: | |||
Augie Fackler
|
r43347 | msg = _(b'%s: not overwriting - file already committed\n') | ||
Martin von Zweigbergk
|
r49061 | # Check if if the target was added in the parent and the | ||
# source already existed in the grandparent. | ||||
looks_like_copy_in_pctx = abstarget in pctx and any( | ||||
abssrc in gpctx and abstarget not in gpctx | ||||
for gpctx in pctx.parents() | ||||
) | ||||
if looks_like_copy_in_pctx: | ||||
if rename: | ||||
hint = _( | ||||
b"('hg rename --at-rev .' to record the rename " | ||||
b"in the parent of the working copy)\n" | ||||
) | ||||
else: | ||||
hint = _( | ||||
b"('hg copy --at-rev .' to record the copy in " | ||||
b"the parent of the working copy)\n" | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r30151 | else: | ||
Martin von Zweigbergk
|
r49061 | if after: | ||
flags = b'--after --force' | ||||
else: | ||||
flags = b'--force' | ||||
if rename: | ||||
hint = ( | ||||
_( | ||||
b"('hg rename %s' to replace the file by " | ||||
b'recording a rename)\n' | ||||
) | ||||
% flags | ||||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r49061 | else: | ||
hint = ( | ||||
_( | ||||
b"('hg copy %s' to replace the file by " | ||||
b'recording a copy)\n' | ||||
) | ||||
% flags | ||||
) | ||||
Augie Fackler
|
r30151 | else: | ||
Augie Fackler
|
r43347 | msg = _(b'%s: not overwriting - file exists\n') | ||
Augie Fackler
|
r30151 | if rename: | ||
Augie Fackler
|
r43347 | hint = _( | ||
b"('hg rename --after' to record the rename)\n" | ||||
) | ||||
Augie Fackler
|
r30151 | else: | ||
Augie Fackler
|
r43347 | hint = _(b"('hg copy --after' to record the copy)\n") | ||
Augie Fackler
|
r30151 | ui.warn(msg % reltarget) | ||
ui.warn(hint) | ||||
Augie Fackler
|
r43346 | return True # report a failure | ||
Matt Mackall
|
r5607 | |||
if after: | ||||
Matt Mackall
|
r5608 | if not exists: | ||
Steve Losh
|
r11152 | if rename: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'%s: not recording move - %s does not exist\n') | ||
Augie Fackler
|
r43346 | % (relsrc, reltarget) | ||
) | ||||
Steve Losh
|
r11152 | else: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'%s: not recording copy - %s does not exist\n') | ||
Augie Fackler
|
r43346 | % (relsrc, reltarget) | ||
) | ||||
return True # report a failure | ||||
Matt Mackall
|
r5608 | elif not dryrun: | ||
Matt Mackall
|
r5589 | try: | ||
Matt Mackall
|
r5608 | if exists: | ||
os.unlink(target) | ||||
Augie Fackler
|
r43347 | targetdir = os.path.dirname(target) or b'.' | ||
Matt Mackall
|
r5608 | if not os.path.isdir(targetdir): | ||
os.makedirs(targetdir) | ||||
Matt Mackall
|
r16283 | if samefile: | ||
Augie Fackler
|
r43347 | tmp = target + b"~hgrename" | ||
Matt Mackall
|
r16283 | os.rename(src, tmp) | ||
os.rename(tmp, target) | ||||
else: | ||||
Kyle Lippincott
|
r37106 | # Preserve stat info on renames, not on copies; this matches | ||
# Linux CLI behavior. | ||||
util.copyfile(src, target, copystat=rename) | ||||
Adrian Buehlmann
|
r14518 | srcexists = True | ||
Gregory Szorc
|
r25660 | except IOError as inst: | ||
Matt Mackall
|
r5589 | if inst.errno == errno.ENOENT: | ||
Augie Fackler
|
r43347 | ui.warn(_(b'%s: deleted in working directory\n') % relsrc) | ||
Adrian Buehlmann
|
r14518 | srcexists = False | ||
Matt Mackall
|
r5589 | else: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'%s: cannot copy - %s\n') | ||
Augie Fackler
|
r43346 | % (relsrc, encoding.strtolocal(inst.strerror)) | ||
) | ||||
return True # report a failure | ||||
Matt Mackall
|
r5607 | |||
Matt Mackall
|
r5589 | if ui.verbose or not exact: | ||
Martin Geisler
|
r7894 | if rename: | ||
Augie Fackler
|
r43347 | ui.status(_(b'moving %s to %s\n') % (relsrc, reltarget)) | ||
Martin Geisler
|
r7894 | else: | ||
Augie Fackler
|
r43347 | ui.status(_(b'copying %s to %s\n') % (relsrc, reltarget)) | ||
Matt Mackall
|
r5608 | |||
Matt Mackall
|
r5589 | targets[abstarget] = abssrc | ||
Matt Mackall
|
r5607 | |||
# fix up dirstate | ||||
Augie Fackler
|
r43346 | scmutil.dirstatecopy( | ||
Martin von Zweigbergk
|
r44840 | ui, repo, ctx, abssrc, abstarget, dryrun=dryrun, cwd=cwd | ||
Augie Fackler
|
r43346 | ) | ||
Matt Mackall
|
r5610 | if rename and not dryrun: | ||
Matt Mackall
|
r16283 | if not after and srcexists and not samefile: | ||
Augie Fackler
|
r43347 | rmdir = repo.ui.configbool(b'experimental', b'removeemptydirs') | ||
Kyle Lippincott
|
r38512 | repo.wvfs.unlinkpath(abssrc, rmdir=rmdir) | ||
Martin von Zweigbergk
|
r44840 | ctx.forget([abssrc]) | ||
Matt Mackall
|
r5589 | |||
# pat: ossep | ||||
# dest ossep | ||||
# srcs: list of (hgsep, hgsep, ossep, bool) | ||||
# return: function that takes hgsep and returns ossep | ||||
def targetpathfn(pat, dest, srcs): | ||||
if os.path.isdir(pat): | ||||
Augie Fackler
|
r20033 | abspfx = pathutil.canonpath(repo.root, cwd, pat) | ||
Matt Mackall
|
r5589 | abspfx = util.localpath(abspfx) | ||
if destdirexists: | ||||
striplen = len(os.path.split(abspfx)[0]) | ||||
else: | ||||
striplen = len(abspfx) | ||||
if striplen: | ||||
Pulkit Goyal
|
r30615 | striplen += len(pycompat.ossep) | ||
Matt Mackall
|
r5589 | res = lambda p: os.path.join(dest, util.localpath(p)[striplen:]) | ||
elif destdirexists: | ||||
Augie Fackler
|
r43346 | res = lambda p: os.path.join( | ||
dest, os.path.basename(util.localpath(p)) | ||||
) | ||||
Matt Mackall
|
r5589 | else: | ||
res = lambda p: dest | ||||
return res | ||||
# pat: ossep | ||||
# dest ossep | ||||
# srcs: list of (hgsep, hgsep, ossep, bool) | ||||
# return: function that takes hgsep and returns ossep | ||||
def targetpathafterfn(pat, dest, srcs): | ||||
Martin Geisler
|
r12085 | if matchmod.patkind(pat): | ||
Matt Mackall
|
r5589 | # a mercurial pattern | ||
Augie Fackler
|
r43346 | res = lambda p: os.path.join( | ||
dest, os.path.basename(util.localpath(p)) | ||||
) | ||||
Matt Mackall
|
r5589 | else: | ||
Augie Fackler
|
r20033 | abspfx = pathutil.canonpath(repo.root, cwd, pat) | ||
Matt Mackall
|
r5589 | if len(abspfx) < len(srcs[0][0]): | ||
# A directory. Either the target path contains the last | ||||
# component of the source path or it does not. | ||||
def evalpath(striplen): | ||||
score = 0 | ||||
for s in srcs: | ||||
t = os.path.join(dest, util.localpath(s[0])[striplen:]) | ||||
Patrick Mezard
|
r12357 | if os.path.lexists(t): | ||
Matt Mackall
|
r5589 | score += 1 | ||
return score | ||||
abspfx = util.localpath(abspfx) | ||||
striplen = len(abspfx) | ||||
if striplen: | ||||
Pulkit Goyal
|
r30615 | striplen += len(pycompat.ossep) | ||
Matt Mackall
|
r5589 | if os.path.isdir(os.path.join(dest, os.path.split(abspfx)[1])): | ||
score = evalpath(striplen) | ||||
striplen1 = len(os.path.split(abspfx)[0]) | ||||
if striplen1: | ||||
Pulkit Goyal
|
r30615 | striplen1 += len(pycompat.ossep) | ||
Matt Mackall
|
r5589 | if evalpath(striplen1) > score: | ||
striplen = striplen1 | ||||
Augie Fackler
|
r43346 | res = lambda p: os.path.join(dest, util.localpath(p)[striplen:]) | ||
Matt Mackall
|
r5589 | else: | ||
# a file | ||||
if destdirexists: | ||||
Augie Fackler
|
r43346 | res = lambda p: os.path.join( | ||
dest, os.path.basename(util.localpath(p)) | ||||
) | ||||
Matt Mackall
|
r5589 | else: | ||
res = lambda p: dest | ||||
return res | ||||
Alexis S. L. Carvalho
|
r6258 | destdirexists = os.path.isdir(dest) and not os.path.islink(dest) | ||
Matt Mackall
|
r5589 | if not destdirexists: | ||
Martin Geisler
|
r12085 | if len(pats) > 1 or matchmod.patkind(pats[0]): | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b'with multiple sources, destination must be an ' | ||
b'existing directory' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Shun-ichi GOTO
|
r5843 | if util.endswithsep(dest): | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
_(b'destination %s is not a directory') % dest | ||||
) | ||||
Matt Mackall
|
r5607 | |||
tfn = targetpathfn | ||||
if after: | ||||
Matt Mackall
|
r5589 | tfn = targetpathafterfn | ||
copylist = [] | ||||
for pat in pats: | ||||
Matt Mackall
|
r5605 | srcs = walkpat(pat) | ||
Matt Mackall
|
r5589 | if not srcs: | ||
continue | ||||
copylist.append((tfn(pat, dest, srcs), srcs)) | ||||
if not copylist: | ||||
Martin von Zweigbergk
|
r47897 | hint = None | ||
if rename: | ||||
hint = _(b'maybe you meant to use --after --at-rev=.') | ||||
raise error.InputError(_(b'no files to copy'), hint=hint) | ||||
Matt Mackall
|
r5589 | |||
Matt Mackall
|
r5606 | errors = 0 | ||
Matt Mackall
|
r5589 | for targetpath, srcs in copylist: | ||
Matt Mackall
|
r5605 | for abssrc, relsrc, exact in srcs: | ||
Matt Mackall
|
r5606 | if copyfile(abssrc, relsrc, targetpath(abssrc), exact): | ||
errors += 1 | ||||
Matt Mackall
|
r5589 | |||
Matt Mackall
|
r11177 | return errors != 0 | ||
Matt Mackall
|
r5589 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r26561 | ## facility to let extension process additional data into an import patch | ||
# list of identifier to be executed in order | ||||
extrapreimport = [] # run before commit | ||||
Augie Fackler
|
r43346 | extrapostimport = [] # run after commit | ||
Pierre-Yves David
|
r26561 | # mapping from identifier to actual import function | ||
# | ||||
# 'preimport' are run before the commit is made and are provided the following | ||||
# arguments: | ||||
# - repo: the localrepository instance, | ||||
# - patchdata: data extracted from patch header (cf m.patch.patchheadermap), | ||||
Mads Kiilerich
|
r26781 | # - extra: the future extra dictionary of the changeset, please mutate it, | ||
Pierre-Yves David
|
r26561 | # - opts: the import options. | ||
# XXX ideally, we would just pass an ctx ready to be computed, that would allow | ||||
# mutation of in memory commit and more. Feel free to rework the code to get | ||||
# there. | ||||
extrapreimportmap = {} | ||||
Pierre-Yves David
|
r26562 | # 'postimport' are run after the commit is made and are provided the following | ||
# argument: | ||||
# - ctx: the changectx created by import. | ||||
extrapostimportmap = {} | ||||
Pierre-Yves David
|
r26561 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r37638 | def tryimportone(ui, repo, patchdata, parents, opts, msgs, updatefunc): | ||
Pierre-Yves David
|
r20500 | """Utility function used by commands.import to import a single patch | ||
This function is explicitly defined here to help the evolve extension to | ||||
wrap this part of the import logic. | ||||
The API is currently a bit ugly because it a simple code translation from | ||||
the import command. Feel free to make it better. | ||||
Gregory Szorc
|
r37638 | :patchdata: a dictionary containing parsed patch data (such as from | ||
``patch.extract()``) | ||||
Pierre-Yves David
|
r20500 | :parents: nodes that will be parent of the created commit | ||
:opts: the full dict of option passed to the import command | ||||
:msgs: list to save commit message to. | ||||
(used in case we need to save it when failing) | ||||
:updatefunc: a function that update a repo to a given node | ||||
updatefunc(<repo>, <node>) | ||||
""" | ||||
Gregory Szorc
|
r25930 | # avoid cycle context -> subrepo -> cmdutil | ||
Gregory Szorc
|
r28322 | from . import context | ||
Gregory Szorc
|
r37638 | |||
Augie Fackler
|
r43347 | tmpname = patchdata.get(b'filename') | ||
message = patchdata.get(b'message') | ||||
user = opts.get(b'user') or patchdata.get(b'user') | ||||
date = opts.get(b'date') or patchdata.get(b'date') | ||||
branch = patchdata.get(b'branch') | ||||
nodeid = patchdata.get(b'nodeid') | ||||
p1 = patchdata.get(b'p1') | ||||
p2 = patchdata.get(b'p2') | ||||
nocommit = opts.get(b'no_commit') | ||||
importbranch = opts.get(b'import_branch') | ||||
update = not opts.get(b'bypass') | ||||
strip = opts[b"strip"] | ||||
prefix = opts[b"prefix"] | ||||
sim = float(opts.get(b'similarity') or 0) | ||||
Gregory Szorc
|
r37639 | |||
Pierre-Yves David
|
r20500 | if not tmpname: | ||
Gregory Szorc
|
r37639 | return None, None, False | ||
Pierre-Yves David
|
r20500 | |||
Pierre-Yves David
|
r21553 | rejects = False | ||
Gregory Szorc
|
r37639 | cmdline_message = logmessage(ui, opts) | ||
if cmdline_message: | ||||
# pickup the cmdline msg | ||||
message = cmdline_message | ||||
elif message: | ||||
# pickup the patch msg | ||||
message = message.strip() | ||||
else: | ||||
# launch the editor | ||||
message = None | ||||
Augie Fackler
|
r43347 | ui.debug(b'message:\n%s\n' % (message or b'')) | ||
Gregory Szorc
|
r37639 | |||
if len(parents) == 1: | ||||
Joerg Sonnenberger
|
r47600 | parents.append(repo[nullrev]) | ||
Augie Fackler
|
r43347 | if opts.get(b'exact'): | ||
Gregory Szorc
|
r37639 | if not nodeid or not p1: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b'not a Mercurial patch')) | ||
Gregory Szorc
|
r37639 | p1 = repo[p1] | ||
Joerg Sonnenberger
|
r47600 | p2 = repo[p2 or nullrev] | ||
Gregory Szorc
|
r37639 | elif p2: | ||
try: | ||||
Pierre-Yves David
|
r20500 | p1 = repo[p1] | ||
Gregory Szorc
|
r37639 | p2 = repo[p2] | ||
# Without any options, consider p2 only if the | ||||
# patch is being applied on top of the recorded | ||||
# first parent. | ||||
if p1 != parents[0]: | ||||
p1 = parents[0] | ||||
Joerg Sonnenberger
|
r47600 | p2 = repo[nullrev] | ||
Gregory Szorc
|
r37639 | except error.RepoError: | ||
p1, p2 = parents | ||||
Joerg Sonnenberger
|
r47600 | if p2.rev() == nullrev: | ||
Augie Fackler
|
r43346 | ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b"warning: import the patch as a normal revision\n" | ||
b"(use --exact to import the patch as a merge)\n" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Gregory Szorc
|
r37639 | else: | ||
p1, p2 = parents | ||||
n = None | ||||
if update: | ||||
if p1 != parents[0]: | ||||
updatefunc(repo, p1.node()) | ||||
if p2 != parents[1]: | ||||
repo.setparents(p1.node(), p2.node()) | ||||
Augie Fackler
|
r43347 | if opts.get(b'exact') or importbranch: | ||
r51158 | repo.dirstate.setbranch( | |||
branch or b'default', repo.currenttransaction() | ||||
) | ||||
Augie Fackler
|
r43347 | |||
partial = opts.get(b'partial', False) | ||||
Gregory Szorc
|
r37639 | files = set() | ||
try: | ||||
Augie Fackler
|
r43346 | patch.patch( | ||
ui, | ||||
repo, | ||||
tmpname, | ||||
strip=strip, | ||||
prefix=prefix, | ||||
files=files, | ||||
eolmode=None, | ||||
similarity=sim / 100.0, | ||||
) | ||||
Martin von Zweigbergk
|
r49186 | except error.PatchParseError as e: | ||
Martin von Zweigbergk
|
r49187 | raise error.InputError( | ||
pycompat.bytestr(e), | ||||
hint=_( | ||||
b'check that whitespace in the patch has not been mangled' | ||||
), | ||||
) | ||||
Martin von Zweigbergk
|
r49186 | except error.PatchApplicationError as e: | ||
Gregory Szorc
|
r37639 | if not partial: | ||
Martin von Zweigbergk
|
r49186 | raise error.StateError(pycompat.bytestr(e)) | ||
Gregory Szorc
|
r37639 | if partial: | ||
rejects = True | ||||
files = list(files) | ||||
if nocommit: | ||||
if message: | ||||
msgs.append(message) | ||||
Pierre-Yves David
|
r20500 | else: | ||
Augie Fackler
|
r43347 | if opts.get(b'exact') or p2: | ||
Gregory Szorc
|
r37639 | # If you got here, you either use --force and know what | ||
# you are doing or used --exact or a merge patch while | ||||
# being updated to its first parent. | ||||
m = None | ||||
else: | ||||
m = scmutil.matchfiles(repo, files or []) | ||||
Augie Fackler
|
r43347 | editform = mergeeditform(repo[None], b'import.normal') | ||
if opts.get(b'exact'): | ||||
Gregory Szorc
|
r37639 | editor = None | ||
else: | ||||
Augie Fackler
|
r43346 | editor = getcommiteditor( | ||
editform=editform, **pycompat.strkwargs(opts) | ||||
) | ||||
Gregory Szorc
|
r37639 | extra = {} | ||
for idfunc in extrapreimport: | ||||
extrapreimportmap[idfunc](repo, patchdata, extra, opts) | ||||
overrides = {} | ||||
if partial: | ||||
Augie Fackler
|
r43347 | overrides[(b'ui', b'allowemptycommit')] = True | ||
Denis Laxalde
|
r44118 | if opts.get(b'secret'): | ||
overrides[(b'phases', b'new-commit')] = b'secret' | ||||
Augie Fackler
|
r43347 | with repo.ui.configoverride(overrides, b'import'): | ||
Augie Fackler
|
r43346 | n = repo.commit( | ||
message, user, date, match=m, editor=editor, extra=extra | ||||
) | ||||
Gregory Szorc
|
r37639 | for idfunc in extrapostimport: | ||
extrapostimportmap[idfunc](repo[n]) | ||||
else: | ||||
Augie Fackler
|
r43347 | if opts.get(b'exact') or importbranch: | ||
branch = branch or b'default' | ||||
Gregory Szorc
|
r37639 | else: | ||
branch = p1.branch() | ||||
store = patch.filestore() | ||||
try: | ||||
Pierre-Yves David
|
r20500 | files = set() | ||
Pierre-Yves David
|
r21553 | try: | ||
Augie Fackler
|
r43346 | patch.patchrepo( | ||
ui, | ||||
repo, | ||||
p1, | ||||
store, | ||||
tmpname, | ||||
strip, | ||||
prefix, | ||||
files, | ||||
eolmode=None, | ||||
) | ||||
Martin von Zweigbergk
|
r49186 | except error.PatchParseError as e: | ||
Martin von Zweigbergk
|
r49187 | raise error.InputError( | ||
stringutil.forcebytestr(e), | ||||
hint=_( | ||||
b'check that whitespace in the patch has not been mangled' | ||||
), | ||||
) | ||||
Martin von Zweigbergk
|
r49186 | except error.PatchApplicationError as e: | ||
raise error.StateError(stringutil.forcebytestr(e)) | ||||
Augie Fackler
|
r43347 | if opts.get(b'exact'): | ||
Gregory Szorc
|
r37639 | editor = None | ||
Pierre-Yves David
|
r20500 | else: | ||
Augie Fackler
|
r43347 | editor = getcommiteditor(editform=b'import.bypass') | ||
Augie Fackler
|
r43346 | memctx = context.memctx( | ||
repo, | ||||
(p1.node(), p2.node()), | ||||
message, | ||||
files=files, | ||||
filectxfn=store, | ||||
user=user, | ||||
date=date, | ||||
branch=branch, | ||||
editor=editor, | ||||
) | ||||
Matt Harbison
|
r44849 | |||
overrides = {} | ||||
if opts.get(b'secret'): | ||||
overrides[(b'phases', b'new-commit')] = b'secret' | ||||
with repo.ui.configoverride(overrides, b'import'): | ||||
n = memctx.commit() | ||||
Gregory Szorc
|
r37639 | finally: | ||
store.close() | ||||
Augie Fackler
|
r43347 | if opts.get(b'exact') and nocommit: | ||
Gregory Szorc
|
r37639 | # --exact with --no-commit is still useful in that it does merge | ||
# and branch bits | ||||
Augie Fackler
|
r43347 | ui.warn(_(b"warning: can't check exact import with --no-commit\n")) | ||
elif opts.get(b'exact') and (not n or hex(n) != nodeid): | ||||
raise error.Abort(_(b'patch is damaged or loses information')) | ||||
msg = _(b'applied to working directory') | ||||
Gregory Szorc
|
r37639 | if n: | ||
# i18n: refers to a short changeset id | ||||
Augie Fackler
|
r43347 | msg = _(b'created %s') % short(n) | ||
Gregory Szorc
|
r37639 | return msg, n, rejects | ||
Pierre-Yves David
|
r20500 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r26545 | # facility to let extensions include additional data in an exported patch | ||
# list of identifiers to be executed in order | ||||
extraexport = [] | ||||
# mapping from identifier to actual export function | ||||
# function as to return a string to be added to the header or None | ||||
# it is given two arguments (sequencenumber, changectx) | ||||
extraexportmap = {} | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37620 | def _exportsingle(repo, ctx, fm, match, switch_parent, seqno, diffopts): | ||
Yuya Nishihara
|
r32660 | node = scmutil.binnode(ctx) | ||
Augie Fackler
|
r32433 | parents = [p.node() for p in ctx.parents() if p] | ||
branch = ctx.branch() | ||||
if switch_parent: | ||||
parents.reverse() | ||||
if parents: | ||||
prev = parents[0] | ||||
else: | ||||
Joerg Sonnenberger
|
r47771 | prev = repo.nullid | ||
Augie Fackler
|
r32433 | |||
Yuya Nishihara
|
r37620 | fm.context(ctx=ctx) | ||
Augie Fackler
|
r43347 | fm.plain(b'# HG changeset patch\n') | ||
fm.write(b'user', b'# User %s\n', ctx.user()) | ||||
fm.plain(b'# Date %d %d\n' % ctx.date()) | ||||
fm.write(b'date', b'# %s\n', fm.formatdate(ctx.date())) | ||||
Augie Fackler
|
r43346 | fm.condwrite( | ||
Augie Fackler
|
r43347 | branch and branch != b'default', b'branch', b'# Branch %s\n', branch | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | fm.write(b'node', b'# Node ID %s\n', hex(node)) | ||
fm.plain(b'# Parent %s\n' % hex(prev)) | ||||
Augie Fackler
|
r32433 | if len(parents) > 1: | ||
Augie Fackler
|
r43347 | fm.plain(b'# Parent %s\n' % hex(parents[1])) | ||
fm.data(parents=fm.formatlist(pycompat.maplist(hex, parents), name=b'node')) | ||||
Yuya Nishihara
|
r37620 | |||
# TODO: redesign extraexportmap function to support formatter | ||||
Augie Fackler
|
r32433 | for headerid in extraexport: | ||
header = extraexportmap[headerid](seqno, ctx) | ||||
if header is not None: | ||||
Augie Fackler
|
r43347 | fm.plain(b'# %s\n' % header) | ||
fm.write(b'desc', b'%s\n', ctx.description().rstrip()) | ||||
fm.plain(b'\n') | ||||
Yuya Nishihara
|
r37620 | |||
if fm.isplain(): | ||||
chunkiter = patch.diffui(repo, prev, node, match, opts=diffopts) | ||||
for chunk, label in chunkiter: | ||||
fm.plain(chunk, label=label) | ||||
else: | ||||
chunkiter = patch.diff(repo, prev, node, match, opts=diffopts) | ||||
# TODO: make it structured? | ||||
fm.data(diff=b''.join(chunkiter)) | ||||
Augie Fackler
|
r32433 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r37622 | def _exportfile(repo, revs, fm, dest, switch_parent, diffopts, match): | ||
Yuya Nishihara
|
r37618 | """Export changesets to stdout or a single file""" | ||
for seqno, rev in enumerate(revs, 1): | ||||
ctx = repo[rev] | ||||
Augie Fackler
|
r43347 | if not dest.startswith(b'<'): | ||
repo.ui.note(b"%s\n" % dest) | ||||
Yuya Nishihara
|
r37620 | fm.startitem() | ||
_exportsingle(repo, ctx, fm, match, switch_parent, seqno, diffopts) | ||||
Yuya Nishihara
|
r37622 | |||
Augie Fackler
|
r43346 | |||
def _exportfntemplate( | ||||
repo, revs, basefm, fntemplate, switch_parent, diffopts, match | ||||
): | ||||
Yuya Nishihara
|
r37618 | """Export changesets to possibly multiple files""" | ||
total = len(revs) | ||||
revwidth = max(len(str(rev)) for rev in revs) | ||||
Yuya Nishihara
|
r37619 | filemap = util.sortdict() # filename: [(seqno, rev), ...] | ||
Yuya Nishihara
|
r37618 | |||
for seqno, rev in enumerate(revs, 1): | ||||
ctx = repo[rev] | ||||
Augie Fackler
|
r43346 | dest = makefilename( | ||
ctx, fntemplate, total=total, seqno=seqno, revwidth=revwidth | ||||
) | ||||
Yuya Nishihara
|
r37619 | filemap.setdefault(dest, []).append((seqno, rev)) | ||
for dest in filemap: | ||||
Yuya Nishihara
|
r37622 | with formatter.maybereopen(basefm, dest) as fm: | ||
Augie Fackler
|
r43347 | repo.ui.note(b"%s\n" % dest) | ||
Yuya Nishihara
|
r37619 | for seqno, rev in filemap[dest]: | ||
Yuya Nishihara
|
r37620 | fm.startitem() | ||
Yuya Nishihara
|
r37619 | ctx = repo[rev] | ||
Augie Fackler
|
r43346 | _exportsingle( | ||
repo, ctx, fm, match, switch_parent, seqno, diffopts | ||||
) | ||||
Yuya Nishihara
|
r37618 | |||
Martin von Zweigbergk
|
r42673 | def _prefetchchangedfiles(repo, revs, match): | ||
allfiles = set() | ||||
for rev in revs: | ||||
for file in repo[rev].files(): | ||||
if not match or match(file): | ||||
allfiles.add(file) | ||||
Rodrigo Damazio Bovendorp
|
r45632 | match = scmutil.matchfiles(repo, allfiles) | ||
revmatches = [(rev, match) for rev in revs] | ||||
scmutil.prefetchfiles(repo, revmatches) | ||||
Martin von Zweigbergk
|
r42673 | |||
Augie Fackler
|
r43346 | |||
def export( | ||||
repo, | ||||
revs, | ||||
basefm, | ||||
Augie Fackler
|
r43347 | fntemplate=b'hg-%h.patch', | ||
Augie Fackler
|
r43346 | switch_parent=False, | ||
opts=None, | ||||
match=None, | ||||
): | ||||
Augie Fackler
|
r46554 | """export changesets as hg patches | ||
Augie Fackler
|
r32430 | |||
Args: | ||||
repo: The repository from which we're exporting revisions. | ||||
revs: A list of revisions to export as revision numbers. | ||||
Yuya Nishihara
|
r37622 | basefm: A formatter to which patches should be written. | ||
Augie Fackler
|
r32431 | fntemplate: An optional string to use for generating patch file names. | ||
Augie Fackler
|
r32430 | switch_parent: If True, show diffs against second parent when not nullid. | ||
Default is false, which always shows diff against p1. | ||||
opts: diff options to use for generating the patch. | ||||
match: If specified, only export changes to files matching this matcher. | ||||
Returns: | ||||
Nothing. | ||||
Side Effect: | ||||
"HG Changeset Patch" data is emitted to one of the following | ||||
destinations: | ||||
Augie Fackler
|
r32431 | fntemplate specified: Each rev is written to a unique file named using | ||
Augie Fackler
|
r32430 | the given template. | ||
Yuya Nishihara
|
r37622 | Otherwise: All revs will be written to basefm. | ||
Augie Fackler
|
r46554 | """ | ||
Martin von Zweigbergk
|
r42673 | _prefetchchangedfiles(repo, revs, match) | ||
Matt Harbison
|
r37781 | |||
Yuya Nishihara
|
r37621 | if not fntemplate: | ||
Augie Fackler
|
r43347 | _exportfile( | ||
repo, revs, basefm, b'<unnamed>', switch_parent, opts, match | ||||
) | ||||
Yuya Nishihara
|
r37618 | else: | ||
Augie Fackler
|
r43346 | _exportfntemplate( | ||
repo, revs, basefm, fntemplate, switch_parent, opts, match | ||||
) | ||||
Benoit Boissinot
|
r10611 | |||
Yuya Nishihara
|
r37621 | def exportfile(repo, revs, fp, switch_parent=False, opts=None, match=None): | ||
"""Export changesets to the given file stream""" | ||||
Martin von Zweigbergk
|
r42673 | _prefetchchangedfiles(repo, revs, match) | ||
Matt Harbison
|
r37781 | |||
Augie Fackler
|
r43347 | dest = getattr(fp, 'name', b'<unnamed>') | ||
with formatter.formatter(repo.ui, fp, b'export', {}) as fm: | ||||
Yuya Nishihara
|
r37622 | _exportfile(repo, revs, fm, dest, switch_parent, opts, match) | ||
Yuya Nishihara
|
r37621 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r29795 | def showmarker(fm, marker, index=None): | ||
Pierre-Yves David
|
r20470 | """utility function to display obsolescence marker in a readable way | ||
To be used by debug function.""" | ||||
Kostia Balytskyi
|
r28613 | if index is not None: | ||
Augie Fackler
|
r43347 | fm.write(b'index', b'%i ', index) | ||
fm.write(b'prednode', b'%s ', hex(marker.prednode())) | ||||
Yuya Nishihara
|
r29795 | succs = marker.succnodes() | ||
Augie Fackler
|
r43346 | fm.condwrite( | ||
Augie Fackler
|
r43347 | succs, | ||
b'succnodes', | ||||
b'%s ', | ||||
fm.formatlist(map(hex, succs), name=b'node'), | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | fm.write(b'flag', b'%X ', marker.flags()) | ||
Pierre-Yves David
|
r22260 | parents = marker.parentnodes() | ||
if parents is not None: | ||||
Augie Fackler
|
r43346 | fm.write( | ||
Augie Fackler
|
r43347 | b'parentnodes', | ||
b'{%s} ', | ||||
fm.formatlist(map(hex, parents), name=b'node', sep=b', '), | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | fm.write(b'date', b'(%s) ', fm.formatdate(marker.date())) | ||
Yuya Nishihara
|
r29795 | meta = marker.metadata().copy() | ||
Augie Fackler
|
r43347 | meta.pop(b'date', None) | ||
Yuya Nishihara
|
r38594 | smeta = pycompat.rapply(pycompat.maybebytestr, meta) | ||
Augie Fackler
|
r43347 | fm.write( | ||
b'metadata', b'{%s}', fm.formatdict(smeta, fmt=b'%r: %r', sep=b', ') | ||||
) | ||||
fm.plain(b'\n') | ||||
Pierre-Yves David
|
r20470 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r3814 | def finddate(ui, repo, date): | ||
"""Find the tipmost changeset that matches the given date spec""" | ||||
Yuya Nishihara
|
r46029 | mrevs = repo.revs(b'date(%s)', date) | ||
try: | ||||
rev = mrevs.max() | ||||
except ValueError: | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"revision matching date not found")) | ||
Yuya Nishihara
|
r46029 | |||
ui.status( | ||||
_(b"found revision %d from %s\n") | ||||
% (rev, dateutil.datestr(repo[rev].date())) | ||||
) | ||||
return b'%d' % rev | ||||
Matt Mackall
|
r3814 | |||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r41799 | def add(ui, repo, match, prefix, uipathfn, explicitonly, **opts): | ||
Martin Geisler
|
r12269 | bad = [] | ||
Matt Harbison
|
r25436 | |||
badfn = lambda x, y: bad.append(x) or match.bad(x, y) | ||||
Martin Geisler
|
r12269 | names = [] | ||
Martin Geisler
|
r12270 | wctx = repo[None] | ||
Adrian Buehlmann
|
r14138 | cca = None | ||
abort, warn = scmutil.checkportabilityalert(ui) | ||||
if abort or warn: | ||||
Joshua Redstone
|
r17201 | cca = scmutil.casecollisionauditor(ui, abort, repo.dirstate) | ||
Matt Harbison
|
r25436 | |||
Martin von Zweigbergk
|
r40123 | match = repo.narrowmatch(match, includeexact=True) | ||
Durham Goode
|
r26206 | badmatch = matchmod.badmatch(match, badfn) | ||
dirstate = repo.dirstate | ||||
# We don't want to just call wctx.walk here, since it would return a lot of | ||||
# clean files, which we aren't interested in and takes time. | ||||
Augie Fackler
|
r43346 | for f in sorted( | ||
dirstate.walk( | ||||
badmatch, | ||||
subrepos=sorted(wctx.substate), | ||||
unknown=True, | ||||
ignored=False, | ||||
full=False, | ||||
) | ||||
): | ||||
Martin von Zweigbergk
|
r52056 | entry = dirstate.get_entry(f) | ||
# We don't want to even attmpt to add back files that have been removed | ||||
# It would lead to a misleading message saying we're adding the path, | ||||
# and can also lead to file/dir conflicts when attempting to add it. | ||||
removed = entry and entry.removed | ||||
Martin Geisler
|
r12269 | exact = match.exact(f) | ||
Martin von Zweigbergk
|
r52056 | if ( | ||
exact | ||||
or not explicitonly | ||||
and f not in wctx | ||||
and repo.wvfs.lexists(f) | ||||
and not removed | ||||
): | ||||
Adrian Buehlmann
|
r14138 | if cca: | ||
cca(f) | ||||
Martin Geisler
|
r12269 | names.append(f) | ||
if ui.verbose or not exact: | ||||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b'adding %s\n') % uipathfn(f), label=b'ui.addremove.added' | ||
Augie Fackler
|
r43346 | ) | ||
Martin Geisler
|
r12270 | |||
Mads Kiilerich
|
r18364 | for subpath in sorted(wctx.substate): | ||
David M. Carr
|
r15410 | sub = wctx.sub(subpath) | ||
try: | ||||
Martin von Zweigbergk
|
r28017 | submatch = matchmod.subdirmatcher(subpath, match) | ||
Martin von Zweigbergk
|
r41777 | subprefix = repo.wvfs.reljoin(prefix, subpath) | ||
Martin von Zweigbergk
|
r41799 | subuipathfn = scmutil.subdiruipathfn(subpath, uipathfn) | ||
Augie Fackler
|
r43906 | if opts.get('subrepos'): | ||
Augie Fackler
|
r43346 | bad.extend( | ||
sub.add(ui, submatch, subprefix, subuipathfn, False, **opts) | ||||
) | ||||
David M. Carr
|
r15410 | else: | ||
Augie Fackler
|
r43346 | bad.extend( | ||
sub.add(ui, submatch, subprefix, subuipathfn, True, **opts) | ||||
) | ||||
David M. Carr
|
r15410 | except error.LookupError: | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b"skipping missing subrepository: %s\n") % uipathfn(subpath) | ||
Augie Fackler
|
r43346 | ) | ||
Martin Geisler
|
r12270 | |||
Augie Fackler
|
r43906 | if not opts.get('dry_run'): | ||
Martin Geisler
|
r12270 | rejected = wctx.add(names, prefix) | ||
Martin Geisler
|
r12269 | bad.extend(f for f in rejected if f in match.files()) | ||
return bad | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r32005 | def addwebdirpath(repo, serverpath, webconf): | ||
webconf[serverpath] = repo.root | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b'adding %s = %s\n' % (serverpath, repo.root)) | ||
for r in repo.revs(b'filelog("path:.hgsub")'): | ||||
Matt Harbison
|
r32005 | ctx = repo[r] | ||
for subpath in ctx.substate: | ||||
ctx.sub(subpath).addwebdirpath(serverpath, webconf) | ||||
Augie Fackler
|
r43346 | |||
def forget( | ||||
ui, repo, match, prefix, uipathfn, explicitonly, dryrun, interactive | ||||
): | ||||
Sushil khanchi
|
r37796 | if dryrun and interactive: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError( | ||
_(b"cannot specify both --dry-run and --interactive") | ||||
) | ||||
David M. Carr
|
r15912 | bad = [] | ||
Matt Harbison
|
r25437 | badfn = lambda x, y: bad.append(x) or match.bad(x, y) | ||
David M. Carr
|
r15912 | wctx = repo[None] | ||
forgot = [] | ||||
Matt Harbison
|
r25437 | |||
s = repo.status(match=matchmod.badmatch(match, badfn), clean=True) | ||||
Martin von Zweigbergk
|
r32174 | forget = sorted(s.modified + s.added + s.deleted + s.clean) | ||
David M. Carr
|
r15912 | if explicitonly: | ||
forget = [f for f in forget if match.exact(f)] | ||||
Mads Kiilerich
|
r18364 | for subpath in sorted(wctx.substate): | ||
David M. Carr
|
r15912 | sub = wctx.sub(subpath) | ||
Martin von Zweigbergk
|
r41776 | submatch = matchmod.subdirmatcher(subpath, match) | ||
subprefix = repo.wvfs.reljoin(prefix, subpath) | ||||
Martin von Zweigbergk
|
r41802 | subuipathfn = scmutil.subdiruipathfn(subpath, uipathfn) | ||
David M. Carr
|
r15912 | try: | ||
Augie Fackler
|
r43346 | subbad, subforgot = sub.forget( | ||
submatch, | ||||
subprefix, | ||||
subuipathfn, | ||||
dryrun=dryrun, | ||||
interactive=interactive, | ||||
) | ||||
Augie Fackler
|
r43347 | bad.extend([subpath + b'/' + f for f in subbad]) | ||
forgot.extend([subpath + b'/' + f for f in subforgot]) | ||||
David M. Carr
|
r15912 | except error.LookupError: | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b"skipping missing subrepository: %s\n") % uipathfn(subpath) | ||
Augie Fackler
|
r43346 | ) | ||
David M. Carr
|
r15912 | |||
FUJIWARA Katsunori
|
r16070 | if not explicitonly: | ||
for f in match.files(): | ||||
Matt Harbison
|
r23673 | if f not in repo.dirstate and not repo.wvfs.isdir(f): | ||
David M. Carr
|
r15912 | if f not in forgot: | ||
Matt Harbison
|
r23673 | if repo.wvfs.exists(f): | ||
Matt Harbison
|
r24548 | # Don't complain if the exact case match wasn't given. | ||
# But don't do this until after checking 'forgot', so | ||||
# that subrepo files aren't normalized, and this op is | ||||
# purely from data cached by the status walk above. | ||||
if repo.dirstate.normalize(f) in repo.dirstate: | ||||
continue | ||||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _( | ||
b'not removing %s: ' | ||||
b'file is already untracked\n' | ||||
) | ||||
Augie Fackler
|
r43346 | % uipathfn(f) | ||
) | ||||
David M. Carr
|
r15912 | bad.append(f) | ||
Sushil khanchi
|
r37796 | if interactive: | ||
Augie Fackler
|
r43346 | responses = _( | ||
Augie Fackler
|
r43347 | b'[Ynsa?]' | ||
b'$$ &Yes, forget this file' | ||||
b'$$ &No, skip this file' | ||||
b'$$ &Skip remaining files' | ||||
b'$$ Include &all remaining files' | ||||
b'$$ &? (display help)' | ||||
Augie Fackler
|
r43346 | ) | ||
Sushil khanchi
|
r37774 | for filename in forget[:]: | ||
Augie Fackler
|
r43346 | r = ui.promptchoice( | ||
Augie Fackler
|
r43347 | _(b'forget %s %s') % (uipathfn(filename), responses) | ||
Augie Fackler
|
r43346 | ) | ||
if r == 4: # ? | ||||
Sushil khanchi
|
r37774 | while r == 4: | ||
for c, t in ui.extractchoices(responses)[1]: | ||||
Augie Fackler
|
r43347 | ui.write(b'%s - %s\n' % (c, encoding.lower(t))) | ||
Augie Fackler
|
r43346 | r = ui.promptchoice( | ||
Augie Fackler
|
r43347 | _(b'forget %s %s') % (uipathfn(filename), responses) | ||
Augie Fackler
|
r43346 | ) | ||
if r == 0: # yes | ||||
Sushil khanchi
|
r37774 | continue | ||
Augie Fackler
|
r43346 | elif r == 1: # no | ||
Sushil khanchi
|
r37774 | forget.remove(filename) | ||
Augie Fackler
|
r43346 | elif r == 2: # Skip | ||
Sushil khanchi
|
r37774 | fnindex = forget.index(filename) | ||
del forget[fnindex:] | ||||
break | ||||
Augie Fackler
|
r43346 | elif r == 3: # All | ||
Sushil khanchi
|
r37774 | break | ||
David M. Carr
|
r15912 | for f in forget: | ||
Sushil khanchi
|
r37796 | if ui.verbose or not match.exact(f) or interactive: | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b'removing %s\n') % uipathfn(f), label=b'ui.addremove.removed' | ||
Augie Fackler
|
r43346 | ) | ||
David M. Carr
|
r15912 | |||
Sushil khanchi
|
r36957 | if not dryrun: | ||
rejected = wctx.forget(forget, prefix) | ||||
bad.extend(f for f in rejected if f in match.files()) | ||||
forgot.extend(f for f in forget if f not in rejected) | ||||
David M. Carr
|
r15912 | return bad, forgot | ||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r41914 | def files(ui, ctx, m, uipathfn, fm, fmt, subrepos): | ||
Matt Harbison
|
r24275 | ret = 1 | ||
Augie Fackler
|
r43347 | needsfctx = ui.verbose or {b'size', b'flags'} & fm.datahint() | ||
Valentin Gatien-Baron
|
r45388 | if fm.isplain() and not needsfctx: | ||
# Fast path. The speed-up comes from skipping the formatter, and batching | ||||
# calls to ui.write. | ||||
buf = [] | ||||
for f in ctx.matches(m): | ||||
buf.append(fmt % uipathfn(f)) | ||||
if len(buf) > 100: | ||||
ui.write(b''.join(buf)) | ||||
del buf[:] | ||||
ret = 0 | ||||
if buf: | ||||
ui.write(b''.join(buf)) | ||||
else: | ||||
for f in ctx.matches(m): | ||||
fm.startitem() | ||||
fm.context(ctx=ctx) | ||||
if needsfctx: | ||||
fc = ctx[f] | ||||
fm.write(b'size flags', b'% 10d % 1s ', fc.size(), fc.flags()) | ||||
fm.data(path=f) | ||||
fm.plain(fmt % uipathfn(f)) | ||||
ret = 0 | ||||
Matt Harbison
|
r24275 | |||
Matt Harbison
|
r25228 | for subpath in sorted(ctx.substate): | ||
Hannes Oldenburg
|
r29802 | submatch = matchmod.subdirmatcher(subpath, m) | ||
Martin von Zweigbergk
|
r41914 | subuipathfn = scmutil.subdiruipathfn(subpath, uipathfn) | ||
Augie Fackler
|
r43346 | if subrepos or m.exact(subpath) or any(submatch.files()): | ||
Matt Harbison
|
r24413 | sub = ctx.sub(subpath) | ||
try: | ||||
Matt Harbison
|
r28387 | recurse = m.exact(subpath) or subrepos | ||
Augie Fackler
|
r43346 | if ( | ||
sub.printfiles(ui, submatch, subuipathfn, fm, fmt, recurse) | ||||
== 0 | ||||
): | ||||
Matt Harbison
|
r24413 | ret = 0 | ||
except error.LookupError: | ||||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b"skipping missing subrepository: %s\n") | ||
Augie Fackler
|
r43346 | % uipathfn(subpath) | ||
) | ||||
Matt Harbison
|
r24413 | |||
Matt Harbison
|
r24275 | return ret | ||
Augie Fackler
|
r43346 | |||
def remove( | ||||
ui, repo, m, prefix, uipathfn, after, force, subrepos, dryrun, warnings=None | ||||
): | ||||
Matt Harbison
|
r23289 | ret = 0 | ||
s = repo.status(match=m, clean=True) | ||||
Augie Fackler
|
r44043 | modified, added, deleted, clean = s.modified, s.added, s.deleted, s.clean | ||
Matt Harbison
|
r23289 | |||
Matt Harbison
|
r23325 | wctx = repo[None] | ||
timeless
|
r28607 | if warnings is None: | ||
warnings = [] | ||||
warn = True | ||||
else: | ||||
warn = False | ||||
timeless
|
r28608 | subs = sorted(wctx.substate) | ||
Augie Fackler
|
r43346 | progress = ui.makeprogress( | ||
Augie Fackler
|
r43347 | _(b'searching'), total=len(subs), unit=_(b'subrepos') | ||
Augie Fackler
|
r43346 | ) | ||
timeless
|
r28608 | for subpath in subs: | ||
Hannes Oldenburg
|
r29802 | submatch = matchmod.subdirmatcher(subpath, m) | ||
Martin von Zweigbergk
|
r41775 | subprefix = repo.wvfs.reljoin(prefix, subpath) | ||
Martin von Zweigbergk
|
r41800 | subuipathfn = scmutil.subdiruipathfn(subpath, uipathfn) | ||
Hannes Oldenburg
|
r29802 | if subrepos or m.exact(subpath) or any(submatch.files()): | ||
Martin von Zweigbergk
|
r38366 | progress.increment() | ||
Matt Harbison
|
r23325 | sub = wctx.sub(subpath) | ||
try: | ||||
Augie Fackler
|
r43346 | if sub.removefiles( | ||
submatch, | ||||
subprefix, | ||||
subuipathfn, | ||||
after, | ||||
force, | ||||
subrepos, | ||||
dryrun, | ||||
warnings, | ||||
): | ||||
Matt Harbison
|
r23325 | ret = 1 | ||
except error.LookupError: | ||||
Augie Fackler
|
r43346 | warnings.append( | ||
Augie Fackler
|
r43347 | _(b"skipping missing subrepository: %s\n") | ||
Augie Fackler
|
r43346 | % uipathfn(subpath) | ||
) | ||||
Martin von Zweigbergk
|
r38392 | progress.complete() | ||
Matt Harbison
|
r23325 | |||
Matt Harbison
|
r23289 | # warn about failure to delete explicit files/dirs | ||
r43923 | deleteddirs = pathutil.dirs(deleted) | |||
timeless
|
r28608 | files = m.files() | ||
Augie Fackler
|
r43347 | progress = ui.makeprogress( | ||
_(b'deleting'), total=len(files), unit=_(b'files') | ||||
) | ||||
timeless
|
r28608 | for f in files: | ||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r23325 | def insubrepo(): | ||
for subpath in wctx.substate: | ||||
Augie Fackler
|
r43347 | if f.startswith(subpath + b'/'): | ||
Matt Harbison
|
r23325 | return True | ||
return False | ||||
Martin von Zweigbergk
|
r38366 | progress.increment() | ||
Martin von Zweigbergk
|
r24955 | isdir = f in deleteddirs or wctx.hasdir(f) | ||
Augie Fackler
|
r43347 | if f in repo.dirstate or isdir or f == b'.' or insubrepo() or f in subs: | ||
Matt Harbison
|
r23289 | continue | ||
Matt Harbison
|
r23325 | |||
Matt Harbison
|
r23674 | if repo.wvfs.exists(f): | ||
if repo.wvfs.isdir(f): | ||||
Augie Fackler
|
r43346 | warnings.append( | ||
Augie Fackler
|
r43347 | _(b'not removing %s: no tracked files\n') % uipathfn(f) | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r23289 | else: | ||
Augie Fackler
|
r43346 | warnings.append( | ||
Augie Fackler
|
r43347 | _(b'not removing %s: file is untracked\n') % uipathfn(f) | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r23289 | # missing files will generate a warning elsewhere | ||
ret = 1 | ||||
Martin von Zweigbergk
|
r38392 | progress.complete() | ||
Matt Harbison
|
r23289 | |||
if force: | ||||
list = modified + deleted + clean + added | ||||
elif after: | ||||
list = deleted | ||||
timeless
|
r28608 | remaining = modified + added + clean | ||
Augie Fackler
|
r43346 | progress = ui.makeprogress( | ||
Augie Fackler
|
r43347 | _(b'skipping'), total=len(remaining), unit=_(b'files') | ||
Augie Fackler
|
r43346 | ) | ||
timeless
|
r28608 | for f in remaining: | ||
Martin von Zweigbergk
|
r38366 | progress.increment() | ||
pavanpc@fb.com
|
r35120 | if ui.verbose or (f in files): | ||
Augie Fackler
|
r43346 | warnings.append( | ||
Augie Fackler
|
r43347 | _(b'not removing %s: file still exists\n') % uipathfn(f) | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r23289 | ret = 1 | ||
Martin von Zweigbergk
|
r38392 | progress.complete() | ||
Matt Harbison
|
r23289 | else: | ||
list = deleted + clean | ||||
Augie Fackler
|
r43346 | progress = ui.makeprogress( | ||
Augie Fackler
|
r43347 | _(b'skipping'), total=(len(modified) + len(added)), unit=_(b'files') | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r23289 | for f in modified: | ||
Martin von Zweigbergk
|
r38366 | progress.increment() | ||
Augie Fackler
|
r43346 | warnings.append( | ||
_( | ||||
Augie Fackler
|
r43347 | b'not removing %s: file is modified (use -f' | ||
b' to force removal)\n' | ||||
Augie Fackler
|
r43346 | ) | ||
% uipathfn(f) | ||||
) | ||||
Matt Harbison
|
r23289 | ret = 1 | ||
for f in added: | ||||
Martin von Zweigbergk
|
r38366 | progress.increment() | ||
Augie Fackler
|
r43346 | warnings.append( | ||
_( | ||||
Augie Fackler
|
r43347 | b"not removing %s: file has been marked for add" | ||
b" (use 'hg forget' to undo add)\n" | ||||
Augie Fackler
|
r43346 | ) | ||
% uipathfn(f) | ||||
) | ||||
Matt Harbison
|
r23289 | ret = 1 | ||
Martin von Zweigbergk
|
r38392 | progress.complete() | ||
timeless
|
r28608 | |||
list = sorted(list) | ||||
Augie Fackler
|
r43347 | progress = ui.makeprogress( | ||
_(b'deleting'), total=len(list), unit=_(b'files') | ||||
) | ||||
timeless
|
r28608 | for f in list: | ||
Matt Harbison
|
r23289 | if ui.verbose or not m.exact(f): | ||
Martin von Zweigbergk
|
r38366 | progress.increment() | ||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b'removing %s\n') % uipathfn(f), label=b'ui.addremove.removed' | ||
Augie Fackler
|
r43346 | ) | ||
Martin von Zweigbergk
|
r38392 | progress.complete() | ||
Matt Harbison
|
r23289 | |||
Sushil khanchi
|
r37168 | if not dryrun: | ||
with repo.wlock(): | ||||
if not after: | ||||
for f in list: | ||||
if f in added: | ||||
Augie Fackler
|
r43346 | continue # we never unlink added files on remove | ||
rmdir = repo.ui.configbool( | ||||
Augie Fackler
|
r43347 | b'experimental', b'removeemptydirs' | ||
Augie Fackler
|
r43346 | ) | ||
Kyle Lippincott
|
r38512 | repo.wvfs.unlinkpath(f, ignoremissing=True, rmdir=rmdir) | ||
Sushil khanchi
|
r37168 | repo[None].forget(list) | ||
Matt Harbison
|
r23289 | |||
timeless
|
r28607 | if warn: | ||
for warning in warnings: | ||||
ui.warn(warning) | ||||
Matt Harbison
|
r23289 | return ret | ||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r42678 | def _catfmtneedsdata(fm): | ||
Augie Fackler
|
r43347 | return not fm.datahint() or b'data' in fm.datahint() | ||
Matt Harbison
|
r42678 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r35680 | def _updatecatformatter(fm, ctx, matcher, path, decode): | ||
"""Hook for adding data to the formatter used by ``hg cat``. | ||||
Extensions (e.g., lfs) can wrap this to inject keywords/data, but must call | ||||
this method first.""" | ||||
Matt Harbison
|
r42678 | |||
# data() can be expensive to fetch (e.g. lfs), so don't fetch it if it | ||||
# wasn't requested. | ||||
data = b'' | ||||
if _catfmtneedsdata(fm): | ||||
data = ctx[path].data() | ||||
if decode: | ||||
data = ctx.repo().wwritedata(path, data) | ||||
Matt Harbison
|
r35680 | fm.startitem() | ||
Yuya Nishihara
|
r38557 | fm.context(ctx=ctx) | ||
Augie Fackler
|
r43347 | fm.write(b'data', b'%s', data) | ||
Yuya Nishihara
|
r39405 | fm.data(path=path) | ||
Matt Harbison
|
r35680 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r32578 | def cat(ui, repo, ctx, matcher, basefm, fntemplate, prefix, **opts): | ||
Matt Harbison
|
r21040 | err = 1 | ||
def write(path): | ||||
Yuya Nishihara
|
r32578 | filename = None | ||
Yuya Nishihara
|
r32576 | if fntemplate: | ||
Augie Fackler
|
r43346 | filename = makefilename( | ||
ctx, fntemplate, pathname=os.path.join(prefix, path) | ||||
) | ||||
Ryan McElroy
|
r35010 | # attempt to create the directory if it does not already exist | ||
try: | ||||
os.makedirs(os.path.dirname(filename)) | ||||
except OSError: | ||||
pass | ||||
Yuya Nishihara
|
r37615 | with formatter.maybereopen(basefm, filename) as fm: | ||
Matt Harbison
|
r51168 | _updatecatformatter(fm, ctx, matcher, path, opts.get('decode')) | ||
Matt Harbison
|
r21040 | |||
# Automation often uses hg cat on single files, so special case it | ||||
# for performance to avoid the cost of parsing the manifest. | ||||
if len(matcher.files()) == 1 and not matcher.anypats(): | ||||
file = matcher.files()[0] | ||||
Durham Goode
|
r30340 | mfl = repo.manifestlog | ||
Yuya Nishihara
|
r24718 | mfnode = ctx.manifestnode() | ||
Durham Goode
|
r30340 | try: | ||
if mfnode and mfl[mfnode].find(file)[0]: | ||||
Matt Harbison
|
r42678 | if _catfmtneedsdata(basefm): | ||
Rodrigo Damazio Bovendorp
|
r45632 | scmutil.prefetchfiles(repo, [(ctx.rev(), matcher)]) | ||
Durham Goode
|
r30340 | write(file) | ||
return 0 | ||||
except KeyError: | ||||
pass | ||||
Matt Harbison
|
r21040 | |||
Matt Harbison
|
r42678 | if _catfmtneedsdata(basefm): | ||
Rodrigo Damazio Bovendorp
|
r45632 | scmutil.prefetchfiles(repo, [(ctx.rev(), matcher)]) | ||
Matt Harbison
|
r37780 | |||
for abs in ctx.walk(matcher): | ||||
Matt Harbison
|
r21040 | write(abs) | ||
err = 0 | ||||
Matt Harbison
|
r21041 | |||
Martin von Zweigbergk
|
r41812 | uipathfn = scmutil.getuipathfn(repo, legacyrelativevalue=True) | ||
Matt Harbison
|
r21041 | for subpath in sorted(ctx.substate): | ||
sub = ctx.sub(subpath) | ||||
try: | ||||
Martin von Zweigbergk
|
r28017 | submatch = matchmod.subdirmatcher(subpath, matcher) | ||
Martin von Zweigbergk
|
r41798 | subprefix = os.path.join(prefix, subpath) | ||
Augie Fackler
|
r43346 | if not sub.cat( | ||
submatch, | ||||
basefm, | ||||
fntemplate, | ||||
subprefix, | ||||
Matt Harbison
|
r51168 | **opts, | ||
Augie Fackler
|
r43346 | ): | ||
Matt Harbison
|
r21041 | err = 0 | ||
except error.RepoLookupError: | ||||
Augie Fackler
|
r43346 | ui.status( | ||
Augie Fackler
|
r43347 | _(b"skipping missing subrepository: %s\n") % uipathfn(subpath) | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r21041 | |||
Matt Harbison
|
r21040 | return err | ||
Augie Fackler
|
r43346 | |||
r50924 | class _AddRemoveContext: | |||
"""a small (hacky) context to deal with lazy opening of context | ||||
This is to be used in the `commit` function right below. This deals with | ||||
lazily open a `changing_files` context inside a `transaction` that span the | ||||
full commit operation. | ||||
We need : | ||||
- a `changing_files` context to wrap the dirstate change within the | ||||
"addremove" operation, | ||||
- a transaction to make sure these change are not written right after the | ||||
addremove, but when the commit operation succeed. | ||||
However it get complicated because: | ||||
- opening a transaction "this early" shuffle hooks order, especially the | ||||
`precommit` one happening after the `pretxtopen` one which I am not too | ||||
enthusiastic about. | ||||
- the `mq` extensions + the `record` extension stacks many layers of call | ||||
to implement `qrefresh --interactive` and this result with `mq` calling a | ||||
`strip` in the middle of this function. Which prevent the existence of | ||||
transaction wrapping all of its function code. (however, `qrefresh` never | ||||
call the `addremove` bits. | ||||
- the largefile extensions (and maybe other extensions?) wraps `addremove` | ||||
so slicing `addremove` in smaller bits is a complex endeavour. | ||||
So I eventually took a this shortcut that open the transaction if we | ||||
actually needs it, not disturbing much of the rest of the code. | ||||
It will result in some hooks order change for `hg commit --addremove`, | ||||
however it seems a corner case enough to ignore that for now (hopefully). | ||||
Notes that None of the above problems seems insurmountable, however I have | ||||
been fighting with this specific piece of code for a couple of day already | ||||
and I need a solution to keep moving forward on the bigger work around | ||||
`changing_files` context that is being introduced at the same time as this | ||||
hack. | ||||
Each problem seems to have a solution: | ||||
- the hook order issue could be solved by refactoring the many-layer stack | ||||
that currently composes a commit and calling them earlier, | ||||
- the mq issue could be solved by refactoring `mq` so that the final strip | ||||
is done after transaction closure. Be warned that the mq code is quite | ||||
antic however. | ||||
- large-file could be reworked in parallel of the `addremove` to be | ||||
friendlier to this. | ||||
However each of these tasks are too much a diversion right now. In addition | ||||
they will be much easier to undertake when the `changing_files` dust has | ||||
settled.""" | ||||
def __init__(self, repo): | ||||
self._repo = repo | ||||
self._transaction = None | ||||
self._dirstate_context = None | ||||
self._state = None | ||||
def __enter__(self): | ||||
assert self._state is None | ||||
self._state = True | ||||
return self | ||||
def open_transaction(self): | ||||
"""open a `transaction` and `changing_files` context | ||||
Call this when you know that change to the dirstate will be needed and | ||||
we need to open the transaction early | ||||
This will also open the dirstate `changing_files` context, so you should | ||||
call `close_dirstate_context` when the distate changes are done. | ||||
""" | ||||
assert self._state is not None | ||||
if self._transaction is None: | ||||
self._transaction = self._repo.transaction(b'commit') | ||||
self._transaction.__enter__() | ||||
if self._dirstate_context is None: | ||||
self._dirstate_context = self._repo.dirstate.changing_files( | ||||
self._repo | ||||
) | ||||
self._dirstate_context.__enter__() | ||||
def close_dirstate_context(self): | ||||
"""close the change_files if any | ||||
Call this after the (potential) `open_transaction` call to close the | ||||
(potential) changing_files context. | ||||
""" | ||||
if self._dirstate_context is not None: | ||||
self._dirstate_context.__exit__(None, None, None) | ||||
self._dirstate_context = None | ||||
def __exit__(self, *args): | ||||
if self._dirstate_context is not None: | ||||
self._dirstate_context.__exit__(*args) | ||||
if self._transaction is not None: | ||||
self._transaction.__exit__(*args) | ||||
Bryan O'Sullivan
|
r5034 | def commit(ui, repo, commitfunc, pats, opts): | ||
'''commit the specified files or all outstanding changes''' | ||||
Augie Fackler
|
r43347 | date = opts.get(b'date') | ||
Thomas Arendsen Hein
|
r6139 | if date: | ||
Augie Fackler
|
r43347 | opts[b'date'] = dateutil.parsedate(date) | ||
Bryan O'Sullivan
|
r5034 | |||
r50924 | with repo.wlock(), repo.lock(): | |||
r50923 | message = logmessage(ui, opts) | |||
matcher = scmutil.match(repo[None], pats, opts) | ||||
r50924 | ||||
with _AddRemoveContext(repo) as c: | ||||
r50923 | # extract addremove carefully -- this function can be called from a | |||
# command that doesn't support addremove | ||||
if opts.get(b'addremove'): | ||||
relative = scmutil.anypats(pats, opts) | ||||
uipathfn = scmutil.getuipathfn( | ||||
repo, | ||||
legacyrelativevalue=relative, | ||||
Augie Fackler
|
r43346 | ) | ||
r50923 | r = scmutil.addremove( | |||
repo, | ||||
matcher, | ||||
b"", | ||||
uipathfn, | ||||
opts, | ||||
r50924 | open_tr=c.open_transaction, | |||
r50923 | ) | |||
m = _(b"failed to mark all new/missing files as added/removed") | ||||
if r != 0: | ||||
raise error.Abort(m) | ||||
r50924 | c.close_dirstate_context() | |||
r50923 | return commitfunc(ui, repo, message, matcher, opts) | |||
Matt Mackall
|
r8407 | |||
Augie Fackler
|
r43346 | |||
Hannes Oldenburg
|
r29819 | def samefile(f, ctx1, ctx2): | ||
if f in ctx1.manifest(): | ||||
a = ctx1.filectx(f) | ||||
if f in ctx2.manifest(): | ||||
b = ctx2.filectx(f) | ||||
Augie Fackler
|
r43346 | return not a.cmp(b) and a.flags() == b.flags() | ||
Hannes Oldenburg
|
r29819 | else: | ||
return False | ||||
else: | ||||
return f not in ctx2.manifest() | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50987 | def amend(ui, repo, old, extra, pats, opts: Dict[str, Any]): | ||
Gregory Szorc
|
r25930 | # avoid cycle context -> subrepo -> cmdutil | ||
Gregory Szorc
|
r28322 | from . import context | ||
Gregory Szorc
|
r25930 | |||
Matt Harbison
|
r23101 | # amend will reuse the existing user if not specified, but the obsolete | ||
# marker creation requires that the current user's name is specified. | ||||
Durham Goode
|
r24379 | if obsolete.isenabled(repo, obsolete.createmarkersopt): | ||
Augie Fackler
|
r43346 | ui.username() # raise exception if username not set | ||
Matt Harbison
|
r23101 | |||
Augie Fackler
|
r43347 | ui.note(_(b'amending changeset %s\n') % old) | ||
Idan Kamara
|
r16458 | base = old.p1() | ||
Augie Fackler
|
r43347 | with repo.wlock(), repo.lock(), repo.transaction(b'amend'): | ||
Jun Wu
|
r33438 | # Participating changesets: | ||
# | ||||
Saurabh Singh
|
r34087 | # wctx o - workingctx that contains changes from working copy | ||
# | to go into amending commit | ||||
Jun Wu
|
r33438 | # | | ||
# old o - changeset to amend | ||||
# | | ||||
Saurabh Singh
|
r34058 | # base o - first parent of the changeset to amend | ||
Saurabh Singh
|
r34087 | wctx = repo[None] | ||
Jun Wu
|
r33438 | |||
Martin von Zweigbergk
|
r35197 | # Copy to avoid mutating input | ||
extra = extra.copy() | ||||
Jun Wu
|
r33438 | # Update extra dict from amended commit (e.g. to preserve graft | ||
# source) | ||||
extra.update(old.extra()) | ||||
Saurabh Singh
|
r34087 | # Also update it from the from the wctx | ||
extra.update(wctx.extra()) | ||||
Matt Harbison
|
r43202 | # date-only change should be ignored? | ||
Martin von Zweigbergk
|
r48225 | datemaydiffer = resolve_commit_options(ui, opts) | ||
opts = pycompat.byteskwargs(opts) | ||||
Matt Harbison
|
r43202 | |||
date = old.date() | ||||
Augie Fackler
|
r43347 | if opts.get(b'date'): | ||
date = dateutil.parsedate(opts.get(b'date')) | ||||
user = opts.get(b'user') or old.user() | ||||
Boris Feld
|
r34123 | |||
Jun Wu
|
r33438 | if len(old.parents()) > 1: | ||
# ctx.files() isn't reliable for merges, so fall back to the | ||||
# slower repo.status() method | ||||
Augie Fackler
|
r44043 | st = base.status(old) | ||
files = set(st.modified) | set(st.added) | set(st.removed) | ||||
Jun Wu
|
r33438 | else: | ||
files = set(old.files()) | ||||
Saurabh Singh
|
r34087 | # add/remove the files to the working copy if the "addremove" option | ||
# was specified. | ||||
matcher = scmutil.match(wctx, pats, opts) | ||||
Martin von Zweigbergk
|
r41801 | relative = scmutil.anypats(pats, opts) | ||
Martin von Zweigbergk
|
r41834 | uipathfn = scmutil.getuipathfn(repo, legacyrelativevalue=relative) | ||
r50937 | if opts.get(b'addremove'): | |||
with repo.dirstate.changing_files(repo): | ||||
if scmutil.addremove(repo, matcher, b"", uipathfn, opts) != 0: | ||||
m = _( | ||||
b"failed to mark all new/missing files as added/removed" | ||||
) | ||||
raise error.Abort(m) | ||||
Saurabh Singh
|
r34087 | |||
Yuya Nishihara
|
r35019 | # Check subrepos. This depends on in-place wctx._status update in | ||
# subrepo.precommit(). To minimize the risk of this hack, we do | ||||
# nothing if .hgsub does not exist. | ||||
Augie Fackler
|
r43347 | if b'.hgsub' in wctx or b'.hgsub' in old: | ||
Yuya Nishihara
|
r36026 | subs, commitsubs, newsubstate = subrepoutil.precommit( | ||
Augie Fackler
|
r43346 | ui, wctx, wctx._status, matcher | ||
) | ||||
Yuya Nishihara
|
r35019 | # amend should abort if commitsubrepos is enabled | ||
assert not commitsubs | ||||
if subs: | ||||
Yuya Nishihara
|
r36026 | subrepoutil.writestate(repo, newsubstate) | ||
Yuya Nishihara
|
r35019 | |||
Augie Fackler
|
r45383 | ms = mergestatemod.mergestate.read(repo) | ||
Yuya Nishihara
|
r36927 | mergeutil.checkunresolved(ms) | ||
Augie Fackler
|
r44937 | filestoamend = {f for f in wctx.files() if matcher(f)} | ||
Saurabh Singh
|
r34087 | |||
Augie Fackler
|
r43346 | changes = len(filestoamend) > 0 | ||
Martin von Zweigbergk
|
r49833 | changeset_copies = ( | ||
repo.ui.config(b'experimental', b'copies.read-from') | ||||
!= b'filelog-only' | ||||
) | ||||
# If there are changes to amend or if copy information needs to be read | ||||
# from the changeset extras, we cannot take the fast path of using | ||||
# filectxs from the old commit. | ||||
if changes or changeset_copies: | ||||
Jun Wu
|
r33438 | # Recompute copies (avoid recording a -> b -> a) | ||
Kyle Lippincott
|
r50074 | copied = copies.pathcopies(base, wctx) | ||
if old.p2(): | ||||
copied.update(copies.pathcopies(old.p2(), wctx)) | ||||
Jun Wu
|
r33438 | |||
# Prune files which were reverted by the updates: if old | ||||
Saurabh Singh
|
r34087 | # introduced file X and the file was renamed in the working | ||
# copy, then those two files are the same and | ||||
Jun Wu
|
r33438 | # we can discard X from our list of files. Likewise if X | ||
Yuya Nishihara
|
r35016 | # was removed, it's no longer relevant. If X is missing (aka | ||
# deleted), old X must be preserved. | ||||
Saurabh Singh
|
r34087 | files.update(filestoamend) | ||
Augie Fackler
|
r43346 | files = [ | ||
f | ||||
for f in files | ||||
if (f not in filestoamend or not samefile(f, wctx, base)) | ||||
] | ||||
Jun Wu
|
r33438 | |||
def filectxfn(repo, ctx_, path): | ||||
try: | ||||
Saurabh Singh
|
r34087 | # If the file being considered is not amongst the files | ||
Martin von Zweigbergk
|
r49833 | # to be amended, we should use the file context from the | ||
Saurabh Singh
|
r34087 | # old changeset. This avoids issues when only some files in | ||
# the working copy are being amended but there are also | ||||
# changes to other files from the old changeset. | ||||
Martin von Zweigbergk
|
r49833 | if path in filestoamend: | ||
Kyle Lippincott
|
r50035 | # Return None for removed files. | ||
if path in wctx.removed(): | ||||
return None | ||||
Martin von Zweigbergk
|
r49833 | fctx = wctx[path] | ||
else: | ||||
fctx = old.filectx(path) | ||||
Jun Wu
|
r33438 | flags = fctx.flags() | ||
Augie Fackler
|
r43346 | mctx = context.memfilectx( | ||
repo, | ||||
ctx_, | ||||
fctx.path(), | ||||
fctx.data(), | ||||
Augie Fackler
|
r43347 | islink=b'l' in flags, | ||
isexec=b'x' in flags, | ||||
Augie Fackler
|
r43346 | copysource=copied.get(path), | ||
) | ||||
Jun Wu
|
r33438 | return mctx | ||
except KeyError: | ||||
return None | ||||
Augie Fackler
|
r43346 | |||
Jun Wu
|
r33438 | else: | ||
Augie Fackler
|
r43347 | ui.note(_(b'copying changeset %s to %s\n') % (old, base)) | ||
Jun Wu
|
r33438 | |||
# Use version of files as in the old cset | ||||
def filectxfn(repo, ctx_, path): | ||||
try: | ||||
return old.filectx(path) | ||||
except KeyError: | ||||
return None | ||||
Saurabh Singh
|
r34087 | # See if we got a message from -m or -l, if not, open the editor with | ||
# the message of the changeset to amend. | ||||
message = logmessage(ui, opts) | ||||
Augie Fackler
|
r43347 | editform = mergeeditform(old, b'commit.amend') | ||
Saurabh Singh
|
r34087 | |||
Jun Wu
|
r33438 | if not message: | ||
message = old.description() | ||||
Kyle Lippincott
|
r42591 | # Default if message isn't provided and --edit is not passed is to | ||
# invoke editor, but allow --no-edit. If somehow we don't have any | ||||
# description, let's always start the editor. | ||||
Augie Fackler
|
r43347 | doedit = not message or opts.get(b'edit') in [True, None] | ||
Kyle Lippincott
|
r42591 | else: | ||
# Default if message is provided is to not invoke editor, but allow | ||||
# --edit. | ||||
Augie Fackler
|
r43347 | doedit = opts.get(b'edit') is True | ||
Kyle Lippincott
|
r42591 | editor = getcommiteditor(edit=doedit, editform=editform) | ||
Jun Wu
|
r33438 | |||
pureextra = extra.copy() | ||||
Augie Fackler
|
r43347 | extra[b'amend_source'] = old.hex() | ||
Jun Wu
|
r33438 | |||
Augie Fackler
|
r43346 | new = context.memctx( | ||
repo, | ||||
parents=[base.node(), old.p2().node()], | ||||
text=message, | ||||
files=files, | ||||
filectxfn=filectxfn, | ||||
user=user, | ||||
date=date, | ||||
extra=extra, | ||||
editor=editor, | ||||
) | ||||
Jun Wu
|
r33438 | |||
newdesc = changelog.stripdesc(new.description()) | ||||
Augie Fackler
|
r43346 | if ( | ||
(not changes) | ||||
Jun Wu
|
r33438 | and newdesc == old.description() | ||
and user == old.user() | ||||
Yuya Nishihara
|
r41159 | and (date == old.date() or datemaydiffer) | ||
Augie Fackler
|
r43346 | and pureextra == old.extra() | ||
): | ||||
Jun Wu
|
r33438 | # nothing changed. continuing here would create a new node | ||
# anyway because of the amend_source noise. | ||||
Pierre-Yves David
|
r17472 | # | ||
Jun Wu
|
r33438 | # This not what we expect from amend. | ||
Yuya Nishihara
|
r41159 | return old.node() | ||
Jun Wu
|
r33438 | |||
Martin von Zweigbergk
|
r38442 | commitphase = None | ||
Augie Fackler
|
r43347 | if opts.get(b'secret'): | ||
Martin von Zweigbergk
|
r38442 | commitphase = phases.secret | ||
Martin von Zweigbergk
|
r50666 | elif opts.get(b'draft'): | ||
commitphase = phases.draft | ||||
Martin von Zweigbergk
|
r38442 | newid = repo.commitctx(new) | ||
Martin von Zweigbergk
|
r45948 | ms.reset() | ||
Saurabh Singh
|
r34057 | |||
r50855 | with repo.dirstate.changing_parents(repo): | |||
r48408 | # Reroute the working copy parent to the new changeset | |||
repo.setparents(newid, repo.nullid) | ||||
# Fixing the dirstate because localrepo.commitctx does not update | ||||
# it. This is rather convenient because we did not need to update | ||||
# the dirstate for all the files in the new commit which commitctx | ||||
# could have done if it updated the dirstate. Now, we can | ||||
# selectively update the dirstate only for the amended files. | ||||
dirstate = repo.dirstate | ||||
# Update the state of the files which were added and modified in the | ||||
# amend to "normal" in the dirstate. We need to use "normallookup" since | ||||
# the files may have changed since the command started; using "normal" | ||||
# would mark them as clean but with uncommitted contents. | ||||
normalfiles = set(wctx.modified() + wctx.added()) & filestoamend | ||||
for f in normalfiles: | ||||
r48528 | dirstate.update_file( | |||
f, p1_tracked=True, wc_tracked=True, possibly_dirty=True | ||||
) | ||||
r48408 | ||||
# Update the state of files which were removed in the amend | ||||
# to "removed" in the dirstate. | ||||
removedfiles = set(wctx.removed()) & filestoamend | ||||
for f in removedfiles: | ||||
r48546 | dirstate.update_file(f, p1_tracked=False, wc_tracked=False) | |||
Saurabh Singh
|
r34087 | |||
Martin von Zweigbergk
|
r47499 | mapping = {old.node(): (newid,)} | ||
obsmetadata = None | ||||
if opts.get(b'note'): | ||||
obsmetadata = {b'note': encoding.fromlocal(opts[b'note'])} | ||||
backup = ui.configbool(b'rewrite', b'backup-bundle') | ||||
scmutil.cleanupnodes( | ||||
repo, | ||||
mapping, | ||||
b'amend', | ||||
metadata=obsmetadata, | ||||
fixphase=True, | ||||
targetphase=commitphase, | ||||
backup=backup, | ||||
) | ||||
Idan Kamara
|
r16458 | return newid | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | def commiteditor(repo, ctx, subs, editform=b''): | ||
Matt Mackall
|
r8407 | if ctx.description(): | ||
return ctx.description() | ||||
Augie Fackler
|
r43346 | return commitforceeditor( | ||
repo, ctx, subs, editform=editform, unchangedmessagedetection=True | ||||
) | ||||
def commitforceeditor( | ||||
repo, | ||||
ctx, | ||||
subs, | ||||
finishdesc=None, | ||||
extramsg=None, | ||||
Augie Fackler
|
r43347 | editform=b'', | ||
Augie Fackler
|
r43346 | unchangedmessagedetection=False, | ||
): | ||||
Matt Mackall
|
r21923 | if not extramsg: | ||
Augie Fackler
|
r43347 | extramsg = _(b"Leave message empty to abort commit.") | ||
forms = [e for e in editform.split(b'.') if e] | ||||
forms.insert(0, b'changeset') | ||||
Tony Tung
|
r26742 | templatetext = None | ||
FUJIWARA Katsunori
|
r22012 | while forms: | ||
Augie Fackler
|
r43347 | ref = b'.'.join(forms) | ||
if repo.ui.config(b'committemplate', ref): | ||||
Tony Tung
|
r26742 | templatetext = committext = buildcommittemplate( | ||
Augie Fackler
|
r43346 | repo, ctx, subs, extramsg, ref | ||
) | ||||
FUJIWARA Katsunori
|
r22012 | break | ||
forms.pop() | ||||
FUJIWARA Katsunori
|
r21924 | else: | ||
committext = buildcommittext(repo, ctx, subs, extramsg) | ||||
FUJIWARA Katsunori
|
r21869 | |||
# run editor in the repository root | ||||
Matt Harbison
|
r39843 | olddir = encoding.getcwd() | ||
FUJIWARA Katsunori
|
r21869 | os.chdir(repo.root) | ||
FUJIWARA Katsunori
|
r26750 | |||
# make in-memory changes visible to external process | ||||
tr = repo.currenttransaction() | ||||
repo.dirstate.write(tr) | ||||
pending = tr and tr.writepending() and repo.root | ||||
Augie Fackler
|
r43346 | editortext = repo.ui.edit( | ||
committext, | ||||
ctx.user(), | ||||
ctx.extra(), | ||||
editform=editform, | ||||
pending=pending, | ||||
repopath=repo.path, | ||||
Augie Fackler
|
r43347 | action=b'commit', | ||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r30724 | text = editortext | ||
Sean Farley
|
r30703 | |||
# strip away anything below this special string (used for editors that want | ||||
# to display the diff) | ||||
Yuya Nishihara
|
r30724 | stripbelow = re.search(_linebelow, text, flags=re.MULTILINE) | ||
Sean Farley
|
r30703 | if stripbelow: | ||
Augie Fackler
|
r43346 | text = text[: stripbelow.start()] | ||
Yuya Nishihara
|
r30724 | |||
Augie Fackler
|
r43347 | text = re.sub(b"(?m)^HG:.*(\n|$)", b"", text) | ||
FUJIWARA Katsunori
|
r21869 | os.chdir(olddir) | ||
if finishdesc: | ||||
text = finishdesc(text) | ||||
if not text.strip(): | ||||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"empty commit message")) | ||
Tony Tung
|
r26742 | if unchangedmessagedetection and editortext == templatetext: | ||
Martin von Zweigbergk
|
r46431 | raise error.InputError(_(b"commit message unchanged")) | ||
FUJIWARA Katsunori
|
r21869 | |||
return text | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r32878 | def buildcommittemplate(repo, ctx, subs, extramsg, ref): | ||
FUJIWARA Katsunori
|
r21924 | ui = repo.ui | ||
Martin von Zweigbergk
|
r45824 | spec = formatter.reference_templatespec(ref) | ||
Yuya Nishihara
|
r35972 | t = logcmdutil.changesettemplater(ui, repo, spec) | ||
Augie Fackler
|
r43346 | t.t.cache.update( | ||
(k, templater.unquotestring(v)) | ||||
Augie Fackler
|
r43347 | for k, v in repo.ui.configitems(b'committemplate') | ||
Augie Fackler
|
r43346 | ) | ||
FUJIWARA Katsunori
|
r22013 | |||
FUJIWARA Katsunori
|
r21924 | if not extramsg: | ||
Augie Fackler
|
r43347 | extramsg = b'' # ensure that extramsg is string | ||
FUJIWARA Katsunori
|
r21924 | |||
ui.pushbuffer() | ||||
t.show(ctx, extramsg=extramsg) | ||||
return ui.popbuffer() | ||||
Augie Fackler
|
r43346 | |||
timeless@mozdev.org
|
r26426 | def hgprefix(msg): | ||
Augie Fackler
|
r43347 | return b"\n".join([b"HG: %s" % a for a in msg.split(b"\n") if a]) | ||
timeless@mozdev.org
|
r26426 | |||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r21869 | def buildcommittext(repo, ctx, subs, extramsg): | ||
Matt Mackall
|
r8407 | edittext = [] | ||
Matt Mackall
|
r8707 | modified, added, removed = ctx.modified(), ctx.added(), ctx.removed() | ||
Matt Mackall
|
r8407 | if ctx.description(): | ||
edittext.append(ctx.description()) | ||||
Augie Fackler
|
r43347 | edittext.append(b"") | ||
edittext.append(b"") # Empty line between message and comments. | ||||
Augie Fackler
|
r43346 | edittext.append( | ||
hgprefix( | ||||
_( | ||||
Augie Fackler
|
r43347 | b"Enter commit message." | ||
b" Lines beginning with 'HG:' are removed." | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
) | ||||
timeless@mozdev.org
|
r26426 | edittext.append(hgprefix(extramsg)) | ||
Augie Fackler
|
r43347 | edittext.append(b"HG: --") | ||
edittext.append(hgprefix(_(b"user: %s") % ctx.user())) | ||||
Matt Mackall
|
r8407 | if ctx.p2(): | ||
Augie Fackler
|
r43347 | edittext.append(hgprefix(_(b"branch merge"))) | ||
Matt Mackall
|
r8407 | if ctx.branch(): | ||
Augie Fackler
|
r43347 | edittext.append(hgprefix(_(b"branch '%s'") % ctx.branch())) | ||
Ryan McElroy
|
r24986 | if bookmarks.isactivewdirparent(repo): | ||
Augie Fackler
|
r43347 | edittext.append(hgprefix(_(b"bookmark '%s'") % repo._activebookmark)) | ||
edittext.extend([hgprefix(_(b"subrepo %s") % s) for s in subs]) | ||||
edittext.extend([hgprefix(_(b"added %s") % f) for f in added]) | ||||
edittext.extend([hgprefix(_(b"changed %s") % f) for f in modified]) | ||||
edittext.extend([hgprefix(_(b"removed %s") % f) for f in removed]) | ||||
Matt Mackall
|
r8707 | if not added and not modified and not removed: | ||
Augie Fackler
|
r43347 | edittext.append(hgprefix(_(b"no files changed"))) | ||
edittext.append(b"") | ||||
return b"\n".join(edittext) | ||||
Adrian Buehlmann
|
r14297 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r51717 | def commitstatus(repo, node, branch, bheads=None, tip=None, **opts): | ||
Kevin Bullock
|
r18688 | ctx = repo[node] | ||
parents = ctx.parents() | ||||
Dan Villiom Podlaski Christiansen
|
r46414 | if tip is not None and repo.changelog.tip() == tip: | ||
# avoid reporting something like "committed new head" when | ||||
# recommitting old changesets, and issue a helpful warning | ||||
# for most instances | ||||
r47204 | repo.ui.warn(_(b"warning: commit already existed in the repository!\n")) | |||
Dan Villiom Podlaski Christiansen
|
r46414 | elif ( | ||
Matt Harbison
|
r51717 | not opts.get('amend') | ||
Augie Fackler
|
r43346 | and bheads | ||
and node not in bheads | ||||
Manuel Jacob
|
r45581 | and not any( | ||
Manuel Jacob
|
r45582 | p.node() in bheads and p.branch() == branch for p in parents | ||
Manuel Jacob
|
r45581 | ) | ||
Augie Fackler
|
r43346 | ): | ||
Augie Fackler
|
r43347 | repo.ui.status(_(b'created new head\n')) | ||
Kevin Bullock
|
r18688 | # The message is not printed for initial roots. For the other | ||
# changesets, it is printed in the following situations: | ||||
# | ||||
# Par column: for the 2 parents with ... | ||||
# N: null or no parent | ||||
# B: parent is on another named branch | ||||
# C: parent is a regular non head changeset | ||||
# H: parent was a branch head of the current branch | ||||
# Msg column: whether we print "created new head" message | ||||
# In the following, it is assumed that there already exists some | ||||
# initial branch heads of the current branch, otherwise nothing is | ||||
# printed anyway. | ||||
# | ||||
# Par Msg Comment | ||||
# N N y additional topo root | ||||
# | ||||
# B N y additional branch root | ||||
# C N y additional topo head | ||||
# H N n usual case | ||||
# | ||||
# B B y weird additional branch root | ||||
# C B y branch merge | ||||
# H B n merge with named branch | ||||
# | ||||
# C C y additional head from merge | ||||
# C H n merge with a head | ||||
# | ||||
# H H n head merge: head count decreases | ||||
Matt Harbison
|
r51717 | if not opts.get('close_branch'): | ||
Kevin Bullock
|
r18688 | for r in parents: | ||
if r.closesbranch() and r.branch() == branch: | ||||
Augie Fackler
|
r43347 | repo.ui.status( | ||
_(b'reopening closed branch head %d\n') % r.rev() | ||||
) | ||||
Kevin Bullock
|
r18688 | |||
if repo.ui.debugflag: | ||||
Augie Fackler
|
r43347 | repo.ui.write( | ||
_(b'committed changeset %d:%s\n') % (ctx.rev(), ctx.hex()) | ||||
) | ||||
Kevin Bullock
|
r18688 | elif repo.ui.verbose: | ||
Augie Fackler
|
r43347 | repo.ui.write(_(b'committed changeset %d:%s\n') % (ctx.rev(), ctx)) | ||
Kevin Bullock
|
r18688 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r27943 | def postcommitstatus(repo, pats, opts): | ||
return repo.status(match=scmutil.match(repo[None], pats, opts)) | ||||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r45935 | def revert(ui, repo, ctx, *pats, **opts): | ||
Pulkit Goyal
|
r35148 | opts = pycompat.byteskwargs(opts) | ||
Martin von Zweigbergk
|
r45935 | parent, p2 = repo.dirstate.parents() | ||
Angel Ezquerra
|
r16304 | node = ctx.node() | ||
mf = ctx.manifest() | ||||
Pierre-Yves David
|
r21579 | if node == p2: | ||
parent = p2 | ||||
Angel Ezquerra
|
r16304 | |||
# need all matching names in dirstate and manifest of target rev, | ||||
# so have to walk both. do not print errors if files exist in one | ||||
Yuya Nishihara
|
r24451 | # but not other. in both cases, filesets should be evaluated against | ||
# workingctx to get consistent result (issue4497). this means 'set:**' | ||||
# cannot be used to select missing files from target rev. | ||||
Angel Ezquerra
|
r16304 | |||
Pierre-Yves David
|
r21575 | # `names` is a mapping for all elements in working copy and target revision | ||
# The mapping is in the form: | ||||
Sushil khanchi
|
r39323 | # <abs path in repo> -> (<path from CWD>, <exactly specified by matcher?>) | ||
Angel Ezquerra
|
r16304 | names = {} | ||
Martin von Zweigbergk
|
r41754 | uipathfn = scmutil.getuipathfn(repo, legacyrelativevalue=True) | ||
Angel Ezquerra
|
r16304 | |||
r50933 | with repo.wlock(), repo.dirstate.changing_files(repo): | |||
Pierre-Yves David
|
r21575 | ## filling of the `names` mapping | ||
# walk dirstate to fill `names` | ||||
Angel Ezquerra
|
r16304 | |||
Augie Fackler
|
r43347 | interactive = opts.get(b'interactive', False) | ||
Martin von Zweigbergk
|
r24449 | wctx = repo[None] | ||
m = scmutil.match(wctx, pats, opts) | ||||
Matt Mackall
|
r24479 | |||
# we'll need this later | ||||
targetsubs = sorted(s for s in wctx.substate if m(s)) | ||||
Martin von Zweigbergk
|
r24450 | if not m.always(): | ||
Augie Fackler
|
r32362 | matcher = matchmod.badmatch(m, lambda x, y: False) | ||
Martin von Zweigbergk
|
r32382 | for abs in wctx.walk(matcher): | ||
Martin von Zweigbergk
|
r41754 | names[abs] = m.exact(abs) | ||
Durham Goode
|
r22573 | |||
# walk target manifest to fill `names` | ||||
def badfn(path, msg): | ||||
if path in names: | ||||
return | ||||
if path in ctx.substate: | ||||
Angel Ezquerra
|
r16304 | return | ||
Augie Fackler
|
r43347 | path_ = path + b'/' | ||
Durham Goode
|
r22573 | for f in names: | ||
if f.startswith(path_): | ||||
return | ||||
Augie Fackler
|
r43347 | ui.warn(b"%s: %s\n" % (uipathfn(path), msg)) | ||
Durham Goode
|
r22573 | |||
Matt Harbison
|
r25439 | for abs in ctx.walk(matchmod.badmatch(m, badfn)): | ||
Durham Goode
|
r22573 | if abs not in names: | ||
Martin von Zweigbergk
|
r41754 | names[abs] = m.exact(abs) | ||
Durham Goode
|
r22573 | |||
# Find status of all file in `names`. | ||||
m = scmutil.matchfiles(repo, names) | ||||
Augie Fackler
|
r43346 | changes = repo.status( | ||
node1=node, match=m, unknown=True, ignored=True, clean=True | ||||
) | ||||
Durham Goode
|
r22573 | else: | ||
Martin von Zweigbergk
|
r24450 | changes = repo.status(node1=node, match=m) | ||
Durham Goode
|
r22573 | for kind in changes: | ||
for abs in kind: | ||||
Martin von Zweigbergk
|
r41754 | names[abs] = m.exact(abs) | ||
Durham Goode
|
r22573 | |||
m = scmutil.matchfiles(repo, names) | ||||
Martin von Zweigbergk
|
r23374 | modified = set(changes.modified) | ||
Augie Fackler
|
r43346 | added = set(changes.added) | ||
removed = set(changes.removed) | ||||
Martin von Zweigbergk
|
r23374 | _deleted = set(changes.deleted) | ||
Augie Fackler
|
r43346 | unknown = set(changes.unknown) | ||
Martin von Zweigbergk
|
r23374 | unknown.update(changes.ignored) | ||
Augie Fackler
|
r43346 | clean = set(changes.clean) | ||
Pierre-Yves David
|
r22610 | modadded = set() | ||
Pierre-Yves David
|
r22185 | |||
Mads Kiilerich
|
r24180 | # We need to account for the state of the file in the dirstate, | ||
# even when we revert against something else than parent. This will | ||||
Pierre-Yves David
|
r22155 | # slightly alter the behavior of revert (doing back up or not, delete | ||
Mads Kiilerich
|
r23139 | # or just forget etc). | ||
Pierre-Yves David
|
r22155 | if parent == node: | ||
dsmodified = modified | ||||
dsadded = added | ||||
dsremoved = removed | ||||
Pierre-Yves David
|
r23403 | # store all local modifications, useful later for rename detection | ||
localchanges = dsmodified | dsadded | ||||
Pierre-Yves David
|
r22155 | modified, added, removed = set(), set(), set() | ||
else: | ||||
changes = repo.status(node1=parent, match=m) | ||||
Martin von Zweigbergk
|
r23374 | dsmodified = set(changes.modified) | ||
Augie Fackler
|
r43346 | dsadded = set(changes.added) | ||
dsremoved = set(changes.removed) | ||||
Pierre-Yves David
|
r23403 | # store all local modifications, useful later for rename detection | ||
localchanges = dsmodified | dsadded | ||||
Angel Ezquerra
|
r16304 | |||
Pierre-Yves David
|
r22188 | # only take into account for removes between wc and target | ||
clean |= dsremoved - removed | ||||
dsremoved &= removed | ||||
# distinct between dirstate remove and other | ||||
removed -= dsremoved | ||||
Pierre-Yves David
|
r22610 | modadded = added & dsmodified | ||
added -= modadded | ||||
Pierre-Yves David
|
r22190 | # tell newly modified apart. | ||
dsmodified &= modified | ||||
Augie Fackler
|
r43346 | dsmodified |= modified & dsadded # dirstate added may need backup | ||
Pierre-Yves David
|
r22190 | modified -= dsmodified | ||
Pierre-Yves David
|
r22488 | # We need to wait for some post-processing to update this set | ||
# before making the distinction. The dirstate will be used for | ||||
# that purpose. | ||||
Pierre-Yves David
|
r22208 | dsadded = added | ||
Pierre-Yves David
|
r22209 | # in case of merge, files that are actually added can be reported as | ||
# modified, we need to post process the result | ||||
Joerg Sonnenberger
|
r47771 | if p2 != repo.nullid: | ||
Durham Goode
|
r31134 | mergeadd = set(dsmodified) | ||
for path in dsmodified: | ||||
if path in mf: | ||||
mergeadd.remove(path) | ||||
Pierre-Yves David
|
r22209 | dsadded |= mergeadd | ||
dsmodified -= mergeadd | ||||
Angel Ezquerra
|
r16304 | |||
Pierre-Yves David
|
r21575 | # if f is a rename, update `names` to also revert the source | ||
Pierre-Yves David
|
r23403 | for f in localchanges: | ||
Angel Ezquerra
|
r16304 | src = repo.dirstate.copied(f) | ||
Pierre-Yves David
|
r22213 | # XXX should we check for rename down to target node? | ||
r48916 | if ( | |||
src | ||||
and src not in names | ||||
and repo.dirstate.get_entry(src).removed | ||||
): | ||||
Pierre-Yves David
|
r22154 | dsremoved.add(src) | ||
Martin von Zweigbergk
|
r41754 | names[src] = True | ||
Angel Ezquerra
|
r16304 | |||
Martin von Zweigbergk
|
r31157 | # determine the exact nature of the deleted changesets | ||
deladded = set(_deleted) | ||||
for path in _deleted: | ||||
if path in mf: | ||||
deladded.remove(path) | ||||
deleted = _deleted - deladded | ||||
Pierre-Yves David
|
r22488 | # distinguish between file to forget and the other | ||
added = set() | ||||
for abs in dsadded: | ||||
r48916 | if not repo.dirstate.get_entry(abs).added: | |||
Pierre-Yves David
|
r22488 | added.add(abs) | ||
dsadded -= added | ||||
Pierre-Yves David
|
r22490 | for abs in deladded: | ||
r48916 | if repo.dirstate.get_entry(abs).added: | |||
Pierre-Yves David
|
r22490 | dsadded.add(abs) | ||
deladded -= dsadded | ||||
Pierre-Yves David
|
r22396 | # For files marked as removed, we check if an unknown file is present at | ||
# the same path. If a such file exists it may need to be backed up. | ||||
# Making the distinction at this stage helps have simpler backup | ||||
# logic. | ||||
removunk = set() | ||||
for abs in removed: | ||||
target = repo.wjoin(abs) | ||||
if os.path.lexists(target): | ||||
removunk.add(abs) | ||||
removed -= removunk | ||||
dsremovunk = set() | ||||
for abs in dsremoved: | ||||
target = repo.wjoin(abs) | ||||
if os.path.lexists(target): | ||||
dsremovunk.add(abs) | ||||
dsremoved -= dsremovunk | ||||
Angel Ezquerra
|
r16304 | |||
Pierre-Yves David
|
r21575 | # action to be actually performed by revert | ||
# (<list of file>, message>) tuple | ||||
Augie Fackler
|
r43346 | actions = { | ||
Augie Fackler
|
r43347 | b'revert': ([], _(b'reverting %s\n')), | ||
b'add': ([], _(b'adding %s\n')), | ||||
b'remove': ([], _(b'removing %s\n')), | ||||
b'drop': ([], _(b'removing %s\n')), | ||||
b'forget': ([], _(b'forgetting %s\n')), | ||||
b'undelete': ([], _(b'undeleting %s\n')), | ||||
b'noop': (None, _(b'no changes needed to %s\n')), | ||||
b'unknown': (None, _(b'file not managed: %s\n')), | ||||
Augie Fackler
|
r43346 | } | ||
Angel Ezquerra
|
r16304 | |||
Pierre-Yves David
|
r22608 | # "constant" that convey the backup strategy. | ||
# All set to `discard` if `no-backup` is set do avoid checking | ||||
# no_backup lower in the code. | ||||
Pierre-Yves David
|
r22609 | # These values are ordered for comparison purposes | ||
Augie Fackler
|
r43346 | backupinteractive = 3 # do backup if interactively modified | ||
Pierre-Yves David
|
r22608 | backup = 2 # unconditionally do backup | ||
Augie Fackler
|
r43346 | check = 1 # check if the existing file differs from target | ||
discard = 0 # never do backup | ||||
Augie Fackler
|
r43347 | if opts.get(b'no_backup'): | ||
skarlage
|
r29498 | backupinteractive = backup = check = discard | ||
if interactive: | ||||
dsmodifiedbackup = backupinteractive | ||||
else: | ||||
dsmodifiedbackup = backup | ||||
tobackup = set() | ||||
Angel Ezquerra
|
r16304 | |||
Augie Fackler
|
r43347 | backupanddel = actions[b'remove'] | ||
if not opts.get(b'no_backup'): | ||||
backupanddel = actions[b'drop'] | ||||
Pierre-Yves David
|
r22611 | |||
Angel Ezquerra
|
r16304 | disptable = ( | ||
# dispatch table: | ||||
# file state | ||||
Pierre-Yves David
|
r22153 | # action | ||
# make backup | ||||
Pierre-Yves David
|
r22371 | ## Sets that results that will change file on disk | ||
# Modified compared to target, no local change | ||||
Augie Fackler
|
r43347 | (modified, actions[b'revert'], discard), | ||
Pierre-Yves David
|
r22397 | # Modified compared to target, but local file is deleted | ||
Augie Fackler
|
r43347 | (deleted, actions[b'revert'], discard), | ||
Pierre-Yves David
|
r22371 | # Modified compared to target, local change | ||
Augie Fackler
|
r43347 | (dsmodified, actions[b'revert'], dsmodifiedbackup), | ||
Pierre-Yves David
|
r22371 | # Added since target | ||
Augie Fackler
|
r43347 | (added, actions[b'remove'], discard), | ||
Pierre-Yves David
|
r22488 | # Added in working directory | ||
Augie Fackler
|
r43347 | (dsadded, actions[b'forget'], discard), | ||
Pierre-Yves David
|
r22610 | # Added since target, have local modification | ||
Augie Fackler
|
r43346 | (modadded, backupanddel, backup), | ||
Pierre-Yves David
|
r22490 | # Added since target but file is missing in working directory | ||
Augie Fackler
|
r43347 | (deladded, actions[b'drop'], discard), | ||
Pierre-Yves David
|
r22371 | # Removed since target, before working copy parent | ||
Augie Fackler
|
r43347 | (removed, actions[b'add'], discard), | ||
Pierre-Yves David
|
r22396 | # Same as `removed` but an unknown file exists at the same path | ||
Augie Fackler
|
r43347 | (removunk, actions[b'add'], check), | ||
Pierre-Yves David
|
r22371 | # Removed since targe, marked as such in working copy parent | ||
Augie Fackler
|
r43347 | (dsremoved, actions[b'undelete'], discard), | ||
Pierre-Yves David
|
r22396 | # Same as `dsremoved` but an unknown file exists at the same path | ||
Augie Fackler
|
r43347 | (dsremovunk, actions[b'undelete'], check), | ||
Pierre-Yves David
|
r22371 | ## the following sets does not result in any file changes | ||
# File with no modification | ||||
Augie Fackler
|
r43347 | (clean, actions[b'noop'], discard), | ||
Pierre-Yves David
|
r22371 | # Existing file, not tracked anywhere | ||
Augie Fackler
|
r43347 | (unknown, actions[b'unknown'], discard), | ||
Augie Fackler
|
r43346 | ) | ||
Angel Ezquerra
|
r16304 | |||
Martin von Zweigbergk
|
r41754 | for abs, exact in sorted(names.items()): | ||
Pierre-Yves David
|
r21575 | # target file to be touch on disk (relative to cwd) | ||
Angel Ezquerra
|
r16304 | target = repo.wjoin(abs) | ||
Pierre-Yves David
|
r21575 | # search the entry in the dispatch table. | ||
Pierre-Yves David
|
r22212 | # if the file is in any of these sets, it was touched in the working | ||
Pierre-Yves David
|
r21575 | # directory parent and we are sure it needs to be reverted. | ||
Pierre-Yves David
|
r22232 | for table, (xlist, msg), dobackup in disptable: | ||
Angel Ezquerra
|
r16304 | if abs not in table: | ||
continue | ||||
Pierre-Yves David
|
r22233 | if xlist is not None: | ||
xlist.append(abs) | ||||
skarlage
|
r29498 | if dobackup: | ||
# If in interactive mode, don't automatically create | ||||
# .orig files (issue4793) | ||||
if dobackup == backupinteractive: | ||||
tobackup.add(abs) | ||||
Augie Fackler
|
r43346 | elif backup <= dobackup or wctx[abs].cmp(ctx[abs]): | ||
Martin von Zweigbergk
|
r41754 | absbakname = scmutil.backuppath(ui, repo, abs) | ||
Augie Fackler
|
r43346 | bakname = os.path.relpath( | ||
absbakname, start=repo.root | ||||
) | ||||
ui.note( | ||||
Augie Fackler
|
r43347 | _(b'saving current version of %s as %s\n') | ||
Augie Fackler
|
r43346 | % (uipathfn(abs), uipathfn(bakname)) | ||
) | ||||
Augie Fackler
|
r43347 | if not opts.get(b'dry_run'): | ||
Laurent Charignon
|
r24475 | if interactive: | ||
Martin von Zweigbergk
|
r41754 | util.copyfile(target, absbakname) | ||
Laurent Charignon
|
r24475 | else: | ||
Martin von Zweigbergk
|
r41754 | util.rename(target, absbakname) | ||
Augie Fackler
|
r43347 | if opts.get(b'dry_run'): | ||
Sushil khanchi
|
r39442 | if ui.verbose or not exact: | ||
Martin von Zweigbergk
|
r41754 | ui.status(msg % uipathfn(abs)) | ||
Pierre-Yves David
|
r22233 | elif exact: | ||
Martin von Zweigbergk
|
r41754 | ui.warn(msg % uipathfn(abs)) | ||
Angel Ezquerra
|
r16304 | break | ||
Pierre-Yves David
|
r21575 | |||
Augie Fackler
|
r43347 | if not opts.get(b'dry_run'): | ||
needdata = (b'revert', b'add', b'undelete') | ||||
Matt Harbison
|
r35941 | oplist = [actions[name][0] for name in needdata] | ||
Matt Harbison
|
r37780 | prefetch = scmutil.prefetchfiles | ||
Rodrigo Damazio Bovendorp
|
r45632 | matchfiles = scmutil.matchfiles( | ||
repo, [f for sublist in oplist for f in sublist] | ||||
) | ||||
Augie Fackler
|
r43346 | prefetch( | ||
Augie Fackler
|
r46554 | repo, | ||
[(ctx.rev(), matchfiles)], | ||||
Augie Fackler
|
r43346 | ) | ||
Denis Laxalde
|
r42238 | match = scmutil.match(repo[None], pats) | ||
Augie Fackler
|
r43346 | _performrevert( | ||
repo, | ||||
ctx, | ||||
names, | ||||
uipathfn, | ||||
actions, | ||||
match, | ||||
interactive, | ||||
tobackup, | ||||
) | ||||
Bryan O'Sullivan
|
r19129 | |||
Matt Harbison
|
r24134 | if targetsubs: | ||
# Revert the subrepos on the revert list | ||||
for sub in targetsubs: | ||||
Matt Harbison
|
r24463 | try: | ||
Augie Fackler
|
r43346 | wctx.sub(sub).revert( | ||
ctx.substate[sub], *pats, **pycompat.strkwargs(opts) | ||||
) | ||||
Matt Harbison
|
r24463 | except KeyError: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | b"subrepository '%s' does not exist in %s!" | ||
Augie Fackler
|
r43346 | % (sub, short(ctx.node())) | ||
) | ||||
def _performrevert( | ||||
repo, | ||||
ctx, | ||||
names, | ||||
uipathfn, | ||||
actions, | ||||
match, | ||||
interactive=False, | ||||
tobackup=None, | ||||
): | ||||
Pierre-Yves David
|
r21576 | """function that actually perform all the actions computed for revert | ||
Pierre-Yves David
|
r20571 | |||
This is an independent function to let extension to plug in and react to | ||||
the imminent revert. | ||||
Mads Kiilerich
|
r21024 | Make sure you have the working directory locked when calling this function. | ||
Pierre-Yves David
|
r20571 | """ | ||
Martin von Zweigbergk
|
r45935 | parent, p2 = repo.dirstate.parents() | ||
Pierre-Yves David
|
r20571 | node = ctx.node() | ||
liscju
|
r27985 | excluded_files = [] | ||
Pierre-Yves David
|
r20571 | def checkout(f): | ||
fc = ctx[f] | ||||
FUJIWARA Katsunori
|
r25755 | repo.wwrite(f, fc.data(), fc.flags()) | ||
Pierre-Yves David
|
r20571 | |||
Denis Laxalde
|
r30532 | def doremove(f): | ||
try: | ||||
Augie Fackler
|
r43347 | rmdir = repo.ui.configbool(b'experimental', b'removeemptydirs') | ||
Kyle Lippincott
|
r38512 | repo.wvfs.unlinkpath(f, rmdir=rmdir) | ||
Denis Laxalde
|
r30532 | except OSError: | ||
pass | ||||
r48402 | repo.dirstate.set_untracked(f) | |||
Denis Laxalde
|
r30532 | |||
Sushil khanchi
|
r39442 | def prntstatusmsg(action, f): | ||
Martin von Zweigbergk
|
r41754 | exact = names[f] | ||
Sushil khanchi
|
r39442 | if repo.ui.verbose or not exact: | ||
Martin von Zweigbergk
|
r41754 | repo.ui.status(actions[action][1] % uipathfn(f)) | ||
Sushil khanchi
|
r39442 | |||
Yuya Nishihara
|
r33722 | audit_path = pathutil.pathauditor(repo.root, cached=True) | ||
Augie Fackler
|
r43347 | for f in actions[b'forget'][0]: | ||
liscju
|
r27985 | if interactive: | ||
Denis Laxalde
|
r30530 | choice = repo.ui.promptchoice( | ||
Augie Fackler
|
r43347 | _(b"forget added file %s (Yn)?$$ &Yes $$ &No") % uipathfn(f) | ||
Augie Fackler
|
r43346 | ) | ||
liscju
|
r27985 | if choice == 0: | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'forget', f) | ||
r48403 | repo.dirstate.set_untracked(f) | |||
liscju
|
r27985 | else: | ||
Denis Laxalde
|
r36212 | excluded_files.append(f) | ||
liscju
|
r27985 | else: | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'forget', f) | ||
r48403 | repo.dirstate.set_untracked(f) | |||
Augie Fackler
|
r43347 | for f in actions[b'remove'][0]: | ||
Pierre-Yves David
|
r20571 | audit_path(f) | ||
Denis Laxalde
|
r30532 | if interactive: | ||
choice = repo.ui.promptchoice( | ||||
Augie Fackler
|
r43347 | _(b"remove added file %s (Yn)?$$ &Yes $$ &No") % uipathfn(f) | ||
Augie Fackler
|
r43346 | ) | ||
Denis Laxalde
|
r30532 | if choice == 0: | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'remove', f) | ||
Denis Laxalde
|
r30532 | doremove(f) | ||
else: | ||||
Denis Laxalde
|
r36212 | excluded_files.append(f) | ||
Denis Laxalde
|
r30532 | else: | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'remove', f) | ||
Denis Laxalde
|
r30532 | doremove(f) | ||
Augie Fackler
|
r43347 | for f in actions[b'drop'][0]: | ||
Pierre-Yves David
|
r22491 | audit_path(f) | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'drop', f) | ||
r48402 | repo.dirstate.set_untracked(f) | |||
Pierre-Yves David
|
r20571 | |||
r49207 | # We are reverting to our parent. If possible, we had like `hg status` | |||
# to report the file as clean. We have to be less agressive for | ||||
# merges to avoid losing information about copy introduced by the merge. | ||||
# This might comes with bugs ? | ||||
reset_copy = p2 == repo.nullid | ||||
def normal(filename): | ||||
return repo.dirstate.set_tracked(filename, reset_copy=reset_copy) | ||||
Laurent Charignon
|
r24359 | |||
Laurent Charignon
|
r25259 | newlyaddedandmodifiedfiles = set() | ||
Laurent Charignon
|
r24359 | if interactive: | ||
# Prompt the user for changes to revert | ||||
Augie Fackler
|
r43347 | torevert = [f for f in actions[b'revert'][0] if f not in excluded_files] | ||
Denis Laxalde
|
r36212 | m = scmutil.matchfiles(repo, torevert) | ||
Augie Fackler
|
r43346 | diffopts = patch.difffeatureopts( | ||
repo.ui, | ||||
whitespace=True, | ||||
Augie Fackler
|
r43347 | section=b'commands', | ||
configprefix=b'revert.interactive.', | ||||
Augie Fackler
|
r43346 | ) | ||
Laurent Charignon
|
r25258 | diffopts.nodates = True | ||
diffopts.git = True | ||||
Augie Fackler
|
r43347 | operation = b'apply' | ||
Martin von Zweigbergk
|
r42154 | if node == parent: | ||
Augie Fackler
|
r43346 | if repo.ui.configbool( | ||
Augie Fackler
|
r43347 | b'experimental', b'revert.interactive.select-to-keep' | ||
Augie Fackler
|
r43346 | ): | ||
Augie Fackler
|
r43347 | operation = b'keep' | ||
Martin von Zweigbergk
|
r42154 | else: | ||
Augie Fackler
|
r43347 | operation = b'discard' | ||
if operation == b'apply': | ||||
Martin von Zweigbergk
|
r42154 | diff = patch.diff(repo, None, ctx.node(), m, opts=diffopts) | ||
else: | ||||
Laurent Charignon
|
r25424 | diff = patch.diff(repo, ctx.node(), None, m, opts=diffopts) | ||
Daniel Ploch
|
r48365 | original_headers = patch.parsepatch(diff) | ||
Laurent Charignon
|
r25424 | |||
Laurent Charignon
|
r24359 | try: | ||
Augie Fackler
|
r43346 | chunks, opts = recordfilter( | ||
Daniel Ploch
|
r48365 | repo.ui, original_headers, match, operation=operation | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | if operation == b'discard': | ||
Laurent Charignon
|
r25424 | chunks = patch.reversehunks(chunks) | ||
Martin von Zweigbergk
|
r49186 | except error.PatchParseError as err: | ||
raise error.InputError(_(b'error parsing patch: %s') % err) | ||||
except error.PatchApplicationError as err: | ||||
raise error.StateError(_(b'error applying patch: %s') % err) | ||||
Laurent Charignon
|
r24359 | |||
Kyle Lippincott
|
r43122 | # FIXME: when doing an interactive revert of a copy, there's no way of | ||
# performing a partial revert of the added file, the only option is | ||||
# "remove added file <name> (Yn)?", so we don't need to worry about the | ||||
# alsorestore value. Ideally we'd be able to partially revert | ||||
# copied/renamed files. | ||||
Daniel Ploch
|
r48365 | newlyaddedandmodifiedfiles, unusedalsorestore = newandmodified(chunks) | ||
skarlage
|
r29498 | if tobackup is None: | ||
tobackup = set() | ||||
Laurent Charignon
|
r24359 | # Apply changes | ||
timeless
|
r28861 | fp = stringio() | ||
Yuya Nishihara
|
r39451 | # chunks are serialized per file, but files aren't sorted | ||
Augie Fackler
|
r44937 | for f in sorted({c.header.filename() for c in chunks if ishunk(c)}): | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'revert', f) | ||
Martin von Zweigbergk
|
r42154 | files = set() | ||
Laurent Charignon
|
r24359 | for c in chunks: | ||
Sushil khanchi
|
r39442 | if ishunk(c): | ||
skarlage
|
r29498 | abs = c.header.filename() | ||
Sushil khanchi
|
r39442 | # Create a backup file only if this hunk should be backed up | ||
if c.header.filename() in tobackup: | ||||
target = repo.wjoin(abs) | ||||
Martin von Zweigbergk
|
r41737 | bakname = scmutil.backuppath(repo.ui, repo, abs) | ||
Sushil khanchi
|
r39442 | util.copyfile(target, bakname) | ||
tobackup.remove(abs) | ||||
Martin von Zweigbergk
|
r42154 | if abs not in files: | ||
files.add(abs) | ||||
Augie Fackler
|
r43347 | if operation == b'keep': | ||
Martin von Zweigbergk
|
r42154 | checkout(abs) | ||
Laurent Charignon
|
r24359 | c.write(fp) | ||
dopatch = fp.tell() | ||||
fp.seek(0) | ||||
if dopatch: | ||||
try: | ||||
patch.internalpatch(repo.ui, repo, fp, 1, eolmode=None) | ||||
Martin von Zweigbergk
|
r49186 | except error.PatchParseError as err: | ||
raise error.InputError(pycompat.bytestr(err)) | ||||
except error.PatchApplicationError as err: | ||||
raise error.StateError(pycompat.bytestr(err)) | ||||
Laurent Charignon
|
r24359 | del fp | ||
else: | ||||
Augie Fackler
|
r43347 | for f in actions[b'revert'][0]: | ||
prntstatusmsg(b'revert', f) | ||||
FUJIWARA Katsunori
|
r25755 | checkout(f) | ||
Laurent Charignon
|
r24359 | if normal: | ||
normal(f) | ||||
Pierre-Yves David
|
r20571 | |||
Augie Fackler
|
r43347 | for f in actions[b'add'][0]: | ||
Laurent Charignon
|
r25259 | # Don't checkout modified files, they are already created by the diff | ||
Martin von Zweigbergk
|
r49895 | if f in newlyaddedandmodifiedfiles: | ||
continue | ||||
Martin von Zweigbergk
|
r49896 | if interactive: | ||
choice = repo.ui.promptchoice( | ||||
_(b"add new file %s (Yn)?$$ &Yes $$ &No") % uipathfn(f) | ||||
) | ||||
if choice != 0: | ||||
continue | ||||
Martin von Zweigbergk
|
r49895 | prntstatusmsg(b'add', f) | ||
checkout(f) | ||||
repo.dirstate.set_tracked(f) | ||||
Pierre-Yves David
|
r20571 | |||
Augie Fackler
|
r43347 | for f in actions[b'undelete'][0]: | ||
Taapas Agrawal
|
r41663 | if interactive: | ||
choice = repo.ui.promptchoice( | ||||
Augie Fackler
|
r43347 | _(b"add back removed file %s (Yn)?$$ &Yes $$ &No") % f | ||
Augie Fackler
|
r43346 | ) | ||
Taapas Agrawal
|
r41663 | if choice == 0: | ||
Augie Fackler
|
r43347 | prntstatusmsg(b'undelete', f) | ||
Taapas Agrawal
|
r41663 | checkout(f) | ||
normal(f) | ||||
else: | ||||
excluded_files.append(f) | ||||
else: | ||||
Augie Fackler
|
r43347 | prntstatusmsg(b'undelete', f) | ||
Taapas Agrawal
|
r41663 | checkout(f) | ||
normal(f) | ||||
Pierre-Yves David
|
r20571 | |||
copied = copies.pathcopies(repo[parent], ctx) | ||||
Augie Fackler
|
r43347 | for f in ( | ||
actions[b'add'][0] + actions[b'undelete'][0] + actions[b'revert'][0] | ||||
): | ||||
Pierre-Yves David
|
r20571 | if f in copied: | ||
repo.dirstate.copy(copied[f], f) | ||||
Augie Fackler
|
r43346 | |||
FUJIWARA Katsunori
|
r21051 | # a list of (ui, repo, otherpeer, opts, missing) functions called by | ||
# commands.outgoing. "missing" is "missing" of the result of | ||||
# "findcommonoutgoing()" | ||||
outgoinghooks = util.hooks() | ||||
Bryan O'Sullivan
|
r19211 | # a list of (ui, repo) functions called by commands.summary | ||
summaryhooks = util.hooks() | ||||
Matt Mackall
|
r19474 | |||
FUJIWARA Katsunori
|
r21047 | # a list of (ui, repo, opts, changes) functions called by commands.summary. | ||
# | ||||
# functions should return tuple of booleans below, if 'changes' is None: | ||||
# (whether-incomings-are-needed, whether-outgoings-are-needed) | ||||
# | ||||
# otherwise, 'changes' is a tuple of tuples below: | ||||
# - (sourceurl, sourcebranch, sourcepeer, incoming) | ||||
# - (desturl, destbranch, destpeer, outgoing) | ||||
summaryremotehooks = util.hooks() | ||||
Taapas Agrawal
|
r42730 | |||
Taapas Agrawal
|
r42732 | def checkunfinished(repo, commit=False, skipmerge=False): | ||
Augie Fackler
|
r46554 | """Look for an unfinished multistep operation, like graft, and abort | ||
Matt Mackall
|
r19474 | if found. It's probably good to check this right before | ||
bailifchanged(). | ||||
Augie Fackler
|
r46554 | """ | ||
Matt Harbison
|
r38220 | # Check for non-clearable states first, so things like rebase will take | ||
# precedence over update. | ||||
Taapas Agrawal
|
r42730 | for state in statemod._unfinishedstates: | ||
Augie Fackler
|
r43346 | if ( | ||
state._clearable | ||||
or (commit and state._allowcommit) | ||||
or state._reportonly | ||||
): | ||||
Matt Harbison
|
r38220 | continue | ||
Taapas Agrawal
|
r42730 | if state.isunfinished(repo): | ||
Martin von Zweigbergk
|
r46444 | raise error.StateError(state.msg(), hint=state.hint()) | ||
Taapas Agrawal
|
r42730 | |||
for s in statemod._unfinishedstates: | ||||
Augie Fackler
|
r43346 | if ( | ||
not s._clearable | ||||
or (commit and s._allowcommit) | ||||
Augie Fackler
|
r43347 | or (s._opname == b'merge' and skipmerge) | ||
Augie Fackler
|
r43346 | or s._reportonly | ||
): | ||||
Matt Mackall
|
r19496 | continue | ||
Taapas Agrawal
|
r42730 | if s.isunfinished(repo): | ||
Martin von Zweigbergk
|
r46444 | raise error.StateError(s.msg(), hint=s.hint()) | ||
Matt Mackall
|
r19474 | |||
Augie Fackler
|
r43346 | |||
Matt Mackall
|
r19474 | def clearunfinished(repo): | ||
Augie Fackler
|
r46554 | """Check for unfinished operations (as above), and clear the ones | ||
Matt Mackall
|
r19474 | that are clearable. | ||
Augie Fackler
|
r46554 | """ | ||
Taapas Agrawal
|
r42730 | for state in statemod._unfinishedstates: | ||
Taapas Agrawal
|
r42732 | if state._reportonly: | ||
continue | ||||
Taapas Agrawal
|
r42730 | if not state._clearable and state.isunfinished(repo): | ||
Martin von Zweigbergk
|
r46444 | raise error.StateError(state.msg(), hint=state.hint()) | ||
Taapas Agrawal
|
r42730 | |||
for s in statemod._unfinishedstates: | ||||
Matt Harbison
|
r47731 | if s._opname == b'merge' or s._reportonly: | ||
Taapas Agrawal
|
r42732 | continue | ||
Taapas Agrawal
|
r42730 | if s._clearable and s.isunfinished(repo): | ||
util.unlink(repo.vfs.join(s._fname)) | ||||
FUJIWARA Katsunori
|
r24991 | |||
Augie Fackler
|
r43346 | |||
Taapas Agrawal
|
r42780 | def getunfinishedstate(repo): | ||
Augie Fackler
|
r46554 | """Checks for unfinished operations and returns statecheck object | ||
for it""" | ||||
Taapas Agrawal
|
r42780 | for state in statemod._unfinishedstates: | ||
if state.isunfinished(repo): | ||||
return state | ||||
return None | ||||
Augie Fackler
|
r43346 | |||
timeless
|
r28120 | def howtocontinue(repo): | ||
Augie Fackler
|
r46554 | """Check for an unfinished operation and return the command to finish | ||
timeless
|
r28120 | it. | ||
Taapas Agrawal
|
r42777 | statemod._unfinishedstates list is checked for an unfinished operation | ||
and the corresponding message to finish it is generated if a method to | ||||
continue is supported by the operation. | ||||
timeless
|
r28120 | |||
Returns a (msg, warning) tuple. 'msg' is a string and 'warning' is | ||||
a boolean. | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43347 | contmsg = _(b"continue: %s") | ||
Taapas Agrawal
|
r42733 | for state in statemod._unfinishedstates: | ||
if not state._continueflag: | ||||
continue | ||||
if state.isunfinished(repo): | ||||
return contmsg % state.continuemsg(), True | ||||
Matt Harbison
|
r33362 | if repo[None].dirty(missing=True, merge=False, branch=False): | ||
Augie Fackler
|
r43347 | return contmsg % _(b"hg commit"), False | ||
timeless
|
r28120 | return None, None | ||
Augie Fackler
|
r43346 | |||
timeless
|
r28120 | def checkafterresolved(repo): | ||
Augie Fackler
|
r46554 | """Inform the user about the next action after completing hg resolve | ||
timeless
|
r28120 | |||
Taapas Agrawal
|
r42777 | If there's a an unfinished operation that supports continue flag, | ||
howtocontinue will yield repo.ui.warn as the reporter. | ||||
timeless
|
r28120 | |||
Otherwise, it will yield repo.ui.note. | ||||
Augie Fackler
|
r46554 | """ | ||
timeless
|
r28120 | msg, warning = howtocontinue(repo) | ||
if msg is not None: | ||||
if warning: | ||||
Augie Fackler
|
r43347 | repo.ui.warn(b"%s\n" % msg) | ||
timeless
|
r28120 | else: | ||
Augie Fackler
|
r43347 | repo.ui.note(b"%s\n" % msg) | ||
timeless
|
r28120 | |||
Augie Fackler
|
r43346 | |||
timeless
|
r28120 | def wrongtooltocontinue(repo, task): | ||
Augie Fackler
|
r46554 | """Raise an abort suggesting how to properly continue if there is an | ||
timeless
|
r28120 | active task. | ||
Uses howtocontinue() to find the active task. | ||||
If there's no task (repo.ui.note for 'hg commit'), it does not offer | ||||
a hint. | ||||
Augie Fackler
|
r46554 | """ | ||
timeless
|
r28120 | after = howtocontinue(repo) | ||
hint = None | ||||
if after[1]: | ||||
hint = after[0] | ||||
Martin von Zweigbergk
|
r46444 | raise error.StateError(_(b'no %s in progress') % task, hint=hint) | ||
Taapas Agrawal
|
r42768 | |||
Augie Fackler
|
r43346 | |||
Taapas Agrawal
|
r42768 | def abortgraft(ui, repo, graftstate): | ||
"""abort the interrupted graft and rollbacks to the state before interrupted | ||||
graft""" | ||||
if not graftstate.exists(): | ||||
Martin von Zweigbergk
|
r46444 | raise error.StateError(_(b"no interrupted graft to abort")) | ||
Taapas Agrawal
|
r42768 | statedata = readgraftstate(repo, graftstate) | ||
Augie Fackler
|
r43347 | newnodes = statedata.get(b'newnodes') | ||
Taapas Agrawal
|
r42768 | if newnodes is None: | ||
# and old graft state which does not have all the data required to abort | ||||
# the graft | ||||
Augie Fackler
|
r43347 | raise error.Abort(_(b"cannot abort using an old graftstate")) | ||
Taapas Agrawal
|
r42768 | |||
# changeset from which graft operation was started | ||||
if len(newnodes) > 0: | ||||
startctx = repo[newnodes[0]].p1() | ||||
else: | ||||
Augie Fackler
|
r43347 | startctx = repo[b'.'] | ||
Taapas Agrawal
|
r42768 | # whether to strip or not | ||
cleanup = False | ||||
Augie Fackler
|
r43346 | |||
Taapas Agrawal
|
r42768 | if newnodes: | ||
newnodes = [repo[r].rev() for r in newnodes] | ||||
cleanup = True | ||||
# checking that none of the newnodes turned public or is public | ||||
immutable = [c for c in newnodes if not repo[c].mutable()] | ||||
if immutable: | ||||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
Augie Fackler
|
r43347 | _(b"cannot clean up public changesets %s\n") | ||
% b', '.join(bytes(repo[r]) for r in immutable), | ||||
hint=_(b"see 'hg help phases' for details"), | ||||
Augie Fackler
|
r43346 | ) | ||
Taapas Agrawal
|
r42768 | cleanup = False | ||
# checking that no new nodes are created on top of grafted revs | ||||
desc = set(repo.changelog.descendants(newnodes)) | ||||
if desc - set(newnodes): | ||||
Augie Fackler
|
r43346 | repo.ui.warn( | ||
_( | ||||
Augie Fackler
|
r43347 | b"new changesets detected on destination " | ||
b"branch, can't strip\n" | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Taapas Agrawal
|
r42768 | cleanup = False | ||
if cleanup: | ||||
with repo.wlock(), repo.lock(): | ||||
Martin von Zweigbergk
|
r46133 | mergemod.clean_update(startctx) | ||
Taapas Agrawal
|
r42768 | # stripping the new nodes created | ||
Augie Fackler
|
r43346 | strippoints = [ | ||
Augie Fackler
|
r43347 | c.node() for c in repo.set(b"roots(%ld)", newnodes) | ||
Augie Fackler
|
r43346 | ] | ||
Taapas Agrawal
|
r42768 | repair.strip(repo.ui, repo, strippoints, backup=False) | ||
if not cleanup: | ||||
# we don't update to the startnode if we can't strip | ||||
Augie Fackler
|
r43347 | startctx = repo[b'.'] | ||
Martin von Zweigbergk
|
r46133 | mergemod.clean_update(startctx) | ||
Taapas Agrawal
|
r42768 | |||
Augie Fackler
|
r43347 | ui.status(_(b"graft aborted\n")) | ||
ui.status(_(b"working directory is now at %s\n") % startctx.hex()[:12]) | ||||
Taapas Agrawal
|
r42768 | graftstate.delete() | ||
return 0 | ||||
Augie Fackler
|
r43346 | |||
r52180 | def readgraftstate( | |||
repo: Any, | ||||
graftstate: statemod.cmdstate, | ||||
) -> Dict[bytes, Any]: | ||||
Taapas Agrawal
|
r42768 | """read the graft state file and return a dict of the data stored in it""" | ||
try: | ||||
return graftstate.read() | ||||
except error.CorruptedState: | ||||
Augie Fackler
|
r43347 | nodes = repo.vfs.read(b'graftstate').splitlines() | ||
return {b'nodes': nodes} | ||||
Taapas Agrawal
|
r42785 | |||
Augie Fackler
|
r43346 | |||
Taapas Agrawal
|
r42785 | def hgabortgraft(ui, repo): | ||
Kyle Lippincott
|
r47856 | """abort logic for aborting graft using 'hg abort'""" | ||
Taapas Agrawal
|
r42785 | with repo.wlock(): | ||
Augie Fackler
|
r43347 | graftstate = statemod.cmdstate(repo, b'graftstate') | ||
Taapas Agrawal
|
r42785 | return abortgraft(ui, repo, graftstate) | ||
r52447 | ||||
def postincoming(ui, repo, modheads, optupdate, checkout, brev): | ||||
"""Run after a changegroup has been added via pull/unbundle | ||||
This takes arguments below: | ||||
:modheads: change of heads by pull/unbundle | ||||
:optupdate: updating working directory is needed or not | ||||
:checkout: update destination revision (or None to default destination) | ||||
:brev: a name, which might be a bookmark to be activated after updating | ||||
return True if update raise any conflict, False otherwise. | ||||
""" | ||||
if modheads == 0: | ||||
return False | ||||
if optupdate: | ||||
# avoid circular import | ||||
from . import hg | ||||
try: | ||||
return hg.updatetotally(ui, repo, checkout, brev) | ||||
except error.UpdateAbort as inst: | ||||
msg = _(b"not updating: %s") % stringutil.forcebytestr(inst) | ||||
hint = inst.hint | ||||
raise error.UpdateAbort(msg, hint=hint) | ||||
if ui.quiet: | ||||
pass # we won't report anything so the other clause are useless. | ||||
elif modheads is not None and modheads > 1: | ||||
currentbranchheads = len(repo.branchheads()) | ||||
if currentbranchheads == modheads: | ||||
ui.status( | ||||
_(b"(run 'hg heads' to see heads, 'hg merge' to merge)\n") | ||||
) | ||||
elif currentbranchheads > 1: | ||||
ui.status( | ||||
_(b"(run 'hg heads .' to see heads, 'hg merge' to merge)\n") | ||||
) | ||||
else: | ||||
ui.status(_(b"(run 'hg heads' to see heads)\n")) | ||||
elif not ui.configbool(b'commands', b'update.requiredest'): | ||||
ui.status(_(b"(run 'hg update' to get a working copy)\n")) | ||||
return False | ||||
r52448 | ||||
def unbundle_files(ui, repo, fnames, unbundle_source=b'unbundle'): | ||||
"""utility for `hg unbundle` and `hg debug::unbundle`""" | ||||
assert fnames | ||||
# avoid circular import | ||||
from . import hg | ||||
with repo.lock(): | ||||
for fname in fnames: | ||||
f = hg.openpath(ui, fname) | ||||
gen = exchange.readbundle(ui, f, fname) | ||||
if isinstance(gen, streamclone.streamcloneapplier): | ||||
raise error.InputError( | ||||
_( | ||||
b'packed bundles cannot be applied with ' | ||||
b'"hg unbundle"' | ||||
), | ||||
hint=_(b'use "hg debugapplystreamclonebundle"'), | ||||
) | ||||
url = b'bundle:' + fname | ||||
try: | ||||
txnname = b'unbundle' | ||||
if not isinstance(gen, bundle2.unbundle20): | ||||
txnname = b'unbundle\n%s' % urlutil.hidepassword(url) | ||||
with repo.transaction(txnname) as tr: | ||||
op = bundle2.applybundle( | ||||
repo, | ||||
gen, | ||||
tr, | ||||
source=unbundle_source, # used by debug::unbundle | ||||
url=url, | ||||
) | ||||
except error.BundleUnknownFeatureError as exc: | ||||
raise error.Abort( | ||||
_(b'%s: unknown bundle feature, %s') % (fname, exc), | ||||
hint=_( | ||||
b"see https://mercurial-scm.org/" | ||||
b"wiki/BundleFeature for more " | ||||
b"information" | ||||
), | ||||
) | ||||
modheads = bundle2.combinechangegroupresults(op) | ||||
return modheads | ||||