2013-03-14 09:57:09 +00:00
|
|
|
package beego
|
|
|
|
|
|
|
|
//@todo add template funcs
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"html/template"
|
2013-09-12 09:20:32 +00:00
|
|
|
"io/ioutil"
|
2013-03-14 09:57:09 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2013-10-29 14:35:04 +00:00
|
|
|
"reflect"
|
2013-09-12 09:20:32 +00:00
|
|
|
"regexp"
|
2013-03-14 09:57:09 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2013-09-12 09:20:32 +00:00
|
|
|
beegoTplFuncMap template.FuncMap
|
|
|
|
BeeTemplates map[string]*template.Template
|
|
|
|
BeeTemplateExt []string
|
2013-03-14 09:57:09 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
BeeTemplates = make(map[string]*template.Template)
|
|
|
|
beegoTplFuncMap = make(template.FuncMap)
|
|
|
|
BeeTemplateExt = make([]string, 0)
|
|
|
|
BeeTemplateExt = append(BeeTemplateExt, "tpl", "html")
|
|
|
|
beegoTplFuncMap["dateformat"] = DateFormat
|
|
|
|
beegoTplFuncMap["date"] = Date
|
|
|
|
beegoTplFuncMap["compare"] = Compare
|
2013-03-14 13:47:39 +00:00
|
|
|
beegoTplFuncMap["substr"] = Substr
|
2013-03-21 05:21:04 +00:00
|
|
|
beegoTplFuncMap["html2str"] = Html2str
|
2013-04-03 15:37:59 +00:00
|
|
|
beegoTplFuncMap["str2html"] = Str2html
|
|
|
|
beegoTplFuncMap["htmlquote"] = Htmlquote
|
|
|
|
beegoTplFuncMap["htmlunquote"] = Htmlunquote
|
2013-08-10 08:33:46 +00:00
|
|
|
beegoTplFuncMap["renderform"] = RenderForm
|
2013-10-29 14:35:04 +00:00
|
|
|
|
|
|
|
// go1.2 added template funcs
|
|
|
|
// Comparisons
|
|
|
|
beegoTplFuncMap["eq"] = eq // ==
|
|
|
|
beegoTplFuncMap["ge"] = ge // >=
|
|
|
|
beegoTplFuncMap["gt"] = gt // >
|
|
|
|
beegoTplFuncMap["le"] = le // <=
|
|
|
|
beegoTplFuncMap["lt"] = lt // <
|
|
|
|
beegoTplFuncMap["ne"] = ne // !=
|
2013-03-14 09:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddFuncMap let user to register a func in the template
|
|
|
|
func AddFuncMap(key string, funname interface{}) error {
|
|
|
|
beegoTplFuncMap[key] = funname
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type templatefile struct {
|
|
|
|
root string
|
|
|
|
files map[string][]string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *templatefile) visit(paths string, f os.FileInfo, err error) error {
|
|
|
|
if f == nil {
|
|
|
|
return err
|
|
|
|
}
|
2013-08-01 03:57:29 +00:00
|
|
|
if f.IsDir() || (f.Mode()&os.ModeSymlink) > 0 {
|
2013-03-14 09:57:09 +00:00
|
|
|
return nil
|
2013-08-01 03:57:29 +00:00
|
|
|
}
|
|
|
|
if !HasTemplateEXt(paths) {
|
2013-03-14 09:57:09 +00:00
|
|
|
return nil
|
2013-08-01 03:57:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
replace := strings.NewReplacer("\\", "/")
|
|
|
|
a := []byte(paths)
|
|
|
|
a = a[len([]byte(self.root)):]
|
2013-09-12 09:20:32 +00:00
|
|
|
file := strings.TrimLeft(replace.Replace(string(a)), "/")
|
|
|
|
subdir := filepath.Dir(file)
|
2013-08-01 03:57:29 +00:00
|
|
|
if _, ok := self.files[subdir]; ok {
|
2013-09-13 05:57:40 +00:00
|
|
|
self.files[subdir] = append(self.files[subdir], file)
|
2013-03-14 09:57:09 +00:00
|
|
|
} else {
|
2013-08-01 03:57:29 +00:00
|
|
|
m := make([]string, 1)
|
2013-09-13 05:57:40 +00:00
|
|
|
m[0] = file
|
2013-08-01 03:57:29 +00:00
|
|
|
self.files[subdir] = m
|
|
|
|
}
|
2013-03-14 09:57:09 +00:00
|
|
|
|
2013-08-01 03:57:29 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func HasTemplateEXt(paths string) bool {
|
|
|
|
for _, v := range BeeTemplateExt {
|
|
|
|
if strings.HasSuffix(paths, "."+v) {
|
|
|
|
return true
|
2013-03-14 09:57:09 +00:00
|
|
|
}
|
|
|
|
}
|
2013-08-01 03:57:29 +00:00
|
|
|
return false
|
2013-03-14 09:57:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func AddTemplateExt(ext string) {
|
|
|
|
for _, v := range BeeTemplateExt {
|
|
|
|
if v == ext {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
BeeTemplateExt = append(BeeTemplateExt, ext)
|
|
|
|
}
|
|
|
|
|
|
|
|
func BuildTemplate(dir string) error {
|
2013-03-21 13:45:48 +00:00
|
|
|
if _, err := os.Stat(dir); err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
2013-04-20 16:49:48 +00:00
|
|
|
return nil
|
2013-03-21 13:45:48 +00:00
|
|
|
} else {
|
|
|
|
return errors.New("dir open err")
|
|
|
|
}
|
2013-03-21 12:46:54 +00:00
|
|
|
}
|
2013-09-12 09:20:32 +00:00
|
|
|
self := &templatefile{
|
2013-03-14 09:57:09 +00:00
|
|
|
root: dir,
|
|
|
|
files: make(map[string][]string),
|
|
|
|
}
|
2013-03-21 13:47:19 +00:00
|
|
|
err := filepath.Walk(dir, func(path string, f os.FileInfo, err error) error {
|
2013-09-12 09:20:32 +00:00
|
|
|
return self.visit(path, f, err)
|
2013-03-14 09:57:09 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("filepath.Walk() returned %v\n", err)
|
|
|
|
return err
|
|
|
|
}
|
2013-09-13 05:57:40 +00:00
|
|
|
for _, v := range self.files {
|
|
|
|
for _, file := range v {
|
|
|
|
t, err := getTemplate(self.root, file, v...)
|
|
|
|
if err != nil {
|
|
|
|
Trace("parse template err:", file, err)
|
|
|
|
} else {
|
|
|
|
BeeTemplates[file] = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-03-14 09:57:09 +00:00
|
|
|
return nil
|
|
|
|
}
|
2013-09-12 09:20:32 +00:00
|
|
|
|
2013-10-30 15:02:53 +00:00
|
|
|
func getTplDeep(root, file, parent string, t *template.Template) (*template.Template, [][]string, error) {
|
|
|
|
var fileabspath string
|
|
|
|
if filepath.HasPrefix(file, "../") {
|
|
|
|
fileabspath = filepath.Join(root, filepath.Dir(parent), file)
|
|
|
|
} else {
|
|
|
|
fileabspath = filepath.Join(root, file)
|
|
|
|
}
|
|
|
|
if e, _ := FileExists(fileabspath); !e {
|
|
|
|
panic("can't find template file" + file)
|
|
|
|
}
|
2013-09-12 09:20:32 +00:00
|
|
|
data, err := ioutil.ReadFile(fileabspath)
|
|
|
|
if err != nil {
|
2013-09-13 09:41:31 +00:00
|
|
|
return nil, [][]string{}, err
|
2013-09-12 09:20:32 +00:00
|
|
|
}
|
|
|
|
t, err = t.New(file).Parse(string(data))
|
|
|
|
if err != nil {
|
2013-09-13 09:41:31 +00:00
|
|
|
return nil, [][]string{}, err
|
2013-09-12 09:20:32 +00:00
|
|
|
}
|
2013-09-12 10:20:29 +00:00
|
|
|
reg := regexp.MustCompile("{{[ ]*template[ ]+\"([^\"]+)\"")
|
2013-09-12 09:20:32 +00:00
|
|
|
allsub := reg.FindAllStringSubmatch(string(data), -1)
|
|
|
|
for _, m := range allsub {
|
|
|
|
if len(m) == 2 {
|
|
|
|
tlook := t.Lookup(m[1])
|
|
|
|
if tlook != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !HasTemplateEXt(m[1]) {
|
|
|
|
continue
|
|
|
|
}
|
2013-10-30 15:02:53 +00:00
|
|
|
t, _, err = getTplDeep(root, m[1], file, t)
|
|
|
|
if err != nil {
|
|
|
|
return nil, [][]string{}, err
|
2013-09-12 09:20:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-09-13 09:41:31 +00:00
|
|
|
return t, allsub, nil
|
2013-09-12 09:20:32 +00:00
|
|
|
}
|
|
|
|
|
2013-09-13 05:57:40 +00:00
|
|
|
func getTemplate(root, file string, others ...string) (t *template.Template, err error) {
|
2013-09-12 09:20:32 +00:00
|
|
|
t = template.New(file).Delims(TemplateLeft, TemplateRight).Funcs(beegoTplFuncMap)
|
2013-09-13 09:41:31 +00:00
|
|
|
var submods [][]string
|
2013-10-30 15:02:53 +00:00
|
|
|
t, submods, err = getTplDeep(root, file, "", t)
|
2013-09-26 10:33:01 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
t, err = _getTemplate(t, root, submods, others...)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func _getTemplate(t0 *template.Template, root string, submods [][]string, others ...string) (t *template.Template, err error) {
|
|
|
|
t = t0
|
2013-09-13 09:41:31 +00:00
|
|
|
for _, m := range submods {
|
|
|
|
if len(m) == 2 {
|
|
|
|
templ := t.Lookup(m[1])
|
|
|
|
if templ != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
//first check filename
|
|
|
|
for _, otherfile := range others {
|
|
|
|
if otherfile == m[1] {
|
2013-09-26 10:33:01 +00:00
|
|
|
var submods1 [][]string
|
2013-10-30 15:02:53 +00:00
|
|
|
t, submods1, err = getTplDeep(root, otherfile, "", t)
|
2013-09-13 09:41:31 +00:00
|
|
|
if err != nil {
|
|
|
|
Trace("template parse file err:", err)
|
2013-09-26 10:33:01 +00:00
|
|
|
} else if submods1 != nil && len(submods1) > 0 {
|
|
|
|
t, err = _getTemplate(t, root, submods1, others...)
|
2013-09-13 09:41:31 +00:00
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//second check define
|
|
|
|
for _, otherfile := range others {
|
|
|
|
fileabspath := filepath.Join(root, otherfile)
|
|
|
|
data, err := ioutil.ReadFile(fileabspath)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
reg := regexp.MustCompile("{{[ ]*define[ ]+\"([^\"]+)\"")
|
|
|
|
allsub := reg.FindAllStringSubmatch(string(data), -1)
|
|
|
|
for _, sub := range allsub {
|
|
|
|
if len(sub) == 2 && sub[1] == m[1] {
|
2013-09-26 10:33:01 +00:00
|
|
|
var submods1 [][]string
|
2013-10-30 15:02:53 +00:00
|
|
|
t, submods1, err = getTplDeep(root, otherfile, "", t)
|
2013-09-13 09:41:31 +00:00
|
|
|
if err != nil {
|
|
|
|
Trace("template parse file err:", err)
|
2013-09-26 10:33:01 +00:00
|
|
|
} else if submods1 != nil && len(submods1) > 0 {
|
|
|
|
t, err = _getTemplate(t, root, submods1, others...)
|
2013-09-13 09:41:31 +00:00
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-09-13 05:57:40 +00:00
|
|
|
}
|
2013-09-13 09:41:31 +00:00
|
|
|
|
2013-09-13 05:57:40 +00:00
|
|
|
}
|
2013-09-12 09:20:32 +00:00
|
|
|
return
|
|
|
|
}
|
2013-10-29 14:35:04 +00:00
|
|
|
|
|
|
|
// go1.2 added template funcs. begin
|
|
|
|
var (
|
|
|
|
errBadComparisonType = errors.New("invalid type for comparison")
|
|
|
|
errBadComparison = errors.New("incompatible types for comparison")
|
|
|
|
errNoComparison = errors.New("missing argument for comparison")
|
|
|
|
)
|
|
|
|
|
|
|
|
type kind int
|
|
|
|
|
|
|
|
const (
|
|
|
|
invalidKind kind = iota
|
|
|
|
boolKind
|
|
|
|
complexKind
|
|
|
|
intKind
|
|
|
|
floatKind
|
|
|
|
integerKind
|
|
|
|
stringKind
|
|
|
|
uintKind
|
|
|
|
)
|
|
|
|
|
|
|
|
func basicKind(v reflect.Value) (kind, error) {
|
|
|
|
switch v.Kind() {
|
|
|
|
case reflect.Bool:
|
|
|
|
return boolKind, nil
|
|
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
|
|
return intKind, nil
|
|
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
|
|
|
return uintKind, nil
|
|
|
|
case reflect.Float32, reflect.Float64:
|
|
|
|
return floatKind, nil
|
|
|
|
case reflect.Complex64, reflect.Complex128:
|
|
|
|
return complexKind, nil
|
|
|
|
case reflect.String:
|
|
|
|
return stringKind, nil
|
|
|
|
}
|
|
|
|
return invalidKind, errBadComparisonType
|
|
|
|
}
|
|
|
|
|
|
|
|
// eq evaluates the comparison a == b || a == c || ...
|
|
|
|
func eq(arg1 interface{}, arg2 ...interface{}) (bool, error) {
|
|
|
|
v1 := reflect.ValueOf(arg1)
|
|
|
|
k1, err := basicKind(v1)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if len(arg2) == 0 {
|
|
|
|
return false, errNoComparison
|
|
|
|
}
|
|
|
|
for _, arg := range arg2 {
|
|
|
|
v2 := reflect.ValueOf(arg)
|
|
|
|
k2, err := basicKind(v2)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if k1 != k2 {
|
|
|
|
return false, errBadComparison
|
|
|
|
}
|
|
|
|
truth := false
|
|
|
|
switch k1 {
|
|
|
|
case boolKind:
|
|
|
|
truth = v1.Bool() == v2.Bool()
|
|
|
|
case complexKind:
|
|
|
|
truth = v1.Complex() == v2.Complex()
|
|
|
|
case floatKind:
|
|
|
|
truth = v1.Float() == v2.Float()
|
|
|
|
case intKind:
|
|
|
|
truth = v1.Int() == v2.Int()
|
|
|
|
case stringKind:
|
|
|
|
truth = v1.String() == v2.String()
|
|
|
|
case uintKind:
|
|
|
|
truth = v1.Uint() == v2.Uint()
|
|
|
|
default:
|
|
|
|
panic("invalid kind")
|
|
|
|
}
|
|
|
|
if truth {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ne evaluates the comparison a != b.
|
|
|
|
func ne(arg1, arg2 interface{}) (bool, error) {
|
|
|
|
// != is the inverse of ==.
|
|
|
|
equal, err := eq(arg1, arg2)
|
|
|
|
return !equal, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// lt evaluates the comparison a < b.
|
|
|
|
func lt(arg1, arg2 interface{}) (bool, error) {
|
|
|
|
v1 := reflect.ValueOf(arg1)
|
|
|
|
k1, err := basicKind(v1)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
v2 := reflect.ValueOf(arg2)
|
|
|
|
k2, err := basicKind(v2)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if k1 != k2 {
|
|
|
|
return false, errBadComparison
|
|
|
|
}
|
|
|
|
truth := false
|
|
|
|
switch k1 {
|
|
|
|
case boolKind, complexKind:
|
|
|
|
return false, errBadComparisonType
|
|
|
|
case floatKind:
|
|
|
|
truth = v1.Float() < v2.Float()
|
|
|
|
case intKind:
|
|
|
|
truth = v1.Int() < v2.Int()
|
|
|
|
case stringKind:
|
|
|
|
truth = v1.String() < v2.String()
|
|
|
|
case uintKind:
|
|
|
|
truth = v1.Uint() < v2.Uint()
|
|
|
|
default:
|
|
|
|
panic("invalid kind")
|
|
|
|
}
|
|
|
|
return truth, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// le evaluates the comparison <= b.
|
|
|
|
func le(arg1, arg2 interface{}) (bool, error) {
|
|
|
|
// <= is < or ==.
|
|
|
|
lessThan, err := lt(arg1, arg2)
|
|
|
|
if lessThan || err != nil {
|
|
|
|
return lessThan, err
|
|
|
|
}
|
|
|
|
return eq(arg1, arg2)
|
|
|
|
}
|
|
|
|
|
|
|
|
// gt evaluates the comparison a > b.
|
|
|
|
func gt(arg1, arg2 interface{}) (bool, error) {
|
|
|
|
// > is the inverse of <=.
|
|
|
|
lessOrEqual, err := le(arg1, arg2)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return !lessOrEqual, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ge evaluates the comparison a >= b.
|
|
|
|
func ge(arg1, arg2 interface{}) (bool, error) {
|
|
|
|
// >= is the inverse of <.
|
|
|
|
lessThan, err := lt(arg1, arg2)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return !lessThan, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// go1.2 added template funcs. end
|