2015-06-30 05:51:55 +00:00
|
|
|
#########################################################################
|
2015-01-22 15:56:23 +00:00
|
|
|
#
|
|
|
|
# pgAdmin 4 - PostgreSQL Tools
|
|
|
|
#
|
2016-01-18 14:48:14 +00:00
|
|
|
# Copyright (C) 2013 - 2016, The pgAdmin Development Team
|
2015-01-22 15:56:23 +00:00
|
|
|
# This software is released under the PostgreSQL Licence
|
|
|
|
#
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
"""Perform the initial setup of the application, by creating the auth
|
|
|
|
and settings database."""
|
|
|
|
|
2015-10-20 07:03:18 +00:00
|
|
|
import getpass
|
2016-06-21 13:12:14 +00:00
|
|
|
import os
|
2015-10-20 07:03:18 +00:00
|
|
|
import random
|
2016-06-10 15:19:46 +00:00
|
|
|
import re
|
2016-06-21 13:12:14 +00:00
|
|
|
import string
|
|
|
|
import sys
|
2015-10-20 07:03:18 +00:00
|
|
|
|
2015-01-22 15:56:23 +00:00
|
|
|
from flask import Flask
|
|
|
|
from flask.ext.security import Security, SQLAlchemyUserDatastore
|
|
|
|
from flask.ext.security.utils import encrypt_password
|
2016-06-21 13:12:14 +00:00
|
|
|
|
2016-03-07 11:48:24 +00:00
|
|
|
from pgadmin.model import db, Role, User, Server, \
|
2015-10-20 07:03:18 +00:00
|
|
|
ServerGroup, Version
|
2015-01-22 15:56:23 +00:00
|
|
|
# Configuration settings
|
|
|
|
import config
|
|
|
|
|
2015-12-04 10:07:04 +00:00
|
|
|
# If script is running under python2 then change the behaviour of functions
|
|
|
|
if hasattr(__builtins__, 'raw_input'):
|
|
|
|
input = raw_input
|
|
|
|
range = xrange
|
2015-10-20 07:03:18 +00:00
|
|
|
|
|
|
|
|
2016-06-21 13:21:06 +00:00
|
|
|
def do_setup(app):
|
2015-03-09 15:06:10 +00:00
|
|
|
"""Create a new settings database from scratch"""
|
2015-06-30 05:51:55 +00:00
|
|
|
if config.SERVER_MODE is False:
|
2015-07-22 16:42:39 +00:00
|
|
|
print("NOTE: Configuring authentication for DESKTOP mode.")
|
2015-06-30 05:51:55 +00:00
|
|
|
email = config.DESKTOP_USER
|
2015-10-20 07:03:18 +00:00
|
|
|
p1 = ''.join([
|
2016-06-21 13:21:06 +00:00
|
|
|
random.choice(string.ascii_letters + string.digits)
|
|
|
|
for n in range(32)
|
|
|
|
])
|
2015-03-09 15:06:10 +00:00
|
|
|
|
|
|
|
else:
|
2015-07-22 16:42:39 +00:00
|
|
|
print("NOTE: Configuring authentication for SERVER mode.\n")
|
2015-03-09 15:06:10 +00:00
|
|
|
|
|
|
|
# Prompt the user for their default username and password.
|
2015-10-20 07:03:18 +00:00
|
|
|
print("""
|
|
|
|
Enter the email address and password to use for the initial pgAdmin user \
|
|
|
|
account:\n""")
|
2016-06-10 15:19:46 +00:00
|
|
|
email_filter = re.compile(
|
|
|
|
"^[a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~-]+@[a-zA-Z0-9]"
|
|
|
|
"(?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9]"
|
|
|
|
"(?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$")
|
|
|
|
|
2015-03-09 15:06:10 +00:00
|
|
|
email = ''
|
2016-06-10 15:19:46 +00:00
|
|
|
input("Email address: ")
|
|
|
|
while email == '' or not email_filter.match(email):
|
|
|
|
print('Invalid email address. Please try again.')
|
2015-11-06 10:23:19 +00:00
|
|
|
email = input("Email address: ")
|
2015-03-09 15:06:10 +00:00
|
|
|
|
2015-10-20 07:03:18 +00:00
|
|
|
def pprompt():
|
|
|
|
return getpass.getpass(), getpass.getpass('Retype password:')
|
2015-03-09 15:06:10 +00:00
|
|
|
|
|
|
|
p1, p2 = pprompt()
|
|
|
|
while p1 != p2:
|
2016-06-10 15:19:46 +00:00
|
|
|
print('Passwords do not match. Please try again.')
|
2015-03-09 15:06:10 +00:00
|
|
|
p1, p2 = pprompt()
|
|
|
|
|
|
|
|
# Setup Flask-Security
|
|
|
|
user_datastore = SQLAlchemyUserDatastore(db, User, Role)
|
2016-05-13 03:19:48 +00:00
|
|
|
Security(app, user_datastore)
|
2015-03-09 15:06:10 +00:00
|
|
|
|
|
|
|
with app.app_context():
|
|
|
|
password = encrypt_password(p1)
|
|
|
|
|
|
|
|
db.create_all()
|
2015-10-20 07:03:18 +00:00
|
|
|
user_datastore.create_role(
|
2016-06-21 13:21:06 +00:00
|
|
|
name='Administrator',
|
|
|
|
description='pgAdmin Administrator Role'
|
|
|
|
)
|
2016-06-06 12:34:08 +00:00
|
|
|
user_datastore.create_role(
|
2016-06-21 13:21:06 +00:00
|
|
|
name='User',
|
|
|
|
description='pgAdmin User Role'
|
|
|
|
)
|
2016-06-06 12:34:08 +00:00
|
|
|
|
2015-03-09 15:06:10 +00:00
|
|
|
user_datastore.create_user(email=email, password=password)
|
2015-09-22 05:46:43 +00:00
|
|
|
db.session.flush()
|
2016-06-06 12:34:08 +00:00
|
|
|
user_datastore.add_role_to_user(email, 'Administrator')
|
2015-06-30 05:51:55 +00:00
|
|
|
|
2015-03-09 15:06:10 +00:00
|
|
|
# Get the user's ID and create the default server group
|
|
|
|
user = User.query.filter_by(email=email).first()
|
|
|
|
server_group = ServerGroup(user_id=user.id, name="Servers")
|
|
|
|
db.session.merge(server_group)
|
|
|
|
|
|
|
|
# Set the schema version
|
2015-10-20 07:03:18 +00:00
|
|
|
version = Version(
|
2016-06-21 13:21:06 +00:00
|
|
|
name='ConfigDB', value=config.SETTINGS_SCHEMA_VERSION
|
|
|
|
)
|
2015-03-09 15:06:10 +00:00
|
|
|
db.session.merge(version)
|
2015-06-30 05:51:55 +00:00
|
|
|
|
2015-03-09 15:06:10 +00:00
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
# Done!
|
2015-07-22 16:42:39 +00:00
|
|
|
print("")
|
2015-10-20 07:03:18 +00:00
|
|
|
print(
|
2016-06-21 13:21:06 +00:00
|
|
|
"The configuration database has been created at {0}".format(
|
|
|
|
config.SQLITE_PATH
|
|
|
|
)
|
|
|
|
)
|
2015-06-30 05:51:55 +00:00
|
|
|
|
|
|
|
|
2015-07-22 16:42:39 +00:00
|
|
|
def do_upgrade(app, datastore, security, version):
|
2015-03-09 15:06:10 +00:00
|
|
|
"""Upgrade an existing settings database"""
|
2015-07-22 16:42:39 +00:00
|
|
|
#######################################################################
|
|
|
|
# Run whatever is required to update the database schema to the current
|
|
|
|
# version.
|
|
|
|
#######################################################################
|
2015-06-30 05:51:55 +00:00
|
|
|
|
2015-10-20 07:03:18 +00:00
|
|
|
with app.app_context():
|
|
|
|
version = Version.query.filter_by(name='ConfigDB').first()
|
|
|
|
|
|
|
|
# Pre-flight checks
|
|
|
|
if int(version.value) > int(config.SETTINGS_SCHEMA_VERSION):
|
|
|
|
print("""
|
|
|
|
The database schema version is {0}, whilst the version required by the \
|
|
|
|
software is {1}.
|
|
|
|
Exiting...""".format(version.value, config.SETTINGS_SCHEMA_VERSION))
|
|
|
|
sys.exit(1)
|
|
|
|
elif int(version.value) == int(config.SETTINGS_SCHEMA_VERSION):
|
|
|
|
print("""
|
|
|
|
The database schema version is {0} as required.
|
|
|
|
Exiting...""".format(version.value))
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
app.logger.info(
|
|
|
|
"NOTE: Upgrading database schema from version %d to %d." %
|
|
|
|
(version.value, config.SETTINGS_SCHEMA_VERSION)
|
2016-06-21 13:21:06 +00:00
|
|
|
)
|
2015-10-20 07:03:18 +00:00
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
# Run whatever is required to update the database schema to the current
|
|
|
|
# version. Always use "< REQUIRED_VERSION" as the test for readability
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
# Changes introduced in schema version 2
|
|
|
|
if int(version.value) < 2:
|
|
|
|
# Create the 'server' table
|
|
|
|
db.metadata.create_all(db.engine, tables=[Server.__table__])
|
|
|
|
if int(version.value) < 3:
|
|
|
|
db.engine.execute(
|
|
|
|
'ALTER TABLE server ADD COLUMN comment TEXT(1024)'
|
2016-06-21 13:21:06 +00:00
|
|
|
)
|
2015-10-20 07:03:18 +00:00
|
|
|
if int(version.value) < 4:
|
|
|
|
db.engine.execute(
|
|
|
|
'ALTER TABLE server ADD COLUMN password TEXT(64)'
|
2016-06-21 13:21:06 +00:00
|
|
|
)
|
2015-10-20 07:03:18 +00:00
|
|
|
if int(version.value) < 5:
|
|
|
|
db.engine.execute('ALTER TABLE server ADD COLUMN role text(64)')
|
2016-03-07 11:48:24 +00:00
|
|
|
if int(version.value) < 6:
|
2015-10-20 07:03:18 +00:00
|
|
|
db.engine.execute("ALTER TABLE server RENAME TO server_old")
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE server (
|
|
|
|
id INTEGER NOT NULL,
|
|
|
|
user_id INTEGER NOT NULL,
|
|
|
|
servergroup_id INTEGER NOT NULL,
|
|
|
|
name VARCHAR(128) NOT NULL,
|
|
|
|
host VARCHAR(128) NOT NULL,
|
|
|
|
port INTEGER NOT NULL CHECK (port >= 1024 AND port <= 65534),
|
|
|
|
maintenance_db VARCHAR(64) NOT NULL,
|
|
|
|
username VARCHAR(64) NOT NULL,
|
|
|
|
ssl_mode VARCHAR(16) NOT NULL CHECK (
|
|
|
|
ssl_mode IN (
|
|
|
|
'allow', 'prefer', 'require', 'disable', 'verify-ca', 'verify-full'
|
|
|
|
)),
|
|
|
|
comment VARCHAR(1024), password TEXT(64), role text(64),
|
|
|
|
PRIMARY KEY (id),
|
|
|
|
FOREIGN KEY(user_id) REFERENCES user (id),
|
|
|
|
FOREIGN KEY(servergroup_id) REFERENCES servergroup (id)
|
|
|
|
)""")
|
|
|
|
db.engine.execute("""
|
|
|
|
INSERT INTO server (
|
|
|
|
id, user_id, servergroup_id, name, host, port, maintenance_db, username,
|
|
|
|
ssl_mode, comment, password, role
|
|
|
|
) SELECT
|
|
|
|
id, user_id, servergroup_id, name, host, port, maintenance_db, username,
|
|
|
|
ssl_mode, comment, password, role
|
|
|
|
FROM server_old""")
|
|
|
|
db.engine.execute("DROP TABLE server_old")
|
|
|
|
|
2016-03-07 11:48:24 +00:00
|
|
|
if int(version.value) < 8:
|
|
|
|
app.logger.info(
|
|
|
|
"Creating the preferences tables..."
|
2016-06-21 13:21:06 +00:00
|
|
|
)
|
2016-03-07 11:48:24 +00:00
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE module_preference(
|
|
|
|
id INTEGER PRIMARY KEY,
|
|
|
|
name VARCHAR(256) NOT NULL
|
|
|
|
)""")
|
|
|
|
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE preference_category(
|
|
|
|
id INTEGER PRIMARY KEY,
|
|
|
|
mid INTEGER,
|
|
|
|
name VARCHAR(256) NOT NULL,
|
|
|
|
|
|
|
|
FOREIGN KEY(mid) REFERENCES module_preference(id)
|
|
|
|
)""")
|
|
|
|
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE preferences (
|
|
|
|
|
|
|
|
id INTEGER PRIMARY KEY,
|
|
|
|
cid INTEGER NOT NULL,
|
|
|
|
name VARCHAR(256) NOT NULL,
|
|
|
|
|
|
|
|
FOREIGN KEY(cid) REFERENCES preference_category (id)
|
|
|
|
)""")
|
|
|
|
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE user_preferences (
|
|
|
|
|
|
|
|
pid INTEGER,
|
|
|
|
uid INTEGER,
|
|
|
|
value VARCHAR(1024) NOT NULL,
|
|
|
|
|
|
|
|
PRIMARY KEY (pid, uid),
|
|
|
|
FOREIGN KEY(pid) REFERENCES preferences (pid),
|
|
|
|
FOREIGN KEY(uid) REFERENCES user (id)
|
2016-04-14 20:36:04 +00:00
|
|
|
)""")
|
|
|
|
|
|
|
|
if int(version.value) < 9:
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE IF NOT EXISTS debugger_function_arguments (
|
|
|
|
server_id INTEGER ,
|
|
|
|
database_id INTEGER ,
|
|
|
|
schema_id INTEGER ,
|
|
|
|
function_id INTEGER ,
|
|
|
|
arg_id INTEGER ,
|
|
|
|
is_null INTEGER NOT NULL CHECK (is_null >= 0 AND is_null <= 1) ,
|
|
|
|
is_expression INTEGER NOT NULL CHECK (is_expression >= 0 AND is_expression <= 1) ,
|
|
|
|
use_default INTEGER NOT NULL CHECK (use_default >= 0 AND use_default <= 1) ,
|
|
|
|
value TEXT,
|
|
|
|
PRIMARY KEY (server_id, database_id, schema_id, function_id, arg_id)
|
2016-05-13 03:19:48 +00:00
|
|
|
)""")
|
|
|
|
|
|
|
|
if int(version.value) < 10:
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE process(
|
|
|
|
user_id INTEGER NOT NULL,
|
|
|
|
pid TEXT NOT NULL,
|
|
|
|
desc TEXT NOT NULL,
|
|
|
|
command TEXT NOT NULL,
|
|
|
|
arguments TEXT,
|
|
|
|
start_time TEXT,
|
|
|
|
end_time TEXT,
|
|
|
|
logdir TEXT,
|
|
|
|
exit_code INTEGER,
|
|
|
|
acknowledge TEXT,
|
|
|
|
PRIMARY KEY(pid),
|
|
|
|
FOREIGN KEY(user_id) REFERENCES user (id)
|
2016-03-07 11:48:24 +00:00
|
|
|
)""")
|
2015-06-30 05:51:55 +00:00
|
|
|
|
2016-06-06 12:34:08 +00:00
|
|
|
if int(version.value) < 11:
|
|
|
|
db.engine.execute("""
|
|
|
|
UPDATE role
|
|
|
|
SET name = 'Administrator',
|
|
|
|
description = 'pgAdmin Administrator Role'
|
|
|
|
WHERE name = 'Administrators'
|
|
|
|
""")
|
|
|
|
|
|
|
|
db.engine.execute("""
|
|
|
|
INSERT INTO role ( name, description )
|
|
|
|
VALUES ('User', 'pgAdmin User Role')
|
|
|
|
""")
|
|
|
|
|
2016-06-23 10:43:50 +00:00
|
|
|
if int(version.value) < 12:
|
|
|
|
db.engine.execute("ALTER TABLE server RENAME TO server_old")
|
|
|
|
db.engine.execute("""
|
|
|
|
CREATE TABLE server (
|
|
|
|
id INTEGER NOT NULL,
|
|
|
|
user_id INTEGER NOT NULL,
|
|
|
|
servergroup_id INTEGER NOT NULL,
|
|
|
|
name VARCHAR(128) NOT NULL,
|
|
|
|
host VARCHAR(128) NOT NULL,
|
|
|
|
port INTEGER NOT NULL CHECK (port >= 1024 AND port <= 65535),
|
|
|
|
maintenance_db VARCHAR(64) NOT NULL,
|
|
|
|
username VARCHAR(64) NOT NULL,
|
|
|
|
ssl_mode VARCHAR(16) NOT NULL CHECK (
|
|
|
|
ssl_mode IN (
|
|
|
|
'allow', 'prefer', 'require', 'disable', 'verify-ca', 'verify-full'
|
|
|
|
)),
|
|
|
|
comment VARCHAR(1024), password TEXT(64), role text(64),
|
|
|
|
PRIMARY KEY (id),
|
|
|
|
FOREIGN KEY(user_id) REFERENCES user (id),
|
|
|
|
FOREIGN KEY(servergroup_id) REFERENCES servergroup (id)
|
|
|
|
)""")
|
|
|
|
db.engine.execute("""
|
|
|
|
INSERT INTO server (
|
|
|
|
id, user_id, servergroup_id, name, host, port, maintenance_db, username,
|
|
|
|
ssl_mode, comment, password, role
|
|
|
|
) SELECT
|
|
|
|
id, user_id, servergroup_id, name, host, port, maintenance_db, username,
|
|
|
|
ssl_mode, comment, password, role
|
|
|
|
FROM server_old""")
|
|
|
|
db.engine.execute("DROP TABLE server_old")
|
|
|
|
|
2016-06-23 15:27:06 +00:00
|
|
|
if int(version.value) < 13:
|
|
|
|
db.engine.execute("""
|
|
|
|
ALTER TABLE SERVER
|
|
|
|
ADD COLUMN discovery_id TEXT
|
|
|
|
""")
|
|
|
|
|
2015-07-22 16:42:39 +00:00
|
|
|
# Finally, update the schema version
|
|
|
|
version.value = config.SETTINGS_SCHEMA_VERSION
|
|
|
|
db.session.merge(version)
|
|
|
|
|
|
|
|
db.session.commit()
|
2015-06-30 05:51:55 +00:00
|
|
|
|
2015-03-09 15:06:10 +00:00
|
|
|
# Done!
|
2015-10-20 07:03:18 +00:00
|
|
|
app.logger.info(
|
|
|
|
"The configuration database %s has been upgraded to version %d" %
|
2016-06-21 13:21:06 +00:00
|
|
|
(config.SQLITE_PATH, config.SETTINGS_SCHEMA_VERSION)
|
|
|
|
)
|
|
|
|
|
2015-06-30 05:51:55 +00:00
|
|
|
|
2015-03-09 15:06:10 +00:00
|
|
|
###############################################################################
|
|
|
|
# Do stuff!
|
|
|
|
###############################################################################
|
2015-07-22 16:42:39 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
app = Flask(__name__)
|
|
|
|
app.config.from_object(config)
|
2015-10-20 07:03:18 +00:00
|
|
|
app.config['SQLALCHEMY_DATABASE_URI'] = \
|
|
|
|
'sqlite:///' + config.SQLITE_PATH.replace('\\', '/')
|
2015-07-22 16:42:39 +00:00
|
|
|
db.init_app(app)
|
|
|
|
|
|
|
|
print("pgAdmin 4 - Application Initialisation")
|
|
|
|
print("======================================\n")
|
|
|
|
|
2015-10-20 07:03:18 +00:00
|
|
|
local_config = os.path.join(
|
2016-06-21 13:21:06 +00:00
|
|
|
os.path.dirname(os.path.realpath(__file__)),
|
|
|
|
'config_local.py'
|
|
|
|
)
|
2015-07-22 16:42:39 +00:00
|
|
|
if not os.path.isfile(local_config):
|
2015-10-20 07:03:18 +00:00
|
|
|
print("""
|
|
|
|
The configuration file - {0} does not exist.
|
|
|
|
Before running this application, ensure that config_local.py has been created
|
|
|
|
and sets values for SECRET_KEY, SECURITY_PASSWORD_SALT and CSRF_SESSION_KEY
|
|
|
|
at bare minimum. See config.py for more information and a complete list of
|
|
|
|
settings. Exiting...""".format(local_config))
|
2015-07-22 16:42:39 +00:00
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
# Check if the database exists. If it does, tell the user and exit.
|
|
|
|
if os.path.isfile(config.SQLITE_PATH):
|
2015-10-20 07:03:18 +00:00
|
|
|
print("""
|
2016-03-07 11:48:24 +00:00
|
|
|
The configuration database '%s' already exists.
|
|
|
|
Entering upgrade mode...""" % config.SQLITE_PATH)
|
2015-07-22 16:42:39 +00:00
|
|
|
|
|
|
|
# Setup Flask-Security
|
|
|
|
user_datastore = SQLAlchemyUserDatastore(db, User, Role)
|
|
|
|
security = Security(app, user_datastore)
|
|
|
|
|
|
|
|
# Always use "< REQUIRED_VERSION" as the test for readability
|
|
|
|
with app.app_context():
|
|
|
|
version = Version.query.filter_by(name='ConfigDB').first()
|
|
|
|
|
|
|
|
# Pre-flight checks
|
|
|
|
if int(version.value) > int(config.SETTINGS_SCHEMA_VERSION):
|
2015-10-20 07:03:18 +00:00
|
|
|
print("""
|
|
|
|
The database schema version is %d, whilst the version required by the \
|
|
|
|
software is %d.
|
2016-03-07 11:48:24 +00:00
|
|
|
Exiting...""" % (version.value, config.SETTINGS_SCHEMA_VERSION))
|
2015-07-22 16:42:39 +00:00
|
|
|
sys.exit(1)
|
|
|
|
elif int(version.value) == int(config.SETTINGS_SCHEMA_VERSION):
|
2015-10-20 07:03:18 +00:00
|
|
|
print("""
|
|
|
|
The database schema version is %d as required.
|
2016-03-07 11:48:24 +00:00
|
|
|
Exiting...""" % (version.value))
|
2015-07-22 16:42:39 +00:00
|
|
|
sys.exit(1)
|
|
|
|
|
2015-10-20 07:03:18 +00:00
|
|
|
print("NOTE: Upgrading database schema from version %d to %d." % (
|
|
|
|
version.value, config.SETTINGS_SCHEMA_VERSION
|
2016-06-21 13:21:06 +00:00
|
|
|
))
|
2015-07-22 16:42:39 +00:00
|
|
|
do_upgrade(app, user_datastore, security, version)
|
|
|
|
else:
|
2016-03-22 15:05:43 +00:00
|
|
|
directory = os.path.dirname(config.SQLITE_PATH)
|
|
|
|
if not os.path.exists(directory):
|
|
|
|
os.makedirs(directory, int('700', 8))
|
|
|
|
db_file = os.open(config.SQLITE_PATH, os.O_CREAT, int('600', 8))
|
|
|
|
os.close(db_file)
|
|
|
|
|
2015-10-20 07:03:18 +00:00
|
|
|
print("""
|
2016-03-07 11:48:24 +00:00
|
|
|
The configuration database - '{0}' does not exist.
|
2015-10-20 07:03:18 +00:00
|
|
|
Entering initial setup mode...""".format(config.SQLITE_PATH))
|
2015-07-22 16:42:39 +00:00
|
|
|
do_setup(app)
|