mirror of
				https://github.com/python/cpython.git
				synced 2025-11-03 23:21:29 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			156 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			156 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""A simple SQLite CLI for the sqlite3 module.
 | 
						|
 | 
						|
Apart from using 'argparse' for the command-line interface,
 | 
						|
this module implements the REPL as a thin wrapper around
 | 
						|
the InteractiveConsole class from the 'code' stdlib module.
 | 
						|
"""
 | 
						|
import sqlite3
 | 
						|
import sys
 | 
						|
 | 
						|
from argparse import ArgumentParser
 | 
						|
from code import InteractiveConsole
 | 
						|
from textwrap import dedent
 | 
						|
from _colorize import get_theme, theme_no_color
 | 
						|
 | 
						|
from ._completer import completer
 | 
						|
 | 
						|
 | 
						|
def execute(c, sql, suppress_errors=True, theme=theme_no_color):
 | 
						|
    """Helper that wraps execution of SQL code.
 | 
						|
 | 
						|
    This is used both by the REPL and by direct execution from the CLI.
 | 
						|
 | 
						|
    'c' may be a cursor or a connection.
 | 
						|
    'sql' is the SQL string to execute.
 | 
						|
    """
 | 
						|
 | 
						|
    try:
 | 
						|
        for row in c.execute(sql):
 | 
						|
            print(row)
 | 
						|
    except sqlite3.Error as e:
 | 
						|
        t = theme.traceback
 | 
						|
        tp = type(e).__name__
 | 
						|
        try:
 | 
						|
            tp += f" ({e.sqlite_errorname})"
 | 
						|
        except AttributeError:
 | 
						|
            pass
 | 
						|
        print(
 | 
						|
            f"{t.type}{tp}{t.reset}: {t.message}{e}{t.reset}", file=sys.stderr
 | 
						|
        )
 | 
						|
        if not suppress_errors:
 | 
						|
            sys.exit(1)
 | 
						|
 | 
						|
 | 
						|
class SqliteInteractiveConsole(InteractiveConsole):
 | 
						|
    """A simple SQLite REPL."""
 | 
						|
 | 
						|
    def __init__(self, connection, use_color=False):
 | 
						|
        super().__init__()
 | 
						|
        self._con = connection
 | 
						|
        self._cur = connection.cursor()
 | 
						|
        self._use_color = use_color
 | 
						|
 | 
						|
    def runsource(self, source, filename="<input>", symbol="single"):
 | 
						|
        """Override runsource, the core of the InteractiveConsole REPL.
 | 
						|
 | 
						|
        Return True if more input is needed; buffering is done automatically.
 | 
						|
        Return False if input is a complete statement ready for execution.
 | 
						|
        """
 | 
						|
        theme = get_theme(force_no_color=not self._use_color)
 | 
						|
 | 
						|
        if not source or source.isspace():
 | 
						|
            return False
 | 
						|
        # Remember to update CLI_COMMANDS in _completer.py
 | 
						|
        if source[0] == ".":
 | 
						|
            match source[1:].strip():
 | 
						|
                case "version":
 | 
						|
                    print(sqlite3.sqlite_version)
 | 
						|
                case "help":
 | 
						|
                    t = theme.syntax
 | 
						|
                    print(f"Enter SQL code or one of the below commands, and press enter.\n\n"
 | 
						|
                          f"{t.builtin}.version{t.reset}    Print underlying SQLite library version\n"
 | 
						|
                          f"{t.builtin}.help{t.reset}       Print this help message\n"
 | 
						|
                          f"{t.builtin}.quit{t.reset}       Exit the CLI, equivalent to CTRL-D\n")
 | 
						|
                case "quit":
 | 
						|
                    sys.exit(0)
 | 
						|
                case "":
 | 
						|
                    pass
 | 
						|
                case _ as unknown:
 | 
						|
                    t = theme.traceback
 | 
						|
                    self.write(f'{t.type}Error{t.reset}: {t.message}unknown '
 | 
						|
                               f'command: "{unknown}"{t.reset}\n')
 | 
						|
        else:
 | 
						|
            if not sqlite3.complete_statement(source):
 | 
						|
                return True
 | 
						|
            execute(self._cur, source, theme=theme)
 | 
						|
        return False
 | 
						|
 | 
						|
 | 
						|
def main(*args):
 | 
						|
    parser = ArgumentParser(
 | 
						|
        description="Python sqlite3 CLI",
 | 
						|
        color=True,
 | 
						|
    )
 | 
						|
    parser.add_argument(
 | 
						|
        "filename", type=str, default=":memory:", nargs="?",
 | 
						|
        help=(
 | 
						|
            "SQLite database to open (defaults to ':memory:'). "
 | 
						|
            "A new database is created if the file does not previously exist."
 | 
						|
        ),
 | 
						|
    )
 | 
						|
    parser.add_argument(
 | 
						|
        "sql", type=str, nargs="?",
 | 
						|
        help=(
 | 
						|
            "An SQL query to execute. "
 | 
						|
            "Any returned rows are printed to stdout."
 | 
						|
        ),
 | 
						|
    )
 | 
						|
    parser.add_argument(
 | 
						|
        "-v", "--version", action="version",
 | 
						|
        version=f"SQLite version {sqlite3.sqlite_version}",
 | 
						|
        help="Print underlying SQLite library version",
 | 
						|
    )
 | 
						|
    args = parser.parse_args(*args)
 | 
						|
 | 
						|
    if args.filename == ":memory:":
 | 
						|
        db_name = "a transient in-memory database"
 | 
						|
    else:
 | 
						|
        db_name = repr(args.filename)
 | 
						|
 | 
						|
    # Prepare REPL banner and prompts.
 | 
						|
    if sys.platform == "win32" and "idlelib.run" not in sys.modules:
 | 
						|
        eofkey = "CTRL-Z"
 | 
						|
    else:
 | 
						|
        eofkey = "CTRL-D"
 | 
						|
    banner = dedent(f"""
 | 
						|
        sqlite3 shell, running on SQLite version {sqlite3.sqlite_version}
 | 
						|
        Connected to {db_name}
 | 
						|
 | 
						|
        Each command will be run using execute() on the cursor.
 | 
						|
        Type ".help" for more information; type ".quit" or {eofkey} to quit.
 | 
						|
    """).strip()
 | 
						|
 | 
						|
    theme = get_theme()
 | 
						|
    s = theme.syntax
 | 
						|
 | 
						|
    sys.ps1 = f"{s.prompt}sqlite> {s.reset}"
 | 
						|
    sys.ps2 = f"{s.prompt}    ... {s.reset}"
 | 
						|
 | 
						|
    con = sqlite3.connect(args.filename, isolation_level=None)
 | 
						|
    try:
 | 
						|
        if args.sql:
 | 
						|
            # SQL statement provided on the command-line; execute it directly.
 | 
						|
            execute(con, args.sql, suppress_errors=False, theme=theme)
 | 
						|
        else:
 | 
						|
            # No SQL provided; start the REPL.
 | 
						|
            with completer():
 | 
						|
                console = SqliteInteractiveConsole(con, use_color=True)
 | 
						|
                console.interact(banner, exitmsg="")
 | 
						|
    finally:
 | 
						|
        con.close()
 | 
						|
 | 
						|
    sys.exit(0)
 | 
						|
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    main(sys.argv[1:])
 |