2019-06-22 20:34:19 +02:00
|
|
|
package view
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path"
|
2020-07-06 20:40:40 +02:00
|
|
|
"path/filepath"
|
2019-06-22 20:34:19 +02:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
|
2019-10-25 00:27:02 +02:00
|
|
|
"github.com/kataras/iris/v12/context"
|
2019-06-22 20:34:19 +02:00
|
|
|
|
2020-07-06 20:40:40 +02:00
|
|
|
"github.com/CloudyKit/jet/v4"
|
2019-06-22 20:34:19 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
const jetEngineName = "jet"
|
|
|
|
|
|
|
|
// JetEngine is the jet template parser's view engine.
|
|
|
|
type JetEngine struct {
|
|
|
|
directory string
|
|
|
|
extension string
|
|
|
|
// physical system files or app-embedded, see `Binary(..., ...)`. Defaults to file system on initialization.
|
|
|
|
loader jet.Loader
|
|
|
|
|
|
|
|
developmentMode bool
|
|
|
|
|
|
|
|
// The Set is the `*jet.Set`, exported to offer any custom capabilities that jet users may want.
|
|
|
|
// Available after `Load`.
|
|
|
|
Set *jet.Set
|
|
|
|
|
|
|
|
// Note that global vars and functions are set in a single spot on the jet parser.
|
|
|
|
// If AddFunc or AddVar called before `Load` then these will be set here to be used via `Load` and clear.
|
|
|
|
vars map[string]interface{}
|
2019-08-09 07:24:58 +02:00
|
|
|
|
2020-07-06 20:40:40 +02:00
|
|
|
jetDataContextKey string
|
2019-06-22 20:34:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ Engine = (*JetEngine)(nil)
|
|
|
|
|
|
|
|
// jet library does not export or give us any option to modify them via Set
|
|
|
|
// (unless we parse the files by ourselves but this is not a smart choice).
|
|
|
|
var jetExtensions = [...]string{
|
|
|
|
".html.jet",
|
|
|
|
".jet.html",
|
|
|
|
".jet",
|
|
|
|
}
|
|
|
|
|
|
|
|
// Jet creates and returns a new jet view engine.
|
|
|
|
func Jet(directory, extension string) *JetEngine {
|
2019-10-24 17:57:05 +02:00
|
|
|
// if _, err := os.Stat(directory); os.IsNotExist(err) {
|
|
|
|
// panic(err)
|
|
|
|
// }
|
|
|
|
|
2019-06-22 20:34:19 +02:00
|
|
|
extOK := false
|
|
|
|
for _, ext := range jetExtensions {
|
|
|
|
if ext == extension {
|
|
|
|
extOK = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !extOK {
|
|
|
|
panic(fmt.Sprintf("%s extension is not a valid jet engine extension[%s]", extension, strings.Join(jetExtensions[0:], ", ")))
|
|
|
|
}
|
|
|
|
|
|
|
|
s := &JetEngine{
|
2020-07-06 20:40:40 +02:00
|
|
|
directory: directory,
|
|
|
|
extension: extension,
|
|
|
|
loader: jet.NewOSFileSystemLoader(directory),
|
|
|
|
jetDataContextKey: "_jet",
|
2019-06-22 20:34:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// String returns the name of this view engine, the "jet".
|
|
|
|
func (s *JetEngine) String() string {
|
|
|
|
return jetEngineName
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ext should return the final file extension which this view engine is responsible to render.
|
|
|
|
func (s *JetEngine) Ext() string {
|
|
|
|
return s.extension
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delims sets the action delimiters to the specified strings, to be used in
|
|
|
|
// templates. An empty delimiter stands for the
|
|
|
|
// corresponding default: {{ or }}.
|
|
|
|
// Should act before `Load` or `iris.Application#RegisterView`.
|
|
|
|
func (s *JetEngine) Delims(left, right string) *JetEngine {
|
|
|
|
s.Set.Delims(left, right)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// JetArguments is a type alias of `jet.Arguments`,
|
|
|
|
// can be used on `AddFunc$funcBody`.
|
|
|
|
type JetArguments = jet.Arguments
|
|
|
|
|
|
|
|
// AddFunc should adds a global function to the jet template set.
|
|
|
|
func (s *JetEngine) AddFunc(funcName string, funcBody interface{}) {
|
|
|
|
// if something like "urlpath" is registered.
|
|
|
|
if generalFunc, ok := funcBody.(func(string, ...interface{}) string); ok {
|
|
|
|
// jet, unlike others does not accept a func(string, ...interface{}) string,
|
|
|
|
// instead it wants:
|
|
|
|
// func(JetArguments) reflect.Value.
|
|
|
|
|
2020-02-05 11:36:21 +01:00
|
|
|
s.AddVar(funcName, jet.Func(func(args JetArguments) reflect.Value {
|
2019-06-22 20:34:19 +02:00
|
|
|
n := args.NumOfArguments()
|
|
|
|
if n == 0 { // no input, don't execute the function, panic instead.
|
|
|
|
panic(funcName + " expects one or more input arguments")
|
|
|
|
}
|
|
|
|
|
|
|
|
firstInput := args.Get(0).String()
|
|
|
|
|
|
|
|
if n == 1 { // if only the first argument is given.
|
|
|
|
return reflect.ValueOf(generalFunc(firstInput))
|
|
|
|
}
|
|
|
|
|
|
|
|
// if has variadic.
|
|
|
|
|
|
|
|
variadicN := n - 1
|
|
|
|
variadicInputs := make([]interface{}, variadicN) // except the first one.
|
|
|
|
|
|
|
|
for i := 0; i < variadicN; i++ {
|
|
|
|
variadicInputs[i] = args.Get(i + 1).Interface()
|
|
|
|
}
|
|
|
|
|
|
|
|
return reflect.ValueOf(generalFunc(firstInput, variadicInputs...))
|
2020-02-05 11:36:21 +01:00
|
|
|
}))
|
2019-06-22 20:34:19 +02:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if jetFunc, ok := funcBody.(jet.Func); !ok {
|
|
|
|
alternativeJetFunc, ok := funcBody.(func(JetArguments) reflect.Value)
|
|
|
|
if !ok {
|
|
|
|
panic(fmt.Sprintf("JetEngine.AddFunc: funcBody argument is not a type of func(JetArguments) reflect.Value. Got %T instead", funcBody))
|
|
|
|
}
|
|
|
|
|
|
|
|
s.AddVar(funcName, jet.Func(alternativeJetFunc))
|
|
|
|
} else {
|
|
|
|
s.AddVar(funcName, jetFunc)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddVar adds a global variable to the jet template set.
|
|
|
|
func (s *JetEngine) AddVar(key string, value interface{}) {
|
|
|
|
if s.Set != nil {
|
|
|
|
s.Set.AddGlobal(key, value)
|
|
|
|
} else {
|
|
|
|
if s.vars == nil {
|
|
|
|
s.vars = make(map[string]interface{})
|
|
|
|
}
|
|
|
|
s.vars[key] = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reload if setted to true the templates are reloading on each render,
|
|
|
|
// use it when you're in development and you're boring of restarting
|
|
|
|
// the whole app when you edit a template file.
|
|
|
|
//
|
|
|
|
// Note that if `true` is passed then only one `View -> ExecuteWriter` will be render each time,
|
|
|
|
// not safe concurrent access across clients, use it only on development state.
|
|
|
|
func (s *JetEngine) Reload(developmentMode bool) *JetEngine {
|
|
|
|
s.developmentMode = developmentMode
|
|
|
|
if s.Set != nil {
|
|
|
|
s.Set.SetDevelopmentMode(developmentMode)
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetLoader can be used when the caller wants to use something like
|
|
|
|
// multi.Loader or httpfs.Loader of the jet subpackages,
|
|
|
|
// overrides any previous loader may set by `Binary` or the default.
|
|
|
|
// Should act before `Load` or `iris.Application#RegisterView`.
|
|
|
|
func (s *JetEngine) SetLoader(loader jet.Loader) *JetEngine {
|
|
|
|
s.loader = loader
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// Binary optionally, use it when template files are distributed
|
|
|
|
// inside the app executable (.go generated files).
|
|
|
|
//
|
|
|
|
// The assetFn and namesFn can come from the go-bindata library.
|
|
|
|
// Should act before `Load` or `iris.Application#RegisterView`.
|
|
|
|
func (s *JetEngine) Binary(assetFn func(name string) ([]byte, error), assetNames func() []string) *JetEngine {
|
|
|
|
// embedded.
|
|
|
|
vdir := s.directory
|
|
|
|
|
|
|
|
if vdir[0] == '.' {
|
|
|
|
vdir = vdir[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
// second check for /something, (or ./something if we had dot on 0 it will be removed)
|
|
|
|
if vdir[0] == '/' || vdir[0] == os.PathSeparator {
|
|
|
|
vdir = vdir[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for trailing slashes because new users may be do that by mistake
|
|
|
|
// although all examples are showing the correct way but you never know
|
|
|
|
// i.e "./assets/" is not correct, if was inside "./assets".
|
|
|
|
// remove last "/".
|
|
|
|
if trailingSlashIdx := len(vdir) - 1; vdir[trailingSlashIdx] == '/' {
|
|
|
|
vdir = vdir[0:trailingSlashIdx]
|
|
|
|
}
|
|
|
|
|
|
|
|
namesSlice := assetNames()
|
|
|
|
names := make(map[string]struct{})
|
|
|
|
for _, name := range namesSlice {
|
|
|
|
if !strings.HasPrefix(name, vdir) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
extOK := false
|
|
|
|
fileExt := path.Ext(name)
|
|
|
|
for _, ext := range jetExtensions {
|
|
|
|
if ext == fileExt {
|
|
|
|
extOK = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !extOK {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
names[name] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(names) == 0 {
|
|
|
|
panic("JetEngine.Binary: no embedded files found in directory: " + vdir)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.loader = &embeddedLoader{
|
|
|
|
vdir: vdir,
|
|
|
|
asset: assetFn,
|
|
|
|
names: names,
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
type (
|
|
|
|
embeddedLoader struct {
|
|
|
|
vdir string
|
|
|
|
asset func(name string) ([]byte, error)
|
|
|
|
names map[string]struct{}
|
|
|
|
}
|
|
|
|
embeddedFile struct {
|
|
|
|
contents []byte // the contents are NOT consumed.
|
|
|
|
readen int64
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
_ jet.Loader = (*embeddedLoader)(nil)
|
|
|
|
_ io.ReadCloser = (*embeddedFile)(nil)
|
|
|
|
)
|
|
|
|
|
|
|
|
func (f *embeddedFile) Close() error { return nil }
|
|
|
|
func (f *embeddedFile) Read(p []byte) (int, error) {
|
|
|
|
if f.readen >= int64(len(f.contents)) {
|
|
|
|
return 0, io.EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
n := copy(p, f.contents[f.readen:])
|
|
|
|
f.readen += int64(n)
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open opens a file from OS file system.
|
|
|
|
func (l *embeddedLoader) Open(name string) (io.ReadCloser, error) {
|
2020-07-06 20:40:40 +02:00
|
|
|
name = path.Join(l.vdir, filepath.ToSlash(name))
|
2019-06-22 20:34:19 +02:00
|
|
|
contents, err := l.asset(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &embeddedFile{
|
|
|
|
contents: contents,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exists checks if the template name exists by walking the list of template paths
|
|
|
|
// returns string with the full path of the template and bool true if the template file was found
|
|
|
|
func (l *embeddedLoader) Exists(name string) (string, bool) {
|
2020-07-06 20:40:40 +02:00
|
|
|
name = path.Join(l.vdir, filepath.ToSlash(name))
|
|
|
|
|
|
|
|
if _, ok := l.names[name]; ok {
|
|
|
|
return name, true
|
2019-06-22 20:34:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return "", false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load should load the templates from a physical system directory or by an embedded one (assets/go-bindata).
|
|
|
|
func (s *JetEngine) Load() error {
|
|
|
|
s.Set = jet.NewHTMLSetLoader(s.loader)
|
|
|
|
s.Set.SetDevelopmentMode(s.developmentMode)
|
|
|
|
|
|
|
|
if s.vars != nil {
|
|
|
|
for key, value := range s.vars {
|
|
|
|
s.Set.AddGlobal(key, value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Note that, unlike the rest of template engines implementations,
|
|
|
|
// we don't call the Set.GetTemplate to parse the templates,
|
|
|
|
// we let it to the jet template parser itself which does that at serve-time and caches each template by itself.
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type (
|
|
|
|
// JetRuntimeVars is a type alias for `jet.VarMap`.
|
|
|
|
// Can be used at `AddJetRuntimeVars/JetEngine.AddRuntimeVars`
|
|
|
|
// to set a runtime variable ${name} to the executing template.
|
|
|
|
JetRuntimeVars = jet.VarMap
|
|
|
|
|
|
|
|
// JetRuntime is a type alias of `jet.Runtime`,
|
|
|
|
// can be used on RuntimeVariable input function.
|
|
|
|
JetRuntime = jet.Runtime
|
|
|
|
)
|
|
|
|
|
|
|
|
// JetRuntimeVarsContextKey is the Iris Context key to keep any custom jet runtime variables.
|
|
|
|
// See `AddJetRuntimeVars` package-level function and `JetEngine.AddRuntimeVars` method.
|
|
|
|
const JetRuntimeVarsContextKey = "iris.jetvarmap"
|
|
|
|
|
|
|
|
// AddJetRuntimeVars sets or inserts runtime jet variables through the Iris Context.
|
|
|
|
// This gives the ability to add runtime variables from different handlers in the request chain,
|
|
|
|
// something that the jet template parser does not offer at all.
|
|
|
|
//
|
|
|
|
// Usage: view.AddJetRuntimeVars(ctx, view.JetRuntimeVars{...}).
|
|
|
|
// See `JetEngine.AddRuntimeVars` too.
|
2020-07-10 22:21:09 +02:00
|
|
|
func AddJetRuntimeVars(ctx *context.Context, jetVarMap JetRuntimeVars) {
|
2019-06-22 20:34:19 +02:00
|
|
|
if v := ctx.Values().Get(JetRuntimeVarsContextKey); v != nil {
|
|
|
|
if vars, ok := v.(JetRuntimeVars); ok {
|
|
|
|
for key, value := range jetVarMap {
|
|
|
|
vars[key] = value
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Values().Set(JetRuntimeVarsContextKey, jetVarMap)
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddRuntimeVars sets or inserts runtime jet variables through the Iris Context.
|
|
|
|
// This gives the ability to add runtime variables from different handlers in the request chain,
|
|
|
|
// something that the jet template parser does not offer at all.
|
|
|
|
//
|
|
|
|
// Usage: view.AddJetRuntimeVars(ctx, view.JetRuntimeVars{...}).
|
|
|
|
// See `view.AddJetRuntimeVars` if package-level access is more meanful to the code flow.
|
2020-07-10 22:21:09 +02:00
|
|
|
func (s *JetEngine) AddRuntimeVars(ctx *context.Context, vars JetRuntimeVars) {
|
2019-06-22 20:34:19 +02:00
|
|
|
AddJetRuntimeVars(ctx, vars)
|
|
|
|
}
|
|
|
|
|
2019-08-09 07:24:58 +02:00
|
|
|
// ExecuteWriter should execute a template by its filename with an optional layout and bindingData.
|
|
|
|
func (s *JetEngine) ExecuteWriter(w io.Writer, filename string, layout string, bindingData interface{}) error {
|
|
|
|
tmpl, err := s.Set.GetTemplate(filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var vars JetRuntimeVars
|
|
|
|
|
2020-07-10 22:21:09 +02:00
|
|
|
if ctx, ok := w.(*context.Context); ok {
|
2019-08-09 07:24:58 +02:00
|
|
|
runtimeVars := ctx.Values().Get(JetRuntimeVarsContextKey)
|
|
|
|
if runtimeVars != nil {
|
|
|
|
if jetVars, ok := runtimeVars.(JetRuntimeVars); ok {
|
|
|
|
vars = jetVars
|
|
|
|
}
|
|
|
|
}
|
2020-07-06 20:40:40 +02:00
|
|
|
|
|
|
|
if v := ctx.Values().Get(s.jetDataContextKey); v != nil {
|
|
|
|
if bindingData == nil {
|
|
|
|
// if bindingData is nil, try to fill them by context key (a middleware can set data).
|
|
|
|
bindingData = v
|
|
|
|
} else if m, ok := bindingData.(context.Map); ok {
|
|
|
|
// else if bindingData are passed to App/Context.View
|
|
|
|
// and it's map try to fill with the new values passed from a middleware.
|
|
|
|
if mv, ok := v.(context.Map); ok {
|
|
|
|
for key, value := range mv {
|
|
|
|
m[key] = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-08-09 07:24:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if bindingData == nil {
|
|
|
|
return tmpl.Execute(w, vars, nil)
|
|
|
|
}
|
|
|
|
|
2020-02-05 11:36:21 +01:00
|
|
|
if vars == nil {
|
|
|
|
vars = make(JetRuntimeVars)
|
2019-08-09 07:24:58 +02:00
|
|
|
}
|
|
|
|
|
2020-07-06 20:40:40 +02:00
|
|
|
/* fixed on jet v4.0.0, so no need of this:
|
2019-08-09 07:24:58 +02:00
|
|
|
if m, ok := bindingData.(context.Map); ok {
|
2020-02-05 11:36:21 +01:00
|
|
|
var jetData interface{}
|
2019-08-09 07:24:58 +02:00
|
|
|
for k, v := range m {
|
2020-07-06 20:40:40 +02:00
|
|
|
if k == s.jetDataContextKey {
|
2020-02-05 11:36:21 +01:00
|
|
|
jetData = v
|
2019-08-09 07:24:58 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-02-05 11:36:21 +01:00
|
|
|
if value, ok := v.(reflect.Value); ok {
|
|
|
|
vars[k] = value
|
|
|
|
} else {
|
|
|
|
vars[k] = reflect.ValueOf(v)
|
2019-08-09 07:24:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-05 11:36:21 +01:00
|
|
|
if jetData != nil {
|
|
|
|
bindingData = jetData
|
|
|
|
}
|
2020-07-06 20:40:40 +02:00
|
|
|
}*/
|
2019-06-22 20:34:19 +02:00
|
|
|
|
|
|
|
return tmpl.Execute(w, vars, bindingData)
|
|
|
|
}
|