Raw File
session.go
// Copyright 2014 The Cayley Authors. All rights reserved.
//
// 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 sexp

// Defines a running session of the sexp query language.

import (
	"context"
	"errors"
	"fmt"
	"sort"

	"github.com/cayleygraph/cayley/graph"
	"github.com/cayleygraph/cayley/graph/iterator"
	"github.com/cayleygraph/cayley/query"
)

const Name = "sexp"

func init() {
	query.RegisterLanguage(query.Language{
		Name: Name,
		Session: func(qs graph.QuadStore) query.Session {
			return NewSession(qs)
		},
	})
}

type Session struct {
	qs graph.QuadStore
}

func NewSession(qs graph.QuadStore) *Session {
	return &Session{qs: qs}
}

func (s *Session) Parse(input string) error {
	var parenDepth int
	for i, x := range input {
		if x == '(' {
			parenDepth++
		}
		if x == ')' {
			parenDepth--
			if parenDepth < 0 {
				min := 0
				if (i - 10) > min {
					min = i - 10
				}
				return fmt.Errorf("too many close parentheses at char %d: %s", i, input[min:i])
			}
		}
	}
	if parenDepth > 0 {
		return query.ErrParseMore
	}
	if len(ParseString(input)) > 0 {
		return nil
	}
	return errors.New("invalid syntax")
}

func (s *Session) Execute(ctx context.Context, input string, opt query.Options) (query.Iterator, error) {
	switch opt.Collation {
	case query.Raw, query.REPL:
	default:
		return nil, &query.ErrUnsupportedCollation{Collation: opt.Collation}
	}
	it := BuildIteratorTreeForQuery(ctx, s.qs, input).Iterate()
	if err := it.Err(); err != nil {
		return nil, err
	}
	if opt.Limit > 0 {
		it = iterator.NewLimitNext(it, int64(opt.Limit))
	}
	return &results{
		s:   s,
		col: opt.Collation,
		it:  it,
	}, nil
}

type results struct {
	s        *Session
	col      query.Collation
	it       iterator.Scanner
	nextPath bool
}

func (it *results) Next(ctx context.Context) bool {
	if it.nextPath && it.it.NextPath(ctx) {
		return true
	}
	it.nextPath = false
	if it.it.Next(ctx) {
		it.nextPath = true
		return true
	}
	return false
}

func (it *results) Result() interface{} {
	m := make(map[string]graph.Ref)
	it.it.TagResults(m)
	if it.col == query.Raw {
		return m
	}
	out := "****\n"
	tagKeys := make([]string, len(m))
	i := 0
	for k := range m {
		tagKeys[i] = k
		i++
	}
	sort.Strings(tagKeys)
	for _, k := range tagKeys {
		if k == "$_" {
			continue
		}
		out += fmt.Sprintf("%s : %s\n", k, it.s.qs.NameOf(m[k]))
	}
	return out
}

func (it *results) Err() error {
	return it.it.Err()
}

func (it *results) Close() error {
	return it.it.Close()
}
back to top