Coverage Report

Created: 2026-06-13 06:28

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/xmlProtoConverter.cpp
Line
Count
Source
1
/*
2
 * Copyright (C) 2019 Google Inc.
3
 *
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
 * http://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
 */
16
17
#include "xmlProtoConverter.h"
18
19
#include <algorithm>
20
21
using namespace std;
22
using namespace xmlProtoFuzzer;
23
24
string ProtoConverter::removeNonAscii(string const& _utf8)
25
1.04M
{
26
1.04M
  string asciiStr{_utf8};
27
23.6M
  asciiStr.erase(remove_if(asciiStr.begin(), asciiStr.end(), [=](char c) -> bool {
28
23.6M
                        return !(std::isalpha(c) || std::isdigit(c));
29
23.6M
                }), asciiStr.end());
30
1.04M
  return asciiStr.empty() ? "fuzz" : asciiStr;
31
1.04M
}
32
33
34
void ProtoConverter::visit(Misc const& _x)
35
4.44k
{
36
4.44k
  switch (_x.misc_oneof_case())
37
4.44k
  {
38
1.95k
  case Misc::kComment:
39
1.95k
    m_output << "<!--" << _x.comment() << "-->\n";
40
1.95k
    break;
41
1.43k
  case Misc::kInst:
42
1.43k
    visit(_x.inst());
43
1.43k
    break;
44
1.05k
  case Misc::MISC_ONEOF_NOT_SET:
45
1.05k
    break;
46
4.44k
  }
47
4.44k
}
48
49
void ProtoConverter::visit(Prolog const& _x)
50
7.76k
{
51
7.76k
  visit(_x.decl());
52
7.76k
  visit(_x.doctype());
53
7.76k
  for (auto const& misc: _x.misc())
54
2.71k
    visit(misc);
55
7.76k
}
56
57
void ProtoConverter::visit(KeyValue const& _x)
58
41.2k
{
59
41.2k
  if (!KeyValue::XmlNamespace_IsValid(_x.type()))
60
355
    return;
61
62
40.9k
  switch (_x.type())
63
40.9k
  {
64
8.64k
  case KeyValue::ATTRIBUTES:
65
8.64k
    m_output << "xml:attributes=\"" << removeNonAscii(_x.value()) << "\" ";
66
8.64k
    break;
67
2.67k
  case KeyValue::BASE:
68
2.67k
    m_output << "xml:base=\"" << removeNonAscii(_x.value()) << "\" ";
69
2.67k
    break;
70
5.92k
  case KeyValue::CATALOG:
71
5.92k
    m_output << "xml:catalog=\"" << removeNonAscii(_x.value()) << "\" ";
72
5.92k
    break;
73
3.28k
  case KeyValue::ID:
74
3.28k
    m_output << "xml:id=\"" << removeNonAscii(_x.value()) << "\" ";
75
3.28k
    break;
76
1.33k
  case KeyValue::LANG:
77
1.33k
    m_output << "xml:lang=\"" << removeNonAscii(_x.value()) << "\" ";
78
1.33k
    break;
79
1.73k
  case KeyValue::LINK:
80
1.73k
    m_output << "xml:link=\"" << removeNonAscii(_x.value()) << "\" ";
81
1.73k
    break;
82
1.21k
  case KeyValue::SPACE:
83
1.21k
    m_output << "xml:space=\"" << removeNonAscii(_x.value()) << "\" ";
84
1.21k
    break;
85
1.03k
  case KeyValue::SPECIAL:
86
1.03k
    m_output << "xml:special=\"" << removeNonAscii(_x.value()) << "\" ";
87
1.03k
    break;
88
3.99k
  case KeyValue::TEST:
89
3.99k
    m_output << "xml:test=\"" << removeNonAscii(_x.value()) << "\" ";
90
3.99k
    break;
91
11.0k
  case KeyValue::FUZZ:
92
11.0k
    if (_x.ByteSizeLong() % 2)
93
2.08k
      m_output << "xmlns:" << removeNonAscii(_x.key()) << "=\"" << removeNonAscii(_x.value()) << "\" ";
94
8.96k
    else
95
8.96k
      m_output << removeNonAscii(_x.key()) << "=\"" << removeNonAscii(_x.value()) << "\" ";
96
11.0k
    break;
97
0
  case KeyValue_XmlNamespace_KeyValue_XmlNamespace_INT_MIN_SENTINEL_DO_NOT_USE_:
98
0
  case KeyValue_XmlNamespace_KeyValue_XmlNamespace_INT_MAX_SENTINEL_DO_NOT_USE_:
99
0
    break;
100
40.9k
  }
101
40.9k
}
102
103
void ProtoConverter::visit(ProcessingInstruction const& _x)
104
1.43k
{
105
1.43k
  m_output << "<?" << removeNonAscii(_x.name()) << " ";
106
1.43k
  for (auto const& prop: _x.kv())
107
1.14k
    visit(prop);
108
1.43k
  m_output << "?>\n";
109
1.43k
}
110
111
void ProtoConverter::visit(Content const& _x)
112
206k
{
113
206k
  switch (_x.content_oneof_case())
114
206k
  {
115
3.31k
  case Content::kStr:
116
3.31k
    m_output << _x.str() << "\n";
117
3.31k
    break;
118
3.26k
  case Content::kE:
119
3.26k
    visit(_x.e());
120
3.26k
    m_output << "\n";
121
3.26k
    break;
122
4.15k
  case Content::kC:
123
4.15k
    visit(_x.c());
124
4.15k
    m_output << "\n";
125
4.15k
    break;
126
195k
  case Content::CONTENT_ONEOF_NOT_SET:
127
195k
    break;
128
206k
  }
129
206k
}
130
131
void ProtoConverter::visit(ElementDecl const& _x)
132
5.48k
{
133
5.48k
  if (!ElementDecl::ContentSpec_IsValid(_x.spec()))
134
290
    return;
135
136
5.19k
  m_output << "<!ELEMENT " << _x.name() << " ";
137
5.19k
  switch (_x.spec())
138
5.19k
  {
139
2.31k
  case ElementDecl::EMPTY:
140
2.31k
    m_output << "EMPTY>";
141
2.31k
    break;
142
380
  case ElementDecl::ANY:
143
380
    m_output << "ANY>";
144
380
    break;
145
277
  case ElementDecl::FUZZ:
146
277
    m_output << "FUZZ>";
147
277
    break;
148
1.18k
  case ElementDecl::MIXED:
149
1.18k
    m_output << "(#PCDATA";
150
1.18k
    for (auto const& pcdata: _x.cdata())
151
3.05k
      m_output << "|" << pcdata;
152
1.18k
    m_output << ")";
153
1.18k
    if (_x.cdata_size() > 0)
154
913
      m_output << "*";
155
1.18k
    m_output << ">";
156
1.18k
    break;
157
1.03k
  case ElementDecl::CHILDREN:
158
1.03k
  {
159
1.03k
    m_output << "(";
160
1.03k
    string delim = "";
161
1.60k
    for (auto const& str: _x.cdata()) {
162
1.60k
      m_output << delim << removeNonAscii(str);
163
1.60k
      delim = ", ";
164
1.60k
    }
165
1.03k
    m_output << ")>";
166
1.03k
    break;
167
0
  }
168
0
  case ElementDecl_ContentSpec_ElementDecl_ContentSpec_INT_MIN_SENTINEL_DO_NOT_USE_:
169
0
  case ElementDecl_ContentSpec_ElementDecl_ContentSpec_INT_MAX_SENTINEL_DO_NOT_USE_:
170
0
    break;
171
5.19k
  }
172
5.19k
}
173
174
void ProtoConverter::visit(AttValue const& _x)
175
4.56k
{
176
4.56k
  if (!isValid(_x))
177
98
    return;
178
179
4.46k
  m_output << "\"";
180
4.46k
  string prefix;
181
4.46k
  switch (_x.type())
182
4.46k
  {
183
2.65k
  case AttValue::ENTITY:
184
2.65k
    prefix = "&";
185
2.65k
    break;
186
1.27k
  case AttValue::CHAR:
187
1.27k
    if (_x.ByteSizeLong() % 2)
188
134
      prefix = "&#";
189
1.14k
    else
190
      // TODO: Value that follows this must be a
191
      // sequence of hex digits.
192
1.14k
      prefix = "&#x";
193
1.27k
    break;
194
541
  case AttValue::FUZZ:
195
541
    prefix = "fuzz";
196
541
    break;
197
0
  case AttValue_Type_AttValue_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
198
0
  case AttValue_Type_AttValue_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
199
0
    break;
200
4.46k
  }
201
4.46k
  for (auto const& name: _x.value())
202
7.38k
    m_output << prefix << removeNonAscii(name) << ";";
203
4.46k
  m_output << "\"";
204
4.46k
}
205
206
void ProtoConverter::visit(DefaultDecl const& _x)
207
21.6k
{
208
21.6k
  if (!isValid(_x))
209
215
    return;
210
211
21.4k
  switch (_x.type())
212
21.4k
  {
213
15.9k
  case DefaultDecl::REQUIRED:
214
15.9k
    m_output << "#REQUIRED";
215
15.9k
    break;
216
485
  case DefaultDecl::IMPLIED:
217
485
    m_output << "#IMPLIED";
218
485
    break;
219
4.56k
  case DefaultDecl::FIXED:
220
4.56k
    m_output << "#FIXED ";
221
4.56k
    visit(_x.att());
222
4.56k
    break;
223
384
  case DefaultDecl::FUZZ:
224
384
    m_output << "#FUZZ";
225
384
    break;
226
0
  case DefaultDecl_Type_DefaultDecl_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
227
0
  case DefaultDecl_Type_DefaultDecl_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
228
0
    break;
229
21.4k
  }
230
21.4k
}
231
232
void ProtoConverter::visit(AttDef const& _x)
233
21.8k
{
234
21.8k
  if (!isValid(_x))
235
197
    return;
236
237
21.6k
  m_output << " " << removeNonAscii(_x.name()) << " ";
238
21.6k
  switch (_x.type())
239
21.6k
  {
240
12.9k
  case AttDef::CDATA:
241
12.9k
    m_output << "CDATA ";
242
12.9k
    break;
243
1.95k
  case AttDef::ID:
244
1.95k
    m_output << "ID ";
245
1.95k
    break;
246
3.05k
  case AttDef::IDREF:
247
3.05k
    m_output << "IDREF ";
248
3.05k
    break;
249
940
  case AttDef::IDREFS:
250
940
    m_output << "IDREFS ";
251
940
    break;
252
967
  case AttDef::ENTITY:
253
967
    m_output << "ENTITY ";
254
967
    break;
255
480
  case AttDef::ENTITIES:
256
480
    m_output << "ENTITIES ";
257
480
    break;
258
572
  case AttDef::NMTOKEN:
259
572
    m_output << "NMTOKEN ";
260
572
    break;
261
601
  case AttDef::NMTOKENS:
262
601
    m_output << "NMTOKENS ";
263
601
    break;
264
100
  case AttDef::FUZZ:
265
100
    m_output << "FUZZ ";
266
100
    break;
267
0
  case AttDef_Type_AttDef_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
268
0
  case AttDef_Type_AttDef_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
269
0
    break;
270
21.6k
  }
271
21.6k
  visit(_x.def());
272
21.6k
}
273
274
void ProtoConverter::visit(AttListDecl const& _x)
275
4.57k
{
276
4.57k
  m_output << "<!ATTLIST " << removeNonAscii(_x.name());
277
4.57k
  for (auto const& att: _x.attdefs())
278
21.8k
    visit(att);
279
4.57k
  m_output << ">";
280
4.57k
}
281
282
void ProtoConverter::visit(NotationDecl const& _x)
283
4.77k
{
284
4.77k
  m_output << "<!NOTATION " << removeNonAscii(_x.name()) << " ";
285
4.77k
  switch (_x.notation_oneof_case())
286
4.77k
  {
287
1.52k
  case NotationDecl::kExt:
288
1.52k
    visit(_x.ext());
289
1.52k
    break;
290
2.16k
  case NotationDecl::kPub:
291
2.16k
    m_output << "PUBLIC " << _x.pub();
292
2.16k
    break;
293
279
  case NotationDecl::kFuzz:
294
279
    m_output << "FUZZ " << _x.fuzz();
295
279
    break;
296
811
  case NotationDecl::NOTATION_ONEOF_NOT_SET:
297
811
    break;
298
4.77k
  }
299
4.77k
  m_output << ">";
300
4.77k
}
301
302
void ProtoConverter::visit(NDataDecl const& _x)
303
3.05k
{
304
3.05k
  m_output << " NDATA " << _x.name();
305
3.05k
}
306
307
void ProtoConverter::visit(EntityDef const& _x)
308
8.37k
{
309
8.37k
  switch (_x.entity_oneof_case())
310
8.37k
  {
311
4.05k
  case EntityDef::kExt:
312
4.05k
    visit(_x.ext());
313
4.05k
    if (_x.ByteSizeLong() % 2)
314
3.05k
      visit(_x.ndata());
315
4.05k
    break;
316
2.06k
  case EntityDef::kVal:
317
2.06k
    visit(_x.val());
318
2.06k
    break;
319
2.25k
  case EntityDef::ENTITY_ONEOF_NOT_SET:
320
2.25k
    break;
321
8.37k
  }
322
8.37k
}
323
324
void ProtoConverter::visit(PEDef const& _x)
325
4.25k
{
326
4.25k
  switch (_x.pedef_oneof_case())
327
4.25k
  {
328
1.87k
  case PEDef::kVal:
329
1.87k
    visit(_x.val());
330
1.87k
    break;
331
1.16k
  case PEDef::kId:
332
1.16k
    visit(_x.id());
333
1.16k
    break;
334
1.21k
  case PEDef::PEDEF_ONEOF_NOT_SET:
335
1.21k
    break;
336
4.25k
  }
337
4.25k
}
338
339
void ProtoConverter::visit(EntityValue const& _x)
340
3.94k
{
341
3.94k
  if (!isValid(_x))
342
185
    return;
343
344
3.75k
  m_output << "\"";
345
3.75k
  string prefix;
346
3.75k
  switch (_x.type())
347
3.75k
  {
348
1.89k
  case EntityValue::ENTITY:
349
1.89k
    prefix = "&";
350
1.89k
    break;
351
1.42k
  case EntityValue::CHAR:
352
1.42k
    if (_x.ByteSizeLong() % 2)
353
723
      prefix = "&#";
354
700
    else
355
700
      prefix = "&#x";
356
1.42k
    break;
357
82
  case EntityValue::PEREF:
358
82
    prefix = "%";
359
82
    break;
360
360
  case EntityValue::FUZZ:
361
360
    prefix = "fuzz";
362
360
    break;
363
0
  case EntityValue_Type_EntityValue_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
364
0
  case EntityValue_Type_EntityValue_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
365
0
    break;
366
3.75k
  }
367
3.75k
  for (auto const& ref: _x.name())
368
6.81k
    m_output << prefix << ref << ";";
369
3.75k
  m_output << "\"";
370
3.75k
}
371
372
void ProtoConverter::visit(EntityDecl const& _x)
373
12.7k
{
374
12.7k
  if (!isValid(_x))
375
81
    return;
376
377
12.6k
  m_output << "<!ENTITY ";
378
12.6k
  switch (_x.type())
379
12.6k
  {
380
8.37k
  case EntityDecl::GEDECL:
381
8.37k
    m_output << _x.name() << " ";
382
8.37k
    visit(_x.ent());
383
8.37k
    break;
384
4.25k
  case EntityDecl::PEDECL:
385
4.25k
    m_output << "% " << _x.name() << " ";
386
4.25k
    visit(_x.pedef());
387
4.25k
    break;
388
0
  case EntityDecl_Type_EntityDecl_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
389
0
  case EntityDecl_Type_EntityDecl_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
390
0
    break;
391
12.6k
  }
392
12.6k
  m_output << ">";
393
12.6k
}
394
395
void ProtoConverter::visit(ConditionalSect const& _x)
396
8.82k
{
397
8.82k
  if (!isValid(_x))
398
70
    return;
399
400
8.75k
  switch (_x.type())
401
8.75k
  {
402
6.01k
  case ConditionalSect::INCLUDE:
403
6.01k
    m_output << "<![ INCLUDE [";
404
6.01k
    visit(_x.ext());
405
6.01k
    m_output << "]]>";
406
6.01k
    break;
407
861
  case ConditionalSect::IGNORE:
408
861
    m_output << "<![ IGNORE [";
409
861
    for (auto const& str: _x.ignores())
410
1.05k
      m_output << "<![" << removeNonAscii(str) << "]]>";
411
861
    m_output << "]]>";
412
861
    break;
413
1.87k
  case ConditionalSect::FUZZ:
414
1.87k
    m_output << "<![ FUZZ [";
415
1.87k
    visit(_x.ext());
416
1.87k
    m_output << "]]>";
417
1.87k
    break;
418
0
  case ConditionalSect_Type_ConditionalSect_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
419
0
  case ConditionalSect_Type_ConditionalSect_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
420
0
    break;
421
8.75k
  }
422
8.75k
}
423
424
425
void ProtoConverter::visit(OneExtSubsetDecl const& _x)
426
39.3k
{
427
39.3k
  switch (_x.extsubset_oneof_case())
428
39.3k
  {
429
21.0k
  case OneExtSubsetDecl::kM:
430
21.0k
    visit(_x.m());
431
21.0k
    break;
432
8.82k
  case OneExtSubsetDecl::kC:
433
8.82k
    visit(_x.c());
434
8.82k
    break;
435
9.45k
  case OneExtSubsetDecl::EXTSUBSET_ONEOF_NOT_SET:
436
9.45k
    break;
437
39.3k
  }
438
39.3k
}
439
440
441
void ProtoConverter::visit(ExtSubsetDecl const& _x)
442
12.8k
{
443
12.8k
  for (auto const& decl: _x.decls())
444
39.3k
    visit(decl);
445
12.8k
}
446
447
void ProtoConverter::visit(CData const& _x)
448
4.15k
{
449
4.15k
  m_output << "<![CDATA[" << removeNonAscii(_x.data()) << "]]>";
450
4.15k
}
451
452
void ProtoConverter::visit(MarkupDecl const& _x)
453
38.9k
{
454
38.9k
  switch (_x.markup_oneof_case())
455
38.9k
  {
456
5.48k
  case MarkupDecl::kE:
457
5.48k
    visit(_x.e());
458
5.48k
    break;
459
4.57k
  case MarkupDecl::kA:
460
4.57k
    visit(_x.a());
461
4.57k
    break;
462
4.77k
  case MarkupDecl::kN:
463
4.77k
    visit(_x.n());
464
4.77k
    break;
465
1.72k
  case MarkupDecl::kM:
466
1.72k
    visit(_x.m());
467
1.72k
    break;
468
12.7k
  case MarkupDecl::kEntity:
469
12.7k
    visit(_x.entity());
470
12.7k
    break;
471
4.96k
  case MarkupDecl::kExt:
472
4.96k
    visit(_x.ext());
473
4.96k
    break;
474
4.70k
  case MarkupDecl::MARKUP_ONEOF_NOT_SET:
475
4.70k
    break;
476
38.9k
  }
477
38.9k
}
478
479
/// Returns predefined element from an Element_Id enum
480
/// @param _x is an enum that holds the desired type of predefined value
481
/// @param _prop is a string that holds the value of the desired type
482
/// @return string holding the predefined value of the form
483
/// name attribute=\"value\"
484
string ProtoConverter::getPredefined(Element_Id _x, string const& _prop)
485
183k
{
486
183k
  string output{};
487
183k
  switch (_x)
488
183k
  {
489
164k
  case Element::XIINCLUDE:
490
167k
  case Element::XIFALLBACK:
491
172k
  case Element::XIHREF:
492
172k
    output = "xi:include href=\"fuzz.xml\"";
493
175k
  case Element::XIPARSE:
494
175k
    output = "xi:include parse=\"xml\"";
495
175k
  case Element::XIXPOINTER:
496
175k
    output = "xi:include xpointer=\"" + removeNonAscii(_prop) + "\"";
497
178k
  case Element::XIENCODING:
498
178k
    output = "xi:include encoding=\"" + removeNonAscii(_prop) + "\"";
499
181k
  case Element::XIACCEPT:
500
181k
    output = "xi:include accept=\"" + removeNonAscii(_prop) + "\"";
501
182k
  case Element::XIACCEPTLANG:
502
182k
    output = "xi:include accept-language=\"" + removeNonAscii(_prop) + "\"";
503
182k
  case Element_Id_Element_Id_INT_MIN_SENTINEL_DO_NOT_USE_:
504
182k
  case Element_Id_Element_Id_INT_MAX_SENTINEL_DO_NOT_USE_:
505
182k
    output = "xi:fuzz xifuzz=\"fuzz\"";
506
183k
  }
507
183k
  return output;
508
183k
}
509
510
/// Returns uri string for a given Element_Id type
511
string ProtoConverter::getUri(Element_Id _x)
512
183k
{
513
183k
  if (!Element::Id_IsValid(_x))
514
340
    return s_XInclude;
515
516
182k
  switch (_x)
517
182k
  {
518
164k
  case Element::XIINCLUDE:
519
167k
  case Element::XIFALLBACK:
520
172k
  case Element::XIHREF:
521
175k
  case Element::XIPARSE:
522
175k
  case Element::XIXPOINTER:
523
178k
  case Element::XIENCODING:
524
181k
  case Element::XIACCEPT:
525
182k
  case Element::XIACCEPTLANG:
526
182k
  case Element_Id_Element_Id_INT_MIN_SENTINEL_DO_NOT_USE_:
527
182k
  case Element_Id_Element_Id_INT_MAX_SENTINEL_DO_NOT_USE_:
528
182k
    return s_XInclude;
529
182k
  }
530
182k
}
531
532
void ProtoConverter::visit(Element const& _x)
533
206k
{
534
206k
  if (!isValid(_x))
535
251
    return;
536
537
  // Predefined child node
538
206k
  string child = {};
539
  // Predefined uri for child node
540
206k
  string pUri = {};
541
  // Element name
542
206k
  string name = removeNonAscii(_x.name());
543
544
206k
  switch (_x.type())
545
206k
  {
546
183k
  case Element::PREDEFINED:
547
183k
    child = getPredefined(_x.id(), _x.childprop());
548
183k
    pUri = getUri(_x.id());
549
183k
    break;
550
23.5k
  case Element::FUZZ:
551
23.5k
  case Element_Type_Element_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
552
23.5k
  case Element_Type_Element_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
553
23.5k
    break;
554
206k
  }
555
556
  // <name k1=v1 k2=v2 k3=v3>
557
  // <content>
558
  // </name>
559
560
  // Start name tag: Must be Ascii?
561
206k
  m_output << "<" << name << " ";
562
563
  // Add uri to name tag
564
206k
  if (!pUri.empty())
565
183k
    m_output << pUri << " ";
566
206k
  for (auto const& prop: _x.kv())
567
40.1k
    visit(prop);
568
206k
  m_output << ">\n";
569
570
  // Add attribute
571
206k
  if (!child.empty())
572
182k
    m_output << "<" << child << "/>\n";
573
574
  // Add content
575
206k
  visit(_x.content());
576
577
  // Close name tag
578
206k
  m_output << "</" << name << ">\n";
579
206k
}
580
581
void ProtoConverter::visit(ExternalId const& _x)
582
14.4k
{
583
14.4k
  if (!isValid(_x))
584
976
    return;
585
586
13.5k
  switch (_x.type())
587
13.5k
  {
588
9.83k
  case ExternalId::SYSTEM:
589
9.83k
    m_output << "SYSTEM " << "\"" << removeNonAscii(_x.system()) << "\"";
590
9.83k
    break;
591
1.23k
  case ExternalId::PUBLIC:
592
1.23k
    m_output << "PUBLIC " << "\"" << removeNonAscii(_x.pub()) << "\""
593
1.23k
      << " " << "\"" << removeNonAscii(_x.system()) << "\"";
594
1.23k
    break;
595
2.44k
  case ExternalId::FUZZ:
596
2.44k
    m_output << "FUZZ " << "\"" << removeNonAscii(_x.pub()) << "\"";
597
2.44k
    break;
598
0
  case ExternalId_Type_ExternalId_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
599
0
  case ExternalId_Type_ExternalId_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
600
0
    break;
601
13.5k
  }
602
13.5k
}
603
604
void ProtoConverter::visit(DocTypeDecl const& _x)
605
7.76k
{
606
7.76k
  m_output << "<!DOCTYPE " << removeNonAscii(_x.name()) << " ";
607
7.76k
  visit(_x.ext());
608
7.76k
  m_output << "[";
609
7.76k
  for (auto const& m: _x.mdecl())
610
17.8k
    visit(m);
611
7.76k
  m_output << "]";
612
7.76k
  m_output << ">\n";
613
7.76k
}
614
615
void ProtoConverter::visit(VersionNum const& _x)
616
7.76k
{
617
7.76k
  if (!isValid(_x))
618
6
    return;
619
620
7.75k
  switch (_x.type())
621
7.75k
  {
622
7.60k
  case VersionNum::STANDARD:
623
7.60k
    m_output << "\"1.0\"";
624
7.60k
    break;
625
152
  case VersionNum::FUZZ:
626
152
  case VersionNum_Type_VersionNum_Type_INT_MIN_SENTINEL_DO_NOT_USE_:
627
152
  case VersionNum_Type_VersionNum_Type_INT_MAX_SENTINEL_DO_NOT_USE_:
628
152
    m_output << "\"" << _x.major() << "." << _x.minor() << "\"";
629
152
    break;
630
7.75k
  }
631
7.75k
}
632
633
void ProtoConverter::visit(Encodings const& _x)
634
7.76k
{
635
7.76k
  if (!Encodings::Enc_IsValid(_x.name()))
636
579
    return;
637
638
7.18k
  m_output << " encoding=\"";
639
7.18k
  switch (_x.name())
640
7.18k
  {
641
6.30k
  case Encodings::BIG5:
642
6.30k
    m_output << "BIG5";
643
6.30k
    break;
644
11
  case Encodings::EUCJP:
645
11
    m_output << "EUC-JP";
646
11
    break;
647
28
  case Encodings::EUCKR:
648
28
    m_output << "EUC-KR";
649
28
    break;
650
18
  case Encodings::GB18030:
651
18
    m_output << "GB18030";
652
18
    break;
653
5
  case Encodings::ISO2022JP:
654
5
    m_output << "ISO-2022-JP";
655
5
    break;
656
134
  case Encodings::ISO2022KR:
657
134
    m_output << "ISO-2022-KR";
658
134
    break;
659
89
  case Encodings::ISO88591:
660
89
    m_output << "ISO-8859-1";
661
89
    break;
662
4
  case Encodings::ISO88592:
663
4
    m_output << "ISO-8859-2";
664
4
    break;
665
7
  case Encodings::ISO88593:
666
7
    m_output << "ISO-8859-3";
667
7
    break;
668
14
  case Encodings::ISO88594:
669
14
    m_output << "ISO-8859-4";
670
14
    break;
671
10
  case Encodings::ISO88595:
672
10
    m_output << "ISO-8859-5";
673
10
    break;
674
6
  case Encodings::ISO88596:
675
6
    m_output << "ISO-8859-6";
676
6
    break;
677
11
  case Encodings::ISO88597:
678
11
    m_output << "ISO-8859-7";
679
11
    break;
680
2
  case Encodings::ISO88598:
681
2
    m_output << "ISO-8859-8";
682
2
    break;
683
12
  case Encodings::ISO88599:
684
12
    m_output << "ISO-8859-9";
685
12
    break;
686
107
  case Encodings::SHIFTJIS:
687
107
    m_output << "SHIFT_JIS";
688
107
    break;
689
14
  case Encodings::TIS620:
690
14
    m_output << "TIS-620";
691
14
    break;
692
80
  case Encodings::USASCII:
693
80
    m_output << "US-ASCII";
694
80
    break;
695
76
  case Encodings::UTF8:
696
76
    m_output << "UTF-8";
697
76
    break;
698
46
  case Encodings::UTF16:
699
46
    m_output << "UTF-16";
700
46
    break;
701
18
  case Encodings::UTF16BE:
702
18
    m_output << "UTF-16BE";
703
18
    break;
704
7
  case Encodings::UTF16LE:
705
7
    m_output << "UTF-16LE";
706
7
    break;
707
41
  case Encodings::WINDOWS31J:
708
41
    m_output << "WINDOWS-31J";
709
41
    break;
710
6
  case Encodings::WINDOWS1255:
711
6
    m_output << "WINDOWS-1255";
712
6
    break;
713
6
  case Encodings::WINDOWS1256:
714
6
    m_output << "WINDOWS-1256";
715
6
    break;
716
127
  case Encodings::FUZZ:
717
127
    m_output << removeNonAscii(_x.fuzz());
718
127
    break;
719
0
  case Encodings_Enc_Encodings_Enc_INT_MIN_SENTINEL_DO_NOT_USE_:
720
0
  case Encodings_Enc_Encodings_Enc_INT_MAX_SENTINEL_DO_NOT_USE_:
721
0
    break;
722
7.18k
  }
723
7.18k
  m_output << "\"";
724
7.18k
}
725
726
void ProtoConverter::visit(XmlDeclaration const& _x)
727
7.76k
{
728
7.76k
  m_output << R"(<?xml version=)";
729
7.76k
  visit(_x.ver());
730
7.76k
  visit(_x.enc());
731
7.76k
  switch (_x.standalone())
732
7.76k
  {
733
7.36k
  case XmlDeclaration::YES:
734
7.36k
    m_output << " standalone=\'yes\'";
735
7.36k
    break;
736
338
  case XmlDeclaration::NO:
737
338
    m_output << " standalone=\'no\'";
738
338
    break;
739
0
  case XmlDeclaration_Standalone_XmlDeclaration_Standalone_INT_MIN_SENTINEL_DO_NOT_USE_:
740
0
  case XmlDeclaration_Standalone_XmlDeclaration_Standalone_INT_MAX_SENTINEL_DO_NOT_USE_:
741
62
  default:
742
62
    break;
743
7.76k
  }
744
7.76k
  m_output << "?>\n";
745
7.76k
}
746
747
void ProtoConverter::visit(XmlDocument const& _x)
748
7.76k
{
749
7.76k
  visit(_x.p());
750
7.76k
  for (auto const& element: _x.e())
751
203k
    visit(element);
752
7.76k
}
753
754
string ProtoConverter::protoToString(XmlDocument const& _x)
755
7.76k
{
756
7.76k
  visit(_x);
757
7.76k
  return m_output.str();
758
7.76k
}