mirror of
				https://github.com/Theodor-Springmann-Stiftung/lenz-web.git
				synced 2025-10-30 01:35:32 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			122 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package templating
 | |
| 
 | |
| import (
 | |
| 	"html/template"
 | |
| 	"io/fs"
 | |
| 	"os"
 | |
| 	"strings"
 | |
| 	"sync"
 | |
| 
 | |
| 	"github.com/yalue/merged_fs"
 | |
| )
 | |
| 
 | |
| // INFO: this ist thread-safe and safe to call in a handler or middleware
 | |
| type TemplateRegistry struct {
 | |
| 	routesFS fs.FS
 | |
| 	once     sync.Once
 | |
| 	// INFO: Template & cache keys are directory routing paths, with '/' as root
 | |
| 	// INFO: we don't need a mutex here since this is set in Load() protected by Once().
 | |
| 	templates map[string]TemplateContext
 | |
| 	cache     sync.Map
 | |
| }
 | |
| 
 | |
| func NewTemplateRegistry(routes fs.FS) *TemplateRegistry {
 | |
| 	return &TemplateRegistry{
 | |
| 		routesFS: routes,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // INFO: This returns a new TemplateRegistry with the new fs added to the existing fs,
 | |
| // merging with the existing FS, possibly overwriting existing files.
 | |
| func (r *TemplateRegistry) Register(path string, fs fs.FS) *TemplateRegistry {
 | |
| 	return NewTemplateRegistry(merged_fs.MergeMultiple(fs, r.routesFS))
 | |
| }
 | |
| 
 | |
| func (r *TemplateRegistry) Reset() *TemplateRegistry {
 | |
| 	return NewTemplateRegistry(r.routesFS)
 | |
| }
 | |
| 
 | |
| func (r *TemplateRegistry) Load() error {
 | |
| 	var outer error
 | |
| 	r.once.Do(func() {
 | |
| 		err := r.load()
 | |
| 		if err != nil {
 | |
| 			outer = err
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	return outer
 | |
| }
 | |
| 
 | |
| // TODO: Throw errors
 | |
| // TODO: what if there is no template in the directory above?
 | |
| // What if a certain path is or should uncallable since it has no index or body?
 | |
| func (r *TemplateRegistry) load() error {
 | |
| 	templates := make(map[string]TemplateContext)
 | |
| 	fs.WalkDir(r.routesFS, ".", func(path string, d fs.DirEntry, err error) error {
 | |
| 		if !d.IsDir() {
 | |
| 			return nil
 | |
| 		}
 | |
| 
 | |
| 		url := FSPathToPath(path)
 | |
| 		tc := NewTemplateContext(url)
 | |
| 
 | |
| 		if path != "." {
 | |
| 			pathelem := strings.Split(path, string(os.PathSeparator))
 | |
| 			pathabove := strings.Join(pathelem[:len(pathelem)-1], string(os.PathSeparator))
 | |
| 			pathabove = FSPathToPath(pathabove)
 | |
| 
 | |
| 			global, ok := templates[pathabove]
 | |
| 			if ok {
 | |
| 				tc.SetGlobals(global.Globals())
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		tc.Parse(r.routesFS)
 | |
| 
 | |
| 		templates[url] = tc
 | |
| 
 | |
| 		return nil
 | |
| 	})
 | |
| 
 | |
| 	r.templates = templates
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // This function takes a template (typically a layout) and adds all the templates of
 | |
| // a given directory path to it. This is useful for adding a layout to a template.
 | |
| func (r *TemplateRegistry) Add(path string, t *template.Template, funcmap *template.FuncMap) error {
 | |
| 	temp, ok := r.cache.Load(path)
 | |
| 	if !ok {
 | |
| 		r.Load()
 | |
| 		tc, ok := r.templates[path]
 | |
| 		if !ok {
 | |
| 			return NewError(NoTemplateError, path)
 | |
| 		}
 | |
| 
 | |
| 		template, err := tc.Template(r.routesFS, funcmap)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 
 | |
| 		r.cache.Store(path, template)
 | |
| 
 | |
| 		return r.Add(path, t, funcmap)
 | |
| 	}
 | |
| 
 | |
| 	casted := temp.(*template.Template)
 | |
| 	for _, st := range casted.Templates() {
 | |
| 		_, err := t.AddParseTree(st.Name(), st.Tree)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // TODO: get for a specific component
 | |
| func (r *TemplateRegistry) Get(path string) error {
 | |
| 	return nil
 | |
| }
 | 
