204 lines
4.5 KiB
Go
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)
|
|
}
|