nokogumbo 1.5.0 → 2.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +237 -26
- data/ext/nokogumbo/extconf.rb +121 -0
- data/ext/nokogumbo/nokogumbo.c +793 -0
- data/gumbo-parser/src/ascii.c +75 -0
- data/gumbo-parser/src/ascii.h +115 -0
- data/gumbo-parser/src/attribute.c +26 -28
- data/gumbo-parser/src/attribute.h +3 -23
- data/gumbo-parser/src/char_ref.c +5972 -6816
- data/gumbo-parser/src/char_ref.h +14 -45
- data/gumbo-parser/src/error.c +510 -163
- data/gumbo-parser/src/error.h +70 -147
- data/gumbo-parser/src/foreign_attrs.c +104 -0
- data/gumbo-parser/src/gumbo.h +577 -305
- data/gumbo-parser/src/insertion_mode.h +4 -28
- data/gumbo-parser/src/macros.h +91 -0
- data/gumbo-parser/src/parser.c +2922 -2228
- data/gumbo-parser/src/parser.h +6 -22
- data/gumbo-parser/src/replacement.h +33 -0
- data/gumbo-parser/src/string_buffer.c +43 -50
- data/gumbo-parser/src/string_buffer.h +24 -40
- data/gumbo-parser/src/string_piece.c +39 -39
- data/gumbo-parser/src/svg_attrs.c +174 -0
- data/gumbo-parser/src/svg_tags.c +137 -0
- data/gumbo-parser/src/tag.c +186 -59
- data/gumbo-parser/src/tag_lookup.c +382 -0
- data/gumbo-parser/src/tag_lookup.h +13 -0
- data/gumbo-parser/src/token_buffer.c +79 -0
- data/gumbo-parser/src/token_buffer.h +71 -0
- data/gumbo-parser/src/token_type.h +1 -25
- data/gumbo-parser/src/tokenizer.c +2127 -1561
- data/gumbo-parser/src/tokenizer.h +41 -52
- data/gumbo-parser/src/tokenizer_states.h +281 -45
- data/gumbo-parser/src/utf8.c +98 -123
- data/gumbo-parser/src/utf8.h +84 -52
- data/gumbo-parser/src/util.c +48 -38
- data/gumbo-parser/src/util.h +10 -40
- data/gumbo-parser/src/vector.c +45 -57
- data/gumbo-parser/src/vector.h +17 -39
- data/lib/nokogumbo.rb +11 -173
- data/lib/nokogumbo/html5.rb +252 -0
- data/lib/nokogumbo/html5/document.rb +53 -0
- data/lib/nokogumbo/html5/document_fragment.rb +62 -0
- data/lib/nokogumbo/html5/node.rb +72 -0
- data/lib/nokogumbo/version.rb +3 -0
- metadata +43 -24
- data/ext/nokogumboc/extconf.rb +0 -60
- data/ext/nokogumboc/nokogumbo.c +0 -295
- data/gumbo-parser/src/char_ref.rl +0 -2554
- data/gumbo-parser/src/string_piece.h +0 -38
- data/gumbo-parser/src/tag.in +0 -150
- data/gumbo-parser/src/tag_enum.h +0 -153
- data/gumbo-parser/src/tag_gperf.h +0 -105
- data/gumbo-parser/src/tag_sizes.h +0 -4
- data/gumbo-parser/src/tag_strings.h +0 -153
- data/gumbo-parser/visualc/include/strings.h +0 -4
- data/test-nokogumbo.rb +0 -190
data/gumbo-parser/src/char_ref.h
CHANGED
@@ -1,60 +1,29 @@
|
|
1
|
-
// Copyright 2011 Google Inc. All Rights Reserved.
|
2
|
-
//
|
3
|
-
// Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
-
// you may not use this file except in compliance with the License.
|
5
|
-
// You may obtain a copy of the License at
|
6
|
-
//
|
7
|
-
// http://www.apache.org/licenses/LICENSE-2.0
|
8
|
-
//
|
9
|
-
// Unless required by applicable law or agreed to in writing, software
|
10
|
-
// distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
-
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
-
// See the License for the specific language governing permissions and
|
13
|
-
// limitations under the License.
|
14
|
-
//
|
15
|
-
// Author: jdtang@google.com (Jonathan Tang)
|
16
|
-
//
|
17
|
-
// Internal header for character reference handling; this should not be exposed
|
18
|
-
// transitively by any public API header. This is why the functions aren't
|
19
|
-
// namespaced.
|
20
|
-
|
21
1
|
#ifndef GUMBO_CHAR_REF_H_
|
22
2
|
#define GUMBO_CHAR_REF_H_
|
23
3
|
|
24
|
-
#include <
|
4
|
+
#include <stdlib.h>
|
25
5
|
|
26
6
|
#ifdef __cplusplus
|
27
7
|
extern "C" {
|
28
8
|
#endif
|
29
9
|
|
30
|
-
struct GumboInternalParser;
|
31
|
-
struct GumboInternalUtf8Iterator;
|
32
|
-
|
33
10
|
// Value that indicates no character was produced.
|
34
|
-
|
11
|
+
#define kGumboNoChar (-1)
|
35
12
|
|
36
|
-
//
|
37
|
-
//
|
38
|
-
//
|
39
|
-
//
|
40
|
-
//
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
// This reads in characters from the input as necessary, and fills in a
|
48
|
-
// OneOrTwoCodepoints struct containing the characters read. It may add parse
|
49
|
-
// errors to the GumboParser's errors vector, if the spec calls for it. Pass a
|
50
|
-
// space for the "additional allowed char" when the spec says "with no
|
51
|
-
// additional allowed char". Returns false on parse error, true otherwise.
|
52
|
-
bool consume_char_ref(struct GumboInternalParser* parser,
|
53
|
-
struct GumboInternalUtf8Iterator* input, int additional_allowed_char,
|
54
|
-
bool is_in_attribute, OneOrTwoCodepoints* output);
|
13
|
+
// On input, str points to the start of the string to match and size is the
|
14
|
+
// size of the string.
|
15
|
+
//
|
16
|
+
// Returns the length of the match or 0 if there is no match.
|
17
|
+
// output[0] contains the first codepoint and output[1] contains the second if
|
18
|
+
// there are two, otherwise output[1] contains kGumboNoChar.
|
19
|
+
size_t match_named_char_ref (
|
20
|
+
const char *str,
|
21
|
+
size_t size,
|
22
|
+
int output[2]
|
23
|
+
);
|
55
24
|
|
56
25
|
#ifdef __cplusplus
|
57
26
|
}
|
58
27
|
#endif
|
59
28
|
|
60
|
-
#endif
|
29
|
+
#endif // GUMBO_CHAR_REF_H_
|
data/gumbo-parser/src/error.c
CHANGED
@@ -1,279 +1,626 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
// Licensed under the Apache License, Version 2.0 (the "License");
|
4
|
-
// you may not use this file except in compliance with the License.
|
5
|
-
// You may obtain a copy of the License at
|
6
|
-
//
|
7
|
-
// http://www.apache.org/licenses/LICENSE-2.0
|
8
|
-
//
|
9
|
-
// Unless required by applicable law or agreed to in writing, software
|
10
|
-
// distributed under the License is distributed on an "AS IS" BASIS,
|
11
|
-
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
12
|
-
// See the License for the specific language governing permissions and
|
13
|
-
// limitations under the License.
|
14
|
-
//
|
15
|
-
// Author: jdtang@google.com (Jonathan Tang)
|
1
|
+
/*
|
2
|
+
Copyright 2010 Google Inc.
|
16
3
|
|
17
|
-
|
4
|
+
Licensed under the Apache License, Version 2.0 (the "License");
|
5
|
+
you may not use this file except in compliance with the License.
|
6
|
+
You may obtain a copy of the License at
|
7
|
+
|
8
|
+
https://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
|
10
|
+
Unless required by applicable law or agreed to in writing, software
|
11
|
+
distributed under the License is distributed on an "AS IS" BASIS,
|
12
|
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
13
|
+
See the License for the specific language governing permissions and
|
14
|
+
limitations under the License.
|
15
|
+
*/
|
18
16
|
|
19
17
|
#include <assert.h>
|
18
|
+
#include <inttypes.h>
|
20
19
|
#include <stdarg.h>
|
21
20
|
#include <stdio.h>
|
22
21
|
#include <string.h>
|
23
|
-
|
22
|
+
#include "ascii.h"
|
23
|
+
#include "error.h"
|
24
24
|
#include "gumbo.h"
|
25
|
+
#include "macros.h"
|
25
26
|
#include "parser.h"
|
26
27
|
#include "string_buffer.h"
|
27
28
|
#include "util.h"
|
28
29
|
#include "vector.h"
|
29
30
|
|
30
|
-
// Prints a formatted message to a StringBuffer.
|
31
|
-
// StringBuffer as necessary to fit the message.
|
31
|
+
// Prints a formatted message to a StringBuffer. This automatically resizes the
|
32
|
+
// StringBuffer as necessary to fit the message. Returns the number of bytes
|
32
33
|
// written.
|
33
|
-
static int print_message(
|
34
|
-
|
34
|
+
static int PRINTF(2) print_message (
|
35
|
+
GumboStringBuffer* output,
|
36
|
+
const char* format,
|
37
|
+
...
|
38
|
+
) {
|
35
39
|
va_list args;
|
36
40
|
int remaining_capacity = output->capacity - output->length;
|
37
41
|
va_start(args, format);
|
38
|
-
int bytes_written = vsnprintf(
|
39
|
-
|
42
|
+
int bytes_written = vsnprintf (
|
43
|
+
output->data + output->length,
|
44
|
+
remaining_capacity,
|
45
|
+
format,
|
46
|
+
args
|
47
|
+
);
|
40
48
|
va_end(args);
|
41
|
-
#
|
49
|
+
#if _MSC_VER && _MSC_VER < 1900
|
42
50
|
if (bytes_written == -1) {
|
43
|
-
// vsnprintf returns -1 on MSVC++ if there's not enough capacity,
|
44
|
-
// returning the number of bytes that would've been written had
|
45
|
-
// enough.
|
46
|
-
// we retry (letting it fail and returning 0 if it doesn't),
|
47
|
-
// no way to smartly resize the buffer.
|
48
|
-
gumbo_string_buffer_reserve(
|
51
|
+
// vsnprintf returns -1 on older MSVC++ if there's not enough capacity,
|
52
|
+
// instead of returning the number of bytes that would've been written had
|
53
|
+
// there been enough. In this case, we'll double the buffer size and hope
|
54
|
+
// it fits when we retry (letting it fail and returning 0 if it doesn't),
|
55
|
+
// since there's no way to smartly resize the buffer.
|
56
|
+
gumbo_string_buffer_reserve(output->capacity * 2, output);
|
49
57
|
va_start(args, format);
|
50
|
-
int result = vsnprintf(
|
51
|
-
|
58
|
+
int result = vsnprintf (
|
59
|
+
output->data + output->length,
|
60
|
+
remaining_capacity,
|
61
|
+
format,
|
62
|
+
args
|
63
|
+
);
|
52
64
|
va_end(args);
|
53
65
|
return result == -1 ? 0 : result;
|
54
66
|
}
|
55
67
|
#else
|
56
|
-
// -1 in standard C99 indicates an encoding error.
|
68
|
+
// -1 in standard C99 indicates an encoding error. Return 0 and do nothing.
|
57
69
|
if (bytes_written == -1) {
|
58
70
|
return 0;
|
59
71
|
}
|
60
72
|
#endif
|
61
73
|
|
62
|
-
if (bytes_written
|
63
|
-
gumbo_string_buffer_reserve(
|
64
|
-
parser, output->capacity + bytes_written, output);
|
74
|
+
if (bytes_written >= remaining_capacity) {
|
75
|
+
gumbo_string_buffer_reserve(output->capacity + bytes_written, output);
|
65
76
|
remaining_capacity = output->capacity - output->length;
|
66
77
|
va_start(args, format);
|
67
|
-
bytes_written = vsnprintf(
|
68
|
-
|
78
|
+
bytes_written = vsnprintf (
|
79
|
+
output->data + output->length,
|
80
|
+
remaining_capacity,
|
81
|
+
format,
|
82
|
+
args
|
83
|
+
);
|
69
84
|
va_end(args);
|
70
85
|
}
|
71
86
|
output->length += bytes_written;
|
72
87
|
return bytes_written;
|
73
88
|
}
|
74
89
|
|
75
|
-
static void print_tag_stack(
|
76
|
-
|
77
|
-
|
90
|
+
static void print_tag_stack (
|
91
|
+
const GumboParserError* error,
|
92
|
+
GumboStringBuffer* output
|
93
|
+
) {
|
94
|
+
print_message(output, " Currently open tags: ");
|
78
95
|
for (unsigned int i = 0; i < error->tag_stack.length; ++i) {
|
79
96
|
if (i) {
|
80
|
-
print_message(
|
97
|
+
print_message(output, ", ");
|
81
98
|
}
|
82
99
|
GumboTag tag = (GumboTag) error->tag_stack.data[i];
|
83
|
-
print_message(
|
100
|
+
print_message(output, "%s", gumbo_normalized_tagname(tag));
|
84
101
|
}
|
85
|
-
gumbo_string_buffer_append_codepoint(
|
102
|
+
gumbo_string_buffer_append_codepoint('.', output);
|
86
103
|
}
|
87
104
|
|
88
|
-
static void
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
105
|
+
static void handle_tokenizer_error (
|
106
|
+
const GumboError* error,
|
107
|
+
GumboStringBuffer* output
|
108
|
+
) {
|
109
|
+
switch (error->type) {
|
110
|
+
case GUMBO_ERR_ABRUPT_CLOSING_OF_EMPTY_COMMENT:
|
111
|
+
print_message(output, "Empty comment abruptly closed by '%s', use '-->'.",
|
112
|
+
error->v.tokenizer.state == GUMBO_LEX_COMMENT_START? ">" : "->");
|
113
|
+
break;
|
114
|
+
case GUMBO_ERR_ABRUPT_DOCTYPE_PUBLIC_IDENTIFIER:
|
115
|
+
print_message (
|
116
|
+
output,
|
117
|
+
"DOCTYPE public identifier missing closing %s.",
|
118
|
+
error->v.tokenizer.state == GUMBO_LEX_DOCTYPE_PUBLIC_ID_DOUBLE_QUOTED?
|
119
|
+
"quotation mark (\")" : "apostrophe (')"
|
120
|
+
);
|
121
|
+
break;
|
122
|
+
case GUMBO_ERR_ABRUPT_DOCTYPE_SYSTEM_IDENTIFIER:
|
123
|
+
print_message (
|
124
|
+
output,
|
125
|
+
"DOCTYPE system identifier missing closing %s.",
|
126
|
+
error->v.tokenizer.state == GUMBO_LEX_DOCTYPE_SYSTEM_ID_DOUBLE_QUOTED?
|
127
|
+
"quotation mark (\")" : "apostrophe (')"
|
128
|
+
);
|
129
|
+
break;
|
130
|
+
case GUMBO_ERR_ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE:
|
131
|
+
print_message (
|
132
|
+
output,
|
133
|
+
"Numeric character reference '%.*s' does not contain any %sdigits.",
|
134
|
+
(int)error->original_text.length, error->original_text.data,
|
135
|
+
error->v.tokenizer.state == GUMBO_LEX_HEXADECIMAL_CHARACTER_REFERENCE_START? "hexadecimal " : ""
|
136
|
+
);
|
137
|
+
break;
|
138
|
+
case GUMBO_ERR_CDATA_IN_HTML_CONTENT:
|
139
|
+
print_message(output, "CDATA section outside foreign (SVG or MathML) content.");
|
140
|
+
break;
|
141
|
+
case GUMBO_ERR_CHARACTER_REFERENCE_OUTSIDE_UNICODE_RANGE:
|
142
|
+
print_message (
|
143
|
+
output,
|
144
|
+
"Numeric character reference '%.*s' references a code point that is outside the valid Unicode range.",
|
145
|
+
(int)error->original_text.length, error->original_text.data
|
146
|
+
);
|
147
|
+
break;
|
148
|
+
case GUMBO_ERR_CONTROL_CHARACTER_IN_INPUT_STREAM:
|
149
|
+
print_message (
|
150
|
+
output,
|
151
|
+
"Input contains prohibited control code point U+%04X.",
|
152
|
+
error->v.tokenizer.codepoint
|
153
|
+
);
|
154
|
+
break;
|
155
|
+
case GUMBO_ERR_CONTROL_CHARACTER_REFERENCE:
|
156
|
+
print_message (
|
157
|
+
output,
|
158
|
+
"Numeric character reference '%.*s' references prohibited control code point U+%04X.",
|
159
|
+
(int)error->original_text.length, error->original_text.data,
|
160
|
+
error->v.tokenizer.codepoint
|
161
|
+
);
|
162
|
+
break;
|
163
|
+
case GUMBO_ERR_END_TAG_WITH_ATTRIBUTES:
|
164
|
+
print_message(output, "End tag contains attributes.");
|
165
|
+
break;
|
166
|
+
case GUMBO_ERR_DUPLICATE_ATTRIBUTE:
|
167
|
+
print_message(output, "Tag contains multiple attributes with the same name.");
|
168
|
+
break;
|
169
|
+
case GUMBO_ERR_END_TAG_WITH_TRAILING_SOLIDUS:
|
170
|
+
print_message(output, "End tag ends with '/>', use '>'.");
|
171
|
+
break;
|
172
|
+
case GUMBO_ERR_EOF_BEFORE_TAG_NAME:
|
173
|
+
print_message(output, "End of input where a tag name is expected.");
|
174
|
+
break;
|
175
|
+
case GUMBO_ERR_EOF_IN_CDATA:
|
176
|
+
print_message(output, "End of input in CDATA section.");
|
177
|
+
break;
|
178
|
+
case GUMBO_ERR_EOF_IN_COMMENT:
|
179
|
+
print_message(output, "End of input in comment.");
|
180
|
+
break;
|
181
|
+
case GUMBO_ERR_EOF_IN_DOCTYPE:
|
182
|
+
print_message(output, "End of input in DOCTYPE.");
|
183
|
+
break;
|
184
|
+
case GUMBO_ERR_EOF_IN_SCRIPT_HTML_COMMENT_LIKE_TEXT:
|
185
|
+
print_message(output, "End of input in text that resembles an HTML comment inside script element content.");
|
186
|
+
break;
|
187
|
+
case GUMBO_ERR_EOF_IN_TAG:
|
188
|
+
print_message(output, "End of input in tag.");
|
189
|
+
break;
|
190
|
+
case GUMBO_ERR_INCORRECTLY_CLOSED_COMMENT:
|
191
|
+
print_message(output, "Comment closed incorrectly by '--!>', use '-->'.");
|
192
|
+
break;
|
193
|
+
case GUMBO_ERR_INCORRECTLY_OPENED_COMMENT:
|
194
|
+
print_message(output, "Comment, DOCTYPE, or CDATA opened incorrectly, use '<!--', '<!DOCTYPE', or '<![CDATA['.");
|
195
|
+
break;
|
196
|
+
case GUMBO_ERR_INVALID_CHARACTER_SEQUENCE_AFTER_DOCTYPE_NAME:
|
197
|
+
print_message(output, "Invalid character sequence after DOCTYPE name, expected 'PUBLIC', 'SYSTEM', or '>'.");
|
198
|
+
break;
|
199
|
+
case GUMBO_ERR_INVALID_FIRST_CHARACTER_OF_TAG_NAME:
|
200
|
+
if (gumbo_ascii_isascii(error->v.tokenizer.codepoint)
|
201
|
+
&& !gumbo_ascii_iscntrl(error->v.tokenizer.codepoint))
|
202
|
+
print_message(output, "Invalid first character of tag name '%c'.", error->v.tokenizer.codepoint);
|
203
|
+
else
|
204
|
+
print_message(output, "Invalid first code point of tag name U+%04X.", error->v.tokenizer.codepoint);
|
205
|
+
break;
|
206
|
+
case GUMBO_ERR_MISSING_ATTRIBUTE_VALUE:
|
207
|
+
print_message(output, "Missing attribute value.");
|
208
|
+
break;
|
209
|
+
case GUMBO_ERR_MISSING_DOCTYPE_NAME:
|
210
|
+
print_message(output, "Missing DOCTYPE name.");
|
211
|
+
break;
|
212
|
+
case GUMBO_ERR_MISSING_DOCTYPE_PUBLIC_IDENTIFIER:
|
213
|
+
print_message(output, "Missing DOCTYPE public identifier.");
|
214
|
+
break;
|
215
|
+
case GUMBO_ERR_MISSING_DOCTYPE_SYSTEM_IDENTIFIER:
|
216
|
+
print_message(output, "Missing DOCTYPE system identifier.");
|
217
|
+
break;
|
218
|
+
case GUMBO_ERR_MISSING_END_TAG_NAME:
|
219
|
+
print_message(output, "Missing end tag name.");
|
220
|
+
break;
|
221
|
+
case GUMBO_ERR_MISSING_QUOTE_BEFORE_DOCTYPE_PUBLIC_IDENTIFIER:
|
222
|
+
print_message(output, "Missing quote before DOCTYPE public identifier.");
|
223
|
+
break;
|
224
|
+
case GUMBO_ERR_MISSING_QUOTE_BEFORE_DOCTYPE_SYSTEM_IDENTIFIER:
|
225
|
+
print_message(output, "Missing quote before DOCTYPE system identifier.");
|
226
|
+
break;
|
227
|
+
case GUMBO_ERR_MISSING_SEMICOLON_AFTER_CHARACTER_REFERENCE:
|
228
|
+
print_message(output, "Missing semicolon after character reference '%.*s'.",
|
229
|
+
(int)error->original_text.length, error->original_text.data);
|
230
|
+
break;
|
231
|
+
case GUMBO_ERR_MISSING_WHITESPACE_AFTER_DOCTYPE_PUBLIC_KEYWORD:
|
232
|
+
print_message(output, "Missing whitespace after 'PUBLIC' keyword.");
|
233
|
+
break;
|
234
|
+
case GUMBO_ERR_MISSING_WHITESPACE_AFTER_DOCTYPE_SYSTEM_KEYWORD:
|
235
|
+
print_message(output, "Missing whitespace after 'SYSTEM' keyword.");
|
236
|
+
break;
|
237
|
+
case GUMBO_ERR_MISSING_WHITESPACE_BEFORE_DOCTYPE_NAME:
|
238
|
+
print_message(output, "Missing whitespace between 'DOCTYPE' keyword and DOCTYPE name.");
|
239
|
+
break;
|
240
|
+
case GUMBO_ERR_MISSING_WHITESPACE_BETWEEN_ATTRIBUTES:
|
241
|
+
print_message(output, "Missing whitespace between attributes.");
|
242
|
+
break;
|
243
|
+
case GUMBO_ERR_MISSING_WHITESPACE_BETWEEN_DOCTYPE_PUBLIC_AND_SYSTEM_IDENTIFIERS:
|
244
|
+
print_message(output, "Missing whitespace between DOCTYPE public and system identifiers.");
|
245
|
+
break;
|
246
|
+
case GUMBO_ERR_NESTED_COMMENT:
|
247
|
+
print_message(output, "Nested comment.");
|
248
|
+
break;
|
249
|
+
case GUMBO_ERR_NONCHARACTER_CHARACTER_REFERENCE:
|
250
|
+
print_message (
|
251
|
+
output,
|
252
|
+
"Numeric character reference '%.*s' references noncharacter U+%04X.",
|
253
|
+
(int)error->original_text.length, error->original_text.data,
|
254
|
+
error->v.tokenizer.codepoint
|
255
|
+
);
|
256
|
+
break;
|
257
|
+
case GUMBO_ERR_NONCHARACTER_IN_INPUT_STREAM:
|
258
|
+
print_message(output, "Input contains noncharacter U+%04X.", error->v.tokenizer.codepoint);
|
259
|
+
break;
|
260
|
+
case GUMBO_ERR_NON_VOID_HTML_ELEMENT_START_TAG_WITH_TRAILING_SOLIDUS:
|
261
|
+
print_message(output, "Start tag of nonvoid HTML element ends with '/>', use '>'.");
|
262
|
+
break;
|
263
|
+
case GUMBO_ERR_NULL_CHARACTER_REFERENCE:
|
264
|
+
print_message(output, "Numeric character reference '%.*s' references U+0000.",
|
265
|
+
(int)error->original_text.length, error->original_text.data);
|
266
|
+
break;
|
267
|
+
case GUMBO_ERR_SURROGATE_CHARACTER_REFERENCE:
|
268
|
+
print_message (
|
269
|
+
output,
|
270
|
+
"Numeric character reference '%.*s' references surrogate U+%4X.",
|
271
|
+
(int)error->original_text.length, error->original_text.data,
|
272
|
+
error->v.tokenizer.codepoint
|
273
|
+
);
|
274
|
+
break;
|
275
|
+
case GUMBO_ERR_SURROGATE_IN_INPUT_STREAM:
|
276
|
+
print_message(output, "Input contains surrogate U+%04X.", error->v.tokenizer.codepoint);
|
277
|
+
break;
|
278
|
+
case GUMBO_ERR_UNEXPECTED_CHARACTER_AFTER_DOCTYPE_SYSTEM_IDENTIFIER:
|
279
|
+
print_message(output, "Unexpected character after DOCTYPE system identifier.");
|
280
|
+
break;
|
281
|
+
case GUMBO_ERR_UNEXPECTED_CHARACTER_IN_ATTRIBUTE_NAME:
|
282
|
+
print_message(output, "Unexpected character (%c) in attribute name.", error->v.tokenizer.codepoint);
|
283
|
+
break;
|
284
|
+
case GUMBO_ERR_UNEXPECTED_CHARACTER_IN_UNQUOTED_ATTRIBUTE_VALUE:
|
285
|
+
print_message(output, "Unexpected character (%c) in unquoted attribute value.", error->v.tokenizer.codepoint);
|
286
|
+
break;
|
287
|
+
case GUMBO_ERR_UNEXPECTED_EQUALS_SIGN_BEFORE_ATTRIBUTE_NAME:
|
288
|
+
print_message(output, "Unexpected '=' before an attribute name.");
|
289
|
+
break;
|
290
|
+
case GUMBO_ERR_UNEXPECTED_NULL_CHARACTER:
|
291
|
+
print_message(output, "Input contains unexpected U+0000.");
|
292
|
+
break;
|
293
|
+
case GUMBO_ERR_UNEXPECTED_QUESTION_MARK_INSTEAD_OF_TAG_NAME:
|
294
|
+
print_message(output, "Unexpected '?' where start tag name is expected.");
|
295
|
+
break;
|
296
|
+
case GUMBO_ERR_UNEXPECTED_SOLIDUS_IN_TAG:
|
297
|
+
print_message(output, "Unexpected '/' in tag.");
|
298
|
+
break;
|
299
|
+
case GUMBO_ERR_UNKNOWN_NAMED_CHARACTER_REFERENCE:
|
300
|
+
print_message(output, "Unknown named character reference '%.*s'.",
|
301
|
+
(int)error->original_text.length, error->original_text.data);
|
302
|
+
break;
|
303
|
+
case GUMBO_ERR_UTF8_INVALID:
|
304
|
+
print_message(output, "Invalid UTF8 encoding.");
|
305
|
+
break;
|
306
|
+
case GUMBO_ERR_UTF8_TRUNCATED:
|
307
|
+
print_message(output, "UTF8 character truncated.");
|
308
|
+
break;
|
309
|
+
case GUMBO_ERR_PARSER:
|
310
|
+
assert(0 && "Unreachable.");
|
311
|
+
}
|
312
|
+
}
|
313
|
+
static void handle_parser_error (
|
314
|
+
const GumboParserError* error,
|
315
|
+
GumboStringBuffer* output
|
316
|
+
) {
|
317
|
+
if (
|
318
|
+
error->parser_state == GUMBO_INSERTION_MODE_INITIAL
|
319
|
+
&& error->input_type != GUMBO_TOKEN_DOCTYPE
|
320
|
+
) {
|
321
|
+
print_message (
|
322
|
+
output,
|
323
|
+
"Expected a doctype token"
|
324
|
+
);
|
94
325
|
return;
|
95
326
|
}
|
96
327
|
|
97
328
|
switch (error->input_type) {
|
98
329
|
case GUMBO_TOKEN_DOCTYPE:
|
99
|
-
print_message(
|
330
|
+
print_message(output, "This is not a legal doctype");
|
100
331
|
return;
|
101
332
|
case GUMBO_TOKEN_COMMENT:
|
102
333
|
// Should never happen; comments are always legal.
|
103
334
|
assert(0);
|
104
335
|
// But just in case...
|
105
|
-
print_message(
|
336
|
+
print_message(output, "Comments aren't legal here");
|
106
337
|
return;
|
107
338
|
case GUMBO_TOKEN_CDATA:
|
108
339
|
case GUMBO_TOKEN_WHITESPACE:
|
109
340
|
case GUMBO_TOKEN_CHARACTER:
|
110
|
-
print_message(
|
341
|
+
print_message(output, "Character tokens aren't legal here");
|
111
342
|
return;
|
112
343
|
case GUMBO_TOKEN_NULL:
|
113
|
-
print_message(
|
344
|
+
print_message(output, "Null bytes are not allowed in HTML5");
|
114
345
|
return;
|
115
346
|
case GUMBO_TOKEN_EOF:
|
116
347
|
if (error->parser_state == GUMBO_INSERTION_MODE_INITIAL) {
|
117
|
-
print_message(
|
348
|
+
print_message(output, "You must provide a doctype");
|
118
349
|
} else {
|
119
|
-
print_message(
|
120
|
-
print_tag_stack(
|
350
|
+
print_message(output, "Premature end of file");
|
351
|
+
print_tag_stack(error, output);
|
121
352
|
}
|
122
353
|
return;
|
123
354
|
case GUMBO_TOKEN_START_TAG:
|
124
355
|
case GUMBO_TOKEN_END_TAG:
|
125
|
-
print_message(
|
126
|
-
print_tag_stack(
|
356
|
+
print_message(output, "That tag isn't allowed here");
|
357
|
+
print_tag_stack(error, output);
|
127
358
|
// TODO(jdtang): Give more specific messaging.
|
128
359
|
return;
|
129
360
|
}
|
130
361
|
}
|
131
362
|
|
132
363
|
// Finds the preceding newline in an original source buffer from a given byte
|
133
|
-
// location.
|
364
|
+
// location. Returns a character pointer to the character after that, or a
|
134
365
|
// pointer to the beginning of the string if this is the first line.
|
135
|
-
static const char*
|
136
|
-
|
137
|
-
|
366
|
+
static const char* find_prev_newline (
|
367
|
+
const char* source_text,
|
368
|
+
size_t source_length,
|
369
|
+
const char* error_location
|
370
|
+
) {
|
371
|
+
const char* source_end = source_text + source_length;
|
372
|
+
assert(error_location >= source_text);
|
373
|
+
assert(error_location <= source_end);
|
138
374
|
const char* c = error_location;
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
return c ==
|
375
|
+
if (c != source_text && (error_location == source_end || *c == '\n'))
|
376
|
+
--c;
|
377
|
+
while (c != source_text && *c != '\n')
|
378
|
+
--c;
|
379
|
+
return c == source_text ? c : c + 1;
|
144
380
|
}
|
145
381
|
|
146
382
|
// Finds the next newline in the original source buffer from a given byte
|
147
|
-
// location.
|
148
|
-
//
|
383
|
+
// location. Returns a character pointer to that newline, or a pointer to
|
384
|
+
// source_text + source_length if this is the last line.
|
149
385
|
static const char* find_next_newline(
|
150
|
-
|
386
|
+
const char* source_text,
|
387
|
+
size_t source_length,
|
388
|
+
const char* error_location
|
389
|
+
) {
|
390
|
+
const char* source_end = source_text + source_length;
|
391
|
+
assert(error_location >= source_text);
|
392
|
+
assert(error_location <= source_end);
|
151
393
|
const char* c = error_location;
|
152
|
-
|
153
|
-
;
|
394
|
+
while (c != source_end && *c != '\n')
|
395
|
+
++c;
|
154
396
|
return c;
|
155
397
|
}
|
156
398
|
|
157
399
|
GumboError* gumbo_add_error(GumboParser* parser) {
|
400
|
+
parser->_output->document_error = true;
|
401
|
+
|
158
402
|
int max_errors = parser->_options->max_errors;
|
159
403
|
if (max_errors >= 0 && parser->_output->errors.length >= (unsigned int) max_errors) {
|
160
404
|
return NULL;
|
161
405
|
}
|
162
|
-
GumboError* error =
|
163
|
-
gumbo_vector_add(
|
406
|
+
GumboError* error = gumbo_alloc(sizeof(GumboError));
|
407
|
+
gumbo_vector_add(error, &parser->_output->errors);
|
164
408
|
return error;
|
165
409
|
}
|
166
410
|
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
411
|
+
GumboSourcePosition gumbo_error_position(const GumboError* error) {
|
412
|
+
return error->position;
|
413
|
+
}
|
414
|
+
|
415
|
+
const char* gumbo_error_code(const GumboError* error) {
|
171
416
|
switch (error->type) {
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
417
|
+
// Defined tokenizer errors.
|
418
|
+
case GUMBO_ERR_ABRUPT_CLOSING_OF_EMPTY_COMMENT:
|
419
|
+
return "abrupt-closing-of-empty-comment";
|
420
|
+
case GUMBO_ERR_ABRUPT_DOCTYPE_PUBLIC_IDENTIFIER:
|
421
|
+
return "abrupt-doctype-public-identifier";
|
422
|
+
case GUMBO_ERR_ABRUPT_DOCTYPE_SYSTEM_IDENTIFIER:
|
423
|
+
return "abrupt-doctype-system-identifier";
|
424
|
+
case GUMBO_ERR_ABSENCE_OF_DIGITS_IN_NUMERIC_CHARACTER_REFERENCE:
|
425
|
+
return "absence-of-digits-in-numeric-character-reference";
|
426
|
+
case GUMBO_ERR_CDATA_IN_HTML_CONTENT:
|
427
|
+
return "cdata-in-html-content";
|
428
|
+
case GUMBO_ERR_CHARACTER_REFERENCE_OUTSIDE_UNICODE_RANGE:
|
429
|
+
return "character-reference-outside-unicode-range";
|
430
|
+
case GUMBO_ERR_CONTROL_CHARACTER_IN_INPUT_STREAM:
|
431
|
+
return "control-character-in-input-stream";
|
432
|
+
case GUMBO_ERR_CONTROL_CHARACTER_REFERENCE:
|
433
|
+
return "control-character-reference";
|
434
|
+
case GUMBO_ERR_END_TAG_WITH_ATTRIBUTES:
|
435
|
+
return "end-tag-with-attributes";
|
436
|
+
case GUMBO_ERR_DUPLICATE_ATTRIBUTE:
|
437
|
+
return "duplicate-attribute";
|
438
|
+
case GUMBO_ERR_END_TAG_WITH_TRAILING_SOLIDUS:
|
439
|
+
return "end-tag-with-trailing-solidus";
|
440
|
+
case GUMBO_ERR_EOF_BEFORE_TAG_NAME:
|
441
|
+
return "eof-before-tag-name";
|
442
|
+
case GUMBO_ERR_EOF_IN_CDATA:
|
443
|
+
return "eof-in-cdata";
|
444
|
+
case GUMBO_ERR_EOF_IN_COMMENT:
|
445
|
+
return "eof-in-comment";
|
446
|
+
case GUMBO_ERR_EOF_IN_DOCTYPE:
|
447
|
+
return "eof-in-doctype";
|
448
|
+
case GUMBO_ERR_EOF_IN_SCRIPT_HTML_COMMENT_LIKE_TEXT:
|
449
|
+
return "eof-in-script-html-comment-like-text";
|
450
|
+
case GUMBO_ERR_EOF_IN_TAG:
|
451
|
+
return "eof-in-tag";
|
452
|
+
case GUMBO_ERR_INCORRECTLY_CLOSED_COMMENT:
|
453
|
+
return "incorrectly-closed-comment";
|
454
|
+
case GUMBO_ERR_INCORRECTLY_OPENED_COMMENT:
|
455
|
+
return "incorrectly-opened-comment";
|
456
|
+
case GUMBO_ERR_INVALID_CHARACTER_SEQUENCE_AFTER_DOCTYPE_NAME:
|
457
|
+
return "invalid-character-sequence-after-doctype-name";
|
458
|
+
case GUMBO_ERR_INVALID_FIRST_CHARACTER_OF_TAG_NAME:
|
459
|
+
return "invalid-first-character-of-tag-name";
|
460
|
+
case GUMBO_ERR_MISSING_ATTRIBUTE_VALUE:
|
461
|
+
return "missing-attribute-value";
|
462
|
+
case GUMBO_ERR_MISSING_DOCTYPE_NAME:
|
463
|
+
return "missing-doctype-name";
|
464
|
+
case GUMBO_ERR_MISSING_DOCTYPE_PUBLIC_IDENTIFIER:
|
465
|
+
return "missing-doctype-public-identifier";
|
466
|
+
case GUMBO_ERR_MISSING_DOCTYPE_SYSTEM_IDENTIFIER:
|
467
|
+
return "missing-doctype-system-identifier";
|
468
|
+
case GUMBO_ERR_MISSING_END_TAG_NAME:
|
469
|
+
return "missing-end-tag-name";
|
470
|
+
case GUMBO_ERR_MISSING_QUOTE_BEFORE_DOCTYPE_PUBLIC_IDENTIFIER:
|
471
|
+
return "missing-quote-before-doctype-public-identifier";
|
472
|
+
case GUMBO_ERR_MISSING_QUOTE_BEFORE_DOCTYPE_SYSTEM_IDENTIFIER:
|
473
|
+
return "missing-quote-before-doctype-system-identifier";
|
474
|
+
case GUMBO_ERR_MISSING_SEMICOLON_AFTER_CHARACTER_REFERENCE:
|
475
|
+
return "missing-semicolon-after-character-reference";
|
476
|
+
case GUMBO_ERR_MISSING_WHITESPACE_AFTER_DOCTYPE_PUBLIC_KEYWORD:
|
477
|
+
return "missing-whitespace-after-doctype-public-keyword";
|
478
|
+
case GUMBO_ERR_MISSING_WHITESPACE_AFTER_DOCTYPE_SYSTEM_KEYWORD:
|
479
|
+
return "missing-whitespace-after-doctype-system-keyword";
|
480
|
+
case GUMBO_ERR_MISSING_WHITESPACE_BEFORE_DOCTYPE_NAME:
|
481
|
+
return "missing-whitespace-before-doctype-name";
|
482
|
+
case GUMBO_ERR_MISSING_WHITESPACE_BETWEEN_ATTRIBUTES:
|
483
|
+
return "missing-whitespace-between-attributes";
|
484
|
+
case GUMBO_ERR_MISSING_WHITESPACE_BETWEEN_DOCTYPE_PUBLIC_AND_SYSTEM_IDENTIFIERS:
|
485
|
+
return "missing-whitespace-between-doctype-public-and-system-identifiers";
|
486
|
+
case GUMBO_ERR_NESTED_COMMENT:
|
487
|
+
return "nested-comment";
|
488
|
+
case GUMBO_ERR_NONCHARACTER_CHARACTER_REFERENCE:
|
489
|
+
return "noncharacter-character-reference";
|
490
|
+
case GUMBO_ERR_NONCHARACTER_IN_INPUT_STREAM:
|
491
|
+
return "noncharacter-in-input-stream";
|
492
|
+
case GUMBO_ERR_NON_VOID_HTML_ELEMENT_START_TAG_WITH_TRAILING_SOLIDUS:
|
493
|
+
return "non-void-html-element-start-tag-with-trailing-solidus";
|
494
|
+
case GUMBO_ERR_NULL_CHARACTER_REFERENCE:
|
495
|
+
return "null-character-reference";
|
496
|
+
case GUMBO_ERR_SURROGATE_CHARACTER_REFERENCE:
|
497
|
+
return "surrogate-character-reference";
|
498
|
+
case GUMBO_ERR_SURROGATE_IN_INPUT_STREAM:
|
499
|
+
return "surrogate-in-input-stream";
|
500
|
+
case GUMBO_ERR_UNEXPECTED_CHARACTER_AFTER_DOCTYPE_SYSTEM_IDENTIFIER:
|
501
|
+
return "unexpected-character-after-doctype-system-identifier";
|
502
|
+
case GUMBO_ERR_UNEXPECTED_CHARACTER_IN_ATTRIBUTE_NAME:
|
503
|
+
return "unexpected-character-in-attribute-name";
|
504
|
+
case GUMBO_ERR_UNEXPECTED_CHARACTER_IN_UNQUOTED_ATTRIBUTE_VALUE:
|
505
|
+
return "unexpected-character-in-unquoted-attribute-value";
|
506
|
+
case GUMBO_ERR_UNEXPECTED_EQUALS_SIGN_BEFORE_ATTRIBUTE_NAME:
|
507
|
+
return "unexpected-equals-sign-before-attribute-name";
|
508
|
+
case GUMBO_ERR_UNEXPECTED_NULL_CHARACTER:
|
509
|
+
return "unexpected-null-character";
|
510
|
+
case GUMBO_ERR_UNEXPECTED_QUESTION_MARK_INSTEAD_OF_TAG_NAME:
|
511
|
+
return "unexpected-question-mark-instead-of-tag-name";
|
512
|
+
case GUMBO_ERR_UNEXPECTED_SOLIDUS_IN_TAG:
|
513
|
+
return "unexpected-solidus-in-tag";
|
514
|
+
case GUMBO_ERR_UNKNOWN_NAMED_CHARACTER_REFERENCE:
|
515
|
+
return "unknown-named-character-reference";
|
516
|
+
|
517
|
+
// Encoding errors.
|
518
|
+
case GUMBO_ERR_UTF8_INVALID:
|
519
|
+
return "utf8-invalid";
|
520
|
+
case GUMBO_ERR_UTF8_TRUNCATED:
|
521
|
+
return "utf8-truncated";
|
522
|
+
|
523
|
+
// Generic parser error.
|
524
|
+
case GUMBO_ERR_PARSER:
|
525
|
+
return "generic-parser";
|
224
526
|
}
|
225
|
-
|
527
|
+
// Silence warning about control reaching end of non-void function.
|
528
|
+
// All errors _should_ be handled in the switch statement.
|
529
|
+
return "generic-parser";
|
226
530
|
}
|
227
531
|
|
228
|
-
void
|
229
|
-
|
230
|
-
|
231
|
-
|
532
|
+
static void error_to_string (
|
533
|
+
const GumboError* error,
|
534
|
+
GumboStringBuffer* output
|
535
|
+
) {
|
536
|
+
if (error->type < GUMBO_ERR_PARSER)
|
537
|
+
handle_tokenizer_error(error, output);
|
538
|
+
else
|
539
|
+
handle_parser_error(&error->v.parser, output);
|
540
|
+
}
|
541
|
+
|
542
|
+
size_t gumbo_error_to_string(const GumboError* error, char** output) {
|
543
|
+
GumboStringBuffer sb;
|
544
|
+
gumbo_string_buffer_init(&sb);
|
545
|
+
error_to_string(error, &sb);
|
546
|
+
*output = sb.data;
|
547
|
+
return sb.length;
|
548
|
+
}
|
232
549
|
|
233
|
-
|
234
|
-
const
|
550
|
+
void caret_diagnostic_to_string (
|
551
|
+
const GumboError* error,
|
552
|
+
const char* source_text,
|
553
|
+
size_t source_length,
|
554
|
+
GumboStringBuffer* output
|
555
|
+
) {
|
556
|
+
error_to_string(error, output);
|
557
|
+
|
558
|
+
const char* error_text = error->original_text.data;
|
559
|
+
const char* line_start = find_prev_newline(source_text, source_length, error_text);
|
560
|
+
const char* line_end = find_next_newline(source_text, source_length, error_text);
|
235
561
|
GumboStringPiece original_line;
|
236
562
|
original_line.data = line_start;
|
237
563
|
original_line.length = line_end - line_start;
|
238
564
|
|
239
|
-
gumbo_string_buffer_append_codepoint(
|
240
|
-
gumbo_string_buffer_append_string(
|
241
|
-
gumbo_string_buffer_append_codepoint(
|
242
|
-
gumbo_string_buffer_reserve(
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
gumbo_string_buffer_append_codepoint(
|
565
|
+
gumbo_string_buffer_append_codepoint('\n', output);
|
566
|
+
gumbo_string_buffer_append_string(&original_line, output);
|
567
|
+
gumbo_string_buffer_append_codepoint('\n', output);
|
568
|
+
gumbo_string_buffer_reserve(output->length + error->position.column, output);
|
569
|
+
if (error->position.column >= 2) {
|
570
|
+
size_t num_spaces = error->position.column - 1;
|
571
|
+
memset(output->data + output->length, ' ', num_spaces);
|
572
|
+
output->length += num_spaces;
|
573
|
+
}
|
574
|
+
gumbo_string_buffer_append_codepoint('^', output);
|
575
|
+
gumbo_string_buffer_append_codepoint('\n', output);
|
249
576
|
}
|
250
577
|
|
251
|
-
|
252
|
-
|
578
|
+
size_t gumbo_caret_diagnostic_to_string (
|
579
|
+
const GumboError* error,
|
580
|
+
const char* source_text,
|
581
|
+
size_t source_length,
|
582
|
+
char **output
|
583
|
+
) {
|
584
|
+
GumboStringBuffer sb;
|
585
|
+
gumbo_string_buffer_init(&sb);
|
586
|
+
caret_diagnostic_to_string(error, source_text, source_length, &sb);
|
587
|
+
*output = sb.data;
|
588
|
+
return sb.length;
|
589
|
+
}
|
590
|
+
|
591
|
+
void gumbo_print_caret_diagnostic (
|
592
|
+
const GumboError* error,
|
593
|
+
const char* source_text,
|
594
|
+
size_t source_length
|
595
|
+
) {
|
253
596
|
GumboStringBuffer text;
|
254
|
-
gumbo_string_buffer_init(
|
255
|
-
|
597
|
+
gumbo_string_buffer_init(&text);
|
598
|
+
print_message (
|
599
|
+
&text,
|
600
|
+
"%lu:%lu: ",
|
601
|
+
(unsigned long)error->position.line,
|
602
|
+
(unsigned long)error->position.column
|
603
|
+
);
|
604
|
+
|
605
|
+
caret_diagnostic_to_string(error, source_text, source_length, &text);
|
256
606
|
printf("%.*s", (int) text.length, text.data);
|
257
|
-
gumbo_string_buffer_destroy(
|
607
|
+
gumbo_string_buffer_destroy(&text);
|
258
608
|
}
|
259
609
|
|
260
|
-
void gumbo_error_destroy(
|
261
|
-
if (error->type == GUMBO_ERR_PARSER
|
262
|
-
|
263
|
-
gumbo_vector_destroy(parser, &error->v.parser.tag_stack);
|
264
|
-
} else if (error->type == GUMBO_ERR_DUPLICATE_ATTR) {
|
265
|
-
gumbo_parser_deallocate(parser, (void*) error->v.duplicate_attr.name);
|
610
|
+
void gumbo_error_destroy(GumboError* error) {
|
611
|
+
if (error->type == GUMBO_ERR_PARSER) {
|
612
|
+
gumbo_vector_destroy(&error->v.parser.tag_stack);
|
266
613
|
}
|
267
|
-
|
614
|
+
gumbo_free(error);
|
268
615
|
}
|
269
616
|
|
270
617
|
void gumbo_init_errors(GumboParser* parser) {
|
271
|
-
gumbo_vector_init(
|
618
|
+
gumbo_vector_init(5, &parser->_output->errors);
|
272
619
|
}
|
273
620
|
|
274
621
|
void gumbo_destroy_errors(GumboParser* parser) {
|
275
622
|
for (unsigned int i = 0; i < parser->_output->errors.length; ++i) {
|
276
|
-
gumbo_error_destroy(parser
|
623
|
+
gumbo_error_destroy(parser->_output->errors.data[i]);
|
277
624
|
}
|
278
|
-
gumbo_vector_destroy(
|
625
|
+
gumbo_vector_destroy(&parser->_output->errors);
|
279
626
|
}
|