121 lines
2.9 KiB
Go
121 lines
2.9 KiB
Go
|
// func-import.go
|
||
|
package expr
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"os"
|
||
|
"path"
|
||
|
"path/filepath"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
const ENV_EXPR_PATH = "EXPR_PATH"
|
||
|
|
||
|
func importFunc(ctx exprContext, name string, args []any) (result any, err error) {
|
||
|
var dirList []string
|
||
|
|
||
|
dirList = addEnvImportDirs(dirList)
|
||
|
dirList = addPresetImportDirs(ctx, dirList)
|
||
|
result, err = doImport(ctx, name, dirList, NewFlatArrayIterator(args))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func checkStringParamExpected(funcName string, paramValue any, paramPos int) (err error) {
|
||
|
if !(isString(paramValue) /*|| isList(paramValue)*/) {
|
||
|
err = fmt.Errorf("%s(): param nr %d has wrong type %T, string expected", funcName, paramPos+1, paramValue)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func addEnvImportDirs(dirList []string) []string {
|
||
|
if dirSpec, exists := os.LookupEnv(ENV_EXPR_PATH); exists {
|
||
|
dirs := strings.Split(dirSpec, ":")
|
||
|
if dirList == nil {
|
||
|
dirList = dirs
|
||
|
} else {
|
||
|
dirList = append(dirList, dirs...)
|
||
|
}
|
||
|
}
|
||
|
return dirList
|
||
|
}
|
||
|
|
||
|
func addPresetImportDirs(ctx exprContext, dirList []string) []string {
|
||
|
if dirSpec, exists := getPresetString(ctx, preset_import_path); exists {
|
||
|
dirs := strings.Split(dirSpec, ":")
|
||
|
if dirList == nil {
|
||
|
dirList = dirs
|
||
|
} else {
|
||
|
dirList = append(dirList, dirs...)
|
||
|
}
|
||
|
}
|
||
|
return dirList
|
||
|
}
|
||
|
|
||
|
func isFile(filePath string) bool {
|
||
|
info, err := os.Stat(filePath)
|
||
|
return (err == nil || errors.Is(err, os.ErrExist)) && info.Mode().IsRegular()
|
||
|
}
|
||
|
|
||
|
func searchAmongPath(filename string, dirList []string) (filePath string) {
|
||
|
for _, dir := range dirList {
|
||
|
if fullPath := path.Join(dir, filename); isFile(fullPath) {
|
||
|
filePath = fullPath
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func isPathRelative(filePath string) bool {
|
||
|
unixPath := filepath.ToSlash(filePath)
|
||
|
return strings.HasPrefix(unixPath, "./") || strings.HasPrefix(unixPath, "../")
|
||
|
}
|
||
|
|
||
|
func makeFilepath(filename string, dirList []string) (filePath string, err error) {
|
||
|
if path.IsAbs(filename) || isPathRelative(filename) {
|
||
|
if isFile(filename) {
|
||
|
filePath = filename
|
||
|
}
|
||
|
} else {
|
||
|
filePath = searchAmongPath(filename, dirList)
|
||
|
}
|
||
|
if len(filePath) == 0 {
|
||
|
err = fmt.Errorf("source file %q not found", filename)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func doImport(ctx exprContext, name string, dirList []string, it Iterator) (result any, err error) {
|
||
|
var v any
|
||
|
var sourceFilepath string
|
||
|
|
||
|
for v, err = it.Next(); err == nil; v, err = it.Next() {
|
||
|
if err = checkStringParamExpected(name, v, it.Index()); err != nil {
|
||
|
break
|
||
|
}
|
||
|
if sourceFilepath, err = makeFilepath(v.(string), dirList); err != nil {
|
||
|
break
|
||
|
}
|
||
|
var file *os.File
|
||
|
if file, err = os.Open(sourceFilepath); err == nil {
|
||
|
defer file.Close()
|
||
|
var expr *ast
|
||
|
scanner := NewScanner(file, DefaultTranslations())
|
||
|
parser := NewParser(ctx)
|
||
|
if expr, err = parser.parse(scanner); err == nil {
|
||
|
_, err = expr.Eval(ctx, false)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if err != nil && err == io.EOF {
|
||
|
err = nil
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func importImportFunc(ctx exprContext) {
|
||
|
ctx.RegisterFunc("import", &simpleFunctor{f: importFunc}, 1, -1)
|
||
|
}
|