#include <stdio.h> #include <string.h> #include <stdbool.h> char const* const commands[] = { #ifdef __aarch64__ # include "commands-aarch64.h" #elif defined __arm__ # include "commands-arm.h" #elif defined __x86_64__ # include "commands-x86_64.h" #elif defined __i686__ # include "commands-i686.h" #else # error Failed to detect arch #endif }; static inline int termux_min3(unsigned int a, unsigned int b, unsigned int c) { return (a < b ? (a < c ? a : c) : (b < c ? b : c)); } static int termux_levenshtein_distance(char const* restrict s1, char const* restrict s2) { unsigned int s1len = strlen(s1); unsigned int s2len = strlen(s2); unsigned int matrix[s2len+1][s1len+1]; matrix[0][0] = 0; for (unsigned x = 1; x <= s2len; x++) matrix[x][0] = matrix[x-1][0] + 1; for (unsigned y = 1; y <= s1len; y++) matrix[0][y] = matrix[0][y-1] + 1; for (unsigned x = 1; x <= s2len; x++) for (unsigned y = 1; y <= s1len; y++) matrix[x][y] = termux_min3(matrix[x-1][y] + 1, matrix[x][y-1] + 1, matrix[x-1][y-1] + (s1[y-1] == s2[x-1] ? 0 : 1)); return matrix[s2len][s1len]; } int main(int argc, char** argv) { if (argc != 2) { fprintf(stderr, "usage: termux-command-not-found <command>\n"); return 1; } char* command_not_found = argv[1]; int best_distance = -1; int guesses_at_best_distance = 0; char current_package[128]; char best_package_guess[128]; char best_command_guess[128]; const int num_commands = sizeof(commands) / sizeof(commands[0]); for (int i = 0; i < num_commands; i++) { char const* current_line = commands[i]; if (current_line[0] == ' ') { // Binary char const* binary_name = current_line + 1; int distance = termux_levenshtein_distance(command_not_found, binary_name); if (distance == 0 && strcmp(command_not_found, binary_name) == 0) { if (best_distance == 0) { fprintf(stderr, "or\n"); } else { fprintf(stderr, "The program '%s' is not installed. Install it by executing:\n", binary_name); } fprintf(stderr, " pkg install %s\n", current_package); best_distance = 0; } else if (best_distance == distance) { guesses_at_best_distance++; } else if (best_distance == -1 || best_distance > distance) { guesses_at_best_distance = 1; best_distance = distance; strncpy(best_command_guess, binary_name, sizeof(best_command_guess)); strncpy(best_package_guess, current_package, sizeof(best_package_guess)); } } else { // Package strncpy(current_package, current_line, sizeof(current_package)); } } if (best_distance == 0) return 127; if (best_distance == -1 || best_distance > 3) { fprintf(stderr, "%s: command not found\n", command_not_found); } else { fprintf(stderr, "No command '%s' found, did you mean:\n", command_not_found); if (guesses_at_best_distance == 1) { // Only one suggestion - show it: fprintf(stderr, " Command '%s' from package '%s'\n", best_command_guess, best_package_guess); } else { // Multiple suggestions at the same distance - show them all: for (int i = 0; i < num_commands; i++) { char const* current_line = commands[i]; if (current_line[0] == ' ') { // Binary char const* binary_name = current_line + 1; int distance = termux_levenshtein_distance(command_not_found, binary_name); if (best_distance == distance) { fprintf(stderr, " Command '%s' from package '%s'\n", binary_name, current_package); } } else { // Package strncpy(current_package, current_line, sizeof(current_package)); } } } } return 127; }