| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | package parser | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2020-07-02 17:25:31 +09:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2024-02-26 01:11:21 -05:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2019-10-23 03:21:42 +09:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	"github.com/goccy/go-yaml/ast" | 
					
						
							| 
									
										
										
										
											2019-10-24 11:07:40 +09:00
										 |  |  | 	"github.com/goccy/go-yaml/internal/errors" | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	"github.com/goccy/go-yaml/lexer" | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	"github.com/goccy/go-yaml/token" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | type Mode uint | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	ParseComments Mode = 1 << iota // parse comments and add them to AST | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ParseBytes parse from byte slice, and returns ast.File | 
					
						
							|  |  |  | func ParseBytes(bytes []byte, mode Mode, opts ...Option) (*ast.File, error) { | 
					
						
							|  |  |  | 	tokens := lexer.Tokenize(string(bytes)) | 
					
						
							|  |  |  | 	f, err := Parse(tokens, mode, opts...) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return f, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse parse from token instances, and returns ast.File | 
					
						
							|  |  |  | func Parse(tokens token.Tokens, mode Mode, opts ...Option) (*ast.File, error) { | 
					
						
							|  |  |  | 	if tk := tokens.InvalidToken(); tk != nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax(tk.Error, tk) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	p, err := newParser(tokens, mode, opts) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f, err := p.parse(newContext()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return f, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parse parse from filename, and returns ast.File | 
					
						
							|  |  |  | func ParseFile(filename string, mode Mode, opts ...Option) (*ast.File, error) { | 
					
						
							|  |  |  | 	file, err := os.ReadFile(filename) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f, err := ParseBytes(file, mode, opts...) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	f.Name = filename | 
					
						
							|  |  |  | 	return f, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | type YAMLVersion string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	YAML10 YAMLVersion = "1.0" | 
					
						
							|  |  |  | 	YAML11 YAMLVersion = "1.1" | 
					
						
							|  |  |  | 	YAML12 YAMLVersion = "1.2" | 
					
						
							|  |  |  | 	YAML13 YAMLVersion = "1.3" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var yamlVersionMap = map[string]YAMLVersion{ | 
					
						
							|  |  |  | 	"1.0": YAML10, | 
					
						
							|  |  |  | 	"1.1": YAML11, | 
					
						
							|  |  |  | 	"1.2": YAML12, | 
					
						
							|  |  |  | 	"1.3": YAML13, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | type parser struct { | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 	tokens                []*Token | 
					
						
							|  |  |  | 	pathMap               map[string]ast.Node | 
					
						
							|  |  |  | 	yamlVersion           YAMLVersion | 
					
						
							|  |  |  | 	allowDuplicateMapKey  bool | 
					
						
							|  |  |  | 	secondaryTagDirective *ast.DirectiveNode | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func newParser(tokens token.Tokens, mode Mode, opts []Option) (*parser, error) { | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	filteredTokens := []*token.Token{} | 
					
						
							|  |  |  | 	if mode&ParseComments != 0 { | 
					
						
							|  |  |  | 		filteredTokens = tokens | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		for _, tk := range tokens { | 
					
						
							|  |  |  | 			if tk.Type == token.CommentType { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// keep prev/next reference between tokens containing comments | 
					
						
							|  |  |  | 			// https://github.com/goccy/go-yaml/issues/254 | 
					
						
							|  |  |  | 			filteredTokens = append(filteredTokens, tk) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	tks, err := createGroupedTokens(token.Tokens(filteredTokens)) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | 	p := &parser{ | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		tokens:  tks, | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | 		pathMap: make(map[string]ast.Node), | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | 	for _, opt := range opts { | 
					
						
							|  |  |  | 		opt(p) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return p, nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parse(ctx *context) (*ast.File, error) { | 
					
						
							|  |  |  | 	file := &ast.File{Docs: []*ast.DocumentNode{}} | 
					
						
							|  |  |  | 	for _, token := range p.tokens { | 
					
						
							|  |  |  | 		doc, err := p.parseDocument(ctx, token.Group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		file.Docs = append(file.Docs, doc) | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return file, nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseDocument(ctx *context, docGroup *TokenGroup) (*ast.DocumentNode, error) { | 
					
						
							|  |  |  | 	if len(docGroup.Tokens) == 0 { | 
					
						
							|  |  |  | 		return ast.Document(docGroup.RawToken(), nil), nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	p.pathMap = make(map[string]ast.Node) | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		tokens = docGroup.Tokens | 
					
						
							|  |  |  | 		start  *token.Token | 
					
						
							|  |  |  | 		end    *token.Token | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if docGroup.First().Type() == token.DocumentHeaderType { | 
					
						
							|  |  |  | 		start = docGroup.First().RawToken() | 
					
						
							|  |  |  | 		tokens = tokens[1:] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if docGroup.Last().Type() == token.DocumentEndType { | 
					
						
							|  |  |  | 		end = docGroup.Last().RawToken() | 
					
						
							|  |  |  | 		tokens = tokens[:len(tokens)-1] | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 		defer func() { | 
					
						
							|  |  |  | 			// clear yaml version value if DocumentEnd token (...) is specified. | 
					
						
							|  |  |  | 			p.yamlVersion = "" | 
					
						
							|  |  |  | 		}() | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if len(tokens) == 0 { | 
					
						
							|  |  |  | 		return ast.Document(docGroup.RawToken(), nil), nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	body, err := p.parseDocumentBody(ctx.withGroup(&TokenGroup{ | 
					
						
							|  |  |  | 		Type:   TokenGroupDocumentBody, | 
					
						
							|  |  |  | 		Tokens: tokens, | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	node := ast.Document(start, body) | 
					
						
							|  |  |  | 	node.End = end | 
					
						
							|  |  |  | 	return node, nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseDocumentBody(ctx *context) (ast.Node, error) { | 
					
						
							|  |  |  | 	node, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if ctx.next() { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("value is not allowed in this context", ctx.currentToken().RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return node, nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseToken(ctx *context, tk *Token) (ast.Node, error) { | 
					
						
							|  |  |  | 	switch tk.GroupType() { | 
					
						
							|  |  |  | 	case TokenGroupMapKey, TokenGroupMapKeyValue: | 
					
						
							|  |  |  | 		return p.parseMap(ctx) | 
					
						
							|  |  |  | 	case TokenGroupDirective: | 
					
						
							|  |  |  | 		node, err := p.parseDirective(ctx.withGroup(tk.Group), tk.Group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 	case TokenGroupDirectiveName: | 
					
						
							|  |  |  | 		node, err := p.parseDirectiveName(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	case TokenGroupAnchor: | 
					
						
							|  |  |  | 		node, err := p.parseAnchor(ctx.withGroup(tk.Group), tk.Group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							|  |  |  | 	case TokenGroupAnchorName: | 
					
						
							|  |  |  | 		anchor, err := p.parseAnchorName(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 		if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("could not find anchor value", tk.RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		value, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		anchor.Value = value | 
					
						
							|  |  |  | 		return anchor, nil | 
					
						
							|  |  |  | 	case TokenGroupAlias: | 
					
						
							|  |  |  | 		node, err := p.parseAlias(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							|  |  |  | 	case TokenGroupLiteral, TokenGroupFolded: | 
					
						
							|  |  |  | 		node, err := p.parseLiteral(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							|  |  |  | 	case TokenGroupScalarTag: | 
					
						
							|  |  |  | 		node, err := p.parseTag(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	switch tk.Type() { | 
					
						
							|  |  |  | 	case token.CommentType: | 
					
						
							|  |  |  | 		return p.parseComment(ctx) | 
					
						
							|  |  |  | 	case token.TagType: | 
					
						
							|  |  |  | 		return p.parseTag(ctx) | 
					
						
							|  |  |  | 	case token.MappingStartType: | 
					
						
							|  |  |  | 		return p.parseFlowMap(ctx) | 
					
						
							|  |  |  | 	case token.SequenceStartType: | 
					
						
							|  |  |  | 		return p.parseFlowSequence(ctx) | 
					
						
							|  |  |  | 	case token.SequenceEntryType: | 
					
						
							|  |  |  | 		return p.parseSequence(ctx) | 
					
						
							|  |  |  | 	case token.SequenceEndType: | 
					
						
							|  |  |  | 		// SequenceEndType is always validated in parseFlowSequence. | 
					
						
							|  |  |  | 		// Therefore, if this is found in other cases, it is treated as a syntax error. | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("could not find '[' character corresponding to ']'", tk.RawToken()) | 
					
						
							|  |  |  | 	case token.MappingEndType: | 
					
						
							|  |  |  | 		// MappingEndType is always validated in parseFlowMap. | 
					
						
							|  |  |  | 		// Therefore, if this is found in other cases, it is treated as a syntax error. | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("could not find '{' character corresponding to '}'", tk.RawToken()) | 
					
						
							|  |  |  | 	case token.MappingValueType: | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("found an invalid key for this map", tk.RawToken()) | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	node, err := p.parseScalarValue(ctx, tk) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() | 
					
						
							|  |  |  | 	return node, nil | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseScalarValue(ctx *context, tk *Token) (ast.ScalarNode, error) { | 
					
						
							|  |  |  | 	if tk.Group != nil { | 
					
						
							|  |  |  | 		switch tk.GroupType() { | 
					
						
							|  |  |  | 		case TokenGroupAnchor: | 
					
						
							|  |  |  | 			return p.parseAnchor(ctx.withGroup(tk.Group), tk.Group) | 
					
						
							|  |  |  | 		case TokenGroupAnchorName: | 
					
						
							|  |  |  | 			anchor, err := p.parseAnchorName(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 			if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 				return nil, errors.ErrSyntax("could not find anchor value", tk.RawToken()) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			value, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			anchor.Value = value | 
					
						
							|  |  |  | 			return anchor, nil | 
					
						
							|  |  |  | 		case TokenGroupAlias: | 
					
						
							|  |  |  | 			return p.parseAlias(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		case TokenGroupLiteral, TokenGroupFolded: | 
					
						
							|  |  |  | 			return p.parseLiteral(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		case TokenGroupScalarTag: | 
					
						
							|  |  |  | 			return p.parseTag(ctx.withGroup(tk.Group)) | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("unexpected scalar value", tk.RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	switch tk.Type() { | 
					
						
							|  |  |  | 	case token.MergeKeyType: | 
					
						
							|  |  |  | 		return newMergeKeyNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.NullType: | 
					
						
							|  |  |  | 		return newNullNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.BoolType: | 
					
						
							|  |  |  | 		return newBoolNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.IntegerType, token.BinaryIntegerType, token.OctetIntegerType, token.HexIntegerType: | 
					
						
							|  |  |  | 		return newIntegerNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.FloatType: | 
					
						
							|  |  |  | 		return newFloatNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.InfinityType: | 
					
						
							|  |  |  | 		return newInfinityNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.NanType: | 
					
						
							|  |  |  | 		return newNanNode(ctx, tk) | 
					
						
							|  |  |  | 	case token.StringType, token.SingleQuoteType, token.DoubleQuoteType: | 
					
						
							|  |  |  | 		return newStringNode(ctx, tk) | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | 	case token.TagType: | 
					
						
							|  |  |  | 		// this case applies when it is a scalar tag and its value does not exist. | 
					
						
							|  |  |  | 		// Examples of cases where the value does not exist include cases like `key: !!str,` or `!!str : value`. | 
					
						
							|  |  |  | 		return p.parseScalarTag(ctx) | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return nil, errors.ErrSyntax("unexpected scalar value type", tk.RawToken()) | 
					
						
							| 
									
										
										
										
											2024-10-31 18:26:08 +09:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseFlowMap(ctx *context) (*ast.MappingNode, error) { | 
					
						
							|  |  |  | 	node, err := newMappingNode(ctx, ctx.currentToken(), true) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ctx.goNext() // skip MappingStart token | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	isFirst := true | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	for ctx.next() { | 
					
						
							|  |  |  | 		tk := ctx.currentToken() | 
					
						
							|  |  |  | 		if tk.Type() == token.MappingEndType { | 
					
						
							|  |  |  | 			node.End = tk.RawToken() | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 			break | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if tk.Type() == token.CollectEntryType { | 
					
						
							|  |  |  | 			ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		} else if !isFirst { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			return nil, errors.ErrSyntax("',' or '}' must be specified", tk.RawToken()) | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		if tk := ctx.currentToken(); tk.Type() == token.MappingEndType { | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 			// this case is here: "{ elem, }". | 
					
						
							|  |  |  | 			// In this case, ignore the last element and break mapping parsing. | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			node.End = tk.RawToken() | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 			break | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		mapKeyTk := ctx.currentToken() | 
					
						
							|  |  |  | 		switch mapKeyTk.GroupType() { | 
					
						
							|  |  |  | 		case TokenGroupMapKeyValue: | 
					
						
							|  |  |  | 			value, err := p.parseMapKeyValue(ctx.withGroup(mapKeyTk.Group), mapKeyTk.Group) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			node.Values = append(node.Values, value) | 
					
						
							|  |  |  | 			ctx.goNext() | 
					
						
							|  |  |  | 		case TokenGroupMapKey: | 
					
						
							|  |  |  | 			key, err := p.parseMapKey(ctx.withGroup(mapKeyTk.Group), mapKeyTk.Group) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-11-26 22:41:11 +09:00
										 |  |  | 			ctx := ctx.withChild(p.mapKeyText(key)) | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			colonTk := mapKeyTk.Group.Last() | 
					
						
							|  |  |  | 			if p.isFlowMapDelim(ctx.nextToken()) { | 
					
						
							|  |  |  | 				value, err := newNullNode(ctx, ctx.insertNullToken(colonTk)) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return nil, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				mapValue, err := newMappingValueNode(ctx, colonTk, key, value) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return nil, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				node.Values = append(node.Values, mapValue) | 
					
						
							|  |  |  | 				ctx.goNext() | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 				if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 					return nil, errors.ErrSyntax("could not find map value", colonTk.RawToken()) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 				value, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return nil, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				mapValue, err := newMappingValueNode(ctx, colonTk, key, value) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return nil, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				node.Values = append(node.Values, mapValue) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			if !p.isFlowMapDelim(ctx.nextToken()) { | 
					
						
							|  |  |  | 				return nil, errors.ErrSyntax("could not find flow map content", mapKeyTk.RawToken()) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			key, err := p.parseScalarValue(ctx, mapKeyTk) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			value, err := newNullNode(ctx, ctx.insertNullToken(mapKeyTk)) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			mapValue, err := newMappingValueNode(ctx, mapKeyTk, key, value) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			node.Values = append(node.Values, mapValue) | 
					
						
							|  |  |  | 			ctx.goNext() | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		isFirst = false | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if node.End == nil { | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		return nil, errors.ErrSyntax("could not find flow mapping end token '}'", node.Start) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() // skip mapping end token. | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 	return node, nil | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) isFlowMapDelim(tk *Token) bool { | 
					
						
							|  |  |  | 	return tk.Type() == token.MappingEndType || tk.Type() == token.CollectEntryType | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-10-31 15:35:43 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseMap(ctx *context) (*ast.MappingNode, error) { | 
					
						
							|  |  |  | 	keyTk := ctx.currentToken() | 
					
						
							|  |  |  | 	if keyTk.Group == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected map key", keyTk.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	var keyValueNode *ast.MappingValueNode | 
					
						
							|  |  |  | 	if keyTk.GroupType() == TokenGroupMapKeyValue { | 
					
						
							|  |  |  | 		node, err := p.parseMapKeyValue(ctx.withGroup(keyTk.Group), keyTk.Group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2024-10-31 15:35:43 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		keyValueNode = node | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		if err := p.validateMapKeyValueNextToken(ctx, keyTk, ctx.currentToken()); err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		key, err := p.parseMapKey(ctx.withGroup(keyTk.Group), keyTk.Group) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		ctx.goNext() | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		valueTk := ctx.currentToken() | 
					
						
							|  |  |  | 		if keyTk.Line() == valueTk.Line() && valueTk.Type() == token.SequenceEntryType { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("block sequence entries are not allowed in this context", valueTk.RawToken()) | 
					
						
							| 
									
										
										
										
											2024-11-12 22:08:26 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		ctx := ctx.withChild(p.mapKeyText(key)) | 
					
						
							|  |  |  | 		value, err := p.parseMapValue(ctx, key, keyTk.Group.Last()) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2020-07-02 17:25:31 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		node, err := newMappingValueNode(ctx, keyTk.Group.Last(), key, value) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2024-11-13 00:40:31 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		keyValueNode = node | 
					
						
							| 
									
										
										
										
											2020-07-02 17:25:31 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	mapNode, err := newMappingNode(ctx, &Token{Token: keyValueNode.GetToken()}, false, keyValueNode) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	var tk *Token | 
					
						
							|  |  |  | 	if ctx.isComment() { | 
					
						
							|  |  |  | 		tk = ctx.nextNotCommentToken() | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		tk = ctx.currentToken() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for tk.Column() == keyTk.Column() { | 
					
						
							| 
									
										
										
										
											2024-11-30 14:47:52 +09:00
										 |  |  | 		typ := tk.Type() | 
					
						
							|  |  |  | 		if ctx.isFlow && typ == token.SequenceEndType { | 
					
						
							|  |  |  | 			// [ | 
					
						
							|  |  |  | 			// key: value | 
					
						
							|  |  |  | 			// ] <= | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		if !p.isMapToken(tk) { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("non-map value is specified", tk.RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cm := p.parseHeadComment(ctx) | 
					
						
							| 
									
										
										
										
											2024-11-30 14:47:52 +09:00
										 |  |  | 		if typ == token.MappingEndType { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			// a: { | 
					
						
							|  |  |  | 			//  b: c | 
					
						
							|  |  |  | 			// } <= | 
					
						
							|  |  |  | 			ctx.goNext() | 
					
						
							|  |  |  | 			break | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		node, err := p.parseMap(ctx) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2023-09-14 12:25:21 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		if len(node.Values) != 0 { | 
					
						
							|  |  |  | 			if err := setHeadComment(cm, node.Values[0]); err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							| 
									
										
										
										
											2023-09-14 12:25:21 +09:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		mapNode.Values = append(mapNode.Values, node.Values...) | 
					
						
							|  |  |  | 		if node.FootComment != nil { | 
					
						
							|  |  |  | 			mapNode.Values[len(mapNode.Values)-1].FootComment = node.FootComment | 
					
						
							| 
									
										
										
										
											2023-09-14 12:25:21 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		tk = ctx.currentToken() | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if ctx.isComment() { | 
					
						
							|  |  |  | 		if keyTk.Column() <= ctx.currentToken().Column() { | 
					
						
							|  |  |  | 			// If the comment is in the same or deeper column as the last element column in map value, | 
					
						
							|  |  |  | 			// treat it as a footer comment for the last element. | 
					
						
							|  |  |  | 			if len(mapNode.Values) == 1 { | 
					
						
							|  |  |  | 				mapNode.Values[0].FootComment = p.parseFootComment(ctx, keyTk.Column()) | 
					
						
							|  |  |  | 				mapNode.Values[0].FootComment.SetPath(mapNode.Values[0].Key.GetPath()) | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				mapNode.FootComment = p.parseFootComment(ctx, keyTk.Column()) | 
					
						
							|  |  |  | 				mapNode.FootComment.SetPath(mapNode.GetPath()) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-11-13 19:45:38 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-09-14 12:25:21 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return mapNode, nil | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) validateMapKeyValueNextToken(ctx *context, keyTk, tk *Token) error { | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 	if tk == nil { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if tk.Column() <= keyTk.Column() { | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if ctx.isFlow && tk.Type() == token.CollectEntryType { | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	// a: b | 
					
						
							|  |  |  | 	//  c <= this token is invalid. | 
					
						
							|  |  |  | 	return errors.ErrSyntax("value is not allowed in this context. map key-value is pre-defined", tk.RawToken()) | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) isMapToken(tk *Token) bool { | 
					
						
							|  |  |  | 	if tk.Group == nil { | 
					
						
							|  |  |  | 		return tk.Type() == token.MappingStartType || tk.Type() == token.MappingEndType | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	g := tk.Group | 
					
						
							|  |  |  | 	return g.Type == TokenGroupMapKey || g.Type == TokenGroupMapKeyValue | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseMapKeyValue(ctx *context, g *TokenGroup) (*ast.MappingValueNode, error) { | 
					
						
							|  |  |  | 	if g.Type != TokenGroupMapKeyValue { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected map key-value pair", g.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if g.First().Group == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected map key", g.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	keyGroup := g.First().Group | 
					
						
							|  |  |  | 	key, err := p.parseMapKey(ctx.withGroup(keyGroup), keyGroup) | 
					
						
							| 
									
										
										
										
											2020-07-02 17:25:31 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	value, err := p.parseToken(ctx.withChild(p.mapKeyText(key)), g.Last()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-23 03:21:42 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return newMappingValueNode(ctx, keyGroup.Last(), key, value) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *parser) parseMapKey(ctx *context, g *TokenGroup) (ast.MapKeyNode, error) { | 
					
						
							|  |  |  | 	if g.Type != TokenGroupMapKey { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected map key", g.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if g.First().Type() == token.MappingKeyType { | 
					
						
							|  |  |  | 		mapKeyTk := g.First() | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | 		if mapKeyTk.Group != nil { | 
					
						
							|  |  |  | 			ctx = ctx.withGroup(mapKeyTk.Group) | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		key, err := newMappingKeyNode(ctx, mapKeyTk) | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		ctx.goNext() // skip mapping key token | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 		if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("could not find value for mapping key", mapKeyTk.RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		scalar, err := p.parseScalarValue(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		key.Value = scalar | 
					
						
							|  |  |  | 		keyText := p.mapKeyText(scalar) | 
					
						
							|  |  |  | 		keyPath := ctx.withChild(keyText).path | 
					
						
							|  |  |  | 		key.SetPath(keyPath) | 
					
						
							|  |  |  | 		if err := p.validateMapKey(key.GetToken(), keyPath); err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		p.pathMap[keyPath] = key | 
					
						
							|  |  |  | 		return key, nil | 
					
						
							| 
									
										
										
										
											2020-05-29 15:32:25 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | 	if g.Last().Type() != token.MappingValueType { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("expected map key-value delimiter ':'", g.Last().RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	scalar, err := p.parseScalarValue(ctx, g.First()) | 
					
						
							| 
									
										
										
										
											2020-06-03 17:09:16 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-23 03:21:42 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	key, ok := scalar.(ast.MapKeyNode) | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("cannot take map-key node", scalar.GetToken()) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	keyText := p.mapKeyText(key) | 
					
						
							|  |  |  | 	keyPath := ctx.withChild(keyText).path | 
					
						
							|  |  |  | 	key.SetPath(keyPath) | 
					
						
							|  |  |  | 	if err := p.validateMapKey(key.GetToken(), keyPath); err != nil { | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2024-11-03 21:32:51 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	p.pathMap[keyPath] = key | 
					
						
							|  |  |  | 	return key, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *parser) validateMapKey(tk *token.Token, keyPath string) error { | 
					
						
							|  |  |  | 	if !p.allowDuplicateMapKey { | 
					
						
							|  |  |  | 		if n, exists := p.pathMap[keyPath]; exists { | 
					
						
							|  |  |  | 			pos := n.GetToken().Position | 
					
						
							|  |  |  | 			return errors.ErrSyntax( | 
					
						
							|  |  |  | 				fmt.Sprintf("mapping key %q already defined at [%d:%d]", tk.Value, pos.Line, pos.Column), | 
					
						
							|  |  |  | 				tk, | 
					
						
							|  |  |  | 			) | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-10-23 20:22:14 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if tk.Type != token.StringType { | 
					
						
							|  |  |  | 		return nil | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	origin := p.removeLeftSideNewLineCharacter(tk.Origin) | 
					
						
							|  |  |  | 	if p.existsNewLineCharacter(origin) { | 
					
						
							|  |  |  | 		return errors.ErrSyntax("unexpected key name", tk) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *parser) removeLeftSideNewLineCharacter(src string) string { | 
					
						
							|  |  |  | 	// CR or LF or CRLF | 
					
						
							|  |  |  | 	return strings.TrimLeft(strings.TrimLeft(strings.TrimLeft(src, "\r"), "\n"), "\r\n") | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) existsNewLineCharacter(src string) bool { | 
					
						
							|  |  |  | 	if strings.Index(src, "\n") > 0 { | 
					
						
							|  |  |  | 		return true | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if strings.Index(src, "\r") > 0 { | 
					
						
							|  |  |  | 		return true | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return false | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) mapKeyText(n ast.Node) string { | 
					
						
							|  |  |  | 	if n == nil { | 
					
						
							|  |  |  | 		return "" | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	switch nn := n.(type) { | 
					
						
							|  |  |  | 	case *ast.MappingKeyNode: | 
					
						
							|  |  |  | 		return p.mapKeyText(nn.Value) | 
					
						
							|  |  |  | 	case *ast.TagNode: | 
					
						
							|  |  |  | 		return p.mapKeyText(nn.Value) | 
					
						
							|  |  |  | 	case *ast.AnchorNode: | 
					
						
							|  |  |  | 		return p.mapKeyText(nn.Value) | 
					
						
							|  |  |  | 	case *ast.AliasNode: | 
					
						
							|  |  |  | 		return p.mapKeyText(nn.Value) | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return n.GetToken().Value | 
					
						
							| 
									
										
										
										
											2024-11-12 10:39:52 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseMapValue(ctx *context, key ast.MapKeyNode, colonTk *Token) (ast.Node, error) { | 
					
						
							|  |  |  | 	tk := ctx.currentToken() | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 	if tk == nil { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		return newNullNode(ctx, ctx.insertNullToken(colonTk)) | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if ctx.isComment() { | 
					
						
							|  |  |  | 		tk = ctx.nextNotCommentToken() | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 22:41:11 +09:00
										 |  |  | 	keyCol := key.GetToken().Position.Column | 
					
						
							|  |  |  | 	keyLine := key.GetToken().Position.Line | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-30 14:47:52 +09:00
										 |  |  | 	if tk.Column() != keyCol && tk.Line() == keyLine && (tk.GroupType() == TokenGroupMapKey || tk.GroupType() == TokenGroupMapKeyValue) { | 
					
						
							|  |  |  | 		// a: b: | 
					
						
							|  |  |  | 		//    ^ | 
					
						
							|  |  |  | 		// | 
					
						
							|  |  |  | 		// a: b: c | 
					
						
							|  |  |  | 		//    ^ | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("mapping value is not allowed in this context", tk.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 22:41:11 +09:00
										 |  |  | 	if tk.Column() == keyCol && p.isMapToken(tk) { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		// key: <value does not defined> | 
					
						
							|  |  |  | 		// next | 
					
						
							|  |  |  | 		return newNullNode(ctx, ctx.insertNullToken(colonTk)) | 
					
						
							| 
									
										
										
										
											2024-11-05 13:28:23 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 22:41:11 +09:00
										 |  |  | 	if tk.Line() == keyLine && tk.GroupType() == TokenGroupAnchorName && | 
					
						
							|  |  |  | 		ctx.nextToken().Column() == keyCol && p.isMapToken(ctx.nextToken()) { | 
					
						
							|  |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		// key: &anchor | 
					
						
							|  |  |  | 		// next | 
					
						
							|  |  |  | 		group := &TokenGroup{ | 
					
						
							|  |  |  | 			Type:   TokenGroupAnchor, | 
					
						
							|  |  |  | 			Tokens: []*Token{tk, ctx.createNullToken(tk)}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		anchor, err := p.parseAnchor(ctx.withGroup(group), group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return anchor, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-30 10:16:17 +09:00
										 |  |  | 	if tk.Column() <= keyCol && tk.GroupType() == TokenGroupAnchorName { | 
					
						
							|  |  |  | 		// key: <value does not defined> | 
					
						
							|  |  |  | 		// &anchor | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("anchor is not allowed in this context", tk.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 22:41:11 +09:00
										 |  |  | 	if tk.Column() < keyCol { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		//   key: <value does not defined> | 
					
						
							|  |  |  | 		// next | 
					
						
							|  |  |  | 		return newNullNode(ctx, ctx.insertNullToken(colonTk)) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 22:41:11 +09:00
										 |  |  | 	if tk.Line() == keyLine && tk.GroupType() == TokenGroupAnchorName && | 
					
						
							|  |  |  | 		ctx.nextToken().Column() < keyCol { | 
					
						
							|  |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		//   key: &anchor | 
					
						
							|  |  |  | 		// next | 
					
						
							|  |  |  | 		group := &TokenGroup{ | 
					
						
							|  |  |  | 			Type:   TokenGroupAnchor, | 
					
						
							|  |  |  | 			Tokens: []*Token{tk, ctx.createNullToken(tk)}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		anchor, err := p.parseAnchor(ctx.withGroup(group), group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return anchor, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	value, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return value, nil | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseAnchor(ctx *context, g *TokenGroup) (*ast.AnchorNode, error) { | 
					
						
							|  |  |  | 	anchorNameGroup := g.First().Group | 
					
						
							|  |  |  | 	anchor, err := p.parseAnchorName(ctx.withGroup(anchorNameGroup)) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 	if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("could not find anchor value", anchor.GetToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	value, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	anchor.Value = value | 
					
						
							|  |  |  | 	return anchor, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseAnchorName(ctx *context) (*ast.AnchorNode, error) { | 
					
						
							|  |  |  | 	anchor, err := newAnchorNode(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 	if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("could not find anchor value", anchor.GetToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	anchorName, err := p.parseScalarValue(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if anchorName == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected anchor. anchor name is not scalar value", ctx.currentToken().RawToken()) | 
					
						
							| 
									
										
										
										
											2019-10-30 16:57:59 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	anchor.Name = anchorName | 
					
						
							|  |  |  | 	return anchor, nil | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseAlias(ctx *context) (*ast.AliasNode, error) { | 
					
						
							|  |  |  | 	alias, err := newAliasNode(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2024-11-17 23:58:35 +09:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 	if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("could not find alias value", alias.GetToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	aliasName, err := p.parseScalarValue(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if aliasName == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected alias. alias name is not scalar value", ctx.currentToken().RawToken()) | 
					
						
							| 
									
										
										
										
											2020-09-09 20:31:42 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	alias.Value = aliasName | 
					
						
							|  |  |  | 	return alias, nil | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-18 13:26:00 +09:00
										 |  |  | func (p *parser) parseLiteral(ctx *context) (*ast.LiteralNode, error) { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	node, err := newLiteralNode(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ctx.goNext() // skip literal/folded token | 
					
						
							| 
									
										
										
										
											2021-07-19 18:48:09 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	tk := ctx.currentToken() | 
					
						
							| 
									
										
										
										
											2024-11-01 15:03:27 +09:00
										 |  |  | 	if tk == nil { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		value, err := newStringNode(ctx, &Token{Token: token.New("", "", node.Start.Position)}) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2021-07-19 18:48:09 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		node.Value = value | 
					
						
							|  |  |  | 		return node, nil | 
					
						
							| 
									
										
										
										
											2021-07-19 18:48:09 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	value, err := p.parseToken(ctx, tk) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	str, ok := value.(*ast.StringNode) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	if !ok { | 
					
						
							| 
									
										
										
										
											2019-10-23 13:30:22 +09:00
										 |  |  | 		return nil, errors.ErrSyntax("unexpected token. required string token", value.GetToken()) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	node.Value = str | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 	return node, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | func (p *parser) parseScalarTag(ctx *context) (*ast.TagNode, error) { | 
					
						
							|  |  |  | 	tag, err := p.parseTag(ctx) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if tag.Value == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("specified not scalar tag", tag.GetToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if _, ok := tag.Value.(ast.ScalarNode); !ok { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("specified not scalar tag", tag.GetToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return tag, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseTag(ctx *context) (*ast.TagNode, error) { | 
					
						
							|  |  |  | 	tagTk := ctx.currentToken() | 
					
						
							|  |  |  | 	tagRawTk := tagTk.RawToken() | 
					
						
							|  |  |  | 	node, err := newTagNode(ctx, tagTk) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2020-05-28 21:39:06 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() | 
					
						
							| 
									
										
										
										
											2020-05-28 21:39:06 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	comment := p.parseHeadComment(ctx) | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var tagValue ast.Node | 
					
						
							|  |  |  | 	if p.secondaryTagDirective != nil { | 
					
						
							|  |  |  | 		value, err := newStringNode(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		tagValue = value | 
					
						
							|  |  |  | 		node.Directive = p.secondaryTagDirective | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		value, err := p.parseTagValue(ctx, tagRawTk, ctx.currentToken()) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		tagValue = value | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 	if err := setHeadComment(comment, tagValue); err != nil { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 	node.Value = tagValue | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	return node, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseTagValue(ctx *context, tagRawTk *token.Token, tk *Token) (ast.Node, error) { | 
					
						
							|  |  |  | 	if tk == nil { | 
					
						
							|  |  |  | 		return newNullNode(ctx, ctx.createNullToken(&Token{Token: tagRawTk})) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	switch token.ReservedTagKeyword(tagRawTk.Value) { | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | 	case token.MappingTag, token.SetTag: | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		if !p.isMapToken(tk) { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("could not find map", tk.RawToken()) | 
					
						
							| 
									
										
										
										
											2020-05-29 15:32:25 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		if tk.Type() == token.MappingStartType { | 
					
						
							|  |  |  | 			return p.parseFlowMap(ctx) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return p.parseMap(ctx) | 
					
						
							|  |  |  | 	case token.IntegerTag, token.FloatTag, token.StringTag, token.BinaryTag, token.TimestampTag, token.BooleanTag, token.NullTag: | 
					
						
							|  |  |  | 		if tk.GroupType() == TokenGroupLiteral || tk.GroupType() == TokenGroupFolded { | 
					
						
							|  |  |  | 			return p.parseLiteral(ctx.withGroup(tk.Group)) | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | 		} else if tk.Type() == token.CollectEntryType || tk.Type() == token.MappingValueType { | 
					
						
							|  |  |  | 			return newTagDefaultScalarValueNode(ctx, tagRawTk) | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		scalar, err := p.parseScalarValue(ctx, tk) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2020-05-29 15:32:25 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return scalar, nil | 
					
						
							| 
									
										
										
										
											2024-11-28 23:34:45 +09:00
										 |  |  | 	case token.SequenceTag, token.OrderedMapTag: | 
					
						
							|  |  |  | 		if tk.Type() == token.SequenceStartType { | 
					
						
							|  |  |  | 			return p.parseFlowSequence(ctx) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return p.parseSequence(ctx) | 
					
						
							| 
									
										
										
										
											2020-05-29 15:32:25 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return p.parseToken(ctx, tk) | 
					
						
							| 
									
										
										
										
											2021-07-16 21:49:17 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseFlowSequence(ctx *context) (*ast.SequenceNode, error) { | 
					
						
							|  |  |  | 	node, err := newSequenceNode(ctx, ctx.currentToken(), true) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ctx.goNext() // skip SequenceStart token | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	isFirst := true | 
					
						
							|  |  |  | 	for ctx.next() { | 
					
						
							|  |  |  | 		tk := ctx.currentToken() | 
					
						
							|  |  |  | 		if tk.Type() == token.SequenceEndType { | 
					
						
							|  |  |  | 			node.End = tk.RawToken() | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if tk.Type() == token.CollectEntryType { | 
					
						
							|  |  |  | 			ctx.goNext() | 
					
						
							|  |  |  | 		} else if !isFirst { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("',' or ']' must be specified", tk.RawToken()) | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 		if tk := ctx.currentToken(); tk.Type() == token.SequenceEndType { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			// this case is here: "[ elem, ]". | 
					
						
							|  |  |  | 			// In this case, ignore the last element and break sequence parsing. | 
					
						
							|  |  |  | 			node.End = tk.RawToken() | 
					
						
							| 
									
										
										
										
											2023-03-01 16:59:07 +09:00
										 |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-27 10:48:36 +09:00
										 |  |  | 		if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		value, err := p.parseToken(ctx.withIndex(uint(len(node.Values))).withFlow(true), ctx.currentToken()) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		node.Values = append(node.Values, value) | 
					
						
							|  |  |  | 		isFirst = false | 
					
						
							| 
									
										
										
										
											2020-05-28 21:39:06 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if node.End == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("sequence end token ']' not found", node.Start) | 
					
						
							| 
									
										
										
										
											2020-05-28 21:39:06 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	ctx.goNext() // skip sequence end token. | 
					
						
							| 
									
										
										
										
											2020-05-28 21:39:06 +09:00
										 |  |  | 	return node, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseSequence(ctx *context) (*ast.SequenceNode, error) { | 
					
						
							|  |  |  | 	seqTk := ctx.currentToken() | 
					
						
							|  |  |  | 	seqNode, err := newSequenceNode(ctx, seqTk, false) | 
					
						
							| 
									
										
										
										
											2020-07-02 17:25:31 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2020-07-02 17:25:31 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	tk := seqTk | 
					
						
							|  |  |  | 	for tk.Type() == token.SequenceEntryType && tk.Column() == seqTk.Column() { | 
					
						
							|  |  |  | 		seqTk := tk | 
					
						
							|  |  |  | 		comment := p.parseHeadComment(ctx) | 
					
						
							|  |  |  | 		ctx.goNext() // skip sequence entry token | 
					
						
							| 
									
										
										
										
											2021-09-07 17:31:17 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-30 14:47:52 +09:00
										 |  |  | 		value, err := p.parseSequenceValue(ctx.withIndex(uint(len(seqNode.Values))), seqTk) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 			return nil, err | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		seqNode.ValueHeadComments = append(seqNode.ValueHeadComments, comment) | 
					
						
							|  |  |  | 		seqNode.Values = append(seqNode.Values, value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if ctx.isComment() { | 
					
						
							|  |  |  | 			tk = ctx.nextNotCommentToken() | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 			tk = ctx.currentToken() | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if ctx.isComment() { | 
					
						
							|  |  |  | 		if seqTk.Column() <= ctx.currentToken().Column() { | 
					
						
							|  |  |  | 			// If the comment is in the same or deeper column as the last element column in sequence value, | 
					
						
							|  |  |  | 			// treat it as a footer comment for the last element. | 
					
						
							|  |  |  | 			seqNode.FootComment = p.parseFootComment(ctx, seqTk.Column()) | 
					
						
							|  |  |  | 			if len(seqNode.Values) != 0 { | 
					
						
							|  |  |  | 				seqNode.FootComment.SetPath(seqNode.Values[len(seqNode.Values)-1].GetPath()) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2024-11-17 23:58:35 +09:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return seqNode, nil | 
					
						
							| 
									
										
										
										
											2024-11-17 23:58:35 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-30 14:47:52 +09:00
										 |  |  | func (p *parser) parseSequenceValue(ctx *context, seqTk *Token) (ast.Node, error) { | 
					
						
							|  |  |  | 	tk := ctx.currentToken() | 
					
						
							|  |  |  | 	if tk == nil { | 
					
						
							|  |  |  | 		return newNullNode(ctx, ctx.insertNullToken(seqTk)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ctx.isComment() { | 
					
						
							|  |  |  | 		tk = ctx.nextNotCommentToken() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	seqCol := seqTk.Column() | 
					
						
							|  |  |  | 	seqLine := seqTk.Line() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if tk.Column() == seqCol && tk.Type() == token.SequenceEntryType { | 
					
						
							|  |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		// - <value does not defined> | 
					
						
							|  |  |  | 		// - | 
					
						
							|  |  |  | 		return newNullNode(ctx, ctx.insertNullToken(seqTk)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if tk.Line() == seqLine && tk.GroupType() == TokenGroupAnchorName && | 
					
						
							|  |  |  | 		ctx.nextToken().Column() == seqCol && ctx.nextToken().Type() == token.SequenceEntryType { | 
					
						
							|  |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		// - &anchor | 
					
						
							|  |  |  | 		// - | 
					
						
							|  |  |  | 		group := &TokenGroup{ | 
					
						
							|  |  |  | 			Type:   TokenGroupAnchor, | 
					
						
							|  |  |  | 			Tokens: []*Token{tk, ctx.createNullToken(tk)}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		anchor, err := p.parseAnchor(ctx.withGroup(group), group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return anchor, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if tk.Column() <= seqCol && tk.GroupType() == TokenGroupAnchorName { | 
					
						
							|  |  |  | 		// - <value does not defined> | 
					
						
							|  |  |  | 		// &anchor | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("anchor is not allowed in this sequence context", tk.RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if tk.Column() < seqCol { | 
					
						
							|  |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		//   - <value does not defined> | 
					
						
							|  |  |  | 		// next | 
					
						
							|  |  |  | 		return newNullNode(ctx, ctx.insertNullToken(seqTk)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if tk.Line() == seqLine && tk.GroupType() == TokenGroupAnchorName && | 
					
						
							|  |  |  | 		ctx.nextToken().Column() < seqCol { | 
					
						
							|  |  |  | 		// in this case, | 
					
						
							|  |  |  | 		// ---- | 
					
						
							|  |  |  | 		//   - &anchor | 
					
						
							|  |  |  | 		// next | 
					
						
							|  |  |  | 		group := &TokenGroup{ | 
					
						
							|  |  |  | 			Type:   TokenGroupAnchor, | 
					
						
							|  |  |  | 			Tokens: []*Token{tk, ctx.createNullToken(tk)}, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		anchor, err := p.parseAnchor(ctx.withGroup(group), group) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 		return anchor, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	value, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return value, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseDirective(ctx *context, g *TokenGroup) (*ast.DirectiveNode, error) { | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 	directiveNameGroup := g.First().Group | 
					
						
							|  |  |  | 	directive, err := p.parseDirectiveName(ctx.withGroup(directiveNameGroup)) | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	switch directive.Name.String() { | 
					
						
							|  |  |  | 	case "YAML": | 
					
						
							|  |  |  | 		if len(g.Tokens) != 2 { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("unexpected format YAML directive", g.First().RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		valueTk := g.Tokens[1] | 
					
						
							|  |  |  | 		valueRawTk := valueTk.RawToken() | 
					
						
							|  |  |  | 		value := valueRawTk.Value | 
					
						
							|  |  |  | 		ver, exists := yamlVersionMap[value] | 
					
						
							|  |  |  | 		if !exists { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax(fmt.Sprintf("unknown YAML version %q", value), valueRawTk) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if p.yamlVersion != "" { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("YAML version has already been specified", valueRawTk) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		p.yamlVersion = ver | 
					
						
							|  |  |  | 		versionNode, err := newStringNode(ctx, valueTk) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		directive.Values = append(directive.Values, versionNode) | 
					
						
							|  |  |  | 	case "TAG": | 
					
						
							|  |  |  | 		if len(g.Tokens) != 3 { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("unexpected format TAG directive", g.First().RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		tagKey, err := newStringNode(ctx, g.Tokens[1]) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if tagKey.Value == "!!" { | 
					
						
							|  |  |  | 			p.secondaryTagDirective = directive | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		tagValue, err := newStringNode(ctx, g.Tokens[2]) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		directive.Values = append(directive.Values, tagKey, tagValue) | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		if len(g.Tokens) > 1 { | 
					
						
							|  |  |  | 			for _, tk := range g.Tokens[1:] { | 
					
						
							|  |  |  | 				value, err := newStringNode(ctx, tk) | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					return nil, err | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				directive.Values = append(directive.Values, value) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return directive, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *parser) parseDirectiveName(ctx *context) (*ast.DirectiveNode, error) { | 
					
						
							|  |  |  | 	directive, err := newDirectiveNode(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-12-09 18:45:26 +09:00
										 |  |  | 	ctx.goNext() | 
					
						
							|  |  |  | 	if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("could not find directive value", directive.GetToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	directiveName, err := p.parseScalarValue(ctx, ctx.currentToken()) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if directiveName == nil { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("unexpected directive. directive name is not scalar value", ctx.currentToken().RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	directive.Name = directiveName | 
					
						
							|  |  |  | 	return directive, nil | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseComment(ctx *context) (ast.Node, error) { | 
					
						
							|  |  |  | 	cm := p.parseHeadComment(ctx) | 
					
						
							| 
									
										
										
										
											2024-12-02 14:45:19 +00:00
										 |  |  | 	if ctx.isTokenNotFound() { | 
					
						
							|  |  |  | 		return cm, nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	node, err := p.parseToken(ctx, ctx.currentToken()) | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if err := setHeadComment(cm, node); err != nil { | 
					
						
							| 
									
										
										
										
											2024-10-28 21:24:15 +09:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	return node, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *parser) parseHeadComment(ctx *context) *ast.CommentGroupNode { | 
					
						
							|  |  |  | 	tks := []*token.Token{} | 
					
						
							|  |  |  | 	for ctx.isComment() { | 
					
						
							|  |  |  | 		tks = append(tks, ctx.currentToken().RawToken()) | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(tks) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ast.CommentGroup(tks) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (p *parser) parseFootComment(ctx *context, col int) *ast.CommentGroupNode { | 
					
						
							|  |  |  | 	tks := []*token.Token{} | 
					
						
							|  |  |  | 	for ctx.isComment() && col <= ctx.currentToken().Column() { | 
					
						
							|  |  |  | 		tks = append(tks, ctx.currentToken().RawToken()) | 
					
						
							|  |  |  | 		ctx.goNext() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(tks) == 0 { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ast.CommentGroup(tks) | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | } |