seaweedfs/go/sequence/sequence.go

79 lines
2.0 KiB
Go
Raw Normal View History

2012-09-04 06:41:24 +08:00
package sequence
import (
2013-01-17 16:56:56 +08:00
"encoding/gob"
"log"
2013-01-17 16:56:56 +08:00
"os"
"path"
"sync"
2012-09-04 06:41:24 +08:00
)
const (
FileIdSaveInterval = 10000
)
type Sequencer interface {
NextFileId(count int) (uint64, int)
}
type SequencerImpl struct {
dir string
fileName string
volumeLock sync.Mutex
sequenceLock sync.Mutex
FileIdSequence uint64
fileIdCounter uint64
}
func NewSequencer(dirname string, filename string) (m *SequencerImpl) {
2013-01-17 16:56:56 +08:00
m = &SequencerImpl{dir: dirname, fileName: filename}
2012-09-04 06:41:24 +08:00
2013-01-17 16:56:56 +08:00
seqFile, se := os.OpenFile(path.Join(m.dir, m.fileName+".seq"), os.O_RDONLY, 0644)
if se != nil {
m.FileIdSequence = FileIdSaveInterval
log.Println("Setting file id sequence", m.FileIdSequence)
} else {
decoder := gob.NewDecoder(seqFile)
defer seqFile.Close()
2013-02-27 14:54:22 +08:00
if se = decoder.Decode(&m.FileIdSequence); se != nil {
log.Printf("error decoding FileIdSequence: %s", se)
m.FileIdSequence = FileIdSaveInterval
log.Println("Setting file id sequence", m.FileIdSequence)
} else {
log.Println("Loading file id sequence", m.FileIdSequence, "=>", m.FileIdSequence+FileIdSaveInterval)
m.FileIdSequence += FileIdSaveInterval
}
2013-01-17 16:56:56 +08:00
//in case the server stops between intervals
}
return
2012-09-04 06:41:24 +08:00
}
//count should be 1 or more
func (m *SequencerImpl) NextFileId(count int) (uint64, int) {
if count <= 0 {
return 0, 0
}
m.sequenceLock.Lock()
defer m.sequenceLock.Unlock()
if m.fileIdCounter < uint64(count) {
m.fileIdCounter = FileIdSaveInterval
m.FileIdSequence += FileIdSaveInterval
m.saveSequence()
}
m.fileIdCounter = m.fileIdCounter - uint64(count)
2013-03-20 04:37:36 +08:00
return m.FileIdSequence - m.fileIdCounter - uint64(count), count
2012-09-04 06:41:24 +08:00
}
func (m *SequencerImpl) saveSequence() {
2013-01-17 16:56:56 +08:00
log.Println("Saving file id sequence", m.FileIdSequence, "to", path.Join(m.dir, m.fileName+".seq"))
seqFile, e := os.OpenFile(path.Join(m.dir, m.fileName+".seq"), os.O_CREATE|os.O_WRONLY, 0644)
if e != nil {
log.Fatalf("Sequence File Save [ERROR] %s\n", e)
}
defer seqFile.Close()
encoder := gob.NewEncoder(seqFile)
2013-02-27 14:54:22 +08:00
if e = encoder.Encode(m.FileIdSequence); e != nil {
log.Fatalf("Sequence File Save [ERROR] %s\n", e)
}
2012-09-04 06:41:24 +08:00
}