2017-07-11 18:09:08 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/url"
|
2018-07-15 15:09:03 +02:00
|
|
|
|
2020-05-28 15:20:58 +02:00
|
|
|
"github.com/google/uuid"
|
2017-07-11 18:09:08 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Generator the type to generate keys(short urls)
|
|
|
|
type Generator func() string
|
|
|
|
|
|
|
|
// DefaultGenerator is the defautl url generator
|
|
|
|
var DefaultGenerator = func() string {
|
2020-05-28 15:20:58 +02:00
|
|
|
id, _ := uuid.NewRandom()
|
2018-04-03 12:07:23 +02:00
|
|
|
return id.String()
|
2017-07-11 18:09:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Factory is responsible to generate keys(short urls)
|
|
|
|
type Factory struct {
|
|
|
|
store Store
|
|
|
|
generator Generator
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFactory receives a generator and a store and returns a new url Factory.
|
|
|
|
func NewFactory(generator Generator, store Store) *Factory {
|
|
|
|
return &Factory{
|
|
|
|
store: store,
|
|
|
|
generator: generator,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gen generates the key.
|
|
|
|
func (f *Factory) Gen(uri string) (key string, err error) {
|
|
|
|
// we don't return the parsed url because #hash are converted to uri-compatible
|
|
|
|
// and we don't want to encode/decode all the time, there is no need for that,
|
|
|
|
// we save the url as the user expects if the uri validation passed.
|
|
|
|
_, err = url.ParseRequestURI(uri)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
key = f.generator()
|
|
|
|
// Make sure that the key is unique
|
|
|
|
for {
|
|
|
|
if v := f.store.Get(key); v == "" {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
key = f.generator()
|
|
|
|
}
|
|
|
|
|
|
|
|
return key, nil
|
|
|
|
}
|