2020-05-20 00:32:52 +00:00
|
|
|
package uri
|
|
|
|
|
|
|
|
import (
|
2021-01-27 04:03:53 +00:00
|
|
|
"bytes"
|
|
|
|
"encoding/hex"
|
|
|
|
"io/ioutil"
|
2020-05-20 00:32:52 +00:00
|
|
|
"net/url"
|
|
|
|
"strings"
|
2021-01-27 04:03:53 +00:00
|
|
|
"unicode"
|
2020-05-20 00:32:52 +00:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// URI implements a parser for a URI format based on the the PKCS #11 URI Scheme
|
|
|
|
// defined in https://tools.ietf.org/html/rfc7512
|
|
|
|
//
|
|
|
|
// These URIs will be used to define the key names in a KMS.
|
|
|
|
type URI struct {
|
|
|
|
*url.URL
|
|
|
|
Values url.Values
|
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new URI from a scheme and key-value pairs.
|
|
|
|
func New(scheme string, values url.Values) *URI {
|
|
|
|
return &URI{
|
|
|
|
URL: &url.URL{
|
|
|
|
Scheme: scheme,
|
|
|
|
Opaque: strings.ReplaceAll(values.Encode(), "&", ";"),
|
|
|
|
},
|
|
|
|
Values: values,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFile creates an uri for a file.
|
|
|
|
func NewFile(path string) *URI {
|
|
|
|
return &URI{
|
|
|
|
URL: &url.URL{
|
|
|
|
Scheme: "file",
|
|
|
|
Path: path,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// HasScheme returns true if the given uri has the given scheme, false otherwise.
|
|
|
|
func HasScheme(scheme, rawuri string) bool {
|
|
|
|
u, err := url.Parse(rawuri)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return strings.EqualFold(u.Scheme, scheme)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse returns the URI for the given string or an error.
|
|
|
|
func Parse(rawuri string) (*URI, error) {
|
|
|
|
u, err := url.Parse(rawuri)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "error parsing %s", rawuri)
|
|
|
|
}
|
|
|
|
if u.Scheme == "" {
|
|
|
|
return nil, errors.Errorf("error parsing %s: scheme is missing", rawuri)
|
|
|
|
}
|
2021-08-17 20:25:55 +00:00
|
|
|
// Starting with Go 1.17 url.ParseQuery returns an error using semicolon as
|
|
|
|
// separator.
|
|
|
|
v, err := url.ParseQuery(strings.ReplaceAll(u.Opaque, ";", "&"))
|
2020-05-20 00:32:52 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "error parsing %s", rawuri)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &URI{
|
|
|
|
URL: u,
|
|
|
|
Values: v,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseWithScheme returns the URI for the given string only if it has the given
|
|
|
|
// scheme.
|
|
|
|
func ParseWithScheme(scheme, rawuri string) (*URI, error) {
|
|
|
|
u, err := Parse(rawuri)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if !strings.EqualFold(u.Scheme, scheme) {
|
|
|
|
return nil, errors.Errorf("error parsing %s: scheme not expected", rawuri)
|
|
|
|
}
|
|
|
|
return u, nil
|
|
|
|
}
|
|
|
|
|
2021-02-12 03:14:15 +00:00
|
|
|
// Get returns the first value in the uri with the given key, it will return
|
2020-05-20 00:32:52 +00:00
|
|
|
// empty string if that field is not present.
|
|
|
|
func (u *URI) Get(key string) string {
|
2021-01-27 04:03:53 +00:00
|
|
|
v := u.Values.Get(key)
|
|
|
|
if v == "" {
|
|
|
|
v = u.URL.Query().Get(key)
|
|
|
|
}
|
2021-02-01 22:16:08 +00:00
|
|
|
return v
|
2021-01-27 04:03:53 +00:00
|
|
|
}
|
|
|
|
|
2021-10-12 22:41:41 +00:00
|
|
|
// GetBool returns true if a given key has the value "true". It returns false
|
|
|
|
// otherwise.
|
2021-10-07 22:48:11 +00:00
|
|
|
func (u *URI) GetBool(key string) bool {
|
|
|
|
v := u.Values.Get(key)
|
|
|
|
if v == "" {
|
|
|
|
v = u.URL.Query().Get(key)
|
|
|
|
}
|
|
|
|
return strings.EqualFold(v, "true")
|
|
|
|
}
|
|
|
|
|
2021-02-12 03:14:15 +00:00
|
|
|
// GetEncoded returns the first value in the uri with the given key, it will
|
2021-02-01 22:16:08 +00:00
|
|
|
// return empty nil if that field is not present or is empty. If the return
|
|
|
|
// value is hex encoded it will decode it and return it.
|
|
|
|
func (u *URI) GetEncoded(key string) []byte {
|
|
|
|
v := u.Get(key)
|
2021-01-27 04:03:53 +00:00
|
|
|
if v == "" {
|
2021-02-01 22:16:08 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if len(v)%2 == 0 {
|
|
|
|
if b, err := hex.DecodeString(v); err == nil {
|
|
|
|
return b
|
|
|
|
}
|
2021-01-27 04:03:53 +00:00
|
|
|
}
|
2021-02-01 22:16:08 +00:00
|
|
|
return []byte(v)
|
2021-01-27 04:03:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Pin returns the pin encoded in the url. It will read the pin from the
|
|
|
|
// pin-value or the pin-source attributes.
|
|
|
|
func (u *URI) Pin() string {
|
|
|
|
if value := u.Get("pin-value"); value != "" {
|
2021-02-01 22:16:08 +00:00
|
|
|
return value
|
2021-01-27 04:03:53 +00:00
|
|
|
}
|
|
|
|
if path := u.Get("pin-source"); path != "" {
|
|
|
|
if b, err := readFile(path); err == nil {
|
|
|
|
return string(bytes.TrimRightFunc(b, unicode.IsSpace))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func readFile(path string) ([]byte, error) {
|
|
|
|
u, err := url.Parse(path)
|
|
|
|
if err == nil && (u.Scheme == "" || u.Scheme == "file") && u.Path != "" {
|
|
|
|
path = u.Path
|
|
|
|
}
|
|
|
|
b, err := ioutil.ReadFile(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "error reading %s", path)
|
|
|
|
}
|
|
|
|
return b, nil
|
|
|
|
}
|