Jeff King <p...@peff.net> writes:

> In a triangular workflow, you may have a distinct
> @{upstream} that you pull changes from, but publish by
> default (if you typed "git push") to a different remote (or
> a different branch on the remote). It may sometimes be
> useful to be able to quickly refer to that publishing point
> (e.g., to see which changes you have that have not yet been
> published).
>
> This patch introduces the <branch>@{publish} shorthand (or
> "@{pu}" to be even shorter). It refers to the tracking

If @{u} can already be used for upstream, why not allow @{p} but
require two letters @{pu}?  Just being curious---I am not advocating
strongly for a shorter short-hand.

Or is @{p} already taken by something and my memory is not
functioning well?

> branch of the remote branch to which you would push if you
> were to push the named branch. That's a mouthful to explain,
> so here's an example:
>
>   $ git checkout -b foo origin/master
>   $ git config remote.pushdefault github
>   $ git push
>
> Signed-off-by: Jeff King <p...@peff.net>
> ---
> The implementation feels weird, like the "where do we push to" code
> should be factored out from somewhere else. I think what we're doing
> here is not _wrong_, but I don't like repeating what "git push" is doing
> elsewhere. And I just punt on "simple" as a result. :)
>
>  sha1_name.c | 76 
> ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++-
>  1 file changed, 75 insertions(+), 1 deletion(-)
>
> diff --git a/sha1_name.c b/sha1_name.c
> index 50df5d4..59ffa93 100644
> --- a/sha1_name.c
> +++ b/sha1_name.c
> @@ -435,6 +435,12 @@ static inline int upstream_mark(const char *string, int 
> len)
>       return at_mark(string, len, suffix, ARRAY_SIZE(suffix));
>  }
>  
> +static inline int publish_mark(const char *string, int len)
> +{
> +     const char *suffix[] = { "@{publish}" };
> +     return at_mark(string, len, suffix, ARRAY_SIZE(suffix));
> +}
> +
>  static int get_sha1_1(const char *name, int len, unsigned char *sha1, 
> unsigned lookup_flags);
>  static int interpret_nth_prior_checkout(const char *name, struct strbuf 
> *buf);
>  
> @@ -481,7 +487,8 @@ static int get_sha1_basic(const char *str, int len, 
> unsigned char *sha1)
>                                       nth_prior = 1;
>                                       continue;
>                               }
> -                             if (!upstream_mark(str + at, len - at)) {
> +                             if (!upstream_mark(str + at, len - at) &&
> +                                 !publish_mark(str + at, len - at)) {
>                                       reflog_len = (len-1) - (at+2);
>                                       len = at;
>                               }
> @@ -1100,6 +1107,69 @@ static int interpret_upstream_mark(const char *name, 
> int namelen,
>       return len + at;
>  }
>  
> +static const char *get_publish_branch(const char *name_buf, int len)
> +{
> +     char *name = xstrndup(name_buf, len);
> +     struct branch *b = branch_get(*name ? name : NULL);
> +     struct remote *remote = b->pushremote;
> +     const char *dst;
> +     const char *track;
> +
> +     free(name);
> +
> +     if (!remote)
> +             die(_("branch '%s' has no remote for pushing"), b->name);
> +
> +     /* Figure out what we would call it on the remote side... */
> +     if (remote->push_refspec_nr)
> +             dst = apply_refspecs(remote->push, remote->push_refspec_nr,
> +                                  b->refname);
> +     else
> +             dst = b->refname;
> +     if (!dst)
> +             die(_("unable to figure out how '%s' would be pushed"),
> +                 b->name);
> +
> +     /* ...and then figure out what we would call that remote here */
> +     track = apply_refspecs(remote->fetch, remote->fetch_refspec_nr, dst);
> +     if (!track)
> +             die(_("%s@{publish} has no tracking branch for '%s'"),
> +                 b->name, dst);
> +
> +     return track;
> +}
> +
> +static int interpret_publish_mark(const char *name, int namelen,
> +                               int at, struct strbuf *buf)
> +{
> +     int len;
> +
> +     len = publish_mark(name + at, namelen - at);
> +     if (!len)
> +             return -1;
> +
> +     switch (push_default) {
> +     case PUSH_DEFAULT_NOTHING:
> +             die(_("cannot use @{publish} with push.default of 'nothing'"));
> +
> +     case PUSH_DEFAULT_UNSPECIFIED:
> +     case PUSH_DEFAULT_MATCHING:
> +     case PUSH_DEFAULT_CURRENT:
> +             set_shortened_ref(buf, get_publish_branch(name, at));
> +             break;
> +
> +     case PUSH_DEFAULT_UPSTREAM:
> +             set_shortened_ref(buf, get_upstream_branch(name, at));
> +             break;
> +
> +     case PUSH_DEFAULT_SIMPLE:
> +             /* ??? */
> +             die("@{publish} with simple unimplemented");
> +     }
> +
> +     return at + len;
> +}
> +
>  /*
>   * This reads short-hand syntax that not only evaluates to a commit
>   * object name, but also can act as if the end user spelled the name
> @@ -1150,6 +1220,10 @@ int interpret_branch_name(const char *name, int 
> namelen, struct strbuf *buf)
>       if (len > 0)
>               return len;
>  
> +     len = interpret_publish_mark(name, namelen, cp - name, buf);
> +     if (len > 0)
> +             return len;
> +
>       return -1;
>  }
--
To unsubscribe from this list: send the line "unsubscribe git" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to