| 
									
										
										
										
											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-10-31 18:26:08 +09:00
										 |  |  | type parser struct { | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	tokens               []*Token | 
					
						
							| 
									
										
										
										
											2024-11-13 18:29:20 +09:00
										 |  |  | 	pathMap              map[string]ast.Node | 
					
						
							|  |  |  | 	allowDuplicateMapKey bool | 
					
						
							| 
									
										
										
										
											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-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 | 
					
						
							|  |  |  | 	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() | 
					
						
							|  |  |  | 		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() | 
					
						
							|  |  |  | 			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) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	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() | 
					
						
							|  |  |  | 				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() { | 
					
						
							|  |  |  | 		if !p.isMapToken(tk) { | 
					
						
							|  |  |  | 			return nil, errors.ErrSyntax("non-map value is specified", tk.RawToken()) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cm := p.parseHeadComment(ctx) | 
					
						
							|  |  |  | 		if tk.Type() == token.MappingEndType { | 
					
						
							|  |  |  | 			// 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 { | 
					
						
							|  |  |  | 	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.Last().Type() != token.MappingValueType { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax("expected map key-value delimiter ':'", g.Last().RawToken()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if g.First().Type() == token.MappingKeyType { | 
					
						
							|  |  |  | 		mapKeyTk := g.First() | 
					
						
							|  |  |  | 		ctx := ctx.withGroup(mapKeyTk.Group) | 
					
						
							|  |  |  | 		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 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		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
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											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-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 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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() | 
					
						
							|  |  |  | 	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() | 
					
						
							|  |  |  | 	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() | 
					
						
							| 
									
										
										
										
											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-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) | 
					
						
							|  |  |  | 	value, err := p.parseTagValue(ctx, tagRawTk, 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(comment, value); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2019-11-05 17:02:55 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	node.Value = value | 
					
						
							| 
									
										
										
										
											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) { | 
					
						
							|  |  |  | 	case token.MappingTag, token.OrderedMapTag: | 
					
						
							|  |  |  | 		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)) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		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 | 
					
						
							|  |  |  | 	case token.SequenceTag, token.SetTag: | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax(fmt.Sprintf("sorry, currently not supported %s tag", tagRawTk.Value), tagRawTk) | 
					
						
							| 
									
										
										
										
											2020-05-29 15:32:25 +09:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 	if strings.HasPrefix(tagRawTk.Value, "!!") { | 
					
						
							|  |  |  | 		return nil, errors.ErrSyntax(fmt.Sprintf("unknown secondary tag name %q specified", tagRawTk.Value), tagRawTk) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	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
										 |  |  | 
 | 
					
						
							|  |  |  | 		if tk := ctx.currentToken(); tk != nil && tk.Type() == token.SequenceEndType { | 
					
						
							|  |  |  | 			// 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-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-26 11:05:45 +09:00
										 |  |  | 		valueTk := ctx.currentToken() | 
					
						
							|  |  |  | 		if valueTk == nil { | 
					
						
							|  |  |  | 			node, err := newNullNode(ctx, ctx.createNullToken(seqTk)) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				return nil, err | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			seqNode.Values = append(seqNode.Values, node) | 
					
						
							|  |  |  | 			break | 
					
						
							| 
									
										
										
										
											2024-11-17 23:58:35 +09:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-10-16 18:21:20 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 11:05:45 +09:00
										 |  |  | 		value, err := p.parseToken(ctx.withIndex(uint(len(seqNode.Values))), valueTk) | 
					
						
							| 
									
										
										
										
											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-26 11:05:45 +09:00
										 |  |  | func (p *parser) parseDirective(ctx *context, g *TokenGroup) (*ast.DirectiveNode, error) { | 
					
						
							|  |  |  | 	node, err := newDirectiveNode(ctx, g.First()) | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 	value, err := p.parseToken(ctx, g.Last()) | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 	node.Value = value | 
					
						
							|  |  |  | 	return node, 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) | 
					
						
							|  |  |  | 	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
										 |  |  | } |