mirror of
https://github.com/mainflux/mainflux.git
synced 2025-04-27 13:48:49 +08:00

* Move Things and Users to Clients Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * NOISSUE - Update Add and Delete Policies (#1792) * Remove Policy Action Ranks Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Rebase Issues Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix CI Test Errors Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Adding Check on Subject For Clients Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Remove Check Client Exists Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Check When Sharing Clients Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Only Add User to Group When Sharing Things Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Remove clientType Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Minor Fix on ShareClient and Fix Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Policies Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Clean Up Things Authorization Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Tests on RetrieveAll Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Test ShareThing Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Merge Conflicts Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Remove Adding Policies. Only Use Ownership Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Check If Subject is same as Object Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Move Back To Union As Sometimes Policy is Empty and Fails to Evaluate on Ownership Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Entity Type For Failing Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix BUG in policy evaluation Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Comments Regarding checkAdmin Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Tests On Rebase Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Combine Authorize For Things and Users Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Tests On Rebase Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Error on Things SVC `unsupported protocol scheme` Signed-off-by: rodneyosodo <blackd0t@protonmail.com> --------- Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * Fix Bug on Things Authorization Cache (#1810) Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * Use Password instead of username in MQTT handler Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * Simplify MQTT authorization Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * Fix MQTT tests Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * NOISSUE - Add More Functions to SDK (#1811) * Add More Functions to SDK Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Examples to GoDoc Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Update Unassign Interface Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Pass Subject as ID and Not Token on List Channels By Thing Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Bootstrap Errors For Element Check Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add empty line Before Return Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Reorder URLS in things mux Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Listing Things Policies Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Share Thing Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Examples to CLI Docs Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Update Identity To Update Another User Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Identify an Update Policies on Things Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Update Things Policies Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix GoDocs on Disconnect Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Change Authorize To Use AccessRequest Signed-off-by: rodneyosodo <blackd0t@protonmail.com> --------- Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * For Evaluate Policy Use AccessRequest (#1814) Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * NOISSUE - Add SDK Tests (#1812) * Add Things Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Channel Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Certs Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Consumer Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Enrich Group Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Tests For Health Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Tests For Tokens Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Rename SDK for Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Policies Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Linter Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Fix Tests Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Make Variable Defination Inline Signed-off-by: rodneyosodo <blackd0t@protonmail.com> --------- Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * NOISSUE - Make Cache Key Duration Configurable (#1815) * Make Cache Key Duration Configurable Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Rename ENV Var Signed-off-by: rodneyosodo <blackd0t@protonmail.com> --------- Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * NOISSUE - Update GoDocs (#1816) * Add GoDocs Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add Missing GoDoc Files Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Enable godot Signed-off-by: rodneyosodo <blackd0t@protonmail.com> * Add License Information Signed-off-by: rodneyosodo <blackd0t@protonmail.com> --------- Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> * NOISSUE - Add Call Home Client to Mainflux services (#1751) * Move Things and Users to Clients Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: SammyOina <sammyoina@gmail.com> * collect and send data package Signed-off-by: SammyOina <sammyoina@gmail.com> * create telemetry migrations Signed-off-by: SammyOina <sammyoina@gmail.com> * add telemetry endpoints Signed-off-by: SammyOina <sammyoina@gmail.com> * add transport Signed-off-by: SammyOina <sammyoina@gmail.com> * create service Signed-off-by: SammyOina <sammyoina@gmail.com> * remove homing server Signed-off-by: SammyOina <sammyoina@gmail.com> * add call home to adapters Signed-off-by: SammyOina <sammyoina@gmail.com> * add last seen Signed-off-by: SammyOina <sammyoina@gmail.com> * rename logger Signed-off-by: SammyOina <sammyoina@gmail.com> * remove homing client Signed-off-by: SammyOina <sammyoina@gmail.com> * use unmerged repo Signed-off-by: SammyOina <sammyoina@gmail.com> * use renamed module Signed-off-by: SammyOina <sammyoina@gmail.com> * update call home version Signed-off-by: SammyOina <sammyoina@gmail.com> * edit documentation Signed-off-by: SammyOina <sammyoina@gmail.com> * align table Signed-off-by: SammyOina <sammyoina@gmail.com> * use alias for call home client Signed-off-by: SammyOina <sammyoina@gmail.com> * update callhome Signed-off-by: SammyOina <sammyoina@gmail.com> * update call home pkg Signed-off-by: SammyOina <sammyoina@gmail.com> * update call home Signed-off-by: SammyOina <sammyoina@gmail.com> * fix modules Signed-off-by: SammyOina <sammyoina@gmail.com> * use mf build version Signed-off-by: SammyOina <sammyoina@gmail.com> * use mf build version Signed-off-by: SammyOina <sammyoina@gmail.com> * restore default Signed-off-by: SammyOina <sammyoina@gmail.com> * add call home for users and things Signed-off-by: SammyOina <sammyoina@gmail.com> * enable opting on call home Signed-off-by: SammyOina <sammyoina@gmail.com> * remove full stops Signed-off-by: SammyOina <sammyoina@gmail.com> * update callhome client Signed-off-by: SammyOina <sammyoina@gmail.com> * add call home to all services Signed-off-by: SammyOina <sammyoina@gmail.com> * fix build Signed-off-by: SammyOina <sammyoina@gmail.com> * restore sdk tests Signed-off-by: SammyOina <sammyoina@gmail.com> * remove unnecessary changes Signed-off-by: SammyOina <sammyoina@gmail.com> * restore health_test.go Signed-off-by: SammyOina <sammyoina@gmail.com> --------- Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: SammyOina <sammyoina@gmail.com> Co-authored-by: b1ackd0t <blackd0t@protonmail.com> Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> --------- Signed-off-by: dusanb94 <dusan.borovcanin@mainflux.com> Signed-off-by: rodneyosodo <blackd0t@protonmail.com> Signed-off-by: SammyOina <sammyoina@gmail.com> Co-authored-by: b1ackd0t <blackd0t@protonmail.com> Co-authored-by: Sammy Kerata Oina <44265300+SammyOina@users.noreply.github.com>
541 lines
13 KiB
Go
541 lines
13 KiB
Go
package env
|
|
|
|
import (
|
|
"encoding"
|
|
"fmt"
|
|
"net/url"
|
|
"os"
|
|
"reflect"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
"unicode"
|
|
)
|
|
|
|
// nolint: gochecknoglobals
|
|
var (
|
|
defaultBuiltInParsers = map[reflect.Kind]ParserFunc{
|
|
reflect.Bool: func(v string) (interface{}, error) {
|
|
return strconv.ParseBool(v)
|
|
},
|
|
reflect.String: func(v string) (interface{}, error) {
|
|
return v, nil
|
|
},
|
|
reflect.Int: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseInt(v, 10, 32)
|
|
return int(i), err
|
|
},
|
|
reflect.Int16: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseInt(v, 10, 16)
|
|
return int16(i), err
|
|
},
|
|
reflect.Int32: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseInt(v, 10, 32)
|
|
return int32(i), err
|
|
},
|
|
reflect.Int64: func(v string) (interface{}, error) {
|
|
return strconv.ParseInt(v, 10, 64)
|
|
},
|
|
reflect.Int8: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseInt(v, 10, 8)
|
|
return int8(i), err
|
|
},
|
|
reflect.Uint: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseUint(v, 10, 32)
|
|
return uint(i), err
|
|
},
|
|
reflect.Uint16: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseUint(v, 10, 16)
|
|
return uint16(i), err
|
|
},
|
|
reflect.Uint32: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseUint(v, 10, 32)
|
|
return uint32(i), err
|
|
},
|
|
reflect.Uint64: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseUint(v, 10, 64)
|
|
return i, err
|
|
},
|
|
reflect.Uint8: func(v string) (interface{}, error) {
|
|
i, err := strconv.ParseUint(v, 10, 8)
|
|
return uint8(i), err
|
|
},
|
|
reflect.Float64: func(v string) (interface{}, error) {
|
|
return strconv.ParseFloat(v, 64)
|
|
},
|
|
reflect.Float32: func(v string) (interface{}, error) {
|
|
f, err := strconv.ParseFloat(v, 32)
|
|
return float32(f), err
|
|
},
|
|
}
|
|
)
|
|
|
|
func defaultTypeParsers() map[reflect.Type]ParserFunc {
|
|
return map[reflect.Type]ParserFunc{
|
|
reflect.TypeOf(url.URL{}): func(v string) (interface{}, error) {
|
|
u, err := url.Parse(v)
|
|
if err != nil {
|
|
return nil, newParseValueError("unable to parse URL", err)
|
|
}
|
|
return *u, nil
|
|
},
|
|
reflect.TypeOf(time.Nanosecond): func(v string) (interface{}, error) {
|
|
s, err := time.ParseDuration(v)
|
|
if err != nil {
|
|
return nil, newParseValueError("unable to parse duration", err)
|
|
}
|
|
return s, err
|
|
},
|
|
}
|
|
}
|
|
|
|
// ParserFunc defines the signature of a function that can be used within `CustomParsers`.
|
|
type ParserFunc func(v string) (interface{}, error)
|
|
|
|
// OnSetFn is a hook that can be run when a value is set.
|
|
type OnSetFn func(tag string, value interface{}, isDefault bool)
|
|
|
|
// Options for the parser.
|
|
type Options struct {
|
|
// Environment keys and values that will be accessible for the service.
|
|
Environment map[string]string
|
|
|
|
// TagName specifies another tagname to use rather than the default env.
|
|
TagName string
|
|
|
|
// RequiredIfNoDef automatically sets all env as required if they do not
|
|
// declare 'envDefault'.
|
|
RequiredIfNoDef bool
|
|
|
|
// OnSet allows to run a function when a value is set.
|
|
OnSet OnSetFn
|
|
|
|
// Prefix define a prefix for each key.
|
|
Prefix string
|
|
|
|
// UseFieldNameByDefault defines whether or not env should use the field
|
|
// name by default if the `env` key is missing.
|
|
UseFieldNameByDefault bool
|
|
|
|
// Sets to true if we have already configured once.
|
|
configured bool
|
|
}
|
|
|
|
// configure will do the basic configurations and defaults.
|
|
func configure(opts []Options) []Options {
|
|
// If we have already configured the first item
|
|
// of options will have been configured set to true.
|
|
if len(opts) > 0 && opts[0].configured {
|
|
return opts
|
|
}
|
|
|
|
// Created options with defaults.
|
|
opt := Options{
|
|
TagName: "env",
|
|
Environment: toMap(os.Environ()),
|
|
configured: true,
|
|
}
|
|
|
|
// Loop over all opts structs and set
|
|
// to opt if value is not default/empty.
|
|
for _, item := range opts {
|
|
if item.Environment != nil {
|
|
opt.Environment = item.Environment
|
|
}
|
|
if item.TagName != "" {
|
|
opt.TagName = item.TagName
|
|
}
|
|
if item.OnSet != nil {
|
|
opt.OnSet = item.OnSet
|
|
}
|
|
if item.Prefix != "" {
|
|
opt.Prefix = item.Prefix
|
|
}
|
|
opt.UseFieldNameByDefault = item.UseFieldNameByDefault
|
|
opt.RequiredIfNoDef = item.RequiredIfNoDef
|
|
}
|
|
|
|
return []Options{opt}
|
|
}
|
|
|
|
func getOnSetFn(opts []Options) OnSetFn {
|
|
return opts[0].OnSet
|
|
}
|
|
|
|
// getTagName returns the tag name.
|
|
func getTagName(opts []Options) string {
|
|
return opts[0].TagName
|
|
}
|
|
|
|
// getEnvironment returns the environment map.
|
|
func getEnvironment(opts []Options) map[string]string {
|
|
return opts[0].Environment
|
|
}
|
|
|
|
// Parse parses a struct containing `env` tags and loads its values from
|
|
// environment variables.
|
|
func Parse(v interface{}, opts ...Options) error {
|
|
return ParseWithFuncs(v, map[reflect.Type]ParserFunc{}, opts...)
|
|
}
|
|
|
|
// ParseWithFuncs is the same as `Parse` except it also allows the user to pass
|
|
// in custom parsers.
|
|
func ParseWithFuncs(v interface{}, funcMap map[reflect.Type]ParserFunc, opts ...Options) error {
|
|
opts = configure(opts)
|
|
|
|
ptrRef := reflect.ValueOf(v)
|
|
if ptrRef.Kind() != reflect.Ptr {
|
|
return newAggregateError(NotStructPtrError{})
|
|
}
|
|
ref := ptrRef.Elem()
|
|
if ref.Kind() != reflect.Struct {
|
|
return newAggregateError(NotStructPtrError{})
|
|
}
|
|
parsers := defaultTypeParsers()
|
|
for k, v := range funcMap {
|
|
parsers[k] = v
|
|
}
|
|
|
|
return doParse(ref, parsers, opts)
|
|
}
|
|
|
|
func doParse(ref reflect.Value, funcMap map[reflect.Type]ParserFunc, opts []Options) error {
|
|
refType := ref.Type()
|
|
|
|
var agrErr AggregateError
|
|
|
|
for i := 0; i < refType.NumField(); i++ {
|
|
refField := ref.Field(i)
|
|
refTypeField := refType.Field(i)
|
|
|
|
if err := doParseField(refField, refTypeField, funcMap, opts); err != nil {
|
|
if val, ok := err.(AggregateError); ok {
|
|
agrErr.Errors = append(agrErr.Errors, val.Errors...)
|
|
} else {
|
|
agrErr.Errors = append(agrErr.Errors, err)
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(agrErr.Errors) == 0 {
|
|
return nil
|
|
}
|
|
|
|
return agrErr
|
|
}
|
|
|
|
func doParseField(refField reflect.Value, refTypeField reflect.StructField, funcMap map[reflect.Type]ParserFunc, opts []Options) error {
|
|
if !refField.CanSet() {
|
|
return nil
|
|
}
|
|
if reflect.Ptr == refField.Kind() && !refField.IsNil() {
|
|
return ParseWithFuncs(refField.Interface(), funcMap, optsWithPrefix(refTypeField, opts)...)
|
|
}
|
|
if reflect.Struct == refField.Kind() && refField.CanAddr() && refField.Type().Name() == "" {
|
|
return ParseWithFuncs(refField.Addr().Interface(), funcMap, optsWithPrefix(refTypeField, opts)...)
|
|
}
|
|
value, err := get(refTypeField, opts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if value != "" {
|
|
return set(refField, refTypeField, value, funcMap)
|
|
}
|
|
|
|
if reflect.Struct == refField.Kind() {
|
|
return doParse(refField, funcMap, optsWithPrefix(refTypeField, opts))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
const underscore rune = '_'
|
|
|
|
func toEnvName(input string) string {
|
|
var output []rune
|
|
for i, c := range input {
|
|
if i > 0 && output[i-1] != underscore && c != underscore && unicode.ToUpper(c) == c {
|
|
output = append(output, underscore)
|
|
}
|
|
output = append(output, unicode.ToUpper(c))
|
|
}
|
|
return string(output)
|
|
}
|
|
|
|
func get(field reflect.StructField, opts []Options) (val string, err error) {
|
|
var exists bool
|
|
var isDefault bool
|
|
var loadFile bool
|
|
var unset bool
|
|
var notEmpty bool
|
|
|
|
required := opts[0].RequiredIfNoDef
|
|
prefix := opts[0].Prefix
|
|
ownKey, tags := parseKeyForOption(field.Tag.Get(getTagName(opts)))
|
|
if ownKey == "" && opts[0].UseFieldNameByDefault {
|
|
ownKey = toEnvName(field.Name)
|
|
}
|
|
key := prefix + ownKey
|
|
for _, tag := range tags {
|
|
switch tag {
|
|
case "":
|
|
continue
|
|
case "file":
|
|
loadFile = true
|
|
case "required":
|
|
required = true
|
|
case "unset":
|
|
unset = true
|
|
case "notEmpty":
|
|
notEmpty = true
|
|
default:
|
|
return "", newNoSupportedTagOptionError(tag)
|
|
}
|
|
}
|
|
expand := strings.EqualFold(field.Tag.Get("envExpand"), "true")
|
|
defaultValue, defExists := field.Tag.Lookup("envDefault")
|
|
val, exists, isDefault = getOr(key, defaultValue, defExists, getEnvironment(opts))
|
|
|
|
if expand {
|
|
val = os.ExpandEnv(val)
|
|
}
|
|
|
|
if unset {
|
|
defer os.Unsetenv(key)
|
|
}
|
|
|
|
if required && !exists && len(ownKey) > 0 {
|
|
return "", newEnvVarIsNotSet(key)
|
|
}
|
|
|
|
if notEmpty && val == "" {
|
|
return "", newEmptyEnvVarError(key)
|
|
}
|
|
|
|
if loadFile && val != "" {
|
|
filename := val
|
|
val, err = getFromFile(filename)
|
|
if err != nil {
|
|
return "", newLoadFileContentError(filename, key, err)
|
|
}
|
|
}
|
|
|
|
if onSetFn := getOnSetFn(opts); onSetFn != nil {
|
|
onSetFn(key, val, isDefault)
|
|
}
|
|
return val, err
|
|
}
|
|
|
|
// split the env tag's key into the expected key and desired option, if any.
|
|
func parseKeyForOption(key string) (string, []string) {
|
|
opts := strings.Split(key, ",")
|
|
return opts[0], opts[1:]
|
|
}
|
|
|
|
func getFromFile(filename string) (value string, err error) {
|
|
b, err := os.ReadFile(filename)
|
|
return string(b), err
|
|
}
|
|
|
|
func getOr(key, defaultValue string, defExists bool, envs map[string]string) (string, bool, bool) {
|
|
value, exists := envs[key]
|
|
switch {
|
|
case (!exists || key == "") && defExists:
|
|
return defaultValue, true, true
|
|
case exists && value == "" && defExists:
|
|
return defaultValue, true, true
|
|
case !exists:
|
|
return "", false, false
|
|
}
|
|
|
|
return value, true, false
|
|
}
|
|
|
|
func set(field reflect.Value, sf reflect.StructField, value string, funcMap map[reflect.Type]ParserFunc) error {
|
|
if tm := asTextUnmarshaler(field); tm != nil {
|
|
if err := tm.UnmarshalText([]byte(value)); err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
typee := sf.Type
|
|
fieldee := field
|
|
if typee.Kind() == reflect.Ptr {
|
|
typee = typee.Elem()
|
|
fieldee = field.Elem()
|
|
}
|
|
|
|
parserFunc, ok := funcMap[typee]
|
|
if ok {
|
|
val, err := parserFunc(value)
|
|
if err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
|
|
fieldee.Set(reflect.ValueOf(val))
|
|
return nil
|
|
}
|
|
|
|
parserFunc, ok = defaultBuiltInParsers[typee.Kind()]
|
|
if ok {
|
|
val, err := parserFunc(value)
|
|
if err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
|
|
fieldee.Set(reflect.ValueOf(val).Convert(typee))
|
|
return nil
|
|
}
|
|
|
|
switch field.Kind() {
|
|
case reflect.Slice:
|
|
return handleSlice(field, value, sf, funcMap)
|
|
case reflect.Map:
|
|
return handleMap(field, value, sf, funcMap)
|
|
}
|
|
|
|
return newNoParserError(sf)
|
|
}
|
|
|
|
func handleSlice(field reflect.Value, value string, sf reflect.StructField, funcMap map[reflect.Type]ParserFunc) error {
|
|
separator := sf.Tag.Get("envSeparator")
|
|
if separator == "" {
|
|
separator = ","
|
|
}
|
|
parts := strings.Split(value, separator)
|
|
|
|
typee := sf.Type.Elem()
|
|
if typee.Kind() == reflect.Ptr {
|
|
typee = typee.Elem()
|
|
}
|
|
|
|
if _, ok := reflect.New(typee).Interface().(encoding.TextUnmarshaler); ok {
|
|
return parseTextUnmarshalers(field, parts, sf)
|
|
}
|
|
|
|
parserFunc, ok := funcMap[typee]
|
|
if !ok {
|
|
parserFunc, ok = defaultBuiltInParsers[typee.Kind()]
|
|
if !ok {
|
|
return newNoParserError(sf)
|
|
}
|
|
}
|
|
|
|
result := reflect.MakeSlice(sf.Type, 0, len(parts))
|
|
for _, part := range parts {
|
|
r, err := parserFunc(part)
|
|
if err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
v := reflect.ValueOf(r).Convert(typee)
|
|
if sf.Type.Elem().Kind() == reflect.Ptr {
|
|
v = reflect.New(typee)
|
|
v.Elem().Set(reflect.ValueOf(r).Convert(typee))
|
|
}
|
|
result = reflect.Append(result, v)
|
|
}
|
|
field.Set(result)
|
|
return nil
|
|
}
|
|
|
|
func handleMap(field reflect.Value, value string, sf reflect.StructField, funcMap map[reflect.Type]ParserFunc) error {
|
|
keyType := sf.Type.Key()
|
|
keyParserFunc, ok := funcMap[keyType]
|
|
if !ok {
|
|
keyParserFunc, ok = defaultBuiltInParsers[keyType.Kind()]
|
|
if !ok {
|
|
return newNoParserError(sf)
|
|
}
|
|
}
|
|
|
|
elemType := sf.Type.Elem()
|
|
elemParserFunc, ok := funcMap[elemType]
|
|
if !ok {
|
|
elemParserFunc, ok = defaultBuiltInParsers[elemType.Kind()]
|
|
if !ok {
|
|
return newNoParserError(sf)
|
|
}
|
|
}
|
|
|
|
separator := sf.Tag.Get("envSeparator")
|
|
if separator == "" {
|
|
separator = ","
|
|
}
|
|
|
|
result := reflect.MakeMap(sf.Type)
|
|
for _, part := range strings.Split(value, separator) {
|
|
pairs := strings.Split(part, ":")
|
|
if len(pairs) != 2 {
|
|
return newParseError(sf, fmt.Errorf(`%q should be in "key:value" format`, part))
|
|
}
|
|
|
|
key, err := keyParserFunc(pairs[0])
|
|
if err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
|
|
elem, err := elemParserFunc(pairs[1])
|
|
if err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
|
|
result.SetMapIndex(reflect.ValueOf(key).Convert(keyType), reflect.ValueOf(elem).Convert(elemType))
|
|
}
|
|
|
|
field.Set(result)
|
|
return nil
|
|
}
|
|
|
|
func asTextUnmarshaler(field reflect.Value) encoding.TextUnmarshaler {
|
|
if reflect.Ptr == field.Kind() {
|
|
if field.IsNil() {
|
|
field.Set(reflect.New(field.Type().Elem()))
|
|
}
|
|
} else if field.CanAddr() {
|
|
field = field.Addr()
|
|
}
|
|
|
|
tm, ok := field.Interface().(encoding.TextUnmarshaler)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return tm
|
|
}
|
|
|
|
func parseTextUnmarshalers(field reflect.Value, data []string, sf reflect.StructField) error {
|
|
s := len(data)
|
|
elemType := field.Type().Elem()
|
|
slice := reflect.MakeSlice(reflect.SliceOf(elemType), s, s)
|
|
for i, v := range data {
|
|
sv := slice.Index(i)
|
|
kind := sv.Kind()
|
|
if kind == reflect.Ptr {
|
|
sv = reflect.New(elemType.Elem())
|
|
} else {
|
|
sv = sv.Addr()
|
|
}
|
|
tm := sv.Interface().(encoding.TextUnmarshaler)
|
|
if err := tm.UnmarshalText([]byte(v)); err != nil {
|
|
return newParseError(sf, err)
|
|
}
|
|
if kind == reflect.Ptr {
|
|
slice.Index(i).Set(sv)
|
|
}
|
|
}
|
|
|
|
field.Set(slice)
|
|
|
|
return nil
|
|
}
|
|
|
|
func optsWithPrefix(field reflect.StructField, opts []Options) []Options {
|
|
subOpts := make([]Options, len(opts))
|
|
copy(subOpts, opts)
|
|
if prefix := field.Tag.Get("envPrefix"); prefix != "" {
|
|
subOpts[0].Prefix += prefix
|
|
}
|
|
return subOpts
|
|
}
|