2017-02-23 10:13:13 +00:00
|
|
|
// Copyright © 2016 Abcum Ltd
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package db
|
|
|
|
|
|
|
|
import (
|
2017-11-16 20:53:39 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"runtime/debug"
|
|
|
|
|
2017-02-23 10:13:13 +00:00
|
|
|
"github.com/abcum/surreal/kvs"
|
2017-11-16 20:53:39 +00:00
|
|
|
"github.com/abcum/surreal/log"
|
2017-02-23 10:13:13 +00:00
|
|
|
"github.com/abcum/surreal/mem"
|
|
|
|
"github.com/abcum/surreal/sql"
|
|
|
|
)
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
type executor struct {
|
|
|
|
dbo *mem.Cache
|
|
|
|
send chan *Response
|
2017-02-23 10:13:13 +00:00
|
|
|
}
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
func newExecutor() (e *executor) {
|
|
|
|
|
|
|
|
e = executorPool.Get().(*executor)
|
|
|
|
|
|
|
|
e.dbo = mem.New()
|
|
|
|
|
|
|
|
e.send = make(chan *Response)
|
|
|
|
|
2017-02-28 00:17:10 +00:00
|
|
|
return
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *executor) execute(ctx context.Context, ast *sql.Query) {
|
|
|
|
|
|
|
|
var err error
|
|
|
|
var now time.Time
|
|
|
|
var rsp *Response
|
|
|
|
var buf []*Response
|
|
|
|
var res []interface{}
|
|
|
|
|
2018-01-31 09:15:29 +00:00
|
|
|
// Get the fibre context ID so that we can use
|
|
|
|
// it to clear or flush websocket notification
|
|
|
|
// changes linked to this context.
|
|
|
|
|
|
|
|
id := ctx.Value(ctxKeyId).(string)
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
// Ensure that the executor is added back into
|
|
|
|
// the executor pool when the executor has
|
|
|
|
// finished processing the request.
|
|
|
|
|
|
|
|
defer executorPool.Put(e)
|
|
|
|
|
|
|
|
// Ensure that the query responses channel is
|
|
|
|
// closed when the full query has been processed
|
|
|
|
// and dealt with.
|
|
|
|
|
|
|
|
defer close(e.send)
|
|
|
|
|
|
|
|
// If we are making use of a global transaction
|
|
|
|
// which is not committed at the end of the
|
|
|
|
// query set, then cancel the transaction.
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if e.dbo.TX != nil {
|
|
|
|
e.dbo.Cancel()
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2017-11-16 20:53:39 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// If we have panicked during query execution
|
|
|
|
// then ensure that we recover from the error
|
|
|
|
// and print the error to the log.
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err := recover(); err != nil {
|
|
|
|
log.WithPrefix("db").WithFields(map[string]interface{}{
|
|
|
|
"id": ctx.Value(ctxKeyId), "stack": string(debug.Stack()),
|
|
|
|
}).Errorln(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Loop over the defined query statements and
|
|
|
|
// process them, while listening for the quit
|
|
|
|
// channel to see if the client has gone away.
|
|
|
|
|
|
|
|
for _, stm := range ast.Statements {
|
|
|
|
|
|
|
|
select {
|
|
|
|
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
// When in debugging mode, log every sql
|
|
|
|
// query, along with the query execution
|
|
|
|
// speed, so we can analyse slow queries.
|
|
|
|
|
|
|
|
log := log.WithPrefix("sql").WithFields(map[string]interface{}{
|
|
|
|
"id": ctx.Value(ctxKeyId),
|
|
|
|
"kind": ctx.Value(ctxKeyKind),
|
|
|
|
})
|
|
|
|
|
|
|
|
if stm, ok := stm.(sql.AuthableStatement); ok {
|
|
|
|
ns, db := stm.Auth()
|
2017-11-28 01:20:30 +00:00
|
|
|
ctx = context.WithValue(ctx, ctxKeyNs, ns)
|
|
|
|
ctx = context.WithValue(ctx, ctxKeyDb, db)
|
2017-11-16 20:53:39 +00:00
|
|
|
log = log.WithField("ns", ns).WithField("db", db)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debugln(stm)
|
|
|
|
|
|
|
|
// If we are not inside a global transaction
|
|
|
|
// then reset the error to nil so that the
|
|
|
|
// next statement is not ignored.
|
|
|
|
|
|
|
|
if e.dbo.TX == nil {
|
|
|
|
err, now = nil, time.Now()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check to see if the current statement is
|
|
|
|
// a TRANSACTION statement, and if it is
|
|
|
|
// then deal with it and move on to the next.
|
|
|
|
|
|
|
|
switch stm.(type) {
|
|
|
|
case *sql.BeginStatement:
|
2018-02-06 17:07:42 +00:00
|
|
|
err = e.begin(ctx, true)
|
2017-11-16 20:53:39 +00:00
|
|
|
continue
|
|
|
|
case *sql.CancelStatement:
|
|
|
|
err, buf = e.cancel(buf, err, e.send)
|
2018-01-10 11:33:26 +00:00
|
|
|
if err != nil {
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
} else {
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
}
|
2017-11-16 20:53:39 +00:00
|
|
|
continue
|
|
|
|
case *sql.CommitStatement:
|
|
|
|
err, buf = e.commit(buf, err, e.send)
|
2018-01-10 11:33:26 +00:00
|
|
|
if err != nil {
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
} else {
|
2018-01-31 09:15:29 +00:00
|
|
|
flush(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
}
|
2017-11-16 20:53:39 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// If an error has occured and we are inside
|
|
|
|
// a global transaction, then ignore all
|
|
|
|
// subsequent statements in the transaction.
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
res, err = e.operate(ctx, stm)
|
|
|
|
} else {
|
|
|
|
res, err = []interface{}{}, queryNotExecuted
|
|
|
|
}
|
|
|
|
|
|
|
|
rsp = &Response{
|
|
|
|
Time: time.Since(now).String(),
|
|
|
|
Status: status(err),
|
|
|
|
Detail: detail(err),
|
|
|
|
Result: append([]interface{}{}, res...),
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we are not inside a global transaction
|
|
|
|
// then we can output the statement response
|
|
|
|
// immediately to the channel.
|
|
|
|
|
|
|
|
if e.dbo.TX == nil {
|
|
|
|
e.send <- rsp
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we are inside a global transaction we
|
|
|
|
// must buffer the responses for output at
|
|
|
|
// the end of the transaction.
|
|
|
|
|
|
|
|
if e.dbo.TX != nil {
|
|
|
|
switch stm.(type) {
|
|
|
|
case *sql.ReturnStatement:
|
|
|
|
buf = groupd(buf, rsp)
|
|
|
|
default:
|
|
|
|
buf = append(buf, rsp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *executor) operate(ctx context.Context, stm sql.Statement) (res []interface{}, err error) {
|
|
|
|
|
|
|
|
var loc bool
|
|
|
|
var trw bool
|
|
|
|
var canc context.CancelFunc
|
|
|
|
|
|
|
|
// If the statement is a UseStatement then
|
|
|
|
// there is no need to create a transaction
|
|
|
|
// as the query does not do anything.
|
|
|
|
|
|
|
|
if _, ok := stm.(*sql.UseStatement); ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we are not inside a global transaction
|
|
|
|
// then grab a new transaction, ensuring that
|
|
|
|
// it is closed at the end.
|
|
|
|
|
|
|
|
if e.dbo.TX == nil {
|
|
|
|
|
|
|
|
loc = true
|
|
|
|
|
|
|
|
switch stm := stm.(type) {
|
|
|
|
case sql.WriteableStatement:
|
|
|
|
trw = stm.Writeable()
|
|
|
|
default:
|
|
|
|
trw = false
|
|
|
|
}
|
|
|
|
|
2018-02-06 17:07:42 +00:00
|
|
|
err = e.begin(ctx, trw)
|
2017-11-16 20:53:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer e.dbo.Cancel()
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark the beginning of this statement so we
|
|
|
|
// can monitor the running time, and ensure
|
|
|
|
// it runs no longer than specified.
|
|
|
|
|
|
|
|
if stm, ok := stm.(sql.KillableStatement); ok {
|
|
|
|
if stm.Duration() > 0 {
|
|
|
|
ctx, canc = context.WithTimeout(ctx, stm.Duration())
|
|
|
|
defer func() {
|
|
|
|
if tim := ctx.Err(); err == nil && tim != nil {
|
|
|
|
res, err = nil, &TimerError{timer: stm.Duration()}
|
|
|
|
}
|
2017-12-06 13:21:12 +00:00
|
|
|
canc()
|
2017-11-16 20:53:39 +00:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-31 09:15:29 +00:00
|
|
|
// Get the fibre context ID so that we can use
|
|
|
|
// it to clear or flush websocket notification
|
|
|
|
// changes linked to this context.
|
|
|
|
|
|
|
|
id := ctx.Value(ctxKeyId).(string)
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
// Execute the defined statement, receiving the
|
|
|
|
// result set, and any errors which occured
|
|
|
|
// while processing the query.
|
|
|
|
|
|
|
|
switch stm := stm.(type) {
|
|
|
|
|
|
|
|
case *sql.IfStatement:
|
|
|
|
res, err = e.executeIf(ctx, stm)
|
|
|
|
|
2018-04-14 17:36:28 +00:00
|
|
|
case *sql.RunStatement:
|
|
|
|
res, err = e.executeRun(ctx, stm)
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
case *sql.InfoStatement:
|
|
|
|
res, err = e.executeInfo(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.LetStatement:
|
|
|
|
res, err = e.executeLet(ctx, stm)
|
|
|
|
case *sql.ReturnStatement:
|
|
|
|
res, err = e.executeReturn(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.LiveStatement:
|
|
|
|
res, err = e.executeLive(ctx, stm)
|
|
|
|
case *sql.KillStatement:
|
|
|
|
res, err = e.executeKill(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.SelectStatement:
|
|
|
|
res, err = e.executeSelect(ctx, stm)
|
|
|
|
case *sql.CreateStatement:
|
|
|
|
res, err = e.executeCreate(ctx, stm)
|
|
|
|
case *sql.UpdateStatement:
|
|
|
|
res, err = e.executeUpdate(ctx, stm)
|
|
|
|
case *sql.DeleteStatement:
|
|
|
|
res, err = e.executeDelete(ctx, stm)
|
|
|
|
case *sql.RelateStatement:
|
|
|
|
res, err = e.executeRelate(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.InsertStatement:
|
|
|
|
res, err = e.executeInsert(ctx, stm)
|
|
|
|
case *sql.UpsertStatement:
|
|
|
|
res, err = e.executeUpsert(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineNamespaceStatement:
|
|
|
|
res, err = e.executeDefineNamespace(ctx, stm)
|
|
|
|
case *sql.RemoveNamespaceStatement:
|
|
|
|
res, err = e.executeRemoveNamespace(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineDatabaseStatement:
|
|
|
|
res, err = e.executeDefineDatabase(ctx, stm)
|
|
|
|
case *sql.RemoveDatabaseStatement:
|
|
|
|
res, err = e.executeRemoveDatabase(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineLoginStatement:
|
|
|
|
res, err = e.executeDefineLogin(ctx, stm)
|
|
|
|
case *sql.RemoveLoginStatement:
|
|
|
|
res, err = e.executeRemoveLogin(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineTokenStatement:
|
|
|
|
res, err = e.executeDefineToken(ctx, stm)
|
|
|
|
case *sql.RemoveTokenStatement:
|
|
|
|
res, err = e.executeRemoveToken(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineScopeStatement:
|
|
|
|
res, err = e.executeDefineScope(ctx, stm)
|
|
|
|
case *sql.RemoveScopeStatement:
|
|
|
|
res, err = e.executeRemoveScope(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineTableStatement:
|
|
|
|
res, err = e.executeDefineTable(ctx, stm)
|
|
|
|
case *sql.RemoveTableStatement:
|
|
|
|
res, err = e.executeRemoveTable(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineEventStatement:
|
|
|
|
res, err = e.executeDefineEvent(ctx, stm)
|
|
|
|
case *sql.RemoveEventStatement:
|
|
|
|
res, err = e.executeRemoveEvent(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineFieldStatement:
|
|
|
|
res, err = e.executeDefineField(ctx, stm)
|
|
|
|
case *sql.RemoveFieldStatement:
|
|
|
|
res, err = e.executeRemoveField(ctx, stm)
|
|
|
|
|
|
|
|
case *sql.DefineIndexStatement:
|
|
|
|
res, err = e.executeDefineIndex(ctx, stm)
|
|
|
|
case *sql.RemoveIndexStatement:
|
|
|
|
res, err = e.executeRemoveIndex(ctx, stm)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-04-14 17:02:58 +00:00
|
|
|
// If the context is already closed or failed,
|
2018-01-10 11:33:26 +00:00
|
|
|
// then ignore this result, clear all queued
|
|
|
|
// changes, and reset the transaction.
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
select {
|
|
|
|
|
|
|
|
case <-ctx.Done():
|
|
|
|
|
|
|
|
e.dbo.Cancel()
|
|
|
|
e.dbo.Reset()
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
|
2018-01-10 11:33:26 +00:00
|
|
|
// If this is a local transaction for only the
|
|
|
|
// current statement, then commit or cancel
|
|
|
|
// depending on the result error.
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
if loc && e.dbo.Closed() == false {
|
2018-01-10 11:33:26 +00:00
|
|
|
|
|
|
|
// As this is a local transaction then
|
|
|
|
// make sure we reset the transaction
|
|
|
|
// context.
|
|
|
|
|
|
|
|
defer e.dbo.Reset()
|
|
|
|
|
|
|
|
// If there was an error with the query
|
|
|
|
// then clear the queued changes and
|
|
|
|
// return immediately.
|
|
|
|
|
|
|
|
if err != nil {
|
2017-11-16 20:53:39 +00:00
|
|
|
e.dbo.Cancel()
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise check if this is a read or
|
|
|
|
// a write transaction, and attempt to
|
|
|
|
// Cancel or Commit, returning any errors.
|
|
|
|
|
|
|
|
if !trw {
|
|
|
|
if err = e.dbo.Cancel(); err != nil {
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
} else {
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
}
|
2017-11-16 20:53:39 +00:00
|
|
|
} else {
|
2018-01-10 11:33:26 +00:00
|
|
|
if err = e.dbo.Commit(); err != nil {
|
2018-01-31 09:15:29 +00:00
|
|
|
clear(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
} else {
|
2018-01-31 09:15:29 +00:00
|
|
|
shift(id)
|
2018-01-10 11:33:26 +00:00
|
|
|
}
|
2017-11-16 20:53:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-02-06 17:07:42 +00:00
|
|
|
func (e *executor) begin(ctx context.Context, rw bool) (err error) {
|
2017-11-16 20:53:39 +00:00
|
|
|
if e.dbo.TX == nil {
|
2018-02-06 17:07:42 +00:00
|
|
|
e.dbo.TX, err = db.Begin(ctx, rw)
|
2017-11-16 20:53:39 +00:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *executor) cancel(buf []*Response, err error, chn chan<- *Response) (error, []*Response) {
|
|
|
|
|
2018-01-10 11:33:26 +00:00
|
|
|
defer e.dbo.Reset()
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
if e.dbo.TX == nil {
|
|
|
|
return nil, buf
|
|
|
|
}
|
|
|
|
|
2018-01-10 11:33:26 +00:00
|
|
|
err = e.dbo.Cancel()
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
for _, v := range buf {
|
|
|
|
v.Status = "ERR"
|
2018-01-10 11:33:26 +00:00
|
|
|
v.Result = []interface{}{}
|
|
|
|
v.Detail = "Transaction cancelled"
|
2017-11-16 20:53:39 +00:00
|
|
|
chn <- v
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := len(buf) - 1; i >= 0; i-- {
|
|
|
|
buf[len(buf)-1] = nil
|
|
|
|
buf = buf[:len(buf)-1]
|
|
|
|
}
|
|
|
|
|
2018-01-10 11:33:26 +00:00
|
|
|
return err, buf
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *executor) commit(buf []*Response, err error, chn chan<- *Response) (error, []*Response) {
|
|
|
|
|
2018-01-10 11:33:26 +00:00
|
|
|
defer e.dbo.Reset()
|
2017-11-16 20:53:39 +00:00
|
|
|
|
|
|
|
if e.dbo.TX == nil {
|
|
|
|
return nil, buf
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2018-01-10 11:33:26 +00:00
|
|
|
err = e.dbo.Cancel()
|
2017-11-16 20:53:39 +00:00
|
|
|
} else {
|
2018-01-10 11:33:26 +00:00
|
|
|
err = e.dbo.Commit()
|
2017-11-16 20:53:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range buf {
|
|
|
|
if err != nil {
|
|
|
|
v.Status = "ERR"
|
2018-01-10 11:33:26 +00:00
|
|
|
v.Result = []interface{}{}
|
|
|
|
v.Detail = "Transaction failed: " + err.Error()
|
2017-11-16 20:53:39 +00:00
|
|
|
}
|
|
|
|
chn <- v
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := len(buf) - 1; i >= 0; i-- {
|
|
|
|
buf[len(buf)-1] = nil
|
|
|
|
buf = buf[:len(buf)-1]
|
|
|
|
}
|
|
|
|
|
2018-01-10 11:33:26 +00:00
|
|
|
return err, buf
|
2017-11-16 20:53:39 +00:00
|
|
|
|
2017-02-28 00:17:10 +00:00
|
|
|
}
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
func status(e error) (s string) {
|
|
|
|
switch e.(type) {
|
|
|
|
default:
|
|
|
|
return "OK"
|
|
|
|
case *kvs.DBError:
|
|
|
|
return "ERR_DB"
|
|
|
|
case *kvs.KVError:
|
|
|
|
return "ERR_KV"
|
|
|
|
case *PermsError:
|
|
|
|
return "ERR_PE"
|
|
|
|
case *ExistError:
|
|
|
|
return "ERR_KV"
|
|
|
|
case *FieldError:
|
|
|
|
return "ERR_FD"
|
|
|
|
case *IndexError:
|
|
|
|
return "ERR_IX"
|
|
|
|
case error:
|
|
|
|
return "ERR"
|
|
|
|
}
|
2017-02-23 10:13:13 +00:00
|
|
|
}
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
func detail(e error) (s string) {
|
|
|
|
switch err := e.(type) {
|
|
|
|
default:
|
|
|
|
return
|
|
|
|
case error:
|
|
|
|
return err.Error()
|
|
|
|
}
|
2017-02-23 10:13:13 +00:00
|
|
|
}
|
|
|
|
|
2017-11-16 20:53:39 +00:00
|
|
|
func groupd(buf []*Response, rsp *Response) []*Response {
|
|
|
|
for i := len(buf) - 1; i >= 0; i-- {
|
|
|
|
buf[len(buf)-1] = nil
|
|
|
|
buf = buf[:len(buf)-1]
|
|
|
|
}
|
|
|
|
return append(buf, rsp)
|
2017-02-23 10:13:13 +00:00
|
|
|
}
|