前言

今天有个同事突然问我:怎么创建一个本地的远程仓库,并且还要能看到仓库文件,能够push的。。
大家都知道Git仓库是分布式的,也可以这么理解每一个本地仓库同时也能是一个远程仓库。
既然这样,我们直接建一个目录,然后创建一个本地仓库就可以了。然后再到另外一个目录进行clone操作就相当于是有远程仓库了。
想法是好的,然而要完整的实现他的需求还是有一些坑要走的,下面来进行具体的操作吧。

创建远程仓库

上面已经把基本概念说了,在这里创建远程仓库其实也就是创建本地仓库。用过git的朋友都知道这个,找到一个地方创建一个空目录,执行以下操作:

lu@E5:~/Temp/git_server/Test$ git init
Initialized empty Git repository in /home/lu/Temp/git_server/Test/.git/

或者创建一个裸仓库

lu@E5:~/Temp/git_server/Test$ git init --bare
Initialized empty Git repository in /home/lu/Temp/git_server/Test/

裸仓库一般是无法直接看到源码的,所以也就不符合需求了,这里就不作过多详述。

关联远程仓库

在创建远程仓库之前,项目其实已经有一些文件了,所以可以直接在他那个项目中再创建一个本地仓库,并把文件提交到本地仓库,最后再push到远程仓库即可,下面来模拟那个环境一步步操作。

  • 创建一个目录并添加一些文件
lu@E5:~/Temp$ mkdir MyTest
lu@E5:~/Temp$ cd MyTest/
lu@E5:~/Temp/MyTest$ echo "test" > test
lu@E5:~/Temp/MyTest$ echo "hello" > hello
  • 创建本地仓库并将文件都提交到仓库
lu@E5:~/Temp/MyTest$ git init
Initialized empty Git repository in /home/lu/Temp/MyTest/.git/

lu@E5:~/Temp/MyTest$ git add .

lu@E5:~/Temp/MyTest$ git commit -m "init"
[master (root-commit) 7be5b2a] init
 2 files changed, 2 insertions(+)
 create mode 100644 hello
 create mode 100644 test
  • 将本地仓库与远程仓库关联
    在这里关联远程(本地)仓库与一般我们关联互联网上的远程仓库有所不同,我们可以直接写本地的仓库路径即可。
lu@E5:~/Temp/MyTest$ git remote add origin ~/Temp/git_server/Test/
  • 提交代码到远程仓库
    注意坑来了,默认情况下直接push到远程(本地)仓库会出错的,如下:
lu@E5:~/Temp/MyTest$ git push -u origin master
Counting objects: 4, done.
Delta compression using up to 16 threads.
Compressing objects: 100% (2/2), done.
Writing objects: 100% (4/4), 245 bytes | 0 bytes/s, done.
Total 4 (delta 0), reused 0 (delta 0)
remote: error: refusing to update checked out branch: refs/heads/master
remote: error: By default, updating the current branch in a non-bare repository
remote: error: is denied, because it will make the index and work tree inconsistent
remote: error: with what you pushed, and will require 'git reset --hard' to match
remote: error: the work tree to HEAD.
remote: error: 
remote: error: You can set 'receive.denyCurrentBranch' configuration variable to
remote: error: 'ignore' or 'warn' in the remote repository to allow pushing into
remote: error: its current branch; however, this is not recommended unless you
remote: error: arranged to update its work tree to match what you pushed in some
remote: error: other way.
remote: error: 
remote: error: To squelch this message and still keep the default behaviour, set
remote: error: 'receive.denyCurrentBranch' configuration variable to 'refuse'.
To /home/lu/Temp/git_server/Test/
 ! [remote rejected] master -> master (branch is currently checked out)
error: failed to push some refs to '/home/lu/Temp/git_server/Test/

因为我们的远程仓库也只是一般的仓库而不是裸仓库,如果是裸仓库的话就不会出现这个问题了。
根据错误提示,我们可以进行相应的配置:

lu@E5:~/Temp/MyTest$ git config --global receive.denyCurrentBranch ignore

也可以直接修改配置文件:

lu@E5:~/Temp/MyTest$ vim ~/.gitconfig
在文件中添加以下内容即可
[receive]
    denyCurrentBranch = ignore

配置完后,我们再试着push一次看看。

lu@E5:~/Temp/MyTest$ git push -u origin master
Counting objects: 4, done.
Delta compression using up to 16 threads.
Compressing objects: 100% (2/2), done.
Writing objects: 100% (4/4), 245 bytes | 0 bytes/s, done.
Total 4 (delta 0), reused 0 (delta 0)
To /home/lu/Temp/git_server/Test/
 * [new branch]      master -> master
Branch master set up to track remote branch master from origin.

终于成功了?还早呢,虽然代码是提交上去了,但是我们再到远程仓库看看有木有文件吧。

lu@E5:~/Temp/MyTest$ cd ../git_server/Test/
lu@E5:~/Temp/git_server/Test$ ls -al
total 12
drwxr-xr-x 3 lu lu 4096 Oct 20 19:53 .
drwxr-xr-x 3 lu lu 4096 Oct 20 19:26 ..
drwxr-xr-x 8 lu lu 4096 Oct 20 20:05 .git

怎么回事呢,我们提交的2个文件并没有看到呢。我们在远程仓库看看状态和log记录。

lu@E5:~/Temp/git_server/Test$ git status
On branch master
Changes to be committed:
  (use "git reset HEAD <file>..." to unstage)

    deleted:    hello
    deleted:    test

lu@E5:~/Temp/git_server/Test$ git log
commit 7be5b2ab70e3a7aa14089e432a4066c935241f30
Author: lusyoe <lusyoe@163.com>
Date:   Thu Oct 20 19:57:26 2016 +0800

    init

可以看到文件是有提交到远程仓库的,只是还没有checkout 出来而已。那如何每次提交到远程仓库都使其自动checkout出文件来呢?这里就需要使用到git hook了,具体的概念这里不多讲,直接说使用吧。

添加git hook

git hook是一个非常强大的功能,它可以监听我们对仓库的修改,继而触发相应的逻辑命令。这里为了完成同事的需求,只说说post-update,就是提交有代码更新到仓库时需要执行的操作。

lu@E5:~/Temp/git_server/Test$ cd .git/hooks/

lu@E5:~/Temp/git_server/Test/.git/hooks$ ls
applypatch-msg.sample  post-update.sample     pre-commit.sample          pre-push.sample    update.sample
commit-msg.sample      pre-applypatch.sample  prepare-commit-msg.sample  pre-rebase.sample

可以看到默认有许多的hook脚本,我们只需要关注post-update.sample即可,下面来根据需求进行操作。

  • 重命名脚本
    首先重命名脚本,因为默认.sample结尾的是不会执行的,名字也不能乱改。
lu@E5:~/Temp/git_server/Test/.git/hooks$ mv post-update.sample post-update
  • 编辑脚本
    我们注释掉默认的操作指令exec git update-server-info,然后输入我们想要执行的操作命令。
lu@E5:~/Temp/git_server/Test/.git/hooks$ vim post-update
  #!/bin/sh
  #
  # An example hook script to prepare a packed repository for use over
  # dumb transports.
  #
  # To enable this hook, rename this file to "post-update".

  #exec git update-server-info
  # 添加以下三行即可
  unset GIT_DIR
  cd ..
  git checkout -f

好了,远程仓库修改完毕。

再次push远程仓库

我们再到本地仓库那添加一个文件然后push到远程仓库看看。

lu@E5:~/Temp/MyTest$ echo "demo" > demo

lu@E5:~/Temp/MyTest$ git add .

lu@E5:~/Temp/MyTest$ git commit -m "add demo"
[master 9da7116] add demo
 1 file changed, 1 insertion(+)
 create mode 100644 demo

lu@E5:~/Temp/MyTest$ git push
Counting objects: 3, done.
Delta compression using up to 16 threads.
Compressing objects: 100% (2/2), done.
Writing objects: 100% (3/3), 288 bytes | 0 bytes/s, done.
Total 3 (delta 0), reused 0 (delta 0)
To /home/lu/Temp/git_server/Test/
   7be5b2a..9da7116  master -> master

去远程仓库看看是否有出现新文件。

lu@E5:~/Temp/git_server/Test$ ls -al
total 24
drwxr-xr-x 3 lu lu 4096 Oct 20 20:33 .
drwxr-xr-x 3 lu lu 4096 Oct 20 19:26 ..
-rw-r--r-- 1 lu lu    5 Oct 20 20:33 demo
drwxr-xr-x 8 lu lu 4096 Oct 20 20:33 .git
-rw-r--r-- 1 lu lu   18 Oct 20 20:33 hello
-rw-r--r-- 1 lu lu    5 Oct 20 20:29 test

后序

OK,非常成功,虽然像这种场景很少遇到,不过通过这一番操作能对git又多一点了解了。不得不说git对于开发人员来说实在是神器啊。