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 } else { 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 }