naming.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. package genopenapi
  2. import (
  3. "reflect"
  4. "regexp"
  5. "strings"
  6. )
  7. // LookupNamingStrategy looks up the given naming strategy and returns the naming
  8. // strategy function for it. The naming strategy function takes in the list of all
  9. // fully-qualified proto message names, and returns a mapping from fully-qualified
  10. // name to OpenAPI name.
  11. func LookupNamingStrategy(strategyName string) func([]string) map[string]string {
  12. switch strings.ToLower(strategyName) {
  13. case "fqn":
  14. return resolveNamesFQN
  15. case "legacy":
  16. return resolveNamesLegacy
  17. case "simple":
  18. return resolveNamesSimple
  19. case "package":
  20. return resolveNamesPackage
  21. }
  22. return nil
  23. }
  24. // resolveNamesFQN uses the fully-qualified proto message name as the
  25. // OpenAPI name, stripping the leading dot.
  26. func resolveNamesFQN(messages []string) map[string]string {
  27. uniqueNames := make(map[string]string, len(messages))
  28. for _, p := range messages {
  29. // strip leading dot from proto fqn
  30. uniqueNames[p] = p[1:]
  31. }
  32. return uniqueNames
  33. }
  34. // resolveNamesLegacy takes the names of all proto messages and generates unique references by
  35. // applying the legacy heuristics for deriving unique names: starting from the bottom of the name hierarchy, it
  36. // determines the minimum number of components necessary to yield a unique name, adds one
  37. // to that number, and then concatenates those last components with no separator in between
  38. // to form a unique name.
  39. //
  40. // E.g., if the fully qualified name is `.a.b.C.D`, and there are other messages with fully
  41. // qualified names ending in `.D` but not in `.C.D`, it assigns the unique name `bCD`.
  42. func resolveNamesLegacy(messages []string) map[string]string {
  43. return resolveNamesUniqueWithContext(messages, 1, "", false)
  44. }
  45. // resolveNamesSimple takes the names of all proto messages and generates unique references by using a simple
  46. // heuristic: starting from the bottom of the name hierarchy, it determines the minimum
  47. // number of components necessary to yield a unique name, and then concatenates those last
  48. // components with a "." separator in between to form a unique name.
  49. //
  50. // E.g., if the fully qualified name is `.a.b.C.D`, and there are other messages with
  51. // fully qualified names ending in `.D` but not in `.C.D`, it assigns the unique name `C.D`.
  52. func resolveNamesSimple(messages []string) map[string]string {
  53. return resolveNamesUniqueWithContext(messages, 0, ".", false)
  54. }
  55. // resolveNamesPackage takes the names of all proto messages and generates unique references by
  56. // starting with the package-scoped name (with nested message types qualified by their containing
  57. // "parent" types), and then following the "simple" heuristic above to add package name components
  58. // until each message has a unique name with a "." between each component.
  59. //
  60. // E.g., if the fully qualified name is `.a.b.C.D`, the name is `C.D` unless there is another
  61. // package-scoped name ending in "C.D", in which case it would be `b.C.D` (unless that also
  62. // conflicted, in which case the name would be the fully-qualified `a.b.C`).
  63. func resolveNamesPackage(messages []string) map[string]string {
  64. return resolveNamesUniqueWithContext(messages, 0, ".", true)
  65. }
  66. // For the "package" naming strategy, we rely on the convention that package names are lowercase
  67. // but message names are capitalized.
  68. var pkgEndRegexp = regexp.MustCompile(`\.[A-Z]`)
  69. // Take the names of every proto message and generates a unique reference by:
  70. // first, separating each message name into its components by splitting at dots. Then,
  71. // take the shortest suffix slice from each components slice that is unique among all
  72. // messages, and convert it into a component name by taking extraContext additional
  73. // components into consideration and joining all components with componentSeparator.
  74. func resolveNamesUniqueWithContext(messages []string, extraContext int, componentSeparator string, qualifyNestedMessages bool) map[string]string {
  75. packagesByDepth := make(map[int][][]string)
  76. uniqueNames := make(map[string]string)
  77. hierarchy := func(pkg string) []string {
  78. if !qualifyNestedMessages {
  79. return strings.Split(pkg, ".")
  80. }
  81. pkgEnd := pkgEndRegexp.FindStringIndex(pkg)
  82. if pkgEnd == nil {
  83. // Fall back to non-qualified behavior if search based on convention fails.
  84. return strings.Split(pkg, ".")
  85. }
  86. // Return each package component as an element, followed by the full message name
  87. // (potentially qualified, if nested) as a single element.
  88. qualifiedPkgName := pkg[:pkgEnd[0]]
  89. nestedTypeName := pkg[pkgEnd[0]+1:]
  90. return append(strings.Split(qualifiedPkgName, "."), nestedTypeName)
  91. }
  92. for _, p := range messages {
  93. h := hierarchy(p)
  94. for depth := range h {
  95. if _, ok := packagesByDepth[depth]; !ok {
  96. packagesByDepth[depth] = make([][]string, 0)
  97. }
  98. packagesByDepth[depth] = append(packagesByDepth[depth], h[len(h)-depth:])
  99. }
  100. }
  101. count := func(list [][]string, item []string) int {
  102. i := 0
  103. for _, element := range list {
  104. if reflect.DeepEqual(element, item) {
  105. i++
  106. }
  107. }
  108. return i
  109. }
  110. for _, p := range messages {
  111. h := hierarchy(p)
  112. depth := 0
  113. for ; depth < len(h); depth++ {
  114. // depth + extraContext > 0 ensures that we only break for values of depth when the
  115. // resulting slice of name components is non-empty. Otherwise, we would return the
  116. // empty string as the concise unique name is len(messages) == 1 (which is
  117. // technically correct).
  118. if depth+extraContext > 0 && count(packagesByDepth[depth], h[len(h)-depth:]) == 1 {
  119. break
  120. }
  121. }
  122. start := len(h) - depth - extraContext
  123. if start < 0 {
  124. start = 0
  125. }
  126. uniqueNames[p] = strings.Join(h[start:], componentSeparator)
  127. }
  128. return uniqueNames
  129. }