toolshed/backend/configure.py

234 lines
8.4 KiB
Python
Raw Normal View History

2023-06-11 08:46:11 +00:00
#!/usr/bin/env python3
2023-10-22 20:46:55 +00:00
import json
2023-06-11 08:46:11 +00:00
import os
import sys
2023-06-22 00:14:52 +00:00
from argparse import ArgumentParser
2023-06-11 08:46:11 +00:00
import dotenv
2023-10-22 20:46:55 +00:00
from django.db import transaction, IntegrityError
2023-06-11 08:46:11 +00:00
class CmdCtx:
2023-06-11 08:46:11 +00:00
def __init__(self, args):
self.args = args
2023-06-11 08:46:11 +00:00
def yesno(self, prompt, default=False):
if not sys.stdin.isatty() or self.args.noninteractive:
return default
elif self.args.yes:
2023-06-11 08:46:11 +00:00
return True
elif self.args.no:
2023-06-11 08:46:11 +00:00
return False
yes = {'yes', 'y', 'ye'}
no = {'no', 'n'}
if default:
yes.add('')
2023-06-11 08:46:11 +00:00
else:
no.add('')
hint = ' [Y/n] ' if default else ' [y/N] '
2023-06-11 08:46:11 +00:00
while True:
choice = input(prompt + hint).lower()
if choice in yes:
return True
elif choice in no:
return False
else:
print('Please respond with "yes" or "no"')
2023-06-11 08:46:11 +00:00
def configure(ctx):
2023-06-11 08:46:11 +00:00
if not os.path.exists('.env'):
if not ctx.yesno("the .env file does not exist, do you want to create it?", default=True):
2023-06-11 08:46:11 +00:00
print('Aborting')
exit(0)
if not os.path.exists('.env.dist'):
print('No .env.dist file found')
exit(1)
else:
from shutil import copyfile
copyfile('.env.dist', '.env')
env = dotenv.load_dotenv('.env')
if not env or not os.getenv('SECRET_KEY'):
from django.core.management.utils import get_random_secret_key
print('No SECRET_KEY found in .env file, generating one...')
with open('.env', 'a') as f:
f.write('\nSECRET_KEY=')
f.write(get_random_secret_key())
f.write('\n')
# TODO rename ALLOWED_HOSTS to something more self-explanatory
current_hosts = os.getenv('ALLOWED_HOSTS')
print('Current ALLOWED_HOSTS: {}'.format(current_hosts))
if ctx.yesno("Do you want to add ALLOWED_HOSTS?"):
2023-06-11 08:46:11 +00:00
hosts = input("Enter a comma-separated list of allowed hosts: ")
joined_hosts = current_hosts + ',' + hosts if current_hosts else hosts
dotenv.set_key('.env', 'ALLOWED_HOSTS', joined_hosts)
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "backend.settings")
import django
django.setup()
if not os.path.exists('db.sqlite3'):
if not ctx.yesno("No database found, do you want to create one?", default=True):
2023-06-11 08:46:11 +00:00
print('Aborting')
exit(0)
from django.core.management import call_command
call_command('migrate')
if ctx.yesno("Do you want to create a superuser?"):
2023-06-11 08:46:11 +00:00
from django.core.management import call_command
call_command('createsuperuser')
call_command('collectstatic', '--no-input')
if ctx.yesno("Do you want to import all categories, properties and tags contained in this repository?",
default=True):
2023-10-22 20:46:55 +00:00
from hostadmin.serializers import CategorySerializer, PropertySerializer, TagSerializer
from hostadmin.models import ImportedIdentifierSets
2024-03-11 16:37:56 +00:00
from hashlib import sha256
2023-10-22 20:46:55 +00:00
if not os.path.exists('shared_data'):
os.mkdir('shared_data')
files = os.listdir('shared_data')
idsets = {}
2024-03-11 16:37:56 +00:00
hashes = {}
2023-10-22 20:46:55 +00:00
for file in files:
if file.endswith('.json'):
name = "git:" + file[:-5]
with open('shared_data/' + file, 'r') as f:
try:
idset = json.load(f)
idsets[name] = idset
2024-03-11 16:37:56 +00:00
f.seek(0)
hashes[name] = sha256(f.read().encode()).hexdigest()
2023-10-22 20:46:55 +00:00
except json.decoder.JSONDecodeError:
print('Error: invalid JSON in file {}'.format(file))
imported_sets = ImportedIdentifierSets.objects.all()
for name in [name for name in idsets.keys() if imported_sets.filter(name=name).exists()]:
print('Identifier set {} already imported, skipping'.format(name))
queue = [name for name in idsets.keys() if not imported_sets.filter(name=name).exists()]
while queue:
name = queue.pop(0)
print('Importing {}...'.format(name))
idset = idsets[name]
if 'depends' in idset:
unmet_deps = [dep for dep in idset['depends'] if not imported_sets.filter(name=dep).exists()]
if unmet_deps:
if all([dep in idsets.keys() for dep in unmet_deps]):
2024-03-11 16:37:56 +00:00
if all([dep in queue for dep in unmet_deps]):
print('Not all dependencies for {} are imported, postponing'.format(name))
queue.append(name)
continue
else:
print('Error: unresolvable dependencies for {}: {}'.format(name, unmet_deps))
continue
2023-10-22 20:46:55 +00:00
else:
print('unknown dependencies for {}: {}'.format(name, unmet_deps))
continue
with transaction.atomic():
try:
if 'categories' in idset:
for category in idset['categories']:
serializer = CategorySerializer(data=category)
if serializer.is_valid():
serializer.save(origin=name)
if 'properties' in idset:
for property in idset['properties']:
serializer = PropertySerializer(data=property)
if serializer.is_valid():
serializer.save(origin=name)
if 'tags' in idset:
for tag in idset['tags']:
serializer = TagSerializer(data=tag)
if serializer.is_valid():
serializer.save(origin=name)
2024-03-11 16:37:56 +00:00
imported_sets.create(name=name, hash=hashes[name])
2023-10-22 20:46:55 +00:00
except IntegrityError:
print('Error: integrity error while importing {}\n\tmight be cause by name conflicts with existing'
' categories, properties or tags'.format(name))
2024-03-11 16:37:56 +00:00
transaction.set_rollback(True)
continue
except Exception as e:
print('Error: {}'.format(e))
transaction.set_rollback(True)
2023-10-22 20:46:55 +00:00
continue
2023-06-22 00:14:52 +00:00
def reset():
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "backend.settings")
import django
2023-10-22 20:46:55 +00:00
import shutil
2023-06-22 00:14:52 +00:00
django.setup()
try:
os.remove('db.sqlite3')
except FileNotFoundError:
pass
2023-10-22 20:46:55 +00:00
for file in os.listdir('userfiles'):
try:
shutil.rmtree('userfiles/' + file)
except FileNotFoundError:
pass
2023-06-22 00:14:52 +00:00
os.system("git clean -f */migrations")
from django.core.management import call_command
2023-10-22 20:46:55 +00:00
apps = ['authentication', 'authtoken', 'sessions', 'hostadmin', 'files', 'toolshed', 'admin']
2023-06-22 00:14:52 +00:00
for app in apps:
call_command('makemigrations', app)
def testdata():
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "backend.settings")
import django
django.setup()
if os.path.exists('testdata.py'):
from testdata import create_test_data
create_test_data()
else:
print('No testdata file found')
print('Please create a file named testdata.py in the shared_data directory. the function create_test_data() '
'will be called automatically and should create all necessary test data.')
def main():
parser = ArgumentParser(description='Toolshed Server Configuration')
parser.add_argument('--yes', '-y', help='Answer yes to all questions', action='store_true')
parser.add_argument('--no', '-n', help='Answer no to all questions', action='store_true')
parser.add_argument('--noninteractive', '-x', help="Run in noninteractive mode", action='store_true')
2023-06-22 00:14:52 +00:00
parser.add_argument('cmd', help='Command', default='configure', nargs='?')
args = parser.parse_args()
if args.yes and args.no:
print('Error: --yes and --no are mutually exclusive')
exit(1)
ctx = CmdCtx(args)
2023-06-22 00:14:52 +00:00
if args.cmd == 'configure':
configure(ctx)
2023-06-22 00:14:52 +00:00
elif args.cmd == 'reset':
reset()
elif args.cmd == 'testdata':
testdata()
2024-02-23 17:51:52 +00:00
elif args.cmd == 'migrate':
print('not implemented yet')
2023-06-22 00:14:52 +00:00
else:
print('Unknown command: {}'.format(args.cmd))
exit(1)
2023-06-11 08:46:11 +00:00
if __name__ == '__main__':
2023-06-22 00:14:52 +00:00
main()