2019-08-16 13:55:03 +00:00
|
|
|
package internal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2021-12-12 11:11:11 +00:00
|
|
|
"fmt"
|
2019-08-16 13:55:03 +00:00
|
|
|
"strings"
|
|
|
|
|
2021-12-12 11:11:11 +00:00
|
|
|
"github.com/gohugoio/hugo/common/collections"
|
2020-12-18 17:20:12 +00:00
|
|
|
"github.com/gohugoio/hugo/common/hexec"
|
2019-08-16 13:55:03 +00:00
|
|
|
"github.com/gohugoio/hugo/markup/converter"
|
|
|
|
)
|
|
|
|
|
|
|
|
func ExternallyRenderContent(
|
|
|
|
cfg converter.ProviderConfig,
|
|
|
|
ctx converter.DocumentContext,
|
2021-12-12 11:11:11 +00:00
|
|
|
content []byte, binaryName string, args []string) ([]byte, error) {
|
2019-08-16 13:55:03 +00:00
|
|
|
logger := cfg.Logger
|
2021-12-12 11:11:11 +00:00
|
|
|
|
|
|
|
if strings.Contains(binaryName, "/") {
|
|
|
|
panic(fmt.Sprintf("should be no slash in %q", binaryName))
|
2020-12-18 17:20:12 +00:00
|
|
|
}
|
2021-12-12 11:11:11 +00:00
|
|
|
|
|
|
|
argsv := collections.StringSliceToInterfaceSlice(args)
|
|
|
|
|
2019-08-16 13:55:03 +00:00
|
|
|
var out, cmderr bytes.Buffer
|
2021-12-12 11:11:11 +00:00
|
|
|
argsv = append(argsv, hexec.WithStdout(&out))
|
|
|
|
argsv = append(argsv, hexec.WithStderr(&cmderr))
|
|
|
|
argsv = append(argsv, hexec.WithStdin(bytes.NewReader(content)))
|
|
|
|
|
|
|
|
cmd, err := cfg.Exec.New(binaryName, argsv...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-12-18 17:20:12 +00:00
|
|
|
err = cmd.Run()
|
2021-12-12 11:11:11 +00:00
|
|
|
|
2019-08-16 13:55:03 +00:00
|
|
|
// Most external helpers exit w/ non-zero exit code only if severe, i.e.
|
|
|
|
// halting errors occurred. -> log stderr output regardless of state of err
|
|
|
|
for _, item := range strings.Split(cmderr.String(), "\n") {
|
|
|
|
item := strings.TrimSpace(item)
|
|
|
|
if item != "" {
|
2021-12-12 11:11:11 +00:00
|
|
|
if err == nil {
|
|
|
|
logger.Warnf("%s: %s", ctx.DocumentName, item)
|
|
|
|
} else {
|
|
|
|
logger.Errorf("%s: %s", ctx.DocumentName, item)
|
|
|
|
}
|
2019-08-16 13:55:03 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-12 11:11:11 +00:00
|
|
|
|
2019-08-16 13:55:03 +00:00
|
|
|
if err != nil {
|
2021-12-12 11:11:11 +00:00
|
|
|
logger.Errorf("%s rendering %s: %v", binaryName, ctx.DocumentName, err)
|
2019-08-16 13:55:03 +00:00
|
|
|
}
|
|
|
|
|
2021-12-12 11:11:11 +00:00
|
|
|
return normalizeExternalHelperLineFeeds(out.Bytes()), nil
|
2019-08-16 13:55:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Strips carriage returns from third-party / external processes (useful for Windows)
|
|
|
|
func normalizeExternalHelperLineFeeds(content []byte) []byte {
|
|
|
|
return bytes.Replace(content, []byte("\r"), []byte(""), -1)
|
|
|
|
}
|
|
|
|
|
2021-12-12 11:11:11 +00:00
|
|
|
var pythonBinaryCandidates = []string{"python", "python.exe"}
|
|
|
|
|
|
|
|
func GetPythonBinaryAndExecPath() (string, string) {
|
|
|
|
for _, p := range pythonBinaryCandidates {
|
|
|
|
if pth := hexec.LookPath(p); pth != "" {
|
|
|
|
return p, pth
|
2019-08-16 13:55:03 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-12 11:11:11 +00:00
|
|
|
return "", ""
|
2019-08-16 13:55:03 +00:00
|
|
|
}
|