Improve SQL INFO command to support namespaces
This commit is contained in:
parent
3ca8fb3acc
commit
2fcc2e3668
3 changed files with 133 additions and 87 deletions
196
db/info.go
196
db/info.go
|
@ -15,100 +15,126 @@
|
|||
package db
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"github.com/abcum/surreal/kvs"
|
||||
"github.com/abcum/surreal/mem"
|
||||
"github.com/abcum/surreal/sql"
|
||||
"github.com/abcum/surreal/util/data"
|
||||
"github.com/abcum/surreal/util/keys"
|
||||
"github.com/abcum/surreal/util/pack"
|
||||
)
|
||||
|
||||
func (e *executor) executeInfoStatement(txn kvs.TX, ast *sql.InfoStatement) (out []interface{}, err error) {
|
||||
|
||||
if ast.What == "" {
|
||||
|
||||
res := data.New()
|
||||
res.Array("scopes")
|
||||
res.Array("tables")
|
||||
res.Array("views")
|
||||
|
||||
// Get the scope definitions
|
||||
sbeg := &keys.SC{KV: ast.KV, NS: ast.NS, DB: ast.DB, SC: keys.Prefix}
|
||||
send := &keys.SC{KV: ast.KV, NS: ast.NS, DB: ast.DB, SC: keys.Suffix}
|
||||
skvs, _ := txn.RGet(sbeg.Encode(), send.Encode(), 0)
|
||||
for _, kv := range skvs {
|
||||
key := &keys.SC{}
|
||||
key.Decode(kv.Key())
|
||||
res.Inc(key.SC, "scopes")
|
||||
}
|
||||
|
||||
// Get the table definitions
|
||||
tbeg := &keys.TB{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: keys.Prefix}
|
||||
tend := &keys.TB{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: keys.Suffix}
|
||||
tkvs, _ := txn.RGet(tbeg.Encode(), tend.Encode(), 0)
|
||||
for _, kv := range tkvs {
|
||||
key := &keys.TB{}
|
||||
key.Decode(kv.Key())
|
||||
res.Inc(key.TB, "tables")
|
||||
}
|
||||
|
||||
// Get the table definitions
|
||||
vbeg := &keys.VW{KV: ast.KV, NS: ast.NS, DB: ast.DB, VW: keys.Prefix}
|
||||
vend := &keys.VW{KV: ast.KV, NS: ast.NS, DB: ast.DB, VW: keys.Suffix}
|
||||
vkvs, _ := txn.RGet(vbeg.Encode(), vend.Encode(), 0)
|
||||
for _, kv := range vkvs {
|
||||
key, val := &keys.VW{}, &sql.DefineViewStatement{}
|
||||
key.Decode(kv.Key())
|
||||
pack.Decode(kv.Val(), val)
|
||||
res.Inc(val, "views")
|
||||
}
|
||||
|
||||
out = append(out, res.Data())
|
||||
|
||||
} else {
|
||||
|
||||
res := data.New()
|
||||
res.Object("rules")
|
||||
res.Array("fields")
|
||||
res.Array("indexes")
|
||||
|
||||
// Get the rules definitions
|
||||
rbeg := &keys.RU{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: ast.What, RU: keys.Prefix}
|
||||
rend := &keys.RU{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: ast.What, RU: keys.Suffix}
|
||||
rkvs, _ := txn.RGet(rbeg.Encode(), rend.Encode(), 0)
|
||||
for _, kv := range rkvs {
|
||||
key, val := &keys.RU{}, &sql.DefineRulesStatement{}
|
||||
key.Decode(kv.Key())
|
||||
pack.Decode(kv.Val(), val)
|
||||
res.Set(val, "rules", fmt.Sprintf("%v", key.RU))
|
||||
}
|
||||
|
||||
// Get the field definitions
|
||||
fbeg := &keys.FD{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: ast.What, FD: keys.Prefix}
|
||||
fend := &keys.FD{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: ast.What, FD: keys.Suffix}
|
||||
fkvs, _ := txn.RGet(fbeg.Encode(), fend.Encode(), 0)
|
||||
for _, kv := range fkvs {
|
||||
key, val := &keys.FD{}, &sql.DefineFieldStatement{}
|
||||
key.Decode(kv.Key())
|
||||
pack.Decode(kv.Val(), val)
|
||||
res.Inc(val, "fields")
|
||||
}
|
||||
|
||||
// Get the field definitions
|
||||
ibeg := &keys.IX{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: ast.What, IX: keys.Prefix}
|
||||
iend := &keys.IX{KV: ast.KV, NS: ast.NS, DB: ast.DB, TB: ast.What, IX: keys.Suffix}
|
||||
ikvs, _ := txn.RGet(ibeg.Encode(), iend.Encode(), 0)
|
||||
for _, kv := range ikvs {
|
||||
key, val := &keys.IX{}, &sql.DefineIndexStatement{}
|
||||
key.Decode(kv.Key())
|
||||
pack.Decode(kv.Val(), val)
|
||||
res.Inc(val, "indexes")
|
||||
}
|
||||
|
||||
out = append(out, res.Data())
|
||||
|
||||
switch ast.Kind {
|
||||
case sql.NAMESPACE:
|
||||
return e.executeInfoNSStatement(txn, ast)
|
||||
case sql.DATABASE:
|
||||
return e.executeInfoDBStatement(txn, ast)
|
||||
case sql.TABLE:
|
||||
return e.executeInfoTBStatement(txn, ast)
|
||||
}
|
||||
|
||||
return
|
||||
|
||||
}
|
||||
|
||||
func (e *executor) executeInfoNSStatement(txn kvs.TX, ast *sql.InfoStatement) (out []interface{}, err error) {
|
||||
|
||||
res := data.New()
|
||||
res.Array("logins")
|
||||
res.Array("tokens")
|
||||
res.Array("databs")
|
||||
|
||||
defer func() {
|
||||
if r := recover(); r != nil {
|
||||
out = append(out, res.Data())
|
||||
}
|
||||
}()
|
||||
|
||||
db := mem.GetNS(ast.NS)
|
||||
|
||||
for _, v := range db.AC {
|
||||
res.Inc(v, "logins")
|
||||
}
|
||||
|
||||
for _, v := range db.TK {
|
||||
res.Inc(v, "tokens")
|
||||
}
|
||||
|
||||
for _, v := range db.DB {
|
||||
res.Inc(v, "databs")
|
||||
}
|
||||
|
||||
out = append(out, res.Data())
|
||||
|
||||
return
|
||||
|
||||
}
|
||||
|
||||
func (e *executor) executeInfoDBStatement(txn kvs.TX, ast *sql.InfoStatement) (out []interface{}, err error) {
|
||||
|
||||
res := data.New()
|
||||
res.Array("logins")
|
||||
res.Array("tokens")
|
||||
res.Array("scopes")
|
||||
res.Array("tables")
|
||||
|
||||
defer func() {
|
||||
if r := recover(); r != nil {
|
||||
out = append(out, res.Data())
|
||||
}
|
||||
}()
|
||||
|
||||
db := mem.GetNS(ast.NS).GetDB(ast.DB)
|
||||
|
||||
for _, v := range db.AC {
|
||||
res.Inc(v, "logins")
|
||||
}
|
||||
|
||||
for _, v := range db.TK {
|
||||
res.Inc(v, "tokens")
|
||||
}
|
||||
|
||||
for _, v := range db.SC {
|
||||
res.Inc(v, "scopes")
|
||||
}
|
||||
|
||||
for _, v := range db.TB {
|
||||
res.Inc(v, "tables")
|
||||
}
|
||||
|
||||
out = append(out, res.Data())
|
||||
|
||||
return
|
||||
|
||||
}
|
||||
|
||||
func (e *executor) executeInfoTBStatement(txn kvs.TX, ast *sql.InfoStatement) (out []interface{}, err error) {
|
||||
|
||||
res := data.New()
|
||||
res.Array("fields")
|
||||
res.Array("indexs")
|
||||
|
||||
defer func() {
|
||||
if r := recover(); r != nil {
|
||||
out = append(out, res.Data())
|
||||
}
|
||||
}()
|
||||
|
||||
tb := mem.GetNS(ast.NS).GetDB(ast.DB).GetTB(ast.What)
|
||||
|
||||
for _, v := range tb.RU {
|
||||
res.Inc(v, "rules")
|
||||
}
|
||||
|
||||
for _, v := range tb.FD {
|
||||
res.Inc(v, "fields")
|
||||
}
|
||||
|
||||
for _, v := range tb.IX {
|
||||
res.Inc(v, "indexs")
|
||||
}
|
||||
|
||||
out = append(out, res.Data())
|
||||
|
||||
return
|
||||
|
||||
}
|
||||
|
|
|
@ -66,6 +66,7 @@ type InfoStatement struct {
|
|||
KV string `cork:"-" codec:"-"`
|
||||
NS string `cork:"-" codec:"-"`
|
||||
DB string `cork:"-" codec:"-"`
|
||||
Kind Token `cork:"-" codec:"-"`
|
||||
What string `cork:"-" codec:"-"`
|
||||
}
|
||||
|
||||
|
|
23
sql/info.go
23
sql/info.go
|
@ -18,11 +18,30 @@ func (p *parser) parseInfoStatement() (stmt *InfoStatement, err error) {
|
|||
|
||||
stmt = &InfoStatement{}
|
||||
|
||||
if stmt.KV, stmt.NS, stmt.DB, err = p.o.get(AuthDB); err != nil {
|
||||
if _, _, err = p.shouldBe(FOR); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if _, _, exi := p.mightBe(FOR); exi {
|
||||
if stmt.Kind, _, err = p.shouldBe(NAMESPACE, DATABASE, TABLE); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if p.is(stmt.Kind, NAMESPACE) {
|
||||
if stmt.KV, stmt.NS, stmt.DB, err = p.o.get(AuthNS); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
if p.is(stmt.Kind, DATABASE) {
|
||||
if stmt.KV, stmt.NS, stmt.DB, err = p.o.get(AuthDB); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
if p.is(stmt.Kind, TABLE) {
|
||||
if stmt.KV, stmt.NS, stmt.DB, err = p.o.get(AuthDB); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if stmt.What, err = p.parseName(); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue