package view import ( "fmt" "io" "os" "path" "reflect" "strings" "github.com/kataras/iris/v12/context" "github.com/CloudyKit/jet" ) 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{} jetRangerRendererContextKey string } 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 { // if _, err := os.Stat(directory); os.IsNotExist(err) { // panic(err) // } 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{ directory: directory, extension: extension, loader: jet.NewOSFileSystemLoader(directory), } return s } // DisableViewDataTypeCheck accepts a context key name to use // to map the jet specific renderer and ranger over context's view data. // // If "jetDataContextKey" is not empty then `ExecuteWriter` will not check for // types to check if an element passed through `Context.ViewData` // contains a jet.Renderer or jet.Ranger or both. // Instead will map those with simple key data naming (faster). // Also it wont check if a value is already a reflect.Value (jet expects this type as values). // // Defaults to empty. func (s *JetEngine) DisableViewDataTypeCheck(jetRangerRendererContextKey string) *JetEngine { s.jetRangerRendererContextKey = jetRangerRendererContextKey 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. s.AddVar(funcName, func(args JetArguments) reflect.Value { 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...)) }) 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) { // name = path.Join(l.vdir, name) 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) { fileName := path.Join(l.vdir, name) if _, ok := l.names[fileName]; ok { return fileName, true } 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. func AddJetRuntimeVars(ctx context.Context, jetVarMap JetRuntimeVars) { 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. func (s *JetEngine) AddRuntimeVars(ctx context.Context, vars JetRuntimeVars) { AddJetRuntimeVars(ctx, vars) } type rangerAndRenderer struct { ranger jet.Ranger renderer jet.Renderer } func (rr rangerAndRenderer) Range() (reflect.Value, reflect.Value, bool) { return rr.ranger.Range() } func (rr rangerAndRenderer) Render(jetRuntime *jet.Runtime) { rr.renderer.Render(jetRuntime) } func rangerRenderer(bindingData interface{}) (interface{}, bool) { if ranger, ok := bindingData.(jet.Ranger); ok { // Externally fixes a BUG on the jet template parser: // eval.go#executeList(list *ListNode):NodeRange.isSet.getRanger(expression = st.evalPrimaryExpressionGroup) // which does not act the "ranger" as element, instead is converted to a value of struct, which makes a jet.Ranger func(*myStruct) Range... // not a compatible jet.Ranger. // getRanger(st.context) should work but author of the jet library is not currently available, // to allow a recommentation or a PR and I don't really want to vendor it because // some end-users may use the jet import path to pass things like Global Funcs and etc. // So to fix it (at least temporarily and only for ref Ranger) we ptr the ptr the "ranger", not the bindingData, and this may // have its downside because the same bindingData may be compatible with other node actions like range or custom Render // but we have no other way at the moment. The same problem exists on the `Renderer` too! // The solution below fixes the above issue but any fields of the struct are not available, // this is ok because most of the times if not always, the users of jet don't use fields on Ranger and custom Renderer inside the templates. if renderer, ok := bindingData.(jet.Renderer); ok { // this can make a Ranger and Renderer both work together, unlike the jet parser itself. return rangerAndRenderer{ranger, renderer}, true } return &ranger, true } if renderer, ok := bindingData.(jet.Renderer); ok { // Here the fields are not available but usually if completes the jet.Renderer no // fields are used in the template. return &renderer, true // see above ^. } return nil, false } // ExecuteWriter should execute a template by its filename with an optional layout and bindingData. // See `DisableViewDataTypeCheck` too. 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 if ctx, ok := w.(context.Context); ok { runtimeVars := ctx.Values().Get(JetRuntimeVarsContextKey) if runtimeVars != nil { if jetVars, ok := runtimeVars.(JetRuntimeVars); ok { vars = jetVars } } } if bindingData == nil { return tmpl.Execute(w, vars, nil) } jetRangerRenderer, ok := rangerRenderer(bindingData) if ok { return tmpl.Execute(w, vars, jetRangerRenderer) } if m, ok := bindingData.(context.Map); ok { for k, v := range m { if s.jetRangerRendererContextKey == "" { switch value := v.(type) { case jet.Ranger, jet.Renderer: jetRangerRenderer, _ = rangerRenderer(value) case reflect.Value: if vars == nil { vars = make(JetRuntimeVars) } // if it's already a reflect value. vars[k] = value default: if vars == nil { vars = make(JetRuntimeVars) } vars.Set(k, v) } continue } if k == s.jetRangerRendererContextKey { jetRangerRenderer = v continue } if vars == nil { vars = make(JetRuntimeVars) } vars.Set(k, v) } return tmpl.Execute(w, vars, jetRangerRenderer) } return tmpl.Execute(w, vars, bindingData) }