517 lines
11 KiB
Go
517 lines
11 KiB
Go
|
/*
|
||
|
** Copyright (c) 2014 Arnaud Ysmal. All Rights Reserved.
|
||
|
**
|
||
|
** Redistribution and use in source and binary forms, with or without
|
||
|
** modification, are permitted provided that the following conditions
|
||
|
** are met:
|
||
|
** 1. Redistributions of source code must retain the above copyright
|
||
|
** notice, this list of conditions and the following disclaimer.
|
||
|
** 2. Redistributions in binary form must reproduce the above copyright
|
||
|
** notice, this list of conditions and the following disclaimer in the
|
||
|
** documentation and/or other materials provided with the distribution.
|
||
|
**
|
||
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
|
||
|
** OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||
|
** WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
** DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
||
|
** FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||
|
** DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||
|
** SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||
|
** HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||
|
** LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||
|
** OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||
|
** SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
package dropbox
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"reflect"
|
||
|
)
|
||
|
|
||
|
type value struct {
|
||
|
values []interface{}
|
||
|
isList bool
|
||
|
}
|
||
|
|
||
|
type fieldOp struct {
|
||
|
Op string
|
||
|
Index int
|
||
|
Index2 int
|
||
|
Data value
|
||
|
}
|
||
|
|
||
|
type opDict map[string]fieldOp
|
||
|
|
||
|
type change struct {
|
||
|
Op string
|
||
|
TID string
|
||
|
RecordID string
|
||
|
Ops opDict
|
||
|
Data Fields
|
||
|
Revert *change
|
||
|
}
|
||
|
type listOfChanges []*change
|
||
|
|
||
|
type changeWork struct {
|
||
|
c *change
|
||
|
out chan error
|
||
|
}
|
||
|
|
||
|
const (
|
||
|
recordDelete = "D"
|
||
|
recordInsert = "I"
|
||
|
recordUpdate = "U"
|
||
|
fieldDelete = "D"
|
||
|
fieldPut = "P"
|
||
|
listCreate = "LC"
|
||
|
listDelete = "LD"
|
||
|
listInsert = "LI"
|
||
|
listMove = "LM"
|
||
|
listPut = "LP"
|
||
|
)
|
||
|
|
||
|
func newValueFromInterface(i interface{}) *value {
|
||
|
if a, ok := i.([]byte); ok {
|
||
|
return &value{
|
||
|
values: []interface{}{a},
|
||
|
isList: false,
|
||
|
}
|
||
|
}
|
||
|
if reflect.TypeOf(i).Kind() == reflect.Slice || reflect.TypeOf(i).Kind() == reflect.Array {
|
||
|
val := reflect.ValueOf(i)
|
||
|
v := &value{
|
||
|
values: make([]interface{}, val.Len()),
|
||
|
isList: true,
|
||
|
}
|
||
|
for i := range v.values {
|
||
|
v.values[i] = val.Index(i).Interface()
|
||
|
}
|
||
|
return v
|
||
|
}
|
||
|
return &value{
|
||
|
values: []interface{}{i},
|
||
|
isList: false,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func newValue(v *value) *value {
|
||
|
var nv *value
|
||
|
|
||
|
nv = &value{
|
||
|
values: make([]interface{}, len(v.values)),
|
||
|
isList: v.isList,
|
||
|
}
|
||
|
copy(nv.values, v.values)
|
||
|
return nv
|
||
|
}
|
||
|
|
||
|
func newFields(f Fields) Fields {
|
||
|
var n Fields
|
||
|
|
||
|
n = make(Fields)
|
||
|
for k, v := range f {
|
||
|
n[k] = *newValue(&v)
|
||
|
}
|
||
|
return n
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) deleteRecord(table, record string) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordDelete,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) insertRecord(table, record string, values Fields) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordInsert,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Data: newFields(values),
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) updateFields(table, record string, values map[string]interface{}) error {
|
||
|
var dsval opDict
|
||
|
|
||
|
dsval = make(opDict)
|
||
|
for k, v := range values {
|
||
|
dsval[k] = fieldOp{
|
||
|
Op: fieldPut,
|
||
|
Data: *newValueFromInterface(v),
|
||
|
}
|
||
|
}
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: dsval,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) updateField(table, record, field string, i interface{}) error {
|
||
|
return ds.updateFields(table, record, map[string]interface{}{field: i})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) deleteField(table, record, field string) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: opDict{
|
||
|
field: fieldOp{
|
||
|
Op: fieldDelete,
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) listCreate(table, record, field string) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: opDict{
|
||
|
field: fieldOp{
|
||
|
Op: listCreate,
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) listDelete(table, record, field string, pos int) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: opDict{
|
||
|
field: fieldOp{
|
||
|
Op: listDelete,
|
||
|
Index: pos,
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) listInsert(table, record, field string, pos int, i interface{}) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: opDict{
|
||
|
field: fieldOp{
|
||
|
Op: listInsert,
|
||
|
Index: pos,
|
||
|
Data: *newValueFromInterface(i),
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) listMove(table, record, field string, from, to int) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: opDict{
|
||
|
field: fieldOp{
|
||
|
Op: listMove,
|
||
|
Index: from,
|
||
|
Index2: to,
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) listPut(table, record, field string, pos int, i interface{}) error {
|
||
|
return ds.handleChange(&change{
|
||
|
Op: recordUpdate,
|
||
|
TID: table,
|
||
|
RecordID: record,
|
||
|
Ops: opDict{
|
||
|
field: fieldOp{
|
||
|
Op: listPut,
|
||
|
Index: pos,
|
||
|
Data: *newValueFromInterface(i),
|
||
|
},
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) handleChange(c *change) error {
|
||
|
var out chan error
|
||
|
|
||
|
if ds.changesQueue == nil {
|
||
|
return fmt.Errorf("datastore is closed")
|
||
|
}
|
||
|
out = make(chan error)
|
||
|
ds.changesQueue <- changeWork{
|
||
|
c: c,
|
||
|
out: out,
|
||
|
}
|
||
|
return <-out
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) doHandleChange() {
|
||
|
var err error
|
||
|
var c *change
|
||
|
|
||
|
q := ds.changesQueue
|
||
|
for cw := range q {
|
||
|
c = cw.c
|
||
|
|
||
|
if err = ds.validateChange(c); err != nil {
|
||
|
cw.out <- err
|
||
|
continue
|
||
|
}
|
||
|
if c.Revert, err = ds.inverseChange(c); err != nil {
|
||
|
cw.out <- err
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
if err = ds.applyChange(c); err != nil {
|
||
|
cw.out <- err
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
ds.changes = append(ds.changes, c)
|
||
|
|
||
|
if ds.autoCommit {
|
||
|
if err = ds.Commit(); err != nil {
|
||
|
cw.out <- err
|
||
|
}
|
||
|
}
|
||
|
close(cw.out)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) validateChange(c *change) error {
|
||
|
var t *Table
|
||
|
var r *Record
|
||
|
var ok bool
|
||
|
|
||
|
if t, ok = ds.tables[c.TID]; !ok {
|
||
|
t = &Table{
|
||
|
datastore: ds,
|
||
|
tableID: c.TID,
|
||
|
records: make(map[string]*Record),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
r = t.records[c.RecordID]
|
||
|
|
||
|
switch c.Op {
|
||
|
case recordInsert, recordDelete:
|
||
|
return nil
|
||
|
case recordUpdate:
|
||
|
if r == nil {
|
||
|
return fmt.Errorf("no such record: %s", c.RecordID)
|
||
|
}
|
||
|
for field, op := range c.Ops {
|
||
|
if op.Op == fieldPut || op.Op == fieldDelete {
|
||
|
continue
|
||
|
}
|
||
|
v, ok := r.fields[field]
|
||
|
if op.Op == listCreate {
|
||
|
if ok {
|
||
|
return fmt.Errorf("field %s already exists", field)
|
||
|
}
|
||
|
continue
|
||
|
}
|
||
|
if !ok {
|
||
|
return fmt.Errorf("no such field: %s", field)
|
||
|
}
|
||
|
if !v.isList {
|
||
|
return fmt.Errorf("field %s is not a list", field)
|
||
|
}
|
||
|
maxIndex := len(v.values) - 1
|
||
|
if op.Op == listInsert {
|
||
|
maxIndex++
|
||
|
}
|
||
|
if op.Index > maxIndex {
|
||
|
return fmt.Errorf("out of bound access index %d on [0:%d]", op.Index, maxIndex)
|
||
|
}
|
||
|
if op.Index2 > maxIndex {
|
||
|
return fmt.Errorf("out of bound access index %d on [0:%d]", op.Index, maxIndex)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) applyChange(c *change) error {
|
||
|
var t *Table
|
||
|
var r *Record
|
||
|
var ok bool
|
||
|
|
||
|
if t, ok = ds.tables[c.TID]; !ok {
|
||
|
t = &Table{
|
||
|
datastore: ds,
|
||
|
tableID: c.TID,
|
||
|
records: make(map[string]*Record),
|
||
|
}
|
||
|
ds.tables[c.TID] = t
|
||
|
}
|
||
|
|
||
|
r = t.records[c.RecordID]
|
||
|
|
||
|
switch c.Op {
|
||
|
case recordInsert:
|
||
|
t.records[c.RecordID] = &Record{
|
||
|
table: t,
|
||
|
recordID: c.RecordID,
|
||
|
fields: newFields(c.Data),
|
||
|
}
|
||
|
case recordDelete:
|
||
|
if r == nil {
|
||
|
return nil
|
||
|
}
|
||
|
r.isDeleted = true
|
||
|
delete(t.records, c.RecordID)
|
||
|
case recordUpdate:
|
||
|
for field, op := range c.Ops {
|
||
|
v, ok := r.fields[field]
|
||
|
switch op.Op {
|
||
|
case fieldPut:
|
||
|
r.fields[field] = *newValue(&op.Data)
|
||
|
case fieldDelete:
|
||
|
if ok {
|
||
|
delete(r.fields, field)
|
||
|
}
|
||
|
case listCreate:
|
||
|
if !ok {
|
||
|
r.fields[field] = value{isList: true}
|
||
|
}
|
||
|
case listDelete:
|
||
|
copy(v.values[op.Index:], v.values[op.Index+1:])
|
||
|
v.values = v.values[:len(v.values)-1]
|
||
|
r.fields[field] = v
|
||
|
case listInsert:
|
||
|
v.values = append(v.values, op.Data)
|
||
|
copy(v.values[op.Index+1:], v.values[op.Index:len(v.values)-1])
|
||
|
v.values[op.Index] = op.Data.values[0]
|
||
|
r.fields[field] = v
|
||
|
case listMove:
|
||
|
val := v.values[op.Index]
|
||
|
if op.Index < op.Index2 {
|
||
|
copy(v.values[op.Index:op.Index2], v.values[op.Index+1:op.Index2+1])
|
||
|
} else {
|
||
|
copy(v.values[op.Index2+1:op.Index+1], v.values[op.Index2:op.Index])
|
||
|
}
|
||
|
v.values[op.Index2] = val
|
||
|
r.fields[field] = v
|
||
|
case listPut:
|
||
|
r.fields[field].values[op.Index] = op.Data.values[0]
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (ds *Datastore) inverseChange(c *change) (*change, error) {
|
||
|
var t *Table
|
||
|
var r *Record
|
||
|
var ok bool
|
||
|
var rev *change
|
||
|
|
||
|
if t, ok = ds.tables[c.TID]; !ok {
|
||
|
t = &Table{
|
||
|
datastore: ds,
|
||
|
tableID: c.TID,
|
||
|
records: make(map[string]*Record),
|
||
|
}
|
||
|
ds.tables[c.TID] = t
|
||
|
}
|
||
|
|
||
|
r = t.records[c.RecordID]
|
||
|
|
||
|
switch c.Op {
|
||
|
case recordInsert:
|
||
|
return &change{
|
||
|
Op: recordDelete,
|
||
|
TID: c.TID,
|
||
|
RecordID: c.RecordID,
|
||
|
}, nil
|
||
|
case recordDelete:
|
||
|
if r == nil {
|
||
|
return nil, nil
|
||
|
}
|
||
|
return &change{
|
||
|
Op: recordInsert,
|
||
|
TID: c.TID,
|
||
|
RecordID: c.RecordID,
|
||
|
Data: newFields(r.fields),
|
||
|
}, nil
|
||
|
case recordUpdate:
|
||
|
rev = &change{
|
||
|
Op: recordUpdate,
|
||
|
TID: c.TID,
|
||
|
RecordID: c.RecordID,
|
||
|
Ops: make(opDict),
|
||
|
}
|
||
|
for field, op := range c.Ops {
|
||
|
switch op.Op {
|
||
|
case fieldPut:
|
||
|
if v, ok := r.fields[field]; ok {
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: fieldPut,
|
||
|
Data: *newValue(&v),
|
||
|
}
|
||
|
} else {
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: fieldDelete,
|
||
|
}
|
||
|
}
|
||
|
case fieldDelete:
|
||
|
if v, ok := r.fields[field]; ok {
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: fieldPut,
|
||
|
Data: *newValue(&v),
|
||
|
}
|
||
|
}
|
||
|
case listCreate:
|
||
|
if _, ok := r.fields[field]; !ok {
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: fieldDelete,
|
||
|
}
|
||
|
}
|
||
|
case listDelete:
|
||
|
v := r.fields[field]
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: listInsert,
|
||
|
Index: op.Index,
|
||
|
Data: value{
|
||
|
values: []interface{}{v.values[op.Index]},
|
||
|
isList: false,
|
||
|
},
|
||
|
}
|
||
|
case listInsert:
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: listDelete,
|
||
|
Index: op.Index,
|
||
|
}
|
||
|
case listMove:
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: listMove,
|
||
|
Index: op.Index2,
|
||
|
Index2: op.Index,
|
||
|
}
|
||
|
case listPut:
|
||
|
v := r.fields[field]
|
||
|
rev.Ops[field] = fieldOp{
|
||
|
Op: listPut,
|
||
|
Index: op.Index,
|
||
|
Data: value{
|
||
|
values: []interface{}{v.values[op.Index]},
|
||
|
isList: false,
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return rev, nil
|
||
|
}
|