Files
mark/macro/macro.go
2026-03-30 11:23:01 +02:00

188 lines
3.7 KiB
Go

package macro
import (
"bytes"
"fmt"
"regexp"
"strings"
"text/template"
"github.com/kovetskiy/mark/v16/includes"
"github.com/rs/zerolog/log"
"go.yaml.in/yaml/v3"
)
var reMacroDirective = regexp.MustCompile(
// <!-- Macro: <regexp>
// Template: <template path>
// <optional yaml data> -->
`(?s)` + // dot capture newlines
/**/ `<!--\s*Macro:\s*(?P<expr>[^\n]+)\n` +
/* */ `\s*Template:\s*(?P<template>.+?)\s*` +
/* */ `(?P<config>\n.*?)?-->`,
)
type Macro struct {
Regexp *regexp.Regexp
Template *template.Template
Config string
}
func (macro *Macro) Apply(
content []byte,
) ([]byte, error) {
var err error
content = macro.Regexp.ReplaceAllFunc(
content,
func(match []byte) []byte {
config := map[string]interface{}{}
err = yaml.Unmarshal([]byte(macro.Config), &config)
if err != nil {
err = fmt.Errorf("unable to unmarshal macros config template: %w", err)
return match
}
var buffer bytes.Buffer
err = macro.Template.Execute(&buffer, macro.configure(
config,
macro.Regexp.FindSubmatch(match),
))
if err != nil {
err = fmt.Errorf("unable to execute macros template: %w", err)
return match
}
return buffer.Bytes()
},
)
return content, err
}
func (macro *Macro) configure(node interface{}, groups [][]byte) interface{} {
switch node := node.(type) {
case map[interface{}]interface{}:
for key, value := range node {
node[key] = macro.configure(value, groups)
}
return node
case map[string]interface{}:
for key, value := range node {
node[key] = macro.configure(value, groups)
}
return node
case []interface{}:
for key, value := range node {
node[key] = macro.configure(value, groups)
}
return node
case string:
for i, group := range groups {
node = strings.ReplaceAll(
node,
fmt.Sprintf("${%d}", i),
string(group),
)
}
return node
}
return node
}
func ExtractMacros(
base string,
includePath string,
contents []byte,
templates *template.Template,
) ([]Macro, []byte, error) {
var err error
var macros []Macro
contents = reMacroDirective.ReplaceAllFunc(
contents,
func(spec []byte) []byte {
if err != nil {
return spec
}
groups := reMacroDirective.FindStringSubmatch(string(spec))
var (
expr = groups[reMacroDirective.SubexpIndex("expr")]
template = groups[reMacroDirective.SubexpIndex("template")]
config = groups[reMacroDirective.SubexpIndex("config")]
)
var macro Macro
if strings.HasPrefix(template, "#") {
cfg := map[string]interface{}{}
err = yaml.Unmarshal([]byte(config), &cfg)
if err != nil {
err = fmt.Errorf("unable to unmarshal macros config template: %w", err)
return nil
}
body, ok := cfg[template[1:]].(string)
if !ok {
err = fmt.Errorf(
"the template config doesn't have '%s' field",
template[1:],
)
return nil
}
macro.Template, err = templates.New(template).Parse(body)
if err != nil {
err = fmt.Errorf("unable to parse template: %w", err)
return nil
}
} else {
macro.Template, err = includes.LoadTemplate(base, includePath, template, "{{", "}}", templates)
if err != nil {
err = fmt.Errorf("unable to load template: %w", err)
return nil
}
}
macro.Regexp, err = regexp.Compile(expr)
if err != nil {
err = fmt.Errorf("unable to compile macros regexp (expr=%q, template=%q): %w", expr, template, err)
return nil
}
macro.Config = config
log.Trace().
Interface("vardump", map[string]interface{}{
"expr": expr,
"template": template,
"config": macro.Config,
}).
Msgf("loaded macro %q", expr)
macros = append(macros, macro)
return []byte{}
},
)
return macros, contents, err
}