You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
250 lines
7.2 KiB
250 lines
7.2 KiB
//
|
|
// DISCLAIMER
|
|
//
|
|
// Copyright 2017 ArangoDB GmbH, Cologne, Germany
|
|
//
|
|
// 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.
|
|
//
|
|
// Copyright holder is ArangoDB GmbH, Cologne, Germany
|
|
//
|
|
// Author Ewout Prangsma
|
|
//
|
|
|
|
package driver
|
|
|
|
import (
|
|
"context"
|
|
"net/http"
|
|
"path"
|
|
)
|
|
|
|
// newDatabase creates a new Database implementation.
|
|
func newDatabase(name string, conn Connection) (Database, error) {
|
|
if name == "" {
|
|
return nil, WithStack(InvalidArgumentError{Message: "name is empty"})
|
|
}
|
|
if conn == nil {
|
|
return nil, WithStack(InvalidArgumentError{Message: "conn is nil"})
|
|
}
|
|
return &database{
|
|
name: name,
|
|
conn: conn,
|
|
}, nil
|
|
}
|
|
|
|
// database implements the Database interface.
|
|
type database struct {
|
|
name string
|
|
conn Connection
|
|
}
|
|
|
|
// relPath creates the relative path to this database (`_db/<name>`)
|
|
func (d *database) relPath() string {
|
|
escapedName := pathEscape(d.name)
|
|
return path.Join("_db", escapedName)
|
|
}
|
|
|
|
// Name returns the name of the database.
|
|
func (d *database) Name() string {
|
|
return d.name
|
|
}
|
|
|
|
// Info fetches information about the database.
|
|
func (d *database) Info(ctx context.Context) (DatabaseInfo, error) {
|
|
req, err := d.conn.NewRequest("GET", path.Join(d.relPath(), "_api/database/current"))
|
|
if err != nil {
|
|
return DatabaseInfo{}, WithStack(err)
|
|
}
|
|
applyContextSettings(ctx, req)
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return DatabaseInfo{}, WithStack(err)
|
|
}
|
|
if err := resp.CheckStatus(200); err != nil {
|
|
return DatabaseInfo{}, WithStack(err)
|
|
}
|
|
var data DatabaseInfo
|
|
if err := resp.ParseBody("result", &data); err != nil {
|
|
return DatabaseInfo{}, WithStack(err)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
// EngineInfo returns information about the database engine being used.
|
|
// Note: When your cluster has multiple endpoints (cluster), you will get information
|
|
// from the server that is currently being used.
|
|
// If you want to know exactly which server the information is from, use a client
|
|
// with only a single endpoint and avoid automatic synchronization of endpoints.
|
|
func (d *database) EngineInfo(ctx context.Context) (EngineInfo, error) {
|
|
req, err := d.conn.NewRequest("GET", path.Join(d.relPath(), "_api/engine"))
|
|
if err != nil {
|
|
return EngineInfo{}, WithStack(err)
|
|
}
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return EngineInfo{}, WithStack(err)
|
|
}
|
|
if err := resp.CheckStatus(200, 404); err != nil {
|
|
return EngineInfo{}, WithStack(err)
|
|
}
|
|
if resp.StatusCode() == 404 {
|
|
// On version 3.1, this endpoint is not yet supported
|
|
return EngineInfo{Type: EngineTypeMMFiles}, nil
|
|
}
|
|
var data EngineInfo
|
|
if err := resp.ParseBody("", &data); err != nil {
|
|
return EngineInfo{}, WithStack(err)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
// Remove removes the entire database.
|
|
// If the database does not exist, a NotFoundError is returned.
|
|
func (d *database) Remove(ctx context.Context) error {
|
|
req, err := d.conn.NewRequest("DELETE", path.Join("_db/_system/_api/database", pathEscape(d.name)))
|
|
if err != nil {
|
|
return WithStack(err)
|
|
}
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return WithStack(err)
|
|
}
|
|
if err := resp.CheckStatus(200); err != nil {
|
|
return WithStack(err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Query performs an AQL query, returning a cursor used to iterate over the returned documents.
|
|
func (d *database) Query(ctx context.Context, query string, bindVars map[string]interface{}) (Cursor, error) {
|
|
req, err := d.conn.NewRequest("POST", path.Join(d.relPath(), "_api/cursor"))
|
|
if err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
input := queryRequest{
|
|
Query: query,
|
|
BindVars: bindVars,
|
|
}
|
|
input.applyContextSettings(ctx)
|
|
if _, err := req.SetBody(input); err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
cs := applyContextSettings(ctx, req)
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
if err := resp.CheckStatus(201); err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
var data cursorData
|
|
if err := resp.ParseBody("", &data); err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
col, err := newCursor(data, resp.Endpoint(), d, cs.AllowDirtyReads)
|
|
if err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
return col, nil
|
|
}
|
|
|
|
// ValidateQuery validates an AQL query.
|
|
// When the query is valid, nil returned, otherwise an error is returned.
|
|
// The query is not executed.
|
|
func (d *database) ValidateQuery(ctx context.Context, query string) error {
|
|
req, err := d.conn.NewRequest("POST", path.Join(d.relPath(), "_api/query"))
|
|
if err != nil {
|
|
return WithStack(err)
|
|
}
|
|
input := parseQueryRequest{
|
|
Query: query,
|
|
}
|
|
if _, err := req.SetBody(input); err != nil {
|
|
return WithStack(err)
|
|
}
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return WithStack(err)
|
|
}
|
|
if err := resp.CheckStatus(200); err != nil {
|
|
return WithStack(err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// OptimizerRulesForQueries returns the available optimizer rules for AQL query
|
|
// returns an array of objects that contain the name of each available rule and its respective flags.
|
|
func (d *database) OptimizerRulesForQueries(ctx context.Context) ([]QueryRule, error) {
|
|
req, err := d.conn.NewRequest("GET", path.Join(d.relPath(), "_api/query/rules"))
|
|
if err != nil {
|
|
return []QueryRule{}, WithStack(err)
|
|
}
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return []QueryRule{}, WithStack(err)
|
|
}
|
|
if err := resp.CheckStatus(200); err != nil {
|
|
return []QueryRule{}, WithStack(err)
|
|
}
|
|
|
|
var data []QueryRule
|
|
responses, err := resp.ParseArrayBody()
|
|
if err != nil {
|
|
return []QueryRule{}, WithStack(err)
|
|
}
|
|
|
|
for _, response := range responses {
|
|
var rule QueryRule
|
|
if err := response.ParseBody("", &rule); err != nil {
|
|
return []QueryRule{}, WithStack(err)
|
|
}
|
|
data = append(data, rule)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
func (d *database) Transaction(ctx context.Context, action string, options *TransactionOptions) (interface{}, error) {
|
|
req, err := d.conn.NewRequest("POST", path.Join(d.relPath(), "_api/transaction"))
|
|
if err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
input := transactionRequest{Action: action}
|
|
if options != nil {
|
|
input.MaxTransactionSize = options.MaxTransactionSize
|
|
input.LockTimeout = options.LockTimeout
|
|
input.WaitForSync = options.WaitForSync
|
|
input.IntermediateCommitCount = options.IntermediateCommitCount
|
|
input.Params = options.Params
|
|
input.IntermediateCommitSize = options.IntermediateCommitSize
|
|
input.Collections.Read = options.ReadCollections
|
|
input.Collections.Write = options.WriteCollections
|
|
input.Collections.Exclusive = options.ExclusiveCollections
|
|
}
|
|
if _, err = req.SetBody(input); err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
resp, err := d.conn.Do(ctx, req)
|
|
if err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
if err = resp.CheckStatus(http.StatusOK); err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
|
|
output := &transactionResponse{}
|
|
if err = resp.ParseBody("", output); err != nil {
|
|
return nil, WithStack(err)
|
|
}
|
|
|
|
return output.Result, nil
|
|
}
|
|
|