Create a gist now

Instantly share code, notes, and snippets.

What would you like to do?
A version of Ubuntu's command-not-found script modified with an option to show all packages identified as possible misspellings. (by default these are only shown if there are <= 15)
# (c) Zygmunt Krynicki 2005, 2006, 2007, 2008
# Licensed under GPL, see COPYING for the whole text
from __future__ import absolute_import, print_function
__version__ = "0.3"
import sys
if sys.path and sys.path[0] == '/usr/lib':
# Avoid ImportError noise due to odd installation location.
if sys.version < '3':
# We might end up being executed with Python 2 due to an old
# /etc/bash.bashrc.
import os
if "COMMAND_NOT_FOUND_FORCE_PYTHON2" not in os.environ:
os.execvp("python3", [sys.argv[0]] + sys.argv)
import gettext
import locale
from optparse import OptionParser
from CommandNotFound.util import crash_guard
from CommandNotFound import CommandNotFound
except KeyboardInterrupt:
import sys
def enable_i18n():
cnf = gettext.translation("command-not-found", fallback=True)
kwargs = {}
if sys.version < '3':
kwargs["unicode"] = True
locale.setlocale(locale.LC_ALL, '')
def fix_sys_argv(encoding=None):
Fix sys.argv to have only unicode strings, not binary strings.
This is required by various places where such argument might be
automatically coerced to unicode string for formatting
if encoding is None:
encoding = locale.getpreferredencoding()
sys.argv = [arg.decode(encoding) for arg in sys.argv]
class LocaleOptionParser(OptionParser):
OptionParser is broken as its implementation of _get_encoding() uses
sys.getdefaultencoding() which is ascii, what it should be using is
locale.getpreferredencoding() which returns value based on LC_CTYPE (most
likely) and allows for UTF-8 encoding to be used.
def _get_encoding(self, file):
encoding = getattr(file, "encoding", None)
if not encoding:
encoding = locale.getpreferredencoding()
return encoding
def main():
if sys.version < '3':
parser = LocaleOptionParser(
usage=_("%prog [options] <command-name>"))
parser.add_option('-d', '--data-dir', action='store',
help=_("use this path to locate data fields"))
parser.add_option('--ignore-installed', '--ignore-installed',
action='store_true', default=False,
help=_("ignore local binaries and display the available packages"))
action='store_true', default=False,
help=_("don't print '<command-name>: command not found'"))
parser.add_option('-n', '--num-matches', type="int", action='store',
default=15, help=_("maximum number of suggested spellings to print"))
parser.add_option('--all', action='store_const', const=99999, dest='num_matches',
help=_("print all suggested spellings"))
(options, args) = parser.parse_args()
class CNFMoreSpellingPrinting(CommandNotFound):
def print_spelling_suggestion(self, word, min_len=3, max_len=options.num_matches):
return CommandNotFound.print_spelling_suggestion(self, word, min_len, max_len)
if len(args) == 1:
cnf = CNFMoreSpellingPrinting(options.data_dir)
if not cnf.advise(args[0], options.ignore_installed) and not options.no_failure_msg:
print(_("%s: command not found") % args[0], file=sys.stderr)
if __name__ == "__main__":
crash_guard(main, BUG_REPORT_URL, __version__)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment