pathspec.c: support adding prefix magic to a pathspec with mnemonic magic

Back in 233c3e6 (parse_pathspec: preserve prefix length via
PATHSPEC_PREFIX_ORIGIN - 2013-07-14), parse_pathspec() is taught to
save prefix length as a dynamic magic. This is needed when the
pathspec is passed to another process and and prefix lenght would be
lost.

Back then we support two cases. If the pathspec is normal, e.g. "abc",
we simply add the prefix to become ":(prefix:2)abc". If the pathspec
contains long magic, e.g. ":(foo,bar)abc" then we turn it to
":(foo,bar,prefix:2)abc". We do not support prefixing on short form,
because the only supported mnemonic '/' disappears after the the
preprocessing steps.

With the introduction of exclude magic with mnemonic '!', we need to
add support for the short form case so that ':!abc' becomes
':(exclude,prefix:2)abc'. Without this, it will break

    cd Documentation
    git add -p -- . ':!technical'

Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Nguyễn Thái Ngọc Duy 2013-12-06 14:30:49 +07:00 committed by Junio C Hamano
parent ef79b1f870
commit 1649612a22

View File

@ -74,6 +74,20 @@ static struct pathspec_magic {
{ PATHSPEC_EXCLUDE, '!', "exclude" }, { PATHSPEC_EXCLUDE, '!', "exclude" },
}; };
static void prefix_short_magic(struct strbuf *sb, int prefixlen,
unsigned short_magic)
{
int i;
strbuf_addstr(sb, ":(");
for (i = 0; i < ARRAY_SIZE(pathspec_magic); i++)
if (short_magic & pathspec_magic[i].bit) {
if (sb->buf[sb->len - 1] != '(')
strbuf_addch(sb, ',');
strbuf_addstr(sb, pathspec_magic[i].name);
}
strbuf_addf(sb, ",prefix:%d)", prefixlen);
}
/* /*
* Take an element of a pathspec and check for magic signatures. * Take an element of a pathspec and check for magic signatures.
* Append the result to the prefix. Return the magic bitmap. * Append the result to the prefix. Return the magic bitmap.
@ -233,22 +247,16 @@ static unsigned prefix_pathspec(struct pathspec_item *item,
*/ */
if (flags & PATHSPEC_PREFIX_ORIGIN) { if (flags & PATHSPEC_PREFIX_ORIGIN) {
struct strbuf sb = STRBUF_INIT; struct strbuf sb = STRBUF_INIT;
const char *start = elt;
if (prefixlen && !literal_global) { if (prefixlen && !literal_global) {
/* Preserve the actual prefix length of each pattern */ /* Preserve the actual prefix length of each pattern */
if (short_magic) if (short_magic)
die("BUG: prefixing on short magic is not supported"); prefix_short_magic(&sb, prefixlen, short_magic);
else if (long_magic_end) { else if (long_magic_end) {
strbuf_add(&sb, start, long_magic_end - start); strbuf_add(&sb, elt, long_magic_end - elt);
strbuf_addf(&sb, ",prefix:%d", prefixlen); strbuf_addf(&sb, ",prefix:%d)", prefixlen);
start = long_magic_end; } else
} else {
if (*start == ':')
start++;
strbuf_addf(&sb, ":(prefix:%d)", prefixlen); strbuf_addf(&sb, ":(prefix:%d)", prefixlen);
}
} }
strbuf_add(&sb, start, copyfrom - start);
strbuf_addstr(&sb, match); strbuf_addstr(&sb, match);
item->original = strbuf_detach(&sb, NULL); item->original = strbuf_detach(&sb, NULL);
} else } else