You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					899 lines
				
				29 KiB
			
		
		
			
		
	
	
					899 lines
				
				29 KiB
			| 
											8 years ago
										 | # -*- coding: utf-8 -*-
 | ||
|  | """
 | ||
|  |     flask.cli
 | ||
|  |     ~~~~~~~~~
 | ||
|  | 
 | ||
|  |     A simple command line application to run flask apps.
 | ||
|  | 
 | ||
|  |     :copyright: © 2010 by the Pallets team.
 | ||
|  |     :license: BSD, see LICENSE for more details.
 | ||
|  | """
 | ||
|  | 
 | ||
|  | from __future__ import print_function
 | ||
|  | 
 | ||
|  | import ast
 | ||
|  | import inspect
 | ||
|  | import os
 | ||
|  | import re
 | ||
|  | import ssl
 | ||
|  | import sys
 | ||
|  | import traceback
 | ||
|  | from functools import update_wrapper
 | ||
|  | from operator import attrgetter
 | ||
|  | from threading import Lock, Thread
 | ||
|  | 
 | ||
|  | import click
 | ||
|  | from werkzeug.utils import import_string
 | ||
|  | 
 | ||
|  | from . import __version__
 | ||
|  | from ._compat import getargspec, iteritems, reraise, text_type
 | ||
|  | from .globals import current_app
 | ||
|  | from .helpers import get_debug_flag, get_env, get_load_dotenv
 | ||
|  | 
 | ||
|  | try:
 | ||
|  |     import dotenv
 | ||
|  | except ImportError:
 | ||
|  |     dotenv = None
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class NoAppException(click.UsageError):
 | ||
|  |     """Raised if an application cannot be found or loaded."""
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def find_best_app(script_info, module):
 | ||
|  |     """Given a module instance this tries to find the best possible
 | ||
|  |     application in the module or raises an exception.
 | ||
|  |     """
 | ||
|  |     from . import Flask
 | ||
|  | 
 | ||
|  |     # Search for the most common names first.
 | ||
|  |     for attr_name in ('app', 'application'):
 | ||
|  |         app = getattr(module, attr_name, None)
 | ||
|  | 
 | ||
|  |         if isinstance(app, Flask):
 | ||
|  |             return app
 | ||
|  | 
 | ||
|  |     # Otherwise find the only object that is a Flask instance.
 | ||
|  |     matches = [
 | ||
|  |         v for k, v in iteritems(module.__dict__) if isinstance(v, Flask)
 | ||
|  |     ]
 | ||
|  | 
 | ||
|  |     if len(matches) == 1:
 | ||
|  |         return matches[0]
 | ||
|  |     elif len(matches) > 1:
 | ||
|  |         raise NoAppException(
 | ||
|  |             'Detected multiple Flask applications in module "{module}". Use '
 | ||
|  |             '"FLASK_APP={module}:name" to specify the correct '
 | ||
|  |             'one.'.format(module=module.__name__)
 | ||
|  |         )
 | ||
|  | 
 | ||
|  |     # Search for app factory functions.
 | ||
|  |     for attr_name in ('create_app', 'make_app'):
 | ||
|  |         app_factory = getattr(module, attr_name, None)
 | ||
|  | 
 | ||
|  |         if inspect.isfunction(app_factory):
 | ||
|  |             try:
 | ||
|  |                 app = call_factory(script_info, app_factory)
 | ||
|  | 
 | ||
|  |                 if isinstance(app, Flask):
 | ||
|  |                     return app
 | ||
|  |             except TypeError:
 | ||
|  |                 if not _called_with_wrong_args(app_factory):
 | ||
|  |                     raise
 | ||
|  |                 raise NoAppException(
 | ||
|  |                     'Detected factory "{factory}" in module "{module}", but '
 | ||
|  |                     'could not call it without arguments. Use '
 | ||
|  |                     '"FLASK_APP=\'{module}:{factory}(args)\'" to specify '
 | ||
|  |                     'arguments.'.format(
 | ||
|  |                         factory=attr_name, module=module.__name__
 | ||
|  |                     )
 | ||
|  |                 )
 | ||
|  | 
 | ||
|  |     raise NoAppException(
 | ||
|  |         'Failed to find Flask application or factory in module "{module}". '
 | ||
|  |         'Use "FLASK_APP={module}:name to specify one.'.format(
 | ||
|  |             module=module.__name__
 | ||
|  |         )
 | ||
|  |     )
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def call_factory(script_info, app_factory, arguments=()):
 | ||
|  |     """Takes an app factory, a ``script_info` object and  optionally a tuple
 | ||
|  |     of arguments. Checks for the existence of a script_info argument and calls
 | ||
|  |     the app_factory depending on that and the arguments provided.
 | ||
|  |     """
 | ||
|  |     args_spec = getargspec(app_factory)
 | ||
|  |     arg_names = args_spec.args
 | ||
|  |     arg_defaults = args_spec.defaults
 | ||
|  | 
 | ||
|  |     if 'script_info' in arg_names:
 | ||
|  |         return app_factory(*arguments, script_info=script_info)
 | ||
|  |     elif arguments:
 | ||
|  |         return app_factory(*arguments)
 | ||
|  |     elif not arguments and len(arg_names) == 1 and arg_defaults is None:
 | ||
|  |         return app_factory(script_info)
 | ||
|  | 
 | ||
|  |     return app_factory()
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def _called_with_wrong_args(factory):
 | ||
|  |     """Check whether calling a function raised a ``TypeError`` because
 | ||
|  |     the call failed or because something in the factory raised the
 | ||
|  |     error.
 | ||
|  | 
 | ||
|  |     :param factory: the factory function that was called
 | ||
|  |     :return: true if the call failed
 | ||
|  |     """
 | ||
|  |     tb = sys.exc_info()[2]
 | ||
|  | 
 | ||
|  |     try:
 | ||
|  |         while tb is not None:
 | ||
|  |             if tb.tb_frame.f_code is factory.__code__:
 | ||
|  |                 # in the factory, it was called successfully
 | ||
|  |                 return False
 | ||
|  | 
 | ||
|  |             tb = tb.tb_next
 | ||
|  | 
 | ||
|  |         # didn't reach the factory
 | ||
|  |         return True
 | ||
|  |     finally:
 | ||
|  |         del tb
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def find_app_by_string(script_info, module, app_name):
 | ||
|  |     """Checks if the given string is a variable name or a function. If it is a
 | ||
|  |     function, it checks for specified arguments and whether it takes a
 | ||
|  |     ``script_info`` argument and calls the function with the appropriate
 | ||
|  |     arguments.
 | ||
|  |     """
 | ||
|  |     from flask import Flask
 | ||
|  |     match = re.match(r'^ *([^ ()]+) *(?:\((.*?) *,? *\))? *$', app_name)
 | ||
|  | 
 | ||
|  |     if not match:
 | ||
|  |         raise NoAppException(
 | ||
|  |             '"{name}" is not a valid variable name or function '
 | ||
|  |             'expression.'.format(name=app_name)
 | ||
|  |         )
 | ||
|  | 
 | ||
|  |     name, args = match.groups()
 | ||
|  | 
 | ||
|  |     try:
 | ||
|  |         attr = getattr(module, name)
 | ||
|  |     except AttributeError as e:
 | ||
|  |         raise NoAppException(e.args[0])
 | ||
|  | 
 | ||
|  |     if inspect.isfunction(attr):
 | ||
|  |         if args:
 | ||
|  |             try:
 | ||
|  |                 args = ast.literal_eval('({args},)'.format(args=args))
 | ||
|  |             except (ValueError, SyntaxError)as e:
 | ||
|  |                 raise NoAppException(
 | ||
|  |                     'Could not parse the arguments in '
 | ||
|  |                     '"{app_name}".'.format(e=e, app_name=app_name)
 | ||
|  |                 )
 | ||
|  |         else:
 | ||
|  |             args = ()
 | ||
|  | 
 | ||
|  |         try:
 | ||
|  |             app = call_factory(script_info, attr, args)
 | ||
|  |         except TypeError as e:
 | ||
|  |             if not _called_with_wrong_args(attr):
 | ||
|  |                 raise
 | ||
|  | 
 | ||
|  |             raise NoAppException(
 | ||
|  |                 '{e}\nThe factory "{app_name}" in module "{module}" could not '
 | ||
|  |                 'be called with the specified arguments.'.format(
 | ||
|  |                     e=e, app_name=app_name, module=module.__name__
 | ||
|  |                 )
 | ||
|  |             )
 | ||
|  |     else:
 | ||
|  |         app = attr
 | ||
|  | 
 | ||
|  |     if isinstance(app, Flask):
 | ||
|  |         return app
 | ||
|  | 
 | ||
|  |     raise NoAppException(
 | ||
|  |         'A valid Flask application was not obtained from '
 | ||
|  |         '"{module}:{app_name}".'.format(
 | ||
|  |             module=module.__name__, app_name=app_name
 | ||
|  |         )
 | ||
|  |     )
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def prepare_import(path):
 | ||
|  |     """Given a filename this will try to calculate the python path, add it
 | ||
|  |     to the search path and return the actual module name that is expected.
 | ||
|  |     """
 | ||
|  |     path = os.path.realpath(path)
 | ||
|  | 
 | ||
|  |     if os.path.splitext(path)[1] == '.py':
 | ||
|  |         path = os.path.splitext(path)[0]
 | ||
|  | 
 | ||
|  |     if os.path.basename(path) == '__init__':
 | ||
|  |         path = os.path.dirname(path)
 | ||
|  | 
 | ||
|  |     module_name = []
 | ||
|  | 
 | ||
|  |     # move up until outside package structure (no __init__.py)
 | ||
|  |     while True:
 | ||
|  |         path, name = os.path.split(path)
 | ||
|  |         module_name.append(name)
 | ||
|  | 
 | ||
|  |         if not os.path.exists(os.path.join(path, '__init__.py')):
 | ||
|  |             break
 | ||
|  | 
 | ||
|  |     if sys.path[0] != path:
 | ||
|  |         sys.path.insert(0, path)
 | ||
|  | 
 | ||
|  |     return '.'.join(module_name[::-1])
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def locate_app(script_info, module_name, app_name, raise_if_not_found=True):
 | ||
|  |     __traceback_hide__ = True
 | ||
|  | 
 | ||
|  |     try:
 | ||
|  |         __import__(module_name)
 | ||
|  |     except ImportError:
 | ||
|  |         # Reraise the ImportError if it occurred within the imported module.
 | ||
|  |         # Determine this by checking whether the trace has a depth > 1.
 | ||
|  |         if sys.exc_info()[-1].tb_next:
 | ||
|  |             raise NoAppException(
 | ||
|  |                 'While importing "{name}", an ImportError was raised:'
 | ||
|  |                 '\n\n{tb}'.format(name=module_name, tb=traceback.format_exc())
 | ||
|  |             )
 | ||
|  |         elif raise_if_not_found:
 | ||
|  |             raise NoAppException(
 | ||
|  |                 'Could not import "{name}".'.format(name=module_name)
 | ||
|  |             )
 | ||
|  |         else:
 | ||
|  |             return
 | ||
|  | 
 | ||
|  |     module = sys.modules[module_name]
 | ||
|  | 
 | ||
|  |     if app_name is None:
 | ||
|  |         return find_best_app(script_info, module)
 | ||
|  |     else:
 | ||
|  |         return find_app_by_string(script_info, module, app_name)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def get_version(ctx, param, value):
 | ||
|  |     if not value or ctx.resilient_parsing:
 | ||
|  |         return
 | ||
|  |     message = 'Flask %(version)s\nPython %(python_version)s'
 | ||
|  |     click.echo(message % {
 | ||
|  |         'version': __version__,
 | ||
|  |         'python_version': sys.version,
 | ||
|  |     }, color=ctx.color)
 | ||
|  |     ctx.exit()
 | ||
|  | 
 | ||
|  | 
 | ||
|  | version_option = click.Option(
 | ||
|  |     ['--version'],
 | ||
|  |     help='Show the flask version',
 | ||
|  |     expose_value=False,
 | ||
|  |     callback=get_version,
 | ||
|  |     is_flag=True,
 | ||
|  |     is_eager=True
 | ||
|  | )
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class DispatchingApp(object):
 | ||
|  |     """Special application that dispatches to a Flask application which
 | ||
|  |     is imported by name in a background thread.  If an error happens
 | ||
|  |     it is recorded and shown as part of the WSGI handling which in case
 | ||
|  |     of the Werkzeug debugger means that it shows up in the browser.
 | ||
|  |     """
 | ||
|  | 
 | ||
|  |     def __init__(self, loader, use_eager_loading=False):
 | ||
|  |         self.loader = loader
 | ||
|  |         self._app = None
 | ||
|  |         self._lock = Lock()
 | ||
|  |         self._bg_loading_exc_info = None
 | ||
|  |         if use_eager_loading:
 | ||
|  |             self._load_unlocked()
 | ||
|  |         else:
 | ||
|  |             self._load_in_background()
 | ||
|  | 
 | ||
|  |     def _load_in_background(self):
 | ||
|  |         def _load_app():
 | ||
|  |             __traceback_hide__ = True
 | ||
|  |             with self._lock:
 | ||
|  |                 try:
 | ||
|  |                     self._load_unlocked()
 | ||
|  |                 except Exception:
 | ||
|  |                     self._bg_loading_exc_info = sys.exc_info()
 | ||
|  |         t = Thread(target=_load_app, args=())
 | ||
|  |         t.start()
 | ||
|  | 
 | ||
|  |     def _flush_bg_loading_exception(self):
 | ||
|  |         __traceback_hide__ = True
 | ||
|  |         exc_info = self._bg_loading_exc_info
 | ||
|  |         if exc_info is not None:
 | ||
|  |             self._bg_loading_exc_info = None
 | ||
|  |             reraise(*exc_info)
 | ||
|  | 
 | ||
|  |     def _load_unlocked(self):
 | ||
|  |         __traceback_hide__ = True
 | ||
|  |         self._app = rv = self.loader()
 | ||
|  |         self._bg_loading_exc_info = None
 | ||
|  |         return rv
 | ||
|  | 
 | ||
|  |     def __call__(self, environ, start_response):
 | ||
|  |         __traceback_hide__ = True
 | ||
|  |         if self._app is not None:
 | ||
|  |             return self._app(environ, start_response)
 | ||
|  |         self._flush_bg_loading_exception()
 | ||
|  |         with self._lock:
 | ||
|  |             if self._app is not None:
 | ||
|  |                 rv = self._app
 | ||
|  |             else:
 | ||
|  |                 rv = self._load_unlocked()
 | ||
|  |             return rv(environ, start_response)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class ScriptInfo(object):
 | ||
|  |     """Help object to deal with Flask applications.  This is usually not
 | ||
|  |     necessary to interface with as it's used internally in the dispatching
 | ||
|  |     to click.  In future versions of Flask this object will most likely play
 | ||
|  |     a bigger role.  Typically it's created automatically by the
 | ||
|  |     :class:`FlaskGroup` but you can also manually create it and pass it
 | ||
|  |     onwards as click object.
 | ||
|  |     """
 | ||
|  | 
 | ||
|  |     def __init__(self, app_import_path=None, create_app=None):
 | ||
|  |         #: Optionally the import path for the Flask application.
 | ||
|  |         self.app_import_path = app_import_path or os.environ.get('FLASK_APP')
 | ||
|  |         #: Optionally a function that is passed the script info to create
 | ||
|  |         #: the instance of the application.
 | ||
|  |         self.create_app = create_app
 | ||
|  |         #: A dictionary with arbitrary data that can be associated with
 | ||
|  |         #: this script info.
 | ||
|  |         self.data = {}
 | ||
|  |         self._loaded_app = None
 | ||
|  | 
 | ||
|  |     def load_app(self):
 | ||
|  |         """Loads the Flask app (if not yet loaded) and returns it.  Calling
 | ||
|  |         this multiple times will just result in the already loaded app to
 | ||
|  |         be returned.
 | ||
|  |         """
 | ||
|  |         __traceback_hide__ = True
 | ||
|  | 
 | ||
|  |         if self._loaded_app is not None:
 | ||
|  |             return self._loaded_app
 | ||
|  | 
 | ||
|  |         app = None
 | ||
|  | 
 | ||
|  |         if self.create_app is not None:
 | ||
|  |             app = call_factory(self, self.create_app)
 | ||
|  |         else:
 | ||
|  |             if self.app_import_path:
 | ||
|  |                 path, name = (self.app_import_path.split(':', 1) + [None])[:2]
 | ||
|  |                 import_name = prepare_import(path)
 | ||
|  |                 app = locate_app(self, import_name, name)
 | ||
|  |             else:
 | ||
|  |                 for path in ('wsgi.py', 'app.py'):
 | ||
|  |                     import_name = prepare_import(path)
 | ||
|  |                     app = locate_app(self, import_name, None,
 | ||
|  |                                      raise_if_not_found=False)
 | ||
|  | 
 | ||
|  |                     if app:
 | ||
|  |                         break
 | ||
|  | 
 | ||
|  |         if not app:
 | ||
|  |             raise NoAppException(
 | ||
|  |                 'Could not locate a Flask application. You did not provide '
 | ||
|  |                 'the "FLASK_APP" environment variable, and a "wsgi.py" or '
 | ||
|  |                 '"app.py" module was not found in the current directory.'
 | ||
|  |             )
 | ||
|  | 
 | ||
|  |         debug = get_debug_flag()
 | ||
|  | 
 | ||
|  |         # Update the app's debug flag through the descriptor so that other
 | ||
|  |         # values repopulate as well.
 | ||
|  |         if debug is not None:
 | ||
|  |             app.debug = debug
 | ||
|  | 
 | ||
|  |         self._loaded_app = app
 | ||
|  |         return app
 | ||
|  | 
 | ||
|  | 
 | ||
|  | pass_script_info = click.make_pass_decorator(ScriptInfo, ensure=True)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def with_appcontext(f):
 | ||
|  |     """Wraps a callback so that it's guaranteed to be executed with the
 | ||
|  |     script's application context.  If callbacks are registered directly
 | ||
|  |     to the ``app.cli`` object then they are wrapped with this function
 | ||
|  |     by default unless it's disabled.
 | ||
|  |     """
 | ||
|  |     @click.pass_context
 | ||
|  |     def decorator(__ctx, *args, **kwargs):
 | ||
|  |         with __ctx.ensure_object(ScriptInfo).load_app().app_context():
 | ||
|  |             return __ctx.invoke(f, *args, **kwargs)
 | ||
|  |     return update_wrapper(decorator, f)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class AppGroup(click.Group):
 | ||
|  |     """This works similar to a regular click :class:`~click.Group` but it
 | ||
|  |     changes the behavior of the :meth:`command` decorator so that it
 | ||
|  |     automatically wraps the functions in :func:`with_appcontext`.
 | ||
|  | 
 | ||
|  |     Not to be confused with :class:`FlaskGroup`.
 | ||
|  |     """
 | ||
|  | 
 | ||
|  |     def command(self, *args, **kwargs):
 | ||
|  |         """This works exactly like the method of the same name on a regular
 | ||
|  |         :class:`click.Group` but it wraps callbacks in :func:`with_appcontext`
 | ||
|  |         unless it's disabled by passing ``with_appcontext=False``.
 | ||
|  |         """
 | ||
|  |         wrap_for_ctx = kwargs.pop('with_appcontext', True)
 | ||
|  |         def decorator(f):
 | ||
|  |             if wrap_for_ctx:
 | ||
|  |                 f = with_appcontext(f)
 | ||
|  |             return click.Group.command(self, *args, **kwargs)(f)
 | ||
|  |         return decorator
 | ||
|  | 
 | ||
|  |     def group(self, *args, **kwargs):
 | ||
|  |         """This works exactly like the method of the same name on a regular
 | ||
|  |         :class:`click.Group` but it defaults the group class to
 | ||
|  |         :class:`AppGroup`.
 | ||
|  |         """
 | ||
|  |         kwargs.setdefault('cls', AppGroup)
 | ||
|  |         return click.Group.group(self, *args, **kwargs)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class FlaskGroup(AppGroup):
 | ||
|  |     """Special subclass of the :class:`AppGroup` group that supports
 | ||
|  |     loading more commands from the configured Flask app.  Normally a
 | ||
|  |     developer does not have to interface with this class but there are
 | ||
|  |     some very advanced use cases for which it makes sense to create an
 | ||
|  |     instance of this.
 | ||
|  | 
 | ||
|  |     For information as of why this is useful see :ref:`custom-scripts`.
 | ||
|  | 
 | ||
|  |     :param add_default_commands: if this is True then the default run and
 | ||
|  |         shell commands wil be added.
 | ||
|  |     :param add_version_option: adds the ``--version`` option.
 | ||
|  |     :param create_app: an optional callback that is passed the script info and
 | ||
|  |         returns the loaded app.
 | ||
|  |     :param load_dotenv: Load the nearest :file:`.env` and :file:`.flaskenv`
 | ||
|  |         files to set environment variables. Will also change the working
 | ||
|  |         directory to the directory containing the first file found.
 | ||
|  | 
 | ||
|  |     .. versionchanged:: 1.0
 | ||
|  |         If installed, python-dotenv will be used to load environment variables
 | ||
|  |         from :file:`.env` and :file:`.flaskenv` files.
 | ||
|  |     """
 | ||
|  | 
 | ||
|  |     def __init__(self, add_default_commands=True, create_app=None,
 | ||
|  |                  add_version_option=True, load_dotenv=True, **extra):
 | ||
|  |         params = list(extra.pop('params', None) or ())
 | ||
|  | 
 | ||
|  |         if add_version_option:
 | ||
|  |             params.append(version_option)
 | ||
|  | 
 | ||
|  |         AppGroup.__init__(self, params=params, **extra)
 | ||
|  |         self.create_app = create_app
 | ||
|  |         self.load_dotenv = load_dotenv
 | ||
|  | 
 | ||
|  |         if add_default_commands:
 | ||
|  |             self.add_command(run_command)
 | ||
|  |             self.add_command(shell_command)
 | ||
|  |             self.add_command(routes_command)
 | ||
|  | 
 | ||
|  |         self._loaded_plugin_commands = False
 | ||
|  | 
 | ||
|  |     def _load_plugin_commands(self):
 | ||
|  |         if self._loaded_plugin_commands:
 | ||
|  |             return
 | ||
|  |         try:
 | ||
|  |             import pkg_resources
 | ||
|  |         except ImportError:
 | ||
|  |             self._loaded_plugin_commands = True
 | ||
|  |             return
 | ||
|  | 
 | ||
|  |         for ep in pkg_resources.iter_entry_points('flask.commands'):
 | ||
|  |             self.add_command(ep.load(), ep.name)
 | ||
|  |         self._loaded_plugin_commands = True
 | ||
|  | 
 | ||
|  |     def get_command(self, ctx, name):
 | ||
|  |         self._load_plugin_commands()
 | ||
|  | 
 | ||
|  |         # We load built-in commands first as these should always be the
 | ||
|  |         # same no matter what the app does.  If the app does want to
 | ||
|  |         # override this it needs to make a custom instance of this group
 | ||
|  |         # and not attach the default commands.
 | ||
|  |         #
 | ||
|  |         # This also means that the script stays functional in case the
 | ||
|  |         # application completely fails.
 | ||
|  |         rv = AppGroup.get_command(self, ctx, name)
 | ||
|  |         if rv is not None:
 | ||
|  |             return rv
 | ||
|  | 
 | ||
|  |         info = ctx.ensure_object(ScriptInfo)
 | ||
|  |         try:
 | ||
|  |             rv = info.load_app().cli.get_command(ctx, name)
 | ||
|  |             if rv is not None:
 | ||
|  |                 return rv
 | ||
|  |         except NoAppException:
 | ||
|  |             pass
 | ||
|  | 
 | ||
|  |     def list_commands(self, ctx):
 | ||
|  |         self._load_plugin_commands()
 | ||
|  | 
 | ||
|  |         # The commands available is the list of both the application (if
 | ||
|  |         # available) plus the builtin commands.
 | ||
|  |         rv = set(click.Group.list_commands(self, ctx))
 | ||
|  |         info = ctx.ensure_object(ScriptInfo)
 | ||
|  |         try:
 | ||
|  |             rv.update(info.load_app().cli.list_commands(ctx))
 | ||
|  |         except Exception:
 | ||
|  |             # Here we intentionally swallow all exceptions as we don't
 | ||
|  |             # want the help page to break if the app does not exist.
 | ||
|  |             # If someone attempts to use the command we try to create
 | ||
|  |             # the app again and this will give us the error.
 | ||
|  |             # However, we will not do so silently because that would confuse
 | ||
|  |             # users.
 | ||
|  |             traceback.print_exc()
 | ||
|  |         return sorted(rv)
 | ||
|  | 
 | ||
|  |     def main(self, *args, **kwargs):
 | ||
|  |         # Set a global flag that indicates that we were invoked from the
 | ||
|  |         # command line interface. This is detected by Flask.run to make the
 | ||
|  |         # call into a no-op. This is necessary to avoid ugly errors when the
 | ||
|  |         # script that is loaded here also attempts to start a server.
 | ||
|  |         os.environ['FLASK_RUN_FROM_CLI'] = 'true'
 | ||
|  | 
 | ||
|  |         if get_load_dotenv(self.load_dotenv):
 | ||
|  |             load_dotenv()
 | ||
|  | 
 | ||
|  |         obj = kwargs.get('obj')
 | ||
|  | 
 | ||
|  |         if obj is None:
 | ||
|  |             obj = ScriptInfo(create_app=self.create_app)
 | ||
|  | 
 | ||
|  |         kwargs['obj'] = obj
 | ||
|  |         kwargs.setdefault('auto_envvar_prefix', 'FLASK')
 | ||
|  |         return super(FlaskGroup, self).main(*args, **kwargs)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def _path_is_ancestor(path, other):
 | ||
|  |     """Take ``other`` and remove the length of ``path`` from it. Then join it
 | ||
|  |     to ``path``. If it is the original value, ``path`` is an ancestor of
 | ||
|  |     ``other``."""
 | ||
|  |     return os.path.join(path, other[len(path):].lstrip(os.sep)) == other
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def load_dotenv(path=None):
 | ||
|  |     """Load "dotenv" files in order of precedence to set environment variables.
 | ||
|  | 
 | ||
|  |     If an env var is already set it is not overwritten, so earlier files in the
 | ||
|  |     list are preferred over later files.
 | ||
|  | 
 | ||
|  |     Changes the current working directory to the location of the first file
 | ||
|  |     found, with the assumption that it is in the top level project directory
 | ||
|  |     and will be where the Python path should import local packages from.
 | ||
|  | 
 | ||
|  |     This is a no-op if `python-dotenv`_ is not installed.
 | ||
|  | 
 | ||
|  |     .. _python-dotenv: https://github.com/theskumar/python-dotenv#readme
 | ||
|  | 
 | ||
|  |     :param path: Load the file at this location instead of searching.
 | ||
|  |     :return: ``True`` if a file was loaded.
 | ||
|  | 
 | ||
|  |     .. versionadded:: 1.0
 | ||
|  |     """
 | ||
|  |     if dotenv is None:
 | ||
|  |         if path or os.path.exists('.env') or os.path.exists('.flaskenv'):
 | ||
|  |             click.secho(
 | ||
|  |                 ' * Tip: There are .env files present.'
 | ||
|  |                 ' Do "pip install python-dotenv" to use them.',
 | ||
|  |                 fg='yellow')
 | ||
|  |         return
 | ||
|  | 
 | ||
|  |     if path is not None:
 | ||
|  |         return dotenv.load_dotenv(path)
 | ||
|  | 
 | ||
|  |     new_dir = None
 | ||
|  | 
 | ||
|  |     for name in ('.env', '.flaskenv'):
 | ||
|  |         path = dotenv.find_dotenv(name, usecwd=True)
 | ||
|  | 
 | ||
|  |         if not path:
 | ||
|  |             continue
 | ||
|  | 
 | ||
|  |         if new_dir is None:
 | ||
|  |             new_dir = os.path.dirname(path)
 | ||
|  | 
 | ||
|  |         dotenv.load_dotenv(path)
 | ||
|  | 
 | ||
|  |     if new_dir and os.getcwd() != new_dir:
 | ||
|  |         os.chdir(new_dir)
 | ||
|  | 
 | ||
|  |     return new_dir is not None  # at least one file was located and loaded
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def show_server_banner(env, debug, app_import_path, eager_loading):
 | ||
|  |     """Show extra startup messages the first time the server is run,
 | ||
|  |     ignoring the reloader.
 | ||
|  |     """
 | ||
|  |     if os.environ.get('WERKZEUG_RUN_MAIN') == 'true':
 | ||
|  |         return
 | ||
|  | 
 | ||
|  |     if app_import_path is not None:
 | ||
|  |         message = ' * Serving Flask app "{0}"'.format(app_import_path)
 | ||
|  | 
 | ||
|  |         if not eager_loading:
 | ||
|  |             message += ' (lazy loading)'
 | ||
|  | 
 | ||
|  |         click.echo(message)
 | ||
|  | 
 | ||
|  |     click.echo(' * Environment: {0}'.format(env))
 | ||
|  | 
 | ||
|  |     if env == 'production':
 | ||
|  |         click.secho(
 | ||
|  |             '   WARNING: Do not use the development server in a production'
 | ||
|  |             ' environment.', fg='red')
 | ||
|  |         click.secho('   Use a production WSGI server instead.', dim=True)
 | ||
|  | 
 | ||
|  |     if debug is not None:
 | ||
|  |         click.echo(' * Debug mode: {0}'.format('on' if debug else 'off'))
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class CertParamType(click.ParamType):
 | ||
|  |     """Click option type for the ``--cert`` option. Allows either an
 | ||
|  |     existing file, the string ``'adhoc'``, or an import for a
 | ||
|  |     :class:`~ssl.SSLContext` object.
 | ||
|  |     """
 | ||
|  | 
 | ||
|  |     name = 'path'
 | ||
|  | 
 | ||
|  |     def __init__(self):
 | ||
|  |         self.path_type = click.Path(
 | ||
|  |             exists=True, dir_okay=False, resolve_path=True)
 | ||
|  | 
 | ||
|  |     def convert(self, value, param, ctx):
 | ||
|  |         try:
 | ||
|  |             return self.path_type(value, param, ctx)
 | ||
|  |         except click.BadParameter:
 | ||
|  |             value = click.STRING(value, param, ctx).lower()
 | ||
|  | 
 | ||
|  |             if value == 'adhoc':
 | ||
|  |                 try:
 | ||
|  |                     import OpenSSL
 | ||
|  |                 except ImportError:
 | ||
|  |                     raise click.BadParameter(
 | ||
|  |                         'Using ad-hoc certificates requires pyOpenSSL.',
 | ||
|  |                         ctx, param)
 | ||
|  | 
 | ||
|  |                 return value
 | ||
|  | 
 | ||
|  |             obj = import_string(value, silent=True)
 | ||
|  | 
 | ||
|  |             if sys.version_info < (2, 7):
 | ||
|  |                 if obj:
 | ||
|  |                     return obj
 | ||
|  |             else:
 | ||
|  |                 if isinstance(obj, ssl.SSLContext):
 | ||
|  |                     return obj
 | ||
|  | 
 | ||
|  |             raise
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def _validate_key(ctx, param, value):
 | ||
|  |     """The ``--key`` option must be specified when ``--cert`` is a file.
 | ||
|  |     Modifies the ``cert`` param to be a ``(cert, key)`` pair if needed.
 | ||
|  |     """
 | ||
|  |     cert = ctx.params.get('cert')
 | ||
|  |     is_adhoc = cert == 'adhoc'
 | ||
|  | 
 | ||
|  |     if sys.version_info < (2, 7):
 | ||
|  |         is_context = cert and not isinstance(cert, (text_type, bytes))
 | ||
|  |     else:
 | ||
|  |         is_context = isinstance(cert, ssl.SSLContext)
 | ||
|  | 
 | ||
|  |     if value is not None:
 | ||
|  |         if is_adhoc:
 | ||
|  |             raise click.BadParameter(
 | ||
|  |                 'When "--cert" is "adhoc", "--key" is not used.',
 | ||
|  |                 ctx, param)
 | ||
|  | 
 | ||
|  |         if is_context:
 | ||
|  |             raise click.BadParameter(
 | ||
|  |                 'When "--cert" is an SSLContext object, "--key is not used.',
 | ||
|  |                 ctx, param)
 | ||
|  | 
 | ||
|  |         if not cert:
 | ||
|  |             raise click.BadParameter(
 | ||
|  |                 '"--cert" must also be specified.',
 | ||
|  |                 ctx, param)
 | ||
|  | 
 | ||
|  |         ctx.params['cert'] = cert, value
 | ||
|  | 
 | ||
|  |     else:
 | ||
|  |         if cert and not (is_adhoc or is_context):
 | ||
|  |             raise click.BadParameter(
 | ||
|  |                 'Required when using "--cert".',
 | ||
|  |                 ctx, param)
 | ||
|  | 
 | ||
|  |     return value
 | ||
|  | 
 | ||
|  | 
 | ||
|  | @click.command('run', short_help='Runs a development server.')
 | ||
|  | @click.option('--host', '-h', default='127.0.0.1',
 | ||
|  |               help='The interface to bind to.')
 | ||
|  | @click.option('--port', '-p', default=5000,
 | ||
|  |               help='The port to bind to.')
 | ||
|  | @click.option('--cert', type=CertParamType(),
 | ||
|  |               help='Specify a certificate file to use HTTPS.')
 | ||
|  | @click.option('--key',
 | ||
|  |               type=click.Path(exists=True, dir_okay=False, resolve_path=True),
 | ||
|  |               callback=_validate_key, expose_value=False,
 | ||
|  |               help='The key file to use when specifying a certificate.')
 | ||
|  | @click.option('--reload/--no-reload', default=None,
 | ||
|  |               help='Enable or disable the reloader. By default the reloader '
 | ||
|  |               'is active if debug is enabled.')
 | ||
|  | @click.option('--debugger/--no-debugger', default=None,
 | ||
|  |               help='Enable or disable the debugger. By default the debugger '
 | ||
|  |               'is active if debug is enabled.')
 | ||
|  | @click.option('--eager-loading/--lazy-loader', default=None,
 | ||
|  |               help='Enable or disable eager loading. By default eager '
 | ||
|  |               'loading is enabled if the reloader is disabled.')
 | ||
|  | @click.option('--with-threads/--without-threads', default=True,
 | ||
|  |               help='Enable or disable multithreading.')
 | ||
|  | @pass_script_info
 | ||
|  | def run_command(info, host, port, reload, debugger, eager_loading,
 | ||
|  |                 with_threads, cert):
 | ||
|  |     """Run a local development server.
 | ||
|  | 
 | ||
|  |     This server is for development purposes only. It does not provide
 | ||
|  |     the stability, security, or performance of production WSGI servers.
 | ||
|  | 
 | ||
|  |     The reloader and debugger are enabled by default if
 | ||
|  |     FLASK_ENV=development or FLASK_DEBUG=1.
 | ||
|  |     """
 | ||
|  |     debug = get_debug_flag()
 | ||
|  | 
 | ||
|  |     if reload is None:
 | ||
|  |         reload = debug
 | ||
|  | 
 | ||
|  |     if debugger is None:
 | ||
|  |         debugger = debug
 | ||
|  | 
 | ||
|  |     if eager_loading is None:
 | ||
|  |         eager_loading = not reload
 | ||
|  | 
 | ||
|  |     show_server_banner(get_env(), debug, info.app_import_path, eager_loading)
 | ||
|  |     app = DispatchingApp(info.load_app, use_eager_loading=eager_loading)
 | ||
|  | 
 | ||
|  |     from werkzeug.serving import run_simple
 | ||
|  |     run_simple(host, port, app, use_reloader=reload, use_debugger=debugger,
 | ||
|  |                threaded=with_threads, ssl_context=cert)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | @click.command('shell', short_help='Runs a shell in the app context.')
 | ||
|  | @with_appcontext
 | ||
|  | def shell_command():
 | ||
|  |     """Runs an interactive Python shell in the context of a given
 | ||
|  |     Flask application.  The application will populate the default
 | ||
|  |     namespace of this shell according to it's configuration.
 | ||
|  | 
 | ||
|  |     This is useful for executing small snippets of management code
 | ||
|  |     without having to manually configure the application.
 | ||
|  |     """
 | ||
|  |     import code
 | ||
|  |     from flask.globals import _app_ctx_stack
 | ||
|  |     app = _app_ctx_stack.top.app
 | ||
|  |     banner = 'Python %s on %s\nApp: %s [%s]\nInstance: %s' % (
 | ||
|  |         sys.version,
 | ||
|  |         sys.platform,
 | ||
|  |         app.import_name,
 | ||
|  |         app.env,
 | ||
|  |         app.instance_path,
 | ||
|  |     )
 | ||
|  |     ctx = {}
 | ||
|  | 
 | ||
|  |     # Support the regular Python interpreter startup script if someone
 | ||
|  |     # is using it.
 | ||
|  |     startup = os.environ.get('PYTHONSTARTUP')
 | ||
|  |     if startup and os.path.isfile(startup):
 | ||
|  |         with open(startup, 'r') as f:
 | ||
|  |             eval(compile(f.read(), startup, 'exec'), ctx)
 | ||
|  | 
 | ||
|  |     ctx.update(app.make_shell_context())
 | ||
|  | 
 | ||
|  |     code.interact(banner=banner, local=ctx)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | @click.command('routes', short_help='Show the routes for the app.')
 | ||
|  | @click.option(
 | ||
|  |     '--sort', '-s',
 | ||
|  |     type=click.Choice(('endpoint', 'methods', 'rule', 'match')),
 | ||
|  |     default='endpoint',
 | ||
|  |     help=(
 | ||
|  |         'Method to sort routes by. "match" is the order that Flask will match '
 | ||
|  |         'routes when dispatching a request.'
 | ||
|  |     )
 | ||
|  | )
 | ||
|  | @click.option(
 | ||
|  |     '--all-methods',
 | ||
|  |     is_flag=True,
 | ||
|  |     help="Show HEAD and OPTIONS methods."
 | ||
|  | )
 | ||
|  | @with_appcontext
 | ||
|  | def routes_command(sort, all_methods):
 | ||
|  |     """Show all registered routes with endpoints and methods."""
 | ||
|  | 
 | ||
|  |     rules = list(current_app.url_map.iter_rules())
 | ||
|  |     if not rules:
 | ||
|  |         click.echo('No routes were registered.')
 | ||
|  |         return
 | ||
|  | 
 | ||
|  |     ignored_methods = set(() if all_methods else ('HEAD', 'OPTIONS'))
 | ||
|  | 
 | ||
|  |     if sort in ('endpoint', 'rule'):
 | ||
|  |         rules = sorted(rules, key=attrgetter(sort))
 | ||
|  |     elif sort == 'methods':
 | ||
|  |         rules = sorted(rules, key=lambda rule: sorted(rule.methods))
 | ||
|  | 
 | ||
|  |     rule_methods = [
 | ||
|  |         ', '.join(sorted(rule.methods - ignored_methods)) for rule in rules
 | ||
|  |     ]
 | ||
|  | 
 | ||
|  |     headers = ('Endpoint', 'Methods', 'Rule')
 | ||
|  |     widths = (
 | ||
|  |         max(len(rule.endpoint) for rule in rules),
 | ||
|  |         max(len(methods) for methods in rule_methods),
 | ||
|  |         max(len(rule.rule) for rule in rules),
 | ||
|  |     )
 | ||
|  |     widths = [max(len(h), w) for h, w in zip(headers, widths)]
 | ||
|  |     row = '{{0:<{0}}}  {{1:<{1}}}  {{2:<{2}}}'.format(*widths)
 | ||
|  | 
 | ||
|  |     click.echo(row.format(*headers).strip())
 | ||
|  |     click.echo(row.format(*('-' * width for width in widths)))
 | ||
|  | 
 | ||
|  |     for rule, methods in zip(rules, rule_methods):
 | ||
|  |         click.echo(row.format(rule.endpoint, methods, rule.rule).rstrip())
 | ||
|  | 
 | ||
|  | 
 | ||
|  | cli = FlaskGroup(help="""\
 | ||
|  | A general utility script for Flask applications.
 | ||
|  | 
 | ||
|  | Provides commands from Flask, extensions, and the application. Loads the
 | ||
|  | application defined in the FLASK_APP environment variable, or from a wsgi.py
 | ||
|  | file. Setting the FLASK_ENV environment variable to 'development' will enable
 | ||
|  | debug mode.
 | ||
|  | 
 | ||
|  | \b
 | ||
|  |   {prefix}{cmd} FLASK_APP=hello.py
 | ||
|  |   {prefix}{cmd} FLASK_ENV=development
 | ||
|  |   {prefix}flask run
 | ||
|  | """.format(
 | ||
|  |     cmd='export' if os.name == 'posix' else 'set',
 | ||
|  |     prefix='$ ' if os.name == 'posix' else '> '
 | ||
|  | ))
 | ||
|  | 
 | ||
|  | 
 | ||
|  | def main(as_module=False):
 | ||
|  |     args = sys.argv[1:]
 | ||
|  | 
 | ||
|  |     if as_module:
 | ||
|  |         this_module = 'flask'
 | ||
|  | 
 | ||
|  |         if sys.version_info < (2, 7):
 | ||
|  |             this_module += '.cli'
 | ||
|  | 
 | ||
|  |         name = 'python -m ' + this_module
 | ||
|  | 
 | ||
|  |         # Python rewrites "python -m flask" to the path to the file in argv.
 | ||
|  |         # Restore the original command so that the reloader works.
 | ||
|  |         sys.argv = ['-m', this_module] + args
 | ||
|  |     else:
 | ||
|  |         name = None
 | ||
|  | 
 | ||
|  |     cli.main(args=args, prog_name=name)
 | ||
|  | 
 | ||
|  | 
 | ||
|  | if __name__ == '__main__':
 | ||
|  |     main(as_module=True)
 |