2014-12-16 15:54:29 +00:00
|
|
|
##########################################################################
|
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2018-01-05 10:42:49 +00:00
|
|
|
# Copyright (C) 2013 - 2018, The pgAdmin Development Team
|
2014-12-16 15:54:29 +00:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
2015-06-29 06:58:41 +00:00
|
|
|
"""This is the main application entry point for pgAdmin 4. If running on
|
2015-01-21 12:00:13 +00:00
|
|
|
a webserver, this will provide the WSGI interface, otherwise, we're going
|
|
|
|
to start a web server."""
|
|
|
|
|
2015-06-30 05:51:55 +00:00
|
|
|
import os
|
|
|
|
import sys
|
2014-12-16 15:54:29 +00:00
|
|
|
|
2017-08-25 09:54:28 +00:00
|
|
|
if sys.version_info[0] >= 3:
|
|
|
|
import builtins
|
|
|
|
else:
|
|
|
|
import __builtin__ as builtins
|
|
|
|
|
2014-12-16 15:54:29 +00:00
|
|
|
# We need to include the root directory in sys.path to ensure that we can
|
|
|
|
# find everything we need when running in the standalone runtime.
|
2015-03-10 13:09:11 +00:00
|
|
|
root = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
if sys.path[0] != root:
|
|
|
|
sys.path.insert(0, root)
|
2014-12-16 15:54:29 +00:00
|
|
|
|
2017-08-25 09:54:28 +00:00
|
|
|
# Grab the SERVER_MODE if it's been set by the runtime
|
|
|
|
if 'SERVER_MODE' in globals():
|
|
|
|
builtins.SERVER_MODE = globals()['SERVER_MODE']
|
|
|
|
else:
|
|
|
|
builtins.SERVER_MODE = None
|
|
|
|
|
2014-12-16 17:14:48 +00:00
|
|
|
import config
|
2014-12-18 17:49:09 +00:00
|
|
|
from pgadmin import create_app
|
2017-06-12 15:51:54 +00:00
|
|
|
from pgadmin.utils import u, fs_encoding, file_quote
|
|
|
|
|
|
|
|
if config.DEBUG:
|
2018-01-26 16:54:21 +00:00
|
|
|
from pgadmin.utils.javascript.javascript_bundler import \
|
|
|
|
JavascriptBundler, JsState
|
2013-10-04 16:12:10 +00:00
|
|
|
|
2016-06-24 11:50:52 +00:00
|
|
|
# Get the config database schema version. We store this in pgadmin.model
|
|
|
|
# as it turns out that putting it in the config files isn't a great idea
|
|
|
|
from pgadmin.model import SCHEMA_VERSION
|
2018-03-08 09:33:43 +00:00
|
|
|
|
2016-06-24 11:50:52 +00:00
|
|
|
config.SETTINGS_SCHEMA_VERSION = SCHEMA_VERSION
|
|
|
|
|
2017-06-12 15:51:54 +00:00
|
|
|
##########################################################################
|
|
|
|
# Sanity checks
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
# Check if the database exists. If it does not, create it.
|
|
|
|
if not os.path.isfile(config.SQLITE_PATH):
|
|
|
|
setupfile = os.path.join(
|
2018-01-26 16:54:21 +00:00
|
|
|
os.path.dirname(os.path.realpath(u(__file__, fs_encoding))),
|
|
|
|
u'setup.py'
|
2017-06-12 15:51:54 +00:00
|
|
|
)
|
2018-03-08 09:33:43 +00:00
|
|
|
exec (open(file_quote(setupfile), 'r').read())
|
2015-01-26 15:20:28 +00:00
|
|
|
|
2014-12-17 15:27:54 +00:00
|
|
|
##########################################################################
|
2017-06-12 15:51:54 +00:00
|
|
|
# Server startup
|
2014-12-17 15:27:54 +00:00
|
|
|
##########################################################################
|
|
|
|
|
2017-06-12 15:51:54 +00:00
|
|
|
# Build Javascript files
|
|
|
|
if config.DEBUG:
|
|
|
|
javascriptBundler = JavascriptBundler()
|
|
|
|
javascriptBundler.bundle()
|
|
|
|
|
2014-12-18 17:49:09 +00:00
|
|
|
# Create the app!
|
|
|
|
app = create_app()
|
|
|
|
|
2015-06-29 08:11:56 +00:00
|
|
|
if config.DEBUG:
|
|
|
|
app.debug = True
|
2015-10-20 07:03:18 +00:00
|
|
|
else:
|
|
|
|
app.debug = False
|
2015-06-29 06:58:41 +00:00
|
|
|
|
2017-06-12 15:51:54 +00:00
|
|
|
# respond to JS
|
|
|
|
if config.DEBUG:
|
|
|
|
if javascriptBundler.report() == JsState.NONE:
|
|
|
|
app.logger.error("Unable to generate javascript")
|
2018-01-26 16:54:21 +00:00
|
|
|
app.logger.error(
|
|
|
|
"To run the app ensure that yarn install command runs successfully"
|
|
|
|
)
|
2017-06-12 15:51:54 +00:00
|
|
|
raise Exception("No generated javascript, aborting")
|
|
|
|
|
2013-10-04 16:12:10 +00:00
|
|
|
# Start the web server. The port number should have already been set by the
|
2015-06-29 06:58:41 +00:00
|
|
|
# runtime if we're running in desktop mode, otherwise we'll just use the
|
2014-12-16 12:53:09 +00:00
|
|
|
# Flask default.
|
2016-06-17 09:03:32 +00:00
|
|
|
PGADMIN_RUNTIME = False
|
2014-12-16 12:53:09 +00:00
|
|
|
if 'PGADMIN_PORT' in globals():
|
2018-01-26 16:54:21 +00:00
|
|
|
app.logger.debug(
|
|
|
|
'Running under the desktop runtime, port: %s',
|
|
|
|
globals()['PGADMIN_PORT']
|
|
|
|
)
|
2015-06-30 05:51:55 +00:00
|
|
|
server_port = int(globals()['PGADMIN_PORT'])
|
2016-06-17 09:03:32 +00:00
|
|
|
PGADMIN_RUNTIME = True
|
2017-02-22 12:41:28 +00:00
|
|
|
elif 'PGADMIN_PORT' in os.environ:
|
|
|
|
port = os.environ['PGADMIN_PORT']
|
|
|
|
app.logger.debug(
|
|
|
|
'Not running under the desktop runtime, port: %s',
|
2018-01-26 16:54:21 +00:00
|
|
|
port
|
|
|
|
)
|
2017-02-22 12:41:28 +00:00
|
|
|
server_port = int(port)
|
2014-12-16 12:53:09 +00:00
|
|
|
else:
|
2015-06-30 05:51:55 +00:00
|
|
|
app.logger.debug(
|
2016-06-17 09:03:32 +00:00
|
|
|
'Not running under the desktop runtime, port: %s',
|
2018-01-26 16:54:21 +00:00
|
|
|
config.DEFAULT_SERVER_PORT
|
|
|
|
)
|
2014-12-16 17:14:48 +00:00
|
|
|
server_port = config.DEFAULT_SERVER_PORT
|
2014-12-16 12:53:09 +00:00
|
|
|
|
2016-06-21 09:42:20 +00:00
|
|
|
# Let the application save the status about the runtime for using it later.
|
|
|
|
app.PGADMIN_RUNTIME = PGADMIN_RUNTIME
|
|
|
|
|
2017-03-06 14:53:49 +00:00
|
|
|
# Set the key if appropriate
|
|
|
|
if 'PGADMIN_KEY' in globals():
|
|
|
|
app.PGADMIN_KEY = globals()['PGADMIN_KEY']
|
|
|
|
app.logger.debug("Desktop security key: %s" % app.PGADMIN_KEY)
|
|
|
|
else:
|
|
|
|
app.PGADMIN_KEY = ''
|
|
|
|
|
2016-08-18 12:43:00 +00:00
|
|
|
# Output a startup message if we're not under the runtime and startup.
|
|
|
|
# If we're under WSGI, we don't need to worry about this
|
|
|
|
if __name__ == '__main__':
|
|
|
|
if not PGADMIN_RUNTIME:
|
2018-01-26 16:54:21 +00:00
|
|
|
print(
|
|
|
|
"Starting %s. Please navigate to http://%s:%d in your browser." %
|
|
|
|
(config.APP_NAME, config.DEFAULT_SERVER, server_port)
|
|
|
|
)
|
2016-08-18 12:43:00 +00:00
|
|
|
sys.stdout.flush()
|
2017-02-10 22:42:44 +00:00
|
|
|
else:
|
|
|
|
# For unknown reason the Qt runtime does not pass the environment
|
|
|
|
# variables (i.e. PYTHONHOME, and PYTHONPATH), to the Python
|
|
|
|
# sub-processes, leading to failures executing background processes.
|
|
|
|
#
|
|
|
|
# This has been observed only on windows. On *nix systems, it is likely
|
|
|
|
# picking the system python environment, which is good enough to run
|
|
|
|
# the process-executor.
|
|
|
|
#
|
|
|
|
# Setting PYTHONHOME launch them properly.
|
2017-03-09 09:54:55 +00:00
|
|
|
from pgadmin.utils import IS_WIN
|
2018-03-08 09:33:43 +00:00
|
|
|
|
2017-03-09 09:54:55 +00:00
|
|
|
if IS_WIN:
|
|
|
|
os.environ['PYTHONHOME'] = sys.prefix
|
2016-06-14 11:00:06 +00:00
|
|
|
|
2017-07-21 08:46:30 +00:00
|
|
|
# Initialize Flask service only once
|
|
|
|
# If `WERKZEUG_RUN_MAIN` is None, i.e: app is initializing for first time
|
|
|
|
# so set `use_reloader` = False, thus reload won't call.
|
|
|
|
# Reference:
|
|
|
|
# https://github.com/pallets/werkzeug/issues/220#issuecomment-11176538
|
2016-08-18 12:43:00 +00:00
|
|
|
try:
|
|
|
|
app.run(
|
|
|
|
host=config.DEFAULT_SERVER,
|
|
|
|
port=server_port,
|
2017-07-21 08:46:30 +00:00
|
|
|
use_reloader=(
|
2018-03-08 09:33:43 +00:00
|
|
|
(not PGADMIN_RUNTIME) and app.debug and
|
|
|
|
os.environ.get("WERKZEUG_RUN_MAIN") is not None
|
2017-07-21 08:46:30 +00:00
|
|
|
),
|
2016-08-18 12:43:00 +00:00
|
|
|
threaded=config.THREADED_MODE
|
|
|
|
)
|
|
|
|
except IOError:
|
|
|
|
app.logger.error("Error starting the app server: %s", sys.exc_info())
|