# NimYAML - YAML implementation in Nim # (c) Copyright 2016 Felix Krause # # See the file "copying.txt", included in this # distribution, for details about the copyright. ## ================== ## Module yaml/parser ## ================== ## ## This is the low-level parser API. A ``YamlParser`` enables you to parse any ## non-nil string or Stream object as YAML character stream. import tables, strutils, macros, streams import stream, private/lex, private/internal, private/escaping, data when defined(nimNoNil): {.experimental: "notnil".} type YamlParser* = object ## A parser object. Retains its ``TagLibrary`` across calls to ## `parse <#parse,YamlParser,Stream>`_. Can be used ## to access anchor names while parsing a YAML character stream, but ## only until the document goes out of scope (i.e. until ## ``yamlEndDocument`` is yielded). issueWarnings: bool YamlLoadingError* = object of ValueError ## Base class for all exceptions that may be raised during the process ## of loading a YAML character stream. mark*: Mark ## position at which the error has occurred. lineContent*: string ## \ ## content of the line where the error was encountered. Includes a ## second line with a marker ``^`` at the position where the error ## was encountered. YamlParserError* = object of YamlLoadingError ## A parser error is raised if the character stream that is parsed is ## not a valid YAML character stream. This stream cannot and will not be ## parsed wholly nor partially and all events that have been emitted by ## the YamlStream the parser provides should be discarded. ## ## A character stream is invalid YAML if and only if at least one of the ## following conditions apply: ## ## - There are invalid characters in an element whose contents is ## restricted to a limited set of characters. For example, there are ## characters in a tag URI which are not valid URI characters. ## - An element has invalid indentation. This can happen for example if ## a block list element indicated by ``"- "`` is less indented than ## the element in the previous line, but there is no block sequence ## list open at the same indentation level. ## - The YAML structure is invalid. For example, an explicit block map ## indicated by ``"? "`` and ``": "`` may not suddenly have a block ## sequence item (``"- "``) at the same indentation level. Another ## possible violation is closing a flow style object with the wrong ## closing character (``}``, ``]``) or not closing it at all. ## - A custom tag shorthand is used that has not previously been ## declared with a ``%TAG`` directive. ## - Multiple tags or anchors are defined for the same node. ## - An alias is used which does not map to any anchor that has ## previously been declared in the same document. ## - An alias has a tag or anchor associated with it. ## ## Some elements in this list are vague. For a detailed description of a ## valid YAML character stream, see the YAML specification. State = proc(c: Context, e: var Event): bool {.gcSafe, raises: [CatchableError].} Level = object state: State indentation: int Context = ref object of YamlStream handles: seq[tuple[handle, uriPrefix: string]] issueWarnings: bool lex: Lexer levels: seq[Level] keyCache: seq[Event] keyCachePos: int caching: bool headerProps, inlineProps: Properties headerStart, inlineStart: Mark blockIndentation: int const defaultProperties = (yAnchorNone, yTagQuestionMark) # parser states {.push gcSafe, raises: [CatchableError], hint[XCannotRaiseY]: off.} proc atStreamStart(c: Context, e: var Event): bool proc atStreamEnd(c: Context, e : var Event): bool proc beforeDoc(c: Context, e: var Event): bool proc beforeDocEnd(c: Context, e: var Event): bool proc afterDirectivesEnd(c: Context, e: var Event): bool proc beforeImplicitRoot(c: Context, e: var Event): bool proc atBlockIndentation(c: Context, e: var Event): bool proc beforeBlockIndentation(c: Context, e: var Event): bool proc beforeNodeProperties(c: Context, e: var Event): bool proc afterCompactParent(c: Context, e: var Event): bool proc afterCompactParentProps(c: Context, e: var Event): bool proc mergePropsOnNewline(c: Context, e: var Event): bool proc beforeFlowItemProps(c: Context, e: var Event): bool proc inBlockSeq(c: Context, e: var Event): bool proc beforeBlockMapValue(c: Context, e: var Event): bool proc atBlockIndentationProps(c: Context, e: var Event): bool proc beforeFlowItem(c: Context, e: var Event): bool proc afterFlowSeqSep(c: Context, e: var Event): bool proc afterFlowMapSep(c: Context, e: var Event): bool proc atBlockMapKeyProps(c: Context, e: var Event): bool proc afterImplicitKey(c: Context, e: var Event): bool proc afterBlockParent(c: Context, e: var Event): bool proc afterBlockParentProps(c: Context, e: var Event): bool proc afterImplicitPairStart(c: Context, e: var Event): bool proc beforePairValue(c: Context, e: var Event): bool proc atEmptyPairKey(c: Context, e: var Event): bool proc afterFlowMapValue(c: Context, e: var Event): bool proc afterFlowSeqSepProps(c: Context, e: var Event): bool proc afterFlowSeqItem(c: Context, e: var Event): bool proc afterPairValue(c: Context, e: var Event): bool proc emitCached(c: Context, e: var Event): bool {.pop.} template pushLevel(c: Context, newState: State, newIndent: int) = debug("parser: push " & newState.astToStr & ", indent = " & $newIndent) c.levels.add(Level(state: newState, indentation: newIndent)) template pushLevel(c: Context, newState: State) = debug("parser: push " & newState.astToStr) c.levels.add(Level(state: newState)) template transition(c: Context, newState: State) = debug("parser: transition " & newState.astToStr) c.levels[^1].state = newState template transition(c: Context, newState: State, newIndent) = debug("parser: transtion " & newState.astToStr & ", indent = " & $newIndent) c.levels[^1] = Level(state: newState, indentation: newIndent) template updateIndentation(c: Context, newIndent: int) = debug("parser: update indent = " & $newIndent) c.levels[^1].indentation = newIndent template popLevel(c: Context) = debug("parser: pop") discard c.levels.pop() proc resolveHandle(c: Context, handle: string): string {.raises: [].} = for item in c.handles: if item.handle == handle: return item.uriPrefix return "" proc init[T](c: Context, p: YamlParser, source: T) {.inline.} = c.pushLevel(atStreamStart, -2) c.nextImpl = proc(s: YamlStream, e: var Event): bool {.raises: [CatchableError].} = let c = Context(s) return c.levels[^1].state(c, e) c.lastTokenContextImpl = proc(s: YamlStream, lineContent: var string): bool = lineContent = Context(s).lex.currentLine() return true c.headerProps = defaultProperties c.inlineProps = defaultProperties c.issueWarnings = p.issueWarnings c.lex.init(source) c.keyCachePos = 0 c.caching = false # interface proc init*(p: var YamlParser, issueWarnings: bool = false) = ## Initializes a YAML parser. p.issueWarnings = issueWarnings proc initYamlParser*(issueWarnings: bool = false): YamlParser = ## Creates an initializes YAML parser and returns it result.issueWarnings = issueWarnings proc parse*(p: YamlParser, s: Stream): YamlStream = let c = new(Context) c.init(p, s) return c proc parse*(p: YamlParser, s: string): YamlStream = let c = new(Context) c.init(p, s) return c # implementation proc isEmpty(props: Properties): bool = result = props.anchor == yAnchorNone and props.tag == yTagQuestionMark proc generateError(c: Context, message: string): ref YamlParserError {.raises: [], .} = result = (ref YamlParserError)( msg: message, parent: nil, mark: c.lex.curStartPos, lineContent: c.lex.currentLine()) proc safeNext(c: Context) = try: c.lex.next() except LexerError as e: raise (ref YamlParserError)( msg: e.msg, parent: nil, mark: (e.line, e.column), lineContent: e.lineContent) proc parseTag(c: Context): Tag = let handle = c.lex.fullLexeme() var uri = c.resolveHandle(handle) if uri == "": raise c.generateError("unknown handle: " & escape(handle)) c.safeNext() if c.lex.cur != Token.Suffix: raise c.generateError("unexpected token (expected tag suffix): " & $c.lex.cur) uri.add(c.lex.evaluated) return Tag(uri) proc toStyle(t: Token): ScalarStyle = return (case t of Plain: ssPlain of SingleQuoted: ssSingleQuoted of DoubleQuoted: ssDoubleQuoted of Literal: ssLiteral of Folded: ssFolded else: ssAny) proc mergeProps(c: Context, src, target: var Properties) = if src.tag != yTagQuestionMark: if target.tag != yTagQuestionMark: raise c.generateError("Only one tag allowed per node") target.tag = src.tag src.tag = yTagQuestionMark if src.anchor != yAnchorNone: if target.anchor != yAnchorNone: raise c.generateError("Only one anchor allowed per node") target.anchor = src.anchor src.anchor = yAnchorNone proc autoScalarTag(props: Properties, t: Token): Properties = result = props if t in {Token.SingleQuoted, Token.DoubleQuoted} and props.tag == yTagQuestionMark: result.tag = yTagExclamationMark proc atStreamStart(c: Context, e: var Event): bool = c.transition(atStreamEnd) c.pushLevel(beforeDoc, -1) e = Event(startPos: c.lex.curStartPos, endPos: c.lex.curStartPos, kind: yamlStartStream) c.safeNext() resetHandles(c.handles) return true proc atStreamEnd(c: Context, e : var Event): bool = e = Event(startPos: c.lex.curStartPos, endPos: c.lex.curStartPos, kind: yamlEndStream) return true proc beforeDoc(c: Context, e: var Event): bool = var version = "" var seenDirectives = false while true: case c.lex.cur of DocumentEnd: if seenDirectives: raise c.generateError("Missing `---` after directives") c.safeNext() of DirectivesEnd: e = startDocEvent(true, version, c.handles, c.lex.curStartPos, c.lex.curEndPos) c.safeNext() c.transition(beforeDocEnd) c.pushLevel(afterDirectivesEnd, -1) return true of StreamEnd: if seenDirectives: raise c.generateError("Missing `---` after directives") c.popLevel() return false of Indentation: e = startDocEvent(false, version, c.handles, c.lex.curStartPos, c.lex.curEndPos) c.transition(beforeDocEnd) c.pushLevel(beforeImplicitRoot, -1) return true of YamlDirective: seenDirectives = true c.safeNext() if c.lex.cur != Token.DirectiveParam: raise c.generateError("Invalid token (expected YAML version string): " & $c.lex.cur) elif version != "": raise c.generateError("Duplicate %YAML") version = c.lex.fullLexeme() if version != "1.2" and c.issueWarnings: discard # TODO c.safeNext() of TagDirective: seenDirectives = true c.safeNext() if c.lex.cur != Token.TagHandle: raise c.generateError("Invalid token (expected tag handle): " & $c.lex.cur) let tagHandle = c.lex.fullLexeme() c.safeNext() if c.lex.cur != Token.Suffix: raise c.generateError("Invalid token (expected tag URI): " & $c.lex.cur) discard registerHandle(c.handles, tagHandle, c.lex.evaluated) c.safeNext() of UnknownDirective: seenDirectives = true # TODO: issue warning while true: c.safeNext() if c.lex.cur != Token.DirectiveParam: break else: raise c.generateError("Unexpected token (expected directive or document start): " & $c.lex.cur) proc afterDirectivesEnd(c: Context, e: var Event): bool = case c.lex.cur of nodePropertyKind: c.inlineStart = c.lex.curStartPos c.pushLevel(beforeNodeProperties) return false of Indentation: c.headerStart = c.inlineStart c.transition(atBlockIndentation) c.pushLevel(beforeBlockIndentation) return false of DocumentEnd, DirectivesEnd, StreamEnd: e = scalarEvent("", c.inlineProps, ssPlain, c.lex.curStartPos, c.lex.curEndPos) c.popLevel() return true of scalarTokenKind: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.lex.curStartPos, c.lex.curEndPos) c.popLevel() c.safeNext() return true else: raise c.generateError("Illegal content at `---`: " & $c.lex.cur) proc beforeImplicitRoot(c: Context, e: var Event): bool = if c.lex.cur != Token.Indentation: raise c.generateError("Unexpected token (expected line start): " & $c.lex.cur) c.inlineStart = c.lex.curEndPos c.headerStart = c.lex.curEndPos c.updateIndentation(c.lex.recentIndentation()) c.safeNext() case c.lex.cur of SeqItemInd, MapKeyInd, MapValueInd: c.transition(afterCompactParent) return false of scalarTokenKind, MapStart, SeqStart: c.transition(atBlockIndentationProps) return false of nodePropertyKind: c.transition(atBlockIndentationProps) c.pushLevel(beforeNodeProperties) else: raise c.generateError("Unexpected token (expected collection start): " & $c.lex.cur) proc atBlockIndentation(c: Context, e: var Event): bool = if c.blockIndentation == c.levels[^1].indentation and (c.lex.cur != Token.SeqItemInd or c.levels[^3].state == inBlockSeq): e = scalarEvent("", c.headerProps, ssPlain, c.headerStart, c.headerStart) c.headerProps = defaultProperties c.popLevel() c.popLevel() return true c.inlineStart = c.lex.curStartPos c.updateIndentation(c.lex.recentIndentation()) case c.lex.cur of nodePropertyKind: if isEmpty(c.headerProps): c.transition(mergePropsOnNewline) else: c.transition(atBlockIndentationProps) c.pushLevel(beforeNodeProperties) return false of SeqItemInd: e = startSeqEvent(csBlock, c.headerProps, c.headerStart, c.lex.curEndPos) c.headerProps = defaultProperties c.transition(inBlockSeq, c.lex.recentIndentation()) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.lex.recentIndentation()) c.safeNext() return true of MapKeyInd: e = startMapEvent(csBlock, c.headerProps, c.headerStart, c.lex.curEndPos) c.headerProps = defaultProperties c.transition(beforeBlockMapValue, c.lex.recentIndentation()) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.lex.recentIndentation()) c.safeNext() return true of Plain, SingleQuoted, DoubleQuoted: c.updateIndentation(c.lex.recentIndentation()) let scalarToken = c.lex.cur e = scalarEvent(c.lex.evaluated, c.headerProps, toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.headerProps = defaultProperties let headerEnd = c.lex.curStartPos c.safeNext() if c.lex.cur == Token.MapValueInd: if c.lex.lastScalarWasMultiline(): raise c.generateError("Implicit mapping key may not be multiline") let props = e.scalarProperties e.scalarProperties = autoScalarTag(defaultProperties, scalarToken) c.keyCache.add(move(e)) e = startMapEvent(csBlock, props, c.headerStart, headerEnd) c.transition(afterImplicitKey) c.pushLevel(emitCached) else: e.scalarProperties = autoScalarTag(e.scalarProperties, scalarToken) c.popLevel() return true of Alias: e = aliasEvent(c.lex.shortLexeme().Anchor, c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties let headerEnd = c.lex.curStartPos c.safeNext() if c.lex.cur == Token.MapValueInd: c.keyCache.add(move(e)) e = startMapEvent(csBlock, c.headerProps, c.headerStart, headerEnd) c.headerProps = defaultProperties c.transition(afterImplicitKey) c.pushLevel(emitCached) elif not isEmpty(c.headerProps): raise c.generateError("Alias may not have properties") else: c.popLevel() return true else: c.transition(atBlockIndentationProps) return false proc atBlockIndentationProps(c: Context, e: var Event): bool = c.updateIndentation(c.lex.recentIndentation()) case c.lex.cur of MapValueInd: c.keyCache.add(scalarEvent("", c.inlineProps, ssPlain, c.inlineStart, c.lex.curEndPos)) c.inlineProps = defaultProperties e = startMapEvent(csBlock, c.headerProps, c.lex.curStartPos, c.lex.curEndPos) c.headerProps = defaultProperties c.transition(afterImplicitKey) c.pushLevel(emitCached) return true of Plain, SingleQuoted, DoubleQuoted: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties let headerEnd = c.lex.curStartPos c.safeNext() if c.lex.cur == Token.MapValueInd: if c.lex.lastScalarWasMultiline(): raise c.generateError("Implicit mapping key may not be multiline") c.keyCache.add(move(e)) e = startMapEvent(csBlock, c.headerProps, c.headerStart, headerEnd) c.headerProps = defaultProperties c.transition(afterImplicitKey) c.pushLevel(emitCached) else: c.mergeProps(c.headerProps, e.scalarProperties) c.popLevel() return true of MapStart, SeqStart: let startPos = c.lex.curStartPos indent = c.lex.currentIndentation() levelDepth = c.levels.len c.transition(beforeFlowItemProps) c.caching = true while c.levels.len >= levelDepth: c.keyCache.add(c.next()) c.caching = false if c.lex.cur == Token.MapValueInd: c.pushLevel(afterImplicitKey, indent) c.pushLevel(emitCached) if c.lex.curStartPos.line != startPos.line: raise c.generateError("Implicit mapping key may not be multiline") e = startMapEvent(csBlock, c.headerProps, c.headerStart, startPos) c.headerProps = defaultProperties return true else: c.pushLevel(emitCached) return false of Literal, Folded: c.mergeProps(c.inlineProps, c.headerProps) e = scalarEvent(c.lex.evaluated, c.headerProps, toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.headerProps = defaultProperties c.safeNext() c.popLevel() return true of Indentation: c.safeNext() c.transition(atBlockIndentation) return false of StreamEnd, DocumentEnd, DirectivesEnd: e = scalarEvent("", c.inlineProps, ssPlain, c.inlineStart, c.lex.curStartPos) c.inlineProps = defaultProperties c.popLevel() return true else: raise c.generateError("Unexpected token (expected block content): " & $c.lex.cur) proc beforeNodeProperties(c: Context, e: var Event): bool = case c.lex.cur of TagHandle: if c.inlineProps.tag != yTagQuestionMark: raise c.generateError("Only one tag allowed per node") c.inlineProps.tag = c.parseTag() of VerbatimTag: if c.inlineProps.tag != yTagQuestionMark: raise c.generateError("Only one tag allowed per node") c.inlineProps.tag = Tag(move(c.lex.evaluated)) of Token.Anchor: if c.inlineProps.anchor != yAnchorNone: raise c.generateError("Only one anchor allowed per node") c.inlineProps.anchor = c.lex.shortLexeme().Anchor of Indentation: c.mergeProps(c.inlineProps, c.headerProps) c.popLevel() return false of Alias: raise c.generateError("Alias may not have node properties") else: c.popLevel() return false c.safeNext() return false proc afterCompactParent(c: Context, e: var Event): bool = c.inlineStart = c.lex.curStartPos case c.lex.cur of nodePropertyKind: c.transition(afterCompactParentProps) c.pushLevel(beforeNodeProperties) of SeqItemInd: e = startSeqEvent(csBlock, c.headerProps, c.headerStart, c.lex.curEndPos) c.headerProps = defaultProperties c.transition(inBlockSeq, c.lex.recentIndentation()) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.lex.recentIndentation()) c.safeNext() return true of MapKeyInd: e = startMapEvent(csBlock, c.headerProps, c.headerStart, c.lex.curEndPos) c.headerProps = defaultProperties c.transition(beforeBlockMapValue, c.lex.recentIndentation()) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.lex.recentIndentation) c.safeNext() return true else: c.transition(afterCompactParentProps) return false proc afterCompactParentProps(c: Context, e: var Event): bool = c.updateIndentation(c.lex.recentIndentation()) case c.lex.cur of nodePropertyKind: c.pushLevel(beforeNodeProperties) return false of Indentation: c.headerStart = c.inlineStart c.transition(atBlockIndentation, c.levels[^3].indentation) c.pushLevel(beforeBlockIndentation) return false of MapValueInd: c.keyCache.add(scalarEvent("", c.inlineProps, ssPlain, c.inlineStart, c.lex.curStartPos)) c.inlineProps = defaultProperties e = startMapEvent(csBlock, defaultProperties, c.lex.curStartPos, c.lex.curStartPos) c.transition(afterImplicitKey) c.pushLevel(emitCached) return true of Alias: e = aliasEvent(c.lex.shortLexeme().Anchor, c.inlineStart, c.lex.curEndPos) let headerEnd = c.lex.curStartPos c.safeNext() if c.lex.cur == Token.MapValueInd: c.keyCache.add(move(e)) e = startMapEvent(csBlock, defaultProperties, headerEnd, headerEnd) c.transition(afterImplicitKey) c.pushLevel(emitCached) else: c.popLevel() return true of scalarTokenKind: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties let headerEnd = c.lex.curStartPos c.updateIndentation(c.lex.recentIndentation()) c.safeNext() if c.lex.cur == Token.MapValueInd: if c.lex.lastScalarWasMultiline(): raise c.generateError("Implicit mapping key may not be multiline") c.keyCache.add(move(e)) e = startMapEvent(csBlock, defaultProperties, headerEnd, headerEnd) c.transition(afterImplicitKey) c.pushLevel(emitCached) else: c.popLevel() return true of MapStart, SeqStart, StreamEnd, DocumentEnd, DirectivesEnd: c.transition(atBlockIndentationProps) return false else: raise c.generateError("Unexpected token (expected newline or flow item start: " & $c.lex.cur) proc afterBlockParent(c: Context, e: var Event): bool = c.inlineStart = c.lex.curStartPos case c.lex.cur of nodePropertyKind: c.transition(afterBlockParentProps) c.pushLevel(beforeNodeProperties) of SeqItemInd, MapKeyInd: raise c.generateError("Compact notation not allowed after implicit key") else: c.transition(afterBlockParentProps) return false proc afterBlockParentProps(c: Context, e: var Event): bool = c.updateIndentation(c.lex.recentIndentation()) case c.lex.cur of nodePropertyKind: c.pushLevel(beforeNodeProperties) return false of MapValueInd: raise c.generateError("Compact notation not allowed after implicit key") of scalarTokenKind: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties c.safeNext() if c.lex.cur == Token.MapValueInd: raise c.generateError("Compact notation not allowed after implicit key") c.popLevel() return true else: c.transition(afterCompactParentProps) return false proc mergePropsOnNewline(c: Context, e: var Event): bool = c.updateIndentation(c.lex.recentIndentation()) if c.lex.cur == Token.Indentation: c.mergeProps(c.inlineProps, c.headerProps) c.transition(afterCompactParentProps) return false proc beforeDocEnd(c: Context, e: var Event): bool = case c.lex.cur of DocumentEnd: e = endDocEvent(true, c.lex.curStartPos, c.lex.curEndPos) c.transition(beforeDoc) c.safeNext() resetHandles(c.handles) of StreamEnd: e = endDocEvent(false, c.lex.curStartPos, c.lex.curEndPos) c.popLevel() of DirectivesEnd: e = endDocEvent(false, c.lex.curStartPos, c.lex.curStartPos) c.transition(beforeDoc) resetHandles(c.handles) else: raise c.generateError("Unexpected token (expected document end): " & $c.lex.cur) return true proc inBlockSeq(c: Context, e: var Event): bool = if c.blockIndentation > c.levels[^1].indentation: raise c.generateError("Invalid indentation: got " & $c.blockIndentation & ", expected " & $c.levels[^1].indentation) case c.lex.cur of SeqItemInd: c.safeNext() c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.blockIndentation) return false else: if c.levels[^3].indentation == c.levels[^1].indentation: e = endSeqEvent(c.lex.curStartPos, c.lex.curEndPos) c.popLevel() c.popLevel() return true else: raise c.generateError("Illegal token (expected block sequence indicator): " & $c.lex.cur) proc beforeBlockMapKey(c: Context, e: var Event): bool = if c.blockIndentation > c.levels[^1].indentation: raise c.generateError("Invalid indentation: got " & $c.blockIndentation & ", expected " & $c.levels[^1].indentation) c.inlineStart = c.lex.curStartPos case c.lex.cur of MapKeyInd: c.transition(beforeBlockMapValue) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.blockIndentation) c.safeNext() return false of nodePropertyKind: c.transition(atBlockMapKeyProps) c.pushLevel(beforeNodeProperties) return false of Plain, SingleQuoted, DoubleQuoted: c.transition(atBlockMapKeyProps) return false of Alias: e = aliasEvent(c.lex.shortLexeme().Anchor, c.inlineStart, c.lex.curEndPos) c.safeNext() c.transition(afterImplicitKey) return true of MapValueInd: e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curEndPos) c.transition(beforeBlockMapValue) return true else: raise c.generateError("Unexpected token (expected mapping key): " & $c.lex.cur) proc atBlockMapKeyProps(c: Context, e: var Event): bool = case c.lex.cur of nodePropertyKind: c.pushLevel(beforeNodeProperties) of Alias: e = aliasEvent(c.lex.shortLexeme().Anchor, c.inlineStart, c.lex.curEndPos) of Plain, SingleQuoted, DoubleQuoted: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties if c.lex.lastScalarWasMultiline(): raise c.generateError("Implicit mapping key may not be multiline") of MapValueInd: e = scalarEvent("", c.inlineProps, ssPlain, c.inlineStart, c.lex.curStartPos) c.inlineProps = defaultProperties c.transition(afterImplicitKey) return true else: raise c.generateError("Unexpected token (expected implicit mapping key): " & $c.lex.cur) c.safeNext() c.transition(afterImplicitKey) return true proc afterImplicitKey(c: Context, e: var Event): bool = if c.lex.cur != Token.MapValueInd: raise c.generateError("Unexpected token (expected ':'): " & $c.lex.cur) c.safeNext() c.transition(beforeBlockMapKey) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterBlockParent, max(0, c.levels[^2].indentation)) return false proc beforeBlockMapValue(c: Context, e: var Event): bool = if c.blockIndentation > c.levels[^1].indentation: raise c.generateError("Invalid indentation") case c.lex.cur of MapValueInd: c.transition(beforeBlockMapKey) c.pushLevel(beforeBlockIndentation) c.pushLevel(afterCompactParent, c.blockIndentation) c.safeNext() of MapKeyInd, Plain, SingleQuoted, DoubleQuoted, nodePropertyKind: # the value is allowed to be missing after an explicit key e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curEndPos) c.transition(beforeBlockMapKey) return true else: raise c.generateError("Unexpected token (expected mapping value): " & $c.lex.cur) proc beforeBlockIndentation(c: Context, e: var Event): bool = proc endBlockNode(e: var Event) = if c.levels[^1].state == beforeBlockMapKey: e = endMapEvent(c.lex.curStartPos, c.lex.curEndPos) elif c.levels[^1].state == beforeBlockMapValue: e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curEndPos) c.transition(beforeBlockMapKey) c.pushLevel(beforeBlockIndentation) return elif c.levels[^1].state == inBlockSeq: e = endSeqEvent(c.lex.curStartPos, c.lex.curEndPos) elif c.levels[^1].state == atBlockIndentation: e = scalarEvent("", c.headerProps, ssPlain, c.headerStart, c.headerStart) c.headerProps = defaultProperties elif c.levels[^1].state == beforeBlockIndentation: raise c.generateError("Unexpected double beforeBlockIndentation") else: raise c.generateError("Internal error (please report this bug): unexpected state at endBlockNode") c.popLevel() c.popLevel() case c.lex.cur of Indentation: c.blockIndentation = c.lex.currentIndentation() if c.blockIndentation < c.levels[^1].indentation: endBlockNode(e) return true else: c.safeNext() return false of StreamEnd, DocumentEnd, DirectivesEnd: c.blockIndentation = 0 if c.levels[^1].state != beforeDocEnd: endBlockNode(e) return true else: return false else: raise c.generateError("Unexpected content after node in block context (expected newline): " & $c.lex.cur) proc beforeFlowItem(c: Context, e: var Event): bool = c.inlineStart = c.lex.curStartPos case c.lex.cur of nodePropertyKind: c.transition(beforeFlowItemProps) c.pushLevel(beforeNodeProperties) of Alias: e = aliasEvent(c.lex.shortLexeme().Anchor, c.inlineStart, c.lex.curEndPos) c.safeNext() c.popLevel() return true else: c.transition(beforeFlowItemProps) return false proc beforeFlowItemProps(c: Context, e: var Event): bool = case c.lex.cur of nodePropertyKind: c.pushLevel(beforeNodeProperties) of Alias: e = aliasEvent(c.lex.shortLexeme().Anchor, c.inlineStart, c.lex.curEndPos) c.safeNext() c.popLevel() of scalarTokenKind: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties c.safeNext() c.popLevel() of MapStart: e = startMapEvent(csFlow, c.inlineProps, c.inlineStart, c.lex.curEndPos) c.transition(afterFlowMapSep) c.safeNext() of SeqStart: e = startSeqEvent(csFlow, c.inlineProps, c.inlineStart, c.lex.curEndPos) c.transition(afterFlowSeqSep) c.safeNext() of MapEnd, SeqEnd, SeqSep, MapValueInd: e = scalarEvent("", c.inlineProps, ssPlain, c.inlineStart, c.lex.curEndPos) c.popLevel() else: raise c.generateError("Unexpected token (expected flow node): " & $c.lex.cur) c.inlineProps = defaultProperties return true proc afterFlowMapKey(c: Context, e: var Event): bool = case c.lex.cur of MapValueInd: c.transition(afterFlowMapValue) c.pushLevel(beforeFlowItem) c.safeNext() return false of SeqSep, MapEnd: e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curEndPos) c.transition(afterFlowMapValue) return true else: raise c.generateError("Unexpected token (expected ':'): " & $c.lex.cur) proc afterFlowMapValue(c: Context, e: var Event): bool = case c.lex.cur of SeqSep: c.transition(afterFlowMapSep) c.safeNext() return false of MapEnd: e = endMapEvent(c.lex.curStartPos, c.lex.curEndPos) c.safeNext() c.popLevel() return true of Plain, SingleQuoted, DoubleQuoted, MapKeyInd, Token.Anchor, Alias, MapStart, SeqStart: raise c.generateError("Missing ','") else: raise c.generateError("Unexpected token (expected ',' or '}'): " & $c.lex.cur) proc afterFlowSeqItem(c: Context, e: var Event): bool = case c.lex.cur of SeqSep: c.transition(afterFlowSeqSep) c.safeNext() return false of SeqEnd: e = endSeqEvent(c.lex.curStartPos, c.lex.curEndPos) c.safeNext() c.popLevel() return true of Plain, SingleQuoted, DoubleQuoted, MapKeyInd, Token.Anchor, Alias, MapStart, SeqStart: raise c.generateError("Missing ','") else: raise c.generateError("Unexpected token (expected ',' or ']'): " & $c.lex.cur) proc afterFlowMapSep(c: Context, e: var Event): bool = case c.lex.cur of MapKeyInd: c.safeNext() of MapEnd: e = endMapEvent(c.lex.curStartPos, c.lex.curEndPos) c.safeNext() c.popLevel() return true of SeqSep: raise c.generateError("Missing mapping entry between commas (use '?' for an empty mapping entry)") else: discard c.transition(afterFlowMapKey) c.pushLevel(beforeFlowItem) return false proc afterFlowSeqSep(c: Context, e: var Event): bool = c.inlineStart = c.lex.curStartPos case c.lex.cur of SeqSep: e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curStartPos) c.safeNext() return true of nodePropertyKind: c.transition(afterFlowSeqSepProps) c.pushLevel(beforeNodeProperties) return false of Plain, SingleQuoted, DoubleQuoted, MapStart, SeqStart: c.transition(afterFlowSeqSepProps) return false of MapKeyInd: c.transition(afterFlowSeqSepProps) e = startMapEvent(csFlow, defaultProperties, c.lex.curStartPos, c.lex.curEndPos) c.safeNext() c.transition(afterFlowSeqItem) c.pushLevel(beforePairValue) c.pushLevel(beforeFlowItem) return true of MapValueInd: c.transition(afterFlowSeqItem) e = startMapEvent(csFlow, defaultProperties, c.lex.curStartPos, c.lex.curEndPos) c.pushLevel(atEmptyPairKey) return true of SeqEnd: e = endSeqEvent(c.lex.curStartPos, c.lex.curEndPos) c.safeNext() c.popLevel() return true else: c.transition(afterFlowSeqItem) c.pushLevel(beforeFlowItem) return false proc afterFlowSeqSepProps(c: Context, e: var Event): bool = # here we handle potential implicit single pairs within flow sequences. c.transition(afterFlowSeqItem) case c.lex.cur of Plain, SingleQuoted, DoubleQuoted: e = scalarEvent(c.lex.evaluated, autoScalarTag(c.inlineProps, c.lex.cur), toStyle(c.lex.cur), c.inlineStart, c.lex.curEndPos) c.inlineProps = defaultProperties c.safeNext() if c.lex.cur == Token.MapValueInd: c.pushLevel(afterImplicitPairStart) if c.caching: c.keyCache.add(startMapEvent(csFlow, defaultProperties, c.lex.curStartPos, c.lex.curStartPos)) else: c.keyCache.add(move(e)) e = startMapEvent(csFlow, defaultProperties, c.lex.curStartPos, c.lex.curStartPos) c.pushLevel(emitCached) return true of MapStart, SeqStart: let startPos = c.lex.curStartPos indent = c.levels[^1].indentation cacheStart = c.keyCache.len levelDepth = c.levels.len alreadyCaching = c.caching c.pushLevel(beforeFlowItemProps) c.caching = true while c.levels.len > levelDepth: c.keyCache.add(c.next()) c.caching = alreadyCaching if c.lex.cur == Token.MapValueInd: c.pushLevel(afterImplicitPairStart, indent) if c.lex.curStartPos.line != startPos.line: raise c.generateError("Implicit mapping key may not be multiline") if not alreadyCaching: c.pushLevel(emitCached) e = startMapEvent(csPair, defaultProperties, startPos, startPos) return true else: # we are already filling a cache. # so we just squeeze the map start in. c.keyCache.insert(startMapEvent(csPair, defaultProperties, startPos, startPos), cacheStart) return false else: if not alreadyCaching: c.pushLevel(emitCached) return false else: c.pushLevel(beforeFlowItem) return false proc atEmptyPairKey(c: Context, e: var Event): bool = c.transition(beforePairValue) e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curStartPos) return true proc beforePairValue(c: Context, e: var Event): bool = if c.lex.cur == Token.MapValueInd: c.transition(afterPairValue) c.pushLevel(beforeFlowItem) c.safeNext() return false else: # pair ends here without value e = scalarEvent("", defaultProperties, ssPlain, c.lex.curStartPos, c.lex.curEndPos) c.popLevel() return true proc afterImplicitPairStart(c: Context, e: var Event): bool = c.safeNext() c.transition(afterPairValue) c.pushLevel(beforeFlowItem) return false proc afterPairValue(c: Context, e: var Event): bool = e = endMapEvent(c.lex.curStartPos, c.lex.curEndPos) c.popLevel() return true proc emitCached(c: Context, e: var Event): bool = debug("emitCollection key: pos = " & $c.keyCachePos & ", len = " & $c.keyCache.len) yAssert(c.keyCachePos < c.keyCache.len) e = move(c.keyCache[c.keyCachePos]) inc(c.keyCachePos) if c.keyCachePos == len(c.keyCache): c.keyCache.setLen(0) c.keyCachePos = 0 c.popLevel() return true proc display*(p: YamlParser, event: Event): string = ## Generate a representation of the given event with proper visualization of ## anchor and tag (if any). The generated representation is conformant to the ## format used in the yaml test suite. ## ## This proc is an informed version of ``$`` on ``YamlStreamEvent`` which can ## properly display the anchor and tag name as it occurs in the input. ## However, it shall only be used while using the streaming API because after ## finishing the parsing of a document, the parser drops all information about ## anchor and tag names. case event.kind of yamlStartStream: result = "+STR" of yamlEndStream: result = "-STR" of yamlEndMap: result = "-MAP" of yamlEndSeq: result = "-SEQ" of yamlStartDoc: result = "+DOC" if event.explicitDirectivesEnd: result &= " ---" of yamlEndDoc: result = "-DOC" if event.explicitDocumentEnd: result &= " ..." of yamlStartMap: result = "+MAP" & renderAttrs(event.mapProperties, true) of yamlStartSeq: result = "+SEQ" & renderAttrs(event.seqProperties, true) of yamlScalar: result = "=VAL" & renderAttrs(event.scalarProperties, event.scalarStyle in {ssPlain, ssFolded, ssLiteral}) case event.scalarStyle of ssPlain, ssAny: result &= " :" of ssSingleQuoted: result &= " \'" of ssDoubleQuoted: result &= " \"" of ssLiteral: result &= " |" of ssFolded: result &= " >" result &= yamlTestSuiteEscape(event.scalarContent) of yamlAlias: result = "=ALI *" & $event.aliasTarget