php-parser/internal/php5/parser.go

207 lines
4.0 KiB
Go
Raw Normal View History

2018-01-26 13:24:56 +00:00
package php5
import (
2020-05-13 17:18:53 +00:00
"bytes"
2020-05-17 19:56:32 +00:00
"fmt"
2020-05-12 21:16:36 +00:00
"github.com/z7zmey/php-parser/internal/positionbuilder"
2020-05-13 17:18:53 +00:00
"github.com/z7zmey/php-parser/internal/scanner"
2020-05-12 21:16:36 +00:00
"github.com/z7zmey/php-parser/pkg/ast"
"github.com/z7zmey/php-parser/pkg/errors"
2020-05-13 17:18:53 +00:00
"github.com/z7zmey/php-parser/pkg/token"
2018-01-26 13:24:56 +00:00
)
// Parser structure
type Parser struct {
2020-05-18 18:44:03 +00:00
Lexer *scanner.Lexer
2018-04-15 20:04:24 +00:00
currentToken *scanner.Token
2019-12-26 15:57:56 +00:00
positionBuilder *positionbuilder.PositionBuilder
2020-05-12 21:16:36 +00:00
rootNode ast.Vertex
errors []*errors.Error
withTokens bool
}
2018-01-26 13:24:56 +00:00
// NewParser creates and returns new Parser
func NewParser(src []byte, v string, withTokens bool) *Parser {
parser := &Parser{
withTokens: withTokens,
}
scannerConfig := scanner.Config{
WithHiddenTokens: withTokens,
ErrHandlerFunc: func(e *errors.Error) {
parser.errors = append(parser.errors, e)
},
}
lexer := scanner.NewLexer(src, v, scannerConfig)
parser.Lexer = lexer
return parser
}
2018-02-04 16:51:44 +00:00
// Lex proxy to scanner Lex
func (l *Parser) Lex(lval *yySymType) int {
t := l.Lexer.Lex()
l.currentToken = t
lval.token = t
return int(t.ID)
}
func (l *Parser) Error(msg string) {
var pos = l.currentToken.Position
2018-11-05 15:14:09 +00:00
l.errors = append(l.errors, errors.NewError(msg, &pos))
}
// GetErrors returns errors list
func (l *Parser) GetErrors() []*errors.Error {
return l.errors
}
// Parse the php7 Parser entrypoint
func (l *Parser) Parse() int {
// init
l.errors = nil
l.rootNode = nil
2019-12-26 15:57:56 +00:00
l.positionBuilder = &positionbuilder.PositionBuilder{}
// parse
return yyParse(l)
2018-01-26 13:24:56 +00:00
}
// GetRootNode returns root node
2020-05-12 21:16:36 +00:00
func (l *Parser) GetRootNode() ast.Vertex {
return l.rootNode
2018-01-26 13:24:56 +00:00
}
2018-06-06 23:25:27 +00:00
// helpers
2020-05-13 17:18:53 +00:00
func lastNode(nn []ast.Vertex) ast.Vertex {
2018-06-26 08:57:17 +00:00
if len(nn) == 0 {
return nil
}
2018-06-06 23:25:27 +00:00
return nn[len(nn)-1]
}
2020-05-13 17:18:53 +00:00
func firstNode(nn []ast.Vertex) ast.Vertex {
2018-06-06 23:25:27 +00:00
return nn[0]
}
2018-06-10 11:53:10 +00:00
func isDollar(r rune) bool {
return r == '$'
}
2020-05-13 17:18:53 +00:00
func (l *Parser) MoveFreeFloating(src ast.Vertex, dst ast.Vertex) {
if l.withTokens == false {
return
}
2020-05-13 17:18:53 +00:00
if src.GetNode().Tokens == nil {
return
}
2020-05-13 17:18:53 +00:00
l.setFreeFloating(dst, token.Start, src.GetNode().Tokens[token.Start])
delete(src.GetNode().Tokens, token.Start)
}
2020-05-13 17:18:53 +00:00
func (l *Parser) setFreeFloating(dst ast.Vertex, p token.Position, strings []token.Token) {
if l.withTokens == false {
return
}
if len(strings) == 0 {
return
}
2020-05-13 17:18:53 +00:00
dstCollection := &dst.GetNode().Tokens
if *dstCollection == nil {
2020-05-13 17:18:53 +00:00
*dstCollection = make(token.Collection)
}
(*dstCollection)[p] = strings
}
2020-05-13 17:18:53 +00:00
func (l *Parser) GetFreeFloatingToken(t *scanner.Token) []token.Token {
if l.withTokens == false {
2020-05-13 17:18:53 +00:00
return []token.Token{}
}
2020-05-17 19:56:32 +00:00
return []token.Token{
{
ID: token.ID(t.ID),
Value: t.Value,
},
}
}
func (l *Parser) addDollarToken(v ast.Vertex) {
if l.withTokens == false {
2020-05-17 19:56:32 +00:00
return
}
2020-05-17 19:56:32 +00:00
l.setFreeFloating(v, token.Dollar, []token.Token{
{
ID: token.ID('$'),
Value: []byte("$"),
},
})
}
2020-05-13 17:18:53 +00:00
func (l *Parser) splitSemiColonAndPhpCloseTag(htmlNode ast.Vertex, prevNode ast.Vertex) {
if l.withTokens == false {
return
}
2020-05-13 17:18:53 +00:00
semiColon := prevNode.GetNode().Tokens[token.SemiColon]
delete(prevNode.GetNode().Tokens, token.SemiColon)
if len(semiColon) == 0 {
return
}
if semiColon[0].Value[0] == ';' {
2020-05-13 17:18:53 +00:00
l.setFreeFloating(prevNode, token.SemiColon, []token.Token{
{
2020-05-13 17:18:53 +00:00
ID: token.ID(';'),
Value: semiColon[0].Value[0:1],
},
})
}
vlen := len(semiColon[0].Value)
2020-05-17 19:56:32 +00:00
fmt.Printf("vlen: %q\n", string(semiColon[0].Value))
tlen := 2
2020-05-13 17:18:53 +00:00
if bytes.HasSuffix(semiColon[0].Value, []byte("?>\n")) {
tlen = 3
}
2020-05-13 17:18:53 +00:00
phpCloseTag := []token.Token{}
if vlen-tlen > 1 {
2020-05-13 17:18:53 +00:00
phpCloseTag = append(phpCloseTag, token.Token{
ID: token.T_WHITESPACE,
Value: semiColon[0].Value[1 : vlen-tlen],
})
}
2020-05-13 17:18:53 +00:00
phpCloseTag = append(phpCloseTag, token.Token{
ID: T_CLOSE_TAG,
Value: semiColon[0].Value[vlen-tlen:],
})
2020-05-13 17:18:53 +00:00
l.setFreeFloating(htmlNode, token.Start, append(phpCloseTag, htmlNode.GetNode().Tokens[token.Start]...))
}
func (p *Parser) returnTokenToPool(yyDollar []yySymType, yyVAL *yySymType) {
for i := 1; i < len(yyDollar); i++ {
if yyDollar[i].token != nil {
2019-03-10 21:37:01 +00:00
p.Lexer.ReturnTokenToPool(yyDollar[i].token)
}
yyDollar[i].token = nil
}
yyVAL.token = nil
}