Merge branch 'bw/pathspec-sans-the-index'
Simplify parse_pathspec() codepath and stop it from looking at the default in-core index. * bw/pathspec-sans-the-index: pathspec: convert find_pathspecs_matching_against_index to take an index pathspec: remove PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP ls-files: prevent prune_cache from overeagerly pruning submodules pathspec: remove PATHSPEC_STRIP_SUBMODULE_SLASH_EXPENSIVE flag submodule: add die_in_unpopulated_submodule function pathspec: provide a more descriptive die message
This commit is contained in:
commit
3c5a78280f
@ -17,6 +17,7 @@
|
|||||||
#include "revision.h"
|
#include "revision.h"
|
||||||
#include "bulk-checkin.h"
|
#include "bulk-checkin.h"
|
||||||
#include "argv-array.h"
|
#include "argv-array.h"
|
||||||
|
#include "submodule.h"
|
||||||
|
|
||||||
static const char * const builtin_add_usage[] = {
|
static const char * const builtin_add_usage[] = {
|
||||||
N_("git add [<options>] [--] <pathspec>..."),
|
N_("git add [<options>] [--] <pathspec>..."),
|
||||||
@ -135,7 +136,7 @@ static char *prune_directory(struct dir_struct *dir, struct pathspec *pathspec,
|
|||||||
*dst++ = entry;
|
*dst++ = entry;
|
||||||
}
|
}
|
||||||
dir->nr = dst - dir->entries;
|
dir->nr = dst - dir->entries;
|
||||||
add_pathspec_matches_against_index(pathspec, seen);
|
add_pathspec_matches_against_index(pathspec, &the_index, seen);
|
||||||
return seen;
|
return seen;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -379,16 +380,19 @@ int cmd_add(int argc, const char **argv, const char *prefix)
|
|||||||
if (read_cache() < 0)
|
if (read_cache() < 0)
|
||||||
die(_("index file corrupt"));
|
die(_("index file corrupt"));
|
||||||
|
|
||||||
|
die_in_unpopulated_submodule(&the_index, prefix);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Check the "pathspec '%s' did not match any files" block
|
* Check the "pathspec '%s' did not match any files" block
|
||||||
* below before enabling new magic.
|
* below before enabling new magic.
|
||||||
*/
|
*/
|
||||||
parse_pathspec(&pathspec, 0,
|
parse_pathspec(&pathspec, 0,
|
||||||
PATHSPEC_PREFER_FULL |
|
PATHSPEC_PREFER_FULL |
|
||||||
PATHSPEC_SYMLINK_LEADING_PATH |
|
PATHSPEC_SYMLINK_LEADING_PATH,
|
||||||
PATHSPEC_STRIP_SUBMODULE_SLASH_EXPENSIVE,
|
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
|
|
||||||
|
die_path_inside_submodule(&the_index, &pathspec);
|
||||||
|
|
||||||
if (add_new_files) {
|
if (add_new_files) {
|
||||||
int baselen;
|
int baselen;
|
||||||
|
|
||||||
@ -414,7 +418,7 @@ int cmd_add(int argc, const char **argv, const char *prefix)
|
|||||||
int i;
|
int i;
|
||||||
|
|
||||||
if (!seen)
|
if (!seen)
|
||||||
seen = find_pathspecs_matching_against_index(&pathspec);
|
seen = find_pathspecs_matching_against_index(&pathspec, &the_index);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* file_exists() assumes exact match
|
* file_exists() assumes exact match
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
#include "quote.h"
|
#include "quote.h"
|
||||||
#include "pathspec.h"
|
#include "pathspec.h"
|
||||||
#include "parse-options.h"
|
#include "parse-options.h"
|
||||||
|
#include "submodule.h"
|
||||||
|
|
||||||
static int quiet, verbose, stdin_paths, show_non_matching, no_index;
|
static int quiet, verbose, stdin_paths, show_non_matching, no_index;
|
||||||
static const char * const check_ignore_usage[] = {
|
static const char * const check_ignore_usage[] = {
|
||||||
@ -87,16 +88,17 @@ static int check_ignore(struct dir_struct *dir,
|
|||||||
parse_pathspec(&pathspec,
|
parse_pathspec(&pathspec,
|
||||||
PATHSPEC_ALL_MAGIC & ~PATHSPEC_FROMTOP,
|
PATHSPEC_ALL_MAGIC & ~PATHSPEC_FROMTOP,
|
||||||
PATHSPEC_SYMLINK_LEADING_PATH |
|
PATHSPEC_SYMLINK_LEADING_PATH |
|
||||||
PATHSPEC_STRIP_SUBMODULE_SLASH_EXPENSIVE |
|
|
||||||
PATHSPEC_KEEP_ORDER,
|
PATHSPEC_KEEP_ORDER,
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
|
|
||||||
|
die_path_inside_submodule(&the_index, &pathspec);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* look for pathspecs matching entries in the index, since these
|
* look for pathspecs matching entries in the index, since these
|
||||||
* should not be ignored, in order to be consistent with
|
* should not be ignored, in order to be consistent with
|
||||||
* 'git status', 'git add' etc.
|
* 'git status', 'git add' etc.
|
||||||
*/
|
*/
|
||||||
seen = find_pathspecs_matching_against_index(&pathspec);
|
seen = find_pathspecs_matching_against_index(&pathspec, &the_index);
|
||||||
for (i = 0; i < pathspec.nr; i++) {
|
for (i = 0; i < pathspec.nr; i++) {
|
||||||
full_path = pathspec.items[i].match;
|
full_path = pathspec.items[i].match;
|
||||||
exclude = NULL;
|
exclude = NULL;
|
||||||
|
@ -97,7 +97,7 @@ static void show_dir_entry(const char *tag, struct dir_entry *ent)
|
|||||||
{
|
{
|
||||||
int len = max_prefix_len;
|
int len = max_prefix_len;
|
||||||
|
|
||||||
if (len >= ent->len)
|
if (len > ent->len)
|
||||||
die("git ls-files: internal error - directory entry not superset of prefix");
|
die("git ls-files: internal error - directory entry not superset of prefix");
|
||||||
|
|
||||||
if (!dir_path_match(ent, &pathspec, len, ps_matched))
|
if (!dir_path_match(ent, &pathspec, len, ps_matched))
|
||||||
@ -238,7 +238,7 @@ static void show_ce_entry(const char *tag, const struct cache_entry *ce)
|
|||||||
strbuf_addstr(&name, super_prefix);
|
strbuf_addstr(&name, super_prefix);
|
||||||
strbuf_addstr(&name, ce->name);
|
strbuf_addstr(&name, ce->name);
|
||||||
|
|
||||||
if (len >= ce_namelen(ce))
|
if (len > ce_namelen(ce))
|
||||||
die("git ls-files: internal error - cache entry not superset of prefix");
|
die("git ls-files: internal error - cache entry not superset of prefix");
|
||||||
|
|
||||||
if (recurse_submodules && S_ISGITLINK(ce->ce_mode) &&
|
if (recurse_submodules && S_ISGITLINK(ce->ce_mode) &&
|
||||||
@ -403,6 +403,25 @@ static void prune_cache(const char *prefix, size_t prefixlen)
|
|||||||
active_nr = last - pos;
|
active_nr = last - pos;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int get_common_prefix_len(const char *common_prefix)
|
||||||
|
{
|
||||||
|
int common_prefix_len;
|
||||||
|
|
||||||
|
if (!common_prefix)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
common_prefix_len = strlen(common_prefix);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the prefix has a trailing slash, strip it so that submodules wont
|
||||||
|
* be pruned from the index.
|
||||||
|
*/
|
||||||
|
if (common_prefix[common_prefix_len - 1] == '/')
|
||||||
|
common_prefix_len--;
|
||||||
|
|
||||||
|
return common_prefix_len;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read the tree specified with --with-tree option
|
* Read the tree specified with --with-tree option
|
||||||
* (typically, HEAD) into stage #1 and then
|
* (typically, HEAD) into stage #1 and then
|
||||||
@ -624,8 +643,7 @@ int cmd_ls_files(int argc, const char **argv, const char *cmd_prefix)
|
|||||||
"--error-unmatch");
|
"--error-unmatch");
|
||||||
|
|
||||||
parse_pathspec(&pathspec, 0,
|
parse_pathspec(&pathspec, 0,
|
||||||
PATHSPEC_PREFER_CWD |
|
PATHSPEC_PREFER_CWD,
|
||||||
PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP,
|
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -637,7 +655,9 @@ int cmd_ls_files(int argc, const char **argv, const char *cmd_prefix)
|
|||||||
max_prefix = NULL;
|
max_prefix = NULL;
|
||||||
else
|
else
|
||||||
max_prefix = common_prefix(&pathspec);
|
max_prefix = common_prefix(&pathspec);
|
||||||
max_prefix_len = max_prefix ? strlen(max_prefix) : 0;
|
max_prefix_len = get_common_prefix_len(max_prefix);
|
||||||
|
|
||||||
|
prune_cache(max_prefix, max_prefix_len);
|
||||||
|
|
||||||
/* Treat unmatching pathspec elements as errors */
|
/* Treat unmatching pathspec elements as errors */
|
||||||
if (pathspec.nr && error_unmatch)
|
if (pathspec.nr && error_unmatch)
|
||||||
@ -651,7 +671,6 @@ int cmd_ls_files(int argc, const char **argv, const char *cmd_prefix)
|
|||||||
show_killed || show_modified || show_resolve_undo))
|
show_killed || show_modified || show_resolve_undo))
|
||||||
show_cached = 1;
|
show_cached = 1;
|
||||||
|
|
||||||
prune_cache(max_prefix, max_prefix_len);
|
|
||||||
if (with_tree) {
|
if (with_tree) {
|
||||||
/*
|
/*
|
||||||
* Basic sanity check; show-stages and show-unmerged
|
* Basic sanity check; show-stages and show-unmerged
|
||||||
|
@ -257,7 +257,6 @@ static void parse_args(struct pathspec *pathspec,
|
|||||||
|
|
||||||
parse_pathspec(pathspec, 0,
|
parse_pathspec(pathspec, 0,
|
||||||
PATHSPEC_PREFER_FULL |
|
PATHSPEC_PREFER_FULL |
|
||||||
PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP |
|
|
||||||
(patch_mode ? PATHSPEC_PREFIX_ORIGIN : 0),
|
(patch_mode ? PATHSPEC_PREFIX_ORIGIN : 0),
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
}
|
}
|
||||||
|
@ -271,8 +271,7 @@ int cmd_rm(int argc, const char **argv, const char *prefix)
|
|||||||
die(_("index file corrupt"));
|
die(_("index file corrupt"));
|
||||||
|
|
||||||
parse_pathspec(&pathspec, 0,
|
parse_pathspec(&pathspec, 0,
|
||||||
PATHSPEC_PREFER_CWD |
|
PATHSPEC_PREFER_CWD,
|
||||||
PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP,
|
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
refresh_index(&the_index, REFRESH_QUIET, &pathspec, NULL, NULL);
|
refresh_index(&the_index, REFRESH_QUIET, &pathspec, NULL, NULL);
|
||||||
|
|
||||||
|
@ -233,8 +233,7 @@ static int module_list_compute(int argc, const char **argv,
|
|||||||
int i, result = 0;
|
int i, result = 0;
|
||||||
char *ps_matched = NULL;
|
char *ps_matched = NULL;
|
||||||
parse_pathspec(pathspec, 0,
|
parse_pathspec(pathspec, 0,
|
||||||
PATHSPEC_PREFER_FULL |
|
PATHSPEC_PREFER_FULL,
|
||||||
PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP,
|
|
||||||
prefix, argv);
|
prefix, argv);
|
||||||
|
|
||||||
if (pathspec->nr)
|
if (pathspec->nr)
|
||||||
|
86
pathspec.c
86
pathspec.c
@ -1,3 +1,4 @@
|
|||||||
|
#define NO_THE_INDEX_COMPATIBILITY_MACROS
|
||||||
#include "cache.h"
|
#include "cache.h"
|
||||||
#include "dir.h"
|
#include "dir.h"
|
||||||
#include "pathspec.h"
|
#include "pathspec.h"
|
||||||
@ -17,6 +18,7 @@
|
|||||||
* to use find_pathspecs_matching_against_index() instead.
|
* to use find_pathspecs_matching_against_index() instead.
|
||||||
*/
|
*/
|
||||||
void add_pathspec_matches_against_index(const struct pathspec *pathspec,
|
void add_pathspec_matches_against_index(const struct pathspec *pathspec,
|
||||||
|
const struct index_state *istate,
|
||||||
char *seen)
|
char *seen)
|
||||||
{
|
{
|
||||||
int num_unmatched = 0, i;
|
int num_unmatched = 0, i;
|
||||||
@ -32,8 +34,8 @@ void add_pathspec_matches_against_index(const struct pathspec *pathspec,
|
|||||||
num_unmatched++;
|
num_unmatched++;
|
||||||
if (!num_unmatched)
|
if (!num_unmatched)
|
||||||
return;
|
return;
|
||||||
for (i = 0; i < active_nr; i++) {
|
for (i = 0; i < istate->cache_nr; i++) {
|
||||||
const struct cache_entry *ce = active_cache[i];
|
const struct cache_entry *ce = istate->cache[i];
|
||||||
ce_path_match(ce, pathspec, seen);
|
ce_path_match(ce, pathspec, seen);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -46,10 +48,11 @@ void add_pathspec_matches_against_index(const struct pathspec *pathspec,
|
|||||||
* nature of the "closest" (i.e. most specific) matches which each of the
|
* nature of the "closest" (i.e. most specific) matches which each of the
|
||||||
* given pathspecs achieves against all items in the index.
|
* given pathspecs achieves against all items in the index.
|
||||||
*/
|
*/
|
||||||
char *find_pathspecs_matching_against_index(const struct pathspec *pathspec)
|
char *find_pathspecs_matching_against_index(const struct pathspec *pathspec,
|
||||||
|
const struct index_state *istate)
|
||||||
{
|
{
|
||||||
char *seen = xcalloc(pathspec->nr, 1);
|
char *seen = xcalloc(pathspec->nr, 1);
|
||||||
add_pathspec_matches_against_index(pathspec, seen);
|
add_pathspec_matches_against_index(pathspec, istate, seen);
|
||||||
return seen;
|
return seen;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -386,65 +389,6 @@ static const char *parse_element_magic(unsigned *magic, int *prefix_len,
|
|||||||
return parse_short_magic(magic, elem);
|
return parse_short_magic(magic, elem);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void strip_submodule_slash_cheap(struct pathspec_item *item)
|
|
||||||
{
|
|
||||||
if (item->len >= 1 && item->match[item->len - 1] == '/') {
|
|
||||||
int i = cache_name_pos(item->match, item->len - 1);
|
|
||||||
|
|
||||||
if (i >= 0 && S_ISGITLINK(active_cache[i]->ce_mode)) {
|
|
||||||
item->len--;
|
|
||||||
item->match[item->len] = '\0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void strip_submodule_slash_expensive(struct pathspec_item *item)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; i < active_nr; i++) {
|
|
||||||
struct cache_entry *ce = active_cache[i];
|
|
||||||
int ce_len = ce_namelen(ce);
|
|
||||||
|
|
||||||
if (!S_ISGITLINK(ce->ce_mode))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (item->len <= ce_len || item->match[ce_len] != '/' ||
|
|
||||||
memcmp(ce->name, item->match, ce_len))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (item->len == ce_len + 1) {
|
|
||||||
/* strip trailing slash */
|
|
||||||
item->len--;
|
|
||||||
item->match[item->len] = '\0';
|
|
||||||
} else {
|
|
||||||
die(_("Pathspec '%s' is in submodule '%.*s'"),
|
|
||||||
item->original, ce_len, ce->name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void die_inside_submodule_path(struct pathspec_item *item)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; i < active_nr; i++) {
|
|
||||||
struct cache_entry *ce = active_cache[i];
|
|
||||||
int ce_len = ce_namelen(ce);
|
|
||||||
|
|
||||||
if (!S_ISGITLINK(ce->ce_mode))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (item->len < ce_len ||
|
|
||||||
!(item->match[ce_len] == '/' || item->match[ce_len] == '\0') ||
|
|
||||||
memcmp(ce->name, item->match, ce_len))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
die(_("Pathspec '%s' is in submodule '%.*s'"),
|
|
||||||
item->original, ce_len, ce->name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Perform the initialization of a pathspec_item based on a pathspec element.
|
* Perform the initialization of a pathspec_item based on a pathspec element.
|
||||||
*/
|
*/
|
||||||
@ -517,12 +461,6 @@ static void init_pathspec_item(struct pathspec_item *item, unsigned flags,
|
|||||||
item->original = xstrdup(elt);
|
item->original = xstrdup(elt);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP)
|
|
||||||
strip_submodule_slash_cheap(item);
|
|
||||||
|
|
||||||
if (flags & PATHSPEC_STRIP_SUBMODULE_SLASH_EXPENSIVE)
|
|
||||||
strip_submodule_slash_expensive(item);
|
|
||||||
|
|
||||||
if (magic & PATHSPEC_LITERAL) {
|
if (magic & PATHSPEC_LITERAL) {
|
||||||
item->nowildcard_len = item->len;
|
item->nowildcard_len = item->len;
|
||||||
} else {
|
} else {
|
||||||
@ -547,15 +485,7 @@ static void init_pathspec_item(struct pathspec_item *item, unsigned flags,
|
|||||||
/* sanity checks, pathspec matchers assume these are sane */
|
/* sanity checks, pathspec matchers assume these are sane */
|
||||||
if (item->nowildcard_len > item->len ||
|
if (item->nowildcard_len > item->len ||
|
||||||
item->prefix > item->len) {
|
item->prefix > item->len) {
|
||||||
/*
|
die ("BUG: error initializing pathspec_item");
|
||||||
* This case can be triggered by the user pointing us to a
|
|
||||||
* pathspec inside a submodule, which is an input error.
|
|
||||||
* Detect that here and complain, but fallback in the
|
|
||||||
* non-submodule case to a BUG, as we have no idea what
|
|
||||||
* would trigger that.
|
|
||||||
*/
|
|
||||||
die_inside_submodule_path(item);
|
|
||||||
die ("BUG: item->nowildcard_len > item->len || item->prefix > item->len)");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
25
pathspec.h
25
pathspec.h
@ -58,27 +58,17 @@ struct pathspec {
|
|||||||
#define PATHSPEC_PREFER_CWD (1<<0) /* No args means match cwd */
|
#define PATHSPEC_PREFER_CWD (1<<0) /* No args means match cwd */
|
||||||
#define PATHSPEC_PREFER_FULL (1<<1) /* No args means match everything */
|
#define PATHSPEC_PREFER_FULL (1<<1) /* No args means match everything */
|
||||||
#define PATHSPEC_MAXDEPTH_VALID (1<<2) /* max_depth field is valid */
|
#define PATHSPEC_MAXDEPTH_VALID (1<<2) /* max_depth field is valid */
|
||||||
/* strip the trailing slash if the given path is a gitlink */
|
|
||||||
#define PATHSPEC_STRIP_SUBMODULE_SLASH_CHEAP (1<<3)
|
|
||||||
/* die if a symlink is part of the given path's directory */
|
/* die if a symlink is part of the given path's directory */
|
||||||
#define PATHSPEC_SYMLINK_LEADING_PATH (1<<4)
|
#define PATHSPEC_SYMLINK_LEADING_PATH (1<<3)
|
||||||
/*
|
#define PATHSPEC_PREFIX_ORIGIN (1<<4)
|
||||||
* This is like a combination of ..LEADING_PATH and .._SLASH_CHEAP
|
#define PATHSPEC_KEEP_ORDER (1<<5)
|
||||||
* (but not the same): it strips the trailing slash if the given path
|
|
||||||
* is a gitlink but also checks and dies if gitlink is part of the
|
|
||||||
* leading path (i.e. the given path goes beyond a submodule). It's
|
|
||||||
* safer than _SLASH_CHEAP and also more expensive.
|
|
||||||
*/
|
|
||||||
#define PATHSPEC_STRIP_SUBMODULE_SLASH_EXPENSIVE (1<<5)
|
|
||||||
#define PATHSPEC_PREFIX_ORIGIN (1<<6)
|
|
||||||
#define PATHSPEC_KEEP_ORDER (1<<7)
|
|
||||||
/*
|
/*
|
||||||
* For the callers that just need pure paths from somewhere else, not
|
* For the callers that just need pure paths from somewhere else, not
|
||||||
* from command line. Global --*-pathspecs options are ignored. No
|
* from command line. Global --*-pathspecs options are ignored. No
|
||||||
* magic is parsed in each pathspec either. If PATHSPEC_LITERAL is
|
* magic is parsed in each pathspec either. If PATHSPEC_LITERAL is
|
||||||
* allowed, then it will automatically set for every pathspec.
|
* allowed, then it will automatically set for every pathspec.
|
||||||
*/
|
*/
|
||||||
#define PATHSPEC_LITERAL_PATH (1<<8)
|
#define PATHSPEC_LITERAL_PATH (1<<6)
|
||||||
|
|
||||||
extern void parse_pathspec(struct pathspec *pathspec,
|
extern void parse_pathspec(struct pathspec *pathspec,
|
||||||
unsigned magic_mask,
|
unsigned magic_mask,
|
||||||
@ -106,7 +96,10 @@ static inline int ps_strcmp(const struct pathspec_item *item,
|
|||||||
return strcmp(s1, s2);
|
return strcmp(s1, s2);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern char *find_pathspecs_matching_against_index(const struct pathspec *pathspec);
|
extern void add_pathspec_matches_against_index(const struct pathspec *pathspec,
|
||||||
extern void add_pathspec_matches_against_index(const struct pathspec *pathspec, char *seen);
|
const struct index_state *istate,
|
||||||
|
char *seen);
|
||||||
|
extern char *find_pathspecs_matching_against_index(const struct pathspec *pathspec,
|
||||||
|
const struct index_state *istate);
|
||||||
|
|
||||||
#endif /* PATHSPEC_H */
|
#endif /* PATHSPEC_H */
|
||||||
|
63
submodule.c
63
submodule.c
@ -282,6 +282,69 @@ int is_submodule_populated_gently(const char *path, int *return_error_code)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dies if the provided 'prefix' corresponds to an unpopulated submodule
|
||||||
|
*/
|
||||||
|
void die_in_unpopulated_submodule(const struct index_state *istate,
|
||||||
|
const char *prefix)
|
||||||
|
{
|
||||||
|
int i, prefixlen;
|
||||||
|
|
||||||
|
if (!prefix)
|
||||||
|
return;
|
||||||
|
|
||||||
|
prefixlen = strlen(prefix);
|
||||||
|
|
||||||
|
for (i = 0; i < istate->cache_nr; i++) {
|
||||||
|
struct cache_entry *ce = istate->cache[i];
|
||||||
|
int ce_len = ce_namelen(ce);
|
||||||
|
|
||||||
|
if (!S_ISGITLINK(ce->ce_mode))
|
||||||
|
continue;
|
||||||
|
if (prefixlen <= ce_len)
|
||||||
|
continue;
|
||||||
|
if (strncmp(ce->name, prefix, ce_len))
|
||||||
|
continue;
|
||||||
|
if (prefix[ce_len] != '/')
|
||||||
|
continue;
|
||||||
|
|
||||||
|
die(_("in unpopulated submodule '%s'"), ce->name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dies if any paths in the provided pathspec descends into a submodule
|
||||||
|
*/
|
||||||
|
void die_path_inside_submodule(const struct index_state *istate,
|
||||||
|
const struct pathspec *ps)
|
||||||
|
{
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
for (i = 0; i < istate->cache_nr; i++) {
|
||||||
|
struct cache_entry *ce = istate->cache[i];
|
||||||
|
int ce_len = ce_namelen(ce);
|
||||||
|
|
||||||
|
if (!S_ISGITLINK(ce->ce_mode))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
for (j = 0; j < ps->nr ; j++) {
|
||||||
|
const struct pathspec_item *item = &ps->items[j];
|
||||||
|
|
||||||
|
if (item->len <= ce_len)
|
||||||
|
continue;
|
||||||
|
if (item->match[ce_len] != '/')
|
||||||
|
continue;
|
||||||
|
if (strncmp(ce->name, item->match, ce_len))
|
||||||
|
continue;
|
||||||
|
if (item->len == ce_len + 1)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
die(_("Pathspec '%s' is in submodule '%.*s'"),
|
||||||
|
item->original, ce_len, ce->name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int parse_submodule_update_strategy(const char *value,
|
int parse_submodule_update_strategy(const char *value,
|
||||||
struct submodule_update_strategy *dst)
|
struct submodule_update_strategy *dst)
|
||||||
{
|
{
|
||||||
|
@ -49,6 +49,10 @@ extern int is_submodule_initialized(const char *path);
|
|||||||
* Otherwise the return error code is the same as of resolve_gitdir_gently.
|
* Otherwise the return error code is the same as of resolve_gitdir_gently.
|
||||||
*/
|
*/
|
||||||
extern int is_submodule_populated_gently(const char *path, int *return_error_code);
|
extern int is_submodule_populated_gently(const char *path, int *return_error_code);
|
||||||
|
extern void die_in_unpopulated_submodule(const struct index_state *istate,
|
||||||
|
const char *prefix);
|
||||||
|
extern void die_path_inside_submodule(const struct index_state *istate,
|
||||||
|
const struct pathspec *ps);
|
||||||
extern int parse_submodule_update_strategy(const char *value,
|
extern int parse_submodule_update_strategy(const char *value,
|
||||||
struct submodule_update_strategy *dst);
|
struct submodule_update_strategy *dst);
|
||||||
extern const char *submodule_strategy_to_string(const struct submodule_update_strategy *s);
|
extern const char *submodule_strategy_to_string(const struct submodule_update_strategy *s);
|
||||||
|
@ -24,13 +24,9 @@ test_expect_success 'error message for path inside submodule' '
|
|||||||
test_i18ncmp expect actual
|
test_i18ncmp expect actual
|
||||||
'
|
'
|
||||||
|
|
||||||
cat <<EOF >expect
|
|
||||||
fatal: Pathspec '.' is in submodule 'sub'
|
|
||||||
EOF
|
|
||||||
|
|
||||||
test_expect_success 'error message for path inside submodule from within submodule' '
|
test_expect_success 'error message for path inside submodule from within submodule' '
|
||||||
test_must_fail git -C sub add . 2>actual &&
|
test_must_fail git -C sub add . 2>actual &&
|
||||||
test_i18ncmp expect actual
|
test_i18ngrep "in unpopulated submodule" actual
|
||||||
'
|
'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
Loading…
Reference in New Issue
Block a user