346 lines
12 KiB
Go
346 lines
12 KiB
Go
package stdlib
|
|
|
|
import (
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"os/exec"
|
|
|
|
"github.com/d5/tengo/objects"
|
|
)
|
|
|
|
var osModule = map[string]objects.Object{
|
|
"o_rdonly": &objects.Int{Value: int64(os.O_RDONLY)},
|
|
"o_wronly": &objects.Int{Value: int64(os.O_WRONLY)},
|
|
"o_rdwr": &objects.Int{Value: int64(os.O_RDWR)},
|
|
"o_append": &objects.Int{Value: int64(os.O_APPEND)},
|
|
"o_create": &objects.Int{Value: int64(os.O_CREATE)},
|
|
"o_excl": &objects.Int{Value: int64(os.O_EXCL)},
|
|
"o_sync": &objects.Int{Value: int64(os.O_SYNC)},
|
|
"o_trunc": &objects.Int{Value: int64(os.O_TRUNC)},
|
|
"mode_dir": &objects.Int{Value: int64(os.ModeDir)},
|
|
"mode_append": &objects.Int{Value: int64(os.ModeAppend)},
|
|
"mode_exclusive": &objects.Int{Value: int64(os.ModeExclusive)},
|
|
"mode_temporary": &objects.Int{Value: int64(os.ModeTemporary)},
|
|
"mode_symlink": &objects.Int{Value: int64(os.ModeSymlink)},
|
|
"mode_device": &objects.Int{Value: int64(os.ModeDevice)},
|
|
"mode_named_pipe": &objects.Int{Value: int64(os.ModeNamedPipe)},
|
|
"mode_socket": &objects.Int{Value: int64(os.ModeSocket)},
|
|
"mode_setuid": &objects.Int{Value: int64(os.ModeSetuid)},
|
|
"mode_setgui": &objects.Int{Value: int64(os.ModeSetgid)},
|
|
"mode_char_device": &objects.Int{Value: int64(os.ModeCharDevice)},
|
|
"mode_sticky": &objects.Int{Value: int64(os.ModeSticky)},
|
|
"mode_type": &objects.Int{Value: int64(os.ModeType)},
|
|
"mode_perm": &objects.Int{Value: int64(os.ModePerm)},
|
|
"path_separator": &objects.Char{Value: os.PathSeparator},
|
|
"path_list_separator": &objects.Char{Value: os.PathListSeparator},
|
|
"dev_null": &objects.String{Value: os.DevNull},
|
|
"seek_set": &objects.Int{Value: int64(io.SeekStart)},
|
|
"seek_cur": &objects.Int{Value: int64(io.SeekCurrent)},
|
|
"seek_end": &objects.Int{Value: int64(io.SeekEnd)},
|
|
"args": &objects.UserFunction{Value: osArgs}, // args() => array(string)
|
|
"chdir": FuncASRE(os.Chdir), // chdir(dir string) => error
|
|
"chmod": osFuncASFmRE(os.Chmod), // chmod(name string, mode int) => error
|
|
"chown": FuncASIIRE(os.Chown), // chown(name string, uid int, gid int) => error
|
|
"clearenv": FuncAR(os.Clearenv), // clearenv()
|
|
"environ": FuncARSs(os.Environ), // environ() => array(string)
|
|
"exit": FuncAIR(os.Exit), // exit(code int)
|
|
"expand_env": FuncASRS(os.ExpandEnv), // expand_env(s string) => string
|
|
"getegid": FuncARI(os.Getegid), // getegid() => int
|
|
"getenv": FuncASRS(os.Getenv), // getenv(s string) => string
|
|
"geteuid": FuncARI(os.Geteuid), // geteuid() => int
|
|
"getgid": FuncARI(os.Getgid), // getgid() => int
|
|
"getgroups": FuncARIsE(os.Getgroups), // getgroups() => array(string)/error
|
|
"getpagesize": FuncARI(os.Getpagesize), // getpagesize() => int
|
|
"getpid": FuncARI(os.Getpid), // getpid() => int
|
|
"getppid": FuncARI(os.Getppid), // getppid() => int
|
|
"getuid": FuncARI(os.Getuid), // getuid() => int
|
|
"getwd": FuncARSE(os.Getwd), // getwd() => string/error
|
|
"hostname": FuncARSE(os.Hostname), // hostname() => string/error
|
|
"lchown": FuncASIIRE(os.Lchown), // lchown(name string, uid int, gid int) => error
|
|
"link": FuncASSRE(os.Link), // link(oldname string, newname string) => error
|
|
"lookup_env": &objects.UserFunction{Value: osLookupEnv}, // lookup_env(key string) => string/false
|
|
"mkdir": osFuncASFmRE(os.Mkdir), // mkdir(name string, perm int) => error
|
|
"mkdir_all": osFuncASFmRE(os.MkdirAll), // mkdir_all(name string, perm int) => error
|
|
"readlink": FuncASRSE(os.Readlink), // readlink(name string) => string/error
|
|
"remove": FuncASRE(os.Remove), // remove(name string) => error
|
|
"remove_all": FuncASRE(os.RemoveAll), // remove_all(name string) => error
|
|
"rename": FuncASSRE(os.Rename), // rename(oldpath string, newpath string) => error
|
|
"setenv": FuncASSRE(os.Setenv), // setenv(key string, value string) => error
|
|
"symlink": FuncASSRE(os.Symlink), // symlink(oldname string newname string) => error
|
|
"temp_dir": FuncARS(os.TempDir), // temp_dir() => string
|
|
"truncate": FuncASI64RE(os.Truncate), // truncate(name string, size int) => error
|
|
"unsetenv": FuncASRE(os.Unsetenv), // unsetenv(key string) => error
|
|
"create": &objects.UserFunction{Value: osCreate}, // create(name string) => imap(file)/error
|
|
"open": &objects.UserFunction{Value: osOpen}, // open(name string) => imap(file)/error
|
|
"open_file": &objects.UserFunction{Value: osOpenFile}, // open_file(name string, flag int, perm int) => imap(file)/error
|
|
"find_process": &objects.UserFunction{Value: osFindProcess}, // find_process(pid int) => imap(process)/error
|
|
"start_process": &objects.UserFunction{Value: osStartProcess}, // start_process(name string, argv array(string), dir string, env array(string)) => imap(process)/error
|
|
"exec_look_path": FuncASRSE(exec.LookPath), // exec_look_path(file) => string/error
|
|
"exec": &objects.UserFunction{Value: osExec}, // exec(name, args...) => command
|
|
"stat": &objects.UserFunction{Value: osStat}, // stat(name) => imap(fileinfo)/error
|
|
"read_file": &objects.UserFunction{Value: osReadFile}, // readfile(name) => array(byte)/error
|
|
}
|
|
|
|
func osReadFile(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 1 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
fname, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
bytes, err := ioutil.ReadFile(fname)
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
return &objects.Bytes{Value: bytes}, nil
|
|
}
|
|
|
|
func osStat(args ...objects.Object) (ret objects.Object, err error) {
|
|
if len(args) != 1 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
fname, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
stat, err := os.Stat(fname)
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
fstat := &objects.ImmutableMap{
|
|
Value: map[string]objects.Object{
|
|
"name": &objects.String{Value: stat.Name()},
|
|
"mtime": &objects.Time{Value: stat.ModTime()},
|
|
"size": &objects.Int{Value: stat.Size()},
|
|
"mode": &objects.Int{Value: int64(stat.Mode())},
|
|
},
|
|
}
|
|
|
|
if stat.IsDir() {
|
|
fstat.Value["directory"] = objects.TrueValue
|
|
} else {
|
|
fstat.Value["directory"] = objects.FalseValue
|
|
}
|
|
|
|
return fstat, nil
|
|
}
|
|
|
|
func osCreate(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 1 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
res, err := os.Create(s1)
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
return makeOSFile(res), nil
|
|
}
|
|
|
|
func osOpen(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 1 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
res, err := os.Open(s1)
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
return makeOSFile(res), nil
|
|
}
|
|
|
|
func osOpenFile(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 3 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
i2, ok := objects.ToInt(args[1])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
i3, ok := objects.ToInt(args[2])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
res, err := os.OpenFile(s1, i2, os.FileMode(i3))
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
return makeOSFile(res), nil
|
|
}
|
|
|
|
func osArgs(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 0 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
arr := &objects.Array{}
|
|
for _, osArg := range os.Args {
|
|
arr.Value = append(arr.Value, &objects.String{Value: osArg})
|
|
}
|
|
|
|
return arr, nil
|
|
}
|
|
|
|
func osFuncASFmRE(fn func(string, os.FileMode) error) *objects.UserFunction {
|
|
return &objects.UserFunction{
|
|
Value: func(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 2 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
i2, ok := objects.ToInt64(args[1])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
return wrapError(fn(s1, os.FileMode(i2))), nil
|
|
},
|
|
}
|
|
}
|
|
|
|
func osLookupEnv(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 1 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
s1, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
res, ok := os.LookupEnv(s1)
|
|
if !ok {
|
|
return objects.FalseValue, nil
|
|
}
|
|
|
|
return &objects.String{Value: res}, nil
|
|
}
|
|
|
|
func osExec(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) == 0 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
name, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
var execArgs []string
|
|
for _, arg := range args[1:] {
|
|
execArg, ok := objects.ToString(arg)
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
execArgs = append(execArgs, execArg)
|
|
}
|
|
|
|
return makeOSExecCommand(exec.Command(name, execArgs...)), nil
|
|
}
|
|
|
|
func osFindProcess(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 1 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
i1, ok := objects.ToInt(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
proc, err := os.FindProcess(i1)
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
return makeOSProcess(proc), nil
|
|
}
|
|
|
|
func osStartProcess(args ...objects.Object) (objects.Object, error) {
|
|
if len(args) != 4 {
|
|
return nil, objects.ErrWrongNumArguments
|
|
}
|
|
|
|
name, ok := objects.ToString(args[0])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
argv, err := stringArray(args[1])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
dir, ok := objects.ToString(args[2])
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
env, err := stringArray(args[3])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
proc, err := os.StartProcess(name, argv, &os.ProcAttr{
|
|
Dir: dir,
|
|
Env: env,
|
|
})
|
|
if err != nil {
|
|
return wrapError(err), nil
|
|
}
|
|
|
|
return makeOSProcess(proc), nil
|
|
}
|
|
|
|
func stringArray(o objects.Object) ([]string, error) {
|
|
arr, ok := o.(*objects.Array)
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
var sarr []string
|
|
for _, elem := range arr.Value {
|
|
str, ok := elem.(*objects.String)
|
|
if !ok {
|
|
return nil, objects.ErrInvalidTypeConversion
|
|
}
|
|
|
|
sarr = append(sarr, str.Value)
|
|
}
|
|
|
|
return sarr, nil
|
|
}
|