aboutsummaryrefslogtreecommitdiffstats
path: root/kube2msb/src/kube2msb/vendor/github.com/coreos/go-oidc/jose/claims.go
blob: 8b48bfd230b075cbd7069c5744737ca4eb84e5e1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
package jose

import (
	"encoding/json"
	"fmt"
	"math"
	"time"
)

type Claims map[string]interface{}

func (c Claims) Add(name string, value interface{}) {
	c[name] = value
}

func (c Claims) StringClaim(name string) (string, bool, error) {
	cl, ok := c[name]
	if !ok {
		return "", false, nil
	}

	v, ok := cl.(string)
	if !ok {
		return "", false, fmt.Errorf("unable to parse claim as string: %v", name)
	}

	return v, true, nil
}

func (c Claims) StringsClaim(name string) ([]string, bool, error) {
	cl, ok := c[name]
	if !ok {
		return nil, false, nil
	}

	if v, ok := cl.([]string); ok {
		return v, true, nil
	}

	// When unmarshaled, []string will become []interface{}.
	if v, ok := cl.([]interface{}); ok {
		var ret []string
		for _, vv := range v {
			str, ok := vv.(string)
			if !ok {
				return nil, false, fmt.Errorf("unable to parse claim as string array: %v", name)
			}
			ret = append(ret, str)
		}
		return ret, true, nil
	}

	return nil, false, fmt.Errorf("unable to parse claim as string array: %v", name)
}

func (c Claims) Int64Claim(name string) (int64, bool, error) {
	cl, ok := c[name]
	if !ok {
		return 0, false, nil
	}

	v, ok := cl.(int64)
	if !ok {
		vf, ok := cl.(float64)
		if !ok {
			return 0, false, fmt.Errorf("unable to parse claim as int64: %v", name)
		}
		v = int64(vf)
	}

	return v, true, nil
}

func (c Claims) Float64Claim(name string) (float64, bool, error) {
	cl, ok := c[name]
	if !ok {
		return 0, false, nil
	}

	v, ok := cl.(float64)
	if !ok {
		vi, ok := cl.(int64)
		if !ok {
			return 0, false, fmt.Errorf("unable to parse claim as float64: %v", name)
		}
		v = float64(vi)
	}

	return v, true, nil
}

func (c Claims) TimeClaim(name string) (time.Time, bool, error) {
	v, ok, err := c.Float64Claim(name)
	if !ok || err != nil {
		return time.Time{}, ok, err
	}

	s := math.Trunc(v)
	ns := (v - s) * math.Pow(10, 9)
	return time.Unix(int64(s), int64(ns)).UTC(), true, nil
}

func decodeClaims(payload []byte) (Claims, error) {
	var c Claims
	if err := json.Unmarshal(payload, &c); err != nil {
		return nil, fmt.Errorf("malformed JWT claims, unable to decode: %v", err)
	}
	return c, nil
}

func marshalClaims(c Claims) ([]byte, error) {
	b, err := json.Marshal(c)
	if err != nil {
		return nil, err
	}
	return b, nil
}

func encodeClaims(c Claims) (string, error) {
	b, err := marshalClaims(c)
	if err != nil {
		return "", err
	}

	return encodeSegment(b), nil
}