summaryrefslogtreecommitdiff
path: root/vendor/github.com/uptrace/bun/dialect/pgdialect/array_parser.go
blob: 56db9fd077d8af40c7cb2fed3c8e9eefce952f1b (plain)
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
115
116
117
118
119
package pgdialect

import (
	"bytes"
	"fmt"
	"io"
)

type arrayParser struct {
	p pgparser

	elem []byte
	err  error

	isJson bool
}

func newArrayParser(b []byte) *arrayParser {
	p := new(arrayParser)

	if b[0] == 'n' {
		p.p.Reset(nil)
		return p
	}

	if len(b) < 2 || (b[0] != '{' && b[0] != '[') || (b[len(b)-1] != '}' && b[len(b)-1] != ']') {
		p.err = fmt.Errorf("pgdialect: can't parse array: %q", b)
		return p
	}
	p.isJson = b[0] == '['

	p.p.Reset(b[1 : len(b)-1])
	return p
}

func (p *arrayParser) Next() bool {
	if p.err != nil {
		return false
	}
	p.err = p.readNext()
	return p.err == nil
}

func (p *arrayParser) Err() error {
	if p.err != io.EOF {
		return p.err
	}
	return nil
}

func (p *arrayParser) Elem() []byte {
	return p.elem
}

func (p *arrayParser) readNext() error {
	ch := p.p.Read()
	if ch == 0 {
		return io.EOF
	}

	switch ch {
	case '}', ']':
		return io.EOF
	case '"':
		b, err := p.p.ReadSubstring(ch)
		if err != nil {
			return err
		}

		if p.p.Peek() == ',' {
			p.p.Advance()
		}

		p.elem = b
		return nil
	case '[', '(':
		rng, err := p.p.ReadRange(ch)
		if err != nil {
			return err
		}

		if p.p.Peek() == ',' {
			p.p.Advance()
		}

		p.elem = rng
		return nil
	default:
		if ch == '{' && p.isJson {
			json, err := p.p.ReadJSON()
			if err != nil {
				return err
			}

			for {
				if p.p.Peek() == ',' || p.p.Peek() == ' ' {
					p.p.Advance()
				} else {
					break
				}
			}

			p.elem = json
			return nil
		} else {
			lit := p.p.ReadLiteral(ch)
			if bytes.Equal(lit, []byte("NULL")) {
				lit = nil
			}

			if p.p.Peek() == ',' {
				p.p.Advance()
			}

			p.elem = lit
			return nil
		}
	}
}