-
-
Notifications
You must be signed in to change notification settings - Fork 4
Expand file tree
/
Copy pathrender.go
More file actions
158 lines (139 loc) · 4.27 KB
/
render.go
File metadata and controls
158 lines (139 loc) · 4.27 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
package render
import (
"bytes"
"encoding/json"
"fmt"
stdio "io"
"os"
"path/filepath"
"text/template"
"github.com/Masterminds/sprig/v3"
"github.com/flowexec/tuikit/views"
"github.com/pkg/errors"
"gopkg.in/yaml.v3"
"github.com/flowexec/flow/internal/context"
"github.com/flowexec/flow/internal/io"
"github.com/flowexec/flow/internal/logger"
"github.com/flowexec/flow/internal/runner"
"github.com/flowexec/flow/internal/runner/engine"
"github.com/flowexec/flow/internal/utils/env"
"github.com/flowexec/flow/types/executable"
)
type renderRunner struct{}
func NewRunner() runner.Runner {
return &renderRunner{}
}
func (r *renderRunner) Name() string {
return "render"
}
func (r *renderRunner) IsCompatible(executable *executable.Executable) bool {
if executable == nil || executable.Render == nil {
return false
}
return true
}
func (r *renderRunner) Exec(
ctx *context.Context,
e *executable.Executable,
_ engine.Engine,
inputEnv map[string]string,
inputArgs []string,
) error {
if !ctx.Config.ShowTUI() {
return fmt.Errorf("unable to render when interactive mode is disabled")
}
renderSpec := e.Render
if err := env.SetEnv(ctx.Config.CurrentVaultName(), e.Env(), inputArgs, inputEnv); err != nil {
return errors.Wrap(err, "unable to set parameters to env")
}
if cb, err := env.CreateTempEnvFiles(
ctx.Config.CurrentVaultName(),
e.FlowFilePath(),
e.WorkspacePath(),
e.Env(),
inputArgs,
inputEnv,
); err != nil {
ctx.AddCallback(cb)
return errors.Wrap(err, "unable to create temporary env files")
} else {
ctx.AddCallback(cb)
}
envMap, err := env.BuildEnvMap(
ctx.Config.CurrentVaultName(), e.Env(), inputArgs, inputEnv, env.DefaultEnv(ctx, e),
)
if err != nil {
return errors.Wrap(err, "unable to set parameters to env")
}
targetDir, isTmp, err := renderSpec.Dir.ExpandDirectory(
e.WorkspacePath(),
e.FlowFilePath(),
ctx.ProcessTmpDir,
envMap,
)
if err != nil {
return errors.Wrap(err, "unable to expand directory")
} else if isTmp {
ctx.ProcessTmpDir = targetDir
}
contentFile := filepath.Clean(filepath.Join(targetDir, renderSpec.TemplateFile))
var templateData interface{}
if renderSpec.TemplateDataFile != "" {
templateData, err = readDataFile(targetDir, renderSpec.TemplateDataFile)
if err != nil {
return err
}
}
tmpl, err := template.New(filepath.Base(renderSpec.TemplateFile)).Funcs(sprig.TxtFuncMap()).ParseFiles(contentFile)
if err != nil {
return errors.Wrapf(err, "unable to parse template file %s", contentFile)
}
var buff bytes.Buffer
if err = tmpl.Execute(&buff, templateData); err != nil {
return errors.Wrapf(err, "unable to execute template file %s", contentFile)
}
logger.Log().Infof("Rendering content from file %s", contentFile)
if os.Getenv(io.DisableInteractiveEnvKey) != "" {
logger.Log().Print("### Rendered Content Start ###")
logger.Log().Print(buff.String())
logger.Log().Print("### Rendered Content End ###")
return nil
}
if err := ctx.TUIContainer.Start(); err != nil {
return errors.Wrapf(err, "unable to open viewer")
}
defer func() {
ctx.TUIContainer.WaitForExit()
}()
filename := filepath.Base(contentFile)
ctx.TUIContainer.SetState("file", filename)
return ctx.TUIContainer.SetView(views.NewMarkdownView(ctx.TUIContainer.RenderState(), buff.String()))
}
func readDataFile(dir, path string) (interface{}, error) {
var templateData interface{}
dataFilePath := filepath.Clean(filepath.Join(dir, path))
if _, err := os.Stat(dataFilePath); os.IsNotExist(err) {
return nil, fmt.Errorf("template data file %s does not exist", dataFilePath)
}
reader, err := os.Open(dataFilePath)
if err != nil {
return nil, errors.Wrapf(err, "unable to open template data file %s", dataFilePath)
}
defer reader.Close()
data, err := stdio.ReadAll(reader)
if err != nil {
return nil, errors.Wrapf(err, "unable to read template data file %s", dataFilePath)
}
extension := filepath.Ext(dataFilePath)
switch extension {
case ".json":
if err = json.Unmarshal(data, &templateData); err != nil {
return nil, errors.Wrapf(err, "unable to unmarshal template data file %s", dataFilePath)
}
case ".yaml", ".yml":
if err = yaml.Unmarshal(data, &templateData); err != nil {
return nil, errors.Wrapf(err, "unable to unmarshal template data file %s", dataFilePath)
}
}
return templateData, nil
}