Skip to content

Commit b0db704

Browse files
bmwillgitster
authored andcommitted
pathspec: allow querying for attributes
The pathspec mechanism is extended via the new ":(attr:eol=input)pattern/to/match" syntax to filter paths so that it requires paths to not just match the given pattern but also have the specified attrs attached for them to be chosen. Based on a patch by Stefan Beller <[email protected]> Signed-off-by: Brandon Williams <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 625568c commit b0db704

File tree

7 files changed

+382
-9
lines changed

7 files changed

+382
-9
lines changed

Documentation/glossary-content.txt

Lines changed: 21 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -384,6 +384,27 @@ full pathname may have special meaning:
384384
+
385385
Glob magic is incompatible with literal magic.
386386

387+
attr;;
388+
After `attr:` comes a space separated list of "attribute
389+
requirements", all of which must be met in order for the
390+
path to be considered a match; this is in addition to the
391+
usual non-magic pathspec pattern matching.
392+
See linkgit:gitattributes[5].
393+
+
394+
Each of the attribute requirements for the path takes one of
395+
these forms:
396+
397+
- "`ATTR`" requires that the attribute `ATTR` be set.
398+
399+
- "`-ATTR`" requires that the attribute `ATTR` be unset.
400+
401+
- "`ATTR=VALUE`" requires that the attribute `ATTR` be
402+
set to the string `VALUE`.
403+
404+
- "`!ATTR`" requires that the attribute `ATTR` be
405+
unspecified.
406+
+
407+
387408
exclude;;
388409
After a path matches any non-exclude pathspec, it will be run
389410
through all exclude pathspec (magic signature: `!` or its

attr.c

Lines changed: 17 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -603,6 +603,23 @@ struct attr_check *attr_check_initl(const char *one, ...)
603603
return check;
604604
}
605605

606+
struct attr_check *attr_check_dup(const struct attr_check *check)
607+
{
608+
struct attr_check *ret;
609+
610+
if (!check)
611+
return NULL;
612+
613+
ret = attr_check_alloc();
614+
615+
ret->nr = check->nr;
616+
ret->alloc = check->alloc;
617+
ALLOC_ARRAY(ret->items, ret->nr);
618+
COPY_ARRAY(ret->items, check->items, ret->nr);
619+
620+
return ret;
621+
}
622+
606623
struct attr_check_item *attr_check_append(struct attr_check *check,
607624
const struct git_attr *attr)
608625
{

attr.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ struct attr_check {
4444

4545
extern struct attr_check *attr_check_alloc(void);
4646
extern struct attr_check *attr_check_initl(const char *, ...);
47+
extern struct attr_check *attr_check_dup(const struct attr_check *check);
4748

4849
extern struct attr_check_item *attr_check_append(struct attr_check *check,
4950
const struct git_attr *attr);

dir.c

Lines changed: 40 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
*/
1010
#include "cache.h"
1111
#include "dir.h"
12+
#include "attr.h"
1213
#include "refs.h"
1314
#include "wildmatch.h"
1415
#include "pathspec.h"
@@ -134,7 +135,8 @@ static size_t common_prefix_len(const struct pathspec *pathspec)
134135
PATHSPEC_LITERAL |
135136
PATHSPEC_GLOB |
136137
PATHSPEC_ICASE |
137-
PATHSPEC_EXCLUDE);
138+
PATHSPEC_EXCLUDE |
139+
PATHSPEC_ATTR);
138140

139141
for (n = 0; n < pathspec->nr; n++) {
140142
size_t i = 0, len = 0, item_len;
@@ -209,6 +211,36 @@ int within_depth(const char *name, int namelen,
209211
#define DO_MATCH_DIRECTORY (1<<1)
210212
#define DO_MATCH_SUBMODULE (1<<2)
211213

214+
static int match_attrs(const char *name, int namelen,
215+
const struct pathspec_item *item)
216+
{
217+
int i;
218+
219+
git_check_attr(name, item->attr_check);
220+
for (i = 0; i < item->attr_match_nr; i++) {
221+
const char *value;
222+
int matched;
223+
enum attr_match_mode match_mode;
224+
225+
value = item->attr_check->items[i].value;
226+
match_mode = item->attr_match[i].match_mode;
227+
228+
if (ATTR_TRUE(value))
229+
matched = (match_mode == MATCH_SET);
230+
else if (ATTR_FALSE(value))
231+
matched = (match_mode == MATCH_UNSET);
232+
else if (ATTR_UNSET(value))
233+
matched = (match_mode == MATCH_UNSPECIFIED);
234+
else
235+
matched = (match_mode == MATCH_VALUE &&
236+
!strcmp(item->attr_match[i].value, value));
237+
if (!matched)
238+
return 0;
239+
}
240+
241+
return 1;
242+
}
243+
212244
/*
213245
* Does 'match' match the given name?
214246
* A match is found if
@@ -261,6 +293,9 @@ static int match_pathspec_item(const struct pathspec_item *item, int prefix,
261293
strncmp(item->match, name - prefix, item->prefix))
262294
return 0;
263295

296+
if (item->attr_match_nr && !match_attrs(name, namelen, item))
297+
return 0;
298+
264299
/* If the match was just the prefix, we matched */
265300
if (!*match)
266301
return MATCHED_RECURSIVELY;
@@ -339,7 +374,8 @@ static int do_match_pathspec(const struct pathspec *ps,
339374
PATHSPEC_LITERAL |
340375
PATHSPEC_GLOB |
341376
PATHSPEC_ICASE |
342-
PATHSPEC_EXCLUDE);
377+
PATHSPEC_EXCLUDE |
378+
PATHSPEC_ATTR);
343379

344380
if (!ps->nr) {
345381
if (!ps->recursive ||
@@ -1361,7 +1397,8 @@ static int simplify_away(const char *path, int pathlen,
13611397
PATHSPEC_LITERAL |
13621398
PATHSPEC_GLOB |
13631399
PATHSPEC_ICASE |
1364-
PATHSPEC_EXCLUDE);
1400+
PATHSPEC_EXCLUDE |
1401+
PATHSPEC_ATTR);
13651402

13661403
for (i = 0; i < pathspec->nr; i++) {
13671404
const struct pathspec_item *item = &pathspec->items[i];

pathspec.c

Lines changed: 108 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
#include "cache.h"
22
#include "dir.h"
33
#include "pathspec.h"
4+
#include "attr.h"
45

56
/*
67
* Finds which of the given pathspecs match items in the index.
@@ -72,6 +73,7 @@ static struct pathspec_magic {
7273
{ PATHSPEC_GLOB, '\0', "glob" },
7374
{ PATHSPEC_ICASE, '\0', "icase" },
7475
{ PATHSPEC_EXCLUDE, '!', "exclude" },
76+
{ PATHSPEC_ATTR, '\0', "attr" },
7577
};
7678

7779
static void prefix_magic(struct strbuf *sb, int prefixlen, unsigned magic)
@@ -87,6 +89,72 @@ static void prefix_magic(struct strbuf *sb, int prefixlen, unsigned magic)
8789
strbuf_addf(sb, ",prefix:%d)", prefixlen);
8890
}
8991

92+
static void parse_pathspec_attr_match(struct pathspec_item *item, const char *value)
93+
{
94+
struct string_list_item *si;
95+
struct string_list list = STRING_LIST_INIT_DUP;
96+
97+
if (item->attr_check || item->attr_match)
98+
die(_("Only one 'attr:' specification is allowed."));
99+
100+
if (!value || !*value)
101+
die(_("attr spec must not be empty"));
102+
103+
string_list_split(&list, value, ' ', -1);
104+
string_list_remove_empty_items(&list, 0);
105+
106+
item->attr_check = attr_check_alloc();
107+
item->attr_match = xcalloc(list.nr, sizeof(struct attr_match));
108+
109+
for_each_string_list_item(si, &list) {
110+
size_t attr_len;
111+
char *attr_name;
112+
const struct git_attr *a;
113+
114+
int j = item->attr_match_nr++;
115+
const char *attr = si->string;
116+
struct attr_match *am = &item->attr_match[j];
117+
118+
switch (*attr) {
119+
case '!':
120+
am->match_mode = MATCH_UNSPECIFIED;
121+
attr++;
122+
attr_len = strlen(attr);
123+
break;
124+
case '-':
125+
am->match_mode = MATCH_UNSET;
126+
attr++;
127+
attr_len = strlen(attr);
128+
break;
129+
default:
130+
attr_len = strcspn(attr, "=");
131+
if (attr[attr_len] != '=')
132+
am->match_mode = MATCH_SET;
133+
else {
134+
am->match_mode = MATCH_VALUE;
135+
am->value = xstrdup(&attr[attr_len + 1]);
136+
if (strchr(am->value, '\\'))
137+
die(_("attr spec values must not contain backslashes"));
138+
}
139+
break;
140+
}
141+
142+
attr_name = xmemdupz(attr, attr_len);
143+
a = git_attr(attr_name);
144+
if (!a)
145+
die(_("invalid attribute name %s"), attr_name);
146+
147+
attr_check_append(item->attr_check, a);
148+
149+
free(attr_name);
150+
}
151+
152+
if (item->attr_check->nr != item->attr_match_nr)
153+
die("BUG: should have same number of entries");
154+
155+
string_list_clear(&list, 0);
156+
}
157+
90158
static inline int get_literal_global(void)
91159
{
92160
static int literal = -1;
@@ -164,6 +232,7 @@ static int get_global_magic(int element_magic)
164232
* returns the position in 'elem' after all magic has been parsed
165233
*/
166234
static const char *parse_long_magic(unsigned *magic, int *prefix_len,
235+
struct pathspec_item *item,
167236
const char *elem)
168237
{
169238
const char *pos;
@@ -189,6 +258,14 @@ static const char *parse_long_magic(unsigned *magic, int *prefix_len,
189258
continue;
190259
}
191260

261+
if (starts_with(pos, "attr:")) {
262+
char *attr_body = xmemdupz(pos + 5, len - 5);
263+
parse_pathspec_attr_match(item, attr_body);
264+
*magic |= PATHSPEC_ATTR;
265+
free(attr_body);
266+
continue;
267+
}
268+
192269
for (i = 0; i < ARRAY_SIZE(pathspec_magic); i++) {
193270
if (strlen(pathspec_magic[i].name) == len &&
194271
!strncmp(pathspec_magic[i].name, pos, len)) {
@@ -252,13 +329,14 @@ static const char *parse_short_magic(unsigned *magic, const char *elem)
252329
}
253330

254331
static const char *parse_element_magic(unsigned *magic, int *prefix_len,
332+
struct pathspec_item *item,
255333
const char *elem)
256334
{
257335
if (elem[0] != ':' || get_literal_global())
258336
return elem; /* nothing to do */
259337
else if (elem[1] == '(')
260338
/* longhand */
261-
return parse_long_magic(magic, prefix_len, elem);
339+
return parse_long_magic(magic, prefix_len, item, elem);
262340
else
263341
/* shorthand */
264342
return parse_short_magic(magic, elem);
@@ -335,12 +413,17 @@ static void init_pathspec_item(struct pathspec_item *item, unsigned flags,
335413
char *match;
336414
int pathspec_prefix = -1;
337415

416+
item->attr_check = NULL;
417+
item->attr_match = NULL;
418+
item->attr_match_nr = 0;
419+
338420
/* PATHSPEC_LITERAL_PATH ignores magic */
339421
if (flags & PATHSPEC_LITERAL_PATH) {
340422
magic = PATHSPEC_LITERAL;
341423
} else {
342424
copyfrom = parse_element_magic(&element_magic,
343425
&pathspec_prefix,
426+
item,
344427
elt);
345428
magic |= element_magic;
346429
magic |= get_global_magic(element_magic);
@@ -565,26 +648,46 @@ void parse_pathspec(struct pathspec *pathspec,
565648

566649
void copy_pathspec(struct pathspec *dst, const struct pathspec *src)
567650
{
568-
int i;
651+
int i, j;
569652

570653
*dst = *src;
571654
ALLOC_ARRAY(dst->items, dst->nr);
572655
COPY_ARRAY(dst->items, src->items, dst->nr);
573656

574657
for (i = 0; i < dst->nr; i++) {
575-
dst->items[i].match = xstrdup(src->items[i].match);
576-
dst->items[i].original = xstrdup(src->items[i].original);
658+
struct pathspec_item *d = &dst->items[i];
659+
struct pathspec_item *s = &src->items[i];
660+
661+
d->match = xstrdup(s->match);
662+
d->original = xstrdup(s->original);
663+
664+
ALLOC_ARRAY(d->attr_match, d->attr_match_nr);
665+
COPY_ARRAY(d->attr_match, s->attr_match, d->attr_match_nr);
666+
for (j = 0; j < d->attr_match_nr; j++) {
667+
const char *value = s->attr_match[j].value;
668+
d->attr_match[j].value = xstrdup_or_null(value);
669+
}
670+
671+
d->attr_check = attr_check_dup(s->attr_check);
577672
}
578673
}
579674

580675
void clear_pathspec(struct pathspec *pathspec)
581676
{
582-
int i;
677+
int i, j;
583678

584679
for (i = 0; i < pathspec->nr; i++) {
585680
free(pathspec->items[i].match);
586681
free(pathspec->items[i].original);
682+
683+
for (j = 0; j < pathspec->items[j].attr_match_nr; j++)
684+
free(pathspec->items[i].attr_match[j].value);
685+
free(pathspec->items[i].attr_match);
686+
687+
if (pathspec->items[i].attr_check)
688+
attr_check_free(pathspec->items[i].attr_check);
587689
}
690+
588691
free(pathspec->items);
589692
pathspec->items = NULL;
590693
pathspec->nr = 0;

pathspec.h

Lines changed: 14 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,15 @@
88
#define PATHSPEC_GLOB (1<<3)
99
#define PATHSPEC_ICASE (1<<4)
1010
#define PATHSPEC_EXCLUDE (1<<5)
11+
#define PATHSPEC_ATTR (1<<6)
1112
#define PATHSPEC_ALL_MAGIC \
1213
(PATHSPEC_FROMTOP | \
1314
PATHSPEC_MAXDEPTH | \
1415
PATHSPEC_LITERAL | \
1516
PATHSPEC_GLOB | \
1617
PATHSPEC_ICASE | \
17-
PATHSPEC_EXCLUDE)
18+
PATHSPEC_EXCLUDE | \
19+
PATHSPEC_ATTR)
1820

1921
#define PATHSPEC_ONESTAR 1 /* the pathspec pattern satisfies GFNM_ONESTAR */
2022

@@ -31,6 +33,17 @@ struct pathspec {
3133
int len, prefix;
3234
int nowildcard_len;
3335
int flags;
36+
int attr_match_nr;
37+
struct attr_match {
38+
char *value;
39+
enum attr_match_mode {
40+
MATCH_SET,
41+
MATCH_UNSET,
42+
MATCH_VALUE,
43+
MATCH_UNSPECIFIED
44+
} match_mode;
45+
} *attr_match;
46+
struct attr_check *attr_check;
3447
} *items;
3548
};
3649

0 commit comments

Comments
 (0)