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
|
package db
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"github.com/abcum/surreal/kvs"
|
"github.com/abcum/surreal/kvs"
|
||||||
|
"github.com/abcum/surreal/mem"
|
||||||
"github.com/abcum/surreal/sql"
|
"github.com/abcum/surreal/sql"
|
||||||
"github.com/abcum/surreal/util/data"
|
"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) {
|
func (e *executor) executeInfoStatement(txn kvs.TX, ast *sql.InfoStatement) (out []interface{}, err error) {
|
||||||
|
|
||||||
if ast.What == "" {
|
switch ast.Kind {
|
||||||
|
case sql.NAMESPACE:
|
||||||
res := data.New()
|
return e.executeInfoNSStatement(txn, ast)
|
||||||
res.Array("scopes")
|
case sql.DATABASE:
|
||||||
res.Array("tables")
|
return e.executeInfoDBStatement(txn, ast)
|
||||||
res.Array("views")
|
case sql.TABLE:
|
||||||
|
return e.executeInfoTBStatement(txn, ast)
|
||||||
// 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())
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return
|
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:"-"`
|
KV string `cork:"-" codec:"-"`
|
||||||
NS string `cork:"-" codec:"-"`
|
NS string `cork:"-" codec:"-"`
|
||||||
DB string `cork:"-" codec:"-"`
|
DB string `cork:"-" codec:"-"`
|
||||||
|
Kind Token `cork:"-" codec:"-"`
|
||||||
What string `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{}
|
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
|
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 {
|
if stmt.What, err = p.parseName(); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue