lua update

This commit is contained in:
siddontang 2014-09-02 22:04:18 +08:00
parent ab1ae62bf7
commit 0af4758e79
3 changed files with 167 additions and 21 deletions

112
ledis/tx.go Normal file
View File

@ -0,0 +1,112 @@
package ledis
import (
"errors"
"fmt"
"github.com/siddontang/ledisdb/store"
)
var (
ErrNestTx = errors.New("nest transaction not supported")
ErrTxDone = errors.New("Transaction has already been committed or rolled back")
)
type Tx struct {
*DB
tx *store.Tx
logs [][]byte
}
func (db *DB) IsTransaction() bool {
return db.status == DBInTransaction
}
// Begin a transaction, it will block all other write operations before calling Commit or Rollback.
// You must be very careful to prevent long-time transaction.
func (db *DB) Begin() (*Tx, error) {
if db.IsTransaction() {
return nil, ErrNestTx
}
tx := new(Tx)
tx.DB = new(DB)
tx.DB.l = db.l
tx.l.wLock.Lock()
tx.DB.sdb = db.sdb
var err error
tx.tx, err = db.sdb.Begin()
if err != nil {
tx.l.wLock.Unlock()
return nil, err
}
tx.DB.bucket = tx.tx
tx.DB.status = DBInTransaction
tx.DB.index = db.index
tx.DB.kvBatch = tx.newBatch()
tx.DB.listBatch = tx.newBatch()
tx.DB.hashBatch = tx.newBatch()
tx.DB.zsetBatch = tx.newBatch()
tx.DB.binBatch = tx.newBatch()
tx.DB.setBatch = tx.newBatch()
return tx, nil
}
func (tx *Tx) Commit() error {
if tx.tx == nil {
return ErrTxDone
}
tx.l.commitLock.Lock()
err := tx.tx.Commit()
tx.tx = nil
if len(tx.logs) > 0 {
tx.l.binlog.Log(tx.logs...)
}
tx.l.commitLock.Unlock()
tx.l.wLock.Unlock()
tx.DB.bucket = nil
return err
}
func (tx *Tx) Rollback() error {
if tx.tx == nil {
return ErrTxDone
}
err := tx.tx.Rollback()
tx.tx = nil
tx.l.wLock.Unlock()
tx.DB.bucket = nil
return err
}
func (tx *Tx) newBatch() *batch {
return tx.l.newBatch(tx.tx.NewWriteBatch(), &txBatchLocker{}, tx)
}
func (tx *Tx) Select(index int) error {
if index < 0 || index >= int(MaxDBNumber) {
return fmt.Errorf("invalid db index %d", index)
}
tx.DB.index = uint8(index)
return nil
}

View File

@ -1,6 +1,7 @@
package server
import (
"crypto/sha1"
"encoding/hex"
"fmt"
"github.com/aarzilli/golua/lua"
@ -64,20 +65,21 @@ func evalGenericCommand(c *client, evalSha1 bool) error {
return err
}
var sha1 string
var key string
if !evalSha1 {
sha1 = hex.EncodeToString(c.args[0])
h := sha1.Sum(c.args[0])
key = hex.EncodeToString(h[0:20])
} else {
sha1 = ledis.String(c.args[0])
key = strings.ToLower(ledis.String(c.args[0]))
}
l.GetGlobal(sha1)
l.GetGlobal(key)
if l.IsNil(-1) {
l.Pop(1)
if evalSha1 {
return fmt.Errorf("missing %s script", sha1)
return fmt.Errorf("missing %s script", key)
}
if r := l.LoadString(ledis.String(c.args[0])); r != 0 {
@ -86,9 +88,9 @@ func evalGenericCommand(c *client, evalSha1 bool) error {
return err
} else {
l.PushValue(-1)
l.SetGlobal(sha1)
l.SetGlobal(key)
s.chunks[sha1] = struct{}{}
s.chunks[key] = struct{}{}
}
}
@ -125,9 +127,6 @@ func scriptCommand(c *client) error {
}()
args := c.args
if len(args) < 1 {
return ErrCmdParams
}
switch strings.ToLower(c.cmd) {
case "script load":
@ -137,7 +136,7 @@ func scriptCommand(c *client) error {
case "script flush":
return scriptFlushCommand(c)
default:
return fmt.Errorf("invalid scirpt cmd %s", args[0])
return fmt.Errorf("invalid script cmd %s", args[0])
}
return nil
@ -151,28 +150,33 @@ func scriptLoadCommand(c *client) error {
return ErrCmdParams
}
sha1 := hex.EncodeToString(c.args[1])
h := sha1.Sum(c.args[0])
key := hex.EncodeToString(h[0:20])
if r := l.LoadString(ledis.String(c.args[1])); r != 0 {
if r := l.LoadString(ledis.String(c.args[0])); r != 0 {
err := fmt.Errorf("%s", l.ToString(-1))
l.Pop(1)
return err
} else {
l.PushValue(-1)
l.SetGlobal(sha1)
l.SetGlobal(key)
s.chunks[sha1] = struct{}{}
s.chunks[key] = struct{}{}
}
c.resp.writeBulk(ledis.Slice(sha1))
c.resp.writeBulk(ledis.Slice(key))
return nil
}
func scriptExistsCommand(c *client) error {
s := c.app.s
ay := make([]interface{}, len(c.args[1:]))
for i, n := range c.args[1:] {
if len(c.args) < 1 {
return ErrCmdParams
}
ay := make([]interface{}, len(c.args))
for i, n := range c.args {
if _, ok := s.chunks[ledis.String(n)]; ok {
ay[i] = int64(1)
} else {
@ -193,6 +197,8 @@ func scriptFlushCommand(c *client) error {
l.SetGlobal(n)
}
s.chunks = map[string]struct{}{}
c.resp.writeStatus(OK)
return nil

View File

@ -13,17 +13,45 @@ func TestCmdEval(t *testing.T) {
if v, err := ledis.Strings(c.Do("eval", "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}", 2, "key1", "key2", "first", "second")); err != nil {
t.Fatal(err)
} else if len(v) != 4 {
t.Fatal(err)
} else if !reflect.DeepEqual(v, []string{"key1", "key2", "first", "second"}) {
t.Fatal(fmt.Sprintf("%v", v))
}
if v, err := ledis.Strings(c.Do("eval", "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}", 2, "key1", "key2", "first", "second")); err != nil {
t.Fatal(err)
} else if len(v) != 4 {
} else if !reflect.DeepEqual(v, []string{"key1", "key2", "first", "second"}) {
t.Fatal(fmt.Sprintf("%v", v))
}
var sha1 string
var err error
if sha1, err = ledis.String(c.Do("script load", "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}")); err != nil {
t.Fatal(err)
} else if len(sha1) != 40 {
t.Fatal(sha1)
}
if v, err := ledis.Strings(c.Do("evalsha", sha1, 2, "key1", "key2", "first", "second")); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(v, []string{"key1", "key2", "first", "second"}) {
t.Fatal(fmt.Sprintf("%v", v))
}
if ay, err := ledis.Values(c.Do("script exists", sha1, "01234567890123456789")); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(ay, []interface{}{int64(1), int64(0)}) {
t.Fatal(fmt.Sprintf("%v", ay))
}
if ok, err := ledis.String(c.Do("script flush")); err != nil {
t.Fatal(err)
} else if ok != "OK" {
t.Fatal(ok)
}
if ay, err := ledis.Values(c.Do("script exists", sha1)); err != nil {
t.Fatal(err)
} else if !reflect.DeepEqual(ay, []interface{}{int64(0)}) {
t.Fatal(fmt.Sprintf("%v", ay))
}
}