2022-03-03 11:50:35 +02:00
|
|
|
|
#include "suggestions.hh"
|
|
|
|
|
#include "ansicolor.hh"
|
|
|
|
|
#include "util.hh"
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
2022-03-03 14:12:27 +02:00
|
|
|
|
int levenshteinDistance(std::string_view first, std::string_view second)
|
2022-03-03 11:50:35 +02:00
|
|
|
|
{
|
|
|
|
|
// Implementation borrowed from
|
|
|
|
|
// https://en.wikipedia.org/wiki/Levenshtein_distance#Iterative_with_two_matrix_rows
|
|
|
|
|
|
|
|
|
|
int m = first.size();
|
|
|
|
|
int n = second.size();
|
|
|
|
|
|
|
|
|
|
auto v0 = std::vector<int>(n+1);
|
|
|
|
|
auto v1 = std::vector<int>(n+1);
|
|
|
|
|
|
|
|
|
|
for (auto i = 0; i <= n; i++)
|
|
|
|
|
v0[i] = i;
|
|
|
|
|
|
|
|
|
|
for (auto i = 0; i < m; i++) {
|
|
|
|
|
v1[0] = i+1;
|
|
|
|
|
|
|
|
|
|
for (auto j = 0; j < n; j++) {
|
|
|
|
|
auto deletionCost = v0[j+1] + 1;
|
|
|
|
|
auto insertionCost = v1[j] + 1;
|
|
|
|
|
auto substitutionCost = first[i] == second[j] ? v0[j] : v0[j] + 1;
|
|
|
|
|
v1[j+1] = std::min({deletionCost, insertionCost, substitutionCost});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::swap(v0, v1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return v0[n];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Suggestions Suggestions::bestMatches (
|
|
|
|
|
std::set<std::string> allMatches,
|
|
|
|
|
std::string query)
|
|
|
|
|
{
|
|
|
|
|
std::set<Suggestion> res;
|
|
|
|
|
for (const auto & possibleMatch : allMatches) {
|
|
|
|
|
res.insert(Suggestion {
|
2022-03-03 14:12:27 +02:00
|
|
|
|
.distance = levenshteinDistance(query, possibleMatch),
|
2022-03-03 11:50:35 +02:00
|
|
|
|
.suggestion = possibleMatch,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return Suggestions { res };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Suggestions Suggestions::trim(int limit, int maxDistance) const
|
|
|
|
|
{
|
|
|
|
|
std::set<Suggestion> res;
|
|
|
|
|
|
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
|
|
for (auto & elt : suggestions) {
|
2022-03-03 14:12:27 +02:00
|
|
|
|
if (count >= limit || elt.distance > maxDistance)
|
2022-03-03 11:50:35 +02:00
|
|
|
|
break;
|
|
|
|
|
count++;
|
|
|
|
|
res.insert(elt);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Suggestions{res};
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-07 11:04:57 +02:00
|
|
|
|
std::string Suggestion::to_string() const
|
2022-03-03 11:50:35 +02:00
|
|
|
|
{
|
|
|
|
|
return ANSI_WARNING + filterANSIEscapes(suggestion) + ANSI_NORMAL;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-07 11:04:57 +02:00
|
|
|
|
std::string Suggestions::to_string() const
|
2022-03-03 11:50:35 +02:00
|
|
|
|
{
|
|
|
|
|
switch (suggestions.size()) {
|
|
|
|
|
case 0:
|
|
|
|
|
return "";
|
|
|
|
|
case 1:
|
2022-03-07 11:04:57 +02:00
|
|
|
|
return suggestions.begin()->to_string();
|
2022-03-03 11:50:35 +02:00
|
|
|
|
default: {
|
|
|
|
|
std::string res = "one of ";
|
|
|
|
|
auto iter = suggestions.begin();
|
2022-03-07 11:04:57 +02:00
|
|
|
|
res += iter->to_string(); // Iter can’t be end() because the container isn’t null
|
2022-03-03 11:50:35 +02:00
|
|
|
|
iter++;
|
|
|
|
|
auto last = suggestions.end(); last--;
|
|
|
|
|
for ( ; iter != suggestions.end() ; iter++) {
|
|
|
|
|
res += (iter == last) ? " or " : ", ";
|
2022-03-07 11:04:57 +02:00
|
|
|
|
res += iter->to_string();
|
2022-03-03 11:50:35 +02:00
|
|
|
|
}
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Suggestions & Suggestions::operator+=(const Suggestions & other)
|
|
|
|
|
{
|
|
|
|
|
suggestions.insert(
|
|
|
|
|
other.suggestions.begin(),
|
|
|
|
|
other.suggestions.end()
|
|
|
|
|
);
|
|
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-07 11:04:57 +02:00
|
|
|
|
std::ostream & operator<<(std::ostream & str, const Suggestion & suggestion)
|
|
|
|
|
{
|
|
|
|
|
return str << suggestion.to_string();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::ostream & operator<<(std::ostream & str, const Suggestions & suggestions)
|
|
|
|
|
{
|
|
|
|
|
return str << suggestions.to_string();
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-03 11:50:35 +02:00
|
|
|
|
}
|