Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
caddy
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
caddy
Commits
9df9ad97
Commit
9df9ad97
authored
May 01, 2015
by
Matt Holt
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #37 from abiosoft/master
git: post pull command. retries after pull failure.
parents
782ba324
9cd1587c
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
138 additions
and
54 deletions
+138
-54
middleware/git/doc.go
middleware/git/doc.go
+5
-2
middleware/git/git.go
middleware/git/git.go
+18
-3
middleware/git/gitclient.go
middleware/git/gitclient.go
+115
-49
No files found.
middleware/git/doc.go
View file @
9df9ad97
...
...
@@ -7,6 +7,7 @@
// branch
// key
// interval
// then command args
// }
// repo - git repository
// compulsory. Both ssh (e.g. git@github.com:user/project.git)
...
...
@@ -15,7 +16,6 @@
//
// path - directory to pull into, relative to site root
// optional. Defaults to site root.
// If set, must be a subdirectory to site root to be valid.
//
// branch - git branch or tag
// optional. Defaults to master
...
...
@@ -26,6 +26,9 @@
// interval- interval between git pulls in seconds
// optional. Defaults to 3600 (1 Hour).
//
// then - command to execute after successful pull
// optional. If set, will execute only when there are new changes.
//
// Examples :
//
// public repo pulled into site root
...
...
@@ -34,7 +37,7 @@
// public repo pulled into <root>/mysite
// git https://github.com/user/myproject mysite
//
// private repo pulled into <root>/mysite with tag v1.0 and interval of 1 day
// private repo pulled into <root>/mysite with tag v1.0 and interval of 1 day
.
// git {
// repo git@github.com:user/myproject
// branch v1.0
...
...
middleware/git/git.go
View file @
9df9ad97
...
...
@@ -4,6 +4,7 @@ import (
"fmt"
"log"
"net/url"
"os"
"path/filepath"
"runtime"
"strconv"
...
...
@@ -56,7 +57,7 @@ func parse(c middleware.Controller) (*Repo, error) {
switch
len
(
args
)
{
case
2
:
repo
.
Path
=
filepath
.
Join
(
c
.
Root
(),
args
[
1
])
repo
.
Path
=
filepath
.
Clean
(
c
.
Root
()
+
string
(
filepath
.
Separator
)
+
args
[
1
])
fallthrough
case
1
:
repo
.
Url
=
args
[
0
]
...
...
@@ -73,7 +74,7 @@ func parse(c middleware.Controller) (*Repo, error) {
if
!
c
.
NextArg
()
{
return
nil
,
c
.
ArgErr
()
}
repo
.
Path
=
filepath
.
Join
(
c
.
Root
(),
c
.
Val
())
repo
.
Path
=
filepath
.
Clean
(
c
.
Root
()
+
string
(
filepath
.
Separator
)
+
c
.
Val
())
case
"branch"
:
if
!
c
.
NextArg
()
{
return
nil
,
c
.
ArgErr
()
...
...
@@ -92,6 +93,12 @@ func parse(c middleware.Controller) (*Repo, error) {
if
t
>
0
{
repo
.
Interval
=
time
.
Duration
(
t
)
*
time
.
Second
}
case
"then"
:
thenArgs
:=
c
.
RemainingArgs
()
if
len
(
thenArgs
)
==
0
{
return
nil
,
c
.
ArgErr
()
}
repo
.
Then
=
strings
.
Join
(
thenArgs
,
" "
)
}
}
}
...
...
@@ -125,7 +132,7 @@ func parse(c middleware.Controller) (*Repo, error) {
return
nil
,
err
}
return
repo
,
prepare
(
repo
)
return
repo
,
repo
.
prepare
(
)
}
// sanitizeHttp cleans up repository url and converts to https format
...
...
@@ -165,3 +172,11 @@ func sanitizeGit(repoUrl string) (string, string, error) {
host
:=
hostUrl
[
:
i
]
return
repoUrl
,
host
,
nil
}
// logger is an helper function to retrieve the available logger
func
logger
()
*
log
.
Logger
{
if
Logger
==
nil
{
Logger
=
log
.
New
(
os
.
Stderr
,
""
,
log
.
LstdFlags
)
}
return
Logger
}
middleware/git/gitclient.go
View file @
9df9ad97
package
git
import
(
"bytes"
"fmt"
"io/ioutil"
"log"
"os"
"os/exec"
"strings"
"sync"
"time"
"github.com/mholt/caddy/middleware"
)
// DefaultInterval is the minimum interval to delay before
// requesting another git pull
const
DefaultInterval
time
.
Duration
=
time
.
Hour
*
1
// Number of retries if git pull fails
const
numRetries
=
3
// gitBinary holds the absolute path to git executable
var
gitBinary
string
...
...
@@ -25,18 +30,21 @@ var initMutex sync.Mutex = sync.Mutex{}
// Repo is the structure that holds required information
// of a git repository.
type
Repo
struct
{
Url
string
// Repository URL
Path
string
// Directory to pull to
Host
string
// Git domain host e.g. github.com
Branch
string
// Git branch
KeyPath
string
// Path to private ssh key
Interval
time
.
Duration
// Interval between pulls
pulled
bool
// true if there was a successful pull
lastPull
time
.
Time
// time of the last successful pull
Url
string
// Repository URL
Path
string
// Directory to pull to
Host
string
// Git domain host e.g. github.com
Branch
string
// Git branch
KeyPath
string
// Path to private ssh key
Interval
time
.
Duration
// Interval between pulls
Then
string
// Command to execute after successful git pull
pulled
bool
// true if there was a successful pull
lastPull
time
.
Time
// time of the last successful pull
lastCommit
string
// hash for the most recent commit
sync
.
Mutex
}
// Pull performs git clone, or git pull if repository exists
// Pull attempts a git clone.
// It retries at most numRetries times if error occurs
func
(
r
*
Repo
)
Pull
()
error
{
r
.
Lock
()
defer
r
.
Unlock
()
...
...
@@ -45,6 +53,33 @@ func (r *Repo) Pull() error {
return
nil
}
// keep last commit hash for comparison later
lastCommit
:=
r
.
lastCommit
var
err
error
// Attempt to pull at most numRetries times
for
i
:=
0
;
i
<
numRetries
;
i
++
{
if
err
=
r
.
pull
();
err
==
nil
{
break
}
logger
()
.
Println
(
err
)
}
if
err
!=
nil
{
return
err
}
// check if there are new changes,
// then execute post pull command
if
r
.
lastCommit
==
lastCommit
{
logger
()
.
Println
(
"No new changes."
)
return
nil
}
return
r
.
postPullCommand
()
}
// Pull performs git clone, or git pull if repository exists
func
(
r
*
Repo
)
pull
()
error
{
params
:=
[]
string
{
"clone"
,
"-b"
,
r
.
Branch
,
r
.
Url
,
r
.
Path
}
if
r
.
pulled
{
params
=
[]
string
{
"pull"
,
"origin"
,
r
.
Branch
}
...
...
@@ -52,35 +87,27 @@ func (r *Repo) Pull() error {
// if key is specified, pull using ssh key
if
r
.
KeyPath
!=
""
{
return
pullWithKey
(
r
,
params
)
return
r
.
pullWithKey
(
params
)
}
cmd
:=
exec
.
Command
(
gitBinary
,
params
...
)
cmd
.
Env
=
os
.
Environ
()
cmd
.
Stdout
=
os
.
Stderr
cmd
.
Stderr
=
os
.
Stderr
dir
:=
""
if
r
.
pulled
{
cmd
.
D
ir
=
r
.
Path
d
ir
=
r
.
Path
}
var
err
error
if
err
=
cmd
.
Start
();
err
!=
nil
{
return
err
}
if
err
=
cmd
.
Wait
();
err
==
nil
{
if
err
=
runCmd
(
gitBinary
,
params
,
dir
);
err
==
nil
{
r
.
pulled
=
true
r
.
lastPull
=
time
.
Now
()
log
.
Printf
(
"%v pulled.
\n
"
,
r
.
Url
)
logger
()
.
Printf
(
"%v pulled.
\n
"
,
r
.
Url
)
r
.
lastCommit
,
err
=
r
.
getMostRecentCommit
()
}
return
err
}
// pullWithKey performs git clone or git pull if repository exists.
// It is used for private repositories and requires an ssh key.
// pullWithKey is used for private repositories and requires an ssh key.
// Note: currently only limited to Linux and OSX.
func
pullWithKey
(
r
*
Repo
,
params
[]
string
)
error
{
func
(
r
*
Repo
)
pullWithKey
(
params
[]
string
)
error
{
var
gitSsh
,
script
*
os
.
File
// ensure temporary files deleted after usage
defer
func
()
{
...
...
@@ -105,30 +132,23 @@ func pullWithKey(r *Repo, params []string) error {
return
err
}
// execute the git clone bash script
cmd
:=
exec
.
Command
(
script
.
Name
())
cmd
.
Env
=
os
.
Environ
()
cmd
.
Stdout
=
os
.
Stderr
cmd
.
Stderr
=
os
.
Stderr
dir
:=
""
if
r
.
pulled
{
cmd
.
Dir
=
r
.
Path
}
if
err
=
cmd
.
Start
();
err
!=
nil
{
return
err
dir
=
r
.
Path
}
if
err
=
cmd
.
Wait
(
);
err
==
nil
{
if
err
=
runCmd
(
script
.
Name
(),
nil
,
dir
);
err
==
nil
{
r
.
pulled
=
true
r
.
lastPull
=
time
.
Now
()
log
.
Printf
(
"%v pulled.
\n
"
,
r
.
Url
)
logger
()
.
Printf
(
"%v pulled.
\n
"
,
r
.
Url
)
r
.
lastCommit
,
err
=
r
.
getMostRecentCommit
()
}
return
err
}
// prepare prepares for a git pull
// and validates the configured directory
func
prepare
(
r
*
Repo
)
error
{
func
(
r
*
Repo
)
prepare
(
)
error
{
// check if directory exists or is empty
// if not, create directory
fs
,
err
:=
ioutil
.
ReadDir
(
r
.
Path
)
...
...
@@ -148,7 +168,7 @@ func prepare(r *Repo) error {
if
isGit
{
// check if same repository
var
repoUrl
string
if
repoUrl
,
err
=
getRepoUrl
(
r
.
Path
);
err
==
nil
&&
repoUrl
==
r
.
Url
{
if
repoUrl
,
err
=
r
.
getRepoUrl
(
);
err
==
nil
&&
repoUrl
==
r
.
Url
{
r
.
pulled
=
true
return
nil
}
...
...
@@ -160,23 +180,42 @@ func prepare(r *Repo) error {
return
fmt
.
Errorf
(
"Cannot git clone into %v, directory not empty."
,
r
.
Path
)
}
// get
RepoUrl retrieves remote origin url for the git repository at path
func
getRepoUrl
(
path
string
)
(
string
,
error
)
{
args
:=
[]
string
{
"config"
,
"--get"
,
"remote.origin.url"
}
_
,
err
:=
os
.
Stat
(
path
)
// get
MostRecentCommit gets the hash of the most recent commit to the
// repository. Useful for checking if changes occur.
func
(
r
*
Repo
)
getMostRecentCommit
()
(
string
,
error
)
{
command
:=
gitBinary
+
` --no-pager log -n 1 --pretty=format:"%H"`
c
,
args
,
err
:=
middleware
.
SplitCommandAndArgs
(
command
)
if
err
!=
nil
{
return
""
,
err
}
return
runCmdOutput
(
c
,
args
,
r
.
Path
)
}
cmd
:=
exec
.
Command
(
gitBinary
,
args
...
)
cmd
.
Dir
=
path
output
,
err
:=
cmd
.
Output
(
)
// getRepoUrl retrieves remote origin url for the git repository at path
func
(
r
*
Repo
)
getRepoUrl
()
(
string
,
error
)
{
_
,
err
:=
os
.
Stat
(
r
.
Path
)
if
err
!=
nil
{
return
""
,
err
}
args
:=
[]
string
{
"config"
,
"--get"
,
"remote.origin.url"
}
return
runCmdOutput
(
gitBinary
,
args
,
r
.
Path
)
}
return
strings
.
TrimSpace
(
string
(
output
)),
nil
// postPullCommand executes r.Then.
// It is trigged after successful git pull
func
(
r
*
Repo
)
postPullCommand
()
error
{
if
r
.
Then
==
""
{
return
nil
}
c
,
args
,
err
:=
middleware
.
SplitCommandAndArgs
(
r
.
Then
)
if
err
!=
nil
{
return
err
}
if
err
=
runCmd
(
c
,
args
,
r
.
Path
);
err
==
nil
{
logger
()
.
Printf
(
"Command %v successful.
\n
"
,
r
.
Then
)
}
return
err
}
// initGit validates git installation and locates the git executable
...
...
@@ -199,6 +238,33 @@ func initGit() error {
}
// runCmd is a helper function to run commands.
// It runs command with args from directory at dir.
// The executed process outputs to os.Stderr
func
runCmd
(
command
string
,
args
[]
string
,
dir
string
)
error
{
cmd
:=
exec
.
Command
(
command
,
args
...
)
cmd
.
Stderr
=
os
.
Stderr
cmd
.
Stdout
=
os
.
Stderr
cmd
.
Dir
=
dir
if
err
:=
cmd
.
Start
();
err
!=
nil
{
return
err
}
return
cmd
.
Wait
()
}
// runCmdOutput is a helper function to run commands and return output.
// It runs command with args from directory at dir.
// If successful, returns output and nil error
func
runCmdOutput
(
command
string
,
args
[]
string
,
dir
string
)
(
string
,
error
)
{
cmd
:=
exec
.
Command
(
command
,
args
...
)
cmd
.
Dir
=
dir
var
err
error
if
output
,
err
:=
cmd
.
Output
();
err
==
nil
{
return
string
(
bytes
.
TrimSpace
(
output
)),
nil
}
return
""
,
err
}
// writeScriptFile writes content to a temporary file.
// It changes the temporary file mode to executable and
// closes it to prepare it for execution.
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment