Compare commits

...

8 commits

Author SHA1 Message Date
55038ea295 🔀 Merge branch 'release/0.4.0' into stable 2025-03-15 21:06:55 -05:00
4d6cd82b74 📝 Update CHANGELOG 2025-03-15 21:06:16 -05:00
7c016df30f Add MultiGeneratorRandomOrder 2025-03-15 21:00:54 -05:00
2440f55563 ♻️ Replace HashType with Hasher
Compatible with crypto.Hash
2025-03-15 20:09:38 -05:00
1008a064d0 Add Make method to Generator 2025-03-15 16:24:37 -05:00
1677a692d1 Add Random Generator 2025-03-15 15:36:43 -05:00
5c1132e414 💡 Fix some doc blocks 2025-03-15 15:35:40 -05:00
10eb3f2491 🔀 Merge tag 'v0.3.0' into develop
🔖 Lots of Generator changes
2025-03-14 21:58:22 -05:00
16 changed files with 307 additions and 152 deletions

View file

@ -1,5 +1,21 @@
# Changelog # Changelog
### [0.4.0] - 2025-03-15
#### Features
- Add Random Generator
- Add Make method to Generator
- Add MultiGeneratorRandomOrder
#### Changes
- Replace HashType with Hasher: This supports all crypto.Hash
#### Support
- Add some missing doc comments
### [0.3.0] - 2025-03-14 ### [0.3.0] - 2025-03-14
#### Features #### Features

View file

@ -26,7 +26,6 @@ tasks:
desc: Vet go code desc: Vet go code
sources: sources:
- '**/*.go' - '**/*.go'
deps: [gen]
cmds: cmds:
- go vet ./... - go vet ./...
@ -82,7 +81,7 @@ tasks:
test: test:
desc: Run unit tests desc: Run unit tests
deps: [fmt, vet, gen] deps: [fmt, vet]
sources: sources:
- '**/*.go' - '**/*.go'
generates: generates:

View file

@ -9,6 +9,7 @@ type Config struct {
generator Generator generator Generator
} }
// NewConfig returns a new Config with the specified Options.
func NewConfig(options ...Option) Config { func NewConfig(options ...Option) Config {
conf := Config{ conf := Config{
extension: ".txt", extension: ".txt",

View file

@ -1,9 +1,7 @@
package nomino package nomino
import ( import (
"crypto/md5" "crypto"
"crypto/sha1"
"crypto/sha256"
"errors" "errors"
"fmt" "fmt"
"hash" "hash"
@ -42,39 +40,35 @@ func Slug(lang ...string) Generator {
// HashingFunc is a function that generates a hash.Hash // HashingFunc is a function that generates a hash.Hash
type HashingFunc func() hash.Hash type HashingFunc func() hash.Hash
//go:generate stringer -type=HashType -trimprefix=Hash // New allows HashingFunc to be used as a Hasher
func (hf HashingFunc) New() hash.Hash {
// HashType represents a particular hashing algorithm return hf()
type HashType uint8
const (
HashMD5 HashType = iota + 1
HashSHA1
HashSHA256
)
// ErrInvalidHashType is returned by the Hash generator when an invalid HashType is passed
var ErrInvalidHashType = errors.New("invalid hash type")
var hashMap = map[HashType]HashingFunc{
HashMD5: md5.New,
HashSHA1: sha1.New,
HashSHA256: sha256.New,
} }
// Hasher is a type returns a hash.Hash.
// All crypto.Hash may be used.
type Hasher interface {
New() hash.Hash
}
// ErrInvalidHash is returned by the Hash generator when an invalid HashType is passed
var ErrInvalidHash = errors.New("invalid hash type")
// Hash generates a name from a hash of the filename. // Hash generates a name from a hash of the filename.
// When this is used, the original filename will be removed from the final filename. // When this is used, the original filename will be removed from the final filename.
func Hash(t HashType) Generator { func Hash(h Hasher) Generator {
f, ok := hashMap[t] if h == nil {
h = crypto.MD5
}
return func(c *Config) (string, error) { return func(c *Config) (string, error) {
if !ok { if h == crypto.MD5SHA1 {
return "", fmt.Errorf("%w: %s", ErrInvalidHashType, t) return "", ErrInvalidHash
} }
name, err := getOriginal(c) name, err := getOriginal(c)
if err != nil { if err != nil {
return "", err return "", err
} }
hs := f() hs := h.New()
hs.Write([]byte(name)) hs.Write([]byte(name))
return fmt.Sprintf("%x", hs.Sum(nil)), nil return fmt.Sprintf("%x", hs.Sum(nil)), nil
} }

View file

@ -1,66 +1,56 @@
package nomino_test package nomino_test
import ( import (
"crypto"
"crypto/hmac"
"fmt" "fmt"
"hash"
"codeberg.org/danjones000/nomino" "codeberg.org/danjones000/nomino"
) )
func ExampleSlug() { func ExampleSlug() {
conf := nomino.NewConfig( str, _ := nomino.Slug().Make(nomino.WithOriginal("My name is Jimmy"))
nomino.WithOriginal("My name is Jimmy"),
nomino.WithGenerator(nomino.Slug()),
)
str, _ := nomino.Make(conf)
fmt.Println(str) fmt.Println(str)
// Output: my-name-is-jimmy.txt // Output: my-name-is-jimmy.txt
} }
func ExampleSlug_withLang() { func ExampleSlug_withLang() {
conf := nomino.NewConfig( str, _ := nomino.Slug("de").
nomino.WithOriginal("Diese & Dass"), Make(nomino.WithOriginal("Diese & Dass"))
nomino.WithGenerator(nomino.Slug("de")),
)
str, _ := nomino.Make(conf)
fmt.Println(str) fmt.Println(str)
// Output: diese-und-dass.txt // Output: diese-und-dass.txt
} }
func ExampleHash_mD5() { func ExampleHash_mD5() {
conf := nomino.NewConfig( str, _ := nomino.Hash(crypto.MD5).
nomino.WithOriginal("foobar"), Make(nomino.WithOriginal("foobar"))
nomino.WithGenerator(
nomino.Hash(nomino.HashMD5),
),
)
str, _ := nomino.Make(conf)
fmt.Println(str) fmt.Println(str)
// Output: 3858f62230ac3c915f300c664312c63f.txt // Output: 3858f62230ac3c915f300c664312c63f.txt
} }
func ExampleHash_sHA1() { func ExampleHash_sHA1() {
conf := nomino.NewConfig( str, _ := nomino.Hash(crypto.SHA1).
nomino.WithOriginal("foobar"), Make(nomino.WithOriginal("foobar"))
nomino.WithGenerator(
nomino.Hash(nomino.HashSHA1),
),
)
str, _ := nomino.Make(conf)
fmt.Println(str) fmt.Println(str)
// Output: 8843d7f92416211de9ebb963ff4ce28125932878.txt // Output: 8843d7f92416211de9ebb963ff4ce28125932878.txt
} }
func ExampleHash_sHA256() { func ExampleHash_sHA256() {
conf := nomino.NewConfig( str, _ := nomino.Hash(crypto.SHA256).
nomino.WithOriginal("foobar"), Make(nomino.WithOriginal("foobar"))
nomino.WithGenerator(
nomino.Hash(nomino.HashSHA256),
),
)
str, _ := nomino.Make(conf)
fmt.Println(str) fmt.Println(str)
// Output: c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2.txt // Output: c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2.txt
} }
func ExampleHashingFunc_hMAC() {
var hasher nomino.HashingFunc = func() hash.Hash {
return hmac.New(crypto.SHA1.New, []byte("hello"))
}
g := nomino.Hash(hasher)
str, _ := g.Make(nomino.WithOriginal("foobar"))
fmt.Println(str)
// Output: 85f767c284c80a3a59a9635194321d20dd90f31b.txt
}

View file

@ -1,6 +1,7 @@
package nomino package nomino
import ( import (
"crypto"
"testing" "testing"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
@ -22,21 +23,15 @@ func TestSlugRemovesOriginal(t *testing.T) {
} }
func TestHashBadHash(t *testing.T) { func TestHashBadHash(t *testing.T) {
conf := NewConfig(WithOriginal("foobar"), WithGenerator(Hash(0))) conf := NewConfig(WithOriginal("foobar"), WithGenerator(Hash(crypto.MD5SHA1)))
st, err := conf.generator(&conf) st, err := conf.generator(&conf)
assert.Equal(t, "", st) assert.Equal(t, "", st)
assert.ErrorIs(t, err, ErrInvalidHashType) assert.ErrorIs(t, err, ErrInvalidHash)
assert.ErrorContains(t, err, "invalid hash type: HashType(0)")
} }
func TestHashMissingOriginal(t *testing.T) { func TestHashMissingOriginal(t *testing.T) {
conf := NewConfig(WithGenerator(Hash(HashMD5))) conf := NewConfig(WithGenerator(Hash(nil)))
st, err := conf.generator(&conf) st, err := conf.generator(&conf)
assert.Equal(t, "", st) assert.Equal(t, "", st)
assert.ErrorIs(t, err, ErrMissingOriginal) assert.ErrorIs(t, err, ErrMissingOriginal)
} }
func TestHashTypeStringer(t *testing.T) {
s := HashMD5.String()
assert.Equal(t, "MD5", s)
}

60
gen_rand.go Normal file
View file

@ -0,0 +1,60 @@
package nomino
import (
"crypto/rand"
"strings"
"github.com/deatil/go-encoding/base62"
"github.com/google/uuid"
)
func uuidGen(*Config) (string, error) {
u, err := uuid.NewRandom()
if err != nil {
return "", err
}
return u.String(), nil
}
// UUID generates a UUIDv4.
func UUID() Generator {
return uuidGen
}
type randConf struct {
length int
}
// RandomOption is an option for the Random Generator
type RandomOption func(*randConf)
// RandomLength controls the length of the string generated by Random
func RandomLength(length int) RandomOption {
return func(c *randConf) {
c.length = length
}
}
func getRandomBytes(l int) []byte {
key := make([]byte, l)
rand.Read(key)
e := base62.StdEncoding.Encode(key)
return e[:l]
}
// Random generates a random string containing the characters [A-Za-z0-9].
// By default, it will be eight characters long.
func Random(opts ...RandomOption) Generator {
c := randConf{8}
for _, opt := range opts {
opt(&c)
}
return func(*Config) (string, error) {
var buff strings.Builder
buff.Grow(c.length)
for buff.Len() < c.length {
buff.Write(getRandomBytes(c.length - buff.Len()))
}
return buff.String(), nil
}
}

36
gen_rand_examples_test.go Normal file
View file

@ -0,0 +1,36 @@
package nomino_test
import (
"fmt"
"codeberg.org/danjones000/nomino"
)
func ExampleUUID() {
option := nomino.WithGenerator(nomino.UUID())
str, _ := nomino.Make(nomino.NewConfig(option))
fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig(option))
fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig(option))
fmt.Println(str)
}
func ExampleRandom() {
option := nomino.WithGenerator(nomino.Random())
str, _ := nomino.Make(nomino.NewConfig(option))
fmt.Println(str)
}
func ExampleRandomLength() {
option := nomino.WithGenerator(nomino.Random(
nomino.RandomLength(32),
))
str, _ := nomino.Make(nomino.NewConfig(option))
fmt.Println(str)
}

42
gen_rand_test.go Normal file
View file

@ -0,0 +1,42 @@
package nomino
import (
"errors"
"testing"
"github.com/google/uuid"
"github.com/stretchr/testify/assert"
)
func TestUUID(t *testing.T) {
st, err := UUID()(nil)
assert.NoError(t, err)
_, parseErr := uuid.Parse(st)
assert.NoError(t, parseErr)
}
type badRead struct{}
func (badRead) Read([]byte) (int, error) {
return 0, errors.New("sorry")
}
func TestUUIDFail(t *testing.T) {
uuid.SetRand(badRead{})
defer uuid.SetRand(nil)
_, err := UUID()(nil)
assert.Equal(t, errors.New("sorry"), err)
}
func TestRand(t *testing.T) {
st, err := Random()(nil)
assert.NoError(t, err)
assert.Len(t, st, 8)
}
func TestRandLen(t *testing.T) {
st, err := Random(RandomLength(32))(nil)
assert.NoError(t, err)
assert.Len(t, st, 32)
}

View file

@ -2,8 +2,7 @@ package nomino
import ( import (
"errors" "errors"
"math/rand"
"github.com/google/uuid"
) )
// Generator is a function that returns the "random" portion of the returned filename. // Generator is a function that returns the "random" portion of the returned filename.
@ -11,6 +10,12 @@ import (
// for example. // for example.
type Generator func(conf *Config) (string, error) type Generator func(conf *Config) (string, error)
// Make allows you to generate a new string directly from a generator.
func (g Generator) Make(opts ...Option) (string, error) {
opts = append(opts, WithGenerator(g))
return Make(NewConfig(opts...))
}
// WithGenerator sets the specified generator // WithGenerator sets the specified generator
func WithGenerator(g Generator) Option { func WithGenerator(g Generator) Option {
return func(c *Config) { return func(c *Config) {
@ -44,15 +49,19 @@ func MultiGeneratorInOrder(gens ...Generator) Generator {
} }
} }
func uuidGen(*Config) (string, error) { // MultiGeneratorRandomOrder allows the use of multiple generators. Each new invokation will use one of the generators randomly.
u, err := uuid.NewRandom() // If none are passed, the generator will always return ErrMissingGenerators.
if err != nil { func MultiGeneratorRandomOrder(gens ...Generator) Generator {
return "", err if len(gens) == 0 {
return missingGen
} }
return u.String(), nil
}
// UUID generates a UUIDv4. if len(gens) == 1 {
func UUID() Generator { return gens[0]
return uuidGen }
return func(c *Config) (string, error) {
idx := rand.Int() % len(gens)
return gens[idx](c)
}
} }

View file

@ -7,18 +7,14 @@ import (
) )
func ExampleWithGenerator_customGenerator() { func ExampleWithGenerator_customGenerator() {
gen := func(*nomino.Config) (string, error) { var gen nomino.Generator = func(*nomino.Config) (string, error) {
return "hello", nil return "hello", nil
} }
option := nomino.WithGenerator(gen)
str, _ := nomino.Make(nomino.NewConfig(option)) str, _ := gen.Make()
fmt.Println(str) fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig( str, _ = gen.Make(nomino.WithoutExtension())
option,
nomino.WithoutExtension(),
))
fmt.Println(str) fmt.Println(str)
// Output: // Output:
@ -26,6 +22,19 @@ func ExampleWithGenerator_customGenerator() {
// hello // hello
} }
func ExampleGenerator_Make() {
g := nomino.Incremental()
st, _ := g.Make()
fmt.Println(st)
st, _ = g.Make(nomino.WithPrefix("foo"))
fmt.Println(st)
// Output:
// 0.txt
// foo_1.txt
}
func ExampleMultiGeneratorInOrder() { func ExampleMultiGeneratorInOrder() {
gen1 := func(*nomino.Config) (string, error) { gen1 := func(*nomino.Config) (string, error) {
return "hello", nil return "hello", nil
@ -34,15 +43,14 @@ func ExampleMultiGeneratorInOrder() {
return "goodbye", nil return "goodbye", nil
} }
gen := nomino.MultiGeneratorInOrder(gen1, gen2) gen := nomino.MultiGeneratorInOrder(gen1, gen2)
option := nomino.WithGenerator(gen)
str, _ := nomino.Make(nomino.NewConfig(option)) str, _ := gen.Make()
fmt.Println(str) fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig(option)) str, _ = gen.Make()
fmt.Println(str) fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig(option)) str, _ = gen.Make()
fmt.Println(str) fmt.Println(str)
// Output: // Output:
@ -51,15 +59,21 @@ func ExampleMultiGeneratorInOrder() {
// hello.txt // hello.txt
} }
func ExampleUUID() { func ExampleMultiGeneratorRandomOrder() {
option := nomino.WithGenerator(nomino.UUID()) gen1 := func(*nomino.Config) (string, error) {
return "hello", nil
}
gen2 := func(*nomino.Config) (string, error) {
return "goodbye", nil
}
gen := nomino.MultiGeneratorRandomOrder(gen1, gen2)
str, _ := nomino.Make(nomino.NewConfig(option)) str, _ := gen.Make()
fmt.Println(str) fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig(option)) str, _ = gen.Make()
fmt.Println(str) fmt.Println(str)
str, _ = nomino.Make(nomino.NewConfig(option)) str, _ = gen.Make()
fmt.Println(str) fmt.Println(str)
} }

View file

@ -4,7 +4,6 @@ import (
"errors" "errors"
"testing" "testing"
"github.com/google/uuid"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
) )
@ -17,26 +16,34 @@ func TestWithGenerator(t *testing.T) {
assert.Equal(t, "abc", st) assert.Equal(t, "abc", st)
} }
const (
out1 string = "abc"
out2 string = "def"
)
var (
outs = []string{out1, out2}
err1 = errors.New("oops")
gen1 Generator = func(*Config) (string, error) { return out1, nil }
gen2 Generator = func(*Config) (string, error) { return out2, nil }
gen3 Generator = func(*Config) (string, error) { return "", err1 }
gens = []Generator{gen1, gen2, gen3}
)
func TestMultiGeneratorInOrder(t *testing.T) { func TestMultiGeneratorInOrder(t *testing.T) {
st1 := "abc" g := MultiGeneratorInOrder(gens...)
st2 := "def"
er1 := errors.New("oops")
g1 := func(*Config) (string, error) { return st1, nil }
g2 := func(*Config) (string, error) { return st2, nil }
g3 := func(*Config) (string, error) { return "", er1 }
g := MultiGeneratorInOrder(g1, g2, g3)
st, err := g(nil) st, err := g(nil)
assert.NoError(t, err) assert.NoError(t, err)
assert.Equal(t, st1, st) assert.Equal(t, out1, st)
st, err = g(nil) st, err = g(nil)
assert.NoError(t, err) assert.NoError(t, err)
assert.Equal(t, st2, st) assert.Equal(t, out2, st)
st, err = g(nil) st, err = g(nil)
assert.Zero(t, st) assert.Zero(t, st)
assert.ErrorIs(t, err, er1) assert.ErrorIs(t, err, err1)
st, err = g(nil) st, err = g(nil)
assert.NoError(t, err) assert.NoError(t, err)
assert.Equal(t, st1, st) assert.Equal(t, out1, st)
} }
func TestMultiGeneratorInOrderOne(t *testing.T) { func TestMultiGeneratorInOrderOne(t *testing.T) {
@ -62,23 +69,38 @@ func TestMultiGeneratorInOrderMissing(t *testing.T) {
assert.ErrorIs(t, err, ErrMissingGenerators) assert.ErrorIs(t, err, ErrMissingGenerators)
} }
func TestUUID(t *testing.T) { func TestMultiGeneratorRandomOrder(t *testing.T) {
st, err := UUID()(nil) g := MultiGeneratorRandomOrder(gens...)
for i := 0; i < 4; i++ {
st, err := g(nil)
if err != nil {
assert.Zero(t, st)
assert.ErrorIs(t, err, err1)
} else {
assert.Contains(t, outs, st)
}
}
}
func TestMultiGeneratorRandomOrderOne(t *testing.T) {
st1 := "abc"
g1 := func(*Config) (string, error) { return st1, nil }
g := MultiGeneratorRandomOrder(g1)
st, err := g(nil)
assert.NoError(t, err) assert.NoError(t, err)
_, parseErr := uuid.Parse(st) assert.Equal(t, st1, st)
assert.NoError(t, parseErr) st, err = g(nil)
assert.NoError(t, err)
assert.Equal(t, st1, st)
} }
type badRead struct{} func TestMultiGeneratorRandomOrderMissing(t *testing.T) {
g := MultiGeneratorRandomOrder()
func (badRead) Read([]byte) (int, error) { st, err := g(nil)
return 0, errors.New("sorry") assert.Zero(t, st)
} assert.ErrorIs(t, err, ErrMissingGenerators)
st, err = g(nil)
func TestUUIDFail(t *testing.T) { assert.Zero(t, st)
uuid.SetRand(badRead{}) assert.ErrorIs(t, err, ErrMissingGenerators)
defer uuid.SetRand(nil)
_, err := UUID()(nil)
assert.Equal(t, errors.New("sorry"), err)
} }

1
go.mod
View file

@ -3,6 +3,7 @@ module codeberg.org/danjones000/nomino
go 1.23.6 go 1.23.6
require ( require (
github.com/deatil/go-encoding v1.0.3003
github.com/google/uuid v1.6.0 github.com/google/uuid v1.6.0
github.com/gosimple/slug v1.15.0 github.com/gosimple/slug v1.15.0
github.com/stretchr/testify v1.10.0 github.com/stretchr/testify v1.10.0

2
go.sum
View file

@ -1,5 +1,7 @@
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/deatil/go-encoding v1.0.3003 h1:2b05UO+5JfVcXcOa8n/X3pm8aC6L6ET0mBZCb1kj3ck=
github.com/deatil/go-encoding v1.0.3003/go.mod h1:lTMMKsG0RRPGZzdW2EPVJCA7HQy4o1ZQKPf5CmVDy2k=
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/gosimple/slug v1.15.0 h1:wRZHsRrRcs6b0XnxMUBM6WK1U1Vg5B0R7VkIf1Xzobo= github.com/gosimple/slug v1.15.0 h1:wRZHsRrRcs6b0XnxMUBM6WK1U1Vg5B0R7VkIf1Xzobo=

View file

@ -1,26 +0,0 @@
// Code generated by "stringer -type=HashType -trimprefix=Hash"; DO NOT EDIT.
package nomino
import "strconv"
func _() {
// An "invalid array index" compiler error signifies that the constant values have changed.
// Re-run the stringer command to generate them again.
var x [1]struct{}
_ = x[HashMD5-1]
_ = x[HashSHA1-2]
_ = x[HashSHA256-3]
}
const _HashType_name = "MD5SHA1SHA256"
var _HashType_index = [...]uint8{0, 3, 7, 13}
func (i HashType) String() string {
i -= 1
if i >= HashType(len(_HashType_index)-1) {
return "HashType(" + strconv.FormatInt(int64(i+1), 10) + ")"
}
return _HashType_name[_HashType_index[i]:_HashType_index[i+1]]
}

View file

@ -2,9 +2,9 @@ package nomino
import "fmt" import "fmt"
// Make generates a random filename. The behavior can be controlled by specifying Options // Make generates a random filename. The behavior can be controlled by specifying Options.
// In general, the final filename will be [prefix]_[generated_string]_[original_filename]_[suffix].[extension]. // In general, the final filename will be [prefix]_[generated_string]_[original_filename]_[suffix].[extension].
// If the name generator returns an error (generally, it shouldn't), that will be returned instead. // If the name generator returns an error (generally, it shouldn't), that error will be returned instead.
func Make(conf Config, opts ...Option) (string, error) { func Make(conf Config, opts ...Option) (string, error) {
for _, opt := range opts { for _, opt := range opts {
opt(&conf) opt(&conf)