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

//
// 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
}