193 lines
4.8 KiB
Go
193 lines
4.8 KiB
Go
package lifted
|
|
|
|
// This code is directly lifted from the argo-cd codebase in order to avoid relying on the lua package.
|
|
// For reference:
|
|
// https://github.com/argoproj/argo-cd/blob/master/util/lua/oslib_safe.go
|
|
|
|
// oslib_safe contains a subset of the lua OS library. For security reasons, we do not expose
|
|
// the entirety of lua OS library to custom actions, such as ones which can exit, read files, etc.
|
|
// Only the safe functions like os.time(), os.date() are exposed. Implementation was copied from
|
|
// github.com/yuin/gopher-lua.
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
lua "github.com/yuin/gopher-lua"
|
|
)
|
|
|
|
// OpenSafeOs open safe os
|
|
func OpenSafeOs(L *lua.LState) int {
|
|
tabmod := L.RegisterModule(lua.TabLibName, osFuncs)
|
|
L.Push(tabmod)
|
|
return 1
|
|
}
|
|
|
|
// SafeOsLoader sofe laoder
|
|
func SafeOsLoader(L *lua.LState) int {
|
|
mod := L.SetFuncs(L.NewTable(), osFuncs)
|
|
L.Push(mod)
|
|
return 1
|
|
}
|
|
|
|
var osFuncs = map[string]lua.LGFunction{
|
|
"time": osTime,
|
|
"date": osDate,
|
|
}
|
|
|
|
func osTime(L *lua.LState) int {
|
|
if L.GetTop() == 0 {
|
|
L.Push(lua.LNumber(time.Now().Unix()))
|
|
} else {
|
|
tbl := L.CheckTable(1)
|
|
sec := getIntField(tbl, "sec", 0)
|
|
min := getIntField(tbl, "min", 0)
|
|
hour := getIntField(tbl, "hour", 12)
|
|
day := getIntField(tbl, "day", -1)
|
|
month := getIntField(tbl, "month", -1)
|
|
year := getIntField(tbl, "year", -1)
|
|
isdst := getBoolField(tbl, "isdst", false)
|
|
t := time.Date(year, time.Month(month), day, hour, min, sec, 0, time.Local)
|
|
// TODO dst
|
|
if false {
|
|
print(isdst)
|
|
}
|
|
L.Push(lua.LNumber(t.Unix()))
|
|
}
|
|
return 1
|
|
}
|
|
|
|
func getIntField(tb *lua.LTable, key string, v int) int {
|
|
ret := tb.RawGetString(key)
|
|
if ln, ok := ret.(lua.LNumber); ok {
|
|
return int(ln)
|
|
}
|
|
return v
|
|
}
|
|
|
|
func getBoolField(tb *lua.LTable, key string, v bool) bool {
|
|
ret := tb.RawGetString(key)
|
|
if lb, ok := ret.(lua.LBool); ok {
|
|
return bool(lb)
|
|
}
|
|
return v
|
|
}
|
|
|
|
func osDate(L *lua.LState) int {
|
|
t := time.Now()
|
|
cfmt := "%c"
|
|
if L.GetTop() >= 1 {
|
|
cfmt = L.CheckString(1)
|
|
if strings.HasPrefix(cfmt, "!") {
|
|
t = time.Now().UTC()
|
|
cfmt = strings.TrimLeft(cfmt, "!")
|
|
}
|
|
if L.GetTop() >= 2 {
|
|
t = time.Unix(L.CheckInt64(2), 0)
|
|
}
|
|
if strings.HasPrefix(cfmt, "*t") {
|
|
ret := L.NewTable()
|
|
ret.RawSetString("year", lua.LNumber(t.Year()))
|
|
ret.RawSetString("month", lua.LNumber(t.Month()))
|
|
ret.RawSetString("day", lua.LNumber(t.Day()))
|
|
ret.RawSetString("hour", lua.LNumber(t.Hour()))
|
|
ret.RawSetString("min", lua.LNumber(t.Minute()))
|
|
ret.RawSetString("sec", lua.LNumber(t.Second()))
|
|
ret.RawSetString("wday", lua.LNumber(t.Weekday()+1))
|
|
// TODO yday & dst
|
|
ret.RawSetString("yday", lua.LNumber(0))
|
|
ret.RawSetString("isdst", lua.LFalse)
|
|
L.Push(ret)
|
|
return 1
|
|
}
|
|
}
|
|
L.Push(lua.LString(strftime(t, cfmt)))
|
|
return 1
|
|
}
|
|
|
|
var cDateFlagToGo = map[byte]string{
|
|
'a': "mon", 'A': "Monday", 'b': "Jan", 'B': "January", 'c': "02 Jan 06 15:04 MST", 'd': "02",
|
|
'F': "2006-01-02", 'H': "15", 'I': "03", 'm': "01", 'M': "04", 'p': "PM", 'P': "pm", 'S': "05",
|
|
'x': "15/04/05", 'X': "15:04:05", 'y': "06", 'Y': "2006", 'z': "-0700", 'Z': "MST"}
|
|
|
|
func strftime(t time.Time, cfmt string) string {
|
|
sc := newFlagScanner('%', "", "", cfmt)
|
|
for c, eos := sc.Next(); !eos; c, eos = sc.Next() {
|
|
if !sc.ChangeFlag {
|
|
if sc.HasFlag {
|
|
if v, ok := cDateFlagToGo[c]; ok {
|
|
sc.AppendString(t.Format(v))
|
|
} else {
|
|
switch c {
|
|
case 'w':
|
|
sc.AppendString(fmt.Sprint(int(t.Weekday())))
|
|
default:
|
|
sc.AppendChar('%')
|
|
sc.AppendChar(c)
|
|
}
|
|
}
|
|
sc.HasFlag = false
|
|
} else {
|
|
sc.AppendChar(c)
|
|
}
|
|
}
|
|
}
|
|
|
|
return sc.String()
|
|
}
|
|
|
|
type flagScanner struct {
|
|
flag byte
|
|
start string
|
|
end string
|
|
buf []byte
|
|
str string
|
|
Length int
|
|
Pos int
|
|
HasFlag bool
|
|
ChangeFlag bool
|
|
}
|
|
|
|
func newFlagScanner(flag byte, start, end, str string) *flagScanner {
|
|
return &flagScanner{flag, start, end, make([]byte, 0, len(str)), str, len(str), 0, false, false}
|
|
}
|
|
|
|
// AppendString append string to fs.buf
|
|
func (fs *flagScanner) AppendString(str string) { fs.buf = append(fs.buf, str...) }
|
|
|
|
// AppendChar append char to fs.buf
|
|
func (fs *flagScanner) AppendChar(ch byte) { fs.buf = append(fs.buf, ch) }
|
|
|
|
// String return the fs.buf of string
|
|
func (fs *flagScanner) String() string { return string(fs.buf) }
|
|
|
|
// Next iterate fs
|
|
func (fs *flagScanner) Next() (byte, bool) {
|
|
c := byte('\000')
|
|
fs.ChangeFlag = false
|
|
if fs.Pos == fs.Length {
|
|
if fs.HasFlag {
|
|
fs.AppendString(fs.end)
|
|
}
|
|
return c, true
|
|
}
|
|
c = fs.str[fs.Pos]
|
|
if c == fs.flag {
|
|
if fs.Pos < (fs.Length-1) && fs.str[fs.Pos+1] == fs.flag {
|
|
fs.HasFlag = false
|
|
fs.AppendChar(fs.flag)
|
|
fs.Pos += 2
|
|
return fs.Next()
|
|
} else if fs.Pos != fs.Length-1 {
|
|
if fs.HasFlag {
|
|
fs.AppendString(fs.end)
|
|
}
|
|
fs.AppendString(fs.start)
|
|
fs.ChangeFlag = true
|
|
fs.HasFlag = true
|
|
}
|
|
}
|
|
fs.Pos++
|
|
return c, false
|
|
}
|