Skip to content
Snippets Groups Projects
parse.go 4.08 KiB
Newer Older
Manish R Jain's avatar
Manish R Jain committed
/*
 * Copyright 2015 DGraph Labs, Inc.
Manish R Jain's avatar
Manish R Jain committed
 *
 * 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 rdf

import (
	"fmt"
Manish R Jain's avatar
Manish R Jain committed

	"github.com/dgraph-io/dgraph/lex"
	"github.com/dgraph-io/dgraph/uid"
	"github.com/dgraph-io/dgraph/x"
Manish R Jain's avatar
Manish R Jain committed
)

type NQuad struct {
	Subject     string
	Predicate   string
	ObjectId    string
	ObjectValue interface{}
	Label       string
}

func getUid(xid string) (uint64, error) {
	if strings.HasPrefix(xid, "_uid_:") {
		return strconv.ParseUint(xid[6:], 0, 64)
// ToEdge is useful when you want to find the UID corresponding to XID for
// just one edge. ToEdgeUsing(map) is useful when you do this conversion
// in bulk, say over a network call. None of these methods generate a UID
// for an XID.
func (nq NQuad) ToEdge() (result x.DirectedEdge, rerr error) {
	if err != nil {
		return result, err
	}
	result.Entity = sid
	if len(nq.ObjectId) > 0 {
		if err != nil {
			return result, err
		}
		result.ValueId = oid
	} else {
		result.Value = nq.ObjectValue
	}
	result.Source = nq.Label
	result.Timestamp = time.Now()
	return result, nil
}

func toUid(xid string, xidToUid map[string]uint64) (uid uint64, rerr error) {
	if id, present := xidToUid[xid]; present {
		return id, nil
	}

	if !strings.HasPrefix(xid, "_uid_:") {
		return 0, fmt.Errorf("Unable to find xid: %v", xid)
	}
	return strconv.ParseUint(xid[6:], 0, 64)
}

func (nq NQuad) ToEdgeUsing(
	xidToUid map[string]uint64) (result x.DirectedEdge, rerr error) {
	uid, err := toUid(nq.Subject, xidToUid)
	if err != nil {
		return result, err
	}
	result.Entity = uid

	if len(nq.ObjectId) == 0 {
		result.Value = nq.ObjectValue
	} else {
		uid, err = toUid(nq.ObjectId, xidToUid)
		if err != nil {
			return result, err
		}
		result.ValueId = uid
	}
	result.Source = nq.Label
	result.Timestamp = time.Now()
	return result, nil
}

Manish R Jain's avatar
Manish R Jain committed
func stripBracketsIfPresent(val string) string {
	if val[0] != '<' {
		return val
	}
	if val[len(val)-1] != '>' {
		return val
	}
	return val[1 : len(val)-1]
}

func Parse(line string) (rnq NQuad, rerr error) {
Manish R Jain's avatar
Manish R Jain committed
	go run(l)
	var oval string
Manish R Jain's avatar
Manish R Jain committed
	var vend bool
Manish R Jain's avatar
Manish R Jain committed
	for item := range l.Items {
		if item.Typ == itemSubject {
			rnq.Subject = stripBracketsIfPresent(item.Val)
		}
		if item.Typ == itemPredicate {
			rnq.Predicate = stripBracketsIfPresent(item.Val)
		}
		if item.Typ == itemObject {
			rnq.ObjectId = stripBracketsIfPresent(item.Val)
		}
		if item.Typ == itemLiteral {
			oval = item.Val
		}
		if item.Typ == itemLanguage {
Manish R Jain's avatar
Manish R Jain committed
		}
		if item.Typ == itemObjectType {
Manish R Jain's avatar
Manish R Jain committed
			// TODO: Strictly parse common types like integers, floats etc.
Manish R Jain's avatar
Manish R Jain committed
			if len(oval) == 0 {
				glog.Fatalf(
					"itemObject should be emitted before itemObjectType. Input: [%s]",
Manish R Jain's avatar
Manish R Jain committed
			}
			oval += "@@" + stripBracketsIfPresent(item.Val)
		}
Manish R Jain's avatar
Manish R Jain committed
		if item.Typ == lex.ItemError {
			return rnq, fmt.Errorf(item.Val)
		}
		if item.Typ == itemValidEnd {
			vend = true
		}
		if item.Typ == itemLabel {
			rnq.Label = stripBracketsIfPresent(item.Val)
		}
Manish R Jain's avatar
Manish R Jain committed
	}
	if !vend {
		return rnq, fmt.Errorf("Invalid end of input. Input: [%s]", line)
Manish R Jain's avatar
Manish R Jain committed
	}
	if len(oval) > 0 {
		rnq.ObjectValue = oval
	}
	if len(rnq.Subject) == 0 || len(rnq.Predicate) == 0 {
		return rnq, fmt.Errorf("Empty required fields in NQuad. Input: [%s]", line)
Manish R Jain's avatar
Manish R Jain committed
	}
	if len(rnq.ObjectId) == 0 && rnq.ObjectValue == nil {
		return rnq, fmt.Errorf("No Object in NQuad. Input: [%s]", line)
Manish R Jain's avatar
Manish R Jain committed
	}
Manish R Jain's avatar
Manish R Jain committed
	return rnq, nil
}

func isNewline(r rune) bool {
	return r == '\n' || r == '\r'
}