| /* |
| * Whitespace rules |
| * |
| * Copyright (c) 2007 Junio C Hamano |
| */ |
| |
| #include "cache.h" |
| #include "attr.h" |
| |
| static struct whitespace_rule { |
| const char *rule_name; |
| unsigned rule_bits; |
| } whitespace_rule_names[] = { |
| { "trailing-space", WS_TRAILING_SPACE }, |
| { "space-before-tab", WS_SPACE_BEFORE_TAB }, |
| { "indent-with-non-tab", WS_INDENT_WITH_NON_TAB }, |
| }; |
| |
| unsigned parse_whitespace_rule(const char *string) |
| { |
| unsigned rule = WS_DEFAULT_RULE; |
| |
| while (string) { |
| int i; |
| size_t len; |
| const char *ep; |
| int negated = 0; |
| |
| string = string + strspn(string, ", \t\n\r"); |
| ep = strchr(string, ','); |
| if (!ep) |
| len = strlen(string); |
| else |
| len = ep - string; |
| |
| if (*string == '-') { |
| negated = 1; |
| string++; |
| len--; |
| } |
| if (!len) |
| break; |
| for (i = 0; i < ARRAY_SIZE(whitespace_rule_names); i++) { |
| if (strncmp(whitespace_rule_names[i].rule_name, |
| string, len)) |
| continue; |
| if (negated) |
| rule &= ~whitespace_rule_names[i].rule_bits; |
| else |
| rule |= whitespace_rule_names[i].rule_bits; |
| break; |
| } |
| string = ep; |
| } |
| return rule; |
| } |
| |
| static void setup_whitespace_attr_check(struct git_attr_check *check) |
| { |
| static struct git_attr *attr_whitespace; |
| |
| if (!attr_whitespace) |
| attr_whitespace = git_attr("whitespace", 10); |
| check[0].attr = attr_whitespace; |
| } |
| |
| unsigned whitespace_rule(const char *pathname) |
| { |
| struct git_attr_check attr_whitespace_rule; |
| |
| setup_whitespace_attr_check(&attr_whitespace_rule); |
| if (!git_checkattr(pathname, 1, &attr_whitespace_rule)) { |
| const char *value; |
| |
| value = attr_whitespace_rule.value; |
| if (ATTR_TRUE(value)) { |
| /* true (whitespace) */ |
| unsigned all_rule = 0; |
| int i; |
| for (i = 0; i < ARRAY_SIZE(whitespace_rule_names); i++) |
| all_rule |= whitespace_rule_names[i].rule_bits; |
| return all_rule; |
| } else if (ATTR_FALSE(value)) { |
| /* false (-whitespace) */ |
| return 0; |
| } else if (ATTR_UNSET(value)) { |
| /* reset to default (!whitespace) */ |
| return whitespace_rule_cfg; |
| } else { |
| /* string */ |
| return parse_whitespace_rule(value); |
| } |
| } else { |
| return whitespace_rule_cfg; |
| } |
| } |
| |
| /* The returned string should be freed by the caller. */ |
| char *whitespace_error_string(unsigned ws) |
| { |
| struct strbuf err; |
| strbuf_init(&err, 0); |
| if (ws & WS_TRAILING_SPACE) |
| strbuf_addstr(&err, "trailing whitespace"); |
| if (ws & WS_SPACE_BEFORE_TAB) { |
| if (err.len) |
| strbuf_addstr(&err, ", "); |
| strbuf_addstr(&err, "space before tab in indent"); |
| } |
| if (ws & WS_INDENT_WITH_NON_TAB) { |
| if (err.len) |
| strbuf_addstr(&err, ", "); |
| strbuf_addstr(&err, "indent with spaces"); |
| } |
| return strbuf_detach(&err, NULL); |
| } |
| |
| /* If stream is non-NULL, emits the line after checking. */ |
| unsigned check_and_emit_line(const char *line, int len, unsigned ws_rule, |
| FILE *stream, const char *set, |
| const char *reset, const char *ws) |
| { |
| unsigned result = 0; |
| int written = 0; |
| int trailing_whitespace = -1; |
| int trailing_newline = 0; |
| int i; |
| |
| /* Logic is simpler if we temporarily ignore the trailing newline. */ |
| if (len > 0 && line[len - 1] == '\n') { |
| trailing_newline = 1; |
| len--; |
| } |
| |
| /* Check for trailing whitespace. */ |
| if (ws_rule & WS_TRAILING_SPACE) { |
| for (i = len - 1; i >= 0; i--) { |
| if (isspace(line[i])) { |
| trailing_whitespace = i; |
| result |= WS_TRAILING_SPACE; |
| } |
| else |
| break; |
| } |
| } |
| |
| /* Check for space before tab in initial indent. */ |
| for (i = 0; i < len; i++) { |
| if (line[i] == ' ') |
| continue; |
| if (line[i] != '\t') |
| break; |
| if ((ws_rule & WS_SPACE_BEFORE_TAB) && written < i) { |
| result |= WS_SPACE_BEFORE_TAB; |
| if (stream) { |
| fputs(ws, stream); |
| fwrite(line + written, i - written, 1, stream); |
| fputs(reset, stream); |
| } |
| } else if (stream) |
| fwrite(line + written, i - written, 1, stream); |
| if (stream) |
| fwrite(line + i, 1, 1, stream); |
| written = i + 1; |
| } |
| |
| /* Check for indent using non-tab. */ |
| if ((ws_rule & WS_INDENT_WITH_NON_TAB) && i - written >= 8) { |
| result |= WS_INDENT_WITH_NON_TAB; |
| if (stream) { |
| fputs(ws, stream); |
| fwrite(line + written, i - written, 1, stream); |
| fputs(reset, stream); |
| } |
| written = i; |
| } |
| |
| if (stream) { |
| /* Now the rest of the line starts at written. |
| * The non-highlighted part ends at trailing_whitespace. */ |
| if (trailing_whitespace == -1) |
| trailing_whitespace = len; |
| |
| /* Emit non-highlighted (middle) segment. */ |
| if (trailing_whitespace - written > 0) { |
| fputs(set, stream); |
| fwrite(line + written, |
| trailing_whitespace - written, 1, stream); |
| fputs(reset, stream); |
| } |
| |
| /* Highlight errors in trailing whitespace. */ |
| if (trailing_whitespace != len) { |
| fputs(ws, stream); |
| fwrite(line + trailing_whitespace, |
| len - trailing_whitespace, 1, stream); |
| fputs(reset, stream); |
| } |
| if (trailing_newline) |
| fputc('\n', stream); |
| } |
| return result; |
| } |