Is git stash stack pushed to the remote repo?

前端 未结 2 2009
深忆病人
深忆病人 2020-12-28 12:57

Is my stash stack pushed to the remote repo? Or is it completely ignored?

I\'m just curious if I should tend to it every once in a while t

相关标签:
2条回答
  • 2020-12-28 13:10

    As a general rule, no. You could push it though, if you wanted.

    Here's the thing about push (and, for that matter, fetch): these work based on "refspecs", where you name a local reference name—or, for push and specific cases, a raw commit-ID—and then also a remote reference name.

    So what the heck is a reference name?

    Most of the time, you name a branch reference, such as master, or a "remote branch" like origin/master. Confusingly, what git calls a "remote branch" is actually a local entity, not a branch on a remote but rather a branch in your repository under a special name.

    A branch is really just a reference whose name starts with refs/heads/. That's really almost all a branch is. (There's one other special thing about a branch: branches automatically move when you make new commits in them. That is, if you're on your master branch and you make a new commit, git updates refs/heads/master to point to the new commit.)

    A "remote branch" is just a reference whose name starts with refs/remotes/ and then contains the name of the remote (usually origin). So refs/remotes/origin/master is a "remote branch": a local entity, a name in your repo, that your git uses to keep track of "where master was on origin last time git checked in with origin". Whenever you fetch or push from/to origin,1 git updates your origin/branch names based on what it sees "over there".

    Tags are simply references that start with refs/tags/.

    Normally you leave out all these prefixes and just write master to mean your master branch in refs/heads/master, origin/master to mean refs/remotes/origin/master, and v2.3 to mean refs/tags/v2.3. Git figures out which one automatically, since it's obvious. The precise rules are described in gitrevisions. (Annoyingly, git checkout and git branch don't always exactly follow the gitrevisions rules: when they know something is meant to be a branch name they just assume the refs/heads/ part. Other git commands do work as described, though.)

    The stash script uses a reference that is simply spelled refs/stash. So you can name this reference by writing stash; git will find that it's not a branch, nor a remote branch, nor a tag, and will finally resort to using refs/stash to resolve the name.2

    ... which means, what?

    This means that if you wrote:

    $ git push origin stash:ssss
    

    git would find your stash (just the one single stash) and attempt to push it to a remote reference named ssss. This would most likely fail; here's what happened when I tried it:

    error: unable to push to unqualified destination: ssss
    The destination refspec neither matches an existing ref on the remote nor
    begins with refs/, and we are unable to guess a prefix based on the source ref.
    error: failed to push some refs to 'ssh://[redacted]/tmp/t'
    

    Just for fun, I tried the next command, and got a different error:

    $ git push origin stash:refs/ssss
    Counting objects: 8, done.
    Delta compression using up to 8 threads.
    Compressing objects: 100% (4/4), done.
    Writing objects: 100% (4/4), 485 bytes | 0 bytes/s, done.
    Total 4 (delta 3), reused 0 (delta 0)
    remote: error: refusing to create funny ref 'refs/ssss' remotely
    To ssh://[redacted]/tmp/t
     ! [remote rejected] stash -> refs/ssss (funny refname)
    error: failed to push some refs to 'ssh://[redacted]/tmp/t'
    

    However, this worked:

    $ git push origin stash:refs/heads/ssss
    

    This time, it created a new branch on the remote, named ssss. It would also work to push to a remote tag (refs/tags/tagname).

    A stash does not make much sense as either a branch or a tag, but you can push it as if it were one. All the remote knows, once you try this, is that you're sending some commit object(s) and their associated trees, blobs, etc., and that it should store the "tip-most" commit under the (full) name you supply.3 This is why you can push raw commit-IDs, e.g., to create a tag on the remote:

    $ git push af7ec93:refs/tags/foo
    

    (assuming there's a commit whose ID starts with af7ec93 of course).

    One other note about the "stash stack"

    The "stack" is made by using the reflog for the refs/stash reference. When you write stash@{1}, this uses the same gitrevisions rules to resolve to a commit-ID. Since git push allows you to use anything that resolves to a valid ID, you could push these to remote names as well. But you probably shouldn't; just like the top-most stash item under refs/stash, these do not make a lot of sense as regular commits. (Stashes are stored internally as merge commits, but their contents are oddly packaged, and trying to use them as ordinary commits produces not-so-useful results.)


    1This is literally true in git since 1.8.2 or so; but older versions of git sometimes skip updating a remote-branch when fetching, particularly when git fetch is invoked by git pull.

    2If you consult gitrevisions, you'll see that this is not quite accurate either: it will find stash as refs/stash before looking for a branch name.

    3git push usually builds the full name automatically: if you're pushing your master, which is refs/heads/master, it knows that this is a branch, so git push master:newbranch creates refs/heads/newbranch on the remote. However, you can spell out a full name, so as to create a tag on the remote instead, for instance. This also works for deletion operations, when you push an empty local reference to delete a remote-ref: git push :refs/tags/delete_me, for example.


    As bcmcfc already said, keeping a large pile of stashes tends to be a bad idea. What I do instead is keep a large pile of branches (which I avoid pushing unless I really want to save them on some remote(s)); these have names, and hence are a little more manageable (but only a little). There's no difference in efficiency, since stashes and branches both merely serve to contain commits—either the special stash merge commits, or ordinary commits. However, you may want to make sure your push.default is configured to avoid pushing all your branches, if you use branches instead of stashes.

    0 讨论(0)
  • 2020-12-28 13:21

    No. Stashes are local.

    $ man git stash:

    Use git stash when you want to record the current state of the working directory and the index, but want to go back to a clean working directory. The command saves your local modifications away reverts the working directory to match the HEAD commit.

    I wouldn't keep too many of them around locally though. You'll lose track of them over time and they'll become somewhat useless.

    0 讨论(0)
提交回复
热议问题