php-parser/php7/parser.go

234 lines
5.0 KiB
Go
Raw Normal View History

2018-01-26 13:24:56 +00:00
package php7
import (
"io"
"strings"
2018-01-26 13:24:56 +00:00
2018-04-09 20:08:29 +00:00
"github.com/z7zmey/php-parser/errors"
"github.com/z7zmey/php-parser/freefloating"
2018-01-26 13:24:56 +00:00
"github.com/z7zmey/php-parser/node"
2018-04-15 18:39:26 +00:00
"github.com/z7zmey/php-parser/parser"
"github.com/z7zmey/php-parser/position"
"github.com/z7zmey/php-parser/scanner"
2018-01-26 13:24:56 +00:00
)
func (lval *yySymType) Token(t *scanner.Token) {
lval.token = t
}
// Parser structure
type Parser struct {
2018-04-10 13:19:47 +00:00
*scanner.Lexer
2018-04-10 17:15:15 +00:00
path string
2018-04-15 20:04:24 +00:00
currentToken *scanner.Token
2018-04-15 19:06:02 +00:00
positionBuilder *parser.PositionBuilder
rootNode node.Node
}
// NewParser creates and returns new Parser
2018-04-10 17:15:15 +00:00
func NewParser(src io.Reader, path string) *Parser {
lexer := scanner.NewLexer(src, path)
return &Parser{
2018-04-10 13:19:47 +00:00
lexer,
2018-04-10 17:15:15 +00:00
path,
nil,
nil,
nil,
}
}
2018-01-26 13:24:56 +00:00
// Lex proxy to lexer Lex
func (l *Parser) Lex(lval *yySymType) int {
t := l.Lexer.Lex(lval)
2018-04-15 20:04:24 +00:00
l.currentToken = lval.token
return t
}
func (l *Parser) Error(msg string) {
2018-11-05 15:14:09 +00:00
pos := &position.Position{
StartLine: l.currentToken.StartLine,
EndLine: l.currentToken.EndLine,
StartPos: l.currentToken.StartPos,
EndPos: l.currentToken.EndPos,
}
l.Lexer.Errors = append(l.Lexer.Errors, errors.NewError(msg, pos))
}
2019-02-25 14:52:47 +00:00
func (l *Parser) WithFreeFloating() {
l.Lexer.WithFreeFloating = true
}
// Parse the php7 Parser entrypoint
func (l *Parser) Parse() int {
// init
l.Lexer.Errors = nil
l.rootNode = nil
2018-06-24 07:19:44 +00:00
l.positionBuilder = &parser.PositionBuilder{}
// parse
return yyParse(l)
2018-01-26 13:24:56 +00:00
}
2018-04-10 17:15:15 +00:00
// GetPath return path to file
func (l *Parser) GetPath() string {
return l.path
}
// GetRootNode returns root node
func (l *Parser) GetRootNode() node.Node {
return l.rootNode
2018-01-26 13:24:56 +00:00
}
// GetErrors returns errors list
func (l *Parser) GetErrors() []*errors.Error {
return l.Lexer.Errors
2018-01-26 13:24:56 +00:00
}
2018-05-27 15:02:58 +00:00
// helpers
func lastNode(nn []node.Node) node.Node {
2018-06-26 08:57:17 +00:00
if len(nn) == 0 {
return nil
}
2018-05-27 15:02:58 +00:00
return nn[len(nn)-1]
}
func firstNode(nn []node.Node) node.Node {
return nn[0]
}
2018-06-10 11:53:10 +00:00
func isDollar(r rune) bool {
return r == '$'
}
func (l *Parser) MoveFreeFloating(src node.Node, dst node.Node) {
2019-02-25 14:52:47 +00:00
if l.Lexer.WithFreeFloating == false {
return
}
if src.GetFreeFloating() == nil {
return
}
l.setFreeFloating(dst, freefloating.Start, (*src.GetFreeFloating())[freefloating.Start])
delete((*src.GetFreeFloating()), freefloating.Start)
2018-12-31 15:06:27 +00:00
}
func (l *Parser) setFreeFloating(dst node.Node, p freefloating.Position, strings []freefloating.String) {
2019-02-25 14:52:47 +00:00
if l.Lexer.WithFreeFloating == false {
return
}
if len(strings) == 0 {
return
}
dstCollection := dst.GetFreeFloating()
if *dstCollection == nil {
*dstCollection = make(freefloating.Collection)
}
(*dstCollection)[p] = strings
}
func (l *Parser) GetFreeFloatingToken(t *scanner.Token) []freefloating.String {
2019-02-25 14:52:47 +00:00
if l.Lexer.WithFreeFloating == false {
return []freefloating.String{}
}
return t.GetFreeFloatingToken()
}
func (l *Parser) addDollarToken(v node.Node) {
2019-02-25 14:52:47 +00:00
if l.Lexer.WithFreeFloating == false {
return
}
l.setFreeFloating(v, freefloating.Dollar, []freefloating.String{
{
StringType: freefloating.TokenType,
Value: "$",
Position: &position.Position{
StartLine: v.GetPosition().StartLine,
EndLine: v.GetPosition().StartLine,
StartPos: v.GetPosition().StartPos,
EndPos: v.GetPosition().StartPos + 1,
},
},
})
}
func (l *Parser) splitSemiColonAndPhpCloseTag(htmlNode node.Node, prevNode node.Node) {
2019-02-25 14:52:47 +00:00
if l.Lexer.WithFreeFloating == false {
return
}
semiColon := (*prevNode.GetFreeFloating())[freefloating.SemiColon]
delete((*prevNode.GetFreeFloating()), freefloating.SemiColon)
if len(semiColon) == 0 {
return
}
p := semiColon[0].Position
if semiColon[0].Value[0] == ';' {
l.setFreeFloating(prevNode, freefloating.SemiColon, []freefloating.String{
{
StringType: freefloating.TokenType,
Value: ";",
Position: &position.Position{
StartLine: p.StartLine,
EndLine: p.StartLine,
StartPos: p.StartPos,
EndPos: p.StartPos + 1,
},
},
})
}
vlen := len(semiColon[0].Value)
tlen := 2
if strings.HasSuffix(semiColon[0].Value, "?>\n") {
tlen = 3
}
phpCloseTag := []freefloating.String{}
if vlen-tlen > 1 {
phpCloseTag = append(phpCloseTag, freefloating.String{
StringType: freefloating.WhiteSpaceType,
Value: semiColon[0].Value[1 : vlen-tlen],
Position: &position.Position{
StartLine: p.StartLine,
EndLine: p.EndLine,
StartPos: p.StartPos + 1,
EndPos: p.EndPos - tlen,
},
})
}
phpCloseTag = append(phpCloseTag, freefloating.String{
StringType: freefloating.WhiteSpaceType,
Value: semiColon[0].Value[vlen-tlen:],
Position: &position.Position{
StartLine: p.EndLine,
EndLine: p.EndLine,
StartPos: p.EndPos - tlen,
EndPos: p.EndPos,
},
})
l.setFreeFloating(htmlNode, freefloating.Start, append(phpCloseTag, (*htmlNode.GetFreeFloating())[freefloating.Start]...))
}
func (p *Parser) returnTokenToPool(yyDollar []yySymType, yyVAL *yySymType) {
for i := 1; i < len(yyDollar); i++ {
if yyDollar[i].token != nil {
p.TokenPool.Put(yyDollar[i].token)
}
yyDollar[i].token = nil
}
yyVAL.token = nil
}