# encoding: utf8 from __future__ import print_function import argparse import os import sys import pokedex.cli.search import pokedex.db import pokedex.db.load import pokedex.db.tables import pokedex.lookup from pokedex import defaults def main(junk, *argv): parser = create_parser() if len(argv) <= 0: parser.print_help() sys.exit() args = parser.parse_args(argv) args.func(parser, args) def setuptools_entry(): main(*sys.argv) def create_parser(): """Build and return an ArgumentParser. """ # Slightly clumsy workaround to make both `setup -v` and `-v setup` work common_parser = argparse.ArgumentParser(add_help=False) common_parser.add_argument( '-e', '--engine', dest='engine_uri', default=None, help=u'By default, all commands try to use a SQLite database ' u'in the pokedex install directory. Use this option (or ' u'a POKEDEX_DB_ENGINE environment variable) to specify an ' u'alternate database.', ) common_parser.add_argument( '-i', '--index', dest='index_dir', default=None, help=u'By default, all commands try to put the lookup index in ' u'the pokedex install directory. Use this option (or a ' u'POKEDEX_INDEX_DIR environment variable) to specify an ' u'alternate loction.', ) common_parser.add_argument( '-q', '--quiet', dest='verbose', action='store_false', help=u'Don\'t print system output. This is the default for ' 'non-system commands and setup.', ) common_parser.add_argument( '-v', '--verbose', dest='verbose', default=False, action='store_true', help=u'Print system output. This is the default for system ' u'commands, except setup.', ) parser = argparse.ArgumentParser( prog='pokedex', description=u'A command-line Pokédex interface', parents=[common_parser], ) cmds = parser.add_subparsers(title='commands', metavar='', help='commands') cmd_help = cmds.add_parser( 'help', help=u'Display this message', parents=[common_parser]) cmd_help.set_defaults(func=command_help) cmd_lookup = cmds.add_parser( 'lookup', help=u'Look up something in the Pokédex', parents=[common_parser]) cmd_lookup.set_defaults(func=command_lookup) cmd_lookup.add_argument('criteria', nargs='+') cmd_search = cmds.add_parser( 'search', help=u'Find things by various criteria', parents=[common_parser]) pokedex.cli.search.configure_parser(cmd_search) cmd_load = cmds.add_parser( 'load', help=u'Load Pokédex data into a database from CSV files', parents=[common_parser]) cmd_load.set_defaults(func=command_load, verbose=True) # TODO get the actual default here cmd_load.add_argument( '-d', '--directory', dest='directory', default=None, help="directory containing the CSV files to load") cmd_load.add_argument( '-D', '--drop-tables', dest='drop_tables', default=False, action='store_true', help="drop all tables before loading data") cmd_load.add_argument( '-r', '--recursive', dest='recursive', default=False, action='store_true', help="load and drop all dependent tables (default is to use exactly the given list)") cmd_load.add_argument( '-S', '--safe', dest='safe', default=False, action='store_true', help="disable database-specific optimizations, such as Postgres's COPY FROM") # TODO need a custom handler for splittin' all of these cmd_load.add_argument( '-l', '--langs', dest='langs', default=None, help="comma-separated list of language codes to load, or 'none' (default: all)") cmd_load.add_argument( 'tables', nargs='*', help="list of database tables to load (default: all)") cmd_dump = cmds.add_parser( 'dump', help=u'Dump Pokédex data from a database into CSV files', parents=[common_parser]) cmd_dump.set_defaults(func=command_dump, verbose=True) cmd_dump.add_argument( '-d', '--directory', dest='directory', default=None, help="directory to place the dumped CSV files") cmd_dump.add_argument( '-l', '--langs', dest='langs', default=None, help=u"comma-separated list of language codes to load, 'none', or 'all' (default: all)") cmd_dump.add_argument( 'tables', nargs='*', help="list of database tables to load (default: all)") cmd_reindex = cmds.add_parser( 'reindex', help=u'Rebuild the lookup index from the database', parents=[common_parser]) cmd_reindex.set_defaults(func=command_reindex, verbose=True) cmd_setup = cmds.add_parser( 'setup', help=u'Combine load and reindex', parents=[common_parser]) cmd_setup.set_defaults(func=command_setup, verbose=False) cmd_status = cmds.add_parser( 'status', help=u'Print which engine, index, and csv directory would be used for other commands', parents=[common_parser]) cmd_status.set_defaults(func=command_status, verbose=True) return parser def get_session(args): """Given a parsed options object, connects to the database and returns a session. """ engine_uri = args.engine_uri got_from = 'command line' if engine_uri is None: engine_uri, got_from = defaults.get_default_db_uri_with_origin() session = pokedex.db.connect(engine_uri) if args.verbose: print("Connected to database %(engine)s (from %(got_from)s)" % dict(engine=session.bind.url, got_from=got_from)) return session def get_lookup(args, session=None, recreate=False): """Given a parsed options object, opens the whoosh index and returns a PokedexLookup object. """ if recreate and not session: raise ValueError("get_lookup() needs an explicit session to regen the index") index_dir = args.index_dir got_from = 'command line' if index_dir is None: index_dir, got_from = defaults.get_default_index_dir_with_origin() if args.verbose: print("Opened lookup index %(index_dir)s (from %(got_from)s)" % dict(index_dir=index_dir, got_from=got_from)) lookup = pokedex.lookup.PokedexLookup(index_dir, session=session) if recreate: lookup.rebuild_index() return lookup def get_csv_directory(args): """Prints and returns the csv directory we're about to use.""" if not args.verbose: return csvdir = args.directory got_from = 'command line' if csvdir is None: csvdir, got_from = defaults.get_default_csv_dir_with_origin() print("Using CSV directory %(csvdir)s (from %(got_from)s)" % dict(csvdir=csvdir, got_from=got_from)) return csvdir ### Plumbing commands def command_dump(parser, args): session = get_session(args) get_csv_directory(args) if args.langs is not None: langs = [l.strip() for l in args.langs.split(',')] # If the langs code is only 'all' then langs is None so that all the tables get dumped. if len(langs) == 1 and langs[0] == 'all': langs = None # Check if either 'all' or 'none' codes are used along side other codes. # If either code is used, an error message will be displayed and the program will close. elif len(langs) > 1 and 'all' in langs: print("\nERROR: The 'all' code should be used by itself.") return elif len(langs) > 1 and 'none' in langs: print("\nERROR: The 'none' code should be used by itself.") return else: langs = None pokedex.db.load.dump( session, directory=args.directory, tables=args.tables, verbose=args.verbose, langs=langs, ) def command_load(parser, args): if not args.engine_uri: print("WARNING: You're reloading the default database, but not the lookup index. They") print(" might get out of sync, and pokedex commands may not work correctly!") print("To fix this, run `pokedex reindex` when this command finishes. Or, just use") print("`pokedex setup` to do both at once.") print() if args.langs == 'none': langs = [] elif args.langs is None: langs = None else: langs = [l.strip() for l in args.langs.split(',')] session = get_session(args) get_csv_directory(args) pokedex.db.load.load( session, directory=args.directory, drop_tables=args.drop_tables, tables=args.tables, verbose=args.verbose, safe=args.safe, recursive=args.recursive, langs=langs, ) def command_reindex(parser, args): session = get_session(args) get_lookup(args, session=session, recreate=True) print("Recreated lookup index.") def command_setup(parser, args): args.directory = None session = get_session(args) get_csv_directory(args) pokedex.db.load.load( session, directory=None, drop_tables=True, verbose=args.verbose, safe=False) get_lookup(args, session=session, recreate=True) print("Recreated lookup index.") def command_status(parser, args): args.directory = None # Database, and a lame check for whether it's been inited at least once session = get_session(args) print(" - OK! Connected successfully.") if pokedex.db.tables.Pokemon.__table__.exists(session.bind): print(" - OK! Database seems to contain some data.") else: print(" - WARNING: Database appears to be empty.") # CSV; simple checks that the dir exists csvdir = get_csv_directory(args) if not os.path.exists(csvdir): print(" - ERROR: No such directory!") elif not os.path.isdir(csvdir): print(" - ERROR: Not a directory!") else: print(" - OK! Directory exists.") if os.access(csvdir, os.R_OK): print(" - OK! Can read from directory.") else: print(" - ERROR: Can't read from directory!") if os.access(csvdir, os.W_OK): print(" - OK! Can write to directory.") else: print(" - WARNING: Can't write to directory! " "`dump` will not work. You may need to sudo.") # Index; the PokedexLookup constructor covers most tests and will # cheerfully bomb if they fail get_lookup(args, recreate=False) print(" - OK! Opened successfully.") ### User-facing commands def command_lookup(parser, args): name = u' '.join(args.criteria) session = get_session(args) lookup = get_lookup(args, session=session, recreate=False) results = lookup.lookup(name) if not results: print("No matches.") elif results[0].exact: print("Matched:") else: print("Fuzzy-matched:") for result in results: if hasattr(result.object, 'full_name'): name = result.object.full_name else: name = result.object.name print("%s: %s" % (result.object.__tablename__, name), end='') if result.language: print("(%s in %s)" % (result.name, result.language)) else: print() def command_help(parser, args): parser.print_help() if __name__ == '__main__': main(*sys.argv)