259

Is there a way to execute Git commands against a repository without being in that repository?

For example something like this: git /home/repo log?

Please do not tell me to cd to it. I'm doing this via an exec call.

6

6 Answers 6

425

Use -C as the first argument to git:

git -C /home/repo log

Per the docs, the effect of this is:

-C <path>

Run as if git was started in <path> instead of the current working directory. ...

This is almost equivalent to --git-dir and --work-tree without appending the usual .git folder. However, the options --git-dir and --work-tree do not exist to access the repository from outside the work tree; they are used to move the .git somewhere else, and they are much more complicated to use in some cases.

For instance, to get the log of /home/repo/subdir only:

git -C /home/repo/subdir log .

or

git -C /home/repo log subdir

It is not possible to use log . with --git-dir or --work-tree. The path must be processed to extract the subpath relative to the top of the work tree, and even in that case, git will not recognize it as a path if you do not use the -- option, so the only possible way is:

git --git-dir /home/repo/.git log -- subdir

Furthermore, --work-tree does not work at all with the log subcommand with my version (git 1.9.1). It is just ignored:

git --git-dir /home/repo/.git --work-tree /home/repo/subdir log -- subdir
git --git-dir /home/repo/.git --work-tree /home/repo/whatever log -- subdir

I do not even understand if this is a bug or a feature... as usual with many git design choices.

Sign up to request clarification or add additional context in comments.

7 Comments

git stash is another example of an subcommand which fails if --git-dir and --work-tree are used.
The -C option was added in version 1.8.5 of git, which was released in 2013.
-C worked better than --git-dir for me ... -C honours git config settings for the repo. Am running git version 1.9.5.msysgit.1
Image
@calandoa do we always need to specify the location from the root , one more thing if the name of the repo contain multiple words how do we put in in this code
Note: you need to pass the -C option to git and not to the subcommand (as correctly done by @calandoa in this answer). Calling git -C /some/dir add . would work as expected while git add -C /some/dir . would throw an error. Just something to be aware of to prevent confusing results.
|
118

Try:

git --git-dir=/home/repo/.git log

It is important to give the path all the way up to the .git directory of your repository. Otherwise you will get only an error message that says something like:

fatal: Not a git repository

4 Comments

This didn't work for me, since it showed all files of the commit as deleted. Seems like it is checking the commit but not the folder contents. Refer to @calandoa 's answer for better performance.
Didn't work for me using the latest git, however using the -C option it worked, example: git -C /home/repo/.git log
this actually did not work as the po expected. why this is any accepted answer? This will use the current dir as work tree and use the specified .git for history, which is completely wrong. If you do not just show log, for example, git status. -C is the correct way to go.
It does not work as people in previous comments already said
18

In fact you need to use --git-dir and --work-tree together. Here is an example:

local [] Desktop: mkdir git
local [] Desktop: cd git
local [] git: touch README.txt
local [] git: git init
Initialized empty Git repository in /Users/albert/Desktop/git/.git/
local [] git: cd ..
local [] Desktop: git --work-tree=git --git-dir=git/.git add .
local [] Desktop: git --work-tree=git --git-dir=git/.git commit -a -m 'initial commit, called from outside the git directory'
[master (root-commit) ee951b1] initial commit, called from outside the git directory
0 files changed, 0 insertions(+), 0 deletions(-)
create mode 100644 README.txt
local [] Desktop: cd git
local [] git: git log --pretty=oneline
ee951b161053e0e0948f9e2a36bfbb60f9c87abe initial commit, called from outside the git di

1 Comment

This is more correct than the answer with --git-dir alone. git-scm.com/blog/2010/04/11/environment.html
13

I have tried many times! I finally got it!

git -C dir --no-pager log --format='%an' -1 filename

do remember, please don't add .git to your

-C dir

Comments

7

For any git command, you can do:

git --git-dir=<PATH-TO-REPO>/.git --work-tree=<PATH-TO-REPO> <git-command>

For example, if you want to do a git status:

 git --git-dir=/home/myrepo/.git --work-tree=/home/myrepo/  status

or if you want to check the branch the repo is in:

git --git-dir=/home/myrepo/.git --work-tree=/home/myrepo/  rev-parse --abbrev-ref HEAD

3 Comments

hey google users, stackoverflow.com/a/35899275/1277898 is the better answer ;)
@Git.Coach, if you have v1.8.5 or higher.
Even 2 years, this answer still being the best one! Thanks
3

This is similar to @max's answer. Unfortunately, --git-dir didn't do what I needed. edit In retrospect, another [previous] answer I hadn't read suggested using --work-tree. I'm not sure whether using environment or flags is more appropriate so I'll leave my answer in case someone finds use in it, but I'll be switching to use --work-tree / --git-dir.


There's two repos, one inside the other but not a submodule; the outer repo ignores it:

tools (outer repo)
  gcc/4.9.2 (inner repo)

What I wanted was the output of git rev-parse --show-prefix relative to the outer repo. Here's what I came up with (bash syntax; split across lines for readability):

prefix_dir=$(cd ..; git rev-parse --show-toplevel)
prefix_dir=$(GIT_WORK_TREE=${prefix_dir} git rev-parse --show-prefix)

When executed from within 4.9.2 it produces the string gcc/4.9.2.

Comments

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.