2018-01-02 12:41:54 +00:00
|
|
|
<!--
|
|
|
|
Title: PHP Parser
|
|
|
|
Description: A Parser for PHP written in Go.
|
|
|
|
Author: Slizov Vadim
|
2018-01-10 22:49:01 +00:00
|
|
|
Keywords: go golang php php-parser ast
|
2018-01-02 12:41:54 +00:00
|
|
|
-->
|
|
|
|
|
2018-03-02 20:43:56 +00:00
|
|
|
<img src="./parser.jpg" alt="A parser for PHP written in Go" width="980"/>
|
2018-01-09 16:36:06 +00:00
|
|
|
|
2018-01-10 22:32:20 +00:00
|
|
|
[![Go Report Card](https://goreportcard.com/badge/github.com/z7zmey/php-parser)](https://goreportcard.com/report/github.com/z7zmey/php-parser)
|
2018-01-09 16:36:06 +00:00
|
|
|
[![Exago](https://api.exago.io:443/badge/tests/github.com/z7zmey/php-parser)](https://exago.io/project/github.com/z7zmey/php-parser)
|
2018-01-12 07:44:56 +00:00
|
|
|
[![Exago](https://api.exago.io:443/badge/cov/github.com/z7zmey/php-parser)](https://exago.io/project/github.com/z7zmey/php-parser)
|
2018-02-19 12:34:49 +00:00
|
|
|
[![GoDoc](https://godoc.org/github.com/z7zmey/php-parser?status.svg)](https://godoc.org/github.com/z7zmey/php-parser)
|
2018-01-09 16:36:06 +00:00
|
|
|
|
2018-03-04 13:47:41 +00:00
|
|
|
#### Try it online: [demo](https://php-parser.com)
|
|
|
|
|
2018-01-09 16:29:05 +00:00
|
|
|
## Features:
|
2018-02-04 19:47:12 +00:00
|
|
|
- Fully support PHP5 and PHP7 syntax
|
2018-01-09 16:29:05 +00:00
|
|
|
- Abstract syntax tree representation
|
2018-01-09 16:36:06 +00:00
|
|
|
- Traversing AST
|
2018-02-27 21:59:27 +00:00
|
|
|
- Namespace resolver
|
2018-01-09 16:29:05 +00:00
|
|
|
|
2018-02-06 10:52:31 +00:00
|
|
|
## Install
|
|
|
|
|
|
|
|
```
|
|
|
|
go get github.com/z7zmey/php-parser
|
|
|
|
```
|
|
|
|
|
2018-01-12 08:04:31 +00:00
|
|
|
## Example
|
|
|
|
```Golang
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2018-04-10 12:51:00 +00:00
|
|
|
"fmt"
|
2018-01-12 08:04:31 +00:00
|
|
|
"bytes"
|
2018-04-07 13:01:59 +00:00
|
|
|
"os"
|
2018-01-12 08:04:31 +00:00
|
|
|
|
2018-02-04 19:49:53 +00:00
|
|
|
"github.com/z7zmey/php-parser/php7"
|
2018-01-12 08:04:31 +00:00
|
|
|
"github.com/z7zmey/php-parser/visitor"
|
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
2018-01-12 08:17:41 +00:00
|
|
|
src := bytes.NewBufferString(`<? echo "Hello world";`)
|
2018-04-09 20:30:44 +00:00
|
|
|
|
2018-04-10 12:51:00 +00:00
|
|
|
parser := php7.NewParser(src, "example.php")
|
|
|
|
parser.Parse()
|
|
|
|
|
|
|
|
for _, e := range parser.GetErrors() {
|
|
|
|
fmt.Println(e)
|
|
|
|
}
|
2018-01-12 08:04:31 +00:00
|
|
|
|
2018-02-04 19:35:46 +00:00
|
|
|
visitor := visitor.Dumper{
|
2018-04-09 20:30:44 +00:00
|
|
|
Writer: os.Stdout,
|
2018-02-04 19:35:46 +00:00
|
|
|
Indent: "",
|
2018-04-10 12:51:00 +00:00
|
|
|
Comments: parser.GetComments(),
|
|
|
|
Positions: parser.GetPositions(),
|
2018-02-04 19:35:46 +00:00
|
|
|
}
|
2018-04-10 12:51:00 +00:00
|
|
|
|
|
|
|
rootNode := parser.GetRootNode()
|
|
|
|
rootNode.Walk(visitor)
|
2018-01-12 08:04:31 +00:00
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2018-02-06 10:52:31 +00:00
|
|
|
## CLI dumper
|
|
|
|
|
|
|
|
```
|
|
|
|
$GOPATH/bin/php-parser /path/to/file/or/dir
|
|
|
|
```
|
|
|
|
|
2018-03-01 21:14:51 +00:00
|
|
|
## Namespace resolver
|
|
|
|
|
|
|
|
Namespace resolver is a visitor that traverses nodes and resolves nodes fully qualified name.
|
|
|
|
It does not change AST but collects resolved names into `map[node.Node]string`
|
|
|
|
|
|
|
|
- For `Class`, `Interface`, `Trait`, `Function`, `ConstList` nodes collects name with current namespace.
|
|
|
|
- For `Name`, `Relative`, `FullyQualified` nodes resolves `use` aliases and collects a fully qualified name.
|
|
|
|
|
2018-04-06 12:23:33 +00:00
|
|
|
## Pretty printer
|
|
|
|
|
|
|
|
```Golang
|
|
|
|
nodes := &stmt.StmtList{
|
|
|
|
Stmts: []node.Node{
|
|
|
|
&stmt.Namespace{
|
|
|
|
NamespaceName: &name.Name{Parts: []node.Node{&name.NamePart{Value: "Foo"}}},
|
|
|
|
},
|
|
|
|
&stmt.Class{
|
|
|
|
Modifiers: []node.Node{&node.Identifier{Value: "abstract"}},
|
|
|
|
ClassName: &name.Name{Parts: []node.Node{&name.NamePart{Value: "Bar"}}},
|
|
|
|
Extends: &name.Name{Parts: []node.Node{&name.NamePart{Value: "Baz"}}},
|
|
|
|
Stmts: []node.Node{
|
|
|
|
&stmt.ClassMethod{
|
|
|
|
Modifiers: []node.Node{&node.Identifier{Value: "public"}},
|
|
|
|
MethodName: &node.Identifier{Value: "greet"},
|
|
|
|
Stmts: []node.Node{
|
|
|
|
&stmt.Echo{
|
|
|
|
Exprs: []node.Node{
|
|
|
|
&scalar.String{Value: "'Hello world'"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
file := os.Stdout
|
|
|
|
p := printer.NewPrinter(file, " ")
|
|
|
|
p.PrintFile(nodes)
|
|
|
|
```
|
|
|
|
|
|
|
|
Output:
|
|
|
|
```PHP
|
|
|
|
<?php
|
|
|
|
namespace Foo;
|
|
|
|
abstract class Bar extends Baz
|
|
|
|
{
|
|
|
|
public function greet()
|
|
|
|
{
|
|
|
|
echo 'Hello world';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2018-01-02 12:41:54 +00:00
|
|
|
## Roadmap
|
2018-01-02 12:34:26 +00:00
|
|
|
- [X] Lexer
|
2018-02-06 10:52:31 +00:00
|
|
|
- [x] PHP 7 syntax analyzer
|
2018-01-02 12:34:26 +00:00
|
|
|
- [x] AST nodes
|
2018-01-02 12:41:54 +00:00
|
|
|
- [x] AST visitor
|
2018-01-04 22:33:50 +00:00
|
|
|
- [x] AST dumper
|
|
|
|
- [x] node position
|
2018-01-05 15:03:59 +00:00
|
|
|
- [x] handling comments
|
2018-02-04 19:47:12 +00:00
|
|
|
- [x] PHP 5 syntax analyzer
|
2018-02-19 09:44:52 +00:00
|
|
|
- [x] Tests
|
2018-02-27 21:59:27 +00:00
|
|
|
- [x] Namespace resolver
|
2018-04-06 12:23:33 +00:00
|
|
|
- [x] Pretty printer
|
2018-02-06 10:52:31 +00:00
|
|
|
- [ ] PhpDocComment parser
|
2018-01-09 16:29:05 +00:00
|
|
|
- [ ] Error handling
|
|
|
|
- [ ] Stabilize api
|
2018-01-02 12:41:54 +00:00
|
|
|
- [ ] Documentation
|
2018-02-04 19:47:12 +00:00
|
|
|
- [ ] Code flow graph
|