blob: d4f185e66a006ad36ddfb5b130092b88889ba67d [file] [log] [blame]
#!/usr/bin/env python3
# Copyright (C) 2020, 2021 Apple Inc. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# 1. Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS "AS IS" AND
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
import argparse
import os
import subprocess
import sys
import time
import threading
import resultsdbpy
from webkitcorepy import AutoInstall
libraries = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
if sys.platform == 'darwin':
is_root = not os.getuid()
does_own_libraries = os.stat(libraries).st_uid == os.getuid()
if (is_root or not does_own_libraries):
libraries = os.path.expanduser('~/Library/webkitpy')
AutoInstall.set_directory(os.path.join(libraries, 'autoinstalled', 'python-{}'.format(sys.version_info[0])))
from resultsdbpy.model.docker import Docker
from multiprocessing import Process
from redis import StrictRedis
class DockerIfNeededContext(object):
def __init__(self, args):
self.docker = Docker.instance() if 'localhost' in [args.cassandra_server, args.redis_server] else None
def __enter__(self):
if self.docker:
self.docker.__enter__()
def __exit__(self, *args, **kwargs):
if self.docker:
self.docker.__exit__(*args, **kwargs)
def has_gunicorn():
return not subprocess.run(['python3', '/usr/local/bin/gunicorns', '--version'], capture_output=True).returncode
def parse_arguments():
parser = argparse.ArgumentParser(description='Run a simple instance of the results database for local development')
parser.add_argument(
'--local-cassandra', help='Use a local Cassandra instance',
dest='cassandra_server', action='store_const', const='localhost',
)
parser.add_argument(
'--mock-cassandra', help='Use a mock Cassandra instance',
dest='cassandra_server', action='store_const', const='mock',
)
parser.set_defaults(cassandra_server='localhost' if Docker.installed() else 'mock')
parser.add_argument(
'--local-redis', help='Use a local Redis instance',
dest='redis_server', action='store_const', const='localhost',
)
parser.add_argument(
'--mock-redis', help='Use a mock Redis instance',
dest='redis_server', action='store_const', const='mock',
)
parser.set_defaults(redis_server='localhost' if Docker.installed() else 'mock')
parser.add_argument(
'--drop-keyspace', help='Drop the existing keyspace before running the program',
dest='drop_keyspace', action='store_true',
)
parser.add_argument(
'--no-drop-keyspace', help='Do NOT drop the existing keyspace before running the program',
dest='drop_keyspace', action='store_false',
)
parser.set_defaults(drop_keyspace=False)
parser.add_argument(
'--clear-redis', help='Clear redis before running the application',
dest='clear_redis', action='store_true',
)
parser.add_argument(
'--no-clear-redis',
help='Do not clear redis before running the application',
dest='clear_redis', action='store_false',
)
parser.set_defaults(clear_redis=True)
parser.add_argument(
'--gunicorn', help='Use the gunicorn webserver',
dest='gunicorn', action='store_true',
)
parser.add_argument(
'--no-gunicorn',
help='Do not use the gunicorn webserver',
dest='gunicorn', action='store_false',
)
parser.set_defaults(gunicorn=has_gunicorn())
return parser.parse_args()
def do_work():
from resultsdbpy.example.worker import main
main()
def web_app():
from resultsdbpy.example.main import main
return main()
def gunicorn():
if not has_gunicorn():
print('gunicorn is not installed, please run `pip3 install gunicorn`')
return 1
# Slight delay so that the worker is started before the webserver
time.sleep(2)
return subprocess.run([
'python3', '/usr/local/bin/gunicorn',
'example.main:app',
'--log-file=-', '--reload',
]).returncode
def main():
args = parse_arguments()
with DockerIfNeededContext(args):
os.environ['CASSANDRA_SERVER'] = args.cassandra_server
os.environ['REDIS_HOST'] = args.redis_server
os.environ['DROP_KEYSPACE'] = '1' if args.drop_keyspace else '0'
os.environ['CREATE_KEYSPACE'] = '1' # Unconditionally create keyspace for testing
if args.clear_redis and args.redis_server != 'mock':
StrictRedis(host=args.redis_server).flushdb()
if args.gunicorn:
if 'mock' in [args.redis_server, args.cassandra_server]:
raise RuntimeError('Cannot run with gunicorn and mock databases, asynchronous workers rely on the real databases')
from resultsdbpy.example.environment import main as environment_main
environment_main()
if not args.gunicorn:
worker = threading.Thread(target=do_work)
worker.daemon = True
worker.start()
return web_app()
worker = Process(target=do_work)
worker.start()
try:
return gunicorn()
finally:
worker.terminate()
worker.join()
if __name__ == '__main__':
sys.exit(main())