From 576fc30e8eb48acd52716109dcba25772b985672 Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Wed, 25 Jan 2017 17:51:53 -0500 Subject: [PATCH 1/2] Corrected URLs for the E-Commerce Service when using Docker Devstack ECOM-6634 --- lms/envs/devstack_docker.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/lms/envs/devstack_docker.py b/lms/envs/devstack_docker.py index 9b9757bbbd..401f326c01 100644 --- a/lms/envs/devstack_docker.py +++ b/lms/envs/devstack_docker.py @@ -13,7 +13,9 @@ HOST = 'edx.devstack.edxapp:18000' SITE_NAME = HOST LMS_ROOT_URL = 'http://{}'.format(HOST) -ECOMMERCE_PUBLIC_URL_ROOT = "http://edx.devstack.ecommerce:18130" +ECOMMERCE_PUBLIC_URL_ROOT = 'http://localhost:18130' +ECOMMERCE_API_URL = 'http://edx.devstack.ecommerce:18130/api/v2' + OAUTH_OIDC_ISSUER = '{}/oauth2'.format(LMS_ROOT_URL) From 47ae3791e92a646503c0fb86e18f6c95b19abb40 Mon Sep 17 00:00:00 2001 From: Clinton Blackburn Date: Thu, 26 Jan 2017 18:04:21 -0500 Subject: [PATCH 2/2] Added test settings for Docker-based devstack Fixes https://github.com/edx/devstack/issues/47 --- lms/envs/test_docker.py | 567 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 567 insertions(+) create mode 100644 lms/envs/test_docker.py diff --git a/lms/envs/test_docker.py b/lms/envs/test_docker.py new file mode 100644 index 0000000000..022085d26d --- /dev/null +++ b/lms/envs/test_docker.py @@ -0,0 +1,567 @@ +# -*- coding: utf-8 -*- +""" Test settings for Docker-based devstack. """ + +# pylint: disable=invalid-name +from uuid import uuid4 +from warnings import filterwarnings, simplefilter + +# NOTE: Importing devstack_docker MUST come first! +from .devstack_docker import * # pylint: disable=wildcard-import, unused-wildcard-import +from openedx.core.lib.tempdir import mkdtemp_clean +from util.db import NoOpMigrationModules +from util.testing import patch_testcase, patch_sessions + + +patch_testcase() +patch_sessions() + +# Silence noisy logs to make troubleshooting easier when tests fail. +import logging + +LOG_OVERRIDES = [ + ('factory.generate', logging.ERROR), + ('factory.containers', logging.ERROR), +] +for log_name, log_level in LOG_OVERRIDES: + logging.getLogger(log_name).setLevel(log_level) + +# Mongo connection settings +# NOTE: This is needed because xmodule/xmodule/modulestore/tests/mongo_connection.py pulls from environment variables! +os.environ['EDXAPP_TEST_MONGO_HOST'] = 'edx.devstack.mongo' +MONGO_PORT_NUM = int(os.environ.get('EDXAPP_TEST_MONGO_PORT', '27017')) +MONGO_HOST = os.environ.get('EDXAPP_TEST_MONGO_HOST', 'edx.devstack.mongo') + +os.environ['DJANGO_LIVE_TEST_SERVER_ADDRESS'] = 'localhost:8000-9000' + +THIS_UUID = uuid4().hex[:5] + +# can't test start dates with this True, but on the other hand, +# can test everything else :) +FEATURES['DISABLE_START_DATES'] = True + +# Most tests don't use the discussion service, so we turn it off to speed them up. +# Tests that do can enable this flag, but must use the UrlResetMixin class to force urls.py +# to reload. For consistency in user-experience, keep the value of this setting in sync with +# the one in cms/envs/test.py +FEATURES['ENABLE_DISCUSSION_SERVICE'] = False + +FEATURES['ENABLE_SERVICE_STATUS'] = True + +FEATURES['ENABLE_SHOPPING_CART'] = True + +FEATURES['ENABLE_VERIFIED_CERTIFICATES'] = True + +# Enable this feature for course staff grade downloads, to enable acceptance tests +FEATURES['ENABLE_GRADE_DOWNLOADS'] = True +FEATURES['ALLOW_COURSE_STAFF_GRADE_DOWNLOADS'] = True + +# Toggles embargo on for testing +FEATURES['EMBARGO'] = True + +FEATURES['ENABLE_COMBINED_LOGIN_REGISTRATION'] = True + +# Enable the milestones app in tests to be consistent with it being enabled in production +FEATURES['MILESTONES_APP'] = True + +# Need wiki for courseware views to work. TODO (vshnayder): shouldn't need it. +WIKI_ENABLED = True + +# Enable a parental consent age limit for testing +PARENTAL_CONSENT_AGE_LIMIT = 13 + +# Makes the tests run much faster... +SOUTH_TESTS_MIGRATE = False # To disable migrations and use syncdb instead + +# Nose Test Runner +TEST_RUNNER = 'openedx.core.djangolib.nose.NoseTestSuiteRunner' + +_SYSTEM = 'lms' + +_REPORT_DIR = REPO_ROOT / 'reports' / _SYSTEM +_REPORT_DIR.makedirs_p() +_NOSEID_DIR = REPO_ROOT / '.testids' / _SYSTEM +_NOSEID_DIR.makedirs_p() + +NOSE_ARGS = [ + '--id-file', _NOSEID_DIR / 'noseids', +] + +NOSE_PLUGINS = [ + 'openedx.core.djangolib.testing.utils.NoseDatabaseIsolation' +] + +# Local Directories +TEST_ROOT = path("test_root") +# Want static files in the same dir for running on jenkins. +STATIC_ROOT = TEST_ROOT / "staticfiles" + +STATUS_MESSAGE_PATH = TEST_ROOT / "status_message.json" + +COURSES_ROOT = TEST_ROOT / "data" +DATA_DIR = COURSES_ROOT + +COMMON_TEST_DATA_ROOT = COMMON_ROOT / "test" / "data" +# Where the content data is checked out. This may not exist on jenkins. +GITHUB_REPO_ROOT = ENV_ROOT / "data" + +USE_I18N = True +LANGUAGE_CODE = 'en' # tests assume they will get English. + +XQUEUE_INTERFACE = { + "url": "http://sandbox-xqueue.edx.org", + "django_auth": { + "username": "lms", + "password": "***REMOVED***" + }, + "basic_auth": ('anant', 'agarwal'), +} +XQUEUE_WAITTIME_BETWEEN_REQUESTS = 5 # seconds + +# Don't rely on a real staff grading backend +MOCK_STAFF_GRADING = True +MOCK_PEER_GRADING = True + +############################ STATIC FILES ############################# + +# TODO (cpennington): We need to figure out how envs/test.py can inject things +# into common.py so that we don't have to repeat this sort of thing +STATICFILES_DIRS = [ + COMMON_ROOT / "static", + PROJECT_ROOT / "static", +] +STATICFILES_DIRS += [ + (course_dir, COMMON_TEST_DATA_ROOT / course_dir) + for course_dir in os.listdir(COMMON_TEST_DATA_ROOT) + if os.path.isdir(COMMON_TEST_DATA_ROOT / course_dir) + ] + +# Avoid having to run collectstatic before the unit test suite +# If we don't add these settings, then Django templates that can't +# find pipelined assets will raise a ValueError. +# http://stackoverflow.com/questions/12816941/unit-testing-with-django-pipeline +STATICFILES_STORAGE = 'pipeline.storage.NonPackagingPipelineStorage' + +# Don't use compression during tests +PIPELINE_JS_COMPRESSOR = None + +update_module_store_settings( + MODULESTORE, + module_store_options={ + 'fs_root': TEST_ROOT / "data", + }, + xml_store_options={ + 'data_dir': mkdtemp_clean(dir=TEST_ROOT), # never inadvertently load all the XML courses + }, + doc_store_settings={ + 'db': 'test_xmodule_{}'.format(THIS_UUID), + 'collection': 'test_modulestore', + }, +) + +CONTENTSTORE['DOC_STORE_CONFIG']['db'] = 'test_xcontent_{}'.format(THIS_UUID) + +DATABASES = { + 'default': { + 'ENGINE': 'django.db.backends.sqlite3', + 'ATOMIC_REQUESTS': True, + }, + 'student_module_history': { + 'ENGINE': 'django.db.backends.sqlite3', + }, +} + +if os.environ.get('DISABLE_MIGRATIONS'): + # Create tables directly from apps' models. This can be removed once we upgrade + # to Django 1.9, which allows setting MIGRATION_MODULES to None in order to skip migrations. + MIGRATION_MODULES = NoOpMigrationModules() + +# Make sure we test with the extended history table +FEATURES['ENABLE_CSMH_EXTENDED'] = True +# INSTALLED_APPS += ('coursewarehistoryextended',) + +CACHES = { + # This is the cache used for most things. + # In staging/prod envs, the sessions also live here. + 'default': { + 'BACKEND': 'django.core.cache.backends.dummy.DummyCache', + }, + + # The general cache is what you get if you use our util.cache. It's used for + # things like caching the course.xml file for different A/B test groups. + # We set it to be a DummyCache to force reloading of course.xml in dev. + # In staging environments, we would grab VERSION from data uploaded by the + # push process. + 'general': { + 'BACKEND': 'django.core.cache.backends.dummy.DummyCache', + }, + + 'mongo_metadata_inheritance': { + 'BACKEND': 'django.core.cache.backends.dummy.DummyCache', + }, + 'loc_cache': { + 'BACKEND': 'django.core.cache.backends.dummy.DummyCache', + }, + 'course_structure_cache': { + 'BACKEND': 'django.core.cache.backends.dummy.DummyCache', + }, +} + +# Dummy secret key for dev +SECRET_KEY = '85920908f28904ed733fe576320db18cabd7b6cd' + +# hide ratelimit warnings while running tests +filterwarnings('ignore', message='No request passed to the backend, unable to rate-limit') + +# Ignore deprecation warnings (so we don't clutter Jenkins builds/production) +# https://docs.python.org/2/library/warnings.html#the-warnings-filter +# Change to "default" to see the first instance of each hit +# or "error" to convert all into errors +simplefilter('ignore') + +############################# SECURITY SETTINGS ################################ +# Default to advanced security in common.py, so tests can reset here to use +# a simpler security model +FEATURES['ENFORCE_PASSWORD_POLICY'] = False +FEATURES['ENABLE_MAX_FAILED_LOGIN_ATTEMPTS'] = False +FEATURES['SQUELCH_PII_IN_LOGS'] = False +FEATURES['PREVENT_CONCURRENT_LOGINS'] = False +FEATURES['ADVANCED_SECURITY'] = False +PASSWORD_MIN_LENGTH = None +PASSWORD_COMPLEXITY = {} + +######### Third-party auth ########## +FEATURES['ENABLE_THIRD_PARTY_AUTH'] = True + +AUTHENTICATION_BACKENDS = ( + 'social.backends.google.GoogleOAuth2', + 'social.backends.linkedin.LinkedinOAuth2', + 'social.backends.facebook.FacebookOAuth2', + 'social.backends.azuread.AzureADOAuth2', + 'social.backends.twitter.TwitterOAuth', + 'third_party_auth.dummy.DummyBackend', + 'third_party_auth.saml.SAMLAuthBackend', + 'third_party_auth.lti.LTIAuthBackend', +) + AUTHENTICATION_BACKENDS + +THIRD_PARTY_AUTH_CUSTOM_AUTH_FORMS = { + 'custom1': { + 'secret_key': 'opensesame', + 'url': '/misc/my-custom-registration-form', + 'error_url': '/misc/my-custom-sso-error-page' + }, +} + +################################## OPENID ##################################### +FEATURES['AUTH_USE_OPENID'] = True +FEATURES['AUTH_USE_OPENID_PROVIDER'] = True + +################################## SHIB ####################################### +FEATURES['AUTH_USE_SHIB'] = True +FEATURES['SHIB_DISABLE_TOS'] = True +FEATURES['RESTRICT_ENROLL_BY_REG_METHOD'] = True + +OPENID_CREATE_USERS = False +OPENID_UPDATE_DETAILS_FROM_SREG = True +OPENID_USE_AS_ADMIN_LOGIN = False +OPENID_PROVIDER_TRUSTED_ROOTS = ['*'] + +############################## OAUTH2 Provider ################################ +FEATURES['ENABLE_OAUTH2_PROVIDER'] = True +# don't cache courses for testing +OIDC_COURSE_HANDLER_CACHE_TIMEOUT = 0 + +########################### External REST APIs ################################# +FEATURES['ENABLE_MOBILE_REST_API'] = True +FEATURES['ENABLE_VIDEO_ABSTRACTION_LAYER_API'] = True + +########################### Grades ################################# +FEATURES['PERSISTENT_GRADES_ENABLED_FOR_ALL_TESTS'] = True + +###################### Payment ##############################3 +# Enable fake payment processing page +FEATURES['ENABLE_PAYMENT_FAKE'] = True + +# Configure the payment processor to use the fake processing page +# Since both the fake payment page and the shoppingcart app are using +# the same settings, we can generate this randomly and guarantee +# that they are using the same secret. +from random import choice +from string import letters, digits, punctuation + +RANDOM_SHARED_SECRET = ''.join( + choice(letters + digits + punctuation) + for x in range(250) +) + +CC_PROCESSOR_NAME = 'CyberSource2' +CC_PROCESSOR['CyberSource2']['SECRET_KEY'] = RANDOM_SHARED_SECRET +CC_PROCESSOR['CyberSource2']['ACCESS_KEY'] = "0123456789012345678901" +CC_PROCESSOR['CyberSource2']['PROFILE_ID'] = "edx" +CC_PROCESSOR['CyberSource2']['PURCHASE_ENDPOINT'] = "/shoppingcart/payment_fake" + +FEATURES['STORE_BILLING_INFO'] = True + +########################### SYSADMIN DASHBOARD ################################ +FEATURES['ENABLE_SYSADMIN_DASHBOARD'] = True +GIT_REPO_DIR = TEST_ROOT / "course_repos" + +################################# CELERY ###################################### + +CELERY_ALWAYS_EAGER = True +CELERY_RESULT_BACKEND = 'djcelery.backends.cache:CacheBackend' + +######################### MARKETING SITE ############################### + +MKTG_URL_LINK_MAP = { + 'ABOUT': 'about', + 'CONTACT': 'contact', + 'HELP_CENTER': 'help-center', + 'COURSES': 'courses', + 'ROOT': 'root', + 'TOS': 'tos', + 'HONOR': 'honor', + 'PRIVACY': 'privacy', + 'CAREERS': 'careers', + 'NEWS': 'news', + 'PRESS': 'press', + 'BLOG': 'blog', + 'DONATE': 'donate', + 'SITEMAP.XML': 'sitemap_xml', + + # Verified Certificates + 'WHAT_IS_VERIFIED_CERT': 'verified-certificate', +} + +SUPPORT_SITE_LINK = 'https://support.example.com' + +############################ STATIC FILES ############################# +DEFAULT_FILE_STORAGE = 'django.core.files.storage.FileSystemStorage' +MEDIA_ROOT = TEST_ROOT / "uploads" +MEDIA_URL = "/static/uploads/" +STATICFILES_DIRS.append(("uploads", MEDIA_ROOT)) + +_NEW_STATICFILES_DIRS = [] +# Strip out any static files that aren't in the repository root +# so that the tests can run with only the edx-platform directory checked out +for static_dir in STATICFILES_DIRS: + # Handle both tuples and non-tuple directory definitions + try: + _, data_dir = static_dir + except ValueError: + data_dir = static_dir + + if data_dir.startswith(REPO_ROOT): + _NEW_STATICFILES_DIRS.append(static_dir) +STATICFILES_DIRS = _NEW_STATICFILES_DIRS + +FILE_UPLOAD_TEMP_DIR = TEST_ROOT / "uploads" +FILE_UPLOAD_HANDLERS = ( + 'django.core.files.uploadhandler.MemoryFileUploadHandler', + 'django.core.files.uploadhandler.TemporaryFileUploadHandler', +) + +########################### Server Ports ################################### + +# These ports are carefully chosen so that if the browser needs to +# access them, they will be available through the SauceLabs SSH tunnel +LETTUCE_SERVER_PORT = 8003 +XQUEUE_PORT = 8040 +YOUTUBE_PORT = 8031 +LTI_PORT = 8765 +VIDEO_SOURCE_PORT = 8777 + +FEATURES['PREVIEW_LMS_BASE'] = "preview.localhost" +############### Module Store Items ########## +PREVIEW_DOMAIN = FEATURES['PREVIEW_LMS_BASE'].split(':')[0] +HOSTNAME_MODULESTORE_DEFAULT_MAPPINGS = { + PREVIEW_DOMAIN: 'draft-preferred' +} + +################### Make tests faster + +# http://slacy.com/blog/2012/04/make-your-tests-faster-in-django-1-4/ +PASSWORD_HASHERS = ( + # 'django.contrib.auth.hashers.PBKDF2PasswordHasher', + # 'django.contrib.auth.hashers.PBKDF2SHA1PasswordHasher', + # 'django.contrib.auth.hashers.BCryptPasswordHasher', + 'django.contrib.auth.hashers.SHA1PasswordHasher', + 'django.contrib.auth.hashers.MD5PasswordHasher', + # 'django.contrib.auth.hashers.CryptPasswordHasher', +) + +### This enables the Metrics tab for the Instructor dashboard ########### +FEATURES['CLASS_DASHBOARD'] = True + +################### Make tests quieter + +# OpenID spews messages like this to stderr, we don't need to see them: +# Generated checkid_setup request to http://testserver/openid/provider/login/ +# with association {HMAC-SHA1}{51d49995}{s/kRmA==} + +import openid.oidutil + +openid.oidutil.log = lambda message, level=0: None + +# Include a non-ascii character in PLATFORM_NAME to uncover possible UnicodeEncodeErrors in tests. +PLATFORM_NAME = u"édX" +SITE_NAME = "edx.org" + +# set up some testing for microsites +FEATURES['USE_MICROSITES'] = True +MICROSITE_ROOT_DIR = COMMON_ROOT / 'test' / 'test_sites' +MICROSITE_CONFIGURATION = { + "test_site": { + "domain_prefix": "test-site", + "university": "test_site", + "platform_name": "Test Site", + "logo_image_url": "test_site/images/header-logo.png", + "email_from_address": "test_site@edx.org", + "payment_support_email": "test_site@edx.org", + "ENABLE_MKTG_SITE": False, + "SITE_NAME": "test_site.localhost", + "course_org_filter": "TestSiteX", + "course_about_show_social_links": False, + "css_overrides_file": "test_site/css/test_site.css", + "show_partners": False, + "show_homepage_promo_video": False, + "course_index_overlay_text": "This is a Test Site Overlay Text.", + "course_index_overlay_logo_file": "test_site/images/header-logo.png", + "homepage_overlay_html": "

This is a Test Site Overlay HTML

", + "ALWAYS_REDIRECT_HOMEPAGE_TO_DASHBOARD_FOR_AUTHENTICATED_USER": False, + "COURSE_CATALOG_VISIBILITY_PERMISSION": "see_in_catalog", + "COURSE_ABOUT_VISIBILITY_PERMISSION": "see_about_page", + "ENABLE_SHOPPING_CART": True, + "ENABLE_PAID_COURSE_REGISTRATION": True, + "SESSION_COOKIE_DOMAIN": "test_site.localhost", + "LINKEDIN_COMPANY_ID": "test", + "FACEBOOK_APP_ID": "12345678908", + "urls": { + 'ABOUT': 'test-site/about', + 'PRIVACY': 'test-site/privacy', + 'TOS_AND_HONOR': 'test-site/tos-and-honor', + }, + }, + "site_with_logistration": { + "domain_prefix": "logistration", + "university": "logistration", + "platform_name": "Test logistration", + "logo_image_url": "test_site/images/header-logo.png", + "email_from_address": "test_site@edx.org", + "payment_support_email": "test_site@edx.org", + "ENABLE_MKTG_SITE": False, + "ENABLE_COMBINED_LOGIN_REGISTRATION": True, + "SITE_NAME": "test_site.localhost", + "course_org_filter": "LogistrationX", + "course_about_show_social_links": False, + "css_overrides_file": "test_site/css/test_site.css", + "show_partners": False, + "show_homepage_promo_video": False, + "course_index_overlay_text": "Logistration.", + "course_index_overlay_logo_file": "test_site/images/header-logo.png", + "homepage_overlay_html": "

This is a Logistration HTML

", + "ALWAYS_REDIRECT_HOMEPAGE_TO_DASHBOARD_FOR_AUTHENTICATED_USER": False, + "COURSE_CATALOG_VISIBILITY_PERMISSION": "see_in_catalog", + "COURSE_ABOUT_VISIBILITY_PERMISSION": "see_about_page", + "ENABLE_SHOPPING_CART": True, + "ENABLE_PAID_COURSE_REGISTRATION": True, + "SESSION_COOKIE_DOMAIN": "test_logistration.localhost", + }, + "default": { + "university": "default_university", + "domain_prefix": "www", + } +} + +MICROSITE_TEST_HOSTNAME = 'test-site.testserver' +MICROSITE_LOGISTRATION_HOSTNAME = 'logistration.testserver' + +TEST_THEME = COMMON_ROOT / "test" / "test-theme" + +# add extra template directory for test-only templates +MAKO_TEMPLATES['main'].extend([ + COMMON_ROOT / 'test' / 'templates', + COMMON_ROOT / 'test' / 'test_sites', + REPO_ROOT / 'openedx' / 'core' / 'djangolib' / 'tests' / 'templates', +]) + +# Setting for the testing of Software Secure Result Callback +VERIFY_STUDENT["SOFTWARE_SECURE"] = { + "API_ACCESS_KEY": "BBBBBBBBBBBBBBBBBBBB", + "API_SECRET_KEY": "CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC", +} + +VIDEO_CDN_URL = { + 'CN': 'http://api.xuetangx.com/edx/video?s3_url=' +} + +######### dashboard git log settings ######### +MONGODB_LOG = { + 'host': MONGO_HOST, + 'port': MONGO_PORT_NUM, + 'user': 'edxapp', + 'password': 'password', + 'db': 'xlog', +} + +NOTES_DISABLED_TABS = [] + +# Enable EdxNotes for tests. +FEATURES['ENABLE_EDXNOTES'] = True + +# Enable teams feature for tests. +FEATURES['ENABLE_TEAMS'] = True + +# Enable courseware search for tests +FEATURES['ENABLE_COURSEWARE_SEARCH'] = True + +# Enable dashboard search for tests +FEATURES['ENABLE_DASHBOARD_SEARCH'] = True + +# Use MockSearchEngine as the search engine for test scenario +SEARCH_ENGINE = "search.tests.mock_search_engine.MockSearchEngine" + +FACEBOOK_APP_SECRET = "Test" +FACEBOOK_APP_ID = "Test" +FACEBOOK_API_VERSION = "v2.2" + +######### custom courses ######### +INSTALLED_APPS += ('lms.djangoapps.ccx', 'openedx.core.djangoapps.ccxcon') +FEATURES['CUSTOM_COURSES_EDX'] = True + +# Set dummy values for profile image settings. +PROFILE_IMAGE_BACKEND = { + 'class': 'storages.backends.overwrite.OverwriteStorage', + 'options': { + 'location': MEDIA_ROOT, + 'base_url': 'http://example-storage.com/profile-images/', + }, +} +PROFILE_IMAGE_DEFAULT_FILENAME = 'default' +PROFILE_IMAGE_DEFAULT_FILE_EXTENSION = 'png' +PROFILE_IMAGE_SECRET_KEY = 'secret' +PROFILE_IMAGE_MAX_BYTES = 1024 * 1024 +PROFILE_IMAGE_MIN_BYTES = 100 + +# Enable the LTI provider feature for testing +FEATURES['ENABLE_LTI_PROVIDER'] = True +INSTALLED_APPS += ('lti_provider',) +AUTHENTICATION_BACKENDS += ('lti_provider.users.LtiBackend',) + +# ORGANIZATIONS +FEATURES['ORGANIZATIONS_APP'] = True + +# Financial assistance page +FEATURES['ENABLE_FINANCIAL_ASSISTANCE_FORM'] = True + +JWT_AUTH.update({ + 'JWT_SECRET_KEY': 'test-secret', + 'JWT_ISSUER': 'https://test-provider/oauth2', + 'JWT_AUDIENCE': 'test-key', +}) + +COURSE_CATALOG_API_URL = 'https://catalog.example.com/api/v1' + +COMPREHENSIVE_THEME_DIRS = [REPO_ROOT / "themes", REPO_ROOT / "common/test"] +COMPREHENSIVE_THEME_LOCALE_PATHS = [REPO_ROOT / "themes/conf/locale", ] + +LMS_ROOT_URL = "http://localhost:8000"