forked from denodrivers/postgres
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharray_parser.ts
119 lines (108 loc) · 3.15 KB
/
array_parser.ts
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
// Based of https://github.com/bendrucker/postgres-array
// Copyright (c) Ben Drucker <[email protected]> (bendrucker.me). MIT License.
type AllowedSeparators = "," | ";";
/** Incorrectly parsed data types default to null */
type ArrayResult<T> = Array<T | null | ArrayResult<T>>;
type Transformer<T> = (value: string) => T;
export type ParseArrayFunction = typeof parseArray;
/**
* Parse a string into an array of values using the provided transform function.
*
* @param source The string to parse
* @param transform A function to transform each value in the array
* @param separator The separator used to split the string into values
* @returns
*/
export function parseArray<T>(
source: string,
transform: Transformer<T>,
separator: AllowedSeparators = ",",
): ArrayResult<T> {
return new ArrayParser(source, transform, separator).parse();
}
class ArrayParser<T> {
position = 0;
entries: ArrayResult<T> = [];
recorded: string[] = [];
dimension = 0;
constructor(
public source: string,
public transform: Transformer<T>,
public separator: AllowedSeparators,
) {}
isEof(): boolean {
return this.position >= this.source.length;
}
nextCharacter() {
const character = this.source[this.position++];
if (character === "\\") {
return {
escaped: true,
value: this.source[this.position++],
};
}
return {
escaped: false,
value: character,
};
}
record(character: string): void {
this.recorded.push(character);
}
newEntry(includeEmpty = false): void {
let entry;
if (this.recorded.length > 0 || includeEmpty) {
entry = this.recorded.join("");
if (entry === "NULL" && !includeEmpty) {
entry = null;
}
if (entry !== null) entry = this.transform(entry);
this.entries.push(entry);
this.recorded = [];
}
}
consumeDimensions(): void {
if (this.source[0] === "[") {
while (!this.isEof()) {
const char = this.nextCharacter();
if (char.value === "=") break;
}
}
}
parse(nested = false): ArrayResult<T> {
let character, parser, quote;
this.consumeDimensions();
while (!this.isEof()) {
character = this.nextCharacter();
if (character.value === "{" && !quote) {
this.dimension++;
if (this.dimension > 1) {
parser = new ArrayParser(
this.source.substr(this.position - 1),
this.transform,
this.separator,
);
this.entries.push(parser.parse(true));
this.position += parser.position - 2;
}
} else if (character.value === "}" && !quote) {
this.dimension--;
if (!this.dimension) {
this.newEntry();
if (nested) return this.entries;
}
} else if (character.value === '"' && !character.escaped) {
if (quote) this.newEntry(true);
quote = !quote;
} else if (character.value === this.separator && !quote) {
this.newEntry();
} else {
this.record(character.value);
}
}
if (this.dimension !== 0) {
throw new Error("array dimension not balanced");
}
return this.entries;
}
}