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

227 lines
5.8 KiB
Go
Raw Permalink 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"
2014-06-09 17:33:04 +08:00
"path"
"sort"
2014-06-09 17:33:04 +08:00
"strings"
"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"
)
func init() {
{{.globalinfo}}
2014-06-09 10:11:37 +08:00
}
`
var (
lastupdateFilename string = "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
2014-11-06 17:30:50 +08:00
const COMMENTFL = "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 {
rep := strings.NewReplacer("/", "_", ".", "_")
2014-11-06 17:30:50 +08:00
commentFilename = COMMENTFL + rep.Replace(pkgpath) + ".go"
if !compareFile(pkgRealpath) {
Info(pkgRealpath + " has not changed, not reloading")
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 {
parserComments(specDecl.Doc, specDecl.Name.String(), fmt.Sprint(specDecl.Recv.List[0].Type.(*ast.StarExpr).X), pkgpath)
}
2014-06-09 17:33:04 +08:00
}
}
}
}
genRouterCode()
savetoFile(pkgRealpath)
2014-06-09 17:33:04 +08:00
return nil
}
func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpath string) error {
if comments != nil && comments.List != nil {
for _, c := range comments.List {
t := strings.TrimSpace(strings.TrimLeft(c.Text, "//"))
if strings.HasPrefix(t, "@router") {
elements := strings.TrimLeft(t, "@router ")
e1 := strings.SplitN(elements, " ", 2)
if len(e1) < 1 {
return errors.New("you should has router infomation")
}
key := pkgpath + ":" + controllerName
cc := ControllerComments{}
cc.Method = funcName
cc.Router = e1[0]
2014-06-09 17:33:04 +08:00
if len(e1) == 2 && e1[1] != "" {
e1 = strings.SplitN(e1[1], " ", 2)
if len(e1) >= 1 {
cc.AllowHTTPMethods = strings.Split(strings.Trim(e1[0], "[]"), ",")
2014-06-09 17:33:04 +08:00
} else {
cc.AllowHTTPMethods = append(cc.AllowHTTPMethods, "get")
2014-06-09 17:33:04 +08:00
}
} else {
cc.AllowHTTPMethods = append(cc.AllowHTTPMethods, "get")
2014-06-09 17:33:04 +08:00
}
if len(e1) == 2 && e1[1] != "" {
keyval := strings.Split(strings.Trim(e1[1], "[]"), " ")
for _, kv := range keyval {
kk := strings.Split(kv, ":")
cc.Params = append(cc.Params, map[string]string{strings.Join(kk[:len(kk)-1], ":"): kk[len(kk)-1]})
2014-06-09 17:33:04 +08:00
}
}
genInfoList[key] = append(genInfoList[key], cc)
}
}
}
return nil
}
func genRouterCode() {
os.Mkdir(path.Join(workPath, "routers"), 0755)
2014-06-09 17:33:04 +08:00
Info("generate router from comments")
var globalinfo string
sortKey := make([]string, 0)
for k, _ := range genInfoList {
sortKey = append(sortKey, k)
}
sort.Strings(sortKey)
for _, k := range sortKey {
cList := genInfoList[k]
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, ",") + "}"
}
globalinfo = globalinfo + `
beego.GlobalControllerRouter["` + k + `"] = append(beego.GlobalControllerRouter["` + k + `"],
beego.ControllerComments{
"` + strings.TrimSpace(c.Method) + `",
2014-08-25 19:47:57 +08:00
` + "`" + c.Router + "`" + `,
` + allmethod + `,
` + params + `})
`
2014-06-10 11:02:41 +08:00
}
}
if globalinfo != "" {
2014-11-03 23:33:11 +08:00
f, err := os.Create(path.Join(workPath, "routers", 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 {
2014-11-03 23:33:11 +08:00
if !utils.FileExists(path.Join(workPath, "routers", commentFilename)) {
return true
}
if utils.FileExists(path.Join(workPath, lastupdateFilename)) {
content, err := ioutil.ReadFile(path.Join(workPath, 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
}
ioutil.WriteFile(path.Join(workPath, 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
}