7 // Flush the buffer if needed.
8 func flush(emitter *yaml_emitter_t) bool {
9 if emitter.buffer_pos+5 >= len(emitter.buffer) {
10 return yaml_emitter_flush(emitter)
15 // Put a character to the output buffer.
16 func put(emitter *yaml_emitter_t, value byte) bool {
17 if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) {
20 emitter.buffer[emitter.buffer_pos] = value
26 // Put a line break to the output buffer.
27 func put_break(emitter *yaml_emitter_t) bool {
28 if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) {
31 switch emitter.line_break {
33 emitter.buffer[emitter.buffer_pos] = '\r'
34 emitter.buffer_pos += 1
36 emitter.buffer[emitter.buffer_pos] = '\n'
37 emitter.buffer_pos += 1
39 emitter.buffer[emitter.buffer_pos+0] = '\r'
40 emitter.buffer[emitter.buffer_pos+1] = '\n'
41 emitter.buffer_pos += 2
43 panic("unknown line break setting")
50 // Copy a character from a string into buffer.
51 func write(emitter *yaml_emitter_t, s []byte, i *int) bool {
52 if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) {
55 p := emitter.buffer_pos
59 emitter.buffer[p+3] = s[*i+3]
62 emitter.buffer[p+2] = s[*i+2]
65 emitter.buffer[p+1] = s[*i+1]
68 emitter.buffer[p+0] = s[*i+0]
70 panic("unknown character width")
73 emitter.buffer_pos += w
78 // Write a whole string into buffer.
79 func write_all(emitter *yaml_emitter_t, s []byte) bool {
80 for i := 0; i < len(s); {
81 if !write(emitter, s, &i) {
88 // Copy a line break character from a string into buffer.
89 func write_break(emitter *yaml_emitter_t, s []byte, i *int) bool {
91 if !put_break(emitter) {
96 if !write(emitter, s, i) {
105 // Set an emitter error and return false.
106 func yaml_emitter_set_emitter_error(emitter *yaml_emitter_t, problem string) bool {
107 emitter.error = yaml_EMITTER_ERROR
108 emitter.problem = problem
113 func yaml_emitter_emit(emitter *yaml_emitter_t, event *yaml_event_t) bool {
114 emitter.events = append(emitter.events, *event)
115 for !yaml_emitter_need_more_events(emitter) {
116 event := &emitter.events[emitter.events_head]
117 if !yaml_emitter_analyze_event(emitter, event) {
120 if !yaml_emitter_state_machine(emitter, event) {
123 yaml_event_delete(event)
124 emitter.events_head++
129 // Check if we need to accumulate more events before emitting.
131 // We accumulate extra
132 // - 1 event for DOCUMENT-START
133 // - 2 events for SEQUENCE-START
134 // - 3 events for MAPPING-START
136 func yaml_emitter_need_more_events(emitter *yaml_emitter_t) bool {
137 if emitter.events_head == len(emitter.events) {
141 switch emitter.events[emitter.events_head].typ {
142 case yaml_DOCUMENT_START_EVENT:
145 case yaml_SEQUENCE_START_EVENT:
148 case yaml_MAPPING_START_EVENT:
154 if len(emitter.events)-emitter.events_head > accumulate {
158 for i := emitter.events_head; i < len(emitter.events); i++ {
159 switch emitter.events[i].typ {
160 case yaml_STREAM_START_EVENT, yaml_DOCUMENT_START_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT:
162 case yaml_STREAM_END_EVENT, yaml_DOCUMENT_END_EVENT, yaml_SEQUENCE_END_EVENT, yaml_MAPPING_END_EVENT:
172 // Append a directive to the directives stack.
173 func yaml_emitter_append_tag_directive(emitter *yaml_emitter_t, value *yaml_tag_directive_t, allow_duplicates bool) bool {
174 for i := 0; i < len(emitter.tag_directives); i++ {
175 if bytes.Equal(value.handle, emitter.tag_directives[i].handle) {
176 if allow_duplicates {
179 return yaml_emitter_set_emitter_error(emitter, "duplicate %TAG directive")
183 // [Go] Do we actually need to copy this given garbage collection
184 // and the lack of deallocating destructors?
185 tag_copy := yaml_tag_directive_t{
186 handle: make([]byte, len(value.handle)),
187 prefix: make([]byte, len(value.prefix)),
189 copy(tag_copy.handle, value.handle)
190 copy(tag_copy.prefix, value.prefix)
191 emitter.tag_directives = append(emitter.tag_directives, tag_copy)
195 // Increase the indentation level.
196 func yaml_emitter_increase_indent(emitter *yaml_emitter_t, flow, indentless bool) bool {
197 emitter.indents = append(emitter.indents, emitter.indent)
198 if emitter.indent < 0 {
200 emitter.indent = emitter.best_indent
204 } else if !indentless {
205 emitter.indent += emitter.best_indent
211 func yaml_emitter_state_machine(emitter *yaml_emitter_t, event *yaml_event_t) bool {
212 switch emitter.state {
214 case yaml_EMIT_STREAM_START_STATE:
215 return yaml_emitter_emit_stream_start(emitter, event)
217 case yaml_EMIT_FIRST_DOCUMENT_START_STATE:
218 return yaml_emitter_emit_document_start(emitter, event, true)
220 case yaml_EMIT_DOCUMENT_START_STATE:
221 return yaml_emitter_emit_document_start(emitter, event, false)
223 case yaml_EMIT_DOCUMENT_CONTENT_STATE:
224 return yaml_emitter_emit_document_content(emitter, event)
226 case yaml_EMIT_DOCUMENT_END_STATE:
227 return yaml_emitter_emit_document_end(emitter, event)
229 case yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE:
230 return yaml_emitter_emit_flow_sequence_item(emitter, event, true)
232 case yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE:
233 return yaml_emitter_emit_flow_sequence_item(emitter, event, false)
235 case yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE:
236 return yaml_emitter_emit_flow_mapping_key(emitter, event, true)
238 case yaml_EMIT_FLOW_MAPPING_KEY_STATE:
239 return yaml_emitter_emit_flow_mapping_key(emitter, event, false)
241 case yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE:
242 return yaml_emitter_emit_flow_mapping_value(emitter, event, true)
244 case yaml_EMIT_FLOW_MAPPING_VALUE_STATE:
245 return yaml_emitter_emit_flow_mapping_value(emitter, event, false)
247 case yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE:
248 return yaml_emitter_emit_block_sequence_item(emitter, event, true)
250 case yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE:
251 return yaml_emitter_emit_block_sequence_item(emitter, event, false)
253 case yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE:
254 return yaml_emitter_emit_block_mapping_key(emitter, event, true)
256 case yaml_EMIT_BLOCK_MAPPING_KEY_STATE:
257 return yaml_emitter_emit_block_mapping_key(emitter, event, false)
259 case yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE:
260 return yaml_emitter_emit_block_mapping_value(emitter, event, true)
262 case yaml_EMIT_BLOCK_MAPPING_VALUE_STATE:
263 return yaml_emitter_emit_block_mapping_value(emitter, event, false)
265 case yaml_EMIT_END_STATE:
266 return yaml_emitter_set_emitter_error(emitter, "expected nothing after STREAM-END")
268 panic("invalid emitter state")
271 // Expect STREAM-START.
272 func yaml_emitter_emit_stream_start(emitter *yaml_emitter_t, event *yaml_event_t) bool {
273 if event.typ != yaml_STREAM_START_EVENT {
274 return yaml_emitter_set_emitter_error(emitter, "expected STREAM-START")
276 if emitter.encoding == yaml_ANY_ENCODING {
277 emitter.encoding = event.encoding
278 if emitter.encoding == yaml_ANY_ENCODING {
279 emitter.encoding = yaml_UTF8_ENCODING
282 if emitter.best_indent < 2 || emitter.best_indent > 9 {
283 emitter.best_indent = 2
285 if emitter.best_width >= 0 && emitter.best_width <= emitter.best_indent*2 {
286 emitter.best_width = 80
288 if emitter.best_width < 0 {
289 emitter.best_width = 1<<31 - 1
291 if emitter.line_break == yaml_ANY_BREAK {
292 emitter.line_break = yaml_LN_BREAK
298 emitter.whitespace = true
299 emitter.indention = true
301 if emitter.encoding != yaml_UTF8_ENCODING {
302 if !yaml_emitter_write_bom(emitter) {
306 emitter.state = yaml_EMIT_FIRST_DOCUMENT_START_STATE
310 // Expect DOCUMENT-START or STREAM-END.
311 func yaml_emitter_emit_document_start(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool {
313 if event.typ == yaml_DOCUMENT_START_EVENT {
315 if event.version_directive != nil {
316 if !yaml_emitter_analyze_version_directive(emitter, event.version_directive) {
321 for i := 0; i < len(event.tag_directives); i++ {
322 tag_directive := &event.tag_directives[i]
323 if !yaml_emitter_analyze_tag_directive(emitter, tag_directive) {
326 if !yaml_emitter_append_tag_directive(emitter, tag_directive, false) {
331 for i := 0; i < len(default_tag_directives); i++ {
332 tag_directive := &default_tag_directives[i]
333 if !yaml_emitter_append_tag_directive(emitter, tag_directive, true) {
338 implicit := event.implicit
339 if !first || emitter.canonical {
343 if emitter.open_ended && (event.version_directive != nil || len(event.tag_directives) > 0) {
344 if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) {
347 if !yaml_emitter_write_indent(emitter) {
352 if event.version_directive != nil {
354 if !yaml_emitter_write_indicator(emitter, []byte("%YAML"), true, false, false) {
357 if !yaml_emitter_write_indicator(emitter, []byte("1.1"), true, false, false) {
360 if !yaml_emitter_write_indent(emitter) {
365 if len(event.tag_directives) > 0 {
367 for i := 0; i < len(event.tag_directives); i++ {
368 tag_directive := &event.tag_directives[i]
369 if !yaml_emitter_write_indicator(emitter, []byte("%TAG"), true, false, false) {
372 if !yaml_emitter_write_tag_handle(emitter, tag_directive.handle) {
375 if !yaml_emitter_write_tag_content(emitter, tag_directive.prefix, true) {
378 if !yaml_emitter_write_indent(emitter) {
384 if yaml_emitter_check_empty_document(emitter) {
388 if !yaml_emitter_write_indent(emitter) {
391 if !yaml_emitter_write_indicator(emitter, []byte("---"), true, false, false) {
394 if emitter.canonical {
395 if !yaml_emitter_write_indent(emitter) {
401 emitter.state = yaml_EMIT_DOCUMENT_CONTENT_STATE
405 if event.typ == yaml_STREAM_END_EVENT {
406 if emitter.open_ended {
407 if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) {
410 if !yaml_emitter_write_indent(emitter) {
414 if !yaml_emitter_flush(emitter) {
417 emitter.state = yaml_EMIT_END_STATE
421 return yaml_emitter_set_emitter_error(emitter, "expected DOCUMENT-START or STREAM-END")
424 // Expect the root node.
425 func yaml_emitter_emit_document_content(emitter *yaml_emitter_t, event *yaml_event_t) bool {
426 emitter.states = append(emitter.states, yaml_EMIT_DOCUMENT_END_STATE)
427 return yaml_emitter_emit_node(emitter, event, true, false, false, false)
430 // Expect DOCUMENT-END.
431 func yaml_emitter_emit_document_end(emitter *yaml_emitter_t, event *yaml_event_t) bool {
432 if event.typ != yaml_DOCUMENT_END_EVENT {
433 return yaml_emitter_set_emitter_error(emitter, "expected DOCUMENT-END")
435 if !yaml_emitter_write_indent(emitter) {
439 // [Go] Allocate the slice elsewhere.
440 if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) {
443 if !yaml_emitter_write_indent(emitter) {
447 if !yaml_emitter_flush(emitter) {
450 emitter.state = yaml_EMIT_DOCUMENT_START_STATE
451 emitter.tag_directives = emitter.tag_directives[:0]
455 // Expect a flow item node.
456 func yaml_emitter_emit_flow_sequence_item(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool {
458 if !yaml_emitter_write_indicator(emitter, []byte{'['}, true, true, false) {
461 if !yaml_emitter_increase_indent(emitter, true, false) {
467 if event.typ == yaml_SEQUENCE_END_EVENT {
469 emitter.indent = emitter.indents[len(emitter.indents)-1]
470 emitter.indents = emitter.indents[:len(emitter.indents)-1]
471 if emitter.canonical && !first {
472 if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) {
475 if !yaml_emitter_write_indent(emitter) {
479 if !yaml_emitter_write_indicator(emitter, []byte{']'}, false, false, false) {
482 emitter.state = emitter.states[len(emitter.states)-1]
483 emitter.states = emitter.states[:len(emitter.states)-1]
489 if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) {
494 if emitter.canonical || emitter.column > emitter.best_width {
495 if !yaml_emitter_write_indent(emitter) {
499 emitter.states = append(emitter.states, yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE)
500 return yaml_emitter_emit_node(emitter, event, false, true, false, false)
503 // Expect a flow key node.
504 func yaml_emitter_emit_flow_mapping_key(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool {
506 if !yaml_emitter_write_indicator(emitter, []byte{'{'}, true, true, false) {
509 if !yaml_emitter_increase_indent(emitter, true, false) {
515 if event.typ == yaml_MAPPING_END_EVENT {
517 emitter.indent = emitter.indents[len(emitter.indents)-1]
518 emitter.indents = emitter.indents[:len(emitter.indents)-1]
519 if emitter.canonical && !first {
520 if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) {
523 if !yaml_emitter_write_indent(emitter) {
527 if !yaml_emitter_write_indicator(emitter, []byte{'}'}, false, false, false) {
530 emitter.state = emitter.states[len(emitter.states)-1]
531 emitter.states = emitter.states[:len(emitter.states)-1]
536 if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) {
540 if emitter.canonical || emitter.column > emitter.best_width {
541 if !yaml_emitter_write_indent(emitter) {
546 if !emitter.canonical && yaml_emitter_check_simple_key(emitter) {
547 emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE)
548 return yaml_emitter_emit_node(emitter, event, false, false, true, true)
550 if !yaml_emitter_write_indicator(emitter, []byte{'?'}, true, false, false) {
553 emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_VALUE_STATE)
554 return yaml_emitter_emit_node(emitter, event, false, false, true, false)
557 // Expect a flow value node.
558 func yaml_emitter_emit_flow_mapping_value(emitter *yaml_emitter_t, event *yaml_event_t, simple bool) bool {
560 if !yaml_emitter_write_indicator(emitter, []byte{':'}, false, false, false) {
564 if emitter.canonical || emitter.column > emitter.best_width {
565 if !yaml_emitter_write_indent(emitter) {
569 if !yaml_emitter_write_indicator(emitter, []byte{':'}, true, false, false) {
573 emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_KEY_STATE)
574 return yaml_emitter_emit_node(emitter, event, false, false, true, false)
577 // Expect a block item node.
578 func yaml_emitter_emit_block_sequence_item(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool {
580 if !yaml_emitter_increase_indent(emitter, false, emitter.mapping_context && !emitter.indention) {
584 if event.typ == yaml_SEQUENCE_END_EVENT {
585 emitter.indent = emitter.indents[len(emitter.indents)-1]
586 emitter.indents = emitter.indents[:len(emitter.indents)-1]
587 emitter.state = emitter.states[len(emitter.states)-1]
588 emitter.states = emitter.states[:len(emitter.states)-1]
591 if !yaml_emitter_write_indent(emitter) {
594 if !yaml_emitter_write_indicator(emitter, []byte{'-'}, true, false, true) {
597 emitter.states = append(emitter.states, yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE)
598 return yaml_emitter_emit_node(emitter, event, false, true, false, false)
601 // Expect a block key node.
602 func yaml_emitter_emit_block_mapping_key(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool {
604 if !yaml_emitter_increase_indent(emitter, false, false) {
608 if event.typ == yaml_MAPPING_END_EVENT {
609 emitter.indent = emitter.indents[len(emitter.indents)-1]
610 emitter.indents = emitter.indents[:len(emitter.indents)-1]
611 emitter.state = emitter.states[len(emitter.states)-1]
612 emitter.states = emitter.states[:len(emitter.states)-1]
615 if !yaml_emitter_write_indent(emitter) {
618 if yaml_emitter_check_simple_key(emitter) {
619 emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE)
620 return yaml_emitter_emit_node(emitter, event, false, false, true, true)
622 if !yaml_emitter_write_indicator(emitter, []byte{'?'}, true, false, true) {
625 emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_VALUE_STATE)
626 return yaml_emitter_emit_node(emitter, event, false, false, true, false)
629 // Expect a block value node.
630 func yaml_emitter_emit_block_mapping_value(emitter *yaml_emitter_t, event *yaml_event_t, simple bool) bool {
632 if !yaml_emitter_write_indicator(emitter, []byte{':'}, false, false, false) {
636 if !yaml_emitter_write_indent(emitter) {
639 if !yaml_emitter_write_indicator(emitter, []byte{':'}, true, false, true) {
643 emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_KEY_STATE)
644 return yaml_emitter_emit_node(emitter, event, false, false, true, false)
648 func yaml_emitter_emit_node(emitter *yaml_emitter_t, event *yaml_event_t,
649 root bool, sequence bool, mapping bool, simple_key bool) bool {
651 emitter.root_context = root
652 emitter.sequence_context = sequence
653 emitter.mapping_context = mapping
654 emitter.simple_key_context = simple_key
657 case yaml_ALIAS_EVENT:
658 return yaml_emitter_emit_alias(emitter, event)
659 case yaml_SCALAR_EVENT:
660 return yaml_emitter_emit_scalar(emitter, event)
661 case yaml_SEQUENCE_START_EVENT:
662 return yaml_emitter_emit_sequence_start(emitter, event)
663 case yaml_MAPPING_START_EVENT:
664 return yaml_emitter_emit_mapping_start(emitter, event)
666 return yaml_emitter_set_emitter_error(emitter,
667 "expected SCALAR, SEQUENCE-START, MAPPING-START, or ALIAS")
673 func yaml_emitter_emit_alias(emitter *yaml_emitter_t, event *yaml_event_t) bool {
674 if !yaml_emitter_process_anchor(emitter) {
677 emitter.state = emitter.states[len(emitter.states)-1]
678 emitter.states = emitter.states[:len(emitter.states)-1]
683 func yaml_emitter_emit_scalar(emitter *yaml_emitter_t, event *yaml_event_t) bool {
684 if !yaml_emitter_select_scalar_style(emitter, event) {
687 if !yaml_emitter_process_anchor(emitter) {
690 if !yaml_emitter_process_tag(emitter) {
693 if !yaml_emitter_increase_indent(emitter, true, false) {
696 if !yaml_emitter_process_scalar(emitter) {
699 emitter.indent = emitter.indents[len(emitter.indents)-1]
700 emitter.indents = emitter.indents[:len(emitter.indents)-1]
701 emitter.state = emitter.states[len(emitter.states)-1]
702 emitter.states = emitter.states[:len(emitter.states)-1]
706 // Expect SEQUENCE-START.
707 func yaml_emitter_emit_sequence_start(emitter *yaml_emitter_t, event *yaml_event_t) bool {
708 if !yaml_emitter_process_anchor(emitter) {
711 if !yaml_emitter_process_tag(emitter) {
714 if emitter.flow_level > 0 || emitter.canonical || event.sequence_style() == yaml_FLOW_SEQUENCE_STYLE ||
715 yaml_emitter_check_empty_sequence(emitter) {
716 emitter.state = yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE
718 emitter.state = yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE
723 // Expect MAPPING-START.
724 func yaml_emitter_emit_mapping_start(emitter *yaml_emitter_t, event *yaml_event_t) bool {
725 if !yaml_emitter_process_anchor(emitter) {
728 if !yaml_emitter_process_tag(emitter) {
731 if emitter.flow_level > 0 || emitter.canonical || event.mapping_style() == yaml_FLOW_MAPPING_STYLE ||
732 yaml_emitter_check_empty_mapping(emitter) {
733 emitter.state = yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE
735 emitter.state = yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE
740 // Check if the document content is an empty scalar.
741 func yaml_emitter_check_empty_document(emitter *yaml_emitter_t) bool {
742 return false // [Go] Huh?
745 // Check if the next events represent an empty sequence.
746 func yaml_emitter_check_empty_sequence(emitter *yaml_emitter_t) bool {
747 if len(emitter.events)-emitter.events_head < 2 {
750 return emitter.events[emitter.events_head].typ == yaml_SEQUENCE_START_EVENT &&
751 emitter.events[emitter.events_head+1].typ == yaml_SEQUENCE_END_EVENT
754 // Check if the next events represent an empty mapping.
755 func yaml_emitter_check_empty_mapping(emitter *yaml_emitter_t) bool {
756 if len(emitter.events)-emitter.events_head < 2 {
759 return emitter.events[emitter.events_head].typ == yaml_MAPPING_START_EVENT &&
760 emitter.events[emitter.events_head+1].typ == yaml_MAPPING_END_EVENT
763 // Check if the next node can be expressed as a simple key.
764 func yaml_emitter_check_simple_key(emitter *yaml_emitter_t) bool {
766 switch emitter.events[emitter.events_head].typ {
767 case yaml_ALIAS_EVENT:
768 length += len(emitter.anchor_data.anchor)
769 case yaml_SCALAR_EVENT:
770 if emitter.scalar_data.multiline {
773 length += len(emitter.anchor_data.anchor) +
774 len(emitter.tag_data.handle) +
775 len(emitter.tag_data.suffix) +
776 len(emitter.scalar_data.value)
777 case yaml_SEQUENCE_START_EVENT:
778 if !yaml_emitter_check_empty_sequence(emitter) {
781 length += len(emitter.anchor_data.anchor) +
782 len(emitter.tag_data.handle) +
783 len(emitter.tag_data.suffix)
784 case yaml_MAPPING_START_EVENT:
785 if !yaml_emitter_check_empty_mapping(emitter) {
788 length += len(emitter.anchor_data.anchor) +
789 len(emitter.tag_data.handle) +
790 len(emitter.tag_data.suffix)
797 // Determine an acceptable scalar style.
798 func yaml_emitter_select_scalar_style(emitter *yaml_emitter_t, event *yaml_event_t) bool {
800 no_tag := len(emitter.tag_data.handle) == 0 && len(emitter.tag_data.suffix) == 0
801 if no_tag && !event.implicit && !event.quoted_implicit {
802 return yaml_emitter_set_emitter_error(emitter, "neither tag nor implicit flags are specified")
805 style := event.scalar_style()
806 if style == yaml_ANY_SCALAR_STYLE {
807 style = yaml_PLAIN_SCALAR_STYLE
809 if emitter.canonical {
810 style = yaml_DOUBLE_QUOTED_SCALAR_STYLE
812 if emitter.simple_key_context && emitter.scalar_data.multiline {
813 style = yaml_DOUBLE_QUOTED_SCALAR_STYLE
816 if style == yaml_PLAIN_SCALAR_STYLE {
817 if emitter.flow_level > 0 && !emitter.scalar_data.flow_plain_allowed ||
818 emitter.flow_level == 0 && !emitter.scalar_data.block_plain_allowed {
819 style = yaml_SINGLE_QUOTED_SCALAR_STYLE
821 if len(emitter.scalar_data.value) == 0 && (emitter.flow_level > 0 || emitter.simple_key_context) {
822 style = yaml_SINGLE_QUOTED_SCALAR_STYLE
824 if no_tag && !event.implicit {
825 style = yaml_SINGLE_QUOTED_SCALAR_STYLE
828 if style == yaml_SINGLE_QUOTED_SCALAR_STYLE {
829 if !emitter.scalar_data.single_quoted_allowed {
830 style = yaml_DOUBLE_QUOTED_SCALAR_STYLE
833 if style == yaml_LITERAL_SCALAR_STYLE || style == yaml_FOLDED_SCALAR_STYLE {
834 if !emitter.scalar_data.block_allowed || emitter.flow_level > 0 || emitter.simple_key_context {
835 style = yaml_DOUBLE_QUOTED_SCALAR_STYLE
839 if no_tag && !event.quoted_implicit && style != yaml_PLAIN_SCALAR_STYLE {
840 emitter.tag_data.handle = []byte{'!'}
842 emitter.scalar_data.style = style
847 func yaml_emitter_process_anchor(emitter *yaml_emitter_t) bool {
848 if emitter.anchor_data.anchor == nil {
852 if emitter.anchor_data.alias {
855 if !yaml_emitter_write_indicator(emitter, c, true, false, false) {
858 return yaml_emitter_write_anchor(emitter, emitter.anchor_data.anchor)
862 func yaml_emitter_process_tag(emitter *yaml_emitter_t) bool {
863 if len(emitter.tag_data.handle) == 0 && len(emitter.tag_data.suffix) == 0 {
866 if len(emitter.tag_data.handle) > 0 {
867 if !yaml_emitter_write_tag_handle(emitter, emitter.tag_data.handle) {
870 if len(emitter.tag_data.suffix) > 0 {
871 if !yaml_emitter_write_tag_content(emitter, emitter.tag_data.suffix, false) {
876 // [Go] Allocate these slices elsewhere.
877 if !yaml_emitter_write_indicator(emitter, []byte("!<"), true, false, false) {
880 if !yaml_emitter_write_tag_content(emitter, emitter.tag_data.suffix, false) {
883 if !yaml_emitter_write_indicator(emitter, []byte{'>'}, false, false, false) {
891 func yaml_emitter_process_scalar(emitter *yaml_emitter_t) bool {
892 switch emitter.scalar_data.style {
893 case yaml_PLAIN_SCALAR_STYLE:
894 return yaml_emitter_write_plain_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context)
896 case yaml_SINGLE_QUOTED_SCALAR_STYLE:
897 return yaml_emitter_write_single_quoted_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context)
899 case yaml_DOUBLE_QUOTED_SCALAR_STYLE:
900 return yaml_emitter_write_double_quoted_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context)
902 case yaml_LITERAL_SCALAR_STYLE:
903 return yaml_emitter_write_literal_scalar(emitter, emitter.scalar_data.value)
905 case yaml_FOLDED_SCALAR_STYLE:
906 return yaml_emitter_write_folded_scalar(emitter, emitter.scalar_data.value)
908 panic("unknown scalar style")
911 // Check if a %YAML directive is valid.
912 func yaml_emitter_analyze_version_directive(emitter *yaml_emitter_t, version_directive *yaml_version_directive_t) bool {
913 if version_directive.major != 1 || version_directive.minor != 1 {
914 return yaml_emitter_set_emitter_error(emitter, "incompatible %YAML directive")
919 // Check if a %TAG directive is valid.
920 func yaml_emitter_analyze_tag_directive(emitter *yaml_emitter_t, tag_directive *yaml_tag_directive_t) bool {
921 handle := tag_directive.handle
922 prefix := tag_directive.prefix
923 if len(handle) == 0 {
924 return yaml_emitter_set_emitter_error(emitter, "tag handle must not be empty")
926 if handle[0] != '!' {
927 return yaml_emitter_set_emitter_error(emitter, "tag handle must start with '!'")
929 if handle[len(handle)-1] != '!' {
930 return yaml_emitter_set_emitter_error(emitter, "tag handle must end with '!'")
932 for i := 1; i < len(handle)-1; i += width(handle[i]) {
933 if !is_alpha(handle, i) {
934 return yaml_emitter_set_emitter_error(emitter, "tag handle must contain alphanumerical characters only")
937 if len(prefix) == 0 {
938 return yaml_emitter_set_emitter_error(emitter, "tag prefix must not be empty")
943 // Check if an anchor is valid.
944 func yaml_emitter_analyze_anchor(emitter *yaml_emitter_t, anchor []byte, alias bool) bool {
945 if len(anchor) == 0 {
946 problem := "anchor value must not be empty"
948 problem = "alias value must not be empty"
950 return yaml_emitter_set_emitter_error(emitter, problem)
952 for i := 0; i < len(anchor); i += width(anchor[i]) {
953 if !is_alpha(anchor, i) {
954 problem := "anchor value must contain alphanumerical characters only"
956 problem = "alias value must contain alphanumerical characters only"
958 return yaml_emitter_set_emitter_error(emitter, problem)
961 emitter.anchor_data.anchor = anchor
962 emitter.anchor_data.alias = alias
966 // Check if a tag is valid.
967 func yaml_emitter_analyze_tag(emitter *yaml_emitter_t, tag []byte) bool {
969 return yaml_emitter_set_emitter_error(emitter, "tag value must not be empty")
971 for i := 0; i < len(emitter.tag_directives); i++ {
972 tag_directive := &emitter.tag_directives[i]
973 if bytes.HasPrefix(tag, tag_directive.prefix) {
974 emitter.tag_data.handle = tag_directive.handle
975 emitter.tag_data.suffix = tag[len(tag_directive.prefix):]
979 emitter.tag_data.suffix = tag
983 // Check if a scalar is valid.
984 func yaml_emitter_analyze_scalar(emitter *yaml_emitter_t, value []byte) bool {
986 block_indicators = false
987 flow_indicators = false
989 special_characters = false
991 leading_space = false
992 leading_break = false
993 trailing_space = false
994 trailing_break = false
998 preceeded_by_whitespace = false
999 followed_by_whitespace = false
1000 previous_space = false
1001 previous_break = false
1004 emitter.scalar_data.value = value
1006 if len(value) == 0 {
1007 emitter.scalar_data.multiline = false
1008 emitter.scalar_data.flow_plain_allowed = false
1009 emitter.scalar_data.block_plain_allowed = true
1010 emitter.scalar_data.single_quoted_allowed = true
1011 emitter.scalar_data.block_allowed = false
1015 if len(value) >= 3 && ((value[0] == '-' && value[1] == '-' && value[2] == '-') || (value[0] == '.' && value[1] == '.' && value[2] == '.')) {
1016 block_indicators = true
1017 flow_indicators = true
1020 preceeded_by_whitespace = true
1021 for i, w := 0, 0; i < len(value); i += w {
1023 followed_by_whitespace = i+w >= len(value) || is_blank(value, i+w)
1027 case '#', ',', '[', ']', '{', '}', '&', '*', '!', '|', '>', '\'', '"', '%', '@', '`':
1028 flow_indicators = true
1029 block_indicators = true
1031 flow_indicators = true
1032 if followed_by_whitespace {
1033 block_indicators = true
1036 if followed_by_whitespace {
1037 flow_indicators = true
1038 block_indicators = true
1043 case ',', '?', '[', ']', '{', '}':
1044 flow_indicators = true
1046 flow_indicators = true
1047 if followed_by_whitespace {
1048 block_indicators = true
1051 if preceeded_by_whitespace {
1052 flow_indicators = true
1053 block_indicators = true
1058 if !is_printable(value, i) || !is_ascii(value, i) && !emitter.unicode {
1059 special_characters = true
1061 if is_space(value, i) {
1063 leading_space = true
1065 if i+width(value[i]) == len(value) {
1066 trailing_space = true
1071 previous_space = true
1072 previous_break = false
1073 } else if is_break(value, i) {
1076 leading_break = true
1078 if i+width(value[i]) == len(value) {
1079 trailing_break = true
1084 previous_space = false
1085 previous_break = true
1087 previous_space = false
1088 previous_break = false
1091 // [Go]: Why 'z'? Couldn't be the end of the string as that's the loop condition.
1092 preceeded_by_whitespace = is_blankz(value, i)
1095 emitter.scalar_data.multiline = line_breaks
1096 emitter.scalar_data.flow_plain_allowed = true
1097 emitter.scalar_data.block_plain_allowed = true
1098 emitter.scalar_data.single_quoted_allowed = true
1099 emitter.scalar_data.block_allowed = true
1101 if leading_space || leading_break || trailing_space || trailing_break {
1102 emitter.scalar_data.flow_plain_allowed = false
1103 emitter.scalar_data.block_plain_allowed = false
1106 emitter.scalar_data.block_allowed = false
1109 emitter.scalar_data.flow_plain_allowed = false
1110 emitter.scalar_data.block_plain_allowed = false
1111 emitter.scalar_data.single_quoted_allowed = false
1113 if space_break || special_characters {
1114 emitter.scalar_data.flow_plain_allowed = false
1115 emitter.scalar_data.block_plain_allowed = false
1116 emitter.scalar_data.single_quoted_allowed = false
1117 emitter.scalar_data.block_allowed = false
1120 emitter.scalar_data.flow_plain_allowed = false
1121 emitter.scalar_data.block_plain_allowed = false
1123 if flow_indicators {
1124 emitter.scalar_data.flow_plain_allowed = false
1126 if block_indicators {
1127 emitter.scalar_data.block_plain_allowed = false
1132 // Check if the event data is valid.
1133 func yaml_emitter_analyze_event(emitter *yaml_emitter_t, event *yaml_event_t) bool {
1135 emitter.anchor_data.anchor = nil
1136 emitter.tag_data.handle = nil
1137 emitter.tag_data.suffix = nil
1138 emitter.scalar_data.value = nil
1141 case yaml_ALIAS_EVENT:
1142 if !yaml_emitter_analyze_anchor(emitter, event.anchor, true) {
1146 case yaml_SCALAR_EVENT:
1147 if len(event.anchor) > 0 {
1148 if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) {
1152 if len(event.tag) > 0 && (emitter.canonical || (!event.implicit && !event.quoted_implicit)) {
1153 if !yaml_emitter_analyze_tag(emitter, event.tag) {
1157 if !yaml_emitter_analyze_scalar(emitter, event.value) {
1161 case yaml_SEQUENCE_START_EVENT:
1162 if len(event.anchor) > 0 {
1163 if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) {
1167 if len(event.tag) > 0 && (emitter.canonical || !event.implicit) {
1168 if !yaml_emitter_analyze_tag(emitter, event.tag) {
1173 case yaml_MAPPING_START_EVENT:
1174 if len(event.anchor) > 0 {
1175 if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) {
1179 if len(event.tag) > 0 && (emitter.canonical || !event.implicit) {
1180 if !yaml_emitter_analyze_tag(emitter, event.tag) {
1188 // Write the BOM character.
1189 func yaml_emitter_write_bom(emitter *yaml_emitter_t) bool {
1190 if !flush(emitter) {
1193 pos := emitter.buffer_pos
1194 emitter.buffer[pos+0] = '\xEF'
1195 emitter.buffer[pos+1] = '\xBB'
1196 emitter.buffer[pos+2] = '\xBF'
1197 emitter.buffer_pos += 3
1201 func yaml_emitter_write_indent(emitter *yaml_emitter_t) bool {
1202 indent := emitter.indent
1206 if !emitter.indention || emitter.column > indent || (emitter.column == indent && !emitter.whitespace) {
1207 if !put_break(emitter) {
1211 for emitter.column < indent {
1212 if !put(emitter, ' ') {
1216 emitter.whitespace = true
1217 emitter.indention = true
1221 func yaml_emitter_write_indicator(emitter *yaml_emitter_t, indicator []byte, need_whitespace, is_whitespace, is_indention bool) bool {
1222 if need_whitespace && !emitter.whitespace {
1223 if !put(emitter, ' ') {
1227 if !write_all(emitter, indicator) {
1230 emitter.whitespace = is_whitespace
1231 emitter.indention = (emitter.indention && is_indention)
1232 emitter.open_ended = false
1236 func yaml_emitter_write_anchor(emitter *yaml_emitter_t, value []byte) bool {
1237 if !write_all(emitter, value) {
1240 emitter.whitespace = false
1241 emitter.indention = false
1245 func yaml_emitter_write_tag_handle(emitter *yaml_emitter_t, value []byte) bool {
1246 if !emitter.whitespace {
1247 if !put(emitter, ' ') {
1251 if !write_all(emitter, value) {
1254 emitter.whitespace = false
1255 emitter.indention = false
1259 func yaml_emitter_write_tag_content(emitter *yaml_emitter_t, value []byte, need_whitespace bool) bool {
1260 if need_whitespace && !emitter.whitespace {
1261 if !put(emitter, ' ') {
1265 for i := 0; i < len(value); {
1268 case ';', '/', '?', ':', '@', '&', '=', '+', '$', ',', '_', '.', '~', '*', '\'', '(', ')', '[', ']':
1271 must_write = is_alpha(value, i)
1274 if !write(emitter, value, &i) {
1278 w := width(value[i])
1279 for k := 0; k < w; k++ {
1282 if !put(emitter, '%') {
1292 if !put(emitter, c) {
1302 if !put(emitter, c) {
1308 emitter.whitespace = false
1309 emitter.indention = false
1313 func yaml_emitter_write_plain_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool {
1314 if !emitter.whitespace {
1315 if !put(emitter, ' ') {
1322 for i := 0; i < len(value); {
1323 if is_space(value, i) {
1324 if allow_breaks && !spaces && emitter.column > emitter.best_width && !is_space(value, i+1) {
1325 if !yaml_emitter_write_indent(emitter) {
1328 i += width(value[i])
1330 if !write(emitter, value, &i) {
1335 } else if is_break(value, i) {
1336 if !breaks && value[i] == '\n' {
1337 if !put_break(emitter) {
1341 if !write_break(emitter, value, &i) {
1344 emitter.indention = true
1348 if !yaml_emitter_write_indent(emitter) {
1352 if !write(emitter, value, &i) {
1355 emitter.indention = false
1361 emitter.whitespace = false
1362 emitter.indention = false
1363 if emitter.root_context {
1364 emitter.open_ended = true
1370 func yaml_emitter_write_single_quoted_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool {
1372 if !yaml_emitter_write_indicator(emitter, []byte{'\''}, true, false, false) {
1378 for i := 0; i < len(value); {
1379 if is_space(value, i) {
1380 if allow_breaks && !spaces && emitter.column > emitter.best_width && i > 0 && i < len(value)-1 && !is_space(value, i+1) {
1381 if !yaml_emitter_write_indent(emitter) {
1384 i += width(value[i])
1386 if !write(emitter, value, &i) {
1391 } else if is_break(value, i) {
1392 if !breaks && value[i] == '\n' {
1393 if !put_break(emitter) {
1397 if !write_break(emitter, value, &i) {
1400 emitter.indention = true
1404 if !yaml_emitter_write_indent(emitter) {
1408 if value[i] == '\'' {
1409 if !put(emitter, '\'') {
1413 if !write(emitter, value, &i) {
1416 emitter.indention = false
1421 if !yaml_emitter_write_indicator(emitter, []byte{'\''}, false, false, false) {
1424 emitter.whitespace = false
1425 emitter.indention = false
1429 func yaml_emitter_write_double_quoted_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool {
1431 if !yaml_emitter_write_indicator(emitter, []byte{'"'}, true, false, false) {
1435 for i := 0; i < len(value); {
1436 if !is_printable(value, i) || (!emitter.unicode && !is_ascii(value, i)) ||
1437 is_bom(value, i) || is_break(value, i) ||
1438 value[i] == '"' || value[i] == '\\' {
1445 case octet&0x80 == 0x00:
1446 w, v = 1, rune(octet&0x7F)
1447 case octet&0xE0 == 0xC0:
1448 w, v = 2, rune(octet&0x1F)
1449 case octet&0xF0 == 0xE0:
1450 w, v = 3, rune(octet&0x0F)
1451 case octet&0xF8 == 0xF0:
1452 w, v = 4, rune(octet&0x07)
1454 for k := 1; k < w; k++ {
1456 v = (v << 6) + (rune(octet) & 0x3F)
1460 if !put(emitter, '\\') {
1467 ok = put(emitter, '0')
1469 ok = put(emitter, 'a')
1471 ok = put(emitter, 'b')
1473 ok = put(emitter, 't')
1475 ok = put(emitter, 'n')
1477 ok = put(emitter, 'v')
1479 ok = put(emitter, 'f')
1481 ok = put(emitter, 'r')
1483 ok = put(emitter, 'e')
1485 ok = put(emitter, '"')
1487 ok = put(emitter, '\\')
1489 ok = put(emitter, 'N')
1491 ok = put(emitter, '_')
1493 ok = put(emitter, 'L')
1495 ok = put(emitter, 'P')
1498 ok = put(emitter, 'x')
1500 } else if v <= 0xFFFF {
1501 ok = put(emitter, 'u')
1504 ok = put(emitter, 'U')
1507 for k := (w - 1) * 4; ok && k >= 0; k -= 4 {
1508 digit := byte((v >> uint(k)) & 0x0F)
1510 ok = put(emitter, digit+'0')
1512 ok = put(emitter, digit+'A'-10)
1520 } else if is_space(value, i) {
1521 if allow_breaks && !spaces && emitter.column > emitter.best_width && i > 0 && i < len(value)-1 {
1522 if !yaml_emitter_write_indent(emitter) {
1525 if is_space(value, i+1) {
1526 if !put(emitter, '\\') {
1530 i += width(value[i])
1531 } else if !write(emitter, value, &i) {
1536 if !write(emitter, value, &i) {
1542 if !yaml_emitter_write_indicator(emitter, []byte{'"'}, false, false, false) {
1545 emitter.whitespace = false
1546 emitter.indention = false
1550 func yaml_emitter_write_block_scalar_hints(emitter *yaml_emitter_t, value []byte) bool {
1551 if is_space(value, 0) || is_break(value, 0) {
1552 indent_hint := []byte{'0' + byte(emitter.best_indent)}
1553 if !yaml_emitter_write_indicator(emitter, indent_hint, false, false, false) {
1558 emitter.open_ended = false
1560 var chomp_hint [1]byte
1561 if len(value) == 0 {
1565 for value[i]&0xC0 == 0x80 {
1568 if !is_break(value, i) {
1572 emitter.open_ended = true
1575 for value[i]&0xC0 == 0x80 {
1578 if is_break(value, i) {
1580 emitter.open_ended = true
1584 if chomp_hint[0] != 0 {
1585 if !yaml_emitter_write_indicator(emitter, chomp_hint[:], false, false, false) {
1592 func yaml_emitter_write_literal_scalar(emitter *yaml_emitter_t, value []byte) bool {
1593 if !yaml_emitter_write_indicator(emitter, []byte{'|'}, true, false, false) {
1596 if !yaml_emitter_write_block_scalar_hints(emitter, value) {
1599 if !put_break(emitter) {
1602 emitter.indention = true
1603 emitter.whitespace = true
1605 for i := 0; i < len(value); {
1606 if is_break(value, i) {
1607 if !write_break(emitter, value, &i) {
1610 emitter.indention = true
1614 if !yaml_emitter_write_indent(emitter) {
1618 if !write(emitter, value, &i) {
1621 emitter.indention = false
1629 func yaml_emitter_write_folded_scalar(emitter *yaml_emitter_t, value []byte) bool {
1630 if !yaml_emitter_write_indicator(emitter, []byte{'>'}, true, false, false) {
1633 if !yaml_emitter_write_block_scalar_hints(emitter, value) {
1637 if !put_break(emitter) {
1640 emitter.indention = true
1641 emitter.whitespace = true
1644 leading_spaces := true
1645 for i := 0; i < len(value); {
1646 if is_break(value, i) {
1647 if !breaks && !leading_spaces && value[i] == '\n' {
1649 for is_break(value, k) {
1650 k += width(value[k])
1652 if !is_blankz(value, k) {
1653 if !put_break(emitter) {
1658 if !write_break(emitter, value, &i) {
1661 emitter.indention = true
1665 if !yaml_emitter_write_indent(emitter) {
1668 leading_spaces = is_blank(value, i)
1670 if !breaks && is_space(value, i) && !is_space(value, i+1) && emitter.column > emitter.best_width {
1671 if !yaml_emitter_write_indent(emitter) {
1674 i += width(value[i])
1676 if !write(emitter, value, &i) {
1680 emitter.indention = false