Skip to content

Commit 34f3d9e

Browse files
committed
Slugs: Use base32 encoding for slugs that don't contain text photoprism#4761
Signed-off-by: Michael Mayer <michael@photoprism.app>
1 parent f1d9e85 commit 34f3d9e

File tree

2 files changed

+38
-5
lines changed

2 files changed

+38
-5
lines changed

pkg/txt/slug.go

+21-1
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,17 @@
11
package txt
22

33
import (
4+
"encoding/base32"
45
"strings"
56

67
"github.com/gosimple/slug"
78
)
89

10+
const SlugCharset = "abcdefghijklmnopqrstuvwxyz123456"
11+
const SlugEncoded = '_'
12+
13+
var SlugEncoding = base32.NewEncoding(SlugCharset).WithPadding(base32.NoPadding)
14+
915
// Slug converts a string to a valid slug with a max length of 80 runes.
1016
func Slug(s string) string {
1117
s = strings.TrimSpace(s)
@@ -14,7 +20,13 @@ func Slug(s string) string {
1420
return ""
1521
}
1622

17-
return Clip(slug.Make(s), ClipSlug)
23+
result := slug.Make(s)
24+
25+
if result == "" {
26+
result = string(SlugEncoded) + SlugEncoding.EncodeToString([]byte(s))
27+
}
28+
29+
return Clip(result, ClipSlug)
1830
}
1931

2032
// SlugToTitle converts a slug back to a title
@@ -23,5 +35,13 @@ func SlugToTitle(s string) string {
2335
return ""
2436
}
2537

38+
if s[0] == SlugEncoded {
39+
title, err := SlugEncoding.DecodeString(s[1:])
40+
41+
if len(title) > 0 && err == nil {
42+
return string(title)
43+
}
44+
}
45+
2646
return Title(strings.Join(Words(s), " "))
2747
}

pkg/txt/slug_test.go

+17-4
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ func TestSlug(t *testing.T) {
1010
t.Run("Empty", func(t *testing.T) {
1111
assert.Equal(t, "", Slug(""))
1212
})
13-
t.Run("BillGates", func(t *testing.T) {
13+
t.Run("Gates", func(t *testing.T) {
1414
assert.Equal(t, "william-henry-gates-iii", Slug("William Henry Gates III"))
1515
})
1616
t.Run("Quotes", func(t *testing.T) {
@@ -19,13 +19,26 @@ func TestSlug(t *testing.T) {
1919
t.Run("Chinese", func(t *testing.T) {
2020
assert.Equal(t, "chen-zhao", Slug(" 陈 赵"))
2121
})
22+
t.Run("Emoji", func(t *testing.T) {
23+
assert.Equal(t, "_5cpzfdq", Slug("💎"))
24+
assert.Equal(t, "_5cpzfea", Slug("💐"))
25+
assert.Equal(t, "_5cpzfea", Slug(" 💐 "))
26+
assert.Equal(t, "_5cpzfdxqt5jja", Slug("💎💐"))
27+
assert.Equal(t, "photoprism", Slug("PhotoPrism 💎"))
28+
})
2229
}
2330

2431
func TestSlugToTitle(t *testing.T) {
25-
t.Run("cute_Kitten", func(t *testing.T) {
32+
t.Run("Empty", func(t *testing.T) {
33+
assert.Equal(t, "", SlugToTitle(""))
34+
})
35+
t.Run("Kitten", func(t *testing.T) {
2636
assert.Equal(t, "Cute-Kitten", SlugToTitle("cute-kitten"))
2737
})
28-
t.Run("empty", func(t *testing.T) {
29-
assert.Equal(t, "", SlugToTitle(""))
38+
t.Run("Emoji", func(t *testing.T) {
39+
assert.Equal(t, "💎", SlugToTitle("_5cpzfdq"))
40+
assert.Equal(t, "💐", SlugToTitle("_5cpzfea"))
41+
assert.Equal(t, "💎💐", SlugToTitle("_5cpzfdxqt5jja"))
42+
assert.Equal(t, "PhotoPrism", SlugToTitle("photoprism"))
3043
})
3144
}

0 commit comments

Comments
 (0)