File: util.go

package info (click to toggle)
golang-github-facebook-ent 0.5.4-3
  • links: PTS, VCS
  • area: main
  • in suites: bookworm
  • size: 14,284 kB
  • sloc: javascript: 349; makefile: 8
file content (114 lines) | stat: -rw-r--r-- 2,688 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
// Copyright 2019-present Facebook Inc. All rights reserved.
// This source code is licensed under the Apache 2.0 license found
// in the LICENSE file in the root directory of this source tree.

package graphson

import (
	"io"
	"unsafe"

	jsoniter "github.com/json-iterator/go"
	"github.com/pkg/errors"
)

// graphson encoding type / value keys
const (
	TypeKey  = "@type"
	ValueKey = "@value"
)

// typeEncoder adds graphson type information to a value encoder.
type typeEncoder struct {
	jsoniter.ValEncoder
	Type Type
}

// Encode belongs to jsoniter.ValEncoder interface.
func (enc typeEncoder) Encode(ptr unsafe.Pointer, stream *jsoniter.Stream) {
	stream.WriteObjectStart()
	stream.WriteObjectField(TypeKey)
	stream.WriteString(enc.Type.String())
	stream.WriteMore()
	stream.WriteObjectField(ValueKey)
	enc.ValEncoder.Encode(ptr, stream)
	stream.WriteObjectEnd()
}

type (
	// typeDecoder decorates a value decoder and adds graphson type verification.
	typeDecoder struct {
		jsoniter.ValDecoder
		typeChecker
	}

	// typeChecker defines an interface for graphson type verification.
	typeChecker interface {
		CheckType(Type) error
	}

	// typeCheckerFunc allows the use of functions as type checkers.
	typeCheckerFunc func(Type) error

	// typeValue defines a graphson type / value pair.
	typeValue struct {
		Type  Type
		Value jsoniter.RawMessage
	}
)

// Decode belongs to jsoniter.ValDecoder interface.
func (dec typeDecoder) Decode(ptr unsafe.Pointer, iter *jsoniter.Iterator) {
	if iter.WhatIsNext() != jsoniter.ObjectValue {
		dec.ValDecoder.Decode(ptr, iter)
		return
	}

	data := iter.SkipAndReturnBytes()
	if iter.Error != nil && iter.Error != io.EOF {
		return
	}

	var tv typeValue
	if err := jsoniter.Unmarshal(data, &tv); err != nil {
		iter.ReportError("unmarshal type value", err.Error())
		return
	}

	if err := dec.CheckType(tv.Type); err != nil {
		iter.ReportError("check type", err.Error())
		return
	}

	it := config.BorrowIterator(tv.Value)
	defer config.ReturnIterator(it)

	dec.ValDecoder.Decode(ptr, it)
	if it.Error != nil && it.Error != io.EOF {
		iter.ReportError("decode value", it.Error.Error())
	}
}

// UnmarshalJSON implements json.Unmarshaler interface.
func (tv *typeValue) UnmarshalJSON(data []byte) error {
	var v struct {
		Type  *Type               `json:"@type"`
		Value jsoniter.RawMessage `json:"@value"`
	}

	if err := jsoniter.Unmarshal(data, &v); err != nil {
		return err
	}
	if v.Type == nil || v.Value == nil {
		return errors.New("missing type or value")
	}

	tv.Type = *v.Type
	tv.Value = v.Value
	return nil
}

// CheckType implements typeChecker interface.
func (f typeCheckerFunc) CheckType(typ Type) error {
	return f(typ)
}