package templating
import (
	"errors"
	"html/template"
	"io"
	"io/fs"
	"log"
	"net/http"
	"strings"
	"sync"
	"github.com/Theodor-Springmann-Stiftung/musenalm/dbmodels"
	"github.com/Theodor-Springmann-Stiftung/musenalm/helpers/functions"
	"github.com/pocketbase/pocketbase/core"
	"golang.org/x/net/websocket"
)
const (
	ASSETS_URL_PREFIX = "/assets"
	RELOAD_TEMPLATE   = `
`
)
type Engine struct {
	regmu  *sync.Mutex
	debug  bool
	ws     *WsServer
	onceWS sync.Once
	// NOTE: LayoutRegistry and TemplateRegistry have their own syncronization & cache and do not require a mutex here
	LayoutRegistry   *LayoutRegistry
	TemplateRegistry *TemplateRegistry
	mu         *sync.Mutex
	FuncMap    template.FuncMap
	GlobalData map[string]any
}
// INFO: We pass the app here to be able to access the config and other data for functions
// which also means we must reload the engine if the app changes
func NewEngine(layouts, templates *fs.FS) *Engine {
	e := Engine{
		regmu:            &sync.Mutex{},
		mu:               &sync.Mutex{},
		LayoutRegistry:   NewLayoutRegistry(*layouts),
		TemplateRegistry: NewTemplateRegistry(*templates),
		FuncMap:          make(template.FuncMap),
		GlobalData:       make(map[string]any),
	}
	e.funcs()
	return &e
}
func (e *Engine) Debug() {
	e.debug = true
	e.onceWS.Do(func() {
		e.ws = NewWsServer()
		go e.startWsServerOnPort9000()
	})
}
func (e *Engine) startWsServerOnPort9000() {
	// We'll create a basic default mux here and mount /pb/reload
	mux := http.NewServeMux()
	mux.Handle("/pb/reload", websocket.Handler(e.ws.Handler))
	log.Println("[Engine Debug] Starting separate WebSocket server on :9000 for live reload...")
	if err := http.ListenAndServe(":9000", mux); err != nil {
		log.Println("[Engine Debug] WebSocket server error:", err)
	}
}
func (e *Engine) funcs() error {
	e.mu.Lock()
	e.mu.Unlock()
	// Passing HTML
	e.AddFunc("Safe", functions.Safe)
	// Creating an array or dict (to pass to a template)
	e.AddFunc("Arr", functions.Arr)
	e.AddFunc("Dict", functions.Dict)
	// Datatype Functions
	e.AddFunc("HasPrefix", strings.HasPrefix)
	e.AddFunc("Contains", functions.Contains)
	e.AddFunc("Add", functions.Add)
	e.AddFunc("Len", functions.Length)
	// String Functions
	e.AddFunc("Lower", functions.Lower)
	e.AddFunc("Upper", functions.Upper)
	e.AddFunc("First", functions.First)
	e.AddFunc("ReplaceSlashParen", functions.ReplaceSlashParen)
	e.AddFunc("ReplaceSlashParenSlash", functions.ReplaceSlashParenSlash)
	e.AddFunc("LinksAnnotation", functions.LinksAnnotation)
	// Time & Date Functions
	e.AddFunc("Today", functions.Today)
	e.AddFunc("GetMonth", functions.GetMonth)
	// TOC
	e.AddFunc("TOCFromHTML", functions.TOCFromHTML)
	// User Functions
	e.AddFunc("IsAdminOrEditor", dbmodels.IsAdminOrEditor)
	e.AddFunc("IsAdmin", dbmodels.IsAdmin)
	return nil
}
func (e *Engine) Globals(data map[string]any) {
	e.mu.Lock()
	defer e.mu.Unlock()
	if e.GlobalData == nil {
		e.GlobalData = data
	} else {
		for k, v := range data {
			(e.GlobalData)[k] = v
		}
	}
}
func (e *Engine) Load() {
	wg := sync.WaitGroup{}
	wg.Add(2)
	go func() {
		defer wg.Done()
		e.LayoutRegistry.Load()
	}()
	go func() {
		defer wg.Done()
		e.TemplateRegistry.Load()
	}()
	wg.Wait()
}
func (e *Engine) Reload() {
	e.regmu.Lock()
	defer e.regmu.Unlock()
	e.LayoutRegistry = e.LayoutRegistry.Reset()
	e.TemplateRegistry = e.TemplateRegistry.Reset()
	e.Load()
}
func (e *Engine) Refresh() {
	if e.debug && e.ws != nil {
		e.ws.BroadcastReload()
	}
}
// INFO: fn is a function that returns either one value or two values, the second one being an error
func (e *Engine) AddFunc(name string, fn any) {
	e.mu.Lock()
	defer e.mu.Unlock()
	e.FuncMap[name] = fn
}
func (e *Engine) AddFuncs(funcs map[string]any) {
	e.mu.Lock()
	defer e.mu.Unlock()
	for k, v := range funcs {
		e.FuncMap[k] = v
	}
}
func (e *Engine) Render(out io.Writer, path string, ld map[string]any, layout ...string) error {
	gd := e.GlobalData
	if ld == nil {
		ld = make(map[string]any)
	}
	// INFO: don't pollute the global data space
	for k, v := range gd {
		_, ok := ld[k]
		if !ok {
			ld[k] = v
		}
	}
	e.mu.Lock()
	defer e.mu.Unlock()
	e.regmu.Lock()
	defer e.regmu.Unlock()
	var l *template.Template
	if layout == nil || len(layout) == 0 {
		lay, err := e.LayoutRegistry.Default(&e.FuncMap)
		if err != nil {
			return err
		}
		l = lay
	} else {
		lay, err := e.LayoutRegistry.Layout(layout[0], &e.FuncMap)
		if err != nil {
			return err
		}
		l = lay
	}
	lay, err := l.Clone()
	if err != nil {
		return err
	}
	err = e.TemplateRegistry.Add(path, lay, &e.FuncMap)
	if err != nil {
		return err
	}
	err = lay.Execute(out, ld)
	if err != nil {
		return err
	}
	return nil
}
func (e *Engine) RenderToString(request *core.RequestEvent, ld map[string]any, path string, layout ...string) (string, error) {
	if ld == nil {
		ld = make(map[string]any)
	}
	r := NewRequest(request)
	ld["request"] = r.Data()
	var builder strings.Builder
	err := e.Render(&builder, path, ld, layout...)
	if err != nil {
		return "", e.Response500(request, err, ld)
	}
	tstring := builder.String()
	if e.debug {
		idx := strings.LastIndex(tstring, "