Slug — Create friendly slugs for your URLs and filenames (Emacs package)

Below you find the latest version of (1) the package's README and (2) its main source file.

For the git repository and issue tracker, see the project's page on sr.ht.

For more packages, see Software.


README.org

Meta

Compact tree-structured view of functions

If you're seeing this as README.org (instead of html), you can:

  1. put point at the beginning of the Functions heading;
  2. maybe C-l (recenter-top-bottom) to put it near the top of the screen;
  3. and then press C-c C-k (outline-show-branches).

You'll then have a compact tree-structured view of all the functions.

To navigate these trees and Org files in general, I wholeheartedly recommend you try the commands fold-and-focus-previous and fold-and-focus-next from my Fold and Focus package. See here how it looks like.

Overview

With this package you can create slugs.

Slugs are user-friendly strings that identify resources.
They can be used, for example:

  • for naming files
  • for creating clean URLs
  • for anchors in html pages

Say you wrote an article for your site. Its title:
The best すし & さしみ 42 € can buy™ [updated: 2042]

How would you name an HTML file to serve that article? Here's an option:
the-best-sushi-and-sashimi-42-eur-can-buy-tm-updated-2042

You can do the above with:

(slug-to-ud-09az "The best すし & さしみ 42 € can buy™ [updated: 2042]")
=> "the-best-sushi-and-sashimi-42-eur-can-buy-tm-updated-2042"

This function combines several others, each of which "cleans" your string in some particular way.

One of them, for example, replaces spaces with dashes. Another removes punctuation. Others deal with non-ASCII characters.

The package provides an assortment of such smaller functions, plus a few that combine them — such as the one above.

See below for usage examples.

Installation

See my page Software for the most up-to-date instructions on how to download and install any of my Emacs packages.

Having downloaded and installed the package and its dependencies, adapt the configurations below to your init.el file.

(use-package slug
  :demand t
  :custom  ; for example:
  (slug-a-hex-digits 12))

Alternatively, if you don't have use-package:

(require 'slug)

Summary of callables

Here's an overview of this package's callables:

Function Summary
slug-call Sequentially call each function from FUNS on string S.
slug-to-ud-09az Unidecode S, then simplify it to 0-9a-z and single dashes.
slug-to-nd-alnum Simplify string S to lowercase alphanumeric and single dashes.
slug-to-nd-09az Simplify string S to 0-9a-z and single dashes.
slug-r-ud-default Transliterate Unicode string S into ASCII.
slug-r-ud-unidecode Transliterate Unicode string S into ASCII.
slug-r-ud-python-unidecode Transliterate Unicode string S into ASCII.
slug-r-expand-german Expand Eszett and umlauted characters in string S.
slug-r-diacritics Remove diacritics from string S.
slug-r-diacritics-decomposable Remove diacritics from decomposable characters in string S.
slug-r-diacritics-undecomposable Remove diacritics from undecomposable characters in string S.
slug-r-strokes Remove strokes from undecomposable characters in string S.
slug-r-expand-ligatures Expand ligatures of undecomposable characters in string S.
slug-r-expand-eszett Expand Eszett in string S.
slug-r-expand-umlauted Expand umlauted characters in string S.
slug-r-org-stats-cookie Remove Org completion stats cookies from string S.
slug-r-html-tags Remove <FOO>..</FOO> HTML tags from string S.
slug-r-urls-org Remove Org URLs from string S.
slug-r-urls-md Remove Markdown URLs from string S.
slug-r-and Replace "&" with " and " in string S.
slug-r-dot Replace "." with " dot " in string S.
slug-r-plus Replace "+" with " plus " in string S.
slug-r-up-to-colon If string S has a colon, remove it and everything after.
slug-r-09Az-or-space Replace with a space all except 0-9A-Za-z in S.
slug-r-09Az-parens-or-space Replace with a space all except 0-9A-Za-z() in S.
slug-r-alnum-or-space Replace with a space all except [:alnum:] in S.
slug-r-alnum-parens-or-space Replace with a space all except [:alnum:]() in S.
slug-r-trim Remove leading and trailing whitespace from string S.
slug-r-one-space Replace two or more spaces with a single space in S.
slug-r-parens-to-dashes Replace parentheses with dashes in parenthesized substrings of S.
slug-r-parens Remove parentheses characters from string S.
slug-r-space-to-dash Replace spaces with dashes in string S.
slug-r-trim-dash Remove leading and trailing dashes from string S.
slug-r-one-dash Replace sequences of dashes in string S with a single one.
slug-a-append-hex-maybe Append random hex to string S if one is not yet present.
slug-a-append-hex Append hex string to string S.
slug-a-prepend-date-maybe Prepend date to string S if one is not yet present.
slug-a-prepend-date Prepend date to string S.
slug-a-append-ts-maybe Append timestamp to string S if one is not yet present.
slug-a-append-ts Append timestamp to string S.
slug-see-readme Open slug's README.org file.
slug-see-news See the News in slug's README.org file.

They are described in more detail below.

Functions

Replacement functions

These are functions to make replacements in strings.

Sequential replacements

These are compositions of smaller, simpler functions.

Call sequentially
slug-call (funs s)

Sequentially call each function from FUNS on string S.

FUNS is a list of functions.

You can use slug-call to compose your own slug-generating functions.

(slug-call (list #'downcase #'reverse)    "RaBOoF")  => "foobar"
(slug-call (list #'slug-r-and #'downcase) "Foo&Bar") => "foo and bar"

;; The order matters:
(slug-call (list #'slug-r-and #'reverse)  "Foo&Bar") => "raB dna ooF"
(slug-call (list #'reverse #'slug-r-and)  "Foo&Bar") => "raB and ooF"
Create lowercase slugs of alphanumerics and dashes

These functions simplify a string to only alphanumeric characters and
dashes.

There are a few variations on how to do this, in particular regarding
how to deal with alphanumeric characters in non-Latin scripts.

Here is a summary, with an example of applying them to this string:
Ął|άε:--~(ióñ)すし,™4௨

Function NLA NLNA Example
slug-to-ud-09az decode decode al-ae-ion-sushi-42-tm
slug-to-nd-alnum leave remove al-αε-ion-すし-4௨
slug-to-nd-09az remove remove al-ion-4

where columns are:

NLA non-Latin alphanumeric characters すし, Здрасти, ௪௨
NLNA non-Latin non-alphanumeric characters € ™ © ®

and actions on the characters are:

decode Unicode すし → sushi, € → EUR, ™ → (tm)
remove character すし → <nothing>, øł → <nothing>
leave unmodified すし → すし, øł → øł

Diacritics are removed by them all.

slug-to-ud-09az (s)

Unidecode S, then simplify it to 0-9a-z and single dashes.

It transliterates Unicode: symbols and alphanumeric characters from
other alphabets are decoded.

Remaining non-alphanumeric characters are removed.

(slug-to-ud-09az "=Ął|άε:--~(ióñ)すし,™4௨=") => "al-ae-ion-sushi-tm-42"

(slug-to-ud-09az
 "The best すし 42-€ can buy™ [updated: 2042]")
=> "the-best-sushi-42-eur-can-buy-tm-updated-2042"

(slug-to-ud-09az
 "Paweł Å Bær Føø's すし résumé -- (2042)")
=> "pawel-a-baer-foo-s-sushi-resume-2042"
slug-to-nd-alnum (s)

Simplify string S to lowercase alphanumeric and single dashes.

It does not transliterate Unicode.

Diacritics are removed: “nd” stands for “no diacritics”.

Alphanumeric characters from other alphabets are left intact;
non-alphanumeric characters are removed.

(slug-to-nd-alnum "=Ął|άε:--~(ióñ)すし,™4௨=") => "al-αε-ion-すし-4௨"

(slug-to-nd-alnum
 "The best すし 42 € can buy™ [updated: 2042]")
=> "the-best-すし-42-can-buy-updated-2042"

(slug-to-nd-alnum
 "Paweł Å Bær Føø's すし résumé™ -- (2042)")
=> "pawel-a-baer-foo-s-すし-resume-2042"
slug-to-nd-09az (s)

Simplify string S to 0-9a-z and single dashes.

It does not transliterate Unicode.

Diacritics are removed: “nd” stands for “no diacritics”.

Symbols, alphanumeric characters from other alphabets, and
non-alphanumeric characters are removed.

(slug-to-nd-09az "=Ął|άε:--~(ióñ)すし,™4௨=") => "al-ion-4"

(slug-to-nd-09az
 "The best すし 42 € can buy™ [updated: 2042]")
=> "the-best-42-can-buy-updated-2042"

(slug-to-nd-09az
 "Paweł Å Bær Føø's すし résumé™ -- (2042)")
=> "pawel-a-baer-foo-s-resume-2042"
Unidecode: transliteration functions

These are functions to transliterate Unicode characters into ASCII.

  • Those in pure Emacs Lisp depend on the elisp package unidecode.
  • Prefer to call Python instead? It depends on python3-unidecode.
Your pick
slug-r-ud-default (s)

Transliterate Unicode string S into ASCII.

Use the function that slug-r-ud-default-function points to.

;;;; Depending on the value of slug-r-ud-default-function, slug-r-ud-default
;;;; will use either slug-r-ud-unidecode or slug-r-ud-python-unidecode,
;;;; which see.
Do it all in Emacs Lisp
slug-r-ud-unidecode (s)

Transliterate Unicode string S into ASCII.

Do so by calling unidecode on string S.

You'd need to have the unidecode elisp package.

(slug-r-ud-unidecode "Hello, (£42)!")     => "Hello, (PS42)!"
(slug-r-ud-unidecode "Dobrý deň, (€42)!") => "Dobry den, (EUR42)!"
(slug-r-ud-unidecode "Здрасти, (ЛВ42)!")  => "Zdrasti, (LV42)!"
(slug-r-ud-unidecode "こんにちは!")        => "konnichiha!"
(slug-r-ud-unidecode "你好™®©")            => "Ni Hao (tm)(r)(c)"
Do it by calling Python
slug-r-ud-python-unidecode (s)

Transliterate Unicode string S into ASCII.

Do so by calling python3-unidecode on string S.

That package would need to be installed on your system.

Replacement of diacritics

These functions remove or expand diacritics from strings.

slug-r-expand-german (s)

Expand Eszett and umlauted characters in string S.

(slug-r-expand-german "Öltanker")    => "Oeltanker"
(slug-r-expand-german "Eiweißstoff") => "Eiweissstoff"
(slug-r-expand-german "GROẞ BERẞEN") => "GROSS BERSSEN"
(slug-r-expand-german
 "Victor jagt zwölf Boxkämpfer quer über den großen Sylter Deich")
=> "Victor jagt zwoelf Boxkaempfer quer ueber den grossen Sylter Deich"
slug-r-diacritics (s)

Remove diacritics from string S.

Note that this is unlikely to preserve the phonology of the original
language in which S is written.

For example, in German it'd do this: schön → schon, but “schoen” would
be the better and more standard diacritics-free conversion. If you need
that, you may want to filter S first through slug-r-expand-german.

For languages using non–Latin-based scripts, try slug-r-ud-default,
which may be able to transliterate S into ASCII reasonably enough.

(slug-r-diacritics
 "Victor jagt zwölf Boxkämpfer quer über den großen Sylter Deich")
=> "Victor jagt zwolf Boxkampfer quer uber den grossen Sylter Deich"

(slug-r-diacritics
 "Paweł Å Bær Føø's すし résumé™")
=> "Pawel A Baer Foo's すし resume"
slug-r-diacritics-decomposable (s)

Remove diacritics from decomposable characters in string S.

(slug-r-diacritics-decomposable
 "Paweł Å Bær Føø's すし résumé™")
=> "Paweł A Bær Føø's すし resume"
slug-r-diacritics-undecomposable (s)

Remove diacritics from undecomposable characters in string S.

Deal with a few non-ASCII characters that aren't Unicode compositions
but could nevertheless be stripped into ASCII ones:

  • đ ħ ł ø æ œ ß → d h l o ae oe ss
  • Đ Ħ Ł Ø Æ Œ ẞ → D H L O Ae Oe SS

The first four are letters with a stroke.
The next two are ligatures.
The last is a German Eszett.

All these are present in a few languages, and are common enough to be
relevant for our sluggifying purposes.

(slug-r-diacritics-undecomposable
 "Paweł Å Bær Føø's すし résumé™")
=> "Pawel Å Baer Foo's すし résumé™"
slug-r-strokes (s)

Remove strokes from undecomposable characters in string S.

These are present in a few languages, and are common enough to be
relevant for our sluggifying purposes:

  • đ ħ ł ø → d h l o
  • Đ Ħ Ł Ø → D H L O.
(slug-r-strokes
 "Metħøđicałły reupħøłsteređ ełectrøcarđiøgrapħs.")
=> "Methodically reupholstered electrocardiographs."

;;;; This example is dedicated to the whole community of Serbian-Polish
;;;; slugophile Emacs Lispers of Maltese-Norwegian descent whose traveling
;;;; choices are affected by NP-hard optimization problems.
(slug-r-strokes
 "His trip from Đurđevo to Łabławki stops at Ħal Għargħur and Øversjødalen.")
=> "His trip from Durdevo to Lablawki stops at Hal Gharghur and Oversjodalen."
slug-r-expand-ligatures (s)

Expand ligatures of undecomposable characters in string S.

These are present in a few languages, and are common enough to be
relevant for our sluggifying purposes:

  • æ œ → ae oe
  • Æ Œ → Ae Oe.
(slug-r-expand-ligatures
 "All amœbæ were subpœnæd.")
=> "All amoebae were subpoenaed."

;;;; This example is dedicated to the whole community of French sisters of
;;;; the Church of Emacs whose interests forcibly exclude Latin epics and
;;;; Greek tragedies because their time is ardently devoted to spreading
;;;; the gospel of Elisp.
(slug-r-expand-ligatures
 "Sœur Lætitia was never interested in the Æneid or Œdipus Rex.")
=> "Soeur Laetitia was never interested in the Aeneid or Oedipus Rex."
slug-r-expand-eszett (s)

Expand Eszett in string S.

  • ß → ss
  • ẞ → SS.
(slug-r-expand-eszett "Eiweißstoff") => "Eiweissstoff"
(slug-r-expand-eszett "GROẞ BERẞEN") => "GROSS BERSSEN"
slug-r-expand-umlauted (s)

Expand umlauted characters in string S.

  • ä ö ü → ae oe ue
  • Ä Ö Ü → Ae Oe Ue.
(slug-r-expand-german
 "Ölüberschussländer")
=> "Oelueberschusslaender"

(slug-r-expand-german
 "Plausibilitätsüberprüfung")
=> "Plausibilitaetsueberpruefung"
Assortment of small replacement functions

These are small replacement functions.
You can easily combine any number of them using slug-call.

Pre-alnum functions

These would be typically used to clean a string before restricting it to
alphanumeric characters.

slug-r-html-tags (s)

Remove <FOO>..</FOO> HTML tags from string S.

(slug-r-html-tags "Only <foo>this is out</foo>that.") => "Only that."
(slug-r-html-tags "No <b>html</b>tags")               => "No tags"
slug-r-urls-org (s)

Remove Org URLs from string S.

[[URL][Description]]

is how they look like.

(slug-r-urls-org "See [[https://example.com][example]] now!")
=> "See [[example]] now!"
(slug-r-urls-org "See [[https://example.com][example]] now!")
=> "See [[example]] now!"
slug-r-urls-md (s)

Remove Markdown URLs from string S.

[Description](URL)

is how they look like.

(slug-r-urls-md "See [example](https://example.com) now!")
=> "See [example] now!"
(slug-r-urls-md "See [example](gemini://example.com) now!")
=> "See [example] now!"
slug-r-and (s)

Replace "&" with " and " in string S.

(slug-r-and "Check this&that.") => "Check this and that."
(slug-r-and "Foobar&Co.")       => "Foobar and Co."
slug-r-dot (s)

Replace "." with " dot " in string S.

(slug-r-dot "www.example.com") => "www dot example dot com"
(slug-r-dot "It's slug.el")    => "It's slug dot el"
slug-r-plus (s)

Replace "+" with " plus " in string S.

(slug-r-plus "a+b")            => "a plus b"
(slug-r-plus "Check+this+out") => "Check plus this plus out"
slug-r-up-to-colon (s)

If string S has a colon, remove it and everything after.

(slug-r-up-to-colon "Part I: Boarding the spaceship") => "Part I"
(slug-r-up-to-colon "Part II: Fighting the aliens")   => "Part II"
Alphanumeric restriction functions

These restrict a string to only alphanumeric-ish characters.

slug-r-09Az-or-space (s)

Replace with a space all except 0-9A-Za-z in S.

(slug-r-09Az-or-space "Hello, (£42)!")       => "Hello    42  "
(slug-r-09Az-or-space "Dobrý deň, (€42)!")   => "Dobr  de     42  "
(slug-r-09Az-or-space "Здрасти, (ЛВ42)!")    => "            42  "
(slug-r-09Az-or-space "வணக்கம், (௹௪௨)!") => "               "
(slug-r-09Az-or-space "今日は、(¥四十二)!")   => "           "
slug-r-09Az-parens-or-space (s)

Replace with a space all except 0-9A-Za-z() in S.

(slug-r-09Az-parens-or-space "Hello, (£42)!")       => "Hello  ( 42) "
(slug-r-09Az-parens-or-space "Dobrý deň, (€42)!")   => "Dobr  de   ( 42) "
(slug-r-09Az-parens-or-space "Здрасти, (ЛВ42)!")    => "         (  42) "
(slug-r-09Az-parens-or-space "வணக்கம், (௹௪௨)!") => "         (   ) "
(slug-r-09Az-parens-or-space "今日は、(¥四十二)!")   => "    (    ) "
slug-r-alnum-or-space (s)

Replace with a space all except [:alnum:] in S.

Note that [:alnum:] is not restricted to ASCII: alphanumeric characters
in non-Latin scripts will be preserved.

(slug-r-alnum-or-space "Hello, (£42)!")       => "Hello    42  "
(slug-r-alnum-or-space "Dobrý deň, (€42)!")   => "Dobrý deň    42  "
(slug-r-alnum-or-space "Здрасти, (ЛВ42)!")    => "Здрасти   ЛВ42  "
(slug-r-alnum-or-space "வணக்கம், (௹௪௨)!") => "வணக்கம்    ௪௨  "
(slug-r-alnum-or-space "今日は、(¥四十二)!")   => "今日は   四十二  "
slug-r-alnum-parens-or-space (s)

Replace with a space all except [:alnum:]() in S.

Note that [:alnum:] is not restricted to ASCII: alphanumeric characters
in non-Latin scripts will be preserved.

(slug-r-alnum-parens-or-space "Hello, (£42)!")       => "Hello  ( 42) "
(slug-r-alnum-parens-or-space "Dobrý deň, (€42)!")   => "Dobrý deň  ( 42) "
(slug-r-alnum-parens-or-space "Здрасти, (ЛВ42)!")    => "Здрасти  (ЛВ42) "
(slug-r-alnum-parens-or-space "வணக்கம், (௹௪௨)!") => "வணக்கம்  ( ௪௨) "
(slug-r-alnum-parens-or-space "今日は、(¥四十二)!")   => "今日は ( 四十二) "
Post-alnum functions

These would be typically used to clean a string after restricting it to
alphanumeric characters.

slug-r-trim (s)

Remove leading and trailing whitespace from string S.

(slug-r-trim "middle  spaces   are ok")     => "middle  spaces   are ok"
(slug-r-trim " surrounding  spaces not   ") => "surrounding  spaces not"
slug-r-one-space (s)

Replace two or more spaces with a single space in S.

(slug-r-one-space "this     is    great")  => "this is great"
(slug-r-one-space " just   one  space  ")  => " just one space "
slug-r-parens-to-dashes (s)

Replace parentheses with dashes in parenthesized substrings of S.

(slug-r-parens-to-dashes "Look (again) here")   => "Look -again- here"
(slug-r-parens-to-dashes "(Parens) here (not)") => " -Parens- here -not- "
slug-r-parens (s)

Remove parentheses characters from string S.

(slug-r-parens "([((Brackets stay))])")         => "[Brackets stay]"
(slug-r-parens "((((Open-close))))")            => "Open-close"
(slug-r-parens "((((Open only")                 => "Open only"
(slug-r-parens "Close only))))")                => "Close only"
(slug-r-parens "A (mix)) of (((open( ()close)") => "A mix of open close"
slug-r-space-to-dash (s)

Replace spaces with dashes in string S.

(slug-r-space-to-dash "No spaces here")      => "No-spaces-here"
(slug-r-space-to-dash "no    spaces   here") => "no----spaces---here"
slug-r-trim-dash (s)

Remove leading and trailing dashes from string S.

(slug-r-trim-dash "-some--thing--")   => "some--thing"
(slug-r-trim-dash "trailing-dashes-") => "trailing-dashes"
(slug-r-trim-dash "-leading-dashes")  => "leading-dashes"
slug-r-one-dash (s)

Replace sequences of dashes in string S with a single one.

(slug-r-one-dash "look---just--one")  => "look-just-one"
(slug-r-one-dash "one-two--three---") => "one-two-three-"

Generation of unique identifiers

These functions generate unique strings, which are useful to prepend or
append to slugs when there's a risk of duplicates.

Random hex digits

Make your slugs unique by appending a random hex string.

slug-a-append-hex-maybe (s)

Append random hex to string S if one is not yet present.

;;;; Since they're random, this test only checks that formats match

(string-match (format "hello-world-[0-9a-f]\\{%d\\}" slug-a-hex-digits)
              (slug-a-append-hex-maybe "hello-world"))
=> 0  ; matches

(slug-a-append-hex-maybe "hello-world-1c3a3f7e") => "hello-world-1c3a3f7e"
slug-a-append-hex (s)

Append hex string to string S.

To avoid appending more than once, use slug-a-append-hex-maybe.

;;;; Since they're random, these tests only check that formats match

(string-match (format "hello-world-[0-9a-f]\\{%d\\}" slug-a-hex-digits)
              (slug-a-append-hex "hello-world"))
=> 0  ; matches

(string-match (format "hello-world-1c3a3f7e-[0-9a-f]\\{%d\\}"
                      slug-a-hex-digits)
              (slug-a-append-hex "hello-world-1c3a3f7e"))
=> 0
Dates

Make your slugs unique by prepending a date.

slug-a-prepend-date-maybe (s)

Prepend date to string S if one is not yet present.

;;;; Since today date will vary, this is tested dynamically

(slug-a-prepend-date-maybe "hello-world")
=> (format "%s-hello-world" (slug--a-make-today))

(slug-a-prepend-date-maybe "2042-04-02-hello-world")
=> "2042-04-02-hello-world"
slug-a-prepend-date (s)

Prepend date to string S.

To avoid prepending more than once, use slug-a-prepend-date-maybe.

;;;; Since today date will vary, this is tested dynamically

(slug-a-prepend-date "hello-world")
=> (format "%s-hello-world" (slug--a-make-today))

(slug-a-prepend-date "2042-04-02-hello-world")
=> (format "%s-2042-04-02-hello-world" (slug--a-make-today))
Timestamps

Make your slugs unique by appending a timestamp.

slug-a-append-ts-maybe (s)

Append timestamp to string S if one is not yet present.

;;;; Since timestamps will vary, this is tested dynamically

(slug-a-append-ts-maybe "hello-world")
=> (format "hello-world-%s" (slug--a-make-now))

(slug-a-append-ts-maybe "hello-world-20420402-040242")
=> "hello-world-20420402-040242"
slug-a-append-ts (s)

Append timestamp to string S.

To avoid appending more than once, use slug-a-append-ts-maybe.

;;;; Since timestamps will vary, this is tested dynamically

(slug-a-append-ts "hello-world")
=> (format "hello-world-%s" (slug--a-make-now))

(slug-a-append-ts "hello-world-20420402-040242")
=> (format "hello-world-20420402-040242-%s" (slug--a-make-now))

Commands

See README

Commands to open slug's README.org. Optionally, find things in it.

slug-see-readme (&optional heading narrow)

Open slug's README.org file.

Search for the file in slug.el's directory.

If found, open it read-only.

If optional argument HEADING is passed, try to navigate to the
heading after opening it. HEADING should be a string.

If optional argument NARROW is non-nil, narrow to that heading.
This argument has no effect if HEADING is nil or not found.

slug-see-news ()

See the News in slug's README.org file.

Contributing

See my page Software for information about how to contribute to any of my Emacs packages.

News

0.2.0

Release

License

This project follows the REUSE Specification (FAQ), which in turn is built upon SPDX.

Therefore, license and copyright information can be found in:

  • each file's comment header, or
  • an adjacent file of the same name with the additional extension .license, or
  • the .reuse/dep5 file

The full text of the licenses can be found in the LICENSES subdirectory.


slug.el

Structure

;;; slug.el --- Create friendly slugs for your URLs and filenames  -*- lexical-binding: t -*-
;;; Commentary:
;;;; For all the details, please do see the README
;;; Acknowledgments:
;;; Code:
;;;; Libraries
;;;; Symbols from other packages
;;;; Package metadata
;;;; Customizable variables
;;;; Functions
;;;;; Description macro
;;;;; Replacement functions
;;;;;; Sequential replacements
;;;;;;; Call sequentially
;;;;;;; Create lowercase slugs of alphanumerics and dashes
;;;;;; Unidecode: transliteration functions
;;;;;;; Your pick
;;;;;;; Do it all in Emacs Lisp
;;;;;;; Do it by calling Python
;;;;;; Replacement of diacritics
;;;;;; Assortment of small replacement functions
;;;;;;; Pre-alnum functions
;;;;;;; Alphanumeric restriction functions
;;;;;;; Post-alnum functions
;;;;; Generation of unique identifiers
;;;;;; Random hex digits
;;;;;; Dates
;;;;;; Timestamps
;;;; Commands
;;;;; See README
;;;; Wrapping up
;;; slug.el ends here

Contents

;;; slug.el --- Create friendly slugs for your URLs and filenames  -*- lexical-binding: t -*-

;; SPDX-FileCopyrightText: © flandrew <https://flandrew.srht.site/listful>

;;---------------------------------------------------------------------------
;; Author:    flandrew
;; Created:   2022-01-22
;; Updated:   2025-03-31
;; Keywords:  extensions, lisp
;; Homepage:  <https://flandrew.srht.site/listful/software.html>
;;---------------------------------------------------------------------------
;; Package-Version:  0.2.0
;; Package-Requires: ((emacs "25.1"))
;;---------------------------------------------------------------------------

;; SPDX-License-Identifier: GPL-3.0-or-later

;; This file is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published
;; by the Free Software Foundation, either version 3 of the License,
;; or (at your option) any later version.
;;
;; This file is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;;
;; You should have received a copy of the GNU General Public License
;; along with this file. If not, see <https://www.gnu.org/licenses/>.

;;; Commentary:
;;
;; With this package you can create slugs.
;;
;; Slugs are user-friendly strings that identify resources.
;; They can be used, for example:
;; - for naming files
;; - for creating clean URLs
;; - for anchors in html pages
;;
;; This package provides "string cleaning" functions.
;; There's an assortment of small ones plus a few that combine them.
;;
;; You can then, for example, transform this title:
;;   The best すし & さしみ 42 € can buy™ [updated: 2042]
;;
;; into this slug:
;;   the-best-sushi-and-sashimi-42-eur-can-buy-tm-updated-2042
;;
;;;; For all the details, please do see the README
;;
;; Open it easily with:
;;   (find-file-read-only "README.org")   <--- C-x C-e here¹
;;
;; or from any buffer:
;;   M-x slug-see-readme
;;
;; or read it online:
;;   <https://flandrew.srht.site/listful/sw-emacs-slug.html>
;;
;; ¹ or the key that ‘eval-last-sexp’ is bound to, if not C-x C-e.
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

;;; Acknowledgments:
;;
;; The replacement function ‘slug-to-nd-alnum’ evolved from a function found
;; in Karl Voit's dot-emacs, which in turn was an adaptation of ox-hugo's
;; org-hugo-slug’.
;;
;; ox-hugo
;;   SPDX-FileCopyrightText:  © Kaushal Modi
;;   SPDX-License-Identifier: GPL-3.0-or-later
;;   Author:                  Kaushal Modi
;;   Homepage:                <https://ox-hugo.scripter.co>
;;
;; dot-emacs
;;   SPDX-FileCopyrightText:  © Karl Voit
;;   SPDX-License-Identifier: GPL-3.0-or-later
;;   Author:                  Karl Voit
;;   Homepage:                <https://github.com/novoid/dot-emacs>
;;
;; ------------------------------------------------------------------------
;;
;; The variable ‘slug-url-rx’ borrows from ffap.el's ‘ffap-url-regexp’.
;;
;; ffap
;;   SPDX-FileCopyrightText:  © Free Software Foundation, Inc.
;;   SPDX-License-Identifier: GPL-3.0-or-later
;;   Author:                  Michelangelo Grigni
;;
;; ------------------------------------------------------------------------
;;
;; The function ‘slug--s-replace-all’ adapts s.el's ‘s-replace-all’.
;;
;; s.el
;;   SPDX-FileCopyrightText:  © Magnar Sveen
;;   SPDX-License-Identifier: GPL-3.0-or-later
;;   Author:                  Magnar Sveen
;;   Homepage:                <https://github.com/magnars/s.el>
;;
;; ------------------------------------------------------------------------
;;
;; The function ‘slug--a-make-hex-n’ is based on org-macs.el's ‘org-id-uuid’.
;;
;; org-macs
;;   SPDX-FileCopyrightText:  © Free Software Foundation, Inc.
;;   SPDX-License-Identifier: GPL-3.0-or-later
;;   Author:                  Carsten Dominik
;;   Homepage:                <https://orgmode.org>
;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;


;;; Code:
;;;; Libraries

(require 'rx)
(require 'lisp-mnt)      ; lm-summary’ ‘lm-homepage’ ‘lm-version’ ‘lm-header
(require 'regexp-opt)    ; regexp-opt
(require 'ucs-normalize) ; ucs-normalize-NFC-string


;;;; Symbols from other packages

;; Silence "not known to be defined" compiler warnings
(declare-function unidecode          "ext:unidecode" (string))
(declare-function unidecode-sanitize "ext:unidecode" (string))


;;;; Package metadata

(defvar slug--name "Slug")

(defvar slug--dot-el
  (format "%s.el" (file-name-sans-extension (eval-and-compile
                                              (or load-file-name
                                                  buffer-file-name)))))
(defvar slug--readme-org
  (expand-file-name "README.org" (file-name-directory slug--dot-el)))

(defvar slug--summary  (lm-summary   slug--dot-el))
(defvar slug--homepage (lm-homepage  slug--dot-el))
(defvar slug--version  (lm-with-file slug--dot-el
                         (or (lm-header "package-version")
                             (lm-version))))


;;;; Customizable variables

(defgroup slug nil
  (format "%s." slug--summary)
  :group 'extensions
  :group 'lisp
  :link  '(emacs-library-link :tag "Lisp file" "slug.el")
  :link  `(file-link :tag "README.org" ,slug--readme-org)
  :link  `(url-link  :tag "Homepage"   ,slug--homepage))

(defcustom slug-r-ud-default-function #'slug-r-ud-unidecode
  "Default function to transliterate Unicode characters.

The choices:
- ‘slug-r-ud-unidecode’ uses only elisp, and depends on package
  unidecode.el. This is the default.
- ‘slug-r-ud-python-unidecode’ uses Python, and depends on your
  having the package python3-unidecode on your system.

This variable will only have an effect when you use a slug function that
does unidecoding.

If you try to use a unidecoding function but the corresponding elisp or
python3 package isn't available, an error message will be shown."
  :package-version '(slug "0.2.0")
  :type '(choice (function-item slug-r-ud-unidecode)
                 (function-item slug-r-ud-python-unidecode)))

(defcustom slug-a-hex-hash-algo 'sha512
  "Algorithm to generate random hex strings."
  :package-version '(slug "0.2.0")
  :type '(choice (const :tag "sha512" sha512)
                 (const :tag "sha256" sha256)
                 (const :tag "sha1"   sha1)
                 (const :tag "md5"    md5)))

(defcustom slug-a-hex-digits 8
  "Number of digits to use in random hex strings."
  :package-version '(slug "0.2.0")
  :type 'natnum)

(defcustom slug-a-hex-digits-min 8
  "Minimal number of digits to match when looking for hex strings.
This variable is used by ‘slug-a-append-hex-maybe’, which checks whether
a string already ends in a sequence of hex digits."
  :package-version '(slug "0.2.0")
  :type 'natnum)

(defcustom slug-a-date-format "%F"
  "Date format used by ‘slug-a-prepend-date’.
The default value is %F, which is the same as %+4Y-%m-%d, a full date.
This is compatible with ‘slug-a-date-rx’'s default."
  :package-version '(slug "0.2.0")
  :type 'string)

(defcustom slug-a-date-rx (rx (: (= 4 digit) "-" (= 2 digit) "-" (= 2 digit)))
  "Regular expression to match when looking for dates.
The default value is compatible with ‘slug-a-date-format’'s default.
This variable is used by ‘slug-a-prepend-date-maybe’, which checks
whether a string starts with a date."
  :package-version '(slug "0.2.0")
  :type 'string)

(defcustom slug-a-ts-format "%Y%m%d-%H%M%S"
  "Timestamp format used by ‘slug-a-append-ts’.
The default value is compatible with ‘slug-a-ts-rx’'s default."
  :package-version '(slug "0.2.0")
  :type 'string)

(defcustom slug-a-ts-rx (rx (: (= 8 digit) "-" (= 6 digit)))
  "Regular expression to match when looking for timestamps.
The default value is compatible with ‘slug-a-ts-format’'s default.
This variable is used by ‘slug-a-append-ts-maybe’, which checks whether
a string already ends in a timestamp."
  :package-version '(slug "0.2.0")
  :type 'string)


;;;; Functions
;;;;; Description macro

(defmacro slug--describe (str)
  "Describe with string STR the contents under this heading.
Think of it as a docstring for the headings of your elisp files.
For the full docstring, look for ‘orgreadme-fy-describe’ in the
package ‘orgreadme-fy’."
  (declare (indent 0))
  (unless (stringp str)
    (user-error "‘slug--describe’ must receive a string")))


;;;;; Replacement functions

(slug--describe
  "These are functions to make replacements in strings.")

;;;;;; Sequential replacements

(slug--describe
  "These are compositions of smaller, simpler functions.")

;;;;;;; Call sequentially

(defun slug-call (funs s)
  "Sequentially call each function from FUNS on string S.
FUNS is a list of functions.

You can use ‘slug-call’ to compose your own slug-generating functions."
  (while funs
    (setq s (funcall (pop funs) s)))
  s)


;;;;;;; Create lowercase slugs of alphanumerics and dashes

(slug--describe
  "These functions simplify a string to only alphanumeric characters and
dashes.

There are a few variations on how to do this, in particular regarding
how to deal with alphanumeric characters in non-Latin scripts.

Here is a summary, with an example of applying them to this string:
=Ął|άε:--~(ióñ)すし,™4௨=

| Function           | NLA    | NLNA   | Example               |
|--------------------+--------+--------+-----------------------|
| ‘slug-to-ud-09az’  | decode | decode | al-ae-ion-sushi-42-tm |
| ‘slug-to-nd-alnum’ | leave  | remove | al-αε-ion-すし-4௨    |
| ‘slug-to-nd-09az’  | remove | remove | al-ion-4              |

where columns are:
| NLA  | non-Latin alphanumeric characters     | すし, Здрасти, ௪௨ |
| NLNA | non-Latin non-alphanumeric characters | € ™ © ®            |

and actions on the characters are:
| decode Unicode   | すし → sushi, € → EUR, ™ → (tm)  |
| remove character | すし → <nothing>, øł → <nothing> |
| leave unmodified | すし → すし, øł → øł              |

Diacritics are removed by them all.")

(defun slug-to-ud-09az (s)
  "Unidecode S, then simplify it to 0-9a-z and single dashes.
It transliterates Unicode: symbols and alphanumeric characters from
other alphabets are decoded.

Remaining non-alphanumeric characters are removed."
  (slug-call (list #'slug-r--pre-alnum
                   #'slug-r-ud-default
                   #'slug-r-09Az-parens-or-space
                   #'slug-r--post-alnum)
             s))

(defun slug-to-nd-alnum (s)
  "Simplify string S to lowercase alphanumeric and single dashes.
It does not transliterate Unicode.

Diacritics are removed: “nd” stands for “no diacritics”.

Alphanumeric characters from other alphabets are left intact;
non-alphanumeric characters are removed."
  (slug-call (list #'slug-r--pre-alnum
                   #'slug-r-diacritics
                   #'slug-r-alnum-parens-or-space
                   #'slug-r--post-alnum)
             s))

(defun slug-to-nd-09az (s)
  "Simplify string S to 0-9a-z and single dashes.
It does not transliterate Unicode.

Diacritics are removed: “nd” stands for “no diacritics”.

Symbols, alphanumeric characters from other alphabets, and
non-alphanumeric characters are removed."
  (slug-call (list #'slug-r--pre-alnum
                   #'slug-r-diacritics
                   #'slug-r-09Az-parens-or-space
                   #'slug-r--post-alnum)
             s))

(defun slug-r--pre-alnum (s)
  "Helper function to process S before alphanumerics."
  (slug-call (list #'slug-r-org-stats-cookie
                   #'slug-r-html-tags
                   #'slug-r-urls-org
                   #'slug-r-urls-md
                   #'slug-r-and
                   #'slug-r-dot
                   #'slug-r-plus)
             s))

(defun slug-r--post-alnum (s)
  "Helper function to process S after alphanumerics."
  (slug-call (list #'slug-r-trim
                   #'slug-r-one-space
                   #'slug-r-parens-to-dashes
                   #'slug-r-parens
                   #'slug-r-space-to-dash
                   #'slug-r-trim-dash
                   #'slug-r-one-dash
                   #'downcase)
             s))


;;;;;; Unidecode: transliteration functions

(slug--describe
  "These are functions to transliterate Unicode characters into ASCII.
- Those in pure Emacs Lisp depend on the elisp package unidecode.
- Prefer to call Python instead? It depends on python3-unidecode.")

;;;;;;; Your pick

(defun slug-r-ud-default (s)
  "Transliterate Unicode string S into ASCII.
Use the function that ‘slug-r-ud-default-function’ points to."
  (funcall slug-r-ud-default-function s))


;;;;;;; Do it all in Emacs Lisp

(defun slug-r-ud-unidecode (s)
  "Transliterate Unicode string S into ASCII.
Do so by calling ‘unidecode’ on string S.

You'd need to have the unidecode elisp package."
  (unless (require 'unidecode nil 'noerror)
    (error "Package unidecode is needed but couldn't be required"))
  (unidecode s))


;;;;;;; Do it by calling Python

(defun slug-r-ud-python-unidecode (s)
  "Transliterate Unicode string S into ASCII.
Do so by calling python3-unidecode on string S.

That package would need to be installed on your system."
  (let* ((cmd (concat "import re, sys, unidecode; "
                      "print (unidecode.unidecode(sys.argv[1]))"))
         (res (with-temp-buffer
                (list :exit-status
                      (call-process "python3" nil t nil "-c" cmd s)
                      :output
                      (slug--chomp (buffer-string))))))
    (pcase (plist-get res :exit-status)
      (0   (plist-get res :output))
      (_   (error "Call to python3-unidecode failed")))))

(defun slug--chomp (s)
  "Remove a trailing newline from string S."
  (replace-regexp-in-string "\\(\r\n\\|\r\\|\n\\)\\'" "" s))


;;;;;; Replacement of diacritics

(slug--describe
  "These functions remove or expand diacritics from strings.")

(defun slug-r-expand-german (s)
  "Expand Eszett and umlauted characters in string S."
  (slug-call (list #'slug-r-expand-eszett
                   #'slug-r-expand-umlauted)
             s))

(defun slug-r-diacritics (s)
  "Remove diacritics from string S.
Note that this is unlikely to preserve the phonology of the original
language in which S is written.

For example, in German it'd do this: schön → schon, but “schoen” would
be the better and more standard diacritics-free conversion. If you need
that, you may want to filter S first through ‘slug-r-expand-german’.

For languages using non–Latin-based scripts, try ‘slug-r-ud-default’,
which may be able to transliterate S into ASCII reasonably enough."
  (slug-call (list #'slug-r-diacritics-undecomposable
                   #'slug-r-diacritics-decomposable)
             s))

(defun slug-r-diacritics-decomposable (s)
  "Remove diacritics from decomposable characters in string S."
  (let ((s (ucs-normalize-NFC-string s))
        cur dec res)
    (dotimes (idx (length s))
      (setq cur (aref s idx))
      (if (< cur ?z)
          (push cur res)
        (setq dec (get-char-code-property cur 'decomposition))
        (unless (symbolp (car dec))  ; e.g. ?™ -> (super 84 77)
          (push (car dec) res))))
    (concat (nreverse res))))

(defun slug-r-diacritics-undecomposable (s)
  "Remove diacritics from undecomposable characters in string S.
Deal with a few non-ASCII characters that aren't Unicode compositions
but could nevertheless be stripped into ASCII ones:
- đ ħ ł ø æ œ ß  →  d h l o ae oe ss
- Đ Ħ Ł Ø Æ Œ ẞ  →  D H L O Ae Oe SS

The first four are letters with a stroke.
The next two are ligatures.
The last is a German Eszett.

All these are present in a few languages, and are common enough to be
relevant for our sluggifying purposes."
  (slug-call (list #'slug-r-strokes
                   #'slug-r-expand-ligatures
                   #'slug-r-expand-eszett)
             s))

(defun slug-r-strokes (s)
  "Remove strokes from undecomposable characters in string S.
These are present in a few languages, and are common enough to be
relevant for our sluggifying purposes:
- đ ħ ł ø  →  d h l o
- Đ Ħ Ł Ø  →  D H L O."
  (slug--s-replace-all '(("đ" . "d")  ("Đ" . "D")
                         ("ħ" . "h")  ("Ħ" . "H")
                         ("ł" . "l")  ("Ł" . "L")
                         ("ø" . "o")  ("Ø" . "O"))
                       s))

(defun slug-r-expand-ligatures (s)
  "Expand ligatures of undecomposable characters in string S.
These are present in a few languages, and are common enough to be
relevant for our sluggifying purposes:
- æ œ  → ae oe
- Æ Œ  → Ae Oe."
  (slug--s-replace-all '(("æ" . "ae") ("Æ" . "Ae")
                         ("œ" . "oe") ("Œ" . "Oe"))
                       s))

(defun slug-r-expand-eszett (s)
  "Expand Eszett in string S.
- ß  →  ss
- ẞ  →  SS."
  (slug--s-replace-all '(("ß" . "ss") ("ẞ" . "SS")) s))

(defun slug-r-expand-umlauted (s)
  "Expand umlauted characters in string S.
- ä ö ü  →  ae oe ue
- Ä Ö Ü  →  Ae Oe Ue."
  (slug--s-replace-all '(("ä" . "ae") ("Ä" . "Ae")
                         ("ö" . "oe") ("Ö" . "Oe")
                         ("ü" . "ue") ("Ü" . "Ue"))
                       s))

(defun slug--s-replace-all (alist s)
  "In string S, replace each KEY from ALIST with its VALUE.
  Adapted from s.el's ‘s-replace-all’."
  (let ((case-fold-search nil))
    (replace-regexp-in-string
     (regexp-opt (mapcar #'car alist))
     (lambda (key) (cdr (assoc-string key alist)))
     s t t)))


;;;;;; Assortment of small replacement functions

(slug--describe
  "These are small replacement functions.
You can easily combine any number of them using ‘slug-call’.")

;;;;;;; Pre-alnum functions

(slug--describe
  "These would be typically used to clean a string before restricting it to
alphanumeric characters.")

(defun slug-r-org-stats-cookie (s)
  "Remove Org completion stats cookies from string S.
For example: [10/42]."
  ;; Regexp taken from ‘org-update-statistics-cookies
  (let ((cookie-rx "\\(\\(\\[[0-9]*%\\]\\)\\|\\(\\[[0-9]*/[0-9]*\\]\\)\\)"))
    (replace-regexp-in-string cookie-rx "" s)))

(defun slug-r-html-tags (s)
  "Remove <FOO>..</FOO> HTML tags from string S."
  (replace-regexp-in-string "<\\(?1:[a-z]+\\)[^>]*>.*</\\1>" "" s))

(defvar slug-url-rx
  (concat
   "\\("
   "news\\(post\\)?:\\|mailto:\\|file:"
   "\\|"
   "\\(ftp\\|https?\\|telnet\\|gopher\\|gemini\\|www\\|wais\\)://"
   "\\)")
  "Regexp matching the beginning of a URI.
Borrowed from ‘ffap-url-regexp’.")

(defun slug-r-urls-org (s)
  "Remove Org URLs from string S.
: [[URL][Description]]
is how they look like."
  (let ((org-url (concat "\\[\\[" slug-url-rx "[^]]+\\]\\[")))
    (replace-regexp-in-string org-url "[[" s)))

(defun slug-r-urls-md (s)
  "Remove Markdown URLs from string S.
: [Description](URL)
is how they look like."
  (let ((md-url (concat "\\](" slug-url-rx "[^)]+)")))
    (replace-regexp-in-string md-url "]" s)))

(defun slug-r-and (s)
  "Replace \"&\" with \" and \" in string S."
  (replace-regexp-in-string "&"   " and "  s))

(defun slug-r-dot (s)
  "Replace \".\" with \" dot \" in string S."
  (replace-regexp-in-string "\\." " dot "  s))

(defun slug-r-plus (s)
  "Replace \"+\" with \" plus \" in string S."
  (replace-regexp-in-string "\\+" " plus " s))

(defun slug-r-up-to-colon (s)
  "If string S has a colon, remove it and everything after."
  (replace-regexp-in-string ":.*" "" s))


;;;;;;; Alphanumeric restriction functions

(slug--describe
  "These restrict a string to only alphanumeric-ish characters.")

(defun slug-r-09Az-or-space (s)
  "Replace with a space all except 0-9A-Za-z in S."
  (replace-regexp-in-string "[^0-9A-Za-z]" " " s))

(defun slug-r-09Az-parens-or-space (s)
  "Replace with a space all except 0-9A-Za-z() in S."
  (replace-regexp-in-string "[^0-9A-Za-z()]" " " s))

(defun slug-r-alnum-or-space (s)
  "Replace with a space all except [:alnum:] in S.
  Note that [:alnum:] is not restricted to ASCII: alphanumeric characters
  in non-Latin scripts will be preserved."
  (replace-regexp-in-string "[^[:alnum:]]" " " s))

(defun slug-r-alnum-parens-or-space (s)
  "Replace with a space all except [:alnum:]() in S.
Note that [:alnum:] is not restricted to ASCII: alphanumeric characters
in non-Latin scripts will be preserved."
  (replace-regexp-in-string "[^[:alnum:]()]" " " s))


;;;;;;; Post-alnum functions

(slug--describe
  "These would be typically used to clean a string after restricting it to
alphanumeric characters.")

(defun slug-r-trim (s)
  "Remove leading and trailing whitespace from string S."
  (replace-regexp-in-string "\\(^[[:space:]]*\\|[[:space:]]*$\\)" "" s))

(defun slug-r-one-space (s)
  "Replace two or more spaces with a single space in S."
  (replace-regexp-in-string "[[:space:]]\\{2,\\}" " " s))

(defun slug-r-parens-to-dashes (s)
  "Replace parentheses with dashes in parenthesized substrings of S."
  (replace-regexp-in-string "\\s-*([[:space:]]*\\([^)]+?\\)[[:space:]]*)\\s-*"
                            " -\\1- " s))

(defun slug-r-parens (s)
  "Remove parentheses characters from string S."
  (replace-regexp-in-string "[()]" "" s))

(defun slug-r-space-to-dash (s)
  "Replace spaces with dashes in string S."
  (replace-regexp-in-string " " "-" s))

(defun slug-r-trim-dash (s)
  "Remove leading and trailing dashes from string S."
  (replace-regexp-in-string "\\(^[-]*\\|[-]*$\\)" "" s))

(defun slug-r-one-dash (s)
  "Replace sequences of dashes in string S with a single one."
  (replace-regexp-in-string "-+" "-" s))


;;;;; Generation of unique identifiers

(slug--describe
  "These functions generate unique strings, which are useful to prepend or
append to slugs when there's a risk of duplicates.")


;;;;;; Random hex digits

(slug--describe
  "Make your slugs unique by appending a random hex string.")

(defun slug-a-append-hex-maybe (s)
  "Append random hex to string S if one is not yet present."
  (let* ((hex-rx (format "[0-9a-f]\\{%s\\}" slug-a-hex-digits-min))
         (post-hex-rx (concat "-" hex-rx "\\'")))
    (if (string-match post-hex-rx s) s (slug-a-append-hex s))))

(defun slug-a-append-hex (s)
  "Append hex string to string S.
To avoid appending more than once, use ‘slug-a-append-hex-maybe’."
  (format "%s-%s" s (slug--a-make-hex)))

(defun slug--a-make-hex ()
  "Create a random string of ‘slug-a-hex-digits’ hex digits."
  (slug--a-make-hex-n slug-a-hex-digits))

(defun slug--a-make-hex-n (n)
  "Create a random string of N hex digits.
If ‘slug-a-hex-hash-algo’ is \\='sha512 (the default), then
N can be between 0 and 128."
  ;; Modified and simplified from ‘org-id-uuid’.
  (let* ((rnd (format "%s%s%s%s%s%s%s"
                      (random)
                      (decode-time)
                      (user-uid)
                      (emacs-pid)
                      (user-full-name)
                      user-mail-address
                      (recent-keys)))
         (hsh (secure-hash slug-a-hex-hash-algo rnd)))
    (substring hsh 0 n)))


;;;;;; Dates

(slug--describe
  "Make your slugs unique by prepending a date.")

(defun slug-a-prepend-date-maybe (s)
  "Prepend date to string S if one is not yet present."
  (let ((pre-date-rx (concat "\\`" slug-a-date-rx "-")))
    (if (string-match pre-date-rx s) s (slug-a-prepend-date s))))

(defun slug-a-prepend-date (s)
  "Prepend date to string S.
To avoid prepending more than once, use ‘slug-a-prepend-date-maybe’."
  (format "%s-%s" (slug--a-make-today) s))

(defun slug--a-make-today ()
  "Create a string of today's date."
  (format-time-string slug-a-date-format))


;;;;;; Timestamps

(slug--describe
  "Make your slugs unique by appending a timestamp.")

(defun slug-a-append-ts-maybe (s)
  "Append timestamp to string S if one is not yet present."
  (let ((post-ts-rx (concat "-" slug-a-ts-rx "\\'")))
    (if (string-match post-ts-rx s) s (slug-a-append-ts s))))

(defun slug-a-append-ts (s)
  "Append timestamp to string S.
To avoid appending more than once, use ‘slug-a-append-ts-maybe’."
  (format "%s-%s" s (slug--a-make-now)))

(defun slug--a-make-now ()
  "Create a timestamp string of now."
  (format-time-string slug-a-ts-format))


;;;; Commands
;;;;; See README

(slug--describe
  "Commands to open slug's README.org. Optionally, find things in it.")

;;;###autoload
(defun slug-see-readme (&optional heading narrow)
  "Open slug's README.org file.
Search for the file in slug.el's directory.

If found, open it read-only.

If optional argument HEADING is passed, try to navigate to the
heading after opening it. HEADING should be a string.

If optional argument NARROW is non-nil, narrow to that heading.
This argument has no effect if HEADING is nil or not found."
  (interactive)
  (let ((readme slug--readme-org))
    (if (file-exists-p readme)
        (let ((pr (make-progress-reporter
                   (format "Opening %s ... "
                           (abbreviate-file-name readme)))))
          (find-file-read-only readme)
          (when heading
            (slug--goto-org-heading heading narrow))
          (progress-reporter-done pr))
      (message "Couldn't find %s's README.org" slug--name))))

;;;###autoload
(defun slug-see-news ()
  "See the News in slug's README.org file."
  (interactive)
  (slug-see-readme "News" 'narrow)
  (slug--display-org-subtree))

(defun slug--display-org-subtree ()
  "Selectively display org subtree."
  (let ((cmds '(outline-hide-subtree
                outline-show-children
                outline-next-heading
                outline-show-branches)))
    (and (fboundp (nth 0 cmds))
         (fboundp (nth 1 cmds))
         (fboundp (nth 2 cmds))
         (fboundp (nth 3 cmds))
         (mapc #'funcall cmds))))

(defun slug--goto-org-heading (heading &optional narrow)
  "Navigate to org HEADING and optionally NARROW to it."
  (let* ((hrx (format "^[*]+ %s" heading))
         (pos (save-match-data
                (save-excursion
                  (save-restriction
                    (widen)
                    (goto-char (point-max))
                    (re-search-backward hrx nil t 1))))))
    (when pos
      (widen)
      (goto-char pos)
      (if (and narrow (fboundp 'org-narrow-to-subtree))
          (org-narrow-to-subtree)
        (recenter-top-bottom 1))
      (when (fboundp 'outline-show-subtree)
        (outline-show-subtree))
      (when (fboundp 'org-flag-drawer)
        (save-excursion
          (forward-line 1)
          (org-flag-drawer t))))))


;;;; Wrapping up

(provide 'slug)

;; Local Variables:
;; coding:                     utf-8
;; indent-tabs-mode:           nil
;; sentence-end-double-space:  nil
;; outline-regexp:             ";;;;* "
;; End:

;;; slug.el ends here