1
0
mirror of https://github.com/astaxie/beego.git synced 2024-11-23 00:30:56 +00:00

add template test and cache module

This commit is contained in:
astaxie 2013-03-14 00:14:09 +08:00
parent db329a48e7
commit 3b89071360
4 changed files with 205 additions and 17 deletions

View File

@ -13,6 +13,8 @@ import (
"strconv" "strconv"
) )
const VERSION = "0.0.3"
var ( var (
BeeApp *App BeeApp *App
AppName string AppName string
@ -218,5 +220,6 @@ func Run() {
GlobalSessions, _ = session.NewManager(SessionProvider, SessionName, SessionGCMaxLifetime) GlobalSessions, _ = session.NewManager(SessionProvider, SessionName, SessionGCMaxLifetime)
go GlobalSessions.GC() go GlobalSessions.GC()
} }
BuildTemplate(ViewsPath)
BeeApp.Run() BeeApp.Run()
} }

132
cache.go Normal file
View File

@ -0,0 +1,132 @@
package beego
import (
"errors"
"fmt"
"strconv"
"time"
)
const (
Kb = 1024
Mb = 1024 * 1024
Gb = 1024 * 1024 * 1024
)
var (
DefaultEvery int = 60 // 1 minute
)
var (
InvalidCacheItem = errors.New("invalid cache item")
ItemIsDirectory = errors.New("can't cache a directory")
ItemNotInCache = errors.New("item not in cache")
ItemTooLarge = errors.New("item too large for cache")
WriteIncomplete = errors.New("incomplete write of cache item")
)
type BeeItem struct {
val interface{}
Lastaccess time.Time
expired int
}
func (itm *BeeItem) Access() interface{} {
itm.Lastaccess = time.Now()
return itm.val
}
type BeeCache struct {
dur time.Duration
items map[string]*BeeItem
Every int // Run an expiration check Every seconds
}
// NewDefaultCache returns a new FileCache with sane defaults.
func NewBeeCache() *BeeCache {
cache := BeeCache{time.Since(time.Now()),
nil,
DefaultEvery}
return &cache
}
func (bc *BeeCache) Get(name string) interface{} {
itm, ok := bc.items[name]
if !ok {
return nil
}
return itm.Access()
}
func (bc *BeeCache) Put(name string, value interface{}, expired int) error {
t := BeeItem{val: value, Lastaccess: time.Now(), expired: expired}
if bc.IsExist(name) {
return errors.New("the key is exist")
} else {
bc.items[name] = &t
}
return nil
}
func (bc *BeeCache) Delete(name string) (ok bool, err error) {
_, ok = bc.items[name]
if !ok {
return
}
delete(bc.items, name)
_, valid := bc.items[name]
if valid {
ok = false
}
return
}
func (bc *BeeCache) IsExist(name string) bool {
_, ok := bc.items[name]
return ok
}
// Start activates the file cache; it will
func (bc *BeeCache) Start() error {
dur, err := time.ParseDuration(fmt.Sprintf("%ds", bc.Every))
if err != nil {
return err
}
bc.dur = dur
bc.items = make(map[string]*BeeItem, 0)
go bc.vaccuum()
return nil
}
func (bc *BeeCache) vaccuum() {
if bc.Every < 1 {
return
}
for {
<-time.After(time.Duration(bc.dur))
if bc.items == nil {
return
}
for name, _ := range bc.items {
if bc.item_expired(name) {
delete(bc.items, name)
}
}
}
}
// item_expired returns true if an item is expired.
func (bc *BeeCache) item_expired(name string) bool {
itm, ok := bc.items[name]
if !ok {
return true
}
dur := time.Now().Sub(itm.Lastaccess)
sec, err := strconv.Atoi(fmt.Sprintf("%0.0f", dur.Seconds()))
if err != nil {
return true
} else if sec >= itm.expired {
return true
}
return false
}

View File

@ -15,7 +15,6 @@ import (
type Controller struct { type Controller struct {
Ctx *Context Ctx *Context
Tpl *template.Template
Data map[interface{}]interface{} Data map[interface{}]interface{}
ChildName string ChildName string
TplNames string TplNames string
@ -39,8 +38,6 @@ type ControllerInterface interface {
func (c *Controller) Init(ctx *Context, cn string) { func (c *Controller) Init(ctx *Context, cn string) {
c.Data = make(map[interface{}]interface{}) c.Data = make(map[interface{}]interface{})
c.Tpl = template.New(cn + ctx.Request.Method)
c.Tpl = c.Tpl.Funcs(beegoTplFuncMap)
c.Layout = "" c.Layout = ""
c.TplNames = "" c.TplNames = ""
c.ChildName = cn c.ChildName = cn
@ -91,17 +88,14 @@ func (c *Controller) Render() error {
if c.TplNames == "" { if c.TplNames == "" {
c.TplNames = c.ChildName + "/" + c.Ctx.Request.Method + "." + c.TplExt c.TplNames = c.ChildName + "/" + c.Ctx.Request.Method + "." + c.TplExt
} }
t, err := c.Tpl.ParseFiles(path.Join(ViewsPath, c.TplNames), path.Join(ViewsPath, c.Layout))
if err != nil {
Trace("template ParseFiles err:", err)
}
_, file := path.Split(c.TplNames) _, file := path.Split(c.TplNames)
subdir := path.Dir(c.TplNames)
newbytes := bytes.NewBufferString("") newbytes := bytes.NewBufferString("")
t.ExecuteTemplate(newbytes, file, c.Data) BeeTemplates[subdir].ExecuteTemplate(newbytes, file, c.Data)
tplcontent, _ := ioutil.ReadAll(newbytes) tplcontent, _ := ioutil.ReadAll(newbytes)
c.Data["LayoutContent"] = template.HTML(string(tplcontent)) c.Data["LayoutContent"] = template.HTML(string(tplcontent))
_, file = path.Split(c.Layout) _, file = path.Split(c.Layout)
err = t.ExecuteTemplate(c.Ctx.ResponseWriter, file, c.Data) err := BeeTemplates[subdir].ExecuteTemplate(c.Ctx.ResponseWriter, file, c.Data)
if err != nil { if err != nil {
Trace("template Execute err:", err) Trace("template Execute err:", err)
} }
@ -109,12 +103,9 @@ func (c *Controller) Render() error {
if c.TplNames == "" { if c.TplNames == "" {
c.TplNames = c.ChildName + "/" + c.Ctx.Request.Method + "." + c.TplExt c.TplNames = c.ChildName + "/" + c.Ctx.Request.Method + "." + c.TplExt
} }
t, err := c.Tpl.ParseFiles(path.Join(ViewsPath, c.TplNames))
if err != nil {
Trace("template ParseFiles err:", err)
}
_, file := path.Split(c.TplNames) _, file := path.Split(c.TplNames)
err = t.ExecuteTemplate(c.Ctx.ResponseWriter, file, c.Data) subdir := path.Dir(c.TplNames)
err := BeeTemplates[subdir].ExecuteTemplate(c.Ctx.ResponseWriter, file, c.Data)
if err != nil { if err != nil {
Trace("template Execute err:", err) Trace("template Execute err:", err)
} }

View File

@ -3,18 +3,27 @@ package beego
//@todo add template funcs //@todo add template funcs
import ( import (
"fmt"
"errors" "errors"
"fmt"
"github.com/russross/blackfriday" "github.com/russross/blackfriday"
"html/template" "html/template"
"os"
"path"
"path/filepath"
"strings" "strings"
"time" "time"
) )
var beegoTplFuncMap template.FuncMap var (
beegoTplFuncMap template.FuncMap
BeeTemplates map[string]*template.Template
BeeTemplateExt string
)
func init() { func init() {
BeeTemplates = make(map[string]*template.Template)
beegoTplFuncMap = make(template.FuncMap) beegoTplFuncMap = make(template.FuncMap)
BeeTemplateExt = "tpl"
beegoTplFuncMap["markdown"] = MarkDown beegoTplFuncMap["markdown"] = MarkDown
beegoTplFuncMap["dateformat"] = DateFormat beegoTplFuncMap["dateformat"] = DateFormat
beegoTplFuncMap["date"] = Date beegoTplFuncMap["date"] = Date
@ -92,3 +101,56 @@ func AddFuncMap(key string, funname interface{}) error {
beegoTplFuncMap[key] = funname beegoTplFuncMap[key] = funname
return nil 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
}
if f.IsDir() {
return nil
} else if (f.Mode() & os.ModeSymlink) > 0 {
return nil
} else {
if strings.HasSuffix(paths, BeeTemplateExt) {
a := []byte(paths)
a = a[len([]byte(self.root)):]
subdir := path.Dir(strings.TrimLeft(string(a), "/"))
if _, ok := self.files[subdir]; ok {
self.files[subdir] = append(self.files[subdir], paths)
} else {
m := make([]string, 1)
m[0] = paths
self.files[subdir] = m
}
}
}
return nil
}
func SetGlobalTemplateExt(ext string) {
BeeTemplateExt = ext
}
func BuildTemplate(dir string) error {
self := templatefile{
root: dir,
files: make(map[string][]string),
}
err := filepath.Walk(dir, func(path string, f os.FileInfo, err error) error {
return self.visit(path, f, err)
})
if err != nil {
fmt.Printf("filepath.Walk() returned %v\n", err)
return err
}
for k, v := range self.files {
BeeTemplates[k] = template.Must(template.New("beegoTemplate" + k).Funcs(beegoTplFuncMap).ParseFiles(v...))
}
return nil
}