import os import sys import subprocess import psycopg2 import urllib.request, urllib.error, urllib.parse import re import time ckan_ini = os.environ.get('CKAN_INI', '/srv/app/production.ini') RETRY = 5 def check_db_connection(retry=None): print('[prerun] Start check_db_connection...') if retry is None: retry = RETRY elif retry == 0: print('[prerun] Giving up after 5 tries...') sys.exit(1) conn_str = os.environ.get('CKAN_SQLALCHEMY_URL', '') try: connection = psycopg2.connect(conn_str) except psycopg2.Error as e: print((str(e))) print('[prerun] Unable to connect to the database...try again in a while.') import time time.sleep(10) check_db_connection(retry = retry - 1) else: connection.close() def check_solr_connection(retry=None): print('[prerun] Start check_solr_connection...') if retry is None: retry = RETRY elif retry == 0: print('[prerun] Giving up after 5 tries...') sys.exit(1) url = os.environ.get('CKAN_SOLR_URL', '') search_url = '{url}/select/?q=*&wt=json'.format(url=url) try: connection = urllib.request.urlopen(search_url) except urllib.error.URLError as e: print((str(e))) print('[prerun] Unable to connect to solr...try again in a while.') import time time.sleep(10) check_solr_connection(retry = retry - 1) else: import re conn_info = connection.read() eval(conn_info) def init_db(): print('[prerun] Start init_db...') db_command = ['ckan', '-c', ckan_ini, 'db', 'init'] print('[prerun] Initializing or upgrading db - start using ckan db init') try: # run init scripts subprocess.check_output(db_command, stderr=subprocess.STDOUT) print('[prerun] Initializing or upgrading db - end') except subprocess.CalledProcessError as e: if 'OperationalError' in e.output: print((e.output)) print('[prerun] Database not ready, waiting a bit before exit...') import time time.sleep(5) sys.exit(1) else: print((e.output)) raise e print('[prerun] Initializing or upgrading db - finish') def init_datastore(): conn_str = os.environ.get('CKAN_DATASTORE_WRITE_URL') if not conn_str: print('[prerun] Skipping datastore initialization') return datastore_perms_command = ['ckan', '-c', ckan_ini, 'datastore', 'set-permissions'] connection = psycopg2.connect(conn_str) cursor = connection.cursor() print('[prerun] Initializing datastore db - start') try: datastore_perms = subprocess.Popen( datastore_perms_command, stdout=subprocess.PIPE) perms_sql = datastore_perms.stdout.read() # Remove internal pg command as psycopg2 does not like it perms_sql = re.sub('\\\\connect \"(.*)\"', '', perms_sql.decode('utf-8')) cursor.execute(perms_sql) for notice in connection.notices: print(notice) connection.commit() print('[prerun] Initializing datastore db - end') print((datastore_perms.stdout.read())) except psycopg2.Error as e: print('[prerun] Could not initialize datastore') print((str(e))) except subprocess.CalledProcessError as e: if 'OperationalError' in e.output: print((e.output)) print('[prerun] Database not ready, waiting a bit before exit...') time.sleep(5) sys.exit(1) else: print((e.output)) raise e finally: cursor.close() connection.close() def create_sysadmin(): print('[prerun] Start create_sysadmin...') name = os.environ.get('CKAN_SYSADMIN_NAME') password = os.environ.get('CKAN_SYSADMIN_PASSWORD') email = os.environ.get('CKAN_SYSADMIN_EMAIL') if name and password and email: # Check if user exists command = ['ckan', '-c', ckan_ini, 'user', 'show', name] out = subprocess.check_output(command) if 'User:None' not in re.sub(r'\s', '', out.decode('utf-8')): print('[prerun] Sysadmin user exists, skipping creation') return # Create user command = ['ckan', '-c', ckan_ini, 'user', 'add', name, 'password=' + password, 'email=' + email] subprocess.call(command) print(('[prerun] Created user {0}'.format(name))) # Make it sysadmin command = ['ckan', '-c', ckan_ini, 'sysadmin', 'add', name] subprocess.call(command) print(('[prerun] Made user {0} a sysadmin'.format(name))) if __name__ == '__main__': maintenance = os.environ.get('MAINTENANCE_MODE', '').lower() == 'true' if maintenance: print('[prerun] Maintenance mode, skipping setup...') else: check_db_connection() check_solr_connection() init_db() if os.environ.get('CKAN_DATASTORE_WRITE_URL'): init_datastore() create_sysadmin() #time.sleep(60000) # don't end the prerun script to allow container dock and debug