Newer
Older
*
* 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
committed
"strconv"
"strings"
"github.com/dgraph-io/dgraph/uid"
"github.com/dgraph-io/dgraph/x"
)
type NQuad struct {
Subject string
Predicate string
ObjectId string
ObjectValue interface{}
Label string
}
Manish R Jain
committed
func getUid(xid string) (uint64, error) {
if strings.HasPrefix(xid, "_uid_:") {
return strconv.ParseUint(xid[6:], 0, 64)
Manish R Jain
committed
}
Manish R Jain
committed
return uid.Get(xid)
Manish R Jain
committed
}
Manish R Jain
committed
// 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) {
Manish R Jain
committed
sid, err := getUid(nq.Subject)
if err != nil {
return result, err
}
result.Entity = sid
if len(nq.ObjectId) > 0 {
Manish R Jain
committed
oid, err := getUid(nq.ObjectId)
if err != nil {
return result, err
}
result.ValueId = oid
} else {
result.Value = nq.ObjectValue
}
Manish R Jain
committed
result.Attribute = nq.Predicate
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
}
Manish R Jain
committed
result.Attribute = nq.Predicate
result.Source = nq.Label
result.Timestamp = time.Now()
return result, nil
}
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
committed
l := &lex.Lexer{}
l.Init(line)
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
committed
rnq.Predicate += "." + item.Val
// TODO: Strictly parse common types like integers, floats etc.
"itemObject should be emitted before itemObjectType. Input: [%s]",
}
oval += "@@" + stripBracketsIfPresent(item.Val)
}
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)
}
return rnq, fmt.Errorf("Invalid end of input. Input: [%s]", line)
}
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)
}
if len(rnq.ObjectId) == 0 && rnq.ObjectValue == nil {
return rnq, fmt.Errorf("No Object in NQuad. Input: [%s]", line)
Manish R Jain
committed
func isNewline(r rune) bool {
return r == '\n' || r == '\r'
}