mirror of
https://github.com/ManyakRus/crud_generator.git
synced 2025-01-20 09:09:22 +02:00
1344 lines
38 KiB
Go
1344 lines
38 KiB
Go
|
package faker
|
||
|
|
||
|
// Faker is a simple fake data generator for your own struct.
|
||
|
// Save your time, and Fake your data for your testing now.
|
||
|
import (
|
||
|
cryptorand "crypto/rand"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
mathrand "math/rand"
|
||
|
"reflect"
|
||
|
"regexp"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
"time"
|
||
|
|
||
|
"github.com/bxcodec/faker/v3/support/slice"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
mu = &sync.Mutex{}
|
||
|
// Sets nil if the value type is struct or map and the size of it equals to zero.
|
||
|
shouldSetNil = false
|
||
|
//Sets random integer generation to zero for slice and maps
|
||
|
testRandZero = false
|
||
|
//Sets the default number of string when it is created randomly.
|
||
|
randomStringLen = 25
|
||
|
//Sets the boundary for random integer value generation. Boundaries can not exceed integer(4 byte...)
|
||
|
iBoundary = intBoundary{start: 0, end: 100}
|
||
|
//Sets the boundary for random float value generation. Boundaries should comply with float values constraints (IEEE 754)
|
||
|
fBoundary = floatBoundary{start: 0, end: 100}
|
||
|
//Sets the random max size for slices and maps.
|
||
|
randomMaxSize = 100
|
||
|
//Sets the random min size for slices and maps.
|
||
|
randomMinSize = 0
|
||
|
// Sets the single fake data generator to generate unique values
|
||
|
generateUniqueValues = false
|
||
|
// Sets whether interface{}s should be ignored.
|
||
|
ignoreInterface = false
|
||
|
// Unique values are kept in memory so the generator retries if the value already exists
|
||
|
uniqueValues = map[string][]interface{}{}
|
||
|
// Lang is selected language for random string generator
|
||
|
lang = LangENG
|
||
|
// How much tries for generating random string
|
||
|
maxGenerateStringRetries = 1000000
|
||
|
)
|
||
|
|
||
|
type intBoundary struct {
|
||
|
start int
|
||
|
end int
|
||
|
}
|
||
|
|
||
|
type floatBoundary struct {
|
||
|
start float64
|
||
|
end float64
|
||
|
}
|
||
|
|
||
|
type langRuneBoundary struct {
|
||
|
start rune
|
||
|
end rune
|
||
|
exclude []rune
|
||
|
}
|
||
|
|
||
|
// Language rune boundaries here
|
||
|
var (
|
||
|
// LangENG is for english language
|
||
|
LangENG = langRuneBoundary{65, 122, []rune{91, 92, 93, 94, 95, 96}}
|
||
|
// LangCHI is for chinese language
|
||
|
LangCHI = langRuneBoundary{19968, 40869, nil}
|
||
|
// LangRUS is for russian language
|
||
|
LangRUS = langRuneBoundary{1025, 1105, nil}
|
||
|
// LangJPN is for japanese Hiragana Katakana language
|
||
|
LangJPN = langRuneBoundary{12353, 12534, []rune{12436, 12437, 12438, 12439, 12440, 12441, 12442, 12443, 12444, 12445, 12446, 12447, 12448}}
|
||
|
// LangKOR is for korean Hangul language
|
||
|
LangKOR = langRuneBoundary{44032, 55203, nil}
|
||
|
// EmotEMJ is for emoticons
|
||
|
EmotEMJ = langRuneBoundary{126976, 129535, nil}
|
||
|
)
|
||
|
|
||
|
// Supported tags
|
||
|
const (
|
||
|
letterIdxBits = 6 // 6 bits to represent a letter index
|
||
|
letterIdxMask = 1<<letterIdxBits - 1 // All 1-bits, as many as letterIdxBits
|
||
|
letterIdxMax = 63 / letterIdxBits // # of letter indices fitting in 63 bits
|
||
|
maxRetry = 10000 // max number of retry for unique values
|
||
|
tagName = "faker"
|
||
|
keep = "keep"
|
||
|
unique = "unique"
|
||
|
ID = "uuid_digit"
|
||
|
HyphenatedID = "uuid_hyphenated"
|
||
|
EmailTag = "email"
|
||
|
MacAddressTag = "mac_address"
|
||
|
DomainNameTag = "domain_name"
|
||
|
UserNameTag = "username"
|
||
|
URLTag = "url"
|
||
|
IPV4Tag = "ipv4"
|
||
|
IPV6Tag = "ipv6"
|
||
|
PASSWORD = "password"
|
||
|
JWT = "jwt"
|
||
|
LATITUDE = "lat"
|
||
|
LONGITUDE = "long"
|
||
|
CreditCardNumber = "cc_number"
|
||
|
CreditCardType = "cc_type"
|
||
|
PhoneNumber = "phone_number"
|
||
|
TollFreeNumber = "toll_free_number"
|
||
|
E164PhoneNumberTag = "e_164_phone_number"
|
||
|
TitleMaleTag = "title_male"
|
||
|
TitleFemaleTag = "title_female"
|
||
|
FirstNameTag = "first_name"
|
||
|
FirstNameMaleTag = "first_name_male"
|
||
|
FirstNameFemaleTag = "first_name_female"
|
||
|
LastNameTag = "last_name"
|
||
|
NAME = "name"
|
||
|
ChineseFirstNameTag = "chinese_first_name"
|
||
|
ChineseLastNameTag = "chinese_last_name"
|
||
|
ChineseNameTag = "chinese_name"
|
||
|
GENDER = "gender"
|
||
|
UnixTimeTag = "unix_time"
|
||
|
DATE = "date"
|
||
|
TIME = "time"
|
||
|
MonthNameTag = "month_name"
|
||
|
YEAR = "year"
|
||
|
DayOfWeekTag = "day_of_week"
|
||
|
DayOfMonthTag = "day_of_month"
|
||
|
TIMESTAMP = "timestamp"
|
||
|
CENTURY = "century"
|
||
|
TIMEZONE = "timezone"
|
||
|
TimePeriodTag = "time_period"
|
||
|
WORD = "word"
|
||
|
SENTENCE = "sentence"
|
||
|
PARAGRAPH = "paragraph"
|
||
|
CurrencyTag = "currency"
|
||
|
AmountTag = "amount"
|
||
|
AmountWithCurrencyTag = "amount_with_currency"
|
||
|
SKIP = "-"
|
||
|
Length = "len"
|
||
|
SliceLength = "slice_len"
|
||
|
Language = "lang"
|
||
|
BoundaryStart = "boundary_start"
|
||
|
BoundaryEnd = "boundary_end"
|
||
|
Equals = "="
|
||
|
comma = ","
|
||
|
colon = ":"
|
||
|
ONEOF = "oneof"
|
||
|
//period = "."
|
||
|
//hyphen = "-"
|
||
|
)
|
||
|
|
||
|
// PriorityTags define the priority order of the tag
|
||
|
var PriorityTags = []string{ID, HyphenatedID, EmailTag, MacAddressTag, DomainNameTag, UserNameTag, URLTag, IPV4Tag,
|
||
|
IPV6Tag, PASSWORD, JWT, LATITUDE, LONGITUDE, CreditCardNumber, CreditCardType, PhoneNumber, TollFreeNumber,
|
||
|
E164PhoneNumberTag, TitleMaleTag, TitleFemaleTag, FirstNameTag, FirstNameMaleTag, FirstNameFemaleTag, LastNameTag,
|
||
|
NAME, ChineseFirstNameTag, ChineseLastNameTag, ChineseNameTag, GENDER, UnixTimeTag, DATE, TIME, MonthNameTag,
|
||
|
YEAR, DayOfWeekTag, DayOfMonthTag, TIMESTAMP, CENTURY, TIMEZONE, TimePeriodTag, WORD, SENTENCE, PARAGRAPH,
|
||
|
CurrencyTag, AmountTag, AmountWithCurrencyTag, SKIP, Length, SliceLength, Language, BoundaryStart, BoundaryEnd, ONEOF,
|
||
|
}
|
||
|
|
||
|
var defaultTag = map[string]string{
|
||
|
EmailTag: EmailTag,
|
||
|
MacAddressTag: MacAddressTag,
|
||
|
DomainNameTag: DomainNameTag,
|
||
|
URLTag: URLTag,
|
||
|
UserNameTag: UserNameTag,
|
||
|
IPV4Tag: IPV4Tag,
|
||
|
IPV6Tag: IPV6Tag,
|
||
|
PASSWORD: PASSWORD,
|
||
|
JWT: JWT,
|
||
|
CreditCardType: CreditCardType,
|
||
|
CreditCardNumber: CreditCardNumber,
|
||
|
LATITUDE: LATITUDE,
|
||
|
LONGITUDE: LONGITUDE,
|
||
|
PhoneNumber: PhoneNumber,
|
||
|
TollFreeNumber: TollFreeNumber,
|
||
|
E164PhoneNumberTag: E164PhoneNumberTag,
|
||
|
TitleMaleTag: TitleMaleTag,
|
||
|
TitleFemaleTag: TitleFemaleTag,
|
||
|
FirstNameTag: FirstNameTag,
|
||
|
FirstNameMaleTag: FirstNameMaleTag,
|
||
|
FirstNameFemaleTag: FirstNameFemaleTag,
|
||
|
LastNameTag: LastNameTag,
|
||
|
NAME: NAME,
|
||
|
ChineseFirstNameTag: ChineseFirstNameTag,
|
||
|
ChineseLastNameTag: ChineseLastNameTag,
|
||
|
ChineseNameTag: ChineseNameTag,
|
||
|
GENDER: GENDER,
|
||
|
UnixTimeTag: UnixTimeTag,
|
||
|
DATE: DATE,
|
||
|
TIME: TimeFormat,
|
||
|
MonthNameTag: MonthNameTag,
|
||
|
YEAR: YearFormat,
|
||
|
DayOfWeekTag: DayOfWeekTag,
|
||
|
DayOfMonthTag: DayOfMonthFormat,
|
||
|
TIMESTAMP: TIMESTAMP,
|
||
|
CENTURY: CENTURY,
|
||
|
TIMEZONE: TIMEZONE,
|
||
|
TimePeriodTag: TimePeriodFormat,
|
||
|
WORD: WORD,
|
||
|
SENTENCE: SENTENCE,
|
||
|
PARAGRAPH: PARAGRAPH,
|
||
|
CurrencyTag: CurrencyTag,
|
||
|
AmountTag: AmountTag,
|
||
|
AmountWithCurrencyTag: AmountWithCurrencyTag,
|
||
|
ID: ID,
|
||
|
HyphenatedID: HyphenatedID,
|
||
|
}
|
||
|
|
||
|
// TaggedFunction used as the standard layout function for tag providers in struct.
|
||
|
// This type also can be used for custom provider.
|
||
|
type TaggedFunction func(v reflect.Value) (interface{}, error)
|
||
|
|
||
|
var mapperTag = map[string]TaggedFunction{
|
||
|
EmailTag: GetNetworker().Email,
|
||
|
MacAddressTag: GetNetworker().MacAddress,
|
||
|
DomainNameTag: GetNetworker().DomainName,
|
||
|
URLTag: GetNetworker().URL,
|
||
|
UserNameTag: GetNetworker().UserName,
|
||
|
IPV4Tag: GetNetworker().IPv4,
|
||
|
IPV6Tag: GetNetworker().IPv6,
|
||
|
PASSWORD: GetNetworker().Password,
|
||
|
JWT: GetNetworker().Jwt,
|
||
|
CreditCardType: GetPayment().CreditCardType,
|
||
|
CreditCardNumber: GetPayment().CreditCardNumber,
|
||
|
LATITUDE: GetAddress().Latitude,
|
||
|
LONGITUDE: GetAddress().Longitude,
|
||
|
PhoneNumber: GetPhoner().PhoneNumber,
|
||
|
TollFreeNumber: GetPhoner().TollFreePhoneNumber,
|
||
|
E164PhoneNumberTag: GetPhoner().E164PhoneNumber,
|
||
|
TitleMaleTag: GetPerson().TitleMale,
|
||
|
TitleFemaleTag: GetPerson().TitleFeMale,
|
||
|
FirstNameTag: GetPerson().FirstName,
|
||
|
FirstNameMaleTag: GetPerson().FirstNameMale,
|
||
|
FirstNameFemaleTag: GetPerson().FirstNameFemale,
|
||
|
LastNameTag: GetPerson().LastName,
|
||
|
NAME: GetPerson().Name,
|
||
|
ChineseFirstNameTag: GetPerson().ChineseFirstName,
|
||
|
ChineseLastNameTag: GetPerson().ChineseLastName,
|
||
|
ChineseNameTag: GetPerson().ChineseName,
|
||
|
GENDER: GetPerson().Gender,
|
||
|
UnixTimeTag: GetDateTimer().UnixTime,
|
||
|
DATE: GetDateTimer().Date,
|
||
|
TIME: GetDateTimer().Time,
|
||
|
MonthNameTag: GetDateTimer().MonthName,
|
||
|
YEAR: GetDateTimer().Year,
|
||
|
DayOfWeekTag: GetDateTimer().DayOfWeek,
|
||
|
DayOfMonthTag: GetDateTimer().DayOfMonth,
|
||
|
TIMESTAMP: GetDateTimer().Timestamp,
|
||
|
CENTURY: GetDateTimer().Century,
|
||
|
TIMEZONE: GetDateTimer().TimeZone,
|
||
|
TimePeriodTag: GetDateTimer().TimePeriod,
|
||
|
WORD: GetLorem().Word,
|
||
|
SENTENCE: GetLorem().Sentence,
|
||
|
PARAGRAPH: GetLorem().Paragraph,
|
||
|
CurrencyTag: GetPrice().Currency,
|
||
|
AmountTag: GetPrice().Amount,
|
||
|
AmountWithCurrencyTag: GetPrice().AmountWithCurrency,
|
||
|
ID: GetIdentifier().Digit,
|
||
|
HyphenatedID: GetIdentifier().Hyphenated,
|
||
|
}
|
||
|
|
||
|
// Generic Error Messages for tags
|
||
|
// ErrUnsupportedKindPtr: Error when get fake from ptr
|
||
|
// ErrUnsupportedKind: Error on passing unsupported kind
|
||
|
// ErrValueNotPtr: Error when value is not pointer
|
||
|
// ErrTagNotSupported: Error when tag is not supported
|
||
|
// ErrTagAlreadyExists: Error when tag exists and call AddProvider
|
||
|
// ErrTagDoesNotExist: Error when tag does not exist and call RemoveProvider
|
||
|
// ErrMoreArguments: Error on passing more arguments
|
||
|
// ErrNotSupportedPointer: Error when passing unsupported pointer
|
||
|
var (
|
||
|
ErrUnsupportedKindPtr = "Unsupported kind: %s Change Without using * (pointer) in Field of %s"
|
||
|
ErrUnsupportedKind = "Unsupported kind: %s"
|
||
|
ErrValueNotPtr = "Not a pointer value"
|
||
|
ErrTagNotSupported = "Tag unsupported: %s"
|
||
|
ErrTagAlreadyExists = "Tag exists"
|
||
|
ErrTagDoesNotExist = "Tag does not exist"
|
||
|
ErrMoreArguments = "Passed more arguments than is possible : (%d)"
|
||
|
ErrNotSupportedPointer = "Use sample:=new(%s)\n faker.FakeData(sample) instead"
|
||
|
ErrSmallerThanZero = "Size:%d is smaller than zero."
|
||
|
ErrSmallerThanOne = "Size:%d is smaller than one."
|
||
|
ErrUniqueFailure = "Failed to generate a unique value for field \"%s\""
|
||
|
|
||
|
ErrStartValueBiggerThanEnd = "Start value can not be bigger than end value."
|
||
|
ErrWrongFormattedTag = "Tag \"%s\" is not written properly"
|
||
|
ErrUnknownType = "Unknown Type"
|
||
|
ErrNotSupportedTypeForTag = "Type is not supported by tag."
|
||
|
ErrUnsupportedTagArguments = "Tag arguments are not compatible with field type."
|
||
|
ErrDuplicateSeparator = "Duplicate separator for tag arguments."
|
||
|
ErrNotEnoughTagArguments = "Not enough arguments for tag."
|
||
|
ErrUnsupportedNumberType = "Unsupported Number type."
|
||
|
)
|
||
|
|
||
|
// Compiled regexp
|
||
|
var (
|
||
|
findLangReg *regexp.Regexp
|
||
|
findLenReg *regexp.Regexp
|
||
|
findSliceLenReg *regexp.Regexp
|
||
|
)
|
||
|
|
||
|
func init() {
|
||
|
rand = mathrand.New(NewSafeSource(mathrand.NewSource(time.Now().UnixNano())))
|
||
|
crypto = cryptorand.Reader
|
||
|
}
|
||
|
|
||
|
func init() {
|
||
|
findLangReg, _ = regexp.Compile("lang=[a-z]{3}")
|
||
|
findLenReg, _ = regexp.Compile(`len=\d+`)
|
||
|
findSliceLenReg, _ = regexp.Compile(`slice_len=\d+`)
|
||
|
|
||
|
randNameFlag = rand.Intn(100) // for person
|
||
|
}
|
||
|
|
||
|
// ResetUnique is used to forget generated unique values.
|
||
|
// Call this when you're done generating a dataset.
|
||
|
func ResetUnique() {
|
||
|
uniqueValues = map[string][]interface{}{}
|
||
|
}
|
||
|
|
||
|
// SetGenerateUniqueValues allows to set the single fake data generator functions to generate unique data.
|
||
|
func SetGenerateUniqueValues(unique bool) {
|
||
|
generateUniqueValues = unique
|
||
|
}
|
||
|
|
||
|
// SetIgnoreInterface allows to set a flag to ignore found interface{}s.
|
||
|
func SetIgnoreInterface(ignore bool) {
|
||
|
ignoreInterface = ignore
|
||
|
}
|
||
|
|
||
|
// SetNilIfLenIsZero allows to set nil for the slice and maps, if size is 0.
|
||
|
func SetNilIfLenIsZero(setNil bool) {
|
||
|
shouldSetNil = setNil
|
||
|
}
|
||
|
|
||
|
// SetRandomStringLength sets a length for random string generation
|
||
|
func SetRandomStringLength(size int) error {
|
||
|
if size < 0 {
|
||
|
return fmt.Errorf(ErrSmallerThanZero, size)
|
||
|
}
|
||
|
randomStringLen = size
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// SetStringLang sets language of random string generation (LangENG, LangCHI, LangRUS, LangJPN, LangKOR, EmotEMJ)
|
||
|
func SetStringLang(l langRuneBoundary) {
|
||
|
lang = l
|
||
|
}
|
||
|
|
||
|
// SetRandomMapAndSliceSize sets the size for maps and slices for random generation.
|
||
|
// deprecates, currently left for old version usage
|
||
|
func SetRandomMapAndSliceSize(size int) error {
|
||
|
return SetRandomMapAndSliceMaxSize(size)
|
||
|
}
|
||
|
|
||
|
// SetRandomMapAndSliceMaxSize sets the max size for maps and slices for random generation.
|
||
|
func SetRandomMapAndSliceMaxSize(size int) error {
|
||
|
if size < 1 {
|
||
|
return fmt.Errorf(ErrSmallerThanOne, size)
|
||
|
}
|
||
|
randomMaxSize = size
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// SetRandomMapAndSliceMinSize sets the min size for maps and slices for random generation.
|
||
|
func SetRandomMapAndSliceMinSize(size int) error {
|
||
|
if size < 0 {
|
||
|
return fmt.Errorf(ErrSmallerThanZero, size)
|
||
|
}
|
||
|
randomMinSize = size
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// SetRandomNumberBoundaries sets boundary for random number generation
|
||
|
func SetRandomNumberBoundaries(start, end int) error {
|
||
|
if start > end {
|
||
|
return errors.New(ErrStartValueBiggerThanEnd)
|
||
|
}
|
||
|
iBoundary = intBoundary{start: start, end: end}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// FakeData is the main function. Will generate a fake data based on your struct. You can use this for automation testing, or anything that need automated data.
|
||
|
// You don't need to Create your own data for your testing.
|
||
|
func FakeData(a interface{}) error {
|
||
|
|
||
|
reflectType := reflect.TypeOf(a)
|
||
|
|
||
|
if reflectType.Kind() != reflect.Ptr {
|
||
|
return errors.New(ErrValueNotPtr)
|
||
|
}
|
||
|
|
||
|
if reflect.ValueOf(a).IsNil() {
|
||
|
return fmt.Errorf(ErrNotSupportedPointer, reflectType.Elem().String())
|
||
|
}
|
||
|
|
||
|
rval := reflect.ValueOf(a)
|
||
|
|
||
|
finalValue, err := getValue(a)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
rval.Elem().Set(finalValue.Elem().Convert(reflectType.Elem()))
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// AddProvider extend faker with tag to generate fake data with specified custom algorithm
|
||
|
// Example:
|
||
|
// type Gondoruwo struct {
|
||
|
// Name string
|
||
|
// Locatadata int
|
||
|
// }
|
||
|
//
|
||
|
// type Sample struct {
|
||
|
// ID int64 `faker:"customIdFaker"`
|
||
|
// Gondoruwo Gondoruwo `faker:"gondoruwo"`
|
||
|
// Danger string `faker:"danger"`
|
||
|
// }
|
||
|
//
|
||
|
// func CustomGenerator() {
|
||
|
// // explicit
|
||
|
// faker.AddProvider("customIdFaker", func(v reflect.Value) (interface{}, error) {
|
||
|
// return int64(43), nil
|
||
|
// })
|
||
|
// // functional
|
||
|
// faker.AddProvider("danger", func() faker.TaggedFunction {
|
||
|
// return func(v reflect.Value) (interface{}, error) {
|
||
|
// return "danger-ranger", nil
|
||
|
// }
|
||
|
// }())
|
||
|
// faker.AddProvider("gondoruwo", func(v reflect.Value) (interface{}, error) {
|
||
|
// obj := Gondoruwo{
|
||
|
// Name: "Power",
|
||
|
// Locatadata: 324,
|
||
|
// }
|
||
|
// return obj, nil
|
||
|
// })
|
||
|
// }
|
||
|
//
|
||
|
// func main() {
|
||
|
// CustomGenerator()
|
||
|
// var sample Sample
|
||
|
// faker.FakeData(&sample)
|
||
|
// fmt.Printf("%+v", sample)
|
||
|
// }
|
||
|
//
|
||
|
// Will print
|
||
|
// {ID:43 Gondoruwo:{Name:Power Locatadata:324} Danger:danger-ranger}
|
||
|
// Notes: when using a custom provider make sure to return the same type as the field
|
||
|
func AddProvider(tag string, provider TaggedFunction) error {
|
||
|
if _, ok := mapperTag[tag]; ok {
|
||
|
return errors.New(ErrTagAlreadyExists)
|
||
|
}
|
||
|
PriorityTags = append(PriorityTags, tag)
|
||
|
mapperTag[tag] = provider
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// RemoveProvider removes existing customization added with AddProvider
|
||
|
func RemoveProvider(tag string) error {
|
||
|
if _, ok := mapperTag[tag]; !ok {
|
||
|
return errors.New(ErrTagDoesNotExist)
|
||
|
}
|
||
|
|
||
|
delete(mapperTag, tag)
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func getValue(a interface{}) (reflect.Value, error) {
|
||
|
t := reflect.TypeOf(a)
|
||
|
if t == nil {
|
||
|
if ignoreInterface {
|
||
|
return reflect.New(reflect.TypeOf(reflect.Struct)), nil
|
||
|
}
|
||
|
return reflect.Value{}, fmt.Errorf("interface{} not allowed")
|
||
|
}
|
||
|
k := t.Kind()
|
||
|
|
||
|
switch k {
|
||
|
case reflect.Ptr:
|
||
|
v := reflect.New(t.Elem())
|
||
|
var val reflect.Value
|
||
|
var err error
|
||
|
if a != reflect.Zero(reflect.TypeOf(a)).Interface() {
|
||
|
val, err = getValue(reflect.ValueOf(a).Elem().Interface())
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
} else {
|
||
|
val, err = getValue(v.Elem().Interface())
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
}
|
||
|
v.Elem().Set(val.Convert(t.Elem()))
|
||
|
return v, nil
|
||
|
case reflect.Struct:
|
||
|
switch t.String() {
|
||
|
case "time.Time":
|
||
|
ft := time.Now().Add(time.Duration(rand.Int63()))
|
||
|
return reflect.ValueOf(ft), nil
|
||
|
default:
|
||
|
originalDataVal := reflect.ValueOf(a)
|
||
|
v := reflect.New(t).Elem()
|
||
|
retry := 0 // error if cannot generate unique value after maxRetry tries
|
||
|
for i := 0; i < v.NumField(); i++ {
|
||
|
if !v.Field(i).CanSet() {
|
||
|
continue // to avoid panic to set on unexported field in struct
|
||
|
}
|
||
|
tags := decodeTags(t, i)
|
||
|
switch {
|
||
|
case tags.keepOriginal:
|
||
|
zero, err := isZero(reflect.ValueOf(a).Field(i))
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
if zero {
|
||
|
err := setDataWithTag(v.Field(i).Addr(), tags.fieldType)
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
continue
|
||
|
}
|
||
|
v.Field(i).Set(reflect.ValueOf(a).Field(i))
|
||
|
case tags.fieldType == "":
|
||
|
val, err := getValue(v.Field(i).Interface())
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
val = val.Convert(v.Field(i).Type())
|
||
|
v.Field(i).Set(val)
|
||
|
case tags.fieldType == SKIP:
|
||
|
item := originalDataVal.Field(i).Interface()
|
||
|
if v.CanSet() && item != nil {
|
||
|
v.Field(i).Set(reflect.ValueOf(item))
|
||
|
}
|
||
|
default:
|
||
|
err := setDataWithTag(v.Field(i).Addr(), tags.fieldType)
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if tags.unique {
|
||
|
|
||
|
if retry >= maxRetry {
|
||
|
return reflect.Value{}, fmt.Errorf(ErrUniqueFailure, reflect.TypeOf(a).Field(i).Name)
|
||
|
}
|
||
|
|
||
|
value := v.Field(i).Interface()
|
||
|
if slice.ContainsValue(uniqueValues[tags.fieldType], value) { // Retry if unique value already found
|
||
|
i--
|
||
|
retry++
|
||
|
continue
|
||
|
}
|
||
|
retry = 0
|
||
|
uniqueValues[tags.fieldType] = append(uniqueValues[tags.fieldType], value)
|
||
|
} else {
|
||
|
retry = 0
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return v, nil
|
||
|
}
|
||
|
|
||
|
case reflect.String:
|
||
|
res, err := randomString(randomStringLen, &lang)
|
||
|
return reflect.ValueOf(res), err
|
||
|
case reflect.Slice:
|
||
|
len := randomSliceAndMapSize()
|
||
|
if shouldSetNil && len == 0 {
|
||
|
return reflect.Zero(t), nil
|
||
|
}
|
||
|
v := reflect.MakeSlice(t, len, len)
|
||
|
for i := 0; i < v.Len(); i++ {
|
||
|
val, err := getValue(v.Index(i).Interface())
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
val = val.Convert(v.Index(i).Type())
|
||
|
v.Index(i).Set(val)
|
||
|
}
|
||
|
return v, nil
|
||
|
case reflect.Array:
|
||
|
v := reflect.New(t).Elem()
|
||
|
for i := 0; i < v.Len(); i++ {
|
||
|
val, err := getValue(v.Index(i).Interface())
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
val = val.Convert(v.Index(i).Type())
|
||
|
v.Index(i).Set(val)
|
||
|
}
|
||
|
return v, nil
|
||
|
case reflect.Int:
|
||
|
return reflect.ValueOf(randomInteger()), nil
|
||
|
case reflect.Int8:
|
||
|
return reflect.ValueOf(int8(randomInteger())), nil
|
||
|
case reflect.Int16:
|
||
|
return reflect.ValueOf(int16(randomInteger())), nil
|
||
|
case reflect.Int32:
|
||
|
return reflect.ValueOf(int32(randomInteger())), nil
|
||
|
case reflect.Int64:
|
||
|
return reflect.ValueOf(int64(randomInteger())), nil
|
||
|
case reflect.Float32:
|
||
|
return reflect.ValueOf(float32(randomFloat())), nil
|
||
|
case reflect.Float64:
|
||
|
return reflect.ValueOf(randomFloat()), nil
|
||
|
case reflect.Bool:
|
||
|
val := rand.Intn(2) > 0
|
||
|
return reflect.ValueOf(val), nil
|
||
|
|
||
|
case reflect.Uint:
|
||
|
return reflect.ValueOf(uint(randomInteger())), nil
|
||
|
|
||
|
case reflect.Uint8:
|
||
|
return reflect.ValueOf(uint8(randomInteger())), nil
|
||
|
|
||
|
case reflect.Uint16:
|
||
|
return reflect.ValueOf(uint16(randomInteger())), nil
|
||
|
|
||
|
case reflect.Uint32:
|
||
|
return reflect.ValueOf(uint32(randomInteger())), nil
|
||
|
|
||
|
case reflect.Uint64:
|
||
|
return reflect.ValueOf(uint64(randomInteger())), nil
|
||
|
|
||
|
case reflect.Map:
|
||
|
len := randomSliceAndMapSize()
|
||
|
if shouldSetNil && len == 0 {
|
||
|
return reflect.Zero(t), nil
|
||
|
}
|
||
|
v := reflect.MakeMap(t)
|
||
|
for i := 0; i < len; i++ {
|
||
|
keyInstance := reflect.New(t.Key()).Elem().Interface()
|
||
|
key, err := getValue(keyInstance)
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
|
||
|
valueInstance := reflect.New(t.Elem()).Elem().Interface()
|
||
|
val, err := getValue(valueInstance)
|
||
|
if err != nil {
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
v.SetMapIndex(key, val)
|
||
|
}
|
||
|
return v, nil
|
||
|
default:
|
||
|
err := fmt.Errorf("no support for kind %+v", t)
|
||
|
return reflect.Value{}, err
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
func isZero(field reflect.Value) (bool, error) {
|
||
|
if field.Kind() == reflect.Map {
|
||
|
return field.Len() == 0, nil
|
||
|
}
|
||
|
|
||
|
for _, kind := range []reflect.Kind{reflect.Struct, reflect.Slice, reflect.Array} {
|
||
|
if kind == field.Kind() {
|
||
|
return false, fmt.Errorf("keep not allowed on struct")
|
||
|
}
|
||
|
}
|
||
|
return reflect.Zero(field.Type()).Interface() == field.Interface(), nil
|
||
|
}
|
||
|
|
||
|
func decodeTags(typ reflect.Type, i int) structTag {
|
||
|
tagField := typ.Field(i).Tag.Get(tagName)
|
||
|
tags := strings.Split(tagField, ",")
|
||
|
|
||
|
keepOriginal := false
|
||
|
uni := false
|
||
|
res := make([]string, 0)
|
||
|
pMap := make(map[string]string)
|
||
|
for _, tag := range tags {
|
||
|
if tag == keep {
|
||
|
keepOriginal = true
|
||
|
continue
|
||
|
} else if tag == unique {
|
||
|
uni = true
|
||
|
continue
|
||
|
}
|
||
|
// res = append(res, tag)
|
||
|
ptag := strings.ToLower(strings.Trim(strings.Split(tag, "=")[0], " "))
|
||
|
pMap[ptag] = tag
|
||
|
ptag = strings.ToLower(strings.Trim(strings.Split(tag, ":")[0], " "))
|
||
|
pMap[ptag] = tag
|
||
|
}
|
||
|
// Priority
|
||
|
for _, ptag := range PriorityTags {
|
||
|
if tag, ok := pMap[ptag]; ok {
|
||
|
if ptag == ONEOF {
|
||
|
res = append(res, tags...)
|
||
|
} else {
|
||
|
res = append(res, tag)
|
||
|
}
|
||
|
delete(pMap, ptag)
|
||
|
}
|
||
|
}
|
||
|
// custom,keep,unique
|
||
|
if len(res) < 1 {
|
||
|
if !keepOriginal && !uni {
|
||
|
res = append(res, tags...)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return structTag{
|
||
|
fieldType: strings.Join(res, ","),
|
||
|
unique: uni,
|
||
|
keepOriginal: keepOriginal,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type structTag struct {
|
||
|
fieldType string
|
||
|
unique bool
|
||
|
keepOriginal bool
|
||
|
}
|
||
|
|
||
|
func setDataWithTag(v reflect.Value, tag string) error {
|
||
|
|
||
|
if v.Kind() != reflect.Ptr {
|
||
|
return errors.New(ErrValueNotPtr)
|
||
|
}
|
||
|
v = reflect.Indirect(v)
|
||
|
switch v.Kind() {
|
||
|
case reflect.Ptr:
|
||
|
if _, exist := mapperTag[tag]; !exist {
|
||
|
return fmt.Errorf(ErrTagNotSupported, tag)
|
||
|
}
|
||
|
if _, def := defaultTag[tag]; !def {
|
||
|
res, err := mapperTag[tag](v)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
v.Set(reflect.ValueOf(res))
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
t := v.Type()
|
||
|
newv := reflect.New(t.Elem())
|
||
|
res, err := mapperTag[tag](newv.Elem())
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
rval := reflect.ValueOf(res)
|
||
|
newv.Elem().Set(rval)
|
||
|
v.Set(newv)
|
||
|
return nil
|
||
|
case reflect.String:
|
||
|
return userDefinedString(v, tag)
|
||
|
case reflect.Int, reflect.Int32, reflect.Int64, reflect.Int8, reflect.Int16, reflect.Uint, reflect.Uint8,
|
||
|
reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Float32, reflect.Float64:
|
||
|
return userDefinedNumber(v, tag)
|
||
|
case reflect.Slice, reflect.Array:
|
||
|
return userDefinedArray(v, tag)
|
||
|
case reflect.Map:
|
||
|
return userDefinedMap(v, tag)
|
||
|
default:
|
||
|
if _, exist := mapperTag[tag]; !exist {
|
||
|
return fmt.Errorf(ErrTagNotSupported, tag)
|
||
|
}
|
||
|
res, err := mapperTag[tag](v)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
v.Set(reflect.ValueOf(res))
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func userDefinedMap(v reflect.Value, tag string) error {
|
||
|
if tagFunc, ok := mapperTag[tag]; ok {
|
||
|
res, err := tagFunc(v)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
v.Set(reflect.ValueOf(res))
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
len := randomSliceAndMapSize()
|
||
|
if shouldSetNil && len == 0 {
|
||
|
v.Set(reflect.Zero(v.Type()))
|
||
|
return nil
|
||
|
}
|
||
|
definedMap := reflect.MakeMap(v.Type())
|
||
|
for i := 0; i < len; i++ {
|
||
|
key, err := getValueWithTag(v.Type().Key(), tag)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
val, err := getValueWithTag(v.Type().Elem(), tag)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
definedMap.SetMapIndex(reflect.ValueOf(key), reflect.ValueOf(val))
|
||
|
}
|
||
|
v.Set(definedMap)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func getValueWithTag(t reflect.Type, tag string) (interface{}, error) {
|
||
|
switch t.Kind() {
|
||
|
case reflect.Int, reflect.Int32, reflect.Int64, reflect.Int8, reflect.Int16, reflect.Uint, reflect.Uint8,
|
||
|
reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Float32, reflect.Float64:
|
||
|
res, err := extractNumberFromTag(tag, t)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return res, nil
|
||
|
case reflect.String:
|
||
|
res, err := extractStringFromTag(tag)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return res, nil
|
||
|
default:
|
||
|
return 0, errors.New(ErrUnknownType)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func userDefinedArray(v reflect.Value, tag string) error {
|
||
|
_, tagExists := mapperTag[tag]
|
||
|
if tagExists {
|
||
|
res, err := mapperTag[tag](v)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
v.Set(reflect.ValueOf(res))
|
||
|
return nil
|
||
|
}
|
||
|
sliceLen, err := extractSliceLengthFromTag(tag)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if shouldSetNil && sliceLen == 0 {
|
||
|
v.Set(reflect.Zero(v.Type()))
|
||
|
return nil
|
||
|
}
|
||
|
array := reflect.MakeSlice(v.Type(), sliceLen, sliceLen)
|
||
|
for i := 0; i < sliceLen; i++ {
|
||
|
res, err := getValueWithTag(v.Type().Elem(), tag)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
array.Index(i).Set(reflect.ValueOf(res))
|
||
|
}
|
||
|
v.Set(array)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func userDefinedString(v reflect.Value, tag string) error {
|
||
|
var res interface{}
|
||
|
var err error
|
||
|
|
||
|
if tagFunc, ok := mapperTag[tag]; ok {
|
||
|
res, err = tagFunc(v)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
} else {
|
||
|
res, err = extractStringFromTag(tag)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
if res == nil {
|
||
|
return fmt.Errorf(ErrTagNotSupported, tag)
|
||
|
}
|
||
|
val, _ := res.(string)
|
||
|
v.SetString(val)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func userDefinedNumber(v reflect.Value, tag string) error {
|
||
|
var res interface{}
|
||
|
var err error
|
||
|
|
||
|
if tagFunc, ok := mapperTag[tag]; ok {
|
||
|
res, err = tagFunc(v)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
} else {
|
||
|
res, err = extractNumberFromTag(tag, v.Type())
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
if res == nil {
|
||
|
return fmt.Errorf(ErrTagNotSupported, tag)
|
||
|
}
|
||
|
|
||
|
v.Set(reflect.ValueOf(res))
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
//extractSliceLengthFromTag checks if the sliceLength tag 'slice_len' is set, if so, returns its value, else return a random length
|
||
|
func extractSliceLengthFromTag(tag string) (int, error) {
|
||
|
if strings.Contains(tag, SliceLength) {
|
||
|
lenParts := strings.SplitN(findSliceLenReg.FindString(tag), Equals, -1)
|
||
|
if len(lenParts) != 2 {
|
||
|
return 0, fmt.Errorf(ErrWrongFormattedTag, tag)
|
||
|
}
|
||
|
sliceLen, err := strconv.Atoi(lenParts[1])
|
||
|
if err != nil {
|
||
|
return 0, fmt.Errorf("the given sliceLength has to be numeric, tag: %s", tag)
|
||
|
}
|
||
|
if sliceLen < 0 {
|
||
|
return 0, fmt.Errorf("slice length can not be negative, tag: %s", tag)
|
||
|
}
|
||
|
return sliceLen, nil
|
||
|
}
|
||
|
|
||
|
return randomSliceAndMapSize(), nil //Returns random slice length if the sliceLength tag isn't set
|
||
|
}
|
||
|
|
||
|
func extractStringFromTag(tag string) (interface{}, error) {
|
||
|
var err error
|
||
|
strlen := randomStringLen
|
||
|
strlng := &lang
|
||
|
isOneOfTag := strings.Contains(tag, ONEOF)
|
||
|
if !strings.Contains(tag, Length) && !strings.Contains(tag, Language) && !isOneOfTag {
|
||
|
return nil, fmt.Errorf(ErrTagNotSupported, tag)
|
||
|
}
|
||
|
if strings.Contains(tag, Length) {
|
||
|
lenParts := strings.SplitN(findLenReg.FindString(tag), Equals, -1)
|
||
|
if len(lenParts) != 2 {
|
||
|
return nil, fmt.Errorf(ErrWrongFormattedTag, tag)
|
||
|
}
|
||
|
strlen, _ = strconv.Atoi(lenParts[1])
|
||
|
}
|
||
|
if strings.Contains(tag, Language) {
|
||
|
strlng, err = extractLangFromTag(tag)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf(ErrWrongFormattedTag, tag)
|
||
|
}
|
||
|
}
|
||
|
if isOneOfTag {
|
||
|
var args []string
|
||
|
args, err = fetchOneOfArgsFromTag(tag)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
toRet := args[rand.Intn(len(args))]
|
||
|
return strings.TrimSpace(toRet), nil
|
||
|
}
|
||
|
res, err := randomString(strlen, strlng)
|
||
|
return res, err
|
||
|
}
|
||
|
|
||
|
func extractLangFromTag(tag string) (*langRuneBoundary, error) {
|
||
|
text := findLangReg.FindString(tag)
|
||
|
texts := strings.SplitN(text, Equals, -1)
|
||
|
if len(texts) != 2 {
|
||
|
return nil, fmt.Errorf(ErrWrongFormattedTag, text)
|
||
|
}
|
||
|
switch strings.ToLower(texts[1]) {
|
||
|
case "eng":
|
||
|
return &LangENG, nil
|
||
|
case "rus":
|
||
|
return &LangRUS, nil
|
||
|
case "chi":
|
||
|
return &LangCHI, nil
|
||
|
case "jpn":
|
||
|
return &LangJPN, nil
|
||
|
case "kor":
|
||
|
return &LangKOR, nil
|
||
|
case "emj":
|
||
|
return &EmotEMJ, nil
|
||
|
default:
|
||
|
return &LangENG, nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func extractNumberFromTag(tag string, t reflect.Type) (interface{}, error) {
|
||
|
hasOneOf := strings.Contains(tag, ONEOF)
|
||
|
hasBoundaryStart := strings.Contains(tag, BoundaryStart)
|
||
|
hasBoundaryEnd := strings.Contains(tag, BoundaryEnd)
|
||
|
usingOneOfTag := hasOneOf && (!hasBoundaryStart && !hasBoundaryEnd)
|
||
|
usingBoundariesTags := !hasOneOf && (hasBoundaryStart && hasBoundaryEnd)
|
||
|
if !usingOneOfTag && !usingBoundariesTags {
|
||
|
return nil, fmt.Errorf(ErrTagNotSupported, tag)
|
||
|
}
|
||
|
|
||
|
// handling oneof tag
|
||
|
if usingOneOfTag {
|
||
|
args, err := fetchOneOfArgsFromTag(tag)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
switch t.Kind() {
|
||
|
case reflect.Float64:
|
||
|
{
|
||
|
toRet, err := extractFloat64FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(float64), nil
|
||
|
}
|
||
|
case reflect.Float32:
|
||
|
{
|
||
|
toRet, err := extractFloat32FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(float32), nil
|
||
|
}
|
||
|
case reflect.Int64:
|
||
|
{
|
||
|
toRet, err := extractInt64FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(int64), nil
|
||
|
}
|
||
|
case reflect.Int32:
|
||
|
{
|
||
|
toRet, err := extractInt32FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(int32), nil
|
||
|
}
|
||
|
case reflect.Int16:
|
||
|
{
|
||
|
toRet, err := extractInt16FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(int16), nil
|
||
|
}
|
||
|
case reflect.Int8:
|
||
|
{
|
||
|
toRet, err := extractInt8FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(int8), nil
|
||
|
}
|
||
|
case reflect.Int:
|
||
|
{
|
||
|
toRet, err := extractIntFromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(int), nil
|
||
|
}
|
||
|
case reflect.Uint64:
|
||
|
{
|
||
|
toRet, err := extractUint64FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(uint64), nil
|
||
|
}
|
||
|
case reflect.Uint32:
|
||
|
{
|
||
|
toRet, err := extractUint32FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(uint32), nil
|
||
|
}
|
||
|
case reflect.Uint16:
|
||
|
{
|
||
|
toRet, err := extractUint16FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(uint16), nil
|
||
|
}
|
||
|
case reflect.Uint8:
|
||
|
{
|
||
|
toRet, err := extractUint8FromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(uint8), nil
|
||
|
}
|
||
|
case reflect.Uint:
|
||
|
{
|
||
|
toRet, err := extractUintFromTagArgs(args)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return toRet.(uint), nil
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
return nil, fmt.Errorf(ErrUnsupportedNumberType)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// handling boundary tags
|
||
|
valuesStr := strings.SplitN(tag, comma, -1)
|
||
|
if len(valuesStr) != 2 {
|
||
|
return nil, fmt.Errorf(ErrWrongFormattedTag, tag)
|
||
|
}
|
||
|
|
||
|
// TODO(Xaspy): When Golang provides generics, we will be able to make this method simpler and more beautiful.
|
||
|
if t.Kind() == reflect.Float64 || t.Kind() == reflect.Float32 {
|
||
|
startBoundary, err := extractFloatFromText(valuesStr[0])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
endBoundary, err := extractFloatFromText(valuesStr[1])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
boundary := floatBoundary{start: startBoundary, end: endBoundary}
|
||
|
switch t.Kind() {
|
||
|
case reflect.Float32:
|
||
|
return float32(randomFloatWithBoundary(boundary)), nil
|
||
|
case reflect.Float64:
|
||
|
return randomFloatWithBoundary(boundary), nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
startBoundary, err := extractIntFromText(valuesStr[0])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
endBoundary, err := extractIntFromText(valuesStr[1])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
boundary := intBoundary{start: startBoundary, end: endBoundary}
|
||
|
switch t.Kind() {
|
||
|
case reflect.Uint:
|
||
|
return uint(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Uint8:
|
||
|
return uint8(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Uint16:
|
||
|
return uint16(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Uint32:
|
||
|
return uint32(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Uint64:
|
||
|
return uint64(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Int:
|
||
|
return randomIntegerWithBoundary(boundary), nil
|
||
|
case reflect.Int8:
|
||
|
return int8(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Int16:
|
||
|
return int16(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Int32:
|
||
|
return int32(randomIntegerWithBoundary(boundary)), nil
|
||
|
case reflect.Int64:
|
||
|
return int64(randomIntegerWithBoundary(boundary)), nil
|
||
|
default:
|
||
|
return nil, errors.New(ErrNotSupportedTypeForTag)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func extractIntFromText(text string) (int, error) {
|
||
|
text = strings.TrimSpace(text)
|
||
|
texts := strings.SplitN(text, Equals, -1)
|
||
|
if len(texts) != 2 {
|
||
|
return 0, fmt.Errorf(ErrWrongFormattedTag, text)
|
||
|
}
|
||
|
return strconv.Atoi(texts[1])
|
||
|
}
|
||
|
|
||
|
func extractFloatFromText(text string) (float64, error) {
|
||
|
text = strings.TrimSpace(text)
|
||
|
texts := strings.SplitN(text, Equals, -1)
|
||
|
if len(texts) != 2 {
|
||
|
return 0, fmt.Errorf(ErrWrongFormattedTag, text)
|
||
|
}
|
||
|
return strconv.ParseFloat(texts[1], 64)
|
||
|
}
|
||
|
|
||
|
func fetchOneOfArgsFromTag(tag string) ([]string, error) {
|
||
|
items := strings.Split(tag, colon)
|
||
|
argsList := items[1:]
|
||
|
if len(argsList) != 1 {
|
||
|
return nil, fmt.Errorf(ErrUnsupportedTagArguments)
|
||
|
}
|
||
|
if strings.Contains(argsList[0], ",,") {
|
||
|
return nil, fmt.Errorf(ErrDuplicateSeparator)
|
||
|
}
|
||
|
if argsList[0] == "" {
|
||
|
return nil, fmt.Errorf(ErrNotEnoughTagArguments)
|
||
|
}
|
||
|
args := strings.Split(argsList[0], comma)
|
||
|
if len(args) < 1 {
|
||
|
return nil, fmt.Errorf(ErrNotEnoughTagArguments)
|
||
|
}
|
||
|
return args, nil
|
||
|
}
|
||
|
|
||
|
func randomString(n int, lang *langRuneBoundary) (string, error) {
|
||
|
b := make([]rune, 0)
|
||
|
set := make(map[rune]struct{})
|
||
|
if lang.exclude != nil {
|
||
|
for _, s := range lang.exclude {
|
||
|
set[s] = struct{}{}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
counter := 0
|
||
|
for i := 0; i < n; {
|
||
|
randRune := rune(rand.Intn(int(lang.end-lang.start)) + int(lang.start))
|
||
|
for slice.ContainsRune(set, randRune) {
|
||
|
if counter++; counter >= maxGenerateStringRetries {
|
||
|
return "", errors.New("Max number of string generation retries exhausted")
|
||
|
}
|
||
|
randRune = rune(rand.Intn(int(lang.end-lang.start)) + int(lang.start))
|
||
|
_, ok := set[randRune]
|
||
|
if !ok {
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
b = append(b, randRune)
|
||
|
i++
|
||
|
}
|
||
|
|
||
|
k := string(b)
|
||
|
return k, nil
|
||
|
}
|
||
|
|
||
|
// randomIntegerWithBoundary returns a random integer between input start and end boundary. [start, end)
|
||
|
func randomIntegerWithBoundary(boundary intBoundary) int {
|
||
|
span := boundary.end - boundary.start
|
||
|
if span <= 0 {
|
||
|
return boundary.start
|
||
|
}
|
||
|
return rand.Intn(span) + boundary.start
|
||
|
}
|
||
|
|
||
|
// randomFloatWithBoundary returns a random float between input start and end boundary. [start, end)
|
||
|
func randomFloatWithBoundary(boundary floatBoundary) float64 {
|
||
|
span := boundary.end - boundary.start
|
||
|
if span <= 0 {
|
||
|
return boundary.start
|
||
|
}
|
||
|
return boundary.start + rand.Float64()*span
|
||
|
}
|
||
|
|
||
|
// randomInteger returns a random integer between start and end boundary. [start, end)
|
||
|
func randomInteger() int {
|
||
|
return randomIntegerWithBoundary(iBoundary)
|
||
|
}
|
||
|
|
||
|
// randomFloat returns a random float between start and end boundary. [start, end)
|
||
|
func randomFloat() float64 {
|
||
|
return randomFloatWithBoundary(fBoundary)
|
||
|
}
|
||
|
|
||
|
// randomSliceAndMapSize returns a random integer between [0,randomSliceAndMapSize). If the testRandZero is set, returns 0
|
||
|
// Written for test purposes for shouldSetNil
|
||
|
func randomSliceAndMapSize() int {
|
||
|
if testRandZero {
|
||
|
return 0
|
||
|
}
|
||
|
r := randomMaxSize - randomMinSize
|
||
|
if r < 1 {
|
||
|
r = 1
|
||
|
}
|
||
|
return randomMinSize + rand.Intn(r)
|
||
|
}
|
||
|
|
||
|
func randomElementFromSliceString(s []string) string {
|
||
|
return s[rand.Int()%len(s)]
|
||
|
}
|
||
|
func randomStringNumber(n int) string {
|
||
|
b := make([]byte, n)
|
||
|
for i, cache, remain := n-1, rand.Int63(), letterIdxMax; i >= 0; {
|
||
|
if remain == 0 {
|
||
|
cache, remain = rand.Int63(), letterIdxMax
|
||
|
}
|
||
|
if idx := int(cache & letterIdxMask); idx < len(numberBytes) {
|
||
|
b[i] = numberBytes[idx]
|
||
|
i--
|
||
|
}
|
||
|
cache >>= letterIdxBits
|
||
|
remain--
|
||
|
}
|
||
|
|
||
|
return string(b)
|
||
|
}
|
||
|
|
||
|
// RandomInt Get three parameters , only first mandatory and the rest are optional
|
||
|
// (minimum_int, maximum_int, count)
|
||
|
// If only set one parameter : An integer greater than minimum_int will be returned
|
||
|
// If only set two parameters : All integers between minimum_int and maximum_int will be returned, in a random order.
|
||
|
// If three parameters: `count` integers between minimum_int and maximum_int will be returned.
|
||
|
func RandomInt(parameters ...int) (p []int, err error) {
|
||
|
switch len(parameters) {
|
||
|
case 1:
|
||
|
minInt := parameters[0]
|
||
|
p = rand.Perm(minInt)
|
||
|
for i := range p {
|
||
|
p[i] += minInt
|
||
|
}
|
||
|
case 2:
|
||
|
minInt, maxInt := parameters[0], parameters[1]
|
||
|
p = rand.Perm(maxInt - minInt + 1)
|
||
|
|
||
|
for i := range p {
|
||
|
p[i] += minInt
|
||
|
}
|
||
|
case 3:
|
||
|
minInt, maxInt := parameters[0], parameters[1]
|
||
|
count := parameters[2]
|
||
|
p = rand.Perm(maxInt - minInt + 1)
|
||
|
|
||
|
for i := range p {
|
||
|
p[i] += minInt
|
||
|
}
|
||
|
p = p[0:count]
|
||
|
default:
|
||
|
err = fmt.Errorf(ErrMoreArguments, len(parameters))
|
||
|
}
|
||
|
return p, err
|
||
|
}
|
||
|
|
||
|
func generateUnique(dataType string, fn func() interface{}) (interface{}, error) {
|
||
|
for i := 0; i < maxRetry; i++ {
|
||
|
value := fn()
|
||
|
if !slice.ContainsValue(uniqueValues[dataType], value) { // Retry if unique value already found
|
||
|
uniqueValues[dataType] = append(uniqueValues[dataType], value)
|
||
|
return value, nil
|
||
|
}
|
||
|
}
|
||
|
return reflect.Value{}, fmt.Errorf(ErrUniqueFailure, dataType)
|
||
|
}
|
||
|
|
||
|
func singleFakeData(dataType string, fn func() interface{}) interface{} {
|
||
|
if generateUniqueValues {
|
||
|
v, err := generateUnique(dataType, fn)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return v
|
||
|
}
|
||
|
return fn()
|
||
|
}
|