package accesslog

import (
	"bytes"
	"io"
	"text/template"
)

// Template is a Formatter.
// It's used to print the Log in a text/template way.
// The caller has full control over the printable result;
// certain fields can be ignored, change the display order and e.t.c.
//
// For faster execution you can create a custom Formatter
// and compile your own quicktemplate: https://github.com/valyala/quicktemplate
//
// This one uses the standard text/template syntax.
type Template struct {
	// Custom template source.
	// Use this or `Tmpl/TmplName` fields.
	Text string
	// Custom template funcs to used when `Text` is not empty.
	Funcs template.FuncMap

	// Custom template to use, overrides the `Text` and `Funcs` fields.
	Tmpl *template.Template
	// If not empty then this named template/block renders the log line.
	TmplName string

	ac *AccessLog
}

// SetOutput creates the default template if missing
// when this formatter is registered.
func (f *Template) SetOutput(dest io.Writer) {
	f.ac, _ = dest.(*AccessLog)

	if f.Tmpl == nil {
		tmpl := template.New("")

		text := f.Text
		if text != "" {
			tmpl.Funcs(f.Funcs)
		} else {
			text = defaultTmplText
		}

		f.Tmpl = template.Must(tmpl.Parse(text))
	}
}

const defaultTmplText = "{{.Now.Format .TimeFormat}}|{{.Latency}}|{{.Code}}|{{.Method}}|{{.Path}}|{{.IP}}|{{.RequestValuesLine}}|{{.BytesReceivedLine}}|{{.BytesSentLine}}|{{.Request}}|{{.Response}}|\n"

func (f *Template) LogText(log *Log) (string, error) {
	var err error

	// A template may be executed safely in parallel, although if parallel
	// executions share a Writer the output may be interleaved.
	// We solve that using a buffer pool, no locks when template is executing (x2 performance boost).
	temp := f.ac.bufPool.Get().(*bytes.Buffer)

	if f.TmplName != "" {
		err = f.Tmpl.ExecuteTemplate(temp, f.TmplName, log)
	} else {
		err = f.Tmpl.Execute(temp, log)
	}

	if err != nil {
		f.ac.bufPool.Put(temp)
		return "", err
	}

	text := temp.String()
	temp.Reset()
	f.ac.bufPool.Put(temp)

	return text, nil
}

// Format prints the logs in text/template format.
func (f *Template) Format(log *Log) (bool, error) {
	var err error

	// A template may be executed safely in parallel, although if parallel
	// executions share a Writer the output may be interleaved.
	// We solve that using a buffer pool, no locks when template is executing (x2 performance boost).
	temp := f.ac.bufPool.Get().(*bytes.Buffer)

	if f.TmplName != "" {
		err = f.Tmpl.ExecuteTemplate(temp, f.TmplName, log)
	} else {
		err = f.Tmpl.Execute(temp, log)
	}

	if err != nil {
		f.ac.bufPool.Put(temp)
		return true, err
	}

	f.ac.Write(temp.Bytes())
	temp.Reset()
	f.ac.bufPool.Put(temp)
	return true, nil
}