-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathid.go
74 lines (65 loc) · 1.87 KB
/
id.go
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
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: MPL-2.0
package oidc
import (
"fmt"
"github.com/hashicorp/cap/oidc/internal/base62"
)
// DefaultIDLength is the default length for generated IDs, which are used for
// state and nonce parameters during OIDC flows.
//
// For ID length requirements see:
// https://tools.ietf.org/html/rfc6749#section-10.10
const DefaultIDLength = 20
// NewID generates a ID with an optional prefix. The ID generated is suitable
// for a Request's State or Nonce. The ID length will be DefaultIDLen, unless an
// optional prefix is provided which will add the prefix's length + an
// underscore. The WithPrefix, WithLen options are supported.
//
// For ID length requirements see:
// https://tools.ietf.org/html/rfc6749#section-10.10
func NewID(opt ...Option) (string, error) {
const op = "NewID"
opts := getIDOpts(opt...)
id, err := base62.Random(opts.withLen)
if err != nil {
return "", fmt.Errorf("%s: unable to generate id: %w", op, err)
}
switch {
case opts.withPrefix != "":
return fmt.Sprintf("%s_%s", opts.withPrefix, id), nil
default:
return id, nil
}
}
// idOptions is the set of available options.
type idOptions struct {
withPrefix string
withLen int
}
// idDefaults is a handy way to get the defaults at runtime and
// during unit tests.
func idDefaults() idOptions {
return idOptions{
withLen: DefaultIDLength,
}
}
// getConfigOpts gets the defaults and applies the opt overrides passed
// in.
func getIDOpts(opt ...Option) idOptions {
opts := idDefaults()
ApplyOpts(&opts, opt...)
return opts
}
// WithPrefix provides an optional prefix for an new ID. When this options is
// provided, NewID will prepend the prefix and an underscore to the new
// identifier.
//
// Valid for: ID
func WithPrefix(prefix string) Option {
return func(o interface{}) {
if o, ok := o.(*idOptions); ok {
o.withPrefix = prefix
}
}
}