Auto generate short links if empty
This commit is contained in:
9
go.mod
9
go.mod
@@ -1,5 +1,10 @@
|
|||||||
module github.com/gopatchy/shortlinks
|
module github.com/gopatchy/shortlinks
|
||||||
|
|
||||||
go 1.22
|
go 1.22.0
|
||||||
|
|
||||||
require github.com/lib/pq v1.10.9
|
toolchain go1.23.3
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/lib/pq v1.10.9
|
||||||
|
golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f
|
||||||
|
)
|
||||||
|
|||||||
2
go.sum
2
go.sum
@@ -1,2 +1,4 @@
|
|||||||
github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
|
github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
|
||||||
github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
|
github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
|
||||||
|
golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f h1:XdNn9LlyWAhLVp6P/i8QYBW+hlyhrhei9uErw2B5GJo=
|
||||||
|
golang.org/x/exp v0.0.0-20241108190413-2d47ceb2692f/go.mod h1:D5SMRVC3C2/4+F/DB1wZsLRnSNimn2Sp/NPsCrsv8ak=
|
||||||
|
|||||||
36
main.go
36
main.go
@@ -7,14 +7,17 @@ import (
|
|||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
_ "github.com/lib/pq"
|
_ "github.com/lib/pq"
|
||||||
|
"golang.org/x/exp/rand"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ShortLinks struct {
|
type ShortLinks struct {
|
||||||
tmpl *template.Template
|
tmpl *template.Template
|
||||||
mux *http.ServeMux
|
mux *http.ServeMux
|
||||||
db *sql.DB
|
db *sql.DB
|
||||||
|
r *rand.Rand
|
||||||
}
|
}
|
||||||
|
|
||||||
type response struct {
|
type response struct {
|
||||||
@@ -33,6 +36,7 @@ func NewShortLinks(db *sql.DB) (*ShortLinks, error) {
|
|||||||
tmpl: tmpl,
|
tmpl: tmpl,
|
||||||
mux: http.NewServeMux(),
|
mux: http.NewServeMux(),
|
||||||
db: db,
|
db: db,
|
||||||
|
r: rand.New(rand.NewSource(uint64(time.Now().UnixNano()))),
|
||||||
}
|
}
|
||||||
|
|
||||||
sl.mux.HandleFunc("GET /{$}", sl.serveRoot)
|
sl.mux.HandleFunc("GET /{$}", sl.serveRoot)
|
||||||
@@ -88,6 +92,14 @@ func (sl *ShortLinks) serveSet(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
short := r.Form.Get("short")
|
short := r.Form.Get("short")
|
||||||
|
|
||||||
|
if short == "" {
|
||||||
|
short, err = sl.genShort()
|
||||||
|
if err != nil {
|
||||||
|
sendError(w, http.StatusInternalServerError, "genShort: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
long := r.Form.Get("long")
|
long := r.Form.Get("long")
|
||||||
if long == "" {
|
if long == "" {
|
||||||
sendError(w, http.StatusBadRequest, "long= param required")
|
sendError(w, http.StatusBadRequest, "long= param required")
|
||||||
@@ -110,6 +122,30 @@ DO UPDATE SET long = $2;
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (sl *ShortLinks) genShort() (string, error) {
|
||||||
|
for chars := 3; chars <= 10; chars++ {
|
||||||
|
b := make([]byte, chars)
|
||||||
|
|
||||||
|
for i := range b {
|
||||||
|
b[i] = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"[sl.r.Intn(62)]
|
||||||
|
}
|
||||||
|
|
||||||
|
short := string(b)
|
||||||
|
|
||||||
|
exists := false
|
||||||
|
err := sl.db.QueryRow("SELECT EXISTS(SELECT 1 FROM links WHERE short = $1)", short).Scan(&exists)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("check exists: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !exists {
|
||||||
|
return short, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", fmt.Errorf("no available short link found")
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
port := os.Getenv("PORT")
|
port := os.Getenv("PORT")
|
||||||
if port == "" {
|
if port == "" {
|
||||||
|
|||||||
@@ -87,7 +87,10 @@ async function set() {
|
|||||||
|
|
||||||
document.getElementById('short-icon').setAttribute('name', 'check-square-fill');
|
document.getElementById('short-icon').setAttribute('name', 'check-square-fill');
|
||||||
document.getElementById('long-icon').setAttribute('name', 'check-square-fill');
|
document.getElementById('long-icon').setAttribute('name', 'check-square-fill');
|
||||||
|
|
||||||
|
if (short != '') {
|
||||||
setShortItem(short, 'check-square-fill');
|
setShortItem(short, 'check-square-fill');
|
||||||
|
}
|
||||||
|
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
params.set('short', short);
|
params.set('short', short);
|
||||||
@@ -98,14 +101,16 @@ async function set() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (resp.status !== 200) {
|
if (resp.status !== 200) {
|
||||||
error('Failed to set', await resp.text());
|
error('Failed to set', (await resp.json()).message);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const newShort = (await resp.json()).short;
|
||||||
|
|
||||||
if (document.getElementById('short').value == short && document.getElementById('long').value == long) {
|
if (document.getElementById('short').value == short && document.getElementById('long').value == long) {
|
||||||
document.getElementById('short-icon').setAttribute('name', 'check-square');
|
document.getElementById('short-icon').setAttribute('name', 'check-square');
|
||||||
document.getElementById('long-icon').setAttribute('name', 'check-square');
|
document.getElementById('long-icon').setAttribute('name', 'check-square');
|
||||||
setShortItem(short, 'check-square');
|
setShortItem(newShort, 'check-square');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user