Skip to content

Commit

Permalink
Merge pull request #583 from mreiferson/data_path_lock_583
Browse files Browse the repository at this point in the history
nsqd: panic when running two instances with same -data-path
  • Loading branch information
jehiah committed Oct 4, 2015
2 parents 27c4556 + c1362f3 commit 5a36653
Show file tree
Hide file tree
Showing 3 changed files with 76 additions and 0 deletions.
38 changes: 38 additions & 0 deletions internal/dirlock/dirlock.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
// +build !windows

package dirlock

import (
"fmt"
"os"
"syscall"
)

type DirLock struct {
dir string
f *os.File
}

func New(dir string) *DirLock {
return &DirLock{
dir: dir,
}
}

func (l *DirLock) Lock() error {
f, err := os.Open(l.dir)
if err != nil {
return err
}
l.f = f
err = syscall.Flock(int(f.Fd()), syscall.LOCK_EX|syscall.LOCK_NB)
if err != nil {
return fmt.Errorf("cannot flock directory %s - %s", l.dir, err)
}
return nil
}

func (l *DirLock) Unlock() error {
defer l.f.Close()
return syscall.Flock(int(l.f.Fd()), syscall.LOCK_UN)
}
21 changes: 21 additions & 0 deletions internal/dirlock/dirlock_windows.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
// +build windows

package dirlock

type DirLock struct {
dir string
}

func New(dir string) *DirLock {
return &DirLock{
dir: dir,
}
}

func (l *DirLock) Lock() error {
return nil
}

func (l *DirLock) Unlock() error {
return nil
}
17 changes: 17 additions & 0 deletions nsqd/nsqd.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import (

"github.com/bitly/go-simplejson"
"github.com/nsqio/nsq/internal/clusterinfo"
"github.com/nsqio/nsq/internal/dirlock"
"github.com/nsqio/nsq/internal/http_api"
"github.com/nsqio/nsq/internal/protocol"
"github.com/nsqio/nsq/internal/statsd"
Expand All @@ -45,6 +46,7 @@ type NSQD struct {

opts atomic.Value

dl *dirlock.DirLock
flag int32
errMtx sync.RWMutex
err error
Expand All @@ -71,6 +73,12 @@ type NSQD struct {
}

func New(opts *Options) *NSQD {
dataPath := opts.DataPath
if opts.DataPath == "" {
cwd, _ := os.Getwd()
dataPath = cwd
}

n := &NSQD{
flag: flagHealthy,
startTime: time.Now(),
Expand All @@ -80,9 +88,16 @@ func New(opts *Options) *NSQD {
notifyChan: make(chan interface{}),
optsNotificationChan: make(chan struct{}, 1),
ci: clusterinfo.New(opts.Logger, http_api.NewClient(nil)),
dl: dirlock.New(dataPath),
}
n.swapOpts(opts)

err := n.dl.Lock()
if err != nil {
n.logf("FATAL: --data-path=%s in use (possibly by another instance of nsqd)", dataPath)
os.Exit(1)
}

if opts.MaxDeflateLevel < 1 || opts.MaxDeflateLevel > 9 {
n.logf("FATAL: --max-deflate-level must be [1,9]")
os.Exit(1)
Expand Down Expand Up @@ -435,6 +450,8 @@ func (n *NSQD) Exit() {
// could potentially starve items in process and deadlock)
close(n.exitChan)
n.waitGroup.Wait()

n.dl.Unlock()
}

// GetTopic performs a thread safe operation
Expand Down

0 comments on commit 5a36653

Please sign in to comment.