1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-11 04:17:13 +00:00
Beego/parser.go

394 lines
9.5 KiB
Go
Raw Normal View History

2014-08-18 16:41:43 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2014-06-08 20:24:01 +08:00
package beego
2014-06-09 10:11:37 +08:00
import (
"encoding/json"
2014-06-09 17:33:04 +08:00
"errors"
"fmt"
"go/ast"
"go/parser"
"go/token"
"io/ioutil"
2014-06-09 10:11:37 +08:00
"os"
2016-04-27 23:57:22 +08:00
"path/filepath"
"regexp"
"sort"
"strconv"
2014-06-09 17:33:04 +08:00
"strings"
"unicode"
2017-04-25 18:39:42 +03:00
"github.com/astaxie/beego/context/param"
"github.com/astaxie/beego/logs"
"github.com/astaxie/beego/utils"
2014-06-09 10:11:37 +08:00
)
2014-06-09 17:33:04 +08:00
var globalRouterTemplate = `package routers
2014-06-09 10:11:37 +08:00
import (
"github.com/astaxie/beego"
2017-04-25 18:39:42 +03:00
"github.com/astaxie/beego/context/param"
2014-06-09 10:11:37 +08:00
)
func init() {
{{.globalinfo}}
2014-06-09 10:11:37 +08:00
}
`
var (
2015-09-08 21:45:45 +08:00
lastupdateFilename = "lastupdate.tmp"
2014-11-05 22:23:54 +08:00
commentFilename string
pkgLastupdate map[string]int64
genInfoList map[string][]ControllerComments
)
2014-06-09 10:11:37 +08:00
2016-09-21 19:33:12 +08:00
const commentPrefix = "commentsRouter_"
2014-11-05 22:23:54 +08:00
2014-06-09 17:33:04 +08:00
func init() {
pkgLastupdate = make(map[string]int64)
2014-06-09 17:33:04 +08:00
}
2014-06-09 10:11:37 +08:00
2014-06-09 17:33:04 +08:00
func parserPkg(pkgRealpath, pkgpath string) error {
2016-05-05 19:28:09 +08:00
rep := strings.NewReplacer("\\", "_", "/", "_", ".", "_")
2016-05-06 13:26:48 +08:00
commentFilename, _ = filepath.Rel(AppPath, pkgRealpath)
2016-09-21 19:33:12 +08:00
commentFilename = commentPrefix + rep.Replace(commentFilename) + ".go"
if !compareFile(pkgRealpath) {
logs.Info(pkgRealpath + " no changed")
return nil
}
2014-11-06 16:25:47 +08:00
genInfoList = make(map[string][]ControllerComments)
2014-06-09 17:33:04 +08:00
fileSet := token.NewFileSet()
astPkgs, err := parser.ParseDir(fileSet, pkgRealpath, func(info os.FileInfo) bool {
name := info.Name()
return !info.IsDir() && !strings.HasPrefix(name, ".") && strings.HasSuffix(name, ".go")
}, parser.ParseComments)
2014-06-09 10:11:37 +08:00
2014-06-09 17:33:04 +08:00
if err != nil {
return err
}
for _, pkg := range astPkgs {
for _, fl := range pkg.Files {
for _, d := range fl.Decls {
switch specDecl := d.(type) {
case *ast.FuncDecl:
2014-08-05 08:56:04 +08:00
if specDecl.Recv != nil {
2015-09-10 11:35:57 +02:00
exp, ok := specDecl.Recv.List[0].Type.(*ast.StarExpr) // Check that the type is correct first beforing throwing to parser
if ok {
parserComments(specDecl, fmt.Sprint(exp.X), pkgpath)
2015-09-10 11:35:57 +02:00
}
2014-08-05 08:56:04 +08:00
}
2014-06-09 17:33:04 +08:00
}
}
}
}
2016-04-27 23:57:22 +08:00
genRouterCode(pkgRealpath)
savetoFile(pkgRealpath)
2014-06-09 17:33:04 +08:00
return nil
}
type parsedComment struct {
routerPath string
methods []string
params map[string]parsedParam
}
type parsedParam struct {
name string
datatype string
location string
defValue string
required bool
}
func parserComments(f *ast.FuncDecl, controllerName, pkgpath string) error {
if f.Doc != nil {
parsedComment, err := parseComment(f.Doc.List)
if err != nil {
return err
}
if parsedComment.routerPath != "" {
key := pkgpath + ":" + controllerName
cc := ControllerComments{}
cc.Method = f.Name.String()
cc.Router = parsedComment.routerPath
cc.AllowHTTPMethods = parsedComment.methods
cc.MethodParams = buildMethodParams(f.Type.Params.List, parsedComment)
genInfoList[key] = append(genInfoList[key], cc)
}
}
return nil
}
func buildMethodParams(funcParams []*ast.Field, pc *parsedComment) []*param.MethodParam {
result := make([]*param.MethodParam, 0, len(funcParams))
for _, fparam := range funcParams {
for _, pName := range fparam.Names {
methodParam := buildMethodParam(fparam, pName.Name, pc)
result = append(result, methodParam)
}
}
return result
}
func buildMethodParam(fparam *ast.Field, name string, pc *parsedComment) *param.MethodParam {
options := []param.MethodParamOption{}
if cparam, ok := pc.params[name]; ok {
//Build param from comment info
name = cparam.name
if cparam.required {
options = append(options, param.IsRequired)
}
switch cparam.location {
case "body":
options = append(options, param.InBody)
case "header":
options = append(options, param.InHeader)
case "path":
options = append(options, param.InPath)
}
if cparam.defValue != "" {
options = append(options, param.Default(cparam.defValue))
}
} else {
if paramInPath(name, pc.routerPath) {
options = append(options, param.InPath)
}
}
return param.New(name, options...)
}
func paramInPath(name, route string) bool {
return strings.HasSuffix(route, ":"+name) ||
strings.Contains(route, ":"+name+"/")
}
var routeRegex = regexp.MustCompile(`@router\s+(\S+)(?:\s+\[(\S+)\])?`)
func parseComment(lines []*ast.Comment) (pc *parsedComment, err error) {
pc = &parsedComment{}
for _, c := range lines {
t := strings.TrimSpace(strings.TrimLeft(c.Text, "//"))
if strings.HasPrefix(t, "@router") {
matches := routeRegex.FindStringSubmatch(t)
if len(matches) == 3 {
pc.routerPath = matches[1]
methods := matches[2]
if methods == "" {
pc.methods = []string{"get"}
//pc.hasGet = true
2014-06-09 17:33:04 +08:00
} else {
pc.methods = strings.Split(methods, ",")
//pc.hasGet = strings.Contains(methods, "get")
2014-06-09 17:33:04 +08:00
}
} else {
return nil, errors.New("Router information is missing")
2014-06-09 17:33:04 +08:00
}
} else if strings.HasPrefix(t, "@Param") {
pv := getparams(strings.TrimSpace(strings.TrimLeft(t, "@Param")))
if len(pv) < 4 {
logs.Error("Invalid @Param format. Needs at least 4 parameters")
}
p := parsedParam{}
2017-04-25 23:42:35 +03:00
names := strings.SplitN(pv[0], "=>", 2)
p.name = names[0]
funcParamName := p.name
if len(names) > 1 {
2017-04-25 23:42:35 +03:00
funcParamName = names[1]
}
p.location = pv[1]
p.datatype = pv[2]
switch len(pv) {
case 5:
p.required, _ = strconv.ParseBool(pv[3])
case 6:
p.defValue = pv[3]
p.required, _ = strconv.ParseBool(pv[4])
}
if pc.params == nil {
pc.params = map[string]parsedParam{}
}
pc.params[funcParamName] = p
2014-06-09 17:33:04 +08:00
}
}
return
}
// direct copy from bee\g_docs.go
2017-10-11 14:35:31 +08:00
// analysis params return []string
// @Param query form string true "The email for login"
// [query form string true "The email for login"]
func getparams(str string) []string {
var s []rune
var j int
var start bool
var r []string
var quoted int8
2017-05-17 20:50:41 +03:00
for _, c := range str {
if unicode.IsSpace(c) && quoted == 0 {
if !start {
continue
} else {
start = false
j++
r = append(r, string(s))
s = make([]rune, 0)
continue
}
}
start = true
if c == '"' {
quoted ^= 1
continue
}
s = append(s, c)
}
if len(s) > 0 {
r = append(r, string(s))
}
return r
2014-06-09 17:33:04 +08:00
}
2016-04-27 23:57:22 +08:00
func genRouterCode(pkgRealpath string) {
os.Mkdir(getRouterDir(pkgRealpath), 0755)
logs.Info("generate router from comments")
2015-09-08 21:45:45 +08:00
var (
globalinfo string
sortKey []string
)
for k := range genInfoList {
sortKey = append(sortKey, k)
}
sort.Strings(sortKey)
for _, k := range sortKey {
cList := genInfoList[k]
2017-07-15 17:26:20 +08:00
sort.Sort(ControllerCommentsSlice(cList))
2014-06-09 17:33:04 +08:00
for _, c := range cList {
allmethod := "nil"
if len(c.AllowHTTPMethods) > 0 {
2014-06-09 17:33:04 +08:00
allmethod = "[]string{"
for _, m := range c.AllowHTTPMethods {
2014-06-09 17:33:04 +08:00
allmethod += `"` + m + `",`
}
allmethod = strings.TrimRight(allmethod, ",") + "}"
}
params := "nil"
if len(c.Params) > 0 {
2014-06-09 17:33:04 +08:00
params = "[]map[string]string{"
for _, p := range c.Params {
2014-06-09 17:33:04 +08:00
for k, v := range p {
params = params + `map[string]string{` + k + `:"` + v + `"},`
}
}
params = strings.TrimRight(params, ",") + "}"
}
methodParams := "param.Make("
if len(c.MethodParams) > 0 {
lines := make([]string, 0, len(c.MethodParams))
for _, m := range c.MethodParams {
lines = append(lines, fmt.Sprint(m))
}
methodParams += "\n " +
strings.Join(lines, ",\n ") +
",\n "
}
methodParams += ")"
globalinfo = globalinfo + `
beego.GlobalControllerRouter["` + k + `"] = append(beego.GlobalControllerRouter["` + k + `"],
beego.ControllerComments{
2016-07-06 12:53:47 +08:00
Method: "` + strings.TrimSpace(c.Method) + `",
` + "Router: `" + c.Router + "`" + `,
AllowHTTPMethods: ` + allmethod + `,
MethodParams: ` + methodParams + `,
2016-07-06 12:53:47 +08:00
Params: ` + params + `})
`
2014-06-10 11:02:41 +08:00
}
}
if globalinfo != "" {
2016-05-06 13:26:48 +08:00
f, err := os.Create(filepath.Join(getRouterDir(pkgRealpath), commentFilename))
2014-06-10 11:02:41 +08:00
if err != nil {
panic(err)
2014-06-09 17:33:04 +08:00
}
2014-06-10 11:02:41 +08:00
defer f.Close()
f.WriteString(strings.Replace(globalRouterTemplate, "{{.globalinfo}}", globalinfo, -1))
2014-06-09 17:33:04 +08:00
}
2014-06-09 10:11:37 +08:00
}
func compareFile(pkgRealpath string) bool {
2016-05-06 13:26:48 +08:00
if !utils.FileExists(filepath.Join(getRouterDir(pkgRealpath), commentFilename)) {
return true
}
2015-12-09 23:35:04 +08:00
if utils.FileExists(lastupdateFilename) {
content, err := ioutil.ReadFile(lastupdateFilename)
if err != nil {
return true
}
json.Unmarshal(content, &pkgLastupdate)
2014-06-30 15:57:36 +08:00
lastupdate, err := getpathTime(pkgRealpath)
if err != nil {
return true
}
if v, ok := pkgLastupdate[pkgRealpath]; ok {
2014-06-30 15:57:36 +08:00
if lastupdate <= v {
return false
}
}
}
return true
}
func savetoFile(pkgRealpath string) {
2014-06-30 15:57:36 +08:00
lastupdate, err := getpathTime(pkgRealpath)
if err != nil {
return
}
2014-06-30 15:57:36 +08:00
pkgLastupdate[pkgRealpath] = lastupdate
d, err := json.Marshal(pkgLastupdate)
if err != nil {
return
}
2015-12-09 23:35:04 +08:00
ioutil.WriteFile(lastupdateFilename, d, os.ModePerm)
}
2014-06-30 15:57:36 +08:00
func getpathTime(pkgRealpath string) (lastupdate int64, err error) {
fl, err := ioutil.ReadDir(pkgRealpath)
if err != nil {
return lastupdate, err
}
for _, f := range fl {
if lastupdate < f.ModTime().UnixNano() {
lastupdate = f.ModTime().UnixNano()
}
}
return lastupdate, nil
}
2016-04-27 23:57:22 +08:00
func getRouterDir(pkgRealpath string) string {
dir := filepath.Dir(pkgRealpath)
for {
2016-05-06 13:26:48 +08:00
d := filepath.Join(dir, "routers")
if utils.FileExists(d) {
2016-04-27 23:57:22 +08:00
return d
}
2016-05-06 13:26:48 +08:00
if r, _ := filepath.Rel(dir, AppPath); r == "." {
2016-04-27 23:57:22 +08:00
return d
}
// Parent dir.
dir = filepath.Dir(dir)
}
}