/names/generator.go

https://codeberg.org/momar/linkding · Go · 107 lines · 85 code · 13 blank · 9 comment · 18 complexity · 5aed72306973fbda3484e5b072f63b6e MD5 · raw file

  1. package names
  2. import (
  3. "crypto/rand"
  4. "math"
  5. "math/big"
  6. "regexp"
  7. "strconv"
  8. "strings"
  9. "codeberg.org/momar/linkding/names/words"
  10. )
  11. // Types maps an expansion identifier to a slice of possible values. `Generate("#x")` for example returns a random value from the `Types['x']` slice
  12. var Types = map[byte][]string{
  13. 'x': strings.Split("abcdefghijklmnopqrstuvwxyz0123456789", ""),
  14. 'd': strings.Split("0123456789", ""),
  15. 'l': strings.Split("abcdefghijklmnopqrstuvwxyz", ""),
  16. 'v': strings.Split("aeiou", ""),
  17. 'c': strings.Split("bcdfghjklmnpqrstvwxyz", ""),
  18. 'w': append(words.Adjectives, words.Nouns...),
  19. 'n': words.Nouns,
  20. 'a': words.Adjectives,
  21. 'f': words.Faust,
  22. }
  23. var MaximumLength = 1024
  24. var specification *regexp.Regexp
  25. var expansion *regexp.Regexp
  26. var repetition = regexp.MustCompile(`(#?.|\[[^\]]*\])\{\d+\}`)
  27. // GenerateSafe generates a safe name according to a set of given specs seperated by "|" by trying to generate `tries` names for each spec and checking if an item with that name already exists using the `exists` function.
  28. func GenerateSafe(specs string, tries uint, exists func(string) bool) string {
  29. for _, spec := range strings.Split(specs, "|") {
  30. // Try a spec (from first to last)
  31. var i uint
  32. for ; i < tries; i++ {
  33. // Try to generate a name
  34. name := Generate(spec)
  35. if !exists(name) {
  36. return name
  37. }
  38. }
  39. }
  40. return ""
  41. }
  42. // Generate a name according to the given spec. A spec can only contain letters (a-z), numbers (0-9), dashes (-), expansions matching a type identifier from the Types map (#d) and repetitions ([text]{5} #d{5} [xyz#d]{5}). An invalid spec returns an empty string, which should be handled.
  43. func Generate(spec string) string {
  44. spec = strings.TrimSpace(spec)
  45. if spec == "" {
  46. spec = "#a-#a-#n"
  47. }
  48. // Generate regular expression with type characters
  49. if expansion == nil {
  50. typeChars := ""
  51. for k := range Types {
  52. typeChars += string(k)
  53. }
  54. expansion = regexp.MustCompile(`#[` + typeChars + `]`)
  55. specification = regexp.MustCompile(`^([a-z0-9-]+|#[` + typeChars + `]|(\[([a-z0-9-]+|#[a-z])+\])?\{\d+\})+$`)
  56. }
  57. // Check syntax
  58. if !specification.MatchString(spec) {
  59. return ""
  60. }
  61. // Expand repetitions
  62. for {
  63. i := repetition.FindStringIndex(spec)
  64. if i == nil {
  65. break
  66. }
  67. placeholder := spec[i[0]:i[1]]
  68. count, _ := strconv.Atoi(placeholder[strings.LastIndex(placeholder, "{")+1 : len(placeholder)-1])
  69. content := strings.TrimPrefix(strings.TrimSuffix(placeholder[:strings.LastIndex(placeholder, "{")], "]"), "[")
  70. spec = spec[:i[0]] + strings.Repeat(content, int(math.Min(float64(count), float64(MaximumLength*2)/float64(len(content))))) + spec[i[1]:]
  71. if len(spec) >= MaximumLength*2 {
  72. break
  73. }
  74. }
  75. // Replace placeholders
  76. for {
  77. i := expansion.FindStringIndex(spec)
  78. if i == nil {
  79. break
  80. }
  81. placeholder := spec[i[0]:i[1]]
  82. words, ok := Types[placeholder[1]]
  83. if !ok {
  84. words = []string{placeholder}
  85. }
  86. n, _ := rand.Int(rand.Reader, big.NewInt(int64(len(words))))
  87. w := words[n.Int64()]
  88. spec = spec[:i[0]] + w + spec[i[1]:]
  89. if i[0]+len(w) >= MaximumLength {
  90. return spec[:MaximumLength]
  91. }
  92. }
  93. return spec
  94. }