feat(cli): add Jaro-Winkler algorithm and getSuggestion method to search helpers
This commit introduces a new file "search.ts" in the CLI helpers. It adds the implementation of the Jaro-Winkler string comparison algorithm and a "getSuggestion" function. The getSuggestion function uses the Jaro-Winkler algorithm for comparing component names against an input string and providing suitable suggestions.
This commit is contained in:
parent
c46cd0b295
commit
48cb320303
61
packages/cli/src/helpers/search.ts
Normal file
61
packages/cli/src/helpers/search.ts
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
export async function jaroWinkler(a: string, b: string): Promise<number> {
|
||||||
|
const p = 0.1
|
||||||
|
|
||||||
|
if (!a.length || !b.length) return 0.0
|
||||||
|
if (a === b) return 1.0
|
||||||
|
|
||||||
|
const range = Math.floor(Math.max(a.length, b.length) / 2) - 1
|
||||||
|
let matches = 0
|
||||||
|
|
||||||
|
let aMatches = new Array(a.length)
|
||||||
|
let bMatches = new Array(b.length)
|
||||||
|
|
||||||
|
for (let i = 0; i < a.length; i++) {
|
||||||
|
const start = i >= range ? i - range : 0
|
||||||
|
const end = i + range <= b.length - 1 ? i + range : b.length - 1
|
||||||
|
|
||||||
|
for (let j = start; j <= end; j++) {
|
||||||
|
if (bMatches[j] !== true && a[i] === b[j]) {
|
||||||
|
++matches
|
||||||
|
aMatches[i] = bMatches[j] = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (matches === 0) return 0.0
|
||||||
|
|
||||||
|
let t = 0
|
||||||
|
|
||||||
|
let point: number
|
||||||
|
|
||||||
|
for (point = 0; point < a.length; point++) if (aMatches[point]) break
|
||||||
|
|
||||||
|
for (let i = point; i < a.length; i++)
|
||||||
|
if (aMatches[i]) {
|
||||||
|
let j
|
||||||
|
for (j = point; j < b.length; j++)
|
||||||
|
if (bMatches[j]) {
|
||||||
|
point = j + 1
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a[i] !== b[j]) ++t
|
||||||
|
}
|
||||||
|
|
||||||
|
t = t / 2.0
|
||||||
|
|
||||||
|
const J = (matches / a.length + matches / b.length + (matches - t) / matches) / 3
|
||||||
|
return J + Math.min((p * t) / matches, 1) * (1 - J)
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getSuggestion(componentNames: string[], input: string): Promise<string[]> {
|
||||||
|
const componentJw = await Promise.all(
|
||||||
|
componentNames.map(async (name) => [name, await jaroWinkler(name, input)] as const),
|
||||||
|
)
|
||||||
|
return componentJw
|
||||||
|
.filter(([_, score]) => score > 0)
|
||||||
|
.sort((a, b) => b[1] - a[1])
|
||||||
|
.slice(0, 5)
|
||||||
|
.map(([name]) => name)
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user