1*0f5c86ddSBaptiste Daroussin #include <yaml.h>
2*0f5c86ddSBaptiste Daroussin
3*0f5c86ddSBaptiste Daroussin #include <stdlib.h>
4*0f5c86ddSBaptiste Daroussin #include <stdio.h>
5*0f5c86ddSBaptiste Daroussin #include <string.h>
6*0f5c86ddSBaptiste Daroussin
7*0f5c86ddSBaptiste Daroussin #ifdef NDEBUG
8*0f5c86ddSBaptiste Daroussin #undef NDEBUG
9*0f5c86ddSBaptiste Daroussin #endif
10*0f5c86ddSBaptiste Daroussin #include <assert.h>
11*0f5c86ddSBaptiste Daroussin
12*0f5c86ddSBaptiste Daroussin #define BUFFER_SIZE 65536
13*0f5c86ddSBaptiste Daroussin #define MAX_DOCUMENTS 16
14*0f5c86ddSBaptiste Daroussin
copy_document(yaml_document_t * document_to,yaml_document_t * document_from)15*0f5c86ddSBaptiste Daroussin int copy_document(yaml_document_t *document_to, yaml_document_t *document_from)
16*0f5c86ddSBaptiste Daroussin {
17*0f5c86ddSBaptiste Daroussin yaml_node_t *node;
18*0f5c86ddSBaptiste Daroussin yaml_node_item_t *item;
19*0f5c86ddSBaptiste Daroussin yaml_node_pair_t *pair;
20*0f5c86ddSBaptiste Daroussin
21*0f5c86ddSBaptiste Daroussin if (!yaml_document_initialize(document_to, document_from->version_directive,
22*0f5c86ddSBaptiste Daroussin document_from->tag_directives.start,
23*0f5c86ddSBaptiste Daroussin document_from->tag_directives.end,
24*0f5c86ddSBaptiste Daroussin document_from->start_implicit, document_from->end_implicit))
25*0f5c86ddSBaptiste Daroussin return 0;
26*0f5c86ddSBaptiste Daroussin
27*0f5c86ddSBaptiste Daroussin for (node = document_from->nodes.start;
28*0f5c86ddSBaptiste Daroussin node < document_from->nodes.top; node ++) {
29*0f5c86ddSBaptiste Daroussin switch (node->type) {
30*0f5c86ddSBaptiste Daroussin case YAML_SCALAR_NODE:
31*0f5c86ddSBaptiste Daroussin if (!yaml_document_add_scalar(document_to, node->tag,
32*0f5c86ddSBaptiste Daroussin node->data.scalar.value, node->data.scalar.length,
33*0f5c86ddSBaptiste Daroussin node->data.scalar.style)) goto error;
34*0f5c86ddSBaptiste Daroussin break;
35*0f5c86ddSBaptiste Daroussin case YAML_SEQUENCE_NODE:
36*0f5c86ddSBaptiste Daroussin if (!yaml_document_add_sequence(document_to, node->tag,
37*0f5c86ddSBaptiste Daroussin node->data.sequence.style)) goto error;
38*0f5c86ddSBaptiste Daroussin break;
39*0f5c86ddSBaptiste Daroussin case YAML_MAPPING_NODE:
40*0f5c86ddSBaptiste Daroussin if (!yaml_document_add_mapping(document_to, node->tag,
41*0f5c86ddSBaptiste Daroussin node->data.mapping.style)) goto error;
42*0f5c86ddSBaptiste Daroussin break;
43*0f5c86ddSBaptiste Daroussin default:
44*0f5c86ddSBaptiste Daroussin assert(0);
45*0f5c86ddSBaptiste Daroussin break;
46*0f5c86ddSBaptiste Daroussin }
47*0f5c86ddSBaptiste Daroussin }
48*0f5c86ddSBaptiste Daroussin
49*0f5c86ddSBaptiste Daroussin for (node = document_from->nodes.start;
50*0f5c86ddSBaptiste Daroussin node < document_from->nodes.top; node ++) {
51*0f5c86ddSBaptiste Daroussin switch (node->type) {
52*0f5c86ddSBaptiste Daroussin case YAML_SEQUENCE_NODE:
53*0f5c86ddSBaptiste Daroussin for (item = node->data.sequence.items.start;
54*0f5c86ddSBaptiste Daroussin item < node->data.sequence.items.top; item ++) {
55*0f5c86ddSBaptiste Daroussin if (!yaml_document_append_sequence_item(document_to,
56*0f5c86ddSBaptiste Daroussin node - document_from->nodes.start + 1,
57*0f5c86ddSBaptiste Daroussin *item)) goto error;
58*0f5c86ddSBaptiste Daroussin }
59*0f5c86ddSBaptiste Daroussin break;
60*0f5c86ddSBaptiste Daroussin case YAML_MAPPING_NODE:
61*0f5c86ddSBaptiste Daroussin for (pair = node->data.mapping.pairs.start;
62*0f5c86ddSBaptiste Daroussin pair < node->data.mapping.pairs.top; pair ++) {
63*0f5c86ddSBaptiste Daroussin if (!yaml_document_append_mapping_pair(document_to,
64*0f5c86ddSBaptiste Daroussin node - document_from->nodes.start + 1,
65*0f5c86ddSBaptiste Daroussin pair->key, pair->value)) goto error;
66*0f5c86ddSBaptiste Daroussin }
67*0f5c86ddSBaptiste Daroussin break;
68*0f5c86ddSBaptiste Daroussin default:
69*0f5c86ddSBaptiste Daroussin break;
70*0f5c86ddSBaptiste Daroussin }
71*0f5c86ddSBaptiste Daroussin }
72*0f5c86ddSBaptiste Daroussin return 1;
73*0f5c86ddSBaptiste Daroussin
74*0f5c86ddSBaptiste Daroussin error:
75*0f5c86ddSBaptiste Daroussin yaml_document_delete(document_to);
76*0f5c86ddSBaptiste Daroussin return 0;
77*0f5c86ddSBaptiste Daroussin }
78*0f5c86ddSBaptiste Daroussin
compare_nodes(yaml_document_t * document1,int index1,yaml_document_t * document2,int index2,int level)79*0f5c86ddSBaptiste Daroussin int compare_nodes(yaml_document_t *document1, int index1,
80*0f5c86ddSBaptiste Daroussin yaml_document_t *document2, int index2, int level)
81*0f5c86ddSBaptiste Daroussin {
82*0f5c86ddSBaptiste Daroussin int k;
83*0f5c86ddSBaptiste Daroussin yaml_node_t *node1;
84*0f5c86ddSBaptiste Daroussin yaml_node_t *node2;
85*0f5c86ddSBaptiste Daroussin if (level++ > 1000) return 0;
86*0f5c86ddSBaptiste Daroussin node1 = yaml_document_get_node(document1, index1);
87*0f5c86ddSBaptiste Daroussin node2 = yaml_document_get_node(document2, index2);
88*0f5c86ddSBaptiste Daroussin
89*0f5c86ddSBaptiste Daroussin assert(node1);
90*0f5c86ddSBaptiste Daroussin assert(node2);
91*0f5c86ddSBaptiste Daroussin
92*0f5c86ddSBaptiste Daroussin if (node1->type != node2->type)
93*0f5c86ddSBaptiste Daroussin return 0;
94*0f5c86ddSBaptiste Daroussin
95*0f5c86ddSBaptiste Daroussin if (strcmp((char *)node1->tag, (char *)node2->tag) != 0) return 0;
96*0f5c86ddSBaptiste Daroussin
97*0f5c86ddSBaptiste Daroussin switch (node1->type) {
98*0f5c86ddSBaptiste Daroussin case YAML_SCALAR_NODE:
99*0f5c86ddSBaptiste Daroussin if (node1->data.scalar.length != node2->data.scalar.length)
100*0f5c86ddSBaptiste Daroussin return 0;
101*0f5c86ddSBaptiste Daroussin if (strncmp((char *)node1->data.scalar.value, (char *)node2->data.scalar.value,
102*0f5c86ddSBaptiste Daroussin node1->data.scalar.length) != 0) return 0;
103*0f5c86ddSBaptiste Daroussin break;
104*0f5c86ddSBaptiste Daroussin case YAML_SEQUENCE_NODE:
105*0f5c86ddSBaptiste Daroussin if ((node1->data.sequence.items.top - node1->data.sequence.items.start) !=
106*0f5c86ddSBaptiste Daroussin (node2->data.sequence.items.top - node2->data.sequence.items.start))
107*0f5c86ddSBaptiste Daroussin return 0;
108*0f5c86ddSBaptiste Daroussin for (k = 0; k < (node1->data.sequence.items.top - node1->data.sequence.items.start); k ++) {
109*0f5c86ddSBaptiste Daroussin if (!compare_nodes(document1, node1->data.sequence.items.start[k],
110*0f5c86ddSBaptiste Daroussin document2, node2->data.sequence.items.start[k], level)) return 0;
111*0f5c86ddSBaptiste Daroussin }
112*0f5c86ddSBaptiste Daroussin break;
113*0f5c86ddSBaptiste Daroussin case YAML_MAPPING_NODE:
114*0f5c86ddSBaptiste Daroussin if ((node1->data.mapping.pairs.top - node1->data.mapping.pairs.start) !=
115*0f5c86ddSBaptiste Daroussin (node2->data.mapping.pairs.top - node2->data.mapping.pairs.start))
116*0f5c86ddSBaptiste Daroussin return 0;
117*0f5c86ddSBaptiste Daroussin for (k = 0; k < (node1->data.mapping.pairs.top - node1->data.mapping.pairs.start); k ++) {
118*0f5c86ddSBaptiste Daroussin if (!compare_nodes(document1, node1->data.mapping.pairs.start[k].key,
119*0f5c86ddSBaptiste Daroussin document2, node2->data.mapping.pairs.start[k].key, level)) return 0;
120*0f5c86ddSBaptiste Daroussin if (!compare_nodes(document1, node1->data.mapping.pairs.start[k].value,
121*0f5c86ddSBaptiste Daroussin document2, node2->data.mapping.pairs.start[k].value, level)) return 0;
122*0f5c86ddSBaptiste Daroussin }
123*0f5c86ddSBaptiste Daroussin break;
124*0f5c86ddSBaptiste Daroussin default:
125*0f5c86ddSBaptiste Daroussin assert(0);
126*0f5c86ddSBaptiste Daroussin break;
127*0f5c86ddSBaptiste Daroussin }
128*0f5c86ddSBaptiste Daroussin return 1;
129*0f5c86ddSBaptiste Daroussin }
130*0f5c86ddSBaptiste Daroussin
compare_documents(yaml_document_t * document1,yaml_document_t * document2)131*0f5c86ddSBaptiste Daroussin int compare_documents(yaml_document_t *document1, yaml_document_t *document2)
132*0f5c86ddSBaptiste Daroussin {
133*0f5c86ddSBaptiste Daroussin int k;
134*0f5c86ddSBaptiste Daroussin
135*0f5c86ddSBaptiste Daroussin if ((document1->version_directive && !document2->version_directive)
136*0f5c86ddSBaptiste Daroussin || (!document1->version_directive && document2->version_directive)
137*0f5c86ddSBaptiste Daroussin || (document1->version_directive && document2->version_directive
138*0f5c86ddSBaptiste Daroussin && (document1->version_directive->major != document2->version_directive->major
139*0f5c86ddSBaptiste Daroussin || document1->version_directive->minor != document2->version_directive->minor)))
140*0f5c86ddSBaptiste Daroussin return 0;
141*0f5c86ddSBaptiste Daroussin
142*0f5c86ddSBaptiste Daroussin if ((document1->tag_directives.end - document1->tag_directives.start) !=
143*0f5c86ddSBaptiste Daroussin (document2->tag_directives.end - document2->tag_directives.start))
144*0f5c86ddSBaptiste Daroussin return 0;
145*0f5c86ddSBaptiste Daroussin for (k = 0; k < (document1->tag_directives.end - document1->tag_directives.start); k ++) {
146*0f5c86ddSBaptiste Daroussin if ((strcmp((char *)document1->tag_directives.start[k].handle,
147*0f5c86ddSBaptiste Daroussin (char *)document2->tag_directives.start[k].handle) != 0)
148*0f5c86ddSBaptiste Daroussin || (strcmp((char *)document1->tag_directives.start[k].prefix,
149*0f5c86ddSBaptiste Daroussin (char *)document2->tag_directives.start[k].prefix) != 0))
150*0f5c86ddSBaptiste Daroussin return 0;
151*0f5c86ddSBaptiste Daroussin }
152*0f5c86ddSBaptiste Daroussin
153*0f5c86ddSBaptiste Daroussin if ((document1->nodes.top - document1->nodes.start) !=
154*0f5c86ddSBaptiste Daroussin (document2->nodes.top - document2->nodes.start))
155*0f5c86ddSBaptiste Daroussin return 0;
156*0f5c86ddSBaptiste Daroussin
157*0f5c86ddSBaptiste Daroussin if (document1->nodes.top != document1->nodes.start) {
158*0f5c86ddSBaptiste Daroussin if (!compare_nodes(document1, 1, document2, 1, 0))
159*0f5c86ddSBaptiste Daroussin return 0;
160*0f5c86ddSBaptiste Daroussin }
161*0f5c86ddSBaptiste Daroussin
162*0f5c86ddSBaptiste Daroussin return 1;
163*0f5c86ddSBaptiste Daroussin }
164*0f5c86ddSBaptiste Daroussin
print_output(char * name,unsigned char * buffer,size_t size,int count)165*0f5c86ddSBaptiste Daroussin int print_output(char *name, unsigned char *buffer, size_t size, int count)
166*0f5c86ddSBaptiste Daroussin {
167*0f5c86ddSBaptiste Daroussin FILE *file;
168*0f5c86ddSBaptiste Daroussin char data[BUFFER_SIZE];
169*0f5c86ddSBaptiste Daroussin size_t data_size = 1;
170*0f5c86ddSBaptiste Daroussin size_t total_size = 0;
171*0f5c86ddSBaptiste Daroussin if (count >= 0) {
172*0f5c86ddSBaptiste Daroussin printf("FAILED (at the document #%d)\nSOURCE:\n", count+1);
173*0f5c86ddSBaptiste Daroussin }
174*0f5c86ddSBaptiste Daroussin file = fopen(name, "rb");
175*0f5c86ddSBaptiste Daroussin assert(file);
176*0f5c86ddSBaptiste Daroussin while (data_size > 0) {
177*0f5c86ddSBaptiste Daroussin data_size = fread(data, 1, BUFFER_SIZE, file);
178*0f5c86ddSBaptiste Daroussin assert(!ferror(file));
179*0f5c86ddSBaptiste Daroussin if (!data_size) break;
180*0f5c86ddSBaptiste Daroussin assert(fwrite(data, 1, data_size, stdout) == data_size);
181*0f5c86ddSBaptiste Daroussin total_size += data_size;
182*0f5c86ddSBaptiste Daroussin if (feof(file)) break;
183*0f5c86ddSBaptiste Daroussin }
184*0f5c86ddSBaptiste Daroussin fclose(file);
185*0f5c86ddSBaptiste Daroussin printf("#### (length: %ld)\n", (long)total_size);
186*0f5c86ddSBaptiste Daroussin printf("OUTPUT:\n%s#### (length: %ld)\n", buffer, (long)size);
187*0f5c86ddSBaptiste Daroussin return 0;
188*0f5c86ddSBaptiste Daroussin }
189*0f5c86ddSBaptiste Daroussin
190*0f5c86ddSBaptiste Daroussin int
main(int argc,char * argv[])191*0f5c86ddSBaptiste Daroussin main(int argc, char *argv[])
192*0f5c86ddSBaptiste Daroussin {
193*0f5c86ddSBaptiste Daroussin int number;
194*0f5c86ddSBaptiste Daroussin int canonical = 0;
195*0f5c86ddSBaptiste Daroussin int unicode = 0;
196*0f5c86ddSBaptiste Daroussin
197*0f5c86ddSBaptiste Daroussin number = 1;
198*0f5c86ddSBaptiste Daroussin while (number < argc) {
199*0f5c86ddSBaptiste Daroussin if (strcmp(argv[number], "-c") == 0) {
200*0f5c86ddSBaptiste Daroussin canonical = 1;
201*0f5c86ddSBaptiste Daroussin }
202*0f5c86ddSBaptiste Daroussin else if (strcmp(argv[number], "-u") == 0) {
203*0f5c86ddSBaptiste Daroussin unicode = 1;
204*0f5c86ddSBaptiste Daroussin }
205*0f5c86ddSBaptiste Daroussin else if (argv[number][0] == '-') {
206*0f5c86ddSBaptiste Daroussin printf("Unknown option: '%s'\n", argv[number]);
207*0f5c86ddSBaptiste Daroussin return 0;
208*0f5c86ddSBaptiste Daroussin }
209*0f5c86ddSBaptiste Daroussin if (argv[number][0] == '-') {
210*0f5c86ddSBaptiste Daroussin if (number < argc-1) {
211*0f5c86ddSBaptiste Daroussin memmove(argv+number, argv+number+1, (argc-number-1)*sizeof(char *));
212*0f5c86ddSBaptiste Daroussin }
213*0f5c86ddSBaptiste Daroussin argc --;
214*0f5c86ddSBaptiste Daroussin }
215*0f5c86ddSBaptiste Daroussin else {
216*0f5c86ddSBaptiste Daroussin number ++;
217*0f5c86ddSBaptiste Daroussin }
218*0f5c86ddSBaptiste Daroussin }
219*0f5c86ddSBaptiste Daroussin
220*0f5c86ddSBaptiste Daroussin if (argc < 2) {
221*0f5c86ddSBaptiste Daroussin printf("Usage: %s [-c] [-u] file1.yaml ...\n", argv[0]);
222*0f5c86ddSBaptiste Daroussin return 0;
223*0f5c86ddSBaptiste Daroussin }
224*0f5c86ddSBaptiste Daroussin
225*0f5c86ddSBaptiste Daroussin for (number = 1; number < argc; number ++)
226*0f5c86ddSBaptiste Daroussin {
227*0f5c86ddSBaptiste Daroussin FILE *file;
228*0f5c86ddSBaptiste Daroussin yaml_parser_t parser;
229*0f5c86ddSBaptiste Daroussin yaml_emitter_t emitter;
230*0f5c86ddSBaptiste Daroussin
231*0f5c86ddSBaptiste Daroussin yaml_document_t document;
232*0f5c86ddSBaptiste Daroussin unsigned char buffer[BUFFER_SIZE+1];
233*0f5c86ddSBaptiste Daroussin size_t written = 0;
234*0f5c86ddSBaptiste Daroussin yaml_document_t documents[MAX_DOCUMENTS];
235*0f5c86ddSBaptiste Daroussin size_t document_number = 0;
236*0f5c86ddSBaptiste Daroussin int done = 0;
237*0f5c86ddSBaptiste Daroussin int count = 0;
238*0f5c86ddSBaptiste Daroussin int error = 0;
239*0f5c86ddSBaptiste Daroussin int k;
240*0f5c86ddSBaptiste Daroussin memset(buffer, 0, BUFFER_SIZE+1);
241*0f5c86ddSBaptiste Daroussin memset(documents, 0, MAX_DOCUMENTS*sizeof(yaml_document_t));
242*0f5c86ddSBaptiste Daroussin
243*0f5c86ddSBaptiste Daroussin printf("[%d] Loading, dumping, and loading again '%s': ", number, argv[number]);
244*0f5c86ddSBaptiste Daroussin fflush(stdout);
245*0f5c86ddSBaptiste Daroussin
246*0f5c86ddSBaptiste Daroussin file = fopen(argv[number], "rb");
247*0f5c86ddSBaptiste Daroussin assert(file);
248*0f5c86ddSBaptiste Daroussin
249*0f5c86ddSBaptiste Daroussin assert(yaml_parser_initialize(&parser));
250*0f5c86ddSBaptiste Daroussin yaml_parser_set_input_file(&parser, file);
251*0f5c86ddSBaptiste Daroussin assert(yaml_emitter_initialize(&emitter));
252*0f5c86ddSBaptiste Daroussin if (canonical) {
253*0f5c86ddSBaptiste Daroussin yaml_emitter_set_canonical(&emitter, 1);
254*0f5c86ddSBaptiste Daroussin }
255*0f5c86ddSBaptiste Daroussin if (unicode) {
256*0f5c86ddSBaptiste Daroussin yaml_emitter_set_unicode(&emitter, 1);
257*0f5c86ddSBaptiste Daroussin }
258*0f5c86ddSBaptiste Daroussin yaml_emitter_set_output_string(&emitter, buffer, BUFFER_SIZE, &written);
259*0f5c86ddSBaptiste Daroussin yaml_emitter_open(&emitter);
260*0f5c86ddSBaptiste Daroussin
261*0f5c86ddSBaptiste Daroussin while (!done)
262*0f5c86ddSBaptiste Daroussin {
263*0f5c86ddSBaptiste Daroussin if (!yaml_parser_load(&parser, &document)) {
264*0f5c86ddSBaptiste Daroussin error = 1;
265*0f5c86ddSBaptiste Daroussin break;
266*0f5c86ddSBaptiste Daroussin }
267*0f5c86ddSBaptiste Daroussin
268*0f5c86ddSBaptiste Daroussin done = (!yaml_document_get_root_node(&document));
269*0f5c86ddSBaptiste Daroussin if (!done) {
270*0f5c86ddSBaptiste Daroussin assert(document_number < MAX_DOCUMENTS);
271*0f5c86ddSBaptiste Daroussin assert(copy_document(&(documents[document_number++]), &document));
272*0f5c86ddSBaptiste Daroussin assert(yaml_emitter_dump(&emitter, &document) ||
273*0f5c86ddSBaptiste Daroussin (yaml_emitter_flush(&emitter) && print_output(argv[number], buffer, written, count)));
274*0f5c86ddSBaptiste Daroussin count ++;
275*0f5c86ddSBaptiste Daroussin }
276*0f5c86ddSBaptiste Daroussin else {
277*0f5c86ddSBaptiste Daroussin yaml_document_delete(&document);
278*0f5c86ddSBaptiste Daroussin }
279*0f5c86ddSBaptiste Daroussin }
280*0f5c86ddSBaptiste Daroussin
281*0f5c86ddSBaptiste Daroussin yaml_parser_delete(&parser);
282*0f5c86ddSBaptiste Daroussin assert(!fclose(file));
283*0f5c86ddSBaptiste Daroussin yaml_emitter_close(&emitter);
284*0f5c86ddSBaptiste Daroussin yaml_emitter_delete(&emitter);
285*0f5c86ddSBaptiste Daroussin
286*0f5c86ddSBaptiste Daroussin if (!error)
287*0f5c86ddSBaptiste Daroussin {
288*0f5c86ddSBaptiste Daroussin count = done = 0;
289*0f5c86ddSBaptiste Daroussin assert(yaml_parser_initialize(&parser));
290*0f5c86ddSBaptiste Daroussin yaml_parser_set_input_string(&parser, buffer, written);
291*0f5c86ddSBaptiste Daroussin
292*0f5c86ddSBaptiste Daroussin while (!done)
293*0f5c86ddSBaptiste Daroussin {
294*0f5c86ddSBaptiste Daroussin assert(yaml_parser_load(&parser, &document) || print_output(argv[number], buffer, written, count));
295*0f5c86ddSBaptiste Daroussin done = (!yaml_document_get_root_node(&document));
296*0f5c86ddSBaptiste Daroussin if (!done) {
297*0f5c86ddSBaptiste Daroussin assert(compare_documents(documents+count, &document) || print_output(argv[number], buffer, written, count));
298*0f5c86ddSBaptiste Daroussin count ++;
299*0f5c86ddSBaptiste Daroussin }
300*0f5c86ddSBaptiste Daroussin yaml_document_delete(&document);
301*0f5c86ddSBaptiste Daroussin }
302*0f5c86ddSBaptiste Daroussin yaml_parser_delete(&parser);
303*0f5c86ddSBaptiste Daroussin }
304*0f5c86ddSBaptiste Daroussin
305*0f5c86ddSBaptiste Daroussin for (k = 0; k < document_number; k ++) {
306*0f5c86ddSBaptiste Daroussin yaml_document_delete(documents+k);
307*0f5c86ddSBaptiste Daroussin }
308*0f5c86ddSBaptiste Daroussin
309*0f5c86ddSBaptiste Daroussin printf("PASSED (length: %ld)\n", (long)written);
310*0f5c86ddSBaptiste Daroussin print_output(argv[number], buffer, written, -1);
311*0f5c86ddSBaptiste Daroussin }
312*0f5c86ddSBaptiste Daroussin
313*0f5c86ddSBaptiste Daroussin return 0;
314*0f5c86ddSBaptiste Daroussin }
315