154 lines
4.5 KiB
Go
154 lines
4.5 KiB
Go
|
package templater
|
||
|
|
||
|
import (
|
||
|
"log"
|
||
|
"path/filepath"
|
||
|
"strconv"
|
||
|
|
||
|
"bytes"
|
||
|
encjson "encoding/json"
|
||
|
"fmt"
|
||
|
"os"
|
||
|
"text/template"
|
||
|
|
||
|
"github.com/hashicorp/hcl/v2"
|
||
|
"github.com/hashicorp/hcl/v2/hclsyntax"
|
||
|
"github.com/zclconf/go-cty/cty"
|
||
|
ctyjson "github.com/zclconf/go-cty/cty/json"
|
||
|
|
||
|
"forge.cadoles.com/pcaseiro/templatefile/pkg/utils"
|
||
|
)
|
||
|
|
||
|
type ConfigFile struct {
|
||
|
Destination string `form:"destination" json:"destination"` // Where do we write the configuration file
|
||
|
Source string `form:"source" json:"source"` // The template file short name
|
||
|
TemplateType string `json:"type"` // The template file type (hcl or gotemplate)
|
||
|
Mode string `form:"mod" json:"mode"` // The configuration file final permissions (mode)
|
||
|
Owner string `json:"owner"` // The configuration file owner
|
||
|
Service string `json:"service"` // Service to restart after configuration generation
|
||
|
Group string `json:"group"` // The configuration file group owner
|
||
|
}
|
||
|
|
||
|
// Generate the configuration file from the template (hcl or json)
|
||
|
func (cf *ConfigFile) Generate(root string, templateDir string, values []byte) error {
|
||
|
var template string
|
||
|
dest := filepath.Join(root, cf.Destination)
|
||
|
source := filepath.Join(templateDir, cf.Source)
|
||
|
intMod, err := strconv.ParseInt(cf.Mode, 8, 64)
|
||
|
if err != nil {
|
||
|
return (err)
|
||
|
}
|
||
|
|
||
|
template, err = cf.ProcessTemplate(source, values)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Process templates failed with error: %v", err)
|
||
|
}
|
||
|
dirname := filepath.Dir(dest)
|
||
|
err = os.MkdirAll(dirname, os.FileMode(int(0700)))
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Process templates failed with error: %v", err)
|
||
|
}
|
||
|
err = os.WriteFile(dest, []byte(template), os.FileMode(intMod))
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Process templates failed with error: %v", err)
|
||
|
}
|
||
|
log.Printf("\tFile %s generated\n", dest)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Process the template with the provided values
|
||
|
func (cf *ConfigFile) ProcessTemplate(source string, values []byte) (string, error) {
|
||
|
var result string
|
||
|
var err error
|
||
|
|
||
|
if cf.TemplateType == "hcl" {
|
||
|
// The template is an hcl template so we call processHCLTemplate
|
||
|
result, err = cf.processHCLTemplate(source, values)
|
||
|
if err != nil {
|
||
|
return "", fmt.Errorf("Process HCL template failed with error: %v", err)
|
||
|
}
|
||
|
} else if cf.TemplateType == "go" {
|
||
|
// The template is a go template so we call processGoTemplate
|
||
|
result, err = cf.processGoTemplate(source, values)
|
||
|
if err != nil {
|
||
|
return "", fmt.Errorf("Process GO template failed with error: %v", err)
|
||
|
}
|
||
|
}
|
||
|
return result, nil
|
||
|
}
|
||
|
|
||
|
// The actual template processing for Go templates
|
||
|
func (cf *ConfigFile) processGoTemplate(file string, configValues []byte) (string, error) {
|
||
|
|
||
|
// The JSON configuration
|
||
|
var confData map[string]interface{}
|
||
|
var res bytes.Buffer
|
||
|
|
||
|
err := encjson.Unmarshal(configValues, &confData)
|
||
|
utils.CheckErr(err)
|
||
|
|
||
|
// Read the template
|
||
|
data, err := os.ReadFile(file)
|
||
|
utils.CheckErr(err)
|
||
|
|
||
|
tpl, err := template.New("conf").Parse(string(data))
|
||
|
utils.CheckErr(err)
|
||
|
|
||
|
utils.CheckErr(tpl.Execute(&res, confData["Config"]))
|
||
|
|
||
|
return res.String(), nil
|
||
|
}
|
||
|
|
||
|
// The actual template processing for HCL templates
|
||
|
func (cf *ConfigFile) processHCLTemplate(file string, config []byte) (string, error) {
|
||
|
|
||
|
fct, err := os.ReadFile(file)
|
||
|
utils.CheckErr(err)
|
||
|
|
||
|
expr, diags := hclsyntax.ParseTemplate(fct, file, hcl.Pos{Line: 0, Column: 1})
|
||
|
utils.CheckDiags(diags)
|
||
|
|
||
|
// Retrieve values from JSON
|
||
|
var varsVal cty.Value
|
||
|
ctyType, err := ctyjson.ImpliedType(config)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
/* Maybe one day
|
||
|
cexpr, diags := hclsyntax.ParseExpression(config, "", hcl.Pos{Line: 0, Column: 1})
|
||
|
if diags.HasErrors() {
|
||
|
panic(diags.Error())
|
||
|
}
|
||
|
varsVal, diags = cexpr.Value(&hcl.EvalContext{})
|
||
|
fmt.Println(cexpr.Variables())
|
||
|
checkDiags(diags)
|
||
|
*/
|
||
|
} else {
|
||
|
varsVal, err = ctyjson.Unmarshal(config, ctyType)
|
||
|
utils.CheckErr(err)
|
||
|
}
|
||
|
|
||
|
ctx := &hcl.EvalContext{
|
||
|
Variables: varsVal.AsValueMap(),
|
||
|
}
|
||
|
|
||
|
for n := range ctx.Variables {
|
||
|
if !hclsyntax.ValidIdentifier(n) {
|
||
|
return "", fmt.Errorf("invalid template variable name %q: must start with a letter, followed by zero or more letters, digits, and underscores", n)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for _, traversal := range expr.Variables() {
|
||
|
root := traversal.RootName()
|
||
|
if _, ok := ctx.Variables[root]; !ok {
|
||
|
return "", fmt.Errorf("vars map does not contain key %q, referenced at %s", root, traversal[0].SourceRange())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
val, diags := expr.Value(ctx)
|
||
|
if diags.HasErrors() {
|
||
|
return "", diags
|
||
|
}
|
||
|
|
||
|
return val.AsString(), nil
|
||
|
}
|