From 1f4648463daa5e99b1c507c497a3aa8970e3781f Mon Sep 17 00:00:00 2001 From: William Mueller Date: Fri, 7 Jan 2022 19:57:13 -0500 Subject: [PATCH 01/21] updating for 22 --- .dockerignore | 4 ++++ .gitignore | 6 +++++- Dockerfile | 14 +++++++++++++ Procfile | 3 +-- administration/models.py | 13 ++++++++++-- hoohacks/secret_example.py | 16 --------------- hoohacks/settings.py | 41 ++++++++++++++++++++++++++------------ 7 files changed, 63 insertions(+), 34 deletions(-) create mode 100644 .dockerignore create mode 100644 Dockerfile delete mode 100644 hoohacks/secret_example.py diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 0000000..b9bd98b --- /dev/null +++ b/.dockerignore @@ -0,0 +1,4 @@ +# Elastic Beanstalk Files +.elasticbeanstalk/* +.git +.gitignore \ No newline at end of file diff --git a/.gitignore b/.gitignore index 741b0dd..824e8e3 100644 --- a/.gitignore +++ b/.gitignore @@ -11,4 +11,8 @@ img.png *.sqlite3 dump.rdb *.dump -keys/* \ No newline at end of file +keys/* +# Elastic Beanstalk Files +.elasticbeanstalk/* +!.elasticbeanstalk/*.cfg.yml +!.elasticbeanstalk/*.global.yml \ No newline at end of file diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..68c7505 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,14 @@ +FROM python:3.7-buster + +RUN mkdir -p /opt/app +RUN mkdir -p /opt/app/pip_cache +RUN mkdir -p /opt/app/hackx +COPY requirements.txt /opt/app/ +RUN pip3 install -r /opt/app/requirements.txt --cache-dir /opt/app/pip_cache +COPY . /opt/app/hackx/ +WORKDIR /opt/app/hackx/ +RUN python3 manage.py collectstatic --no-input +# RUN chown -R www-data:www-data /opt/app + +EXPOSE 5000 +ENTRYPOINT ["/opt/app/hackx/prodrunserver.sh"] \ No newline at end of file diff --git a/Procfile b/Procfile index 43494a9..4438a4a 100644 --- a/Procfile +++ b/Procfile @@ -1,3 +1,2 @@ release: python3 manage.py migrate -web: bin/start-pgbouncer-stunnel daphne hoohacks.asgi:application --port $PORT --bind 0.0.0.0 -v2 -worker: python manage.py runworker channels -v2 \ No newline at end of file +web: daphne -b :: -p 5000 hoohacks.asgi:application \ No newline at end of file diff --git a/administration/models.py b/administration/models.py index 75a78b6..d0e5f09 100644 --- a/administration/models.py +++ b/administration/models.py @@ -11,9 +11,18 @@ class Settings(models.Model): APPLICATION_CONFIRMATION_DEADLINE = tz.localize(datetime.datetime(2019, 11, 21, 23, 59, 59, 0)) APPLICATION_CONFIRMATION_DEADLINE_FMT = APPLICATION_CONFIRMATION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") ''' - application_submission_deadline = models.DateTimeField(auto_now_add=True) - application_confirmation_deadline = models.DateTimeField(auto_now_add=True) + + ''' + CODE AS OF 12/21/2021, edited + application_submission_deadline = models.DateTimeField(auto_now_add=True) + application_confirmation_deadline = models.DateTimeField(auto_now_add=True) + judging_deadline = models.DateTimeField(auto_now_add=True) + ''' + + application_submission_deadline = (settings.TZ).localize(datetime.datetime(2022, 1, 31, 23, 59, 59, 0)) + application_confirmation_deadline = (settings.TZ).localize(datetime.datetime(2022, 1, 31, 23, 59, 59, 0)) judging_deadline = models.DateTimeField(auto_now_add=True) + def application_submission_deadline_fmt(self): return self.application_submission_deadline.astimezone(settings.TZ).strftime("%B %d, %Y %I:%M %p %Z") diff --git a/hoohacks/secret_example.py b/hoohacks/secret_example.py deleted file mode 100644 index a754323..0000000 --- a/hoohacks/secret_example.py +++ /dev/null @@ -1,16 +0,0 @@ -ADMIN_PASSWORD = '' -GMAIL_USERNAME = '' -GMAIL_PASSWORD = '' -MENTOR_PASSWORD = '' -VOLUNTEER_PASSWORD = '' -JUDGING_PASSWORD = '' -DROPBOX_ACCESS_TOKEN = '' -SLACK_ENABLED = False -SLACK_API_TOKEN = '' -SENDGRID_HOST_PASSWORD = '' -TEXTING_FROM_EMAIL = '' -TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = '' -TWITTER_CONSUMER_SECRET = '' -TWITTER_ACCESS_TOKEN = '' -TWITTER_ACCESS_TOKEN_SECRET = '' diff --git a/hoohacks/settings.py b/hoohacks/settings.py index d1e518a..7eb1320 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -30,7 +30,7 @@ # SECURITY WARNING: don't run with debug turned on in production -ON_HEROKU = 'ON_HEROKU' in os.environ +ON_HEROKU = 'ON_HEROKU' in os.environ or 'ON_AWS' in os.environ DEBUG = False if ON_HEROKU else True @@ -104,6 +104,18 @@ } } +if 'RDS_HOSTNAME' in os.environ: + DATABASES = { + 'default': { + 'ENGINE': 'django.db.backends.mysql', + 'NAME': os.environ['RDS_DB_NAME'], + 'USER': os.environ['RDS_USERNAME'], + 'PASSWORD': os.environ['RDS_PASSWORD'], + 'HOST': os.environ['RDS_HOSTNAME'], + 'PORT': os.environ['RDS_PORT'], + } + } + # Password validation # https://docs.djangoproject.com/en/2.1/ref/settings/#auth-password-validators @@ -175,21 +187,21 @@ EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') TZ = timezone('US/Eastern') -APPLICATION_SUBMISSION_DEADLINE = TZ.localize(datetime.datetime(2019, 11, 20, 23, 59, 59, 0)) +APPLICATION_SUBMISSION_DEADLINE = TZ.localize(datetime.datetime(2022, 1, 20, 23, 59, 59, 0)) APPLICATION_SUBMISSION_DEADLINE_FMT = APPLICATION_SUBMISSION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") -APPLICATION_CONFIRMATION_DEADLINE = TZ.localize(datetime.datetime(2019, 11, 21, 23, 59, 59, 0)) +APPLICATION_CONFIRMATION_DEADLINE = TZ.localize(datetime.datetime(2022, 1, 21, 23, 59, 59, 0)) APPLICATION_CONFIRMATION_DEADLINE_FMT = APPLICATION_CONFIRMATION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") SCHOOLS = [] -GRADUATION_YEARS = [2021, 2022, 2023, 2024, 2025, 2026] +GRADUATION_YEARS = [2022, 2023, 2024, 2025, 2026, 2027] GRADUATION_YEARS_TITLES = { - "2021": "Senior", - "2022": "Junior", - "2023": "Sophomore", - "2024": "Freshman", - "2025": "High School Senior", - "2026": "High School Junior" + "2022": "Senior", + "2023": "Junior", + "2024": "Sophomore", + "2025": "Freshman", + "2026": "High School Senior", + "2027": "High School Junior" } RACES = ["African American", "Native American", "Asian", "Hispanic", "Native Hawaiian", "White", "Other"] @@ -279,6 +291,11 @@ PROD_URL = os.environ.get('PROD_URL', 'http://localhost:8000/') +if not DEBUG: + SECURE_SSL_REDIRECT = True + SECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https') + + USE_PROD_DB = False if USE_PROD_DB and DEBUG: @@ -302,10 +319,8 @@ dsn=os.environ['SENTRY_DSN'], integrations=[DjangoIntegration()] ) - SECURE_SSL_REDIRECT = True - SECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https') except ImportError: - found = False + pass ALLOWED_HOSTS = [ PROD_URL, From 875436d4a0bb406843d6006f86125c98aa25e121 Mon Sep 17 00:00:00 2001 From: William Mueller Date: Fri, 7 Jan 2022 20:26:08 -0500 Subject: [PATCH 02/21] adding sh file --- prodrunserver.sh | 3 +++ 1 file changed, 3 insertions(+) create mode 100644 prodrunserver.sh diff --git a/prodrunserver.sh b/prodrunserver.sh new file mode 100644 index 0000000..877576e --- /dev/null +++ b/prodrunserver.sh @@ -0,0 +1,3 @@ +#!/bin/bash + +daphne -b :: -p 5000 hoohacks.asgi:application \ No newline at end of file From bd081c9bb1ad6c57dbf57d5a33ee52cd66db9838 Mon Sep 17 00:00:00 2001 From: William Mueller Date: Fri, 7 Jan 2022 21:44:16 -0500 Subject: [PATCH 03/21] updating for 22 --- .dockerignore | 4 ---- .gitignore | 6 +----- Dockerfile | 14 -------------- Procfile | 3 ++- hoohacks/settings.py | 23 ++++------------------- prodrunserver.sh | 3 --- runtime.txt | 2 +- 7 files changed, 8 insertions(+), 47 deletions(-) delete mode 100644 .dockerignore delete mode 100644 Dockerfile delete mode 100644 prodrunserver.sh diff --git a/.dockerignore b/.dockerignore deleted file mode 100644 index b9bd98b..0000000 --- a/.dockerignore +++ /dev/null @@ -1,4 +0,0 @@ -# Elastic Beanstalk Files -.elasticbeanstalk/* -.git -.gitignore \ No newline at end of file diff --git a/.gitignore b/.gitignore index 824e8e3..741b0dd 100644 --- a/.gitignore +++ b/.gitignore @@ -11,8 +11,4 @@ img.png *.sqlite3 dump.rdb *.dump -keys/* -# Elastic Beanstalk Files -.elasticbeanstalk/* -!.elasticbeanstalk/*.cfg.yml -!.elasticbeanstalk/*.global.yml \ No newline at end of file +keys/* \ No newline at end of file diff --git a/Dockerfile b/Dockerfile deleted file mode 100644 index 68c7505..0000000 --- a/Dockerfile +++ /dev/null @@ -1,14 +0,0 @@ -FROM python:3.7-buster - -RUN mkdir -p /opt/app -RUN mkdir -p /opt/app/pip_cache -RUN mkdir -p /opt/app/hackx -COPY requirements.txt /opt/app/ -RUN pip3 install -r /opt/app/requirements.txt --cache-dir /opt/app/pip_cache -COPY . /opt/app/hackx/ -WORKDIR /opt/app/hackx/ -RUN python3 manage.py collectstatic --no-input -# RUN chown -R www-data:www-data /opt/app - -EXPOSE 5000 -ENTRYPOINT ["/opt/app/hackx/prodrunserver.sh"] \ No newline at end of file diff --git a/Procfile b/Procfile index 4438a4a..43494a9 100644 --- a/Procfile +++ b/Procfile @@ -1,2 +1,3 @@ release: python3 manage.py migrate -web: daphne -b :: -p 5000 hoohacks.asgi:application \ No newline at end of file +web: bin/start-pgbouncer-stunnel daphne hoohacks.asgi:application --port $PORT --bind 0.0.0.0 -v2 +worker: python manage.py runworker channels -v2 \ No newline at end of file diff --git a/hoohacks/settings.py b/hoohacks/settings.py index 7eb1320..5bffce6 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -30,7 +30,7 @@ # SECURITY WARNING: don't run with debug turned on in production -ON_HEROKU = 'ON_HEROKU' in os.environ or 'ON_AWS' in os.environ +ON_HEROKU = 'ON_HEROKU' in os.environ DEBUG = False if ON_HEROKU else True @@ -104,18 +104,6 @@ } } -if 'RDS_HOSTNAME' in os.environ: - DATABASES = { - 'default': { - 'ENGINE': 'django.db.backends.mysql', - 'NAME': os.environ['RDS_DB_NAME'], - 'USER': os.environ['RDS_USERNAME'], - 'PASSWORD': os.environ['RDS_PASSWORD'], - 'HOST': os.environ['RDS_HOSTNAME'], - 'PORT': os.environ['RDS_PORT'], - } - } - # Password validation # https://docs.djangoproject.com/en/2.1/ref/settings/#auth-password-validators @@ -291,11 +279,6 @@ PROD_URL = os.environ.get('PROD_URL', 'http://localhost:8000/') -if not DEBUG: - SECURE_SSL_REDIRECT = True - SECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https') - - USE_PROD_DB = False if USE_PROD_DB and DEBUG: @@ -319,8 +302,10 @@ dsn=os.environ['SENTRY_DSN'], integrations=[DjangoIntegration()] ) + SECURE_SSL_REDIRECT = True + SECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https') except ImportError: - pass + found = False ALLOWED_HOSTS = [ PROD_URL, diff --git a/prodrunserver.sh b/prodrunserver.sh deleted file mode 100644 index 877576e..0000000 --- a/prodrunserver.sh +++ /dev/null @@ -1,3 +0,0 @@ -#!/bin/bash - -daphne -b :: -p 5000 hoohacks.asgi:application \ No newline at end of file diff --git a/runtime.txt b/runtime.txt index e0108bf..8234723 100644 --- a/runtime.txt +++ b/runtime.txt @@ -1 +1 @@ -python-3.6.10 \ No newline at end of file +python-3.6.8 \ No newline at end of file From 243c54e303bfffccd5f405c4eda908fae31d96ed Mon Sep 17 00:00:00 2001 From: William Mueller Date: Fri, 7 Jan 2022 21:50:42 -0500 Subject: [PATCH 04/21] updated python version --- runtime.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/runtime.txt b/runtime.txt index 8234723..651f607 100644 --- a/runtime.txt +++ b/runtime.txt @@ -1 +1 @@ -python-3.6.8 \ No newline at end of file +python-3.7.2 \ No newline at end of file From 9d05268b4b075211048a8c727037e7807d476376 Mon Sep 17 00:00:00 2001 From: William Mueller Date: Fri, 7 Jan 2022 21:53:53 -0500 Subject: [PATCH 05/21] updated python version --- runtime.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/runtime.txt b/runtime.txt index 651f607..0c9f406 100644 --- a/runtime.txt +++ b/runtime.txt @@ -1 +1 @@ -python-3.7.2 \ No newline at end of file +python-3.10.1 \ No newline at end of file From 9325aa3cc0f0ed44be9d8324a22de7a1076ff389 Mon Sep 17 00:00:00 2001 From: dylanherbig Date: Thu, 8 Dec 2022 23:59:52 -0500 Subject: [PATCH 06/21] initial commit for new version - changed runtime.txt to reflect current version of python (3.10.2) - removed css mapping - temporarily changed secret key structure, will change later - updated requirements.txt file - changed heroku configuration - changed web procfile to include gunicorn - removed routing.py - modified asgi.py file to import correct package --- Procfile | 2 +- administration/models.py | 15 ++- administration/urls.py | 2 +- applications/urls.py | 2 +- hoohacks/asgi.py | 19 +++- hoohacks/data/{schools.csv => schools.txt} | 2 +- hoohacks/routing.py | 13 --- hoohacks/secret_example.py | 17 ++++ hoohacks/settings.py | 99 ++++++++++++++----- hoohacks/urls.py | 13 +-- judging/urls.py | 2 +- mentors/urls.py | 2 +- requirements.txt | 108 ++++++++++++--------- runtime.txt | 2 +- static/css/bootstrap.min.css | 1 - templates/dashboard.html | 5 +- users/models.py | 2 +- users/urls.py | 2 +- users/views.py | 6 +- 19 files changed, 196 insertions(+), 118 deletions(-) rename hoohacks/data/{schools.csv => schools.txt} (99%) delete mode 100644 hoohacks/routing.py create mode 100644 hoohacks/secret_example.py diff --git a/Procfile b/Procfile index 43494a9..2fcc971 100644 --- a/Procfile +++ b/Procfile @@ -1,3 +1,3 @@ release: python3 manage.py migrate -web: bin/start-pgbouncer-stunnel daphne hoohacks.asgi:application --port $PORT --bind 0.0.0.0 -v2 +web: gunicorn -k uvicorn.workers.UvicornWorker hoohacks.asgi --preload worker: python manage.py runworker channels -v2 \ No newline at end of file diff --git a/administration/models.py b/administration/models.py index d0e5f09..5a44de0 100644 --- a/administration/models.py +++ b/administration/models.py @@ -1,7 +1,6 @@ from django.db import models from django.conf import settings -import datetime -from pytz import timezone +from datetime import datetime class Settings(models.Model): ''' @@ -13,14 +12,14 @@ class Settings(models.Model): ''' ''' - CODE AS OF 12/21/2021, edited - application_submission_deadline = models.DateTimeField(auto_now_add=True) - application_confirmation_deadline = models.DateTimeField(auto_now_add=True) - judging_deadline = models.DateTimeField(auto_now_add=True) + CODE AS OF 12/08/2022, edited + application_submission_deadline = settings.TZ.localize(datetime(2023, 11, 20, 23, 59, 59, 0)) + application_confirmation_deadline = settings.TZ.localize(datetime(2023, 11, 20, 23, 59, 59, 0)) + judging_deadline = models.DateTimeField(auto_now_add=True) ''' - application_submission_deadline = (settings.TZ).localize(datetime.datetime(2022, 1, 31, 23, 59, 59, 0)) - application_confirmation_deadline = (settings.TZ).localize(datetime.datetime(2022, 1, 31, 23, 59, 59, 0)) + application_submission_deadline = models.DateTimeField() + application_confirmation_deadline = models.DateTimeField() judging_deadline = models.DateTimeField(auto_now_add=True) diff --git a/administration/urls.py b/administration/urls.py index 04f3d4a..ef01b90 100644 --- a/administration/urls.py +++ b/administration/urls.py @@ -14,7 +14,7 @@ 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) """ from django.contrib import admin -from django.conf.urls import url, include +from django.conf.urls import include from django.urls import path from . import views diff --git a/applications/urls.py b/applications/urls.py index 906b697..0ac855e 100644 --- a/applications/urls.py +++ b/applications/urls.py @@ -14,7 +14,7 @@ 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) """ from django.contrib import admin -from django.conf.urls import url, include +from django.conf.urls import include from django.urls import path from . import views diff --git a/hoohacks/asgi.py b/hoohacks/asgi.py index 77260ac..2081daf 100644 --- a/hoohacks/asgi.py +++ b/hoohacks/asgi.py @@ -1,7 +1,16 @@ +""" +ASGI config for wordOfmouth project. + +It exposes the ASGI callable as a module-level variable named ``application``. + +For more information on this file, see +https://docs.djangoproject.com/en/3.2/howto/deployment/asgi/ +""" + import os -import django -from channels.routing import get_default_application -os.environ.setdefault("DJANGO_SETTINGS_MODULE", "hoohacks.settings") -django.setup() -application = get_default_application() \ No newline at end of file +from django.core.asgi import get_asgi_application + +os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'hoohacks.settings') + +application = get_asgi_application() \ No newline at end of file diff --git a/hoohacks/data/schools.csv b/hoohacks/data/schools.txt similarity index 99% rename from hoohacks/data/schools.csv rename to hoohacks/data/schools.txt index a28fefb..b0d3e0d 100644 --- a/hoohacks/data/schools.csv +++ b/hoohacks/data/schools.txt @@ -1201,4 +1201,4 @@ Maine South High School American High School Chantilly High School Oakton High School -Other +Other \ No newline at end of file diff --git a/hoohacks/routing.py b/hoohacks/routing.py deleted file mode 100644 index 0f4ea10..0000000 --- a/hoohacks/routing.py +++ /dev/null @@ -1,13 +0,0 @@ -# mysite/routing.py -from channels.auth import AuthMiddlewareStack -from channels.routing import ProtocolTypeRouter, URLRouter -import applications.routing - -application = ProtocolTypeRouter({ - # (http->django views is added by default) - 'websocket': AuthMiddlewareStack( - URLRouter( - applications.routing.websocket_urlpatterns - ) - ), -}) diff --git a/hoohacks/secret_example.py b/hoohacks/secret_example.py new file mode 100644 index 0000000..1f9dd35 --- /dev/null +++ b/hoohacks/secret_example.py @@ -0,0 +1,17 @@ +ADMIN_PASSWORD = 'admin' +GMAIL_USERNAME = 'test@gmail.com' +GMAIL_PASSWORD = 'password' +MENTOR_PASSWORD = 'password' +VOLUNTEER_PASSWORD = 'password' +JUDGING_PASSWORD = 'password' +DROPBOX_ACCESS_TOKEN = 'token' +SLACK_ENABLED = False +SLACK_API_TOKEN = 'token' +SENDGRID_HOST_PASSWORD = 'token' +TEXTING_FROM_EMAIL = 'test@gmail.com' +TWITTER_ENABLED = False +TWITTER_CONSUMER_KEY = 'test' +TWITTER_CONSUMER_SECRET = 'test' +TWITTER_ACCESS_TOKEN = 'token' +TWITTER_ACCESS_TOKEN_SECRET = 'token' +PROD_DB_URL = 'token' \ No newline at end of file diff --git a/hoohacks/settings.py b/hoohacks/settings.py index 5bffce6..d0f44b3 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -144,41 +144,68 @@ LOGIN_URL = '/users/login' LOGIN_REDIRECT_URL = '/dashboard' -ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) -MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) -VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) -JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) -DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) - -SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) +# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) +# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) +# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) +# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) +# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) + +ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') +MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') +VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') +JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') +DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') + +# SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) +# if SLACK_ENABLED == "True": +# SLACK_ENABLED = True +# if SLACK_ENABLED != True: +# SLACK_ENABLED = False +# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) +# SLACK_MENTOR_TICKET_CHANNEL = "" +# if SLACK_ENABLED: +# SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" +# SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" + +SLACK_ENABLED = os.environ.get('SLACK_ENABLED') if SLACK_ENABLED == "True": SLACK_ENABLED = True if SLACK_ENABLED != True: SLACK_ENABLED = False -SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) +SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') SLACK_MENTOR_TICKET_CHANNEL = "" if SLACK_ENABLED: SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) +# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) +# if TWITTER_ENABLED == "True": +# TWITTER_ENABLED = True +# if TWITTER_ENABLED != True: +# TWITTER_ENABLED = False +# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) +# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) +# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) +# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) + +TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') if TWITTER_ENABLED == "True": TWITTER_ENABLED = True if TWITTER_ENABLED != True: TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) -TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) -TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) -TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) +TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') +TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') +TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') +TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') MAX_NUMBER_TICKETS = 2 EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') TZ = timezone('US/Eastern') -APPLICATION_SUBMISSION_DEADLINE = TZ.localize(datetime.datetime(2022, 1, 20, 23, 59, 59, 0)) +APPLICATION_SUBMISSION_DEADLINE = TZ.localize(datetime.datetime(2024, 2, 20, 23, 59, 59, 0)) APPLICATION_SUBMISSION_DEADLINE_FMT = APPLICATION_SUBMISSION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") -APPLICATION_CONFIRMATION_DEADLINE = TZ.localize(datetime.datetime(2022, 1, 21, 23, 59, 59, 0)) +APPLICATION_CONFIRMATION_DEADLINE = TZ.localize(datetime.datetime(2024, 2, 21, 23, 59, 59, 0)) APPLICATION_CONFIRMATION_DEADLINE_FMT = APPLICATION_CONFIRMATION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") SCHOOLS = [] @@ -199,17 +226,19 @@ DIETARY_RESTRICTIONS = ["None", "Vegetarian", "Vegan", "Nut Allergy", "Halal", "Gluten Free", "Other"] CITIES = ["Pittsburgh", "Washington, DC", "Richmond", "NYC", "Charlottesville"] # Test Cities -f = open(BASE_DIR + "/hoohacks/data/schools.csv", "r") -for school in f: - SCHOOLS.append(school.strip()) +with open(BASE_DIR + "/hoohacks/data/schools.txt", encoding='utf8') as f: + for school in f.read(): + SCHOOLS.append(school.strip()) f.close() EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' EMAIL_HOST = 'smtp.gmail.com' EMAIL_USE_TLS = True EMAIL_PORT = 587 -EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) -EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) +# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) +# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) +EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') +EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') CONTACT_EMAIL = "travel@hoohacks.io" @@ -256,17 +285,21 @@ SENDGRID_HOST = 'smtp.sendgrid.net' SENDGRID_PORT = 587 SENDGRID_HOST_USER = 'apikey' -SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) +# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) +SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') + SENDGRID_FROM_EMAIL = 'team@hoohacks.io' TEXTING_ENABLED = True -TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) +# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) +TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') + REQUIRE_EMAIL_VERIFICATION = False -ASGI_APPLICATION = 'hoohacks.routing.application' +WSGI_APPLICATION = 'hoohacks.wsgi.application' CHANNEL_LAYERS = { 'default': { @@ -281,9 +314,12 @@ USE_PROD_DB = False + if USE_PROD_DB and DEBUG: import dj_database_url - DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) + # DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) + DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) + try: # Configure Django App for Heroku. @@ -307,6 +343,21 @@ except ImportError: found = False + +# Activate Django-Heroku. +# Use this code to avoid the psycopg2 / django-heroku error! +# Do NOT import django-heroku above! +# try: +# if 'HEROKU' in os.environ: +# import django_heroku +# django_heroku.settings(locals()) +# if 'DATABASE_URL' in os.environ: +# import dj_database_url +# DATABASES = {'default': dj_database_url.config()} +# except ImportError: +# found = False + + ALLOWED_HOSTS = [ PROD_URL, '*' diff --git a/hoohacks/urls.py b/hoohacks/urls.py index ff7d710..f35c69b 100644 --- a/hoohacks/urls.py +++ b/hoohacks/urls.py @@ -15,7 +15,8 @@ """ from django.contrib import admin from django.urls import path -from django.conf.urls import url, include +from django.conf.urls import include +from django.urls import re_path from django.contrib.auth import views as auth_views import users import applications @@ -29,11 +30,11 @@ urlpatterns = [ path('administration/', admin.site.urls), - url(r'^users/', include('users.urls')), - url(r'^apps/', include('applications.urls')), - url(r'^admin/', include('administration.urls')), - url(r'^mentor/', include('mentors.urls')), - url(r'^judging/', include('judging.urls')), + re_path(r'^users/', include('users.urls')), + re_path(r'^apps/', include('applications.urls')), + re_path(r'^admin/', include('administration.urls')), + re_path(r'^mentor/', include('mentors.urls')), + re_path(r'^judging/', include('judging.urls')), path('', users.views.redirect_dashboard), path('emails/', views.receive_email), path('reset/', users.views.reset_password), diff --git a/judging/urls.py b/judging/urls.py index 3da9a22..444b516 100644 --- a/judging/urls.py +++ b/judging/urls.py @@ -14,7 +14,7 @@ 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) """ from django.contrib import admin -from django.conf.urls import url, include +from django.conf.urls import include from django.urls import path from . import views diff --git a/mentors/urls.py b/mentors/urls.py index a40fdf7..81f6f43 100644 --- a/mentors/urls.py +++ b/mentors/urls.py @@ -14,7 +14,7 @@ 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) """ from django.contrib import admin -from django.conf.urls import url, include +from django.conf.urls import include from django.urls import path from . import views diff --git a/requirements.txt b/requirements.txt index 4623a3c..8746c32 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,55 +1,69 @@ -aiohttp==3.7.3 -asgiref==3.3.1 -async-timeout==3.0.1 -attrs==20.3.0 -autobahn==20.7.1 -Automat==20.2.0 -beautifulsoup4==4.9.3 +aiohttp==3.8.3 +aiosignal==1.2.0 +asgiref==3.5.2 +async-timeout==4.0.2 +attrs==22.1.0 +autobahn==22.7.1 +Automat==22.10.0 +beautifulsoup4==4.11.1 bs4==0.0.1 -certifi==2020.12.5 -cffi==1.14.4 -channels-redis==2.4.0 -channels==2.3.0 -chardet==3.0.4 +certifi==2022.9.24 +cffi==1.15.1 +channels==4.0.0 +channels-redis==4.0.0 +chardet==5.0.0 +charset-normalizer==2.1.1 +colorama==0.4.6 constantly==15.1.0 -cryptography==3.3.2 -daphne==2.3.0 +cryptography==38.0.3 +daphne==4.0.0 DateJS==0.5 -Django==2.2.13 -django-heroku -dj_database_url -dropbox==11.0.0 -hyperlink==20.0.1 -idna==2.10 -incremental==17.5.0 +Deprecated==1.2.13 +dj-database-url==1.0.0 +Django==4.1.3 +django-heroku==0.3.1 +dropbox==11.35.0 +frozenlist==1.3.1 +hyperlink==21.0.0 +idna==3.4 +incremental==22.10.0 JSPy==1.0.0 -multidict==5.1.0 -oauthlib==3.1.0 -Pillow==8.0.1 +msgpack==1.0.4 +multidict==6.0.2 +oauthlib==3.2.2 +packaging==21.3 +Pillow==9.3.0 ply==3.11 +psycopg2==2.9.5 pyasn1==0.4.8 pyasn1-modules==0.2.8 -pycparser==2.20 -PyHamcrest==2.0.2 -pyOpenSSL==20.0.0 +pycparser==2.21 +PyHamcrest==2.0.4 +pyOpenSSL==22.1.0 +pyparsing==3.0.9 PySocks==1.7.1 -python-http-client==3.3.1 -pytz==2020.4 -qrcode==6.1 -requests==2.25.0 -requests-oauthlib==1.3.0 -sendgrid==6.4.8 -service-identity==18.1.0 -six==1.15.0 -slackclient==2.9.3 -soupsieve==2.0.1 -sqlparse==0.4.1 -starkbank-ecdsa==1.1.0 -stone==3.2.1 -tweepy==3.9.0 -Twisted==19.7.0 -txaio==20.4.1 -typing-extensions==3.7.4.3 -urllib3==1.26.2 -yarl==1.6.3 -zope.interface==5.2.0 +python-http-client==3.3.7 +pytz==2022.6 +qrcode==7.3.1 +redis==4.3.4 +requests==2.28.1 +requests-oauthlib==1.3.1 +sendgrid==6.9.7 +service-identity==21.1.0 +six==1.16.0 +slackclient==2.9.4 +soupsieve==2.3.2.post1 +sqlparse==0.4.3 +starkbank-ecdsa==2.2.0 +stone==3.3.1 +tweepy==4.12.0 +txaio==22.2.1 +typing_extensions==4.4.0 +tzdata==2022.6 +urllib3==1.26.12 +uvicorn==0.20.0 +whitenoise==6.2.0 +wrapt==1.14.1 +yarl==1.8.1 +zope.interface==5.5.1 +gunicorn==20.1.0 \ No newline at end of file diff --git a/runtime.txt b/runtime.txt index 0c9f406..e730234 100644 --- a/runtime.txt +++ b/runtime.txt @@ -1 +1 @@ -python-3.10.1 \ No newline at end of file +python-3.10.9 \ No newline at end of file diff --git a/static/css/bootstrap.min.css b/static/css/bootstrap.min.css index e490fd0..2561c55 100644 --- a/static/css/bootstrap.min.css +++ b/static/css/bootstrap.min.css @@ -4,4 +4,3 @@ * Copyright 2011-2018 Twitter, Inc. * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) */:root{--blue:#007bff;--indigo:#6610f2;--purple:#6f42c1;--pink:#e83e8c;--red:#dc3545;--orange:#fd7e14;--yellow:#ffc107;--green:#28a745;--teal:#20c997;--cyan:#17a2b8;--white:#fff;--gray:#6c757d;--gray-dark:#343a40;--primary:#007bff;--secondary:#6c757d;--success:#28a745;--info:#17a2b8;--warning:#ffc107;--danger:#dc3545;--light:#f8f9fa;--dark:#343a40;--breakpoint-xs:0;--breakpoint-sm:576px;--breakpoint-md:768px;--breakpoint-lg:992px;--breakpoint-xl:1200px;--font-family-sans-serif:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";--font-family-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace}*,::after,::before{box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%;-ms-overflow-style:scrollbar;-webkit-tap-highlight-color:transparent}@-ms-viewport{width:device-width}article,aside,dialog,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";font-size:1rem;font-weight:400;line-height:1.5;color:#212529;text-align:left;background-color:#fff}[tabindex="-1"]:focus{outline:0!important}hr{box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem}p{margin-top:0;margin-bottom:1rem}abbr[data-original-title],abbr[title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;border-bottom:0}address{margin-bottom:1rem;font-style:normal;line-height:inherit}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}dfn{font-style:italic}b,strong{font-weight:bolder}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#007bff;text-decoration:none;background-color:transparent;-webkit-text-decoration-skip:objects}a:hover{color:#0056b3;text-decoration:underline}a:not([href]):not([tabindex]){color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus,a:not([href]):not([tabindex]):hover{color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus{outline:0}code,kbd,pre,samp{font-family:monospace,monospace;font-size:1em}pre{margin-top:0;margin-bottom:1rem;overflow:auto;-ms-overflow-style:scrollbar}figure{margin:0 0 1rem}img{vertical-align:middle;border-style:none}svg:not(:root){overflow:hidden}table{border-collapse:collapse}caption{padding-top:.75rem;padding-bottom:.75rem;color:#6c757d;text-align:left;caption-side:bottom}th{text-align:inherit}label{display:inline-block;margin-bottom:.5rem}button{border-radius:0}button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}[type=reset],[type=submit],button,html [type=button]{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{padding:0;border-style:none}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=date],input[type=datetime-local],input[type=month],input[type=time]{-webkit-appearance:listbox}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;max-width:100%;padding:0;margin-bottom:.5rem;font-size:1.5rem;line-height:inherit;color:inherit;white-space:normal}progress{vertical-align:baseline}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:none}[type=search]::-webkit-search-cancel-button,[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item;cursor:pointer}template{display:none}[hidden]{display:none!important}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-bottom:.5rem;font-family:inherit;font-weight:500;line-height:1.2;color:inherit}.h1,h1{font-size:2.5rem}.h2,h2{font-size:2rem}.h3,h3{font-size:1.75rem}.h4,h4{font-size:1.5rem}.h5,h5{font-size:1.25rem}.h6,h6{font-size:1rem}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:6rem;font-weight:300;line-height:1.2}.display-2{font-size:5.5rem;font-weight:300;line-height:1.2}.display-3{font-size:4.5rem;font-weight:300;line-height:1.2}.display-4{font-size:3.5rem;font-weight:300;line-height:1.2}hr{margin-top:1rem;margin-bottom:1rem;border:0;border-top:1px solid rgba(0,0,0,.1)}.small,small{font-size:80%;font-weight:400}.mark,mark{padding:.2em;background-color:#fcf8e3}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:90%;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote-footer{display:block;font-size:80%;color:#6c757d}.blockquote-footer::before{content:"\2014 \00A0"}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:#fff;border:1px solid #dee2e6;border-radius:.25rem;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:90%;color:#6c757d}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace}code{font-size:87.5%;color:#e83e8c;word-break:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:87.5%;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:100%;font-weight:700}pre{display:block;font-size:87.5%;color:#212529}pre code{font-size:inherit;color:inherit;word-break:normal}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:576px){.container{max-width:540px}}@media (min-width:768px){.container{max-width:720px}}@media (min-width:992px){.container{max-width:960px}}@media (min-width:1200px){.container{max-width:1140px}}.container-fluid{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-15px;margin-left:-15px}.no-gutters{margin-right:0;margin-left:0}.no-gutters>.col,.no-gutters>[class*=col-]{padding-right:0;padding-left:0}.col,.col-1,.col-10,.col-11,.col-12,.col-2,.col-3,.col-4,.col-5,.col-6,.col-7,.col-8,.col-9,.col-auto,.col-lg,.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-auto,.col-md,.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-auto,.col-sm,.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-auto,.col-xl,.col-xl-1,.col-xl-10,.col-xl-11,.col-xl-12,.col-xl-2,.col-xl-3,.col-xl-4,.col-xl-5,.col-xl-6,.col-xl-7,.col-xl-8,.col-xl-9,.col-xl-auto{position:relative;width:100%;min-height:1px;padding-right:15px;padding-left:15px}.col{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:none}.col-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-first{-ms-flex-order:-1;order:-1}.order-last{-ms-flex-order:13;order:13}.order-0{-ms-flex-order:0;order:0}.order-1{-ms-flex-order:1;order:1}.order-2{-ms-flex-order:2;order:2}.order-3{-ms-flex-order:3;order:3}.order-4{-ms-flex-order:4;order:4}.order-5{-ms-flex-order:5;order:5}.order-6{-ms-flex-order:6;order:6}.order-7{-ms-flex-order:7;order:7}.order-8{-ms-flex-order:8;order:8}.order-9{-ms-flex-order:9;order:9}.order-10{-ms-flex-order:10;order:10}.order-11{-ms-flex-order:11;order:11}.order-12{-ms-flex-order:12;order:12}.offset-1{margin-left:8.333333%}.offset-2{margin-left:16.666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.333333%}.offset-5{margin-left:41.666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.333333%}.offset-8{margin-left:66.666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.333333%}.offset-11{margin-left:91.666667%}@media (min-width:576px){.col-sm{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-sm-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:none}.col-sm-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-sm-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-sm-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-sm-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-sm-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-sm-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-sm-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-sm-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-sm-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-sm-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-sm-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-sm-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-sm-first{-ms-flex-order:-1;order:-1}.order-sm-last{-ms-flex-order:13;order:13}.order-sm-0{-ms-flex-order:0;order:0}.order-sm-1{-ms-flex-order:1;order:1}.order-sm-2{-ms-flex-order:2;order:2}.order-sm-3{-ms-flex-order:3;order:3}.order-sm-4{-ms-flex-order:4;order:4}.order-sm-5{-ms-flex-order:5;order:5}.order-sm-6{-ms-flex-order:6;order:6}.order-sm-7{-ms-flex-order:7;order:7}.order-sm-8{-ms-flex-order:8;order:8}.order-sm-9{-ms-flex-order:9;order:9}.order-sm-10{-ms-flex-order:10;order:10}.order-sm-11{-ms-flex-order:11;order:11}.order-sm-12{-ms-flex-order:12;order:12}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.333333%}.offset-sm-2{margin-left:16.666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.333333%}.offset-sm-5{margin-left:41.666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.333333%}.offset-sm-8{margin-left:66.666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.333333%}.offset-sm-11{margin-left:91.666667%}}@media (min-width:768px){.col-md{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-md-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:none}.col-md-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-md-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-md-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-md-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-md-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-md-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-md-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-md-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-md-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-md-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-md-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-md-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-md-first{-ms-flex-order:-1;order:-1}.order-md-last{-ms-flex-order:13;order:13}.order-md-0{-ms-flex-order:0;order:0}.order-md-1{-ms-flex-order:1;order:1}.order-md-2{-ms-flex-order:2;order:2}.order-md-3{-ms-flex-order:3;order:3}.order-md-4{-ms-flex-order:4;order:4}.order-md-5{-ms-flex-order:5;order:5}.order-md-6{-ms-flex-order:6;order:6}.order-md-7{-ms-flex-order:7;order:7}.order-md-8{-ms-flex-order:8;order:8}.order-md-9{-ms-flex-order:9;order:9}.order-md-10{-ms-flex-order:10;order:10}.order-md-11{-ms-flex-order:11;order:11}.order-md-12{-ms-flex-order:12;order:12}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.333333%}.offset-md-2{margin-left:16.666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.333333%}.offset-md-5{margin-left:41.666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.333333%}.offset-md-8{margin-left:66.666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.333333%}.offset-md-11{margin-left:91.666667%}}@media (min-width:992px){.col-lg{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-lg-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:none}.col-lg-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-lg-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-lg-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-lg-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-lg-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-lg-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-lg-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-lg-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-lg-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-lg-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-lg-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-lg-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-lg-first{-ms-flex-order:-1;order:-1}.order-lg-last{-ms-flex-order:13;order:13}.order-lg-0{-ms-flex-order:0;order:0}.order-lg-1{-ms-flex-order:1;order:1}.order-lg-2{-ms-flex-order:2;order:2}.order-lg-3{-ms-flex-order:3;order:3}.order-lg-4{-ms-flex-order:4;order:4}.order-lg-5{-ms-flex-order:5;order:5}.order-lg-6{-ms-flex-order:6;order:6}.order-lg-7{-ms-flex-order:7;order:7}.order-lg-8{-ms-flex-order:8;order:8}.order-lg-9{-ms-flex-order:9;order:9}.order-lg-10{-ms-flex-order:10;order:10}.order-lg-11{-ms-flex-order:11;order:11}.order-lg-12{-ms-flex-order:12;order:12}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.333333%}.offset-lg-2{margin-left:16.666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.333333%}.offset-lg-5{margin-left:41.666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.333333%}.offset-lg-8{margin-left:66.666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.333333%}.offset-lg-11{margin-left:91.666667%}}@media (min-width:1200px){.col-xl{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-xl-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:none}.col-xl-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-xl-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-xl-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-xl-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-xl-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-xl-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-xl-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-xl-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-xl-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-xl-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-xl-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-xl-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-xl-first{-ms-flex-order:-1;order:-1}.order-xl-last{-ms-flex-order:13;order:13}.order-xl-0{-ms-flex-order:0;order:0}.order-xl-1{-ms-flex-order:1;order:1}.order-xl-2{-ms-flex-order:2;order:2}.order-xl-3{-ms-flex-order:3;order:3}.order-xl-4{-ms-flex-order:4;order:4}.order-xl-5{-ms-flex-order:5;order:5}.order-xl-6{-ms-flex-order:6;order:6}.order-xl-7{-ms-flex-order:7;order:7}.order-xl-8{-ms-flex-order:8;order:8}.order-xl-9{-ms-flex-order:9;order:9}.order-xl-10{-ms-flex-order:10;order:10}.order-xl-11{-ms-flex-order:11;order:11}.order-xl-12{-ms-flex-order:12;order:12}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.333333%}.offset-xl-2{margin-left:16.666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.333333%}.offset-xl-5{margin-left:41.666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.333333%}.offset-xl-8{margin-left:66.666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.333333%}.offset-xl-11{margin-left:91.666667%}}.table{width:100%;max-width:100%;margin-bottom:1rem;background-color:transparent}.table td,.table th{padding:.75rem;vertical-align:top;border-top:1px solid #dee2e6}.table thead th{vertical-align:bottom;border-bottom:2px solid #dee2e6}.table tbody+tbody{border-top:2px solid #dee2e6}.table .table{background-color:#fff}.table-sm td,.table-sm th{padding:.3rem}.table-bordered{border:1px solid #dee2e6}.table-bordered td,.table-bordered th{border:1px solid #dee2e6}.table-bordered thead td,.table-bordered thead th{border-bottom-width:2px}.table-borderless tbody+tbody,.table-borderless td,.table-borderless th,.table-borderless thead th{border:0}.table-striped tbody tr:nth-of-type(odd){background-color:rgba(0,0,0,.05)}.table-hover tbody tr:hover{background-color:rgba(0,0,0,.075)}.table-primary,.table-primary>td,.table-primary>th{background-color:#b8daff}.table-hover .table-primary:hover{background-color:#9fcdff}.table-hover .table-primary:hover>td,.table-hover .table-primary:hover>th{background-color:#9fcdff}.table-secondary,.table-secondary>td,.table-secondary>th{background-color:#d6d8db}.table-hover .table-secondary:hover{background-color:#c8cbcf}.table-hover .table-secondary:hover>td,.table-hover .table-secondary:hover>th{background-color:#c8cbcf}.table-success,.table-success>td,.table-success>th{background-color:#c3e6cb}.table-hover .table-success:hover{background-color:#b1dfbb}.table-hover .table-success:hover>td,.table-hover .table-success:hover>th{background-color:#b1dfbb}.table-info,.table-info>td,.table-info>th{background-color:#bee5eb}.table-hover .table-info:hover{background-color:#abdde5}.table-hover .table-info:hover>td,.table-hover .table-info:hover>th{background-color:#abdde5}.table-warning,.table-warning>td,.table-warning>th{background-color:#ffeeba}.table-hover .table-warning:hover{background-color:#ffe8a1}.table-hover .table-warning:hover>td,.table-hover .table-warning:hover>th{background-color:#ffe8a1}.table-danger,.table-danger>td,.table-danger>th{background-color:#f5c6cb}.table-hover .table-danger:hover{background-color:#f1b0b7}.table-hover .table-danger:hover>td,.table-hover .table-danger:hover>th{background-color:#f1b0b7}.table-light,.table-light>td,.table-light>th{background-color:#fdfdfe}.table-hover .table-light:hover{background-color:#ececf6}.table-hover .table-light:hover>td,.table-hover .table-light:hover>th{background-color:#ececf6}.table-dark,.table-dark>td,.table-dark>th{background-color:#c6c8ca}.table-hover .table-dark:hover{background-color:#b9bbbe}.table-hover .table-dark:hover>td,.table-hover .table-dark:hover>th{background-color:#b9bbbe}.table-active,.table-active>td,.table-active>th{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover>td,.table-hover .table-active:hover>th{background-color:rgba(0,0,0,.075)}.table .thead-dark th{color:#fff;background-color:#212529;border-color:#32383e}.table .thead-light th{color:#495057;background-color:#e9ecef;border-color:#dee2e6}.table-dark{color:#fff;background-color:#212529}.table-dark td,.table-dark th,.table-dark thead th{border-color:#32383e}.table-dark.table-bordered{border:0}.table-dark.table-striped tbody tr:nth-of-type(odd){background-color:rgba(255,255,255,.05)}.table-dark.table-hover tbody tr:hover{background-color:rgba(255,255,255,.075)}@media (max-width:575.98px){.table-responsive-sm{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-sm>.table-bordered{border:0}}@media (max-width:767.98px){.table-responsive-md{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-md>.table-bordered{border:0}}@media (max-width:991.98px){.table-responsive-lg{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-lg>.table-bordered{border:0}}@media (max-width:1199.98px){.table-responsive-xl{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-xl>.table-bordered{border:0}}.table-responsive{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive>.table-bordered{border:0}.form-control{display:block;width:100%;padding:.375rem .75rem;font-size:1rem;line-height:1.5;color:#495057;background-color:#fff;background-clip:padding-box;border:1px solid #ced4da;border-radius:.25rem;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media screen and (prefers-reduced-motion:reduce){.form-control{transition:none}}.form-control::-ms-expand{background-color:transparent;border:0}.form-control:focus{color:#495057;background-color:#fff;border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.form-control::-webkit-input-placeholder{color:#6c757d;opacity:1}.form-control::-moz-placeholder{color:#6c757d;opacity:1}.form-control:-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::placeholder{color:#6c757d;opacity:1}.form-control:disabled,.form-control[readonly]{background-color:#e9ecef;opacity:1}select.form-control:not([size]):not([multiple]){height:calc(2.25rem + 2px)}select.form-control:focus::-ms-value{color:#495057;background-color:#fff}.form-control-file,.form-control-range{display:block;width:100%}.col-form-label{padding-top:calc(.375rem + 1px);padding-bottom:calc(.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + 1px);padding-bottom:calc(.5rem + 1px);font-size:1.25rem;line-height:1.5}.col-form-label-sm{padding-top:calc(.25rem + 1px);padding-bottom:calc(.25rem + 1px);font-size:.875rem;line-height:1.5}.form-control-plaintext{display:block;width:100%;padding-top:.375rem;padding-bottom:.375rem;margin-bottom:0;line-height:1.5;color:#212529;background-color:transparent;border:solid transparent;border-width:1px 0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm,.input-group-lg>.form-control-plaintext.form-control,.input-group-lg>.input-group-append>.form-control-plaintext.btn,.input-group-lg>.input-group-append>.form-control-plaintext.input-group-text,.input-group-lg>.input-group-prepend>.form-control-plaintext.btn,.input-group-lg>.input-group-prepend>.form-control-plaintext.input-group-text,.input-group-sm>.form-control-plaintext.form-control,.input-group-sm>.input-group-append>.form-control-plaintext.btn,.input-group-sm>.input-group-append>.form-control-plaintext.input-group-text,.input-group-sm>.input-group-prepend>.form-control-plaintext.btn,.input-group-sm>.input-group-prepend>.form-control-plaintext.input-group-text{padding-right:0;padding-left:0}.form-control-sm,.input-group-sm>.form-control,.input-group-sm>.input-group-append>.btn,.input-group-sm>.input-group-append>.input-group-text,.input-group-sm>.input-group-prepend>.btn,.input-group-sm>.input-group-prepend>.input-group-text{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.input-group-sm>.input-group-append>select.btn:not([size]):not([multiple]),.input-group-sm>.input-group-append>select.input-group-text:not([size]):not([multiple]),.input-group-sm>.input-group-prepend>select.btn:not([size]):not([multiple]),.input-group-sm>.input-group-prepend>select.input-group-text:not([size]):not([multiple]),.input-group-sm>select.form-control:not([size]):not([multiple]),select.form-control-sm:not([size]):not([multiple]){height:calc(1.8125rem + 2px)}.form-control-lg,.input-group-lg>.form-control,.input-group-lg>.input-group-append>.btn,.input-group-lg>.input-group-append>.input-group-text,.input-group-lg>.input-group-prepend>.btn,.input-group-lg>.input-group-prepend>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.input-group-lg>.input-group-append>select.btn:not([size]):not([multiple]),.input-group-lg>.input-group-append>select.input-group-text:not([size]):not([multiple]),.input-group-lg>.input-group-prepend>select.btn:not([size]):not([multiple]),.input-group-lg>.input-group-prepend>select.input-group-text:not([size]):not([multiple]),.input-group-lg>select.form-control:not([size]):not([multiple]),select.form-control-lg:not([size]):not([multiple]){height:calc(2.875rem + 2px)}.form-group{margin-bottom:1rem}.form-text{display:block;margin-top:.25rem}.form-row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-5px;margin-left:-5px}.form-row>.col,.form-row>[class*=col-]{padding-right:5px;padding-left:5px}.form-check{position:relative;display:block;padding-left:1.25rem}.form-check-input{position:absolute;margin-top:.3rem;margin-left:-1.25rem}.form-check-input:disabled~.form-check-label{color:#6c757d}.form-check-label{margin-bottom:0}.form-check-inline{display:-ms-inline-flexbox;display:inline-flex;-ms-flex-align:center;align-items:center;padding-left:0;margin-right:.75rem}.form-check-inline .form-check-input{position:static;margin-top:0;margin-right:.3125rem;margin-left:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#28a745}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.5rem;margin-top:.1rem;font-size:.875rem;line-height:1;color:#fff;background-color:rgba(40,167,69,.8);border-radius:.2rem}.custom-select.is-valid,.form-control.is-valid,.was-validated .custom-select:valid,.was-validated .form-control:valid{border-color:#28a745}.custom-select.is-valid:focus,.form-control.is-valid:focus,.was-validated .custom-select:valid:focus,.was-validated .form-control:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.custom-select.is-valid~.valid-feedback,.custom-select.is-valid~.valid-tooltip,.form-control.is-valid~.valid-feedback,.form-control.is-valid~.valid-tooltip,.was-validated .custom-select:valid~.valid-feedback,.was-validated .custom-select:valid~.valid-tooltip,.was-validated .form-control:valid~.valid-feedback,.was-validated .form-control:valid~.valid-tooltip{display:block}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:#28a745}.form-check-input.is-valid~.valid-feedback,.form-check-input.is-valid~.valid-tooltip,.was-validated .form-check-input:valid~.valid-feedback,.was-validated .form-check-input:valid~.valid-tooltip{display:block}.custom-control-input.is-valid~.custom-control-label,.was-validated .custom-control-input:valid~.custom-control-label{color:#28a745}.custom-control-input.is-valid~.custom-control-label::before,.was-validated .custom-control-input:valid~.custom-control-label::before{background-color:#71dd8a}.custom-control-input.is-valid~.valid-feedback,.custom-control-input.is-valid~.valid-tooltip,.was-validated .custom-control-input:valid~.valid-feedback,.was-validated .custom-control-input:valid~.valid-tooltip{display:block}.custom-control-input.is-valid:checked~.custom-control-label::before,.was-validated .custom-control-input:valid:checked~.custom-control-label::before{background-color:#34ce57}.custom-control-input.is-valid:focus~.custom-control-label::before,.was-validated .custom-control-input:valid:focus~.custom-control-label::before{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(40,167,69,.25)}.custom-file-input.is-valid~.custom-file-label,.was-validated .custom-file-input:valid~.custom-file-label{border-color:#28a745}.custom-file-input.is-valid~.custom-file-label::before,.was-validated .custom-file-input:valid~.custom-file-label::before{border-color:inherit}.custom-file-input.is-valid~.valid-feedback,.custom-file-input.is-valid~.valid-tooltip,.was-validated .custom-file-input:valid~.valid-feedback,.was-validated .custom-file-input:valid~.valid-tooltip{display:block}.custom-file-input.is-valid:focus~.custom-file-label,.was-validated .custom-file-input:valid:focus~.custom-file-label{box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#dc3545}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.5rem;margin-top:.1rem;font-size:.875rem;line-height:1;color:#fff;background-color:rgba(220,53,69,.8);border-radius:.2rem}.custom-select.is-invalid,.form-control.is-invalid,.was-validated .custom-select:invalid,.was-validated .form-control:invalid{border-color:#dc3545}.custom-select.is-invalid:focus,.form-control.is-invalid:focus,.was-validated .custom-select:invalid:focus,.was-validated .form-control:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.custom-select.is-invalid~.invalid-feedback,.custom-select.is-invalid~.invalid-tooltip,.form-control.is-invalid~.invalid-feedback,.form-control.is-invalid~.invalid-tooltip,.was-validated .custom-select:invalid~.invalid-feedback,.was-validated .custom-select:invalid~.invalid-tooltip,.was-validated .form-control:invalid~.invalid-feedback,.was-validated .form-control:invalid~.invalid-tooltip{display:block}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:#dc3545}.form-check-input.is-invalid~.invalid-feedback,.form-check-input.is-invalid~.invalid-tooltip,.was-validated .form-check-input:invalid~.invalid-feedback,.was-validated .form-check-input:invalid~.invalid-tooltip{display:block}.custom-control-input.is-invalid~.custom-control-label,.was-validated .custom-control-input:invalid~.custom-control-label{color:#dc3545}.custom-control-input.is-invalid~.custom-control-label::before,.was-validated .custom-control-input:invalid~.custom-control-label::before{background-color:#efa2a9}.custom-control-input.is-invalid~.invalid-feedback,.custom-control-input.is-invalid~.invalid-tooltip,.was-validated .custom-control-input:invalid~.invalid-feedback,.was-validated .custom-control-input:invalid~.invalid-tooltip{display:block}.custom-control-input.is-invalid:checked~.custom-control-label::before,.was-validated .custom-control-input:invalid:checked~.custom-control-label::before{background-color:#e4606d}.custom-control-input.is-invalid:focus~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus~.custom-control-label::before{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(220,53,69,.25)}.custom-file-input.is-invalid~.custom-file-label,.was-validated .custom-file-input:invalid~.custom-file-label{border-color:#dc3545}.custom-file-input.is-invalid~.custom-file-label::before,.was-validated .custom-file-input:invalid~.custom-file-label::before{border-color:inherit}.custom-file-input.is-invalid~.invalid-feedback,.custom-file-input.is-invalid~.invalid-tooltip,.was-validated .custom-file-input:invalid~.invalid-feedback,.was-validated .custom-file-input:invalid~.invalid-tooltip{display:block}.custom-file-input.is-invalid:focus~.custom-file-label,.was-validated .custom-file-input:invalid:focus~.custom-file-label{box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-inline{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center}.form-inline .form-check{width:100%}@media (min-width:576px){.form-inline label{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;margin-bottom:0}.form-inline .form-group{display:-ms-flexbox;display:flex;-ms-flex:0 0 auto;flex:0 0 auto;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center;margin-bottom:0}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-plaintext{display:inline-block}.form-inline .custom-select,.form-inline .input-group{width:auto}.form-inline .form-check{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:auto;padding-left:0}.form-inline .form-check-input{position:relative;margin-top:0;margin-right:.25rem;margin-left:0}.form-inline .custom-control{-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center}.form-inline .custom-control-label{margin-bottom:0}}.btn{display:inline-block;font-weight:400;text-align:center;white-space:nowrap;vertical-align:middle;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;border:1px solid transparent;padding:.375rem .75rem;font-size:1rem;line-height:1.5;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media screen and (prefers-reduced-motion:reduce){.btn{transition:none}}.btn:focus,.btn:hover{text-decoration:none}.btn.focus,.btn:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.btn.disabled,.btn:disabled{opacity:.65}.btn:not(:disabled):not(.disabled){cursor:pointer}.btn:not(:disabled):not(.disabled).active,.btn:not(:disabled):not(.disabled):active{background-image:none}a.btn.disabled,fieldset:disabled a.btn{pointer-events:none}.btn-primary{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:hover{color:#fff;background-color:#0069d9;border-color:#0062cc}.btn-primary.focus,.btn-primary:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-primary.disabled,.btn-primary:disabled{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:not(:disabled):not(.disabled).active,.btn-primary:not(:disabled):not(.disabled):active,.show>.btn-primary.dropdown-toggle{color:#fff;background-color:#0062cc;border-color:#005cbf}.btn-primary:not(:disabled):not(.disabled).active:focus,.btn-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-secondary{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:hover{color:#fff;background-color:#5a6268;border-color:#545b62}.btn-secondary.focus,.btn-secondary:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-secondary.disabled,.btn-secondary:disabled{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:not(:disabled):not(.disabled).active,.btn-secondary:not(:disabled):not(.disabled):active,.show>.btn-secondary.dropdown-toggle{color:#fff;background-color:#545b62;border-color:#4e555b}.btn-secondary:not(:disabled):not(.disabled).active:focus,.btn-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-success{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:hover{color:#fff;background-color:#218838;border-color:#1e7e34}.btn-success.focus,.btn-success:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-success.disabled,.btn-success:disabled{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:not(:disabled):not(.disabled).active,.btn-success:not(:disabled):not(.disabled):active,.show>.btn-success.dropdown-toggle{color:#fff;background-color:#1e7e34;border-color:#1c7430}.btn-success:not(:disabled):not(.disabled).active:focus,.btn-success:not(:disabled):not(.disabled):active:focus,.show>.btn-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-info{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:hover{color:#fff;background-color:#138496;border-color:#117a8b}.btn-info.focus,.btn-info:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-info.disabled,.btn-info:disabled{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:not(:disabled):not(.disabled).active,.btn-info:not(:disabled):not(.disabled):active,.show>.btn-info.dropdown-toggle{color:#fff;background-color:#117a8b;border-color:#10707f}.btn-info:not(:disabled):not(.disabled).active:focus,.btn-info:not(:disabled):not(.disabled):active:focus,.show>.btn-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-warning{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:hover{color:#212529;background-color:#e0a800;border-color:#d39e00}.btn-warning.focus,.btn-warning:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-warning.disabled,.btn-warning:disabled{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:not(:disabled):not(.disabled).active,.btn-warning:not(:disabled):not(.disabled):active,.show>.btn-warning.dropdown-toggle{color:#212529;background-color:#d39e00;border-color:#c69500}.btn-warning:not(:disabled):not(.disabled).active:focus,.btn-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-danger{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:hover{color:#fff;background-color:#c82333;border-color:#bd2130}.btn-danger.focus,.btn-danger:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-danger.disabled,.btn-danger:disabled{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:not(:disabled):not(.disabled).active,.btn-danger:not(:disabled):not(.disabled):active,.show>.btn-danger.dropdown-toggle{color:#fff;background-color:#bd2130;border-color:#b21f2d}.btn-danger:not(:disabled):not(.disabled).active:focus,.btn-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-light{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:hover{color:#212529;background-color:#e2e6ea;border-color:#dae0e5}.btn-light.focus,.btn-light:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-light.disabled,.btn-light:disabled{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:not(:disabled):not(.disabled).active,.btn-light:not(:disabled):not(.disabled):active,.show>.btn-light.dropdown-toggle{color:#212529;background-color:#dae0e5;border-color:#d3d9df}.btn-light:not(:disabled):not(.disabled).active:focus,.btn-light:not(:disabled):not(.disabled):active:focus,.show>.btn-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-dark{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:hover{color:#fff;background-color:#23272b;border-color:#1d2124}.btn-dark.focus,.btn-dark:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-dark.disabled,.btn-dark:disabled{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:not(:disabled):not(.disabled).active,.btn-dark:not(:disabled):not(.disabled):active,.show>.btn-dark.dropdown-toggle{color:#fff;background-color:#1d2124;border-color:#171a1d}.btn-dark:not(:disabled):not(.disabled).active:focus,.btn-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-outline-primary{color:#007bff;background-color:transparent;background-image:none;border-color:#007bff}.btn-outline-primary:hover{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary.focus,.btn-outline-primary:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-primary.disabled,.btn-outline-primary:disabled{color:#007bff;background-color:transparent}.btn-outline-primary:not(:disabled):not(.disabled).active,.btn-outline-primary:not(:disabled):not(.disabled):active,.show>.btn-outline-primary.dropdown-toggle{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary:not(:disabled):not(.disabled).active:focus,.btn-outline-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-secondary{color:#6c757d;background-color:transparent;background-image:none;border-color:#6c757d}.btn-outline-secondary:hover{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary.focus,.btn-outline-secondary:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-secondary.disabled,.btn-outline-secondary:disabled{color:#6c757d;background-color:transparent}.btn-outline-secondary:not(:disabled):not(.disabled).active,.btn-outline-secondary:not(:disabled):not(.disabled):active,.show>.btn-outline-secondary.dropdown-toggle{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary:not(:disabled):not(.disabled).active:focus,.btn-outline-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-success{color:#28a745;background-color:transparent;background-image:none;border-color:#28a745}.btn-outline-success:hover{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success.focus,.btn-outline-success:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-success.disabled,.btn-outline-success:disabled{color:#28a745;background-color:transparent}.btn-outline-success:not(:disabled):not(.disabled).active,.btn-outline-success:not(:disabled):not(.disabled):active,.show>.btn-outline-success.dropdown-toggle{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success:not(:disabled):not(.disabled).active:focus,.btn-outline-success:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-info{color:#17a2b8;background-color:transparent;background-image:none;border-color:#17a2b8}.btn-outline-info:hover{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info.focus,.btn-outline-info:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-info.disabled,.btn-outline-info:disabled{color:#17a2b8;background-color:transparent}.btn-outline-info:not(:disabled):not(.disabled).active,.btn-outline-info:not(:disabled):not(.disabled):active,.show>.btn-outline-info.dropdown-toggle{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info:not(:disabled):not(.disabled).active:focus,.btn-outline-info:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-warning{color:#ffc107;background-color:transparent;background-image:none;border-color:#ffc107}.btn-outline-warning:hover{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning.focus,.btn-outline-warning:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-warning.disabled,.btn-outline-warning:disabled{color:#ffc107;background-color:transparent}.btn-outline-warning:not(:disabled):not(.disabled).active,.btn-outline-warning:not(:disabled):not(.disabled):active,.show>.btn-outline-warning.dropdown-toggle{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning:not(:disabled):not(.disabled).active:focus,.btn-outline-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-danger{color:#dc3545;background-color:transparent;background-image:none;border-color:#dc3545}.btn-outline-danger:hover{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger.focus,.btn-outline-danger:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-danger.disabled,.btn-outline-danger:disabled{color:#dc3545;background-color:transparent}.btn-outline-danger:not(:disabled):not(.disabled).active,.btn-outline-danger:not(:disabled):not(.disabled):active,.show>.btn-outline-danger.dropdown-toggle{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger:not(:disabled):not(.disabled).active:focus,.btn-outline-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-light{color:#f8f9fa;background-color:transparent;background-image:none;border-color:#f8f9fa}.btn-outline-light:hover{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light.focus,.btn-outline-light:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-light.disabled,.btn-outline-light:disabled{color:#f8f9fa;background-color:transparent}.btn-outline-light:not(:disabled):not(.disabled).active,.btn-outline-light:not(:disabled):not(.disabled):active,.show>.btn-outline-light.dropdown-toggle{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:not(:disabled):not(.disabled).active:focus,.btn-outline-light:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-dark{color:#343a40;background-color:transparent;background-image:none;border-color:#343a40}.btn-outline-dark:hover{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark.focus,.btn-outline-dark:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-outline-dark.disabled,.btn-outline-dark:disabled{color:#343a40;background-color:transparent}.btn-outline-dark:not(:disabled):not(.disabled).active,.btn-outline-dark:not(:disabled):not(.disabled):active,.show>.btn-outline-dark.dropdown-toggle{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark:not(:disabled):not(.disabled).active:focus,.btn-outline-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-link{font-weight:400;color:#007bff;background-color:transparent}.btn-link:hover{color:#0056b3;text-decoration:underline;background-color:transparent;border-color:transparent}.btn-link.focus,.btn-link:focus{text-decoration:underline;border-color:transparent;box-shadow:none}.btn-link.disabled,.btn-link:disabled{color:#6c757d;pointer-events:none}.btn-group-lg>.btn,.btn-lg{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.btn-group-sm>.btn,.btn-sm{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:.5rem}input[type=button].btn-block,input[type=reset].btn-block,input[type=submit].btn-block{width:100%}.fade{transition:opacity .15s linear}@media screen and (prefers-reduced-motion:reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{position:relative;height:0;overflow:hidden;transition:height .35s ease}@media screen and (prefers-reduced-motion:reduce){.collapsing{transition:none}}.dropdown,.dropleft,.dropright,.dropup{position:relative}.dropdown-toggle::after{display:inline-block;width:0;height:0;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:10rem;padding:.5rem 0;margin:.125rem 0 0;font-size:1rem;color:#212529;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.15);border-radius:.25rem}.dropdown-menu-right{right:0;left:auto}.dropup .dropdown-menu{top:auto;bottom:100%;margin-top:0;margin-bottom:.125rem}.dropup .dropdown-toggle::after{display:inline-block;width:0;height:0;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-menu{top:0;right:auto;left:100%;margin-top:0;margin-left:.125rem}.dropright .dropdown-toggle::after{display:inline-block;width:0;height:0;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropright .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-toggle::after{vertical-align:0}.dropleft .dropdown-menu{top:0;right:100%;left:auto;margin-top:0;margin-right:.125rem}.dropleft .dropdown-toggle::after{display:inline-block;width:0;height:0;margin-left:.255em;vertical-align:.255em;content:""}.dropleft .dropdown-toggle::after{display:none}.dropleft .dropdown-toggle::before{display:inline-block;width:0;height:0;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropleft .dropdown-toggle:empty::after{margin-left:0}.dropleft .dropdown-toggle::before{vertical-align:0}.dropdown-menu[x-placement^=bottom],.dropdown-menu[x-placement^=left],.dropdown-menu[x-placement^=right],.dropdown-menu[x-placement^=top]{right:auto;bottom:auto}.dropdown-divider{height:0;margin:.5rem 0;overflow:hidden;border-top:1px solid #e9ecef}.dropdown-item{display:block;width:100%;padding:.25rem 1.5rem;clear:both;font-weight:400;color:#212529;text-align:inherit;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:focus,.dropdown-item:hover{color:#16181b;text-decoration:none;background-color:#f8f9fa}.dropdown-item.active,.dropdown-item:active{color:#fff;text-decoration:none;background-color:#007bff}.dropdown-item.disabled,.dropdown-item:disabled{color:#6c757d;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:.5rem 1.5rem;margin-bottom:0;font-size:.875rem;color:#6c757d;white-space:nowrap}.dropdown-item-text{display:block;padding:.25rem 1.5rem;color:#212529}.btn-group,.btn-group-vertical{position:relative;display:-ms-inline-flexbox;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;-ms-flex:0 1 auto;flex:0 1 auto}.btn-group-vertical>.btn:hover,.btn-group>.btn:hover{z-index:1}.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus{z-index:1}.btn-group .btn+.btn,.btn-group .btn+.btn-group,.btn-group .btn-group+.btn,.btn-group .btn-group+.btn-group,.btn-group-vertical .btn+.btn,.btn-group-vertical .btn+.btn-group,.btn-group-vertical .btn-group+.btn,.btn-group-vertical .btn-group+.btn-group{margin-left:-1px}.btn-toolbar{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-pack:start;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group>.btn:first-child{margin-left:0}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropright .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropleft .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:center;justify-content:center}.btn-group-vertical .btn,.btn-group-vertical .btn-group{width:100%}.btn-group-vertical>.btn+.btn,.btn-group-vertical>.btn+.btn-group,.btn-group-vertical>.btn-group+.btn,.btn-group-vertical>.btn-group+.btn-group{margin-top:-1px;margin-left:0}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn:not(:first-child){border-top-left-radius:0;border-top-right-radius:0}.btn-group-toggle>.btn,.btn-group-toggle>.btn-group>.btn{margin-bottom:0}.btn-group-toggle>.btn input[type=checkbox],.btn-group-toggle>.btn input[type=radio],.btn-group-toggle>.btn-group>.btn input[type=checkbox],.btn-group-toggle>.btn-group>.btn input[type=radio]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:stretch;align-items:stretch;width:100%}.input-group>.custom-file,.input-group>.custom-select,.input-group>.form-control{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;width:1%;margin-bottom:0}.input-group>.custom-file:focus,.input-group>.custom-select:focus,.input-group>.form-control:focus{z-index:3}.input-group>.custom-file+.custom-file,.input-group>.custom-file+.custom-select,.input-group>.custom-file+.form-control,.input-group>.custom-select+.custom-file,.input-group>.custom-select+.custom-select,.input-group>.custom-select+.form-control,.input-group>.form-control+.custom-file,.input-group>.form-control+.custom-select,.input-group>.form-control+.form-control{margin-left:-1px}.input-group>.custom-select:not(:last-child),.input-group>.form-control:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-select:not(:first-child),.input-group>.form-control:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.custom-file{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center}.input-group>.custom-file:not(:last-child) .custom-file-label,.input-group>.custom-file:not(:last-child) .custom-file-label::after{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-file:not(:first-child) .custom-file-label,.input-group>.custom-file:not(:first-child) .custom-file-label::after{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-append,.input-group-prepend{display:-ms-flexbox;display:flex}.input-group-append .btn,.input-group-prepend .btn{position:relative;z-index:2}.input-group-append .btn+.btn,.input-group-append .btn+.input-group-text,.input-group-append .input-group-text+.btn,.input-group-append .input-group-text+.input-group-text,.input-group-prepend .btn+.btn,.input-group-prepend .btn+.input-group-text,.input-group-prepend .input-group-text+.btn,.input-group-prepend .input-group-text+.input-group-text{margin-left:-1px}.input-group-prepend{margin-right:-1px}.input-group-append{margin-left:-1px}.input-group-text{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.375rem .75rem;margin-bottom:0;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;text-align:center;white-space:nowrap;background-color:#e9ecef;border:1px solid #ced4da;border-radius:.25rem}.input-group-text input[type=checkbox],.input-group-text input[type=radio]{margin-top:0}.input-group>.input-group-append:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group>.input-group-append:last-child>.input-group-text:not(:last-child),.input-group>.input-group-append:not(:last-child)>.btn,.input-group>.input-group-append:not(:last-child)>.input-group-text,.input-group>.input-group-prepend>.btn,.input-group>.input-group-prepend>.input-group-text{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.input-group-append>.btn,.input-group>.input-group-append>.input-group-text,.input-group>.input-group-prepend:first-child>.btn:not(:first-child),.input-group>.input-group-prepend:first-child>.input-group-text:not(:first-child),.input-group>.input-group-prepend:not(:first-child)>.btn,.input-group>.input-group-prepend:not(:first-child)>.input-group-text{border-top-left-radius:0;border-bottom-left-radius:0}.custom-control{position:relative;display:block;min-height:1.5rem;padding-left:1.5rem}.custom-control-inline{display:-ms-inline-flexbox;display:inline-flex;margin-right:1rem}.custom-control-input{position:absolute;z-index:-1;opacity:0}.custom-control-input:checked~.custom-control-label::before{color:#fff;background-color:#007bff}.custom-control-input:focus~.custom-control-label::before{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-control-input:active~.custom-control-label::before{color:#fff;background-color:#b3d7ff}.custom-control-input:disabled~.custom-control-label{color:#6c757d}.custom-control-input:disabled~.custom-control-label::before{background-color:#e9ecef}.custom-control-label{margin-bottom:0}.custom-control-label::before{position:absolute;top:.25rem;left:0;display:block;width:1rem;height:1rem;pointer-events:none;content:"";-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:#dee2e6}.custom-control-label::after{position:absolute;top:.25rem;left:0;display:block;width:1rem;height:1rem;content:"";background-repeat:no-repeat;background-position:center center;background-size:50% 50%}.custom-checkbox .custom-control-label::before{border-radius:.25rem}.custom-checkbox .custom-control-input:checked~.custom-control-label::before{background-color:#007bff}.custom-checkbox .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3E%3Cpath fill='%23fff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26 2.974 7.25 8 2.193z'/%3E%3C/svg%3E")}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::before{background-color:#007bff}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::after{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 4'%3E%3Cpath stroke='%23fff' d='M0 2h4'/%3E%3C/svg%3E")}.custom-checkbox .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-checkbox .custom-control-input:disabled:indeterminate~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-radio .custom-control-label::before{border-radius:50%}.custom-radio .custom-control-input:checked~.custom-control-label::before{background-color:#007bff}.custom-radio .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3E%3Ccircle r='3' fill='%23fff'/%3E%3C/svg%3E")}.custom-radio .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-select{display:inline-block;width:100%;height:calc(2.25rem + 2px);padding:.375rem 1.75rem .375rem .75rem;line-height:1.5;color:#495057;vertical-align:middle;background:#fff url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3E%3Cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3E%3C/svg%3E") no-repeat right .75rem center;background-size:8px 10px;border:1px solid #ced4da;border-radius:.25rem;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-select:focus{border-color:#80bdff;outline:0;box-shadow:inset 0 1px 2px rgba(0,0,0,.075),0 0 5px rgba(128,189,255,.5)}.custom-select:focus::-ms-value{color:#495057;background-color:#fff}.custom-select[multiple],.custom-select[size]:not([size="1"]){height:auto;padding-right:.75rem;background-image:none}.custom-select:disabled{color:#6c757d;background-color:#e9ecef}.custom-select::-ms-expand{opacity:0}.custom-select-sm{height:calc(1.8125rem + 2px);padding-top:.375rem;padding-bottom:.375rem;font-size:75%}.custom-select-lg{height:calc(2.875rem + 2px);padding-top:.375rem;padding-bottom:.375rem;font-size:125%}.custom-file{position:relative;display:inline-block;width:100%;height:calc(2.25rem + 2px);margin-bottom:0}.custom-file-input{position:relative;z-index:2;width:100%;height:calc(2.25rem + 2px);margin:0;opacity:0}.custom-file-input:focus~.custom-file-label{border-color:#80bdff;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-file-input:focus~.custom-file-label::after{border-color:#80bdff}.custom-file-input:lang(en)~.custom-file-label::after{content:"Browse"}.custom-file-label{position:absolute;top:0;right:0;left:0;z-index:1;height:calc(2.25rem + 2px);padding:.375rem .75rem;line-height:1.5;color:#495057;background-color:#fff;border:1px solid #ced4da;border-radius:.25rem}.custom-file-label::after{position:absolute;top:0;right:0;bottom:0;z-index:3;display:block;height:calc(calc(2.25rem + 2px) - 1px * 2);padding:.375rem .75rem;line-height:1.5;color:#495057;content:"Browse";background-color:#e9ecef;border-left:1px solid #ced4da;border-radius:0 .25rem .25rem 0}.custom-range{width:100%;padding-left:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-range:focus{outline:0}.custom-range::-moz-focus-outer{border:0}.custom-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;background-color:#007bff;border:0;border-radius:1rem;-webkit-appearance:none;appearance:none}.custom-range::-webkit-slider-thumb:focus{outline:0;box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range::-webkit-slider-thumb:active{background-color:#b3d7ff}.custom-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-moz-range-thumb{width:1rem;height:1rem;background-color:#007bff;border:0;border-radius:1rem;-moz-appearance:none;appearance:none}.custom-range::-moz-range-thumb:focus{outline:0;box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range::-moz-range-thumb:active{background-color:#b3d7ff}.custom-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-ms-thumb{width:1rem;height:1rem;background-color:#007bff;border:0;border-radius:1rem;appearance:none}.custom-range::-ms-thumb:focus{outline:0;box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range::-ms-thumb:active{background-color:#b3d7ff}.custom-range::-ms-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:transparent;border-color:transparent;border-width:.5rem}.custom-range::-ms-fill-lower{background-color:#dee2e6;border-radius:1rem}.custom-range::-ms-fill-upper{margin-right:15px;background-color:#dee2e6;border-radius:1rem}.nav{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:.5rem 1rem}.nav-link:focus,.nav-link:hover{text-decoration:none}.nav-link.disabled{color:#6c757d}.nav-tabs{border-bottom:1px solid #dee2e6}.nav-tabs .nav-item{margin-bottom:-1px}.nav-tabs .nav-link{border:1px solid transparent;border-top-left-radius:.25rem;border-top-right-radius:.25rem}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{border-color:#e9ecef #e9ecef #dee2e6}.nav-tabs .nav-link.disabled{color:#6c757d;background-color:transparent;border-color:transparent}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:#495057;background-color:#fff;border-color:#dee2e6 #dee2e6 #fff}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.nav-pills .nav-link{border-radius:.25rem}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:#fff;background-color:#007bff}.nav-fill .nav-item{-ms-flex:1 1 auto;flex:1 1 auto;text-align:center}.nav-justified .nav-item{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;text-align:center}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between;padding:.5rem 1rem}.navbar>.container,.navbar>.container-fluid{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between}.navbar-brand{display:inline-block;padding-top:.3125rem;padding-bottom:.3125rem;margin-right:1rem;font-size:1.25rem;line-height:inherit;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{text-decoration:none}.navbar-nav{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link{padding-right:0;padding-left:0}.navbar-nav .dropdown-menu{position:static;float:none}.navbar-text{display:inline-block;padding-top:.5rem;padding-bottom:.5rem}.navbar-collapse{-ms-flex-preferred-size:100%;flex-basis:100%;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center}.navbar-toggler{padding:.25rem .75rem;font-size:1.25rem;line-height:1;background-color:transparent;border:1px solid transparent;border-radius:.25rem}.navbar-toggler:focus,.navbar-toggler:hover{text-decoration:none}.navbar-toggler:not(:disabled):not(.disabled){cursor:pointer}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;content:"";background:no-repeat center center;background-size:100% 100%}@media (max-width:575.98px){.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:576px){.navbar-expand-sm{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-sm .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-sm .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}}@media (max-width:767.98px){.navbar-expand-md>.container,.navbar-expand-md>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:768px){.navbar-expand-md{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-md .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-md>.container,.navbar-expand-md>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-md .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}}@media (max-width:991.98px){.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:992px){.navbar-expand-lg{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-lg .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-lg .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}}@media (max-width:1199.98px){.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid{padding-right:0;padding-left:0}}@media (min-width:1200px){.navbar-expand-xl{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-xl .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-xl .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}}.navbar-expand{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand>.container,.navbar-expand>.container-fluid{padding-right:0;padding-left:0}.navbar-expand .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand>.container,.navbar-expand>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-light .navbar-brand{color:rgba(0,0,0,.9)}.navbar-light .navbar-brand:focus,.navbar-light .navbar-brand:hover{color:rgba(0,0,0,.9)}.navbar-light .navbar-nav .nav-link{color:rgba(0,0,0,.5)}.navbar-light .navbar-nav .nav-link:focus,.navbar-light .navbar-nav .nav-link:hover{color:rgba(0,0,0,.7)}.navbar-light .navbar-nav .nav-link.disabled{color:rgba(0,0,0,.3)}.navbar-light .navbar-nav .active>.nav-link,.navbar-light .navbar-nav .nav-link.active,.navbar-light .navbar-nav .nav-link.show,.navbar-light .navbar-nav .show>.nav-link{color:rgba(0,0,0,.9)}.navbar-light .navbar-toggler{color:rgba(0,0,0,.5);border-color:rgba(0,0,0,.1)}.navbar-light .navbar-toggler-icon{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg viewBox='0 0 30 30' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath stroke='rgba(0, 0, 0, 0.5)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 7h22M4 15h22M4 23h22'/%3E%3C/svg%3E")}.navbar-light .navbar-text{color:rgba(0,0,0,.5)}.navbar-light .navbar-text a{color:rgba(0,0,0,.9)}.navbar-light .navbar-text a:focus,.navbar-light .navbar-text a:hover{color:rgba(0,0,0,.9)}.navbar-dark .navbar-brand{color:#fff}.navbar-dark .navbar-brand:focus,.navbar-dark .navbar-brand:hover{color:#fff}.navbar-dark .navbar-nav .nav-link{color:rgba(255,255,255,.5)}.navbar-dark .navbar-nav .nav-link:focus,.navbar-dark .navbar-nav .nav-link:hover{color:rgba(255,255,255,.75)}.navbar-dark .navbar-nav .nav-link.disabled{color:rgba(255,255,255,.25)}.navbar-dark .navbar-nav .active>.nav-link,.navbar-dark .navbar-nav .nav-link.active,.navbar-dark .navbar-nav .nav-link.show,.navbar-dark .navbar-nav .show>.nav-link{color:#fff}.navbar-dark .navbar-toggler{color:rgba(255,255,255,.5);border-color:rgba(255,255,255,.1)}.navbar-dark .navbar-toggler-icon{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg viewBox='0 0 30 30' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath stroke='rgba(255, 255, 255, 0.5)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 7h22M4 15h22M4 23h22'/%3E%3C/svg%3E")}.navbar-dark .navbar-text{color:rgba(255,255,255,.5)}.navbar-dark .navbar-text a{color:#fff}.navbar-dark .navbar-text a:focus,.navbar-dark .navbar-text a:hover{color:#fff}.card{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;min-width:0;word-wrap:break-word;background-color:#fff;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}.card>hr{margin-right:0;margin-left:0}.card>.list-group:first-child .list-group-item:first-child{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.card>.list-group:last-child .list-group-item:last-child{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.card-body{-ms-flex:1 1 auto;flex:1 1 auto;padding:1.25rem}.card-title{margin-bottom:.75rem}.card-subtitle{margin-top:-.375rem;margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link:hover{text-decoration:none}.card-link+.card-link{margin-left:1.25rem}.card-header{padding:.75rem 1.25rem;margin-bottom:0;background-color:rgba(0,0,0,.03);border-bottom:1px solid rgba(0,0,0,.125)}.card-header:first-child{border-radius:calc(.25rem - 1px) calc(.25rem - 1px) 0 0}.card-header+.list-group .list-group-item:first-child{border-top:0}.card-footer{padding:.75rem 1.25rem;background-color:rgba(0,0,0,.03);border-top:1px solid rgba(0,0,0,.125)}.card-footer:last-child{border-radius:0 0 calc(.25rem - 1px) calc(.25rem - 1px)}.card-header-tabs{margin-right:-.625rem;margin-bottom:-.75rem;margin-left:-.625rem;border-bottom:0}.card-header-pills{margin-right:-.625rem;margin-left:-.625rem}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:1.25rem}.card-img{width:100%;border-radius:calc(.25rem - 1px)}.card-img-top{width:100%;border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card-img-bottom{width:100%;border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card-deck{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}.card-deck .card{margin-bottom:15px}@media (min-width:576px){.card-deck{-ms-flex-flow:row wrap;flex-flow:row wrap;margin-right:-15px;margin-left:-15px}.card-deck .card{display:-ms-flexbox;display:flex;-ms-flex:1 0 0%;flex:1 0 0%;-ms-flex-direction:column;flex-direction:column;margin-right:15px;margin-bottom:0;margin-left:15px}}.card-group{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column}.card-group>.card{margin-bottom:15px}@media (min-width:576px){.card-group{-ms-flex-flow:row wrap;flex-flow:row wrap}.card-group>.card{-ms-flex:1 0 0%;flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:first-child{border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:first-child .card-header,.card-group>.card:first-child .card-img-top{border-top-right-radius:0}.card-group>.card:first-child .card-footer,.card-group>.card:first-child .card-img-bottom{border-bottom-right-radius:0}.card-group>.card:last-child{border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:last-child .card-header,.card-group>.card:last-child .card-img-top{border-top-left-radius:0}.card-group>.card:last-child .card-footer,.card-group>.card:last-child .card-img-bottom{border-bottom-left-radius:0}.card-group>.card:only-child{border-radius:.25rem}.card-group>.card:only-child .card-header,.card-group>.card:only-child .card-img-top{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.card-group>.card:only-child .card-footer,.card-group>.card:only-child .card-img-bottom{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.card-group>.card:not(:first-child):not(:last-child):not(:only-child){border-radius:0}.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-footer,.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-header,.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-img-bottom,.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-img-top{border-radius:0}}.card-columns .card{margin-bottom:.75rem}@media (min-width:576px){.card-columns{-webkit-column-count:3;-moz-column-count:3;column-count:3;-webkit-column-gap:1.25rem;-moz-column-gap:1.25rem;column-gap:1.25rem;orphans:1;widows:1}.card-columns .card{display:inline-block;width:100%}}.accordion .card:not(:first-of-type):not(:last-of-type){border-bottom:0;border-radius:0}.accordion .card:not(:first-of-type) .card-header:first-child{border-radius:0}.accordion .card:first-of-type{border-bottom:0;border-bottom-right-radius:0;border-bottom-left-radius:0}.accordion .card:last-of-type{border-top-left-radius:0;border-top-right-radius:0}.breadcrumb{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding:.75rem 1rem;margin-bottom:1rem;list-style:none;background-color:#e9ecef;border-radius:.25rem}.breadcrumb-item+.breadcrumb-item{padding-left:.5rem}.breadcrumb-item+.breadcrumb-item::before{display:inline-block;padding-right:.5rem;color:#6c757d;content:"/"}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:underline}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:none}.breadcrumb-item.active{color:#6c757d}.pagination{display:-ms-flexbox;display:flex;padding-left:0;list-style:none;border-radius:.25rem}.page-link{position:relative;display:block;padding:.5rem .75rem;margin-left:-1px;line-height:1.25;color:#007bff;background-color:#fff;border:1px solid #dee2e6}.page-link:hover{z-index:2;color:#0056b3;text-decoration:none;background-color:#e9ecef;border-color:#dee2e6}.page-link:focus{z-index:2;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.page-link:not(:disabled):not(.disabled){cursor:pointer}.page-item:first-child .page-link{margin-left:0;border-top-left-radius:.25rem;border-bottom-left-radius:.25rem}.page-item:last-child .page-link{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.page-item.active .page-link{z-index:1;color:#fff;background-color:#007bff;border-color:#007bff}.page-item.disabled .page-link{color:#6c757d;pointer-events:none;cursor:auto;background-color:#fff;border-color:#dee2e6}.pagination-lg .page-link{padding:.75rem 1.5rem;font-size:1.25rem;line-height:1.5}.pagination-lg .page-item:first-child .page-link{border-top-left-radius:.3rem;border-bottom-left-radius:.3rem}.pagination-lg .page-item:last-child .page-link{border-top-right-radius:.3rem;border-bottom-right-radius:.3rem}.pagination-sm .page-link{padding:.25rem .5rem;font-size:.875rem;line-height:1.5}.pagination-sm .page-item:first-child .page-link{border-top-left-radius:.2rem;border-bottom-left-radius:.2rem}.pagination-sm .page-item:last-child .page-link{border-top-right-radius:.2rem;border-bottom-right-radius:.2rem}.badge{display:inline-block;padding:.25em .4em;font-size:75%;font-weight:700;line-height:1;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25rem}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.badge-pill{padding-right:.6em;padding-left:.6em;border-radius:10rem}.badge-primary{color:#fff;background-color:#007bff}.badge-primary[href]:focus,.badge-primary[href]:hover{color:#fff;text-decoration:none;background-color:#0062cc}.badge-secondary{color:#fff;background-color:#6c757d}.badge-secondary[href]:focus,.badge-secondary[href]:hover{color:#fff;text-decoration:none;background-color:#545b62}.badge-success{color:#fff;background-color:#28a745}.badge-success[href]:focus,.badge-success[href]:hover{color:#fff;text-decoration:none;background-color:#1e7e34}.badge-info{color:#fff;background-color:#17a2b8}.badge-info[href]:focus,.badge-info[href]:hover{color:#fff;text-decoration:none;background-color:#117a8b}.badge-warning{color:#212529;background-color:#ffc107}.badge-warning[href]:focus,.badge-warning[href]:hover{color:#212529;text-decoration:none;background-color:#d39e00}.badge-danger{color:#fff;background-color:#dc3545}.badge-danger[href]:focus,.badge-danger[href]:hover{color:#fff;text-decoration:none;background-color:#bd2130}.badge-light{color:#212529;background-color:#f8f9fa}.badge-light[href]:focus,.badge-light[href]:hover{color:#212529;text-decoration:none;background-color:#dae0e5}.badge-dark{color:#fff;background-color:#343a40}.badge-dark[href]:focus,.badge-dark[href]:hover{color:#fff;text-decoration:none;background-color:#1d2124}.jumbotron{padding:2rem 1rem;margin-bottom:2rem;background-color:#e9ecef;border-radius:.3rem}@media (min-width:576px){.jumbotron{padding:4rem 2rem}}.jumbotron-fluid{padding-right:0;padding-left:0;border-radius:0}.alert{position:relative;padding:.75rem 1.25rem;margin-bottom:1rem;border:1px solid transparent;border-radius:.25rem}.alert-heading{color:inherit}.alert-link{font-weight:700}.alert-dismissible{padding-right:4rem}.alert-dismissible .close{position:absolute;top:0;right:0;padding:.75rem 1.25rem;color:inherit}.alert-primary{color:#004085;background-color:#cce5ff;border-color:#b8daff}.alert-primary hr{border-top-color:#9fcdff}.alert-primary .alert-link{color:#002752}.alert-secondary{color:#383d41;background-color:#e2e3e5;border-color:#d6d8db}.alert-secondary hr{border-top-color:#c8cbcf}.alert-secondary .alert-link{color:#202326}.alert-success{color:#155724;background-color:#d4edda;border-color:#c3e6cb}.alert-success hr{border-top-color:#b1dfbb}.alert-success .alert-link{color:#0b2e13}.alert-info{color:#0c5460;background-color:#d1ecf1;border-color:#bee5eb}.alert-info hr{border-top-color:#abdde5}.alert-info .alert-link{color:#062c33}.alert-warning{color:#856404;background-color:#fff3cd;border-color:#ffeeba}.alert-warning hr{border-top-color:#ffe8a1}.alert-warning .alert-link{color:#533f03}.alert-danger{color:#721c24;background-color:#f8d7da;border-color:#f5c6cb}.alert-danger hr{border-top-color:#f1b0b7}.alert-danger .alert-link{color:#491217}.alert-light{color:#818182;background-color:#fefefe;border-color:#fdfdfe}.alert-light hr{border-top-color:#ececf6}.alert-light .alert-link{color:#686868}.alert-dark{color:#1b1e21;background-color:#d6d8d9;border-color:#c6c8ca}.alert-dark hr{border-top-color:#b9bbbe}.alert-dark .alert-link{color:#040505}@-webkit-keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}.progress{display:-ms-flexbox;display:flex;height:1rem;overflow:hidden;font-size:.75rem;background-color:#e9ecef;border-radius:.25rem}.progress-bar{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;color:#fff;text-align:center;white-space:nowrap;background-color:#007bff;transition:width .6s ease}@media screen and (prefers-reduced-motion:reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:1rem 1rem}.progress-bar-animated{-webkit-animation:progress-bar-stripes 1s linear infinite;animation:progress-bar-stripes 1s linear infinite}.media{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start}.media-body{-ms-flex:1;flex:1}.list-group{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0}.list-group-item-action{width:100%;color:#495057;text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{color:#495057;text-decoration:none;background-color:#f8f9fa}.list-group-item-action:active{color:#212529;background-color:#e9ecef}.list-group-item{position:relative;display:block;padding:.75rem 1.25rem;margin-bottom:-1px;background-color:#fff;border:1px solid rgba(0,0,0,.125)}.list-group-item:first-child{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.list-group-item:focus,.list-group-item:hover{z-index:1;text-decoration:none}.list-group-item.disabled,.list-group-item:disabled{color:#6c757d;background-color:#fff}.list-group-item.active{z-index:2;color:#fff;background-color:#007bff;border-color:#007bff}.list-group-flush .list-group-item{border-right:0;border-left:0;border-radius:0}.list-group-flush:first-child .list-group-item:first-child{border-top:0}.list-group-flush:last-child .list-group-item:last-child{border-bottom:0}.list-group-item-primary{color:#004085;background-color:#b8daff}.list-group-item-primary.list-group-item-action:focus,.list-group-item-primary.list-group-item-action:hover{color:#004085;background-color:#9fcdff}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#004085;border-color:#004085}.list-group-item-secondary{color:#383d41;background-color:#d6d8db}.list-group-item-secondary.list-group-item-action:focus,.list-group-item-secondary.list-group-item-action:hover{color:#383d41;background-color:#c8cbcf}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#383d41;border-color:#383d41}.list-group-item-success{color:#155724;background-color:#c3e6cb}.list-group-item-success.list-group-item-action:focus,.list-group-item-success.list-group-item-action:hover{color:#155724;background-color:#b1dfbb}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#155724;border-color:#155724}.list-group-item-info{color:#0c5460;background-color:#bee5eb}.list-group-item-info.list-group-item-action:focus,.list-group-item-info.list-group-item-action:hover{color:#0c5460;background-color:#abdde5}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#0c5460;border-color:#0c5460}.list-group-item-warning{color:#856404;background-color:#ffeeba}.list-group-item-warning.list-group-item-action:focus,.list-group-item-warning.list-group-item-action:hover{color:#856404;background-color:#ffe8a1}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#856404;border-color:#856404}.list-group-item-danger{color:#721c24;background-color:#f5c6cb}.list-group-item-danger.list-group-item-action:focus,.list-group-item-danger.list-group-item-action:hover{color:#721c24;background-color:#f1b0b7}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#721c24;border-color:#721c24}.list-group-item-light{color:#818182;background-color:#fdfdfe}.list-group-item-light.list-group-item-action:focus,.list-group-item-light.list-group-item-action:hover{color:#818182;background-color:#ececf6}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#818182;border-color:#818182}.list-group-item-dark{color:#1b1e21;background-color:#c6c8ca}.list-group-item-dark.list-group-item-action:focus,.list-group-item-dark.list-group-item-action:hover{color:#1b1e21;background-color:#b9bbbe}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#1b1e21;border-color:#1b1e21}.close{float:right;font-size:1.5rem;font-weight:700;line-height:1;color:#000;text-shadow:0 1px 0 #fff;opacity:.5}.close:focus,.close:hover{color:#000;text-decoration:none;opacity:.75}.close:not(:disabled):not(.disabled){cursor:pointer}button.close{padding:0;background-color:transparent;border:0;-webkit-appearance:none}.modal-open{overflow:hidden}.modal{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1050;display:none;overflow:hidden;outline:0}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal-dialog{position:relative;width:auto;margin:.5rem;pointer-events:none}.modal.fade .modal-dialog{transition:-webkit-transform .3s ease-out;transition:transform .3s ease-out;transition:transform .3s ease-out,-webkit-transform .3s ease-out;-webkit-transform:translate(0,-25%);transform:translate(0,-25%)}@media screen and (prefers-reduced-motion:reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{-webkit-transform:translate(0,0);transform:translate(0,0)}.modal-dialog-centered{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;min-height:calc(100% - (.5rem * 2))}.modal-content{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;width:100%;pointer-events:auto;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem;outline:0}.modal-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1040;background-color:#000}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:.5}.modal-header{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:justify;justify-content:space-between;padding:1rem;border-bottom:1px solid #e9ecef;border-top-left-radius:.3rem;border-top-right-radius:.3rem}.modal-header .close{padding:1rem;margin:-1rem -1rem -1rem auto}.modal-title{margin-bottom:0;line-height:1.5}.modal-body{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;padding:1rem}.modal-footer{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:end;justify-content:flex-end;padding:1rem;border-top:1px solid #e9ecef}.modal-footer>:not(:first-child){margin-left:.25rem}.modal-footer>:not(:last-child){margin-right:.25rem}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width:576px){.modal-dialog{max-width:500px;margin:1.75rem auto}.modal-dialog-centered{min-height:calc(100% - (1.75rem * 2))}.modal-sm{max-width:300px}}@media (min-width:992px){.modal-lg{max-width:800px}}.tooltip{position:absolute;z-index:1070;display:block;margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;opacity:0}.tooltip.show{opacity:.9}.tooltip .arrow{position:absolute;display:block;width:.8rem;height:.4rem}.tooltip .arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[x-placement^=top],.bs-tooltip-top{padding:.4rem 0}.bs-tooltip-auto[x-placement^=top] .arrow,.bs-tooltip-top .arrow{bottom:0}.bs-tooltip-auto[x-placement^=top] .arrow::before,.bs-tooltip-top .arrow::before{top:0;border-width:.4rem .4rem 0;border-top-color:#000}.bs-tooltip-auto[x-placement^=right],.bs-tooltip-right{padding:0 .4rem}.bs-tooltip-auto[x-placement^=right] .arrow,.bs-tooltip-right .arrow{left:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=right] .arrow::before,.bs-tooltip-right .arrow::before{right:0;border-width:.4rem .4rem .4rem 0;border-right-color:#000}.bs-tooltip-auto[x-placement^=bottom],.bs-tooltip-bottom{padding:.4rem 0}.bs-tooltip-auto[x-placement^=bottom] .arrow,.bs-tooltip-bottom .arrow{top:0}.bs-tooltip-auto[x-placement^=bottom] .arrow::before,.bs-tooltip-bottom .arrow::before{bottom:0;border-width:0 .4rem .4rem;border-bottom-color:#000}.bs-tooltip-auto[x-placement^=left],.bs-tooltip-left{padding:0 .4rem}.bs-tooltip-auto[x-placement^=left] .arrow,.bs-tooltip-left .arrow{right:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=left] .arrow::before,.bs-tooltip-left .arrow::before{left:0;border-width:.4rem 0 .4rem .4rem;border-left-color:#000}.tooltip-inner{max-width:200px;padding:.25rem .5rem;color:#fff;text-align:center;background-color:#000;border-radius:.25rem}.popover{position:absolute;top:0;left:0;z-index:1060;display:block;max-width:276px;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem}.popover .arrow{position:absolute;display:block;width:1rem;height:.5rem;margin:0 .3rem}.popover .arrow::after,.popover .arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid}.bs-popover-auto[x-placement^=top],.bs-popover-top{margin-bottom:.5rem}.bs-popover-auto[x-placement^=top] .arrow,.bs-popover-top .arrow{bottom:calc((.5rem + 1px) * -1)}.bs-popover-auto[x-placement^=top] .arrow::after,.bs-popover-auto[x-placement^=top] .arrow::before,.bs-popover-top .arrow::after,.bs-popover-top .arrow::before{border-width:.5rem .5rem 0}.bs-popover-auto[x-placement^=top] .arrow::before,.bs-popover-top .arrow::before{bottom:0;border-top-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=top] .arrow::after,.bs-popover-top .arrow::after{bottom:1px;border-top-color:#fff}.bs-popover-auto[x-placement^=right],.bs-popover-right{margin-left:.5rem}.bs-popover-auto[x-placement^=right] .arrow,.bs-popover-right .arrow{left:calc((.5rem + 1px) * -1);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=right] .arrow::after,.bs-popover-auto[x-placement^=right] .arrow::before,.bs-popover-right .arrow::after,.bs-popover-right .arrow::before{border-width:.5rem .5rem .5rem 0}.bs-popover-auto[x-placement^=right] .arrow::before,.bs-popover-right .arrow::before{left:0;border-right-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=right] .arrow::after,.bs-popover-right .arrow::after{left:1px;border-right-color:#fff}.bs-popover-auto[x-placement^=bottom],.bs-popover-bottom{margin-top:.5rem}.bs-popover-auto[x-placement^=bottom] .arrow,.bs-popover-bottom .arrow{top:calc((.5rem + 1px) * -1)}.bs-popover-auto[x-placement^=bottom] .arrow::after,.bs-popover-auto[x-placement^=bottom] .arrow::before,.bs-popover-bottom .arrow::after,.bs-popover-bottom .arrow::before{border-width:0 .5rem .5rem .5rem}.bs-popover-auto[x-placement^=bottom] .arrow::before,.bs-popover-bottom .arrow::before{top:0;border-bottom-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=bottom] .arrow::after,.bs-popover-bottom .arrow::after{top:1px;border-bottom-color:#fff}.bs-popover-auto[x-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:1rem;margin-left:-.5rem;content:"";border-bottom:1px solid #f7f7f7}.bs-popover-auto[x-placement^=left],.bs-popover-left{margin-right:.5rem}.bs-popover-auto[x-placement^=left] .arrow,.bs-popover-left .arrow{right:calc((.5rem + 1px) * -1);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=left] .arrow::after,.bs-popover-auto[x-placement^=left] .arrow::before,.bs-popover-left .arrow::after,.bs-popover-left .arrow::before{border-width:.5rem 0 .5rem .5rem}.bs-popover-auto[x-placement^=left] .arrow::before,.bs-popover-left .arrow::before{right:0;border-left-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=left] .arrow::after,.bs-popover-left .arrow::after{right:1px;border-left-color:#fff}.popover-header{padding:.5rem .75rem;margin-bottom:0;font-size:1rem;color:inherit;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.popover-header:empty{display:none}.popover-body{padding:.5rem .75rem;color:#212529}.carousel{position:relative}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-item{position:relative;display:none;-ms-flex-align:center;align-items:center;width:100%;transition:-webkit-transform .6s ease;transition:transform .6s ease;transition:transform .6s ease,-webkit-transform .6s ease;-webkit-backface-visibility:hidden;backface-visibility:hidden;-webkit-perspective:1000px;perspective:1000px}@media screen and (prefers-reduced-motion:reduce){.carousel-item{transition:none}}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.carousel-item-next,.carousel-item-prev{position:absolute;top:0}.carousel-item-next.carousel-item-left,.carousel-item-prev.carousel-item-right{-webkit-transform:translateX(0);transform:translateX(0)}@supports ((-webkit-transform-style:preserve-3d) or (transform-style:preserve-3d)){.carousel-item-next.carousel-item-left,.carousel-item-prev.carousel-item-right{-webkit-transform:translate3d(0,0,0);transform:translate3d(0,0,0)}}.active.carousel-item-right,.carousel-item-next{-webkit-transform:translateX(100%);transform:translateX(100%)}@supports ((-webkit-transform-style:preserve-3d) or (transform-style:preserve-3d)){.active.carousel-item-right,.carousel-item-next{-webkit-transform:translate3d(100%,0,0);transform:translate3d(100%,0,0)}}.active.carousel-item-left,.carousel-item-prev{-webkit-transform:translateX(-100%);transform:translateX(-100%)}@supports ((-webkit-transform-style:preserve-3d) or (transform-style:preserve-3d)){.active.carousel-item-left,.carousel-item-prev{-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}}.carousel-fade .carousel-item{opacity:0;transition-duration:.6s;transition-property:opacity}.carousel-fade .carousel-item-next.carousel-item-left,.carousel-fade .carousel-item-prev.carousel-item-right,.carousel-fade .carousel-item.active{opacity:1}.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{opacity:0}.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-prev,.carousel-fade .carousel-item-next,.carousel-fade .carousel-item-prev,.carousel-fade .carousel-item.active{-webkit-transform:translateX(0);transform:translateX(0)}@supports ((-webkit-transform-style:preserve-3d) or (transform-style:preserve-3d)){.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-prev,.carousel-fade .carousel-item-next,.carousel-fade .carousel-item-prev,.carousel-fade .carousel-item.active{-webkit-transform:translate3d(0,0,0);transform:translate3d(0,0,0)}}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:15%;color:#fff;text-align:center;opacity:.5}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:20px;height:20px;background:transparent no-repeat center center;background-size:100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 8 8'%3E%3Cpath d='M5.25 0l-4 4 4 4 1.5-1.5-2.5-2.5 2.5-2.5-1.5-1.5z'/%3E%3C/svg%3E")}.carousel-control-next-icon{background-image:url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 8 8'%3E%3Cpath d='M2.75 0l-1.5 1.5 2.5 2.5-2.5 2.5 1.5 1.5 4-4-4-4z'/%3E%3C/svg%3E")}.carousel-indicators{position:absolute;right:0;bottom:10px;left:0;z-index:15;display:-ms-flexbox;display:flex;-ms-flex-pack:center;justify-content:center;padding-left:0;margin-right:15%;margin-left:15%;list-style:none}.carousel-indicators li{position:relative;-ms-flex:0 1 auto;flex:0 1 auto;width:30px;height:3px;margin-right:3px;margin-left:3px;text-indent:-999px;background-color:rgba(255,255,255,.5)}.carousel-indicators li::before{position:absolute;top:-10px;left:0;display:inline-block;width:100%;height:10px;content:""}.carousel-indicators li::after{position:absolute;bottom:-10px;left:0;display:inline-block;width:100%;height:10px;content:""}.carousel-indicators .active{background-color:#fff}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.bg-primary{background-color:#007bff!important}a.bg-primary:focus,a.bg-primary:hover,button.bg-primary:focus,button.bg-primary:hover{background-color:#0062cc!important}.bg-secondary{background-color:#6c757d!important}a.bg-secondary:focus,a.bg-secondary:hover,button.bg-secondary:focus,button.bg-secondary:hover{background-color:#545b62!important}.bg-success{background-color:#28a745!important}a.bg-success:focus,a.bg-success:hover,button.bg-success:focus,button.bg-success:hover{background-color:#1e7e34!important}.bg-info{background-color:#17a2b8!important}a.bg-info:focus,a.bg-info:hover,button.bg-info:focus,button.bg-info:hover{background-color:#117a8b!important}.bg-warning{background-color:#ffc107!important}a.bg-warning:focus,a.bg-warning:hover,button.bg-warning:focus,button.bg-warning:hover{background-color:#d39e00!important}.bg-danger{background-color:#dc3545!important}a.bg-danger:focus,a.bg-danger:hover,button.bg-danger:focus,button.bg-danger:hover{background-color:#bd2130!important}.bg-light{background-color:#f8f9fa!important}a.bg-light:focus,a.bg-light:hover,button.bg-light:focus,button.bg-light:hover{background-color:#dae0e5!important}.bg-dark{background-color:#343a40!important}a.bg-dark:focus,a.bg-dark:hover,button.bg-dark:focus,button.bg-dark:hover{background-color:#1d2124!important}.bg-white{background-color:#fff!important}.bg-transparent{background-color:transparent!important}.border{border:1px solid #dee2e6!important}.border-top{border-top:1px solid #dee2e6!important}.border-right{border-right:1px solid #dee2e6!important}.border-bottom{border-bottom:1px solid #dee2e6!important}.border-left{border-left:1px solid #dee2e6!important}.border-0{border:0!important}.border-top-0{border-top:0!important}.border-right-0{border-right:0!important}.border-bottom-0{border-bottom:0!important}.border-left-0{border-left:0!important}.border-primary{border-color:#007bff!important}.border-secondary{border-color:#6c757d!important}.border-success{border-color:#28a745!important}.border-info{border-color:#17a2b8!important}.border-warning{border-color:#ffc107!important}.border-danger{border-color:#dc3545!important}.border-light{border-color:#f8f9fa!important}.border-dark{border-color:#343a40!important}.border-white{border-color:#fff!important}.rounded{border-radius:.25rem!important}.rounded-top{border-top-left-radius:.25rem!important;border-top-right-radius:.25rem!important}.rounded-right{border-top-right-radius:.25rem!important;border-bottom-right-radius:.25rem!important}.rounded-bottom{border-bottom-right-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-left{border-top-left-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-circle{border-radius:50%!important}.rounded-0{border-radius:0!important}.clearfix::after{display:block;clear:both;content:""}.d-none{display:none!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:-ms-flexbox!important;display:flex!important}.d-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}@media (min-width:576px){.d-sm-none{display:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:-ms-flexbox!important;display:flex!important}.d-sm-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:768px){.d-md-none{display:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:-ms-flexbox!important;display:flex!important}.d-md-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:992px){.d-lg-none{display:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:-ms-flexbox!important;display:flex!important}.d-lg-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:1200px){.d-xl-none{display:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:-ms-flexbox!important;display:flex!important}.d-xl-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media print{.d-print-none{display:none!important}.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:-ms-flexbox!important;display:flex!important}.d-print-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}.embed-responsive{position:relative;display:block;width:100%;padding:0;overflow:hidden}.embed-responsive::before{display:block;content:""}.embed-responsive .embed-responsive-item,.embed-responsive embed,.embed-responsive iframe,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-21by9::before{padding-top:42.857143%}.embed-responsive-16by9::before{padding-top:56.25%}.embed-responsive-4by3::before{padding-top:75%}.embed-responsive-1by1::before{padding-top:100%}.flex-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-center{-ms-flex-align:center!important;align-items:center!important}.align-items-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}@media (min-width:576px){.flex-sm-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-sm-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-sm-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-sm-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-sm-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-sm-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-sm-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-sm-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-sm-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-sm-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-sm-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-sm-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-sm-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-sm-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-sm-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-sm-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-sm-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-sm-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-sm-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-sm-center{-ms-flex-align:center!important;align-items:center!important}.align-items-sm-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-sm-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-sm-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-sm-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-sm-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-sm-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-sm-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-sm-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-sm-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-sm-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-sm-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-sm-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-sm-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-sm-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:768px){.flex-md-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-md-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-md-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-md-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-md-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-md-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-md-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-md-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-md-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-md-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-md-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-md-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-md-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-md-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-md-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-md-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-md-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-md-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-md-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-md-center{-ms-flex-align:center!important;align-items:center!important}.align-items-md-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-md-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-md-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-md-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-md-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-md-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-md-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-md-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-md-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-md-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-md-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-md-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-md-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-md-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:992px){.flex-lg-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-lg-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-lg-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-lg-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-lg-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-lg-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-lg-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-lg-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-lg-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-lg-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-lg-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-lg-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-lg-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-lg-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-lg-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-lg-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-lg-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-lg-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-lg-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-lg-center{-ms-flex-align:center!important;align-items:center!important}.align-items-lg-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-lg-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-lg-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-lg-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-lg-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-lg-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-lg-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-lg-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-lg-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-lg-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-lg-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-lg-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-lg-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-lg-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:1200px){.flex-xl-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-xl-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-xl-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-xl-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-xl-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-xl-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-xl-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-xl-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-xl-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-xl-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-xl-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-xl-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-xl-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-xl-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-xl-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-xl-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-xl-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-xl-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-xl-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-xl-center{-ms-flex-align:center!important;align-items:center!important}.align-items-xl-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-xl-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-xl-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-xl-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-xl-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-xl-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-xl-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-xl-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-xl-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-xl-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-xl-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-xl-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-xl-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-xl-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}.float-left{float:left!important}.float-right{float:right!important}.float-none{float:none!important}@media (min-width:576px){.float-sm-left{float:left!important}.float-sm-right{float:right!important}.float-sm-none{float:none!important}}@media (min-width:768px){.float-md-left{float:left!important}.float-md-right{float:right!important}.float-md-none{float:none!important}}@media (min-width:992px){.float-lg-left{float:left!important}.float-lg-right{float:right!important}.float-lg-none{float:none!important}}@media (min-width:1200px){.float-xl-left{float:left!important}.float-xl-right{float:right!important}.float-xl-none{float:none!important}}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}@supports ((position:-webkit-sticky) or (position:sticky)){.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}.sr-only{position:absolute;width:1px;height:1px;padding:0;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;overflow:visible;clip:auto;white-space:normal}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075)!important}.shadow{box-shadow:0 .5rem 1rem rgba(0,0,0,.15)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175)!important}.shadow-none{box-shadow:none!important}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mw-100{max-width:100%!important}.mh-100{max-height:100%!important}.m-0{margin:0!important}.mt-0,.my-0{margin-top:0!important}.mr-0,.mx-0{margin-right:0!important}.mb-0,.my-0{margin-bottom:0!important}.ml-0,.mx-0{margin-left:0!important}.m-1{margin:.25rem!important}.mt-1,.my-1{margin-top:.25rem!important}.mr-1,.mx-1{margin-right:.25rem!important}.mb-1,.my-1{margin-bottom:.25rem!important}.ml-1,.mx-1{margin-left:.25rem!important}.m-2{margin:.5rem!important}.mt-2,.my-2{margin-top:.5rem!important}.mr-2,.mx-2{margin-right:.5rem!important}.mb-2,.my-2{margin-bottom:.5rem!important}.ml-2,.mx-2{margin-left:.5rem!important}.m-3{margin:1rem!important}.mt-3,.my-3{margin-top:1rem!important}.mr-3,.mx-3{margin-right:1rem!important}.mb-3,.my-3{margin-bottom:1rem!important}.ml-3,.mx-3{margin-left:1rem!important}.m-4{margin:1.5rem!important}.mt-4,.my-4{margin-top:1.5rem!important}.mr-4,.mx-4{margin-right:1.5rem!important}.mb-4,.my-4{margin-bottom:1.5rem!important}.ml-4,.mx-4{margin-left:1.5rem!important}.m-5{margin:3rem!important}.mt-5,.my-5{margin-top:3rem!important}.mr-5,.mx-5{margin-right:3rem!important}.mb-5,.my-5{margin-bottom:3rem!important}.ml-5,.mx-5{margin-left:3rem!important}.p-0{padding:0!important}.pt-0,.py-0{padding-top:0!important}.pr-0,.px-0{padding-right:0!important}.pb-0,.py-0{padding-bottom:0!important}.pl-0,.px-0{padding-left:0!important}.p-1{padding:.25rem!important}.pt-1,.py-1{padding-top:.25rem!important}.pr-1,.px-1{padding-right:.25rem!important}.pb-1,.py-1{padding-bottom:.25rem!important}.pl-1,.px-1{padding-left:.25rem!important}.p-2{padding:.5rem!important}.pt-2,.py-2{padding-top:.5rem!important}.pr-2,.px-2{padding-right:.5rem!important}.pb-2,.py-2{padding-bottom:.5rem!important}.pl-2,.px-2{padding-left:.5rem!important}.p-3{padding:1rem!important}.pt-3,.py-3{padding-top:1rem!important}.pr-3,.px-3{padding-right:1rem!important}.pb-3,.py-3{padding-bottom:1rem!important}.pl-3,.px-3{padding-left:1rem!important}.p-4{padding:1.5rem!important}.pt-4,.py-4{padding-top:1.5rem!important}.pr-4,.px-4{padding-right:1.5rem!important}.pb-4,.py-4{padding-bottom:1.5rem!important}.pl-4,.px-4{padding-left:1.5rem!important}.p-5{padding:3rem!important}.pt-5,.py-5{padding-top:3rem!important}.pr-5,.px-5{padding-right:3rem!important}.pb-5,.py-5{padding-bottom:3rem!important}.pl-5,.px-5{padding-left:3rem!important}.m-auto{margin:auto!important}.mt-auto,.my-auto{margin-top:auto!important}.mr-auto,.mx-auto{margin-right:auto!important}.mb-auto,.my-auto{margin-bottom:auto!important}.ml-auto,.mx-auto{margin-left:auto!important}@media (min-width:576px){.m-sm-0{margin:0!important}.mt-sm-0,.my-sm-0{margin-top:0!important}.mr-sm-0,.mx-sm-0{margin-right:0!important}.mb-sm-0,.my-sm-0{margin-bottom:0!important}.ml-sm-0,.mx-sm-0{margin-left:0!important}.m-sm-1{margin:.25rem!important}.mt-sm-1,.my-sm-1{margin-top:.25rem!important}.mr-sm-1,.mx-sm-1{margin-right:.25rem!important}.mb-sm-1,.my-sm-1{margin-bottom:.25rem!important}.ml-sm-1,.mx-sm-1{margin-left:.25rem!important}.m-sm-2{margin:.5rem!important}.mt-sm-2,.my-sm-2{margin-top:.5rem!important}.mr-sm-2,.mx-sm-2{margin-right:.5rem!important}.mb-sm-2,.my-sm-2{margin-bottom:.5rem!important}.ml-sm-2,.mx-sm-2{margin-left:.5rem!important}.m-sm-3{margin:1rem!important}.mt-sm-3,.my-sm-3{margin-top:1rem!important}.mr-sm-3,.mx-sm-3{margin-right:1rem!important}.mb-sm-3,.my-sm-3{margin-bottom:1rem!important}.ml-sm-3,.mx-sm-3{margin-left:1rem!important}.m-sm-4{margin:1.5rem!important}.mt-sm-4,.my-sm-4{margin-top:1.5rem!important}.mr-sm-4,.mx-sm-4{margin-right:1.5rem!important}.mb-sm-4,.my-sm-4{margin-bottom:1.5rem!important}.ml-sm-4,.mx-sm-4{margin-left:1.5rem!important}.m-sm-5{margin:3rem!important}.mt-sm-5,.my-sm-5{margin-top:3rem!important}.mr-sm-5,.mx-sm-5{margin-right:3rem!important}.mb-sm-5,.my-sm-5{margin-bottom:3rem!important}.ml-sm-5,.mx-sm-5{margin-left:3rem!important}.p-sm-0{padding:0!important}.pt-sm-0,.py-sm-0{padding-top:0!important}.pr-sm-0,.px-sm-0{padding-right:0!important}.pb-sm-0,.py-sm-0{padding-bottom:0!important}.pl-sm-0,.px-sm-0{padding-left:0!important}.p-sm-1{padding:.25rem!important}.pt-sm-1,.py-sm-1{padding-top:.25rem!important}.pr-sm-1,.px-sm-1{padding-right:.25rem!important}.pb-sm-1,.py-sm-1{padding-bottom:.25rem!important}.pl-sm-1,.px-sm-1{padding-left:.25rem!important}.p-sm-2{padding:.5rem!important}.pt-sm-2,.py-sm-2{padding-top:.5rem!important}.pr-sm-2,.px-sm-2{padding-right:.5rem!important}.pb-sm-2,.py-sm-2{padding-bottom:.5rem!important}.pl-sm-2,.px-sm-2{padding-left:.5rem!important}.p-sm-3{padding:1rem!important}.pt-sm-3,.py-sm-3{padding-top:1rem!important}.pr-sm-3,.px-sm-3{padding-right:1rem!important}.pb-sm-3,.py-sm-3{padding-bottom:1rem!important}.pl-sm-3,.px-sm-3{padding-left:1rem!important}.p-sm-4{padding:1.5rem!important}.pt-sm-4,.py-sm-4{padding-top:1.5rem!important}.pr-sm-4,.px-sm-4{padding-right:1.5rem!important}.pb-sm-4,.py-sm-4{padding-bottom:1.5rem!important}.pl-sm-4,.px-sm-4{padding-left:1.5rem!important}.p-sm-5{padding:3rem!important}.pt-sm-5,.py-sm-5{padding-top:3rem!important}.pr-sm-5,.px-sm-5{padding-right:3rem!important}.pb-sm-5,.py-sm-5{padding-bottom:3rem!important}.pl-sm-5,.px-sm-5{padding-left:3rem!important}.m-sm-auto{margin:auto!important}.mt-sm-auto,.my-sm-auto{margin-top:auto!important}.mr-sm-auto,.mx-sm-auto{margin-right:auto!important}.mb-sm-auto,.my-sm-auto{margin-bottom:auto!important}.ml-sm-auto,.mx-sm-auto{margin-left:auto!important}}@media (min-width:768px){.m-md-0{margin:0!important}.mt-md-0,.my-md-0{margin-top:0!important}.mr-md-0,.mx-md-0{margin-right:0!important}.mb-md-0,.my-md-0{margin-bottom:0!important}.ml-md-0,.mx-md-0{margin-left:0!important}.m-md-1{margin:.25rem!important}.mt-md-1,.my-md-1{margin-top:.25rem!important}.mr-md-1,.mx-md-1{margin-right:.25rem!important}.mb-md-1,.my-md-1{margin-bottom:.25rem!important}.ml-md-1,.mx-md-1{margin-left:.25rem!important}.m-md-2{margin:.5rem!important}.mt-md-2,.my-md-2{margin-top:.5rem!important}.mr-md-2,.mx-md-2{margin-right:.5rem!important}.mb-md-2,.my-md-2{margin-bottom:.5rem!important}.ml-md-2,.mx-md-2{margin-left:.5rem!important}.m-md-3{margin:1rem!important}.mt-md-3,.my-md-3{margin-top:1rem!important}.mr-md-3,.mx-md-3{margin-right:1rem!important}.mb-md-3,.my-md-3{margin-bottom:1rem!important}.ml-md-3,.mx-md-3{margin-left:1rem!important}.m-md-4{margin:1.5rem!important}.mt-md-4,.my-md-4{margin-top:1.5rem!important}.mr-md-4,.mx-md-4{margin-right:1.5rem!important}.mb-md-4,.my-md-4{margin-bottom:1.5rem!important}.ml-md-4,.mx-md-4{margin-left:1.5rem!important}.m-md-5{margin:3rem!important}.mt-md-5,.my-md-5{margin-top:3rem!important}.mr-md-5,.mx-md-5{margin-right:3rem!important}.mb-md-5,.my-md-5{margin-bottom:3rem!important}.ml-md-5,.mx-md-5{margin-left:3rem!important}.p-md-0{padding:0!important}.pt-md-0,.py-md-0{padding-top:0!important}.pr-md-0,.px-md-0{padding-right:0!important}.pb-md-0,.py-md-0{padding-bottom:0!important}.pl-md-0,.px-md-0{padding-left:0!important}.p-md-1{padding:.25rem!important}.pt-md-1,.py-md-1{padding-top:.25rem!important}.pr-md-1,.px-md-1{padding-right:.25rem!important}.pb-md-1,.py-md-1{padding-bottom:.25rem!important}.pl-md-1,.px-md-1{padding-left:.25rem!important}.p-md-2{padding:.5rem!important}.pt-md-2,.py-md-2{padding-top:.5rem!important}.pr-md-2,.px-md-2{padding-right:.5rem!important}.pb-md-2,.py-md-2{padding-bottom:.5rem!important}.pl-md-2,.px-md-2{padding-left:.5rem!important}.p-md-3{padding:1rem!important}.pt-md-3,.py-md-3{padding-top:1rem!important}.pr-md-3,.px-md-3{padding-right:1rem!important}.pb-md-3,.py-md-3{padding-bottom:1rem!important}.pl-md-3,.px-md-3{padding-left:1rem!important}.p-md-4{padding:1.5rem!important}.pt-md-4,.py-md-4{padding-top:1.5rem!important}.pr-md-4,.px-md-4{padding-right:1.5rem!important}.pb-md-4,.py-md-4{padding-bottom:1.5rem!important}.pl-md-4,.px-md-4{padding-left:1.5rem!important}.p-md-5{padding:3rem!important}.pt-md-5,.py-md-5{padding-top:3rem!important}.pr-md-5,.px-md-5{padding-right:3rem!important}.pb-md-5,.py-md-5{padding-bottom:3rem!important}.pl-md-5,.px-md-5{padding-left:3rem!important}.m-md-auto{margin:auto!important}.mt-md-auto,.my-md-auto{margin-top:auto!important}.mr-md-auto,.mx-md-auto{margin-right:auto!important}.mb-md-auto,.my-md-auto{margin-bottom:auto!important}.ml-md-auto,.mx-md-auto{margin-left:auto!important}}@media (min-width:992px){.m-lg-0{margin:0!important}.mt-lg-0,.my-lg-0{margin-top:0!important}.mr-lg-0,.mx-lg-0{margin-right:0!important}.mb-lg-0,.my-lg-0{margin-bottom:0!important}.ml-lg-0,.mx-lg-0{margin-left:0!important}.m-lg-1{margin:.25rem!important}.mt-lg-1,.my-lg-1{margin-top:.25rem!important}.mr-lg-1,.mx-lg-1{margin-right:.25rem!important}.mb-lg-1,.my-lg-1{margin-bottom:.25rem!important}.ml-lg-1,.mx-lg-1{margin-left:.25rem!important}.m-lg-2{margin:.5rem!important}.mt-lg-2,.my-lg-2{margin-top:.5rem!important}.mr-lg-2,.mx-lg-2{margin-right:.5rem!important}.mb-lg-2,.my-lg-2{margin-bottom:.5rem!important}.ml-lg-2,.mx-lg-2{margin-left:.5rem!important}.m-lg-3{margin:1rem!important}.mt-lg-3,.my-lg-3{margin-top:1rem!important}.mr-lg-3,.mx-lg-3{margin-right:1rem!important}.mb-lg-3,.my-lg-3{margin-bottom:1rem!important}.ml-lg-3,.mx-lg-3{margin-left:1rem!important}.m-lg-4{margin:1.5rem!important}.mt-lg-4,.my-lg-4{margin-top:1.5rem!important}.mr-lg-4,.mx-lg-4{margin-right:1.5rem!important}.mb-lg-4,.my-lg-4{margin-bottom:1.5rem!important}.ml-lg-4,.mx-lg-4{margin-left:1.5rem!important}.m-lg-5{margin:3rem!important}.mt-lg-5,.my-lg-5{margin-top:3rem!important}.mr-lg-5,.mx-lg-5{margin-right:3rem!important}.mb-lg-5,.my-lg-5{margin-bottom:3rem!important}.ml-lg-5,.mx-lg-5{margin-left:3rem!important}.p-lg-0{padding:0!important}.pt-lg-0,.py-lg-0{padding-top:0!important}.pr-lg-0,.px-lg-0{padding-right:0!important}.pb-lg-0,.py-lg-0{padding-bottom:0!important}.pl-lg-0,.px-lg-0{padding-left:0!important}.p-lg-1{padding:.25rem!important}.pt-lg-1,.py-lg-1{padding-top:.25rem!important}.pr-lg-1,.px-lg-1{padding-right:.25rem!important}.pb-lg-1,.py-lg-1{padding-bottom:.25rem!important}.pl-lg-1,.px-lg-1{padding-left:.25rem!important}.p-lg-2{padding:.5rem!important}.pt-lg-2,.py-lg-2{padding-top:.5rem!important}.pr-lg-2,.px-lg-2{padding-right:.5rem!important}.pb-lg-2,.py-lg-2{padding-bottom:.5rem!important}.pl-lg-2,.px-lg-2{padding-left:.5rem!important}.p-lg-3{padding:1rem!important}.pt-lg-3,.py-lg-3{padding-top:1rem!important}.pr-lg-3,.px-lg-3{padding-right:1rem!important}.pb-lg-3,.py-lg-3{padding-bottom:1rem!important}.pl-lg-3,.px-lg-3{padding-left:1rem!important}.p-lg-4{padding:1.5rem!important}.pt-lg-4,.py-lg-4{padding-top:1.5rem!important}.pr-lg-4,.px-lg-4{padding-right:1.5rem!important}.pb-lg-4,.py-lg-4{padding-bottom:1.5rem!important}.pl-lg-4,.px-lg-4{padding-left:1.5rem!important}.p-lg-5{padding:3rem!important}.pt-lg-5,.py-lg-5{padding-top:3rem!important}.pr-lg-5,.px-lg-5{padding-right:3rem!important}.pb-lg-5,.py-lg-5{padding-bottom:3rem!important}.pl-lg-5,.px-lg-5{padding-left:3rem!important}.m-lg-auto{margin:auto!important}.mt-lg-auto,.my-lg-auto{margin-top:auto!important}.mr-lg-auto,.mx-lg-auto{margin-right:auto!important}.mb-lg-auto,.my-lg-auto{margin-bottom:auto!important}.ml-lg-auto,.mx-lg-auto{margin-left:auto!important}}@media (min-width:1200px){.m-xl-0{margin:0!important}.mt-xl-0,.my-xl-0{margin-top:0!important}.mr-xl-0,.mx-xl-0{margin-right:0!important}.mb-xl-0,.my-xl-0{margin-bottom:0!important}.ml-xl-0,.mx-xl-0{margin-left:0!important}.m-xl-1{margin:.25rem!important}.mt-xl-1,.my-xl-1{margin-top:.25rem!important}.mr-xl-1,.mx-xl-1{margin-right:.25rem!important}.mb-xl-1,.my-xl-1{margin-bottom:.25rem!important}.ml-xl-1,.mx-xl-1{margin-left:.25rem!important}.m-xl-2{margin:.5rem!important}.mt-xl-2,.my-xl-2{margin-top:.5rem!important}.mr-xl-2,.mx-xl-2{margin-right:.5rem!important}.mb-xl-2,.my-xl-2{margin-bottom:.5rem!important}.ml-xl-2,.mx-xl-2{margin-left:.5rem!important}.m-xl-3{margin:1rem!important}.mt-xl-3,.my-xl-3{margin-top:1rem!important}.mr-xl-3,.mx-xl-3{margin-right:1rem!important}.mb-xl-3,.my-xl-3{margin-bottom:1rem!important}.ml-xl-3,.mx-xl-3{margin-left:1rem!important}.m-xl-4{margin:1.5rem!important}.mt-xl-4,.my-xl-4{margin-top:1.5rem!important}.mr-xl-4,.mx-xl-4{margin-right:1.5rem!important}.mb-xl-4,.my-xl-4{margin-bottom:1.5rem!important}.ml-xl-4,.mx-xl-4{margin-left:1.5rem!important}.m-xl-5{margin:3rem!important}.mt-xl-5,.my-xl-5{margin-top:3rem!important}.mr-xl-5,.mx-xl-5{margin-right:3rem!important}.mb-xl-5,.my-xl-5{margin-bottom:3rem!important}.ml-xl-5,.mx-xl-5{margin-left:3rem!important}.p-xl-0{padding:0!important}.pt-xl-0,.py-xl-0{padding-top:0!important}.pr-xl-0,.px-xl-0{padding-right:0!important}.pb-xl-0,.py-xl-0{padding-bottom:0!important}.pl-xl-0,.px-xl-0{padding-left:0!important}.p-xl-1{padding:.25rem!important}.pt-xl-1,.py-xl-1{padding-top:.25rem!important}.pr-xl-1,.px-xl-1{padding-right:.25rem!important}.pb-xl-1,.py-xl-1{padding-bottom:.25rem!important}.pl-xl-1,.px-xl-1{padding-left:.25rem!important}.p-xl-2{padding:.5rem!important}.pt-xl-2,.py-xl-2{padding-top:.5rem!important}.pr-xl-2,.px-xl-2{padding-right:.5rem!important}.pb-xl-2,.py-xl-2{padding-bottom:.5rem!important}.pl-xl-2,.px-xl-2{padding-left:.5rem!important}.p-xl-3{padding:1rem!important}.pt-xl-3,.py-xl-3{padding-top:1rem!important}.pr-xl-3,.px-xl-3{padding-right:1rem!important}.pb-xl-3,.py-xl-3{padding-bottom:1rem!important}.pl-xl-3,.px-xl-3{padding-left:1rem!important}.p-xl-4{padding:1.5rem!important}.pt-xl-4,.py-xl-4{padding-top:1.5rem!important}.pr-xl-4,.px-xl-4{padding-right:1.5rem!important}.pb-xl-4,.py-xl-4{padding-bottom:1.5rem!important}.pl-xl-4,.px-xl-4{padding-left:1.5rem!important}.p-xl-5{padding:3rem!important}.pt-xl-5,.py-xl-5{padding-top:3rem!important}.pr-xl-5,.px-xl-5{padding-right:3rem!important}.pb-xl-5,.py-xl-5{padding-bottom:3rem!important}.pl-xl-5,.px-xl-5{padding-left:3rem!important}.m-xl-auto{margin:auto!important}.mt-xl-auto,.my-xl-auto{margin-top:auto!important}.mr-xl-auto,.mx-xl-auto{margin-right:auto!important}.mb-xl-auto,.my-xl-auto{margin-bottom:auto!important}.ml-xl-auto,.mx-xl-auto{margin-left:auto!important}}.text-monospace{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace}.text-justify{text-align:justify!important}.text-nowrap{white-space:nowrap!important}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.text-left{text-align:left!important}.text-right{text-align:right!important}.text-center{text-align:center!important}@media (min-width:576px){.text-sm-left{text-align:left!important}.text-sm-right{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.text-md-left{text-align:left!important}.text-md-right{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.text-lg-left{text-align:left!important}.text-lg-right{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.text-xl-left{text-align:left!important}.text-xl-right{text-align:right!important}.text-xl-center{text-align:center!important}}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.font-weight-light{font-weight:300!important}.font-weight-normal{font-weight:400!important}.font-weight-bold{font-weight:700!important}.font-italic{font-style:italic!important}.text-white{color:#fff!important}.text-primary{color:#007bff!important}a.text-primary:focus,a.text-primary:hover{color:#0062cc!important}.text-secondary{color:#6c757d!important}a.text-secondary:focus,a.text-secondary:hover{color:#545b62!important}.text-success{color:#28a745!important}a.text-success:focus,a.text-success:hover{color:#1e7e34!important}.text-info{color:#17a2b8!important}a.text-info:focus,a.text-info:hover{color:#117a8b!important}.text-warning{color:#ffc107!important}a.text-warning:focus,a.text-warning:hover{color:#d39e00!important}.text-danger{color:#dc3545!important}a.text-danger:focus,a.text-danger:hover{color:#bd2130!important}.text-light{color:#f8f9fa!important}a.text-light:focus,a.text-light:hover{color:#dae0e5!important}.text-dark{color:#343a40!important}a.text-dark:focus,a.text-dark:hover{color:#1d2124!important}.text-body{color:#212529!important}.text-muted{color:#6c757d!important}.text-black-50{color:rgba(0,0,0,.5)!important}.text-white-50{color:rgba(255,255,255,.5)!important}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.visible{visibility:visible!important}.invisible{visibility:hidden!important}@media print{*,::after,::before{text-shadow:none!important;box-shadow:none!important}a:not(.btn){text-decoration:underline}abbr[title]::after{content:" (" attr(title) ")"}pre{white-space:pre-wrap!important}blockquote,pre{border:1px solid #adb5bd;page-break-inside:avoid}thead{display:table-header-group}img,tr{page-break-inside:avoid}h2,h3,p{orphans:3;widows:3}h2,h3{page-break-after:avoid}@page{size:a3}body{min-width:992px!important}.container{min-width:992px!important}.navbar{display:none}.badge{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered td,.table-bordered th{border:1px solid #dee2e6!important}} -/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file diff --git a/templates/dashboard.html b/templates/dashboard.html index 36f729c..54e18f2 100755 --- a/templates/dashboard.html +++ b/templates/dashboard.html @@ -88,9 +88,10 @@

Refer a friend for free raffle tickets!



-
+ {% comment %} ================= UNCOMMENT ONCE NEW FACEBOOK GROUP HAS BEEN CREATED ================================ {% endcomment %} + {% comment %}
HooHacks 2021 Attendees
-
+
{% endcomment %} diff --git a/users/models.py b/users/models.py index 4da836c..effe3f2 100644 --- a/users/models.py +++ b/users/models.py @@ -1,6 +1,6 @@ from django.db import models from django.contrib.auth.models import AbstractUser -from django.utils.translation import ugettext_lazy as _ +from django.utils.translation import gettext_lazy as _ from django.core.mail import send_mail from django.conf import settings from datetime import datetime diff --git a/users/urls.py b/users/urls.py index 82b730a..b184ac3 100644 --- a/users/urls.py +++ b/users/urls.py @@ -14,7 +14,7 @@ 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) """ from django.contrib import admin -from django.conf.urls import url, include +from django.conf.urls import include from django.urls import path from . import views diff --git a/users/views.py b/users/views.py index 518e9b6..3466049 100644 --- a/users/views.py +++ b/users/views.py @@ -14,9 +14,9 @@ from administration.models import Settings from judging.models import Organization import re -import time import math + def redirect_dashboard(request): return redirect('/dashboard') @@ -25,8 +25,8 @@ def setup(request): if len(Settings.objects.all()) == 0: tz = settings.TZ Settings.objects.create( - application_submission_deadline=datetime.now(), - application_confirmation_deadline=datetime.now(), + application_submission_deadline=datetime(2023, 11, 20, 23, 59, 59, 0), + application_confirmation_deadline=datetime(2023, 11, 20, 23, 59, 59, 0), judging_deadline=datetime.now() ) o = Organization.objects.create(name="Organizers") From 1dec5b56becf139c37c951547470cefdd43dd9fe Mon Sep 17 00:00:00 2001 From: dylanherbig Date: Sat, 17 Dec 2022 17:25:32 -0500 Subject: [PATCH 07/21] - updated sendgrid api key - changed dashboard bullet point to hoohacks.io/apply --- hoohacks/secret_example.py | 17 ----------------- hoohacks/settings.py | 20 ++++++++++---------- templates/dashboard.html | 2 +- 3 files changed, 11 insertions(+), 28 deletions(-) delete mode 100644 hoohacks/secret_example.py diff --git a/hoohacks/secret_example.py b/hoohacks/secret_example.py deleted file mode 100644 index 1f9dd35..0000000 --- a/hoohacks/secret_example.py +++ /dev/null @@ -1,17 +0,0 @@ -ADMIN_PASSWORD = 'admin' -GMAIL_USERNAME = 'test@gmail.com' -GMAIL_PASSWORD = 'password' -MENTOR_PASSWORD = 'password' -VOLUNTEER_PASSWORD = 'password' -JUDGING_PASSWORD = 'password' -DROPBOX_ACCESS_TOKEN = 'token' -SLACK_ENABLED = False -SLACK_API_TOKEN = 'token' -SENDGRID_HOST_PASSWORD = 'token' -TEXTING_FROM_EMAIL = 'test@gmail.com' -TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = 'test' -TWITTER_CONSUMER_SECRET = 'test' -TWITTER_ACCESS_TOKEN = 'token' -TWITTER_ACCESS_TOKEN_SECRET = 'token' -PROD_DB_URL = 'token' \ No newline at end of file diff --git a/hoohacks/settings.py b/hoohacks/settings.py index d0f44b3..cb509fa 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -11,10 +11,10 @@ """ import os -try: - from . import secret -except ImportError: - from . import secret_example as secret +# try: +# from . import secret +# except ImportError: +# from . import secret_example as secret import datetime from pytz import timezone @@ -227,8 +227,8 @@ CITIES = ["Pittsburgh", "Washington, DC", "Richmond", "NYC", "Charlottesville"] # Test Cities with open(BASE_DIR + "/hoohacks/data/schools.txt", encoding='utf8') as f: - for school in f.read(): - SCHOOLS.append(school.strip()) + for school in f.read().split("\n"): + SCHOOLS.append(school) f.close() EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' @@ -289,7 +289,7 @@ SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') -SENDGRID_FROM_EMAIL = 'team@hoohacks.io' +SENDGRID_FROM_EMAIL = 'hackathon.virginia@gmail.com' TEXTING_ENABLED = True @@ -297,7 +297,7 @@ TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') -REQUIRE_EMAIL_VERIFICATION = False +REQUIRE_EMAIL_VERIFICATION = True WSGI_APPLICATION = 'hoohacks.wsgi.application' @@ -317,8 +317,8 @@ if USE_PROD_DB and DEBUG: import dj_database_url - # DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) - DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) + DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) + # DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) try: diff --git a/templates/dashboard.html b/templates/dashboard.html index 54e18f2..9633ff4 100755 --- a/templates/dashboard.html +++ b/templates/dashboard.html @@ -79,7 +79,7 @@

Refer a friend for free raffle tickets!

Get yourself 2 free tickets and the friend you refer 1 free ticket for our raffle, where we are giving away an Xbox, Powerbeats, and more. Here are instructions on how:
  1. Make sure you have applied, been accepted, and confirmed that you are coming to HooHacks.
  2. -
  3. Tell your friend to apply for HooHacks by linking them to hoohacks.io!
  4. +
  5. Tell your friend to apply for HooHacks by linking them to hoohacks.io/apply!
  6. Send your friend your raffle ID ({{ user.raffle_id }}). In their application, there is a field for them to include the ID.
  7. Once they submit, you’re all done! The tickets will be recorded under you and your friend’s name!
From c6fb5ffae2548faf8b301298fc9261d462ec997e Mon Sep 17 00:00:00 2001 From: dylanherbig Date: Fri, 23 Dec 2022 05:24:40 +0100 Subject: [PATCH 08/21] - updated application to include travel and dietary restriction - updated application and confirmation deadline - updated admin user view to reflect application changes - modifed dashboard text --- ...lication_confirmation_deadline_and_more.py | 23 ++++++++++ ...lication_confirmation_deadline_and_more.py | 21 +++++++++ administration/models.py | 10 ++-- .../0011_application_dietary_restriction.py | 18 ++++++++ applications/models.py | 1 + applications/views.py | 20 ++++---- hoohacks/settings.py | 4 +- templates/admin-users.html | 13 ++---- templates/application.html | 46 ++++++++++++++++++- templates/dashboard.html | 2 +- users/views.py | 4 +- 11 files changed, 131 insertions(+), 31 deletions(-) create mode 100644 administration/migrations/0003_alter_settings_application_confirmation_deadline_and_more.py create mode 100644 administration/migrations/0004_remove_settings_application_confirmation_deadline_and_more.py create mode 100644 applications/migrations/0011_application_dietary_restriction.py diff --git a/administration/migrations/0003_alter_settings_application_confirmation_deadline_and_more.py b/administration/migrations/0003_alter_settings_application_confirmation_deadline_and_more.py new file mode 100644 index 0000000..f14bf3b --- /dev/null +++ b/administration/migrations/0003_alter_settings_application_confirmation_deadline_and_more.py @@ -0,0 +1,23 @@ +# Generated by Django 4.1.3 on 2022-12-21 10:56 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('administration', '0002_settings_judging_deadline'), + ] + + operations = [ + migrations.AlterField( + model_name='settings', + name='application_confirmation_deadline', + field=models.DateTimeField(), + ), + migrations.AlterField( + model_name='settings', + name='application_submission_deadline', + field=models.DateTimeField(), + ), + ] diff --git a/administration/migrations/0004_remove_settings_application_confirmation_deadline_and_more.py b/administration/migrations/0004_remove_settings_application_confirmation_deadline_and_more.py new file mode 100644 index 0000000..2debd88 --- /dev/null +++ b/administration/migrations/0004_remove_settings_application_confirmation_deadline_and_more.py @@ -0,0 +1,21 @@ +# Generated by Django 4.1.3 on 2022-12-21 10:58 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('administration', '0003_alter_settings_application_confirmation_deadline_and_more'), + ] + + operations = [ + migrations.RemoveField( + model_name='settings', + name='application_confirmation_deadline', + ), + migrations.RemoveField( + model_name='settings', + name='application_submission_deadline', + ), + ] diff --git a/administration/models.py b/administration/models.py index 5a44de0..214954b 100644 --- a/administration/models.py +++ b/administration/models.py @@ -13,13 +13,13 @@ class Settings(models.Model): ''' CODE AS OF 12/08/2022, edited - application_submission_deadline = settings.TZ.localize(datetime(2023, 11, 20, 23, 59, 59, 0)) - application_confirmation_deadline = settings.TZ.localize(datetime(2023, 11, 20, 23, 59, 59, 0)) - judging_deadline = models.DateTimeField(auto_now_add=True) + application_submission_deadline = settings.TZ.localize(datetime(2023, 11, 20, 23, 59, 59, 0)) + application_confirmation_deadline = settings.TZ.localize(datetime(2023, 11, 20, 23, 59, 59, 0)) + judging_deadline = models.DateTimeField(auto_now_add=True) ''' - application_submission_deadline = models.DateTimeField() - application_confirmation_deadline = models.DateTimeField() + application_submission_deadline = settings.TZ.localize(datetime(2023, 2, 25, 23, 59, 59, 0)) + application_confirmation_deadline = settings.TZ.localize(datetime(2023, 3, 4, 23, 59, 59, 0)) judging_deadline = models.DateTimeField(auto_now_add=True) diff --git a/applications/migrations/0011_application_dietary_restriction.py b/applications/migrations/0011_application_dietary_restriction.py new file mode 100644 index 0000000..7520885 --- /dev/null +++ b/applications/migrations/0011_application_dietary_restriction.py @@ -0,0 +1,18 @@ +# Generated by Django 4.1.3 on 2022-12-21 10:56 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('applications', '0010_confirmation_discord_id'), + ] + + operations = [ + migrations.AddField( + model_name='application', + name='dietary_restriction', + field=models.CharField(default='', max_length=100), + ), + ] diff --git a/applications/models.py b/applications/models.py index 37b2b9c..11f2337 100644 --- a/applications/models.py +++ b/applications/models.py @@ -18,6 +18,7 @@ class Application(models.Model): describe = models.CharField(max_length=200, default="") why = models.CharField(max_length=1800, default="") major = models.CharField(max_length=1000, default="") + dietary_restriction = models.CharField(max_length=100, default="") birthday = models.CharField(max_length=1000, default="") mlh_rules = models.BooleanField(default=False) diff --git a/applications/views.py b/applications/views.py index ac1e2a4..0a1ffc1 100644 --- a/applications/views.py +++ b/applications/views.py @@ -42,6 +42,7 @@ def application(request, msg=''): "grad_year": settings.GRADUATION_YEARS, "highlight": "application", "travel_methods": settings.TRAVEL_METHODS, + "dietary_restrictions": settings.DIETARY_RESTRICTIONS, "msg": msg, "allow": ALLOW }) @@ -77,6 +78,11 @@ def application(request, msg=''): mlh = request.POST.get('mlh', '') mlh_consent = request.POST.get('mlh-consent', '') referrer = request.POST.get('referrer', '') + dietary_restriction = request.POST.get('dietary-restriction', '') + travel = True if request.POST.get('travel', '') == "Yes" else False + + if dietary_restriction == "Other": + dietary_restriction = request.POST.get('dietary-other', '') a.first_name = first_name a.last_name = last_name @@ -89,6 +95,8 @@ def application(request, msg=''): a.major = major a.hackathons = hackathons a.why = why + a.dietary_restriction = dietary_restriction + a.travel = travel friends = User.objects.filter(email=base64.b64decode(referrer).decode("utf-8", "ignore")) if friends.count() == 1 and friends.first() != u and not a.referrer_locked: @@ -117,18 +125,6 @@ def application(request, msg=''): a.save() return redirect(application, msg='saved') - # render(request, "application.html", { - # "user": u, - # "app": a, - # "schools": settings.SCHOOLS, - # "genders": settings.GENDERS, - # "races": settings.RACES, - # "grad_year": settings.GRADUATION_YEARS, - # "highlight": "application", - # "travel_methods": settings.TRAVEL_METHODS, - # "msg": "Your application has been submitted!", - # "allow": ALLOW - # }) @login_required def confirmation(request): diff --git a/hoohacks/settings.py b/hoohacks/settings.py index cb509fa..50c37da 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -202,10 +202,10 @@ EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') TZ = timezone('US/Eastern') -APPLICATION_SUBMISSION_DEADLINE = TZ.localize(datetime.datetime(2024, 2, 20, 23, 59, 59, 0)) +APPLICATION_SUBMISSION_DEADLINE = TZ.localize(datetime.datetime(2023, 2, 25, 23, 59, 59, 0)) APPLICATION_SUBMISSION_DEADLINE_FMT = APPLICATION_SUBMISSION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") -APPLICATION_CONFIRMATION_DEADLINE = TZ.localize(datetime.datetime(2024, 2, 21, 23, 59, 59, 0)) +APPLICATION_CONFIRMATION_DEADLINE = TZ.localize(datetime.datetime(2023, 3, 4, 23, 59, 59, 0)) APPLICATION_CONFIRMATION_DEADLINE_FMT = APPLICATION_CONFIRMATION_DEADLINE.strftime("%B %d, %Y %I:%M:%S %Z") SCHOOLS = [] diff --git a/templates/admin-users.html b/templates/admin-users.html index 1ad1a1e..3bd60a2 100644 --- a/templates/admin-users.html +++ b/templates/admin-users.html @@ -33,7 +33,6 @@ {% for u in all_hackers %} {% with a=u.application %} - View Application Birthday:

+ Dietary Restrictions: +
+
Travel Reimbursement Needed:

@@ -150,9 +150,6 @@ T-Shirt Size:

- Dietary Restrictions: -
-
Phone:

@@ -214,6 +211,7 @@ $('#describe').html($(this).data('describe')); $('#why').html($(this).data('why')); $('#major').html($(this).data('major')); + $('#dietary').html($(this).data('dietary')); $('#birthday').html($(this).data('birthday')); $('#travel').html($(this).data('travel')); $('#travel-method').html($(this).data('travel-method')); @@ -239,7 +237,6 @@ $('#status').html("Incomplete Application"); } $('#tshirt').html($(this).data('tshirt')); - $('#dietary').html($(this).data('dietary')); $('#phone').html($(this).data('phone')); $('#github').html($(this).data('github')); if ($(this).data('resume-file-name') == ""){ diff --git a/templates/application.html b/templates/application.html index 80060e4..be2f6fd 100644 --- a/templates/application.html +++ b/templates/application.html @@ -122,6 +122,49 @@

name="why">{% if app.app_complete %}{{ app.why }}{% endif %}

+ + + + +
+
+ Yes, I need travel reimbursement. +
+ No, I do not need travel reimbursement. +
+
+ +
+ +
+ +
+ + + +

@@ -262,7 +305,8 @@


{% if allow %}

By clicking the Submit Application button, I agree to share the information I provided with the {{ event_name }} - organization, along with the event sponsors and the Major League Hacking (MLH) organization.

+ organization, along with the event sponsors and the Major League Hacking (MLH) organization. You also confirm that you are + 18+ years of age.

{% else %} diff --git a/templates/dashboard.html b/templates/dashboard.html index 9633ff4..e5c112d 100755 --- a/templates/dashboard.html +++ b/templates/dashboard.html @@ -76,7 +76,7 @@

Your application status is:

Refer a friend for free raffle tickets!

- Get yourself 2 free tickets and the friend you refer 1 free ticket for our raffle, where we are giving away an Xbox, Powerbeats, and more. Here are instructions on how: + Get yourself 2 free tickets and the friend you refer 1 free ticket for our raffle, where we are giving away exciting prices. Here are instructions on how:

  1. Make sure you have applied, been accepted, and confirmed that you are coming to HooHacks.
  2. Tell your friend to apply for HooHacks by linking them to hoohacks.io/apply!
  3. diff --git a/users/views.py b/users/views.py index 3466049..a847ed6 100644 --- a/users/views.py +++ b/users/views.py @@ -25,8 +25,8 @@ def setup(request): if len(Settings.objects.all()) == 0: tz = settings.TZ Settings.objects.create( - application_submission_deadline=datetime(2023, 11, 20, 23, 59, 59, 0), - application_confirmation_deadline=datetime(2023, 11, 20, 23, 59, 59, 0), + application_submission_deadline=datetime(2023, 2, 25, 23, 59, 59, 0), + application_confirmation_deadline=datetime(2023, 3, 4, 23, 59, 59, 0), judging_deadline=datetime.now() ) o = Organization.objects.create(name="Organizers") From 3faa8a6818b72f35a26f9cf097efc4d6565a559e Mon Sep 17 00:00:00 2001 From: dylanherbig Date: Sun, 22 Jan 2023 16:50:44 +0100 Subject: [PATCH 09/21] created a registration page for onboarding: - previous iteration was confusing - split onboarding process into two sub-pages instead of a singular one - added "confirm-password" input to ensure that user has correct password --- hoohacks/settings.py | 98 ++++++++++++++-------------- templates/login_page.html | 2 +- templates/register_page.html | 60 ++++++++++++++++++ users/urls.py | 1 + users/views.py | 120 +++++++++++++++++++---------------- 5 files changed, 177 insertions(+), 104 deletions(-) create mode 100644 templates/register_page.html diff --git a/hoohacks/settings.py b/hoohacks/settings.py index 50c37da..bc2db6d 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -11,10 +11,10 @@ """ import os -# try: -# from . import secret -# except ImportError: -# from . import secret_example as secret +try: + from . import secret +except ImportError: + from . import secret_example as secret import datetime from pytz import timezone @@ -144,59 +144,59 @@ LOGIN_URL = '/users/login' LOGIN_REDIRECT_URL = '/dashboard' -# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) -# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) -# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) -# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) -# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) +ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) +MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) +VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) +JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) +DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) -ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') -MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') -VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') -JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') -DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') +# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') +# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') +# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') +# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') +# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') -# SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) -# if SLACK_ENABLED == "True": -# SLACK_ENABLED = True -# if SLACK_ENABLED != True: -# SLACK_ENABLED = False -# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) -# SLACK_MENTOR_TICKET_CHANNEL = "" -# if SLACK_ENABLED: -# SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" -# SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" - -SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) if SLACK_ENABLED == "True": SLACK_ENABLED = True if SLACK_ENABLED != True: SLACK_ENABLED = False -SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) SLACK_MENTOR_TICKET_CHANNEL = "" if SLACK_ENABLED: SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) -# if TWITTER_ENABLED == "True": -# TWITTER_ENABLED = True -# if TWITTER_ENABLED != True: -# TWITTER_ENABLED = False -# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) -# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) -# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) -# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) +# SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +# if SLACK_ENABLED == "True": +# SLACK_ENABLED = True +# if SLACK_ENABLED != True: +# SLACK_ENABLED = False +# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +# SLACK_MENTOR_TICKET_CHANNEL = "" +# if SLACK_ENABLED: +# SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" +# SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) if TWITTER_ENABLED == "True": TWITTER_ENABLED = True if TWITTER_ENABLED != True: TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') -TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') -TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') -TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') +TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) +TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) +TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) +TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) + +# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +# if TWITTER_ENABLED == "True": +# TWITTER_ENABLED = True +# if TWITTER_ENABLED != True: +# TWITTER_ENABLED = False +# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') +# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') +# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') +# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') MAX_NUMBER_TICKETS = 2 EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') @@ -235,10 +235,10 @@ EMAIL_HOST = 'smtp.gmail.com' EMAIL_USE_TLS = True EMAIL_PORT = 587 -# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) -# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) -EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') -EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') +EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) +EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) +# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') +# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') CONTACT_EMAIL = "travel@hoohacks.io" @@ -285,16 +285,16 @@ SENDGRID_HOST = 'smtp.sendgrid.net' SENDGRID_PORT = 587 SENDGRID_HOST_USER = 'apikey' -# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) -SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') +SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) +# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') SENDGRID_FROM_EMAIL = 'hackathon.virginia@gmail.com' TEXTING_ENABLED = True -# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) -TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') +TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) +# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') REQUIRE_EMAIL_VERIFICATION = True diff --git a/templates/login_page.html b/templates/login_page.html index ee45433..8dc9c00 100644 --- a/templates/login_page.html +++ b/templates/login_page.html @@ -46,7 +46,7 @@
- + Register
Forgot my password diff --git a/templates/register_page.html b/templates/register_page.html new file mode 100644 index 0000000..6b7d260 --- /dev/null +++ b/templates/register_page.html @@ -0,0 +1,60 @@ +{% load static %} + + + + + HooHacks + + + + + + + + + + +
+
+
+ +
+
+
+ + \ No newline at end of file diff --git a/users/urls.py b/users/urls.py index b184ac3..322712d 100644 --- a/users/urls.py +++ b/users/urls.py @@ -21,4 +21,5 @@ urlpatterns = [ path(r'login/', views.login_page, name='login_page'), path(r'forgot/', views.forgot_password, name='forgot_password'), + path(r'register/', views.register_page, name='register_page'), ] diff --git a/users/views.py b/users/views.py index a847ed6..11d1009 100644 --- a/users/views.py +++ b/users/views.py @@ -304,67 +304,79 @@ def puzzle_leaderboard(request): } return render(request, "puzzle_leaderboard.html", context) -def login_page(request): +def register_page(request): if request.method == "GET": if request.user.is_authenticated: return redirect("/dashboard") else: - return render(request, "login_page.html", {"message": "None"}) + return render(request, "register_page.html", {"message": "None"}) else: - if request.POST.get('button-type') == "register": - email = request.POST.get('email', '') - f = User.objects.filter(email=email) - if not f.count() == 0: - return render(request, "login_page.html", {"message": "There is already an account found with this email address!"}) - - email = request.POST.get('email', '') - password = request.POST.get('password', '') - u = User.objects.filter(email=email) - if len(u) != 0: - return redirect('index') - - u = User.objects.create( + email = request.POST.get('email', '') + f = User.objects.filter(email=email) + if not f.count() == 0: + return render(request, "register_page.html", {"message": "There is already an account found with this email address!"}) + + password = request.POST.get('password', '') + confirmPassword = request.POST.get('confirmPassword', '') + + u = User.objects.filter(email=email) + if len(u) != 0: + return redirect('index') + + if password != confirmPassword: + return render(request, "register_page.html", {"message": "The passwords do not match. Please try again."}) + + u = User.objects.create( username=email, email=email, password=password, verified=False) - u.set_password(password) - if not settings.REQUIRE_EMAIL_VERIFICATION: - u.verified = True - u.save() + u.set_password(password) + if not settings.REQUIRE_EMAIL_VERIFICATION: + u.verified = True + u.save() - a = Application.objects.create(user=u) - c = Confirmation.objects.create(user=u) - tz = timezone('US/Eastern') - email_uuid = uuid.uuid1() - e = EmailView.objects.create( - uuid_confirmation=email_uuid, - subject="HooHacks Email Verification", - message=settings.VERIFY_EMAIL.format( - u.email, settings.PROD_URL, email_uuid), - action="verify", - sent=tz.localize(datetime.now()), - redirect_url="/users/login/", - user=u - ) - e.send_email() - if not settings.REQUIRE_EMAIL_VERIFICATION: - return render(request, "login_page.html", {"message": "User account created! Please log in to fill out your application!"}) - else: - return render(request, "login_page.html", {"message": "User account created! Please check your email to confirm your account!"}) - elif request.POST.get('button-type') == "login": - email = request.POST.get('email', '') - password = request.POST.get('password', '') - u = User.objects.filter(email__iexact=email) - if u.count() == 0: - return render(request, "login_page.html", {"message": "No account found with this email address!"}) - u = u.first() - user = authenticate( - request, username=u.username, password=password) - if user is not None: - if not u.verified: - return render(request, "login_page.html", {"message": "Your account was not verified! Please check your email (and spam folder) to confirm your account!"}) - login(request, user) - next_url = request.GET.get('next', '/dashboard') - return redirect(next_url) - return render(request, "login_page.html", {"message": "Incorrect Password!"}) + a = Application.objects.create(user=u) + c = Confirmation.objects.create(user=u) + tz = timezone('US/Eastern') + email_uuid = uuid.uuid1() + e = EmailView.objects.create( + uuid_confirmation=email_uuid, + subject="HooHacks Email Verification", + message=settings.VERIFY_EMAIL.format( + u.email, settings.PROD_URL, email_uuid), + action="verify", + sent=tz.localize(datetime.now()), + redirect_url="/users/login/", + user=u + ) + e.send_email() + if not settings.REQUIRE_EMAIL_VERIFICATION: + return render(request, "login_page.html", {"message": "User account created! Please log in to fill out your application!"}) + else: + return render(request, "login_page.html", {"message": "User account created! Please check your email to confirm your account!"}) + + +def login_page(request): + if request.method == "GET": + if request.user.is_authenticated: + return redirect("/dashboard") + else: + return render(request, "login_page.html", {"message": "None"}) + else: + + email = request.POST.get('email', '') + password = request.POST.get('password', '') + u = User.objects.filter(email__iexact=email) + if u.count() == 0: + return render(request, "login_page.html", {"message": "No account found with this email address!"}) + u = u.first() + user = authenticate( + request, username=u.username, password=password) + if user is not None: + if not u.verified: + return render(request, "login_page.html", {"message": "Your account was not verified! Please check your email (and spam folder) to confirm your account!"}) + login(request, user) + next_url = request.GET.get('next', '/dashboard') + return redirect(next_url) + return render(request, "login_page.html", {"message": "Incorrect Password!"}) def forgot_password(request): From 3d3ce4553b7fa54d311be55a4ae10151b01850aa Mon Sep 17 00:00:00 2001 From: dylanherbig Date: Sun, 22 Jan 2023 17:01:02 +0100 Subject: [PATCH 10/21] merged registration into v2 --- hoohacks/settings.py | 102 +++++++++++++++++++++---------------------- 1 file changed, 51 insertions(+), 51 deletions(-) diff --git a/hoohacks/settings.py b/hoohacks/settings.py index bc2db6d..7fabe04 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -11,10 +11,10 @@ """ import os -try: - from . import secret -except ImportError: - from . import secret_example as secret +# try: +# from . import secret +# except ImportError: +# from . import secret_example as secret import datetime from pytz import timezone @@ -144,59 +144,59 @@ LOGIN_URL = '/users/login' LOGIN_REDIRECT_URL = '/dashboard' -ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) -MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) -VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) -JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) -DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) +# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) +# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) +# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) +# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) +# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) -# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') -# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') -# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') -# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') -# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') +ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') +MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') +VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') +JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') +DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') -SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) -if SLACK_ENABLED == "True": - SLACK_ENABLED = True -if SLACK_ENABLED != True: - SLACK_ENABLED = False -SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) -SLACK_MENTOR_TICKET_CHANNEL = "" -if SLACK_ENABLED: - SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" - SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" - -# SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +# SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) # if SLACK_ENABLED == "True": # SLACK_ENABLED = True # if SLACK_ENABLED != True: # SLACK_ENABLED = False -# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) # SLACK_MENTOR_TICKET_CHANNEL = "" # if SLACK_ENABLED: # SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" # SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) -if TWITTER_ENABLED == "True": - TWITTER_ENABLED = True -if TWITTER_ENABLED != True: - TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) -TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) -TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) -TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) +SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +if SLACK_ENABLED == "True": + SLACK_ENABLED = True +if SLACK_ENABLED != True: + SLACK_ENABLED = False +SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +SLACK_MENTOR_TICKET_CHANNEL = "" +if SLACK_ENABLED: + SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" + SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) # if TWITTER_ENABLED == "True": # TWITTER_ENABLED = True # if TWITTER_ENABLED != True: # TWITTER_ENABLED = False -# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') -# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') -# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') -# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') +# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) +# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) +# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) +# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) + +TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +if TWITTER_ENABLED == "True": + TWITTER_ENABLED = True +if TWITTER_ENABLED != True: + TWITTER_ENABLED = False +TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') +TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') +TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') +TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') MAX_NUMBER_TICKETS = 2 EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') @@ -235,10 +235,10 @@ EMAIL_HOST = 'smtp.gmail.com' EMAIL_USE_TLS = True EMAIL_PORT = 587 -EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) -EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) -# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') -# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') +# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) +# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) +EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') +EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') CONTACT_EMAIL = "travel@hoohacks.io" @@ -285,16 +285,16 @@ SENDGRID_HOST = 'smtp.sendgrid.net' SENDGRID_PORT = 587 SENDGRID_HOST_USER = 'apikey' -SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) -# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') +# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) +SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') SENDGRID_FROM_EMAIL = 'hackathon.virginia@gmail.com' TEXTING_ENABLED = True -TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) -# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') +# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) +TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') REQUIRE_EMAIL_VERIFICATION = True @@ -317,8 +317,8 @@ if USE_PROD_DB and DEBUG: import dj_database_url - DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) - # DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) + # DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) + DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) try: From e31f5c9da067d4ebf1abfcf415d35aa981671b71 Mon Sep 17 00:00:00 2001 From: Dylan Herbig <83849210+dylanherbig@users.noreply.github.com> Date: Mon, 30 Jan 2023 13:18:05 +0100 Subject: [PATCH 11/21] fixed too many fatal requests error Django left requests open for 600 seconds which would cause too many connection requests (maxed out at 20 per user) and would cause multiple instances of 500 errors. Now, django requests are immediately closed after a successful request has been made and the 500 error should be mitigated. --- hoohacks/settings.py | 16 +--------------- 1 file changed, 1 insertion(+), 15 deletions(-) diff --git a/hoohacks/settings.py b/hoohacks/settings.py index 7fabe04..0de348a 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -325,6 +325,7 @@ # Configure Django App for Heroku. import django_heroku django_heroku.settings(locals()) + DATABASES['default']['CONN_MAX_AGE'] = 0 if ON_HEROKU: try: del DATABASES['default']['OPTIONS']['sslmode'] @@ -343,21 +344,6 @@ except ImportError: found = False - -# Activate Django-Heroku. -# Use this code to avoid the psycopg2 / django-heroku error! -# Do NOT import django-heroku above! -# try: -# if 'HEROKU' in os.environ: -# import django_heroku -# django_heroku.settings(locals()) -# if 'DATABASE_URL' in os.environ: -# import dj_database_url -# DATABASES = {'default': dj_database_url.config()} -# except ImportError: -# found = False - - ALLOWED_HOSTS = [ PROD_URL, '*' From 8c3161f0dd66ce44f04af7d217b18d995ec426c7 Mon Sep 17 00:00:00 2001 From: Dylan Herbig <83849210+dylanherbig@users.noreply.github.com> Date: Thu, 9 Feb 2023 16:51:35 +0100 Subject: [PATCH 12/21] =?UTF-8?q?fixed=20=E2=80=98confirm=20password?= =?UTF-8?q?=E2=80=99=20input=20box?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit registration experienced issue where the users are not able to select the “confirm password” box - label id has been changed --- templates/register_page.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/templates/register_page.html b/templates/register_page.html index 6b7d260..17ffb00 100644 --- a/templates/register_page.html +++ b/templates/register_page.html @@ -42,7 +42,7 @@
- +
@@ -57,4 +57,4 @@
- \ No newline at end of file + From 871661a38aa0ed1403b75d3fbdea3d22d542d187 Mon Sep 17 00:00:00 2001 From: William Mueller Date: Thu, 9 Feb 2023 11:17:51 -0500 Subject: [PATCH 13/21] adding dietary restriction to csv, updating confirmation email --- administration/views.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/administration/views.py b/administration/views.py index 29a03aa..cb07f28 100644 --- a/administration/views.py +++ b/administration/views.py @@ -356,10 +356,10 @@ def admin_export_csv(request): response['Content-Disposition'] = 'inline; filename="data.csv"' writer = csv.writer(response) writer.writerow(['First Name', 'Last name', 'Email Address', 'School', 'Grad Year', - 'Gender', 'Race', 'Describe', 'Why', 'Major', 'Birthday', 'Travel', 'Where From', 'Travel Method', + 'Gender', 'Race', 'Describe', 'Dietary Restriction', 'Why', 'Major', 'Birthday', 'Travel', 'Where From', 'Travel Method', 'Miles', 'Cost', 'Accepted', 'Waitlisted', 'Rejected']) users = Application.objects.all().values_list('first_name', 'last_name', 'user', 'school', 'grad_year', - 'gender', 'race', 'describe', 'why', 'major', 'birthday', 'travel', 'where_from', 'travel_method', + 'gender', 'race', 'describe', 'dietary_restriction', 'why', 'major', 'birthday', 'travel', 'where_from', 'travel_method', 'miles', 'cost', 'accepted', 'waitlisted', 'rejected') for user in users: app_write = list(user) @@ -547,7 +547,7 @@ def accept_user(request, user_id): e = EmailView.objects.create( uuid_confirmation=email_uuid, subject="HooHacks Status Update", - message=settings.ACCEPTED_EMAIL.format(h.application.full_name, Settings.objects.all()[0].application_submission_deadline_fmt(), settings.PROD_URL, email_uuid), + message=settings.ACCEPTED_EMAIL.format(h.application.full_name, Settings.objects.all()[0].application_confirmation_deadline_fmt(), settings.PROD_URL, email_uuid), action="accepted", redirect_url="/dashboard", user=h From d47ee34c1286ffe4b8b16929641b73337e52526d Mon Sep 17 00:00:00 2001 From: William Mueller <37251567+wmueller49@users.noreply.github.com> Date: Thu, 9 Feb 2023 12:15:34 -0500 Subject: [PATCH 14/21] Revert "adding dietary restriction to csv, updating confirmation email" --- administration/views.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/administration/views.py b/administration/views.py index cb07f28..29a03aa 100644 --- a/administration/views.py +++ b/administration/views.py @@ -356,10 +356,10 @@ def admin_export_csv(request): response['Content-Disposition'] = 'inline; filename="data.csv"' writer = csv.writer(response) writer.writerow(['First Name', 'Last name', 'Email Address', 'School', 'Grad Year', - 'Gender', 'Race', 'Describe', 'Dietary Restriction', 'Why', 'Major', 'Birthday', 'Travel', 'Where From', 'Travel Method', + 'Gender', 'Race', 'Describe', 'Why', 'Major', 'Birthday', 'Travel', 'Where From', 'Travel Method', 'Miles', 'Cost', 'Accepted', 'Waitlisted', 'Rejected']) users = Application.objects.all().values_list('first_name', 'last_name', 'user', 'school', 'grad_year', - 'gender', 'race', 'describe', 'dietary_restriction', 'why', 'major', 'birthday', 'travel', 'where_from', 'travel_method', + 'gender', 'race', 'describe', 'why', 'major', 'birthday', 'travel', 'where_from', 'travel_method', 'miles', 'cost', 'accepted', 'waitlisted', 'rejected') for user in users: app_write = list(user) @@ -547,7 +547,7 @@ def accept_user(request, user_id): e = EmailView.objects.create( uuid_confirmation=email_uuid, subject="HooHacks Status Update", - message=settings.ACCEPTED_EMAIL.format(h.application.full_name, Settings.objects.all()[0].application_confirmation_deadline_fmt(), settings.PROD_URL, email_uuid), + message=settings.ACCEPTED_EMAIL.format(h.application.full_name, Settings.objects.all()[0].application_submission_deadline_fmt(), settings.PROD_URL, email_uuid), action="accepted", redirect_url="/dashboard", user=h From 661d3e9062c9c147ede07117cd85ab3e6a4a6eb8 Mon Sep 17 00:00:00 2001 From: William Mueller Date: Thu, 9 Feb 2023 22:30:15 -0500 Subject: [PATCH 15/21] removing verification email requirment, adding resend verification option --- hoohacks/settings.py | 105 +++++++++++++++-------------- templates/login_page.html | 1 + templates/resend_verification.html | 47 +++++++++++++ users/urls.py | 1 + users/views.py | 58 +++++++++++----- 5 files changed, 145 insertions(+), 67 deletions(-) create mode 100644 templates/resend_verification.html diff --git a/hoohacks/settings.py b/hoohacks/settings.py index 7fabe04..0a3c476 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -11,10 +11,10 @@ """ import os -# try: -# from . import secret -# except ImportError: -# from . import secret_example as secret +try: + from . import secret +except ImportError: + from . import secret_example as secret import datetime from pytz import timezone @@ -144,59 +144,59 @@ LOGIN_URL = '/users/login' LOGIN_REDIRECT_URL = '/dashboard' -# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) -# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) -# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) -# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) -# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) +ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) +MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) +VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) +JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) +DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) -ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') -MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') -VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') -JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') -DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') +# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') +# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') +# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') +# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') +# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') -# SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) -# if SLACK_ENABLED == "True": -# SLACK_ENABLED = True -# if SLACK_ENABLED != True: -# SLACK_ENABLED = False -# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) -# SLACK_MENTOR_TICKET_CHANNEL = "" -# if SLACK_ENABLED: -# SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" -# SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" - -SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) if SLACK_ENABLED == "True": SLACK_ENABLED = True if SLACK_ENABLED != True: SLACK_ENABLED = False -SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) SLACK_MENTOR_TICKET_CHANNEL = "" if SLACK_ENABLED: SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) -# if TWITTER_ENABLED == "True": -# TWITTER_ENABLED = True -# if TWITTER_ENABLED != True: -# TWITTER_ENABLED = False -# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) -# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) -# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) -# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) +# SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +# if SLACK_ENABLED == "True": +# SLACK_ENABLED = True +# if SLACK_ENABLED != True: +# SLACK_ENABLED = False +# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +# SLACK_MENTOR_TICKET_CHANNEL = "" +# if SLACK_ENABLED: +# SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" +# SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) if TWITTER_ENABLED == "True": TWITTER_ENABLED = True if TWITTER_ENABLED != True: TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') -TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') -TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') -TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') +TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) +TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) +TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) +TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) + +# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +# if TWITTER_ENABLED == "True": +# TWITTER_ENABLED = True +# if TWITTER_ENABLED != True: +# TWITTER_ENABLED = False +# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') +# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') +# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') +# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') MAX_NUMBER_TICKETS = 2 EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') @@ -235,10 +235,10 @@ EMAIL_HOST = 'smtp.gmail.com' EMAIL_USE_TLS = True EMAIL_PORT = 587 -# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) -# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) -EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') -EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') +EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) +EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) +# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') +# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') CONTACT_EMAIL = "travel@hoohacks.io" @@ -285,19 +285,20 @@ SENDGRID_HOST = 'smtp.sendgrid.net' SENDGRID_PORT = 587 SENDGRID_HOST_USER = 'apikey' -# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) -SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') +SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) +# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') SENDGRID_FROM_EMAIL = 'hackathon.virginia@gmail.com' TEXTING_ENABLED = True -# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) -TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') +TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) +# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') -REQUIRE_EMAIL_VERIFICATION = True +# FALSE TO AVOID SENDGRID ISSUES +REQUIRE_EMAIL_VERIFICATION = False WSGI_APPLICATION = 'hoohacks.wsgi.application' @@ -317,8 +318,8 @@ if USE_PROD_DB and DEBUG: import dj_database_url - # DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) - DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) + DATABASES['default'] = dj_database_url.config(default=secret.PROD_DB_URL) + # DATABASES['default'] = dj_database_url.config(default=os.environ.get("PROD_DB_URL")) try: diff --git a/templates/login_page.html b/templates/login_page.html index 8dc9c00..3e59963 100644 --- a/templates/login_page.html +++ b/templates/login_page.html @@ -49,6 +49,7 @@ Register
Forgot my password + Resend Verification diff --git a/templates/resend_verification.html b/templates/resend_verification.html new file mode 100644 index 0000000..c77faf1 --- /dev/null +++ b/templates/resend_verification.html @@ -0,0 +1,47 @@ +{% load static %} + + + + + HooHacks + + + + + + + + + +
+
+
+ +
+
+
+ + diff --git a/users/urls.py b/users/urls.py index 322712d..3126bf3 100644 --- a/users/urls.py +++ b/users/urls.py @@ -22,4 +22,5 @@ path(r'login/', views.login_page, name='login_page'), path(r'forgot/', views.forgot_password, name='forgot_password'), path(r'register/', views.register_page, name='register_page'), + path(r'resend/', views.resend_verification, name='resend_verification') ] diff --git a/users/views.py b/users/views.py index 11d1009..3e9f4f7 100644 --- a/users/views.py +++ b/users/views.py @@ -25,8 +25,8 @@ def setup(request): if len(Settings.objects.all()) == 0: tz = settings.TZ Settings.objects.create( - application_submission_deadline=datetime(2023, 2, 25, 23, 59, 59, 0), - application_confirmation_deadline=datetime(2023, 3, 4, 23, 59, 59, 0), + # application_submission_deadline=datetime(2023, 2, 25, 23, 59, 59, 0), + # application_confirmation_deadline=datetime(2023, 3, 4, 23, 59, 59, 0), judging_deadline=datetime.now() ) o = Organization.objects.create(name="Organizers") @@ -335,19 +335,22 @@ def register_page(request): a = Application.objects.create(user=u) c = Confirmation.objects.create(user=u) - tz = timezone('US/Eastern') - email_uuid = uuid.uuid1() - e = EmailView.objects.create( - uuid_confirmation=email_uuid, - subject="HooHacks Email Verification", - message=settings.VERIFY_EMAIL.format( - u.email, settings.PROD_URL, email_uuid), - action="verify", - sent=tz.localize(datetime.now()), - redirect_url="/users/login/", - user=u - ) - e.send_email() + + # COMMENTED OUT TO AVOID SENDGRID ISSUES + + # tz = timezone('US/Eastern') + # email_uuid = uuid.uuid1() + # e = EmailView.objects.create( + # uuid_confirmation=email_uuid, + # subject="HooHacks Email Verification", + # message=settings.VERIFY_EMAIL.format( + # u.email, settings.PROD_URL, email_uuid), + # action="verify", + # sent=tz.localize(datetime.now()), + # redirect_url="/users/login/", + # user=u + # ) + # e.send_email() if not settings.REQUIRE_EMAIL_VERIFICATION: return render(request, "login_page.html", {"message": "User account created! Please log in to fill out your application!"}) else: @@ -404,6 +407,31 @@ def forgot_password(request): print(e.uuid_confirmation) return render(request, "forgot_password.html", {"message": "Check your email to reset your password!", "show": False}) +def resend_verification(request): + if request.method == "GET": + return render(request, "resend_verification.html", {"message": "Please enter your email address to resend the verification email!", "show": True}) + if request.method == "POST": + email = request.POST.get('email', '') + u = User.objects.filter(email=email) + if u.count() == 0: + return render(request, "resend_verification.html", {"message": "There is no user associated with that email address!", "show": True}) + u = u.first() + tz = timezone('US/Eastern') + email_uuid = uuid.uuid1() + e = EmailView.objects.create( + uuid_confirmation=email_uuid, + subject="HooHacks Email Verification", + message=settings.VERIFY_EMAIL.format( + u.email, settings.PROD_URL, email_uuid), + action="verify", + sent=tz.localize(datetime.now()), + redirect_url="/users/login/", + user=u + ) + e.send_email() + print(e.uuid_confirmation) + return render(request, "resend_verification.html", {"message": "Check your email for your verification!", "show": False}) + def reset_password(request, email_uuid): if request.method == "GET": From 3bace6f4399a5ba2a16cc108982ba42ac6c6463c Mon Sep 17 00:00:00 2001 From: William Mueller Date: Thu, 9 Feb 2023 22:36:01 -0500 Subject: [PATCH 16/21] fixed settings --- hoohacks/settings.py | 94 ++++++++++++++++++++++---------------------- 1 file changed, 47 insertions(+), 47 deletions(-) diff --git a/hoohacks/settings.py b/hoohacks/settings.py index 0a3c476..c5f06a5 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -144,59 +144,59 @@ LOGIN_URL = '/users/login' LOGIN_REDIRECT_URL = '/dashboard' -ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) -MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) -VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) -JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) -DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) - -# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') -# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') -# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') -# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') -# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') - -SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) -if SLACK_ENABLED == "True": - SLACK_ENABLED = True -if SLACK_ENABLED != True: - SLACK_ENABLED = False -SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) -SLACK_MENTOR_TICKET_CHANNEL = "" -if SLACK_ENABLED: - SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" - SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" - -# SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +# ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD', secret.ADMIN_PASSWORD) +# MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD', secret.MENTOR_PASSWORD) +# VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD', secret.VOLUNTEER_PASSWORD) +# JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD', secret.JUDGING_PASSWORD) +# DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN', secret.DROPBOX_ACCESS_TOKEN) + +ADMIN_PASSWORD = os.environ.get('ADMIN_PASSWORD') +MENTOR_PASSWORD = os.environ.get('MENTOR_PASSWORD') +VOLUNTEER_PASSWORD = os.environ.get('VOLUNTEER_PASSWORD') +JUDGING_PASSWORD = os.environ.get('JUDGING_PASSWORD') +DROPBOX_ACCESS_TOKEN = os.environ.get('DROPBOX_ACCESS_TOKEN') + +# SLACK_ENABLED = os.environ.get('SLACK_ENABLED', secret.SLACK_ENABLED) # if SLACK_ENABLED == "True": # SLACK_ENABLED = True # if SLACK_ENABLED != True: # SLACK_ENABLED = False -# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +# SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN', secret.SLACK_API_TOKEN) # SLACK_MENTOR_TICKET_CHANNEL = "" # if SLACK_ENABLED: # SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" # SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) -if TWITTER_ENABLED == "True": - TWITTER_ENABLED = True -if TWITTER_ENABLED != True: - TWITTER_ENABLED = False -TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) -TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) -TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) -TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) +SLACK_ENABLED = os.environ.get('SLACK_ENABLED') +if SLACK_ENABLED == "True": + SLACK_ENABLED = True +if SLACK_ENABLED != True: + SLACK_ENABLED = False +SLACK_API_TOKEN = os.environ.get('SLACK_API_TOKEN') +SLACK_MENTOR_TICKET_CHANNEL = "" +if SLACK_ENABLED: + SLACK_MENTOR_TICKET_CHANNEL = "G010LQSEU75" + SLACK_NOTIFICATIONS_CHANNEL = "CSQKP19EC" -# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +# TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED', secret.TWITTER_ENABLED) # if TWITTER_ENABLED == "True": # TWITTER_ENABLED = True # if TWITTER_ENABLED != True: # TWITTER_ENABLED = False -# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') -# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') -# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') -# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') +# TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY', secret.TWITTER_CONSUMER_KEY) +# TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET', secret.TWITTER_CONSUMER_SECRET) +# TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN', secret.TWITTER_ACCESS_TOKEN) +# TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET', secret.TWITTER_ACCESS_TOKEN_SECRET) + +TWITTER_ENABLED = os.environ.get('TWITTER_ENABLED') +if TWITTER_ENABLED == "True": + TWITTER_ENABLED = True +if TWITTER_ENABLED != True: + TWITTER_ENABLED = False +TWITTER_CONSUMER_KEY = os.environ.get('TWITTER_CONSUMER_KEY') +TWITTER_CONSUMER_SECRET = os.environ.get('TWITTER_CONSUMER_SECRET') +TWITTER_ACCESS_TOKEN = os.environ.get('TWITTER_ACCESS_TOKEN') +TWITTER_ACCESS_TOKEN_SECRET = os.environ.get('TWITTER_ACCESS_TOKEN_SECRET') MAX_NUMBER_TICKETS = 2 EVENT_NAME = os.environ.get('EVENT_NAME', 'HooHacks') @@ -235,10 +235,10 @@ EMAIL_HOST = 'smtp.gmail.com' EMAIL_USE_TLS = True EMAIL_PORT = 587 -EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) -EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) -# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') -# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') +# EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME', secret.GMAIL_USERNAME) +# EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD', secret.GMAIL_PASSWORD) +EMAIL_HOST_USER = os.environ.get('GMAIL_USERNAME') +EMAIL_HOST_PASSWORD = os.environ.get('GMAIL_PASSWORD') CONTACT_EMAIL = "travel@hoohacks.io" @@ -285,16 +285,16 @@ SENDGRID_HOST = 'smtp.sendgrid.net' SENDGRID_PORT = 587 SENDGRID_HOST_USER = 'apikey' -SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) -# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') +# SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD', secret.SENDGRID_HOST_PASSWORD) +SENDGRID_HOST_PASSWORD = os.environ.get('SENDGRID_HOST_PASSWORD') SENDGRID_FROM_EMAIL = 'hackathon.virginia@gmail.com' TEXTING_ENABLED = True -TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) -# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') +# TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL', secret.TEXTING_FROM_EMAIL) +TEXTING_FROM_EMAIL = os.environ.get('TEXTING_FROM_EMAIL') # FALSE TO AVOID SENDGRID ISSUES From eb7289f59d415b747946b2993df9268dfe1d54c1 Mon Sep 17 00:00:00 2001 From: William Mueller Date: Thu, 9 Feb 2023 22:40:23 -0500 Subject: [PATCH 17/21] fixing settings pt 2 --- hoohacks/settings.py | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/hoohacks/settings.py b/hoohacks/settings.py index c5f06a5..7a6b18d 100644 --- a/hoohacks/settings.py +++ b/hoohacks/settings.py @@ -11,10 +11,10 @@ """ import os -try: - from . import secret -except ImportError: - from . import secret_example as secret +# try: +# from . import secret +# except ImportError: +# from . import secret_example as secret import datetime from pytz import timezone From 8caa314cd28b431dfeb924bab0b034741398939d Mon Sep 17 00:00:00 2001 From: William Mueller Date: Thu, 9 Feb 2023 23:09:00 -0500 Subject: [PATCH 18/21] fixed weird redirect bug after registration --- users/views.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/users/views.py b/users/views.py index 3e9f4f7..a1d9392 100644 --- a/users/views.py +++ b/users/views.py @@ -352,7 +352,8 @@ def register_page(request): # ) # e.send_email() if not settings.REQUIRE_EMAIL_VERIFICATION: - return render(request, "login_page.html", {"message": "User account created! Please log in to fill out your application!"}) + return redirect("/users/login") # Fixes weird bug where URL doesn't change + #return redirect("/users/login", {"message": "User account created! Please log in to fill out your application!"}) else: return render(request, "login_page.html", {"message": "User account created! Please check your email to confirm your account!"}) From be4f1ca47280ea9ddcd01923bbb7af5339b9986c Mon Sep 17 00:00:00 2001 From: William Mueller Date: Sun, 12 Feb 2023 16:32:58 -0500 Subject: [PATCH 19/21] fixing resend verification --- templates/resend_verification.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/templates/resend_verification.html b/templates/resend_verification.html index c77faf1..7e98f64 100644 --- a/templates/resend_verification.html +++ b/templates/resend_verification.html @@ -28,7 +28,7 @@ {% endif %} {% if show %} -