Merge pull request #116 from abiosoft/master

Git: fix for data races.
This commit is contained in:
Matt Holt 2015-06-07 14:32:20 -06:00
commit 20b6e971c0
7 changed files with 65 additions and 34 deletions

View file

@ -2,7 +2,6 @@ package setup
import ( import (
"io/ioutil" "io/ioutil"
"log"
"strings" "strings"
"testing" "testing"
"time" "time"
@ -42,7 +41,7 @@ func TestIntervals(t *testing.T) {
} }
for i, test := range tests { for i, test := range tests {
git.Logger = nil git.SetLogger(gittest.NewLogger(gittest.Open("file")))
c1 := newTestController(test) c1 := newTestController(test)
repo, err := gitParse(c1) repo, err := gitParse(c1)
@ -61,7 +60,7 @@ func TestIntervals(t *testing.T) {
// switch logger to test file // switch logger to test file
logFile := gittest.Open("file") logFile := gittest.Open("file")
git.Logger = log.New(logFile, "", 0) git.SetLogger(gittest.NewLogger(logFile))
// sleep for the interval // sleep for the interval
gittest.Sleep(repo.Interval) gittest.Sleep(repo.Interval)

View file

@ -3,7 +3,6 @@ package git
import ( import (
"bytes" "bytes"
"fmt" "fmt"
"log"
"os" "os"
"strings" "strings"
"sync" "sync"
@ -30,21 +29,10 @@ var shell string
// git requirements. // git requirements.
var initMutex = sync.Mutex{} var initMutex = sync.Mutex{}
// Logger is used to log errors; if nil, the default log.Logger is used.
var Logger *log.Logger
// Services holds all git pulling services and provides the function to // Services holds all git pulling services and provides the function to
// stop them. // stop them.
var Services = &services{} var Services = &services{}
// 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
}
// Repo is the structure that holds required information // Repo is the structure that holds required information
// of a git repository. // of a git repository.
type Repo struct { type Repo struct {
@ -84,7 +72,7 @@ func (r *Repo) Pull() error {
if err = r.pull(); err == nil { if err = r.pull(); err == nil {
break break
} }
logger().Println(err) Logger().Println(err)
} }
if err != nil { if err != nil {
@ -94,7 +82,7 @@ func (r *Repo) Pull() error {
// check if there are new changes, // check if there are new changes,
// then execute post pull command // then execute post pull command
if r.lastCommit == lastCommit { if r.lastCommit == lastCommit {
logger().Println("No new changes.") Logger().Println("No new changes.")
return nil return nil
} }
return r.postPullCommand() return r.postPullCommand()
@ -121,7 +109,7 @@ func (r *Repo) pull() error {
if err = runCmd(gitBinary, params, dir); err == nil { if err = runCmd(gitBinary, params, dir); err == nil {
r.pulled = true r.pulled = true
r.lastPull = time.Now() r.lastPull = time.Now()
logger().Printf("%v pulled.\n", r.URL) Logger().Printf("%v pulled.\n", r.URL)
r.lastCommit, err = r.getMostRecentCommit() r.lastCommit, err = r.getMostRecentCommit()
} }
return err return err
@ -162,7 +150,7 @@ func (r *Repo) pullWithKey(params []string) error {
if err = runCmd(script.Name(), nil, dir); err == nil { if err = runCmd(script.Name(), nil, dir); err == nil {
r.pulled = true r.pulled = true
r.lastPull = time.Now() r.lastPull = time.Now()
logger().Printf("%v pulled.\n", r.URL) Logger().Printf("%v pulled.\n", r.URL)
r.lastCommit, err = r.getMostRecentCommit() r.lastCommit, err = r.getMostRecentCommit()
} }
return err return err
@ -241,7 +229,7 @@ func (r *Repo) postPullCommand() error {
} }
if err = runCmd(c, args, r.Path); err == nil { if err = runCmd(c, args, r.Path); err == nil {
logger().Printf("Command %v successful.\n", r.Then) Logger().Printf("Command %v successful.\n", r.Then)
} }
return err return err
} }

View file

@ -73,7 +73,7 @@ func TestGit(t *testing.T) {
// pull with success // pull with success
logFile := gittest.Open("file") logFile := gittest.Open("file")
Logger = log.New(logFile, "", 0) SetLogger(log.New(logFile, "", 0))
tests := []struct { tests := []struct {
repo *Repo repo *Repo
output string output string

View file

@ -4,7 +4,9 @@ package gittest
import ( import (
"io" "io"
"log"
"os" "os"
"sync"
"time" "time"
"github.com/mholt/caddy/middleware/git/gitos" "github.com/mholt/caddy/middleware/git/gitos"
@ -39,12 +41,18 @@ func Sleep(d time.Duration) {
FakeOS.Sleep(d) FakeOS.Sleep(d)
} }
// NewLogger creates a logger that logs to f
func NewLogger(f gitos.File) *log.Logger {
return log.New(f, "", 0)
}
// fakeFile is a mock gitos.File. // fakeFile is a mock gitos.File.
type fakeFile struct { type fakeFile struct {
name string name string
dir bool dir bool
content []byte content []byte
info fakeInfo info fakeInfo
sync.Mutex
} }
func (f fakeFile) Name() string { func (f fakeFile) Name() string {
@ -65,6 +73,8 @@ func (f fakeFile) Chmod(mode os.FileMode) error {
} }
func (f *fakeFile) Read(b []byte) (int, error) { func (f *fakeFile) Read(b []byte) (int, error) {
f.Lock()
defer f.Unlock()
if len(f.content) == 0 { if len(f.content) == 0 {
return 0, io.EOF return 0, io.EOF
} }
@ -74,6 +84,8 @@ func (f *fakeFile) Read(b []byte) (int, error) {
} }
func (f *fakeFile) Write(b []byte) (int, error) { func (f *fakeFile) Write(b []byte) (int, error) {
f.Lock()
defer f.Unlock()
f.content = append(f.content, b...) f.content = append(f.content, b...)
return len(b), nil return len(b), nil
} }

38
middleware/git/logger.go Normal file
View file

@ -0,0 +1,38 @@
package git
import (
"log"
"os"
"sync"
)
// logger is used to log errors
var logger = &gitLogger{l: log.New(os.Stderr, "", log.LstdFlags)}
// gitLogger wraps log.Logger with mutex for thread safety.
type gitLogger struct {
l *log.Logger
sync.RWMutex
}
func (g *gitLogger) logger() *log.Logger {
g.RLock()
defer g.RUnlock()
return g.l
}
func (g *gitLogger) setLogger(l *log.Logger) {
g.Lock()
g.l = l
g.Unlock()
}
// Logger gets the currently available logger
func Logger() *log.Logger {
return logger.logger()
}
// SetLogger sets the current logger to l
func SetLogger(l *log.Logger) {
logger.setLogger(l)
}

View file

@ -27,7 +27,7 @@ func Start(repo *Repo) {
case <-s.ticker.C(): case <-s.ticker.C():
err := repo.Pull() err := repo.Pull()
if err != nil { if err != nil {
logger().Println(err) Logger().Println(err)
} }
case <-s.halt: case <-s.halt:
s.ticker.Stop() s.ticker.Stop()

View file

@ -6,12 +6,12 @@ import (
"encoding/hex" "encoding/hex"
"encoding/json" "encoding/json"
"errors" "errors"
"github.com/mholt/caddy/middleware/git"
"io/ioutil" "io/ioutil"
"log" "log"
"net/http" "net/http"
"os"
"strings" "strings"
"github.com/mholt/caddy/middleware/git"
) )
type GithubHook struct{} type GithubHook struct{}
@ -28,15 +28,9 @@ type ghPush struct {
Ref string `json:"ref"` Ref string `json:"ref"`
} }
// Logger is used to log errors; if nil, the default log.Logger is used.
var Logger *log.Logger
// logger is an helper function to retrieve the available logger // logger is an helper function to retrieve the available logger
func logger() *log.Logger { func logger() *log.Logger {
if Logger == nil { return git.Logger()
Logger = log.New(os.Stderr, "", log.LstdFlags)
}
return Logger
} }
func (g GithubHook) DoesHandle(h http.Header) bool { func (g GithubHook) DoesHandle(h http.Header) bool {
@ -97,7 +91,7 @@ func (g GithubHook) handleSignature(r *http.Request, body []byte, secret string)
signature := r.Header.Get("X-Hub-Signature") signature := r.Header.Get("X-Hub-Signature")
if signature != "" { if signature != "" {
if secret == "" { if secret == "" {
logger().Print("Unable to verify request signature. Secret not set in caddyfile!") logger().Print("Unable to verify request signature. Secret not set in caddyfile!\n")
} else { } else {
mac := hmac.New(sha1.New, []byte(secret)) mac := hmac.New(sha1.New, []byte(secret))
mac.Write(body) mac.Write(body)
@ -129,7 +123,7 @@ func (g GithubHook) handlePush(body []byte, repo *git.Repo) error {
branch := refSlice[2] branch := refSlice[2]
if branch == repo.Branch { if branch == repo.Branch {
logger().Print("Received pull notification for the tracking branch, updating...") logger().Print("Received pull notification for the tracking branch, updating...\n")
repo.Pull() repo.Pull()
} }
@ -148,7 +142,7 @@ func (g GithubHook) handleRelease(body []byte, repo *git.Repo) error {
return errors.New("The release request contained an invalid TagName.") return errors.New("The release request contained an invalid TagName.")
} }
logger().Printf("Received new release '%s'. -> Updating local repository to this release.", release.Release.Name) logger().Printf("Received new release '%s'. -> Updating local repository to this release.\n", release.Release.Name)
// Update the local branch to the release tag name // Update the local branch to the release tag name
// this will pull the release tag. // this will pull the release tag.