Merge branch 'tb/grep-column'
"git grep" learned the "--column" option that gives not just the line number but the column number of the hit. * tb/grep-column: contrib/git-jump/git-jump: jump to exact location grep.c: add configuration variables to show matched option builtin/grep.c: add '--column' option to 'git-grep(1)' grep.c: display column number of first match grep.[ch]: extend grep_opt to allow showing matched column grep.c: expose {,inverted} match column in match_line() Documentation/config.txt: camel-case lineNumber for consistency
This commit is contained in:
commit
d036d667b7
@ -1182,8 +1182,10 @@ color.grep.<slot>::
|
|||||||
filename prefix (when not using `-h`)
|
filename prefix (when not using `-h`)
|
||||||
`function`;;
|
`function`;;
|
||||||
function name lines (when using `-p`)
|
function name lines (when using `-p`)
|
||||||
`linenumber`;;
|
`lineNumber`;;
|
||||||
line number prefix (when using `-n`)
|
line number prefix (when using `-n`)
|
||||||
|
`column`;;
|
||||||
|
column number prefix (when using `--column`)
|
||||||
`match`;;
|
`match`;;
|
||||||
matching text (same as setting `matchContext` and `matchSelected`)
|
matching text (same as setting `matchContext` and `matchSelected`)
|
||||||
`matchContext`;;
|
`matchContext`;;
|
||||||
@ -1798,6 +1800,9 @@ gitweb.snapshot::
|
|||||||
grep.lineNumber::
|
grep.lineNumber::
|
||||||
If set to true, enable `-n` option by default.
|
If set to true, enable `-n` option by default.
|
||||||
|
|
||||||
|
grep.column::
|
||||||
|
If set to true, enable the `--column` option by default.
|
||||||
|
|
||||||
grep.patternType::
|
grep.patternType::
|
||||||
Set the default matching behavior. Using a value of 'basic', 'extended',
|
Set the default matching behavior. Using a value of 'basic', 'extended',
|
||||||
'fixed', or 'perl' will enable the `--basic-regexp`, `--extended-regexp`,
|
'fixed', or 'perl' will enable the `--basic-regexp`, `--extended-regexp`,
|
||||||
|
@ -13,7 +13,7 @@ SYNOPSIS
|
|||||||
[-v | --invert-match] [-h|-H] [--full-name]
|
[-v | --invert-match] [-h|-H] [--full-name]
|
||||||
[-E | --extended-regexp] [-G | --basic-regexp]
|
[-E | --extended-regexp] [-G | --basic-regexp]
|
||||||
[-P | --perl-regexp]
|
[-P | --perl-regexp]
|
||||||
[-F | --fixed-strings] [-n | --line-number]
|
[-F | --fixed-strings] [-n | --line-number] [--column]
|
||||||
[-l | --files-with-matches] [-L | --files-without-match]
|
[-l | --files-with-matches] [-L | --files-without-match]
|
||||||
[(-O | --open-files-in-pager) [<pager>]]
|
[(-O | --open-files-in-pager) [<pager>]]
|
||||||
[-z | --null]
|
[-z | --null]
|
||||||
@ -44,6 +44,9 @@ CONFIGURATION
|
|||||||
grep.lineNumber::
|
grep.lineNumber::
|
||||||
If set to true, enable `-n` option by default.
|
If set to true, enable `-n` option by default.
|
||||||
|
|
||||||
|
grep.column::
|
||||||
|
If set to true, enable the `--column` option by default.
|
||||||
|
|
||||||
grep.patternType::
|
grep.patternType::
|
||||||
Set the default matching behavior. Using a value of 'basic', 'extended',
|
Set the default matching behavior. Using a value of 'basic', 'extended',
|
||||||
'fixed', or 'perl' will enable the `--basic-regexp`, `--extended-regexp`,
|
'fixed', or 'perl' will enable the `--basic-regexp`, `--extended-regexp`,
|
||||||
@ -169,6 +172,10 @@ providing this option will cause it to die.
|
|||||||
--line-number::
|
--line-number::
|
||||||
Prefix the line number to matching lines.
|
Prefix the line number to matching lines.
|
||||||
|
|
||||||
|
--column::
|
||||||
|
Prefix the 1-indexed byte-offset of the first match from the start of the
|
||||||
|
matching line.
|
||||||
|
|
||||||
-l::
|
-l::
|
||||||
--files-with-matches::
|
--files-with-matches::
|
||||||
--name-only::
|
--name-only::
|
||||||
|
@ -828,6 +828,7 @@ int cmd_grep(int argc, const char **argv, const char *prefix)
|
|||||||
GREP_PATTERN_TYPE_PCRE),
|
GREP_PATTERN_TYPE_PCRE),
|
||||||
OPT_GROUP(""),
|
OPT_GROUP(""),
|
||||||
OPT_BOOL('n', "line-number", &opt.linenum, N_("show line numbers")),
|
OPT_BOOL('n', "line-number", &opt.linenum, N_("show line numbers")),
|
||||||
|
OPT_BOOL(0, "column", &opt.columnnum, N_("show column number of first match")),
|
||||||
OPT_NEGBIT('h', NULL, &opt.pathname, N_("don't show filenames"), 1),
|
OPT_NEGBIT('h', NULL, &opt.pathname, N_("don't show filenames"), 1),
|
||||||
OPT_BIT('H', NULL, &opt.pathname, N_("show filenames"), 1),
|
OPT_BIT('H', NULL, &opt.pathname, N_("show filenames"), 1),
|
||||||
OPT_NEGBIT(0, "full-name", &opt.relative,
|
OPT_NEGBIT(0, "full-name", &opt.relative,
|
||||||
|
@ -25,6 +25,13 @@ git-jump will feed this to the editor:
|
|||||||
foo.c:2: printf("hello word!\n");
|
foo.c:2: printf("hello word!\n");
|
||||||
-----------------------------------
|
-----------------------------------
|
||||||
|
|
||||||
|
Or, when running 'git jump grep', column numbers will also be emitted,
|
||||||
|
e.g. `git jump grep "hello"` would return:
|
||||||
|
|
||||||
|
-----------------------------------
|
||||||
|
foo.c:2:9: printf("hello word!\n");
|
||||||
|
-----------------------------------
|
||||||
|
|
||||||
Obviously this trivial case isn't that interesting; you could just open
|
Obviously this trivial case isn't that interesting; you could just open
|
||||||
`foo.c` yourself. But when you have many changes scattered across a
|
`foo.c` yourself. But when you have many changes scattered across a
|
||||||
project, you can use the editor's support to "jump" from point to point.
|
project, you can use the editor's support to "jump" from point to point.
|
||||||
@ -35,7 +42,8 @@ Git-jump can generate four types of interesting lists:
|
|||||||
|
|
||||||
2. The beginning of any merge conflict markers.
|
2. The beginning of any merge conflict markers.
|
||||||
|
|
||||||
3. Any grep matches.
|
3. Any grep matches, including the column of the first match on a
|
||||||
|
line.
|
||||||
|
|
||||||
4. Any whitespace errors detected by `git diff --check`.
|
4. Any whitespace errors detected by `git diff --check`.
|
||||||
|
|
||||||
@ -82,7 +90,7 @@ which does something similar to `git jump grep`. However, it is limited
|
|||||||
to positioning the cursor to the correct line in only the first file,
|
to positioning the cursor to the correct line in only the first file,
|
||||||
leaving you to locate subsequent hits in that file or other files using
|
leaving you to locate subsequent hits in that file or other files using
|
||||||
the editor or pager. By contrast, git-jump provides the editor with a
|
the editor or pager. By contrast, git-jump provides the editor with a
|
||||||
complete list of files and line numbers for each match.
|
complete list of files, lines, and a column number for each match.
|
||||||
|
|
||||||
|
|
||||||
Limitations
|
Limitations
|
||||||
|
@ -52,7 +52,7 @@ mode_merge() {
|
|||||||
# editor shows them to us in the status bar.
|
# editor shows them to us in the status bar.
|
||||||
mode_grep() {
|
mode_grep() {
|
||||||
cmd=$(git config jump.grepCmd)
|
cmd=$(git config jump.grepCmd)
|
||||||
test -n "$cmd" || cmd="git grep -n"
|
test -n "$cmd" || cmd="git grep -n --column"
|
||||||
$cmd "$@" |
|
$cmd "$@" |
|
||||||
perl -pe '
|
perl -pe '
|
||||||
s/[ \t]+/ /g;
|
s/[ \t]+/ /g;
|
||||||
|
130
grep.c
130
grep.c
@ -20,6 +20,7 @@ static const char *color_grep_slots[] = {
|
|||||||
[GREP_COLOR_FILENAME] = "filename",
|
[GREP_COLOR_FILENAME] = "filename",
|
||||||
[GREP_COLOR_FUNCTION] = "function",
|
[GREP_COLOR_FUNCTION] = "function",
|
||||||
[GREP_COLOR_LINENO] = "lineNumber",
|
[GREP_COLOR_LINENO] = "lineNumber",
|
||||||
|
[GREP_COLOR_COLUMNNO] = "column",
|
||||||
[GREP_COLOR_MATCH_CONTEXT] = "matchContext",
|
[GREP_COLOR_MATCH_CONTEXT] = "matchContext",
|
||||||
[GREP_COLOR_MATCH_SELECTED] = "matchSelected",
|
[GREP_COLOR_MATCH_SELECTED] = "matchSelected",
|
||||||
[GREP_COLOR_SELECTED] = "selected",
|
[GREP_COLOR_SELECTED] = "selected",
|
||||||
@ -59,6 +60,7 @@ void init_grep_defaults(void)
|
|||||||
color_set(opt->colors[GREP_COLOR_FILENAME], "");
|
color_set(opt->colors[GREP_COLOR_FILENAME], "");
|
||||||
color_set(opt->colors[GREP_COLOR_FUNCTION], "");
|
color_set(opt->colors[GREP_COLOR_FUNCTION], "");
|
||||||
color_set(opt->colors[GREP_COLOR_LINENO], "");
|
color_set(opt->colors[GREP_COLOR_LINENO], "");
|
||||||
|
color_set(opt->colors[GREP_COLOR_COLUMNNO], "");
|
||||||
color_set(opt->colors[GREP_COLOR_MATCH_CONTEXT], GIT_COLOR_BOLD_RED);
|
color_set(opt->colors[GREP_COLOR_MATCH_CONTEXT], GIT_COLOR_BOLD_RED);
|
||||||
color_set(opt->colors[GREP_COLOR_MATCH_SELECTED], GIT_COLOR_BOLD_RED);
|
color_set(opt->colors[GREP_COLOR_MATCH_SELECTED], GIT_COLOR_BOLD_RED);
|
||||||
color_set(opt->colors[GREP_COLOR_SELECTED], "");
|
color_set(opt->colors[GREP_COLOR_SELECTED], "");
|
||||||
@ -110,6 +112,10 @@ int grep_config(const char *var, const char *value, void *cb)
|
|||||||
opt->linenum = git_config_bool(var, value);
|
opt->linenum = git_config_bool(var, value);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
if (!strcmp(var, "grep.column")) {
|
||||||
|
opt->columnnum = git_config_bool(var, value);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
if (!strcmp(var, "grep.fullname")) {
|
if (!strcmp(var, "grep.fullname")) {
|
||||||
opt->relative = !git_config_bool(var, value);
|
opt->relative = !git_config_bool(var, value);
|
||||||
@ -157,6 +163,7 @@ void grep_init(struct grep_opt *opt, const char *prefix)
|
|||||||
opt->extended_regexp_option = def->extended_regexp_option;
|
opt->extended_regexp_option = def->extended_regexp_option;
|
||||||
opt->pattern_type_option = def->pattern_type_option;
|
opt->pattern_type_option = def->pattern_type_option;
|
||||||
opt->linenum = def->linenum;
|
opt->linenum = def->linenum;
|
||||||
|
opt->columnnum = def->columnnum;
|
||||||
opt->max_depth = def->max_depth;
|
opt->max_depth = def->max_depth;
|
||||||
opt->pathname = def->pathname;
|
opt->pathname = def->pathname;
|
||||||
opt->relative = def->relative;
|
opt->relative = def->relative;
|
||||||
@ -1244,11 +1251,11 @@ static int match_one_pattern(struct grep_pat *p, char *bol, char *eol,
|
|||||||
return hit;
|
return hit;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int match_expr_eval(struct grep_expr *x, char *bol, char *eol,
|
static int match_expr_eval(struct grep_opt *opt, struct grep_expr *x, char *bol,
|
||||||
enum grep_context ctx, int collect_hits)
|
char *eol, enum grep_context ctx, ssize_t *col,
|
||||||
|
ssize_t *icol, int collect_hits)
|
||||||
{
|
{
|
||||||
int h = 0;
|
int h = 0;
|
||||||
regmatch_t match;
|
|
||||||
|
|
||||||
if (!x)
|
if (!x)
|
||||||
die("Not a valid grep expression");
|
die("Not a valid grep expression");
|
||||||
@ -1257,25 +1264,52 @@ static int match_expr_eval(struct grep_expr *x, char *bol, char *eol,
|
|||||||
h = 1;
|
h = 1;
|
||||||
break;
|
break;
|
||||||
case GREP_NODE_ATOM:
|
case GREP_NODE_ATOM:
|
||||||
h = match_one_pattern(x->u.atom, bol, eol, ctx, &match, 0);
|
{
|
||||||
|
regmatch_t tmp;
|
||||||
|
h = match_one_pattern(x->u.atom, bol, eol, ctx,
|
||||||
|
&tmp, 0);
|
||||||
|
if (h && (*col < 0 || tmp.rm_so < *col))
|
||||||
|
*col = tmp.rm_so;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case GREP_NODE_NOT:
|
case GREP_NODE_NOT:
|
||||||
h = !match_expr_eval(x->u.unary, bol, eol, ctx, 0);
|
/*
|
||||||
|
* Upon visiting a GREP_NODE_NOT, col and icol become swapped.
|
||||||
|
*/
|
||||||
|
h = !match_expr_eval(opt, x->u.unary, bol, eol, ctx, icol, col,
|
||||||
|
0);
|
||||||
break;
|
break;
|
||||||
case GREP_NODE_AND:
|
case GREP_NODE_AND:
|
||||||
if (!match_expr_eval(x->u.binary.left, bol, eol, ctx, 0))
|
h = match_expr_eval(opt, x->u.binary.left, bol, eol, ctx, col,
|
||||||
return 0;
|
icol, 0);
|
||||||
h = match_expr_eval(x->u.binary.right, bol, eol, ctx, 0);
|
if (h || opt->columnnum) {
|
||||||
|
/*
|
||||||
|
* Don't short-circuit AND when given --column, since a
|
||||||
|
* NOT earlier in the tree may turn this into an OR. In
|
||||||
|
* this case, see the below comment.
|
||||||
|
*/
|
||||||
|
h &= match_expr_eval(opt, x->u.binary.right, bol, eol,
|
||||||
|
ctx, col, icol, 0);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case GREP_NODE_OR:
|
case GREP_NODE_OR:
|
||||||
if (!collect_hits)
|
if (!(collect_hits || opt->columnnum)) {
|
||||||
return (match_expr_eval(x->u.binary.left,
|
/*
|
||||||
bol, eol, ctx, 0) ||
|
* Don't short-circuit OR when given --column (or
|
||||||
match_expr_eval(x->u.binary.right,
|
* collecting hits) to ensure we don't skip a later
|
||||||
bol, eol, ctx, 0));
|
* child that would produce an earlier match.
|
||||||
h = match_expr_eval(x->u.binary.left, bol, eol, ctx, 0);
|
*/
|
||||||
|
return (match_expr_eval(opt, x->u.binary.left, bol, eol,
|
||||||
|
ctx, col, icol, 0) ||
|
||||||
|
match_expr_eval(opt, x->u.binary.right, bol,
|
||||||
|
eol, ctx, col, icol, 0));
|
||||||
|
}
|
||||||
|
h = match_expr_eval(opt, x->u.binary.left, bol, eol, ctx, col,
|
||||||
|
icol, 0);
|
||||||
|
if (collect_hits)
|
||||||
x->u.binary.left->hit |= h;
|
x->u.binary.left->hit |= h;
|
||||||
h |= match_expr_eval(x->u.binary.right, bol, eol, ctx, 1);
|
h |= match_expr_eval(opt, x->u.binary.right, bol, eol, ctx, col,
|
||||||
|
icol, collect_hits);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
die("Unexpected node type (internal error) %d", x->node);
|
die("Unexpected node type (internal error) %d", x->node);
|
||||||
@ -1286,27 +1320,43 @@ static int match_expr_eval(struct grep_expr *x, char *bol, char *eol,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int match_expr(struct grep_opt *opt, char *bol, char *eol,
|
static int match_expr(struct grep_opt *opt, char *bol, char *eol,
|
||||||
enum grep_context ctx, int collect_hits)
|
enum grep_context ctx, ssize_t *col,
|
||||||
|
ssize_t *icol, int collect_hits)
|
||||||
{
|
{
|
||||||
struct grep_expr *x = opt->pattern_expression;
|
struct grep_expr *x = opt->pattern_expression;
|
||||||
return match_expr_eval(x, bol, eol, ctx, collect_hits);
|
return match_expr_eval(opt, x, bol, eol, ctx, col, icol, collect_hits);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int match_line(struct grep_opt *opt, char *bol, char *eol,
|
static int match_line(struct grep_opt *opt, char *bol, char *eol,
|
||||||
|
ssize_t *col, ssize_t *icol,
|
||||||
enum grep_context ctx, int collect_hits)
|
enum grep_context ctx, int collect_hits)
|
||||||
{
|
{
|
||||||
struct grep_pat *p;
|
struct grep_pat *p;
|
||||||
regmatch_t match;
|
int hit = 0;
|
||||||
|
|
||||||
if (opt->extended)
|
if (opt->extended)
|
||||||
return match_expr(opt, bol, eol, ctx, collect_hits);
|
return match_expr(opt, bol, eol, ctx, col, icol,
|
||||||
|
collect_hits);
|
||||||
|
|
||||||
/* we do not call with collect_hits without being extended */
|
/* we do not call with collect_hits without being extended */
|
||||||
for (p = opt->pattern_list; p; p = p->next) {
|
for (p = opt->pattern_list; p; p = p->next) {
|
||||||
if (match_one_pattern(p, bol, eol, ctx, &match, 0))
|
regmatch_t tmp;
|
||||||
return 1;
|
if (match_one_pattern(p, bol, eol, ctx, &tmp, 0)) {
|
||||||
|
hit |= 1;
|
||||||
|
if (!opt->columnnum) {
|
||||||
|
/*
|
||||||
|
* Without --column, any single match on a line
|
||||||
|
* is enough to know that it needs to be
|
||||||
|
* printed. With --column, scan _all_ patterns
|
||||||
|
* to find the earliest.
|
||||||
|
*/
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return 0;
|
if (*col < 0 || tmp.rm_so < *col)
|
||||||
|
*col = tmp.rm_so;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return hit;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int match_next_pattern(struct grep_pat *p, char *bol, char *eol,
|
static int match_next_pattern(struct grep_pat *p, char *bol, char *eol,
|
||||||
@ -1355,7 +1405,7 @@ static int next_match(struct grep_opt *opt, char *bol, char *eol,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void show_line(struct grep_opt *opt, char *bol, char *eol,
|
static void show_line(struct grep_opt *opt, char *bol, char *eol,
|
||||||
const char *name, unsigned lno, char sign)
|
const char *name, unsigned lno, ssize_t cno, char sign)
|
||||||
{
|
{
|
||||||
int rest = eol - bol;
|
int rest = eol - bol;
|
||||||
const char *match_color, *line_color = NULL;
|
const char *match_color, *line_color = NULL;
|
||||||
@ -1390,6 +1440,17 @@ static void show_line(struct grep_opt *opt, char *bol, char *eol,
|
|||||||
output_color(opt, buf, strlen(buf), opt->colors[GREP_COLOR_LINENO]);
|
output_color(opt, buf, strlen(buf), opt->colors[GREP_COLOR_LINENO]);
|
||||||
output_sep(opt, sign);
|
output_sep(opt, sign);
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
|
* Treat 'cno' as the 1-indexed offset from the start of a non-context
|
||||||
|
* line to its first match. Otherwise, 'cno' is 0 indicating that we are
|
||||||
|
* being called with a context line.
|
||||||
|
*/
|
||||||
|
if (opt->columnnum && cno) {
|
||||||
|
char buf[32];
|
||||||
|
xsnprintf(buf, sizeof(buf), "%"PRIuMAX, (uintmax_t)cno);
|
||||||
|
output_color(opt, buf, strlen(buf), opt->colors[GREP_COLOR_COLUMNNO]);
|
||||||
|
output_sep(opt, sign);
|
||||||
|
}
|
||||||
if (opt->color) {
|
if (opt->color) {
|
||||||
regmatch_t match;
|
regmatch_t match;
|
||||||
enum grep_context ctx = GREP_CONTEXT_BODY;
|
enum grep_context ctx = GREP_CONTEXT_BODY;
|
||||||
@ -1495,7 +1556,7 @@ static void show_funcname_line(struct grep_opt *opt, struct grep_source *gs,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
if (match_funcname(opt, gs, bol, eol)) {
|
if (match_funcname(opt, gs, bol, eol)) {
|
||||||
show_line(opt, bol, eol, gs->name, lno, '=');
|
show_line(opt, bol, eol, gs->name, lno, 0, '=');
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1560,7 +1621,7 @@ static void show_pre_context(struct grep_opt *opt, struct grep_source *gs,
|
|||||||
|
|
||||||
while (*eol != '\n')
|
while (*eol != '\n')
|
||||||
eol++;
|
eol++;
|
||||||
show_line(opt, bol, eol, gs->name, cur, sign);
|
show_line(opt, bol, eol, gs->name, cur, 0, sign);
|
||||||
bol = eol + 1;
|
bol = eol + 1;
|
||||||
cur++;
|
cur++;
|
||||||
}
|
}
|
||||||
@ -1759,6 +1820,8 @@ static int grep_source_1(struct grep_opt *opt, struct grep_source *gs, int colle
|
|||||||
while (left) {
|
while (left) {
|
||||||
char *eol, ch;
|
char *eol, ch;
|
||||||
int hit;
|
int hit;
|
||||||
|
ssize_t cno;
|
||||||
|
ssize_t col = -1, icol = -1;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* look_ahead() skips quickly to the line that possibly
|
* look_ahead() skips quickly to the line that possibly
|
||||||
@ -1782,7 +1845,7 @@ static int grep_source_1(struct grep_opt *opt, struct grep_source *gs, int colle
|
|||||||
if ((ctx == GREP_CONTEXT_HEAD) && (eol == bol))
|
if ((ctx == GREP_CONTEXT_HEAD) && (eol == bol))
|
||||||
ctx = GREP_CONTEXT_BODY;
|
ctx = GREP_CONTEXT_BODY;
|
||||||
|
|
||||||
hit = match_line(opt, bol, eol, ctx, collect_hits);
|
hit = match_line(opt, bol, eol, &col, &icol, ctx, collect_hits);
|
||||||
*eol = ch;
|
*eol = ch;
|
||||||
|
|
||||||
if (collect_hits)
|
if (collect_hits)
|
||||||
@ -1823,7 +1886,18 @@ static int grep_source_1(struct grep_opt *opt, struct grep_source *gs, int colle
|
|||||||
show_pre_context(opt, gs, bol, eol, lno);
|
show_pre_context(opt, gs, bol, eol, lno);
|
||||||
else if (opt->funcname)
|
else if (opt->funcname)
|
||||||
show_funcname_line(opt, gs, bol, lno);
|
show_funcname_line(opt, gs, bol, lno);
|
||||||
show_line(opt, bol, eol, gs->name, lno, ':');
|
cno = opt->invert ? icol : col;
|
||||||
|
if (cno < 0) {
|
||||||
|
/*
|
||||||
|
* A negative cno indicates that there was no
|
||||||
|
* match on the line. We are thus inverted and
|
||||||
|
* being asked to show all lines that _don't_
|
||||||
|
* match a given expression. Therefore, set cno
|
||||||
|
* to 0 to suggest the whole line matches.
|
||||||
|
*/
|
||||||
|
cno = 0;
|
||||||
|
}
|
||||||
|
show_line(opt, bol, eol, gs->name, lno, cno + 1, ':');
|
||||||
last_hit = lno;
|
last_hit = lno;
|
||||||
if (opt->funcbody)
|
if (opt->funcbody)
|
||||||
show_function = 1;
|
show_function = 1;
|
||||||
@ -1852,7 +1926,7 @@ static int grep_source_1(struct grep_opt *opt, struct grep_source *gs, int colle
|
|||||||
/* If the last hit is within the post context,
|
/* If the last hit is within the post context,
|
||||||
* we need to show this line.
|
* we need to show this line.
|
||||||
*/
|
*/
|
||||||
show_line(opt, bol, eol, gs->name, lno, '-');
|
show_line(opt, bol, eol, gs->name, lno, col + 1, '-');
|
||||||
}
|
}
|
||||||
|
|
||||||
next_line:
|
next_line:
|
||||||
|
2
grep.h
2
grep.h
@ -67,6 +67,7 @@ enum grep_color {
|
|||||||
GREP_COLOR_FILENAME,
|
GREP_COLOR_FILENAME,
|
||||||
GREP_COLOR_FUNCTION,
|
GREP_COLOR_FUNCTION,
|
||||||
GREP_COLOR_LINENO,
|
GREP_COLOR_LINENO,
|
||||||
|
GREP_COLOR_COLUMNNO,
|
||||||
GREP_COLOR_MATCH_CONTEXT,
|
GREP_COLOR_MATCH_CONTEXT,
|
||||||
GREP_COLOR_MATCH_SELECTED,
|
GREP_COLOR_MATCH_SELECTED,
|
||||||
GREP_COLOR_SELECTED,
|
GREP_COLOR_SELECTED,
|
||||||
@ -139,6 +140,7 @@ struct grep_opt {
|
|||||||
int prefix_length;
|
int prefix_length;
|
||||||
regex_t regexp;
|
regex_t regexp;
|
||||||
int linenum;
|
int linenum;
|
||||||
|
int columnnum;
|
||||||
int invert;
|
int invert;
|
||||||
int ignore_case;
|
int ignore_case;
|
||||||
int status_only;
|
int status_only;
|
||||||
|
@ -99,6 +99,101 @@ do
|
|||||||
test_cmp expected actual
|
test_cmp expected actual
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep -w $L (with --column)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:5:foo mmap bar
|
||||||
|
echo ${HC}file:14:foo_mmap bar mmap
|
||||||
|
echo ${HC}file:5:foo mmap bar_mmap
|
||||||
|
echo ${HC}file:14:foo_mmap bar mmap baz
|
||||||
|
} >expected &&
|
||||||
|
git grep --column -w -e mmap $H >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep -w $L (with --column, extended OR)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:14:foo_mmap bar mmap
|
||||||
|
echo ${HC}file:19:foo_mmap bar mmap baz
|
||||||
|
} >expected &&
|
||||||
|
git grep --column -w -e mmap$ --or -e baz $H >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep -w $L (with --column, --invert)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:1:foo mmap bar
|
||||||
|
echo ${HC}file:1:foo_mmap bar
|
||||||
|
echo ${HC}file:1:foo_mmap bar mmap
|
||||||
|
echo ${HC}file:1:foo mmap bar_mmap
|
||||||
|
} >expected &&
|
||||||
|
git grep --column --invert -w -e baz $H -- file >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep $L (with --column, --invert, extended OR)" '
|
||||||
|
{
|
||||||
|
echo ${HC}hello_world:6:HeLLo_world
|
||||||
|
} >expected &&
|
||||||
|
git grep --column --invert -e ll --or --not -e _ $H -- hello_world \
|
||||||
|
>actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep $L (with --column, --invert, extended AND)" '
|
||||||
|
{
|
||||||
|
echo ${HC}hello_world:3:Hello world
|
||||||
|
echo ${HC}hello_world:3:Hello_world
|
||||||
|
echo ${HC}hello_world:6:HeLLo_world
|
||||||
|
} >expected &&
|
||||||
|
git grep --column --invert --not -e _ --and --not -e ll $H -- hello_world \
|
||||||
|
>actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep $L (with --column, double-negation)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:1:foo_mmap bar mmap baz
|
||||||
|
} >expected &&
|
||||||
|
git grep --column --not \( --not -e foo --or --not -e baz \) $H -- file \
|
||||||
|
>actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep -w $L (with --column, -C)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:5:foo mmap bar
|
||||||
|
echo ${HC}file-foo_mmap bar
|
||||||
|
echo ${HC}file:14:foo_mmap bar mmap
|
||||||
|
echo ${HC}file:5:foo mmap bar_mmap
|
||||||
|
echo ${HC}file:14:foo_mmap bar mmap baz
|
||||||
|
} >expected &&
|
||||||
|
git grep --column -w -C1 -e mmap $H >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep -w $L (with --line-number, --column)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:1:5:foo mmap bar
|
||||||
|
echo ${HC}file:3:14:foo_mmap bar mmap
|
||||||
|
echo ${HC}file:4:5:foo mmap bar_mmap
|
||||||
|
echo ${HC}file:5:14:foo_mmap bar mmap baz
|
||||||
|
} >expected &&
|
||||||
|
git grep -n --column -w -e mmap $H >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success "grep -w $L (with non-extended patterns, --column)" '
|
||||||
|
{
|
||||||
|
echo ${HC}file:5:foo mmap bar
|
||||||
|
echo ${HC}file:10:foo_mmap bar
|
||||||
|
echo ${HC}file:10:foo_mmap bar mmap
|
||||||
|
echo ${HC}file:5:foo mmap bar_mmap
|
||||||
|
echo ${HC}file:10:foo_mmap bar mmap baz
|
||||||
|
} >expected &&
|
||||||
|
git grep --column -w -e bar -e mmap $H >actual &&
|
||||||
|
test_cmp expected actual
|
||||||
|
'
|
||||||
|
|
||||||
test_expect_success "grep -w $L" '
|
test_expect_success "grep -w $L" '
|
||||||
{
|
{
|
||||||
echo ${HC}file:1:foo mmap bar
|
echo ${HC}file:1:foo mmap bar
|
||||||
|
Loading…
Reference in New Issue
Block a user