Git: fix for data races.

This commit is contained in:
Abiola Ibrahim 2015-06-07 20:31:34 +01:00
parent 995a2ea618
commit c42e60a3d2
7 changed files with 65 additions and 34 deletions

View file

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

View file

@ -3,7 +3,6 @@ package git
import (
"bytes"
"fmt"
"log"
"os"
"strings"
"sync"
@ -30,21 +29,10 @@ var shell string
// git requirements.
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
// stop them.
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
// of a git repository.
type Repo struct {
@ -84,7 +72,7 @@ func (r *Repo) Pull() error {
if err = r.pull(); err == nil {
break
}
logger().Println(err)
Logger().Println(err)
}
if err != nil {
@ -94,7 +82,7 @@ func (r *Repo) Pull() error {
// check if there are new changes,
// then execute post pull command
if r.lastCommit == lastCommit {
logger().Println("No new changes.")
Logger().Println("No new changes.")
return nil
}
return r.postPullCommand()
@ -121,7 +109,7 @@ func (r *Repo) pull() error {
if err = runCmd(gitBinary, params, dir); err == nil {
r.pulled = true
r.lastPull = time.Now()
logger().Printf("%v pulled.\n", r.URL)
Logger().Printf("%v pulled.\n", r.URL)
r.lastCommit, err = r.getMostRecentCommit()
}
return err
@ -162,7 +150,7 @@ func (r *Repo) pullWithKey(params []string) error {
if err = runCmd(script.Name(), nil, dir); err == nil {
r.pulled = true
r.lastPull = time.Now()
logger().Printf("%v pulled.\n", r.URL)
Logger().Printf("%v pulled.\n", r.URL)
r.lastCommit, err = r.getMostRecentCommit()
}
return err
@ -241,7 +229,7 @@ func (r *Repo) postPullCommand() error {
}
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
}

View file

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

View file

@ -4,7 +4,9 @@ package gittest
import (
"io"
"log"
"os"
"sync"
"time"
"github.com/mholt/caddy/middleware/git/gitos"
@ -39,12 +41,18 @@ func Sleep(d time.Duration) {
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.
type fakeFile struct {
name string
dir bool
content []byte
info fakeInfo
sync.Mutex
}
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) {
f.Lock()
defer f.Unlock()
if len(f.content) == 0 {
return 0, io.EOF
}
@ -74,6 +84,8 @@ func (f *fakeFile) Read(b []byte) (int, error) {
}
func (f *fakeFile) Write(b []byte) (int, error) {
f.Lock()
defer f.Unlock()
f.content = append(f.content, b...)
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():
err := repo.Pull()
if err != nil {
logger().Println(err)
Logger().Println(err)
}
case <-s.halt:
s.ticker.Stop()

View file

@ -6,12 +6,12 @@ import (
"encoding/hex"
"encoding/json"
"errors"
"github.com/mholt/caddy/middleware/git"
"io/ioutil"
"log"
"net/http"
"os"
"strings"
"github.com/mholt/caddy/middleware/git"
)
type GithubHook struct{}
@ -28,15 +28,9 @@ type ghPush struct {
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
func logger() *log.Logger {
if Logger == nil {
Logger = log.New(os.Stderr, "", log.LstdFlags)
}
return Logger
return git.Logger()
}
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")
if signature != "" {
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 {
mac := hmac.New(sha1.New, []byte(secret))
mac.Write(body)
@ -129,7 +123,7 @@ func (g GithubHook) handlePush(body []byte, repo *git.Repo) error {
branch := refSlice[2]
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()
}
@ -148,7 +142,7 @@ func (g GithubHook) handleRelease(body []byte, repo *git.Repo) error {
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
// this will pull the release tag.