scratch/scribble.go

211 lines
3.1 KiB
Go
Raw Normal View History

2014-12-01 22:55:19 +00:00
package scribble
import (
2014-12-16 16:59:23 +00:00
"encoding/json"
"errors"
2014-12-01 22:55:19 +00:00
"fmt"
"io/ioutil"
"os"
"strings"
"sync"
2014-12-01 22:55:19 +00:00
"github.com/pagodabox/golang-hatchet"
2014-12-01 23:12:45 +00:00
)
2014-12-16 16:59:23 +00:00
//
const Version = "0.0.1"
2014-12-01 22:55:19 +00:00
//
type (
2014-12-16 16:59:23 +00:00
// Driver
2014-12-01 22:55:19 +00:00
Driver struct {
mutexes map[string]sync.Mutex
2014-12-01 22:55:19 +00:00
dir string
log hatchet.Logger
2014-12-01 22:55:19 +00:00
}
// Transaction represents
Transaction struct {
Action string
Collection string
ResourceID string
2014-12-01 22:55:19 +00:00
Container interface{}
}
)
2014-12-16 16:59:23 +00:00
// New
func New(dir string, logger hatchet.Logger) (*Driver, error) {
fmt.Printf("Creating database directory at '%v'...\n", dir)
2014-12-01 22:55:19 +00:00
//
if logger == nil {
logger = hatchet.DevNullLogger{}
}
//
scribble := &Driver{
dir: dir,
mutexes: make(map[string]sync.Mutex),
log: logger,
}
2014-12-01 22:55:19 +00:00
//
2014-12-16 16:59:23 +00:00
if err := mkDir(scribble.dir); err != nil {
return nil, err
2014-12-01 22:55:19 +00:00
}
//
2014-12-16 16:59:23 +00:00
return scribble, nil
2014-12-01 22:55:19 +00:00
}
// Transact
2014-12-16 16:59:23 +00:00
func (d *Driver) Transact(trans Transaction) error {
2014-12-01 22:55:19 +00:00
//
switch trans.Action {
case "write":
return d.write(trans)
2014-12-01 22:55:19 +00:00
case "read":
return d.read(trans)
2014-12-01 22:55:19 +00:00
case "readall":
return d.readAll(trans)
2014-12-01 22:55:19 +00:00
case "delete":
return d.delete(trans)
2014-12-01 22:55:19 +00:00
default:
return errors.New(fmt.Sprintf("Unsupported action %+v", trans.Action))
2014-12-01 22:55:19 +00:00
}
return nil
2014-12-01 22:55:19 +00:00
}
// private
// write
func (d *Driver) write(trans Transaction) error {
mutex := d.getOrCreateMutex(trans.Collection)
mutex.Lock()
2014-12-01 22:55:19 +00:00
//
dir := d.dir + "/" + trans.Collection
//
b, err := json.MarshalIndent(trans.Container, "", "\t")
if err != nil {
return err
2014-12-01 22:55:19 +00:00
}
//
if err := mkDir(dir); err != nil {
return err
2014-12-01 22:55:19 +00:00
}
//
2014-12-16 21:32:14 +00:00
if err := ioutil.WriteFile(dir + "/" + trans.ResourceID, b, 0666); err != nil {
return err
2014-12-16 16:59:23 +00:00
}
2014-12-01 22:55:19 +00:00
mutex.Unlock()
2014-12-01 22:55:19 +00:00
return nil
2014-12-01 22:55:19 +00:00
}
// read
func (d *Driver) read(trans Transaction) error {
2014-12-01 22:55:19 +00:00
dir := d.dir + "/" + trans.Collection
b, err := ioutil.ReadFile(dir + "/" + trans.ResourceID)
2014-12-01 22:55:19 +00:00
if err != nil {
return err
2014-12-01 22:55:19 +00:00
}
2014-12-16 16:59:23 +00:00
if err := json.Unmarshal(b, trans.Container); err != nil {
return err
2014-12-01 22:55:19 +00:00
}
return nil
2014-12-01 22:55:19 +00:00
}
// readAll
func (d *Driver) readAll(trans Transaction) error {
2014-12-01 22:55:19 +00:00
dir := d.dir + "/" + trans.Collection
//
files, err := ioutil.ReadDir(dir)
2014-12-16 16:59:23 +00:00
// an error here just means an empty collection so do nothing
2014-12-01 22:55:19 +00:00
if err != nil {
}
var f []string
for _, file := range files {
b, err := ioutil.ReadFile(dir + "/" + file.Name())
if err != nil {
return err
2014-12-01 22:55:19 +00:00
}
f = append(f, string(b))
}
//
2014-12-16 16:59:23 +00:00
if err := json.Unmarshal([]byte("["+strings.Join(f, ",")+"]"), trans.Container); err != nil {
return err
2014-12-01 22:55:19 +00:00
}
return nil
2014-12-01 22:55:19 +00:00
}
// delete
func (d *Driver) delete(trans Transaction) error {
mutex := d.getOrCreateMutex(trans.Collection)
mutex.Lock()
2014-12-01 22:55:19 +00:00
dir := d.dir + "/" + trans.Collection
err := os.Remove(dir + "/" + trans.ResourceID)
2014-12-01 22:55:19 +00:00
if err != nil {
return err
2014-12-01 22:55:19 +00:00
}
mutex.Unlock()
return nil
2014-12-01 22:55:19 +00:00
}
// helpers
// getOrCreateMutex
func (d *Driver) getOrCreateMutex(collection string) sync.Mutex {
2014-12-01 22:55:19 +00:00
c, ok := d.mutexes[collection]
2014-12-01 22:55:19 +00:00
// if the mutex doesn't exist make it
2014-12-01 22:55:19 +00:00
if !ok {
d.mutexes[collection] = sync.Mutex{}
return d.mutexes[collection]
2014-12-01 22:55:19 +00:00
}
return c
}
// mkDir
func mkDir(d string) error {
//
dir, _ := os.Stat(d)
if dir == nil {
err := os.MkdirAll(d, 0755)
if err != nil {
return err
}
}
return nil
}