This parameter allows the branchname validation functions to optionally return a flag specifying the reason for failure, when requested. This allows the caller to know why it was about to die. This allows more useful error messages to be given to the user when trying to rename a branch. The flags are specified in the form of an enum and values for success flags have been assigned explicitly to clearly express that certain callers rely on those values and they cannot be arbitrary. Only the logic has been added but no caller has been made to use it, yet. So, no functional changes. Signed-off-by: Kaartic Sivaraam <kaartic.sivaraam@xxxxxxxxx> --- branch.c | 59 ++++++++++++++++++++++++-------------------- branch.h | 61 +++++++++++++++++++++++++++++++++++++--------- builtin/branch.c | 4 +-- builtin/checkout.c | 5 ++-- 4 files changed, 88 insertions(+), 41 deletions(-) diff --git a/branch.c b/branch.c index 2672054f0..0eeb8403e 100644 --- a/branch.c +++ b/branch.c @@ -178,41 +178,48 @@ int read_branch_desc(struct strbuf *buf, const char *branch_name) return 0; } -/* - * Check if 'name' can be a valid name for a branch; die otherwise. - * Return 1 if the named branch already exists; return 0 otherwise. - * Fill ref with the full refname for the branch. - */ -int validate_branchname(const char *name, struct strbuf *ref) +enum branch_validation_result validate_branchname(const char *name, struct strbuf *ref, unsigned gently) { - if (strbuf_check_branch_ref(ref, name)) - die(_("'%s' is not a valid branch name."), name); + if (strbuf_check_branch_ref(ref, name)) { + if (gently) + return VALIDATION_FATAL_INVALID_BRANCH_NAME; + else + die(_("'%s' is not a valid branch name."), name); + } - return ref_exists(ref->buf); + return ref_exists(ref->buf) ? VALIDATION_PASS_BRANCH_EXISTS : VALIDATION_PASS_BRANCH_DOESNT_EXIST; } -/* - * Check if a branch 'name' can be created as a new branch; die otherwise. - * 'force' can be used when it is OK for the named branch already exists. - * Return 1 if the named branch already exists; return 0 otherwise. - * Fill ref with the full refname for the branch. - */ -int validate_new_branchname(const char *name, struct strbuf *ref, int force) +enum branch_validation_result validate_new_branchname(const char *name, struct strbuf *ref, int force, unsigned gently) { const char *head; - if (!validate_branchname(name, ref)) - return 0; + if (gently) { + enum branch_validation_result res = validate_branchname(name, ref, 1); + if (res == VALIDATION_FATAL_INVALID_BRANCH_NAME || res == VALIDATION_PASS_BRANCH_DOESNT_EXIST) + return res; + } else { + if (validate_branchname(name, ref, 0) == VALIDATION_PASS_BRANCH_DOESNT_EXIST) + return VALIDATION_PASS_BRANCH_DOESNT_EXIST; + } - if (!force) - die(_("A branch named '%s' already exists."), - ref->buf + strlen("refs/heads/")); + if (!force) { + if (gently) + return VALIDATION_FATAL_BRANCH_EXISTS_NO_FORCE; + else + die(_("A branch named '%s' already exists."), + ref->buf + strlen("refs/heads/")); + } head = resolve_ref_unsafe("HEAD", 0, NULL, NULL); - if (!is_bare_repository() && head && !strcmp(head, ref->buf)) - die(_("Cannot force update the current branch.")); + if (!is_bare_repository() && head && !strcmp(head, ref->buf)) { + if (gently) + return VALIDATION_FATAL_CANNOT_FORCE_UPDATE_CURRENT_BRANCH; + else + die(_("Cannot force update the current branch.")); + } - return 1; + return VALIDATION_WARN_BRANCH_EXISTS; } static int check_tracking_branch(struct remote *remote, void *cb_data) @@ -259,8 +266,8 @@ void create_branch(const char *name, const char *start_name, explicit_tracking = 1; if ((track == BRANCH_TRACK_OVERRIDE || clobber_head_ok) - ? validate_branchname(name, &ref) - : validate_new_branchname(name, &ref, force)) { + ? validate_branchname(name, &ref, 0) + : validate_new_branchname(name, &ref, force, 0)) { if (!force) dont_change_ref = 1; else diff --git a/branch.h b/branch.h index 473d0a93e..ee5f1c0e7 100644 --- a/branch.h +++ b/branch.h @@ -28,20 +28,59 @@ void create_branch(const char *name, const char *start_name, int force, int clobber_head_ok, int reflog, int quiet, enum branch_track track); -/* - * Check if 'name' can be a valid name for a branch; die otherwise. - * Return 1 if the named branch already exists; return 0 otherwise. - * Fill ref with the full refname for the branch. - */ -extern int validate_branchname(const char *name, struct strbuf *ref); +enum branch_validation_result { + /* Flags that convey there are fatal errors */ + VALIDATION_FATAL_BRANCH_EXISTS_NO_FORCE = -3, + VALIDATION_FATAL_CANNOT_FORCE_UPDATE_CURRENT_BRANCH, + VALIDATION_FATAL_INVALID_BRANCH_NAME, + /* Flags that convey there are no fatal errors */ + VALIDATION_PASS_BRANCH_DOESNT_EXIST = 0, + VALIDATION_PASS_BRANCH_EXISTS = 1, + VALIDATION_WARN_BRANCH_EXISTS = 2 +}; /* - * Check if a branch 'name' can be created as a new branch; die otherwise. - * 'force' can be used when it is OK for the named branch already exists. - * Return 1 if the named branch already exists; return 0 otherwise. - * Fill ref with the full refname for the branch. + * Check if 'name' can be a valid name for a branch; die otherwise. + * + * - name is the new branch name + * + * - ref is used to return the full refname for the branch + * + * The return values have the following meaning, + * + * - If 'gently' is 0, the function dies in case of a fatal error and returns + * flags of 'branch_validation_result' that indicate nonfatal cases, otherwise. + * The positive non-zero flag implies that the branch exists. + * + * - If 'gently' is 1, the function doesn't die in case of a fatal error but returns + * flags of 'branch_validaton_result' that identify the fatal error. The behaviour + * in case of success is same as above. + * */ -extern int validate_new_branchname(const char *name, struct strbuf *ref, int force); +extern enum branch_validation_result validate_branchname(const char *name, struct strbuf *ref, unsigned gently); + +/* + * Check if a branch 'name' can be created as a new branch. + * + * - name is the new branch name + * + * - ref is used to return the full refname for the branch + * + * - force can be used when it is OK if the named branch already exists. + * the currently checkout branch; with 'shouldnt_exist', it has no effect. + * + * The return values have the following meaning, + * + * - If 'gently' is 0, the function dies in case of a fatal error and returns + * flags of 'branch_validation_result' that indicate nonfatal cases, otherwise. + * The positive non-zero flag implies that the branch can be force updated. + * + * - If 'gently' is 1, the function doesn't die in case of a fatal error but returns + * flags of 'branch_validaton_result' that identify the fatal error. The behaviour + * in case of success is same as above. + * + */ +extern enum branch_validation_result validate_new_branchname(const char *name, struct strbuf *ref, int force, unsigned gently); /* * Remove information about the state of working on the current diff --git a/builtin/branch.c b/builtin/branch.c index 8dcc2ed05..5412aa78f 100644 --- a/builtin/branch.c +++ b/builtin/branch.c @@ -489,9 +489,9 @@ static void copy_or_rename_branch(const char *oldname, const char *newname, int * cause the worktree to become inconsistent with HEAD, so allow it. */ if (!strcmp(oldname, newname)) - validate_branchname(newname, &newref); + validate_branchname(newname, &newref, 0); else - validate_new_branchname(newname, &newref, force); + validate_new_branchname(newname, &newref, force, 0); reject_rebase_or_bisect_branch(oldref.buf); diff --git a/builtin/checkout.c b/builtin/checkout.c index 8f4dfb104..c16455ff0 100644 --- a/builtin/checkout.c +++ b/builtin/checkout.c @@ -1243,10 +1243,11 @@ int cmd_checkout(int argc, const char **argv, const char *prefix) struct strbuf buf = STRBUF_INIT; if (opts.new_branch_force) - opts.branch_exists = validate_branchname(opts.new_branch, &buf); + opts.branch_exists = validate_branchname(opts.new_branch, &buf, 0); else opts.branch_exists = - validate_new_branchname(opts.new_branch, &buf, 0); + validate_new_branchname(opts.new_branch, &buf, 0, 0); + strbuf_release(&buf); } -- 2.16.1.291.g4437f3f13