Compare commits
8 commits
fd5413ab21
...
55038ea295
| Author | SHA1 | Date | |
|---|---|---|---|
| 55038ea295 | |||
| 4d6cd82b74 | |||
| 7c016df30f | |||
| 2440f55563 | |||
| 1008a064d0 | |||
| 1677a692d1 | |||
| 5c1132e414 | |||
| 10eb3f2491 |
16 changed files with 307 additions and 152 deletions
16
CHANGELOG.md
16
CHANGELOG.md
|
|
@ -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
|
||||||
|
|
|
||||||
|
|
@ -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:
|
||||||
|
|
|
||||||
|
|
@ -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",
|
||||||
|
|
|
||||||
46
gen_file.go
46
gen_file.go
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -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
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -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
60
gen_rand.go
Normal 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
36
gen_rand_examples_test.go
Normal 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
42
gen_rand_test.go
Normal 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)
|
||||||
|
}
|
||||||
|
|
@ -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)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -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
1
go.mod
|
|
@ -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
2
go.sum
|
|
@ -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=
|
||||||
|
|
|
||||||
|
|
@ -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]]
|
|
||||||
}
|
|
||||||
4
make.go
4
make.go
|
|
@ -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)
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue