Files
kubesphere/vendor/github.com/open-policy-agent/opa/topdown/time.go
hongming 9769357005 update
Signed-off-by: hongming <talonwan@yunify.com>
2020-03-20 02:16:11 +08:00

204 lines
4.5 KiB
Go

// Copyright 2017 The OPA Authors. All rights reserved.
// Use of this source code is governed by an Apache2
// license that can be found in the LICENSE file.
package topdown
import (
"encoding/json"
"fmt"
"math/big"
"strconv"
"sync"
"time"
"github.com/open-policy-agent/opa/ast"
"github.com/open-policy-agent/opa/topdown/builtins"
)
type nowKeyID string
var nowKey = nowKeyID("time.now_ns")
var tzCache map[string]*time.Location
var tzCacheMutex *sync.Mutex
func builtinTimeNowNanos(bctx BuiltinContext, _ []*ast.Term, iter func(*ast.Term) error) error {
exist, ok := bctx.Cache.Get(nowKey)
var now *ast.Term
if !ok {
curr := time.Now()
now = ast.NewTerm(ast.Number(int64ToJSONNumber(curr.UnixNano())))
bctx.Cache.Put(nowKey, now)
} else {
now = exist.(*ast.Term)
}
return iter(now)
}
func builtinTimeParseNanos(a, b ast.Value) (ast.Value, error) {
format, err := builtins.StringOperand(a, 1)
if err != nil {
return nil, err
}
value, err := builtins.StringOperand(b, 2)
if err != nil {
return nil, err
}
result, err := time.Parse(string(format), string(value))
if err != nil {
return nil, err
}
return ast.Number(int64ToJSONNumber(result.UnixNano())), nil
}
func builtinTimeParseRFC3339Nanos(a ast.Value) (ast.Value, error) {
value, err := builtins.StringOperand(a, 1)
if err != nil {
return nil, err
}
result, err := time.Parse(time.RFC3339, string(value))
if err != nil {
return nil, err
}
return ast.Number(int64ToJSONNumber(result.UnixNano())), nil
}
func builtinParseDurationNanos(a ast.Value) (ast.Value, error) {
duration, err := builtins.StringOperand(a, 1)
if err != nil {
return nil, err
}
value, err := time.ParseDuration(string(duration))
if err != nil {
return nil, err
}
return ast.Number(int64ToJSONNumber(int64(value))), nil
}
func builtinDate(a ast.Value) (ast.Value, error) {
t, err := tzTime(a)
if err != nil {
return nil, err
}
year, month, day := t.Date()
result := ast.Array{ast.IntNumberTerm(year), ast.IntNumberTerm(int(month)), ast.IntNumberTerm(day)}
return result, nil
}
func builtinClock(a ast.Value) (ast.Value, error) {
t, err := tzTime(a)
if err != nil {
return nil, err
}
hour, minute, second := t.Clock()
result := ast.Array{ast.IntNumberTerm(hour), ast.IntNumberTerm(minute), ast.IntNumberTerm(second)}
return result, nil
}
func builtinWeekday(a ast.Value) (ast.Value, error) {
t, err := tzTime(a)
if err != nil {
return nil, err
}
weekday := t.Weekday().String()
return ast.String(weekday), nil
}
func tzTime(a ast.Value) (t time.Time, err error) {
var nVal ast.Value
loc := time.UTC
switch va := a.(type) {
case ast.Array:
if len(va) == 0 {
return time.Time{}, builtins.NewOperandTypeErr(1, a, "either number (ns) or [number (ns), string (tz)]")
}
nVal, err = builtins.NumberOperand(va[0].Value, 1)
if err != nil {
return time.Time{}, err
}
if len(va) > 1 {
tzVal, err := builtins.StringOperand(va[1].Value, 1)
if err != nil {
return time.Time{}, err
}
tzName := string(tzVal)
switch tzName {
case "", "UTC":
// loc is already UTC
case "Local":
loc = time.Local
default:
var ok bool
tzCacheMutex.Lock()
loc, ok = tzCache[tzName]
if !ok {
loc, err = time.LoadLocation(tzName)
if err != nil {
tzCacheMutex.Unlock()
return time.Time{}, err
}
tzCache[tzName] = loc
}
tzCacheMutex.Unlock()
}
}
case ast.Number:
nVal = a
default:
return time.Time{}, builtins.NewOperandTypeErr(1, a, "either number (ns) or [number (ns), string (tz)]")
}
value, err := builtins.NumberOperand(nVal, 1)
if err != nil {
return time.Time{}, err
}
f := builtins.NumberToFloat(value)
i64, acc := f.Int64()
if acc != big.Exact {
return time.Time{}, fmt.Errorf("timestamp too big")
}
t = time.Unix(0, i64).In(loc)
return t, nil
}
func int64ToJSONNumber(i int64) json.Number {
return json.Number(strconv.FormatInt(i, 10))
}
func init() {
RegisterBuiltinFunc(ast.NowNanos.Name, builtinTimeNowNanos)
RegisterFunctionalBuiltin1(ast.ParseRFC3339Nanos.Name, builtinTimeParseRFC3339Nanos)
RegisterFunctionalBuiltin2(ast.ParseNanos.Name, builtinTimeParseNanos)
RegisterFunctionalBuiltin1(ast.ParseDurationNanos.Name, builtinParseDurationNanos)
RegisterFunctionalBuiltin1(ast.Date.Name, builtinDate)
RegisterFunctionalBuiltin1(ast.Clock.Name, builtinClock)
RegisterFunctionalBuiltin1(ast.Weekday.Name, builtinWeekday)
tzCacheMutex = &sync.Mutex{}
tzCache = make(map[string]*time.Location)
}