Skip to content

Commit

Permalink
feat(devcontainer): Enable developers to easily install Sefaria
Browse files Browse the repository at this point in the history
  • Loading branch information
orxai committed Oct 11, 2024
1 parent be0f579 commit b2b78ba
Show file tree
Hide file tree
Showing 4 changed files with 355 additions and 0 deletions.
6 changes: 6 additions & 0 deletions .devcontainer/devcontainer.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
{
"image": "ghcr.io/orxaicom/sefaria-container-unofficial:latest",
"forwardPorts": [8000],
"postCreateCommand": "chmod +x .devcontainer/postCreate.sh && .devcontainer/postCreate.sh",
"postStartCommand": "chmod +x .devcontainer/postStart.sh && .devcontainer/postStart.sh"
}
336 changes: 336 additions & 0 deletions .devcontainer/local_settings.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,336 @@
# An example of settings needed in a local_settings.py file.
# copy this file to sefaria/local_settings.py and provide local info to run.
from datetime import timedelta
import sys
import structlog
import sefaria.system.logging as sefaria_logging
import os

# These are things you need to change!

################
# YOU ONLY NEED TO CHANGE "NAME" TO THE PATH OF YOUR SQLITE DATA FILE
# If the db.sqlite file does not exist, simply list a path where it can be created.
# You can set the path to /path/to/Sefaria-Project/db.sqlite, since we git-ignore all sqlite files
# (you do not need to create the empty db.sqlite file, as Django will handle that later)
# ########################################
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.sqlite3', # Add 'postgresql_psycopg2', 'mysql', 'sqlite3' or 'oracle'.
'NAME': 'db.sqlite', # Path to where you would like the database to be created including a file name, or path to an existing database file if using sqlite3.
'USER': '', # Not used with sqlite3.
'PASSWORD': '', # Not used with sqlite3.
'HOST': '', # Set to empty string for localhost. Not used with sqlite3.
'PORT': '', # Set to empty string for default. Not used with sqlite3.
}
}

"""
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.postgresql',
'NAME': 'name of db table here',
'USER': 'name of db user here',
'PASSWORD': 'password here',
'HOST': '127.0.0.1',
'PORT': '',
}
}"""

# Map domain to an interface language that the domain should be pinned to.
# Leave as {} to prevent language pinning, in which case one domain can serve either Hebrew or English
DOMAIN_LANGUAGES = {
"http://hebrew.example.org": "hebrew",
"http://english.example.org": "english",
}


################ These are things you can change! ###########################################################################
SILENCED_SYSTEM_CHECKS = ['captcha.recaptcha_test_key_error']

ALLOWED_HOSTS = ["localhost", "127.0.0.1","0.0.0.0"]

ADMINS = (
('Your Name', '[email protected]'),
)
ADMIN_PATH = 'somethingsomething' #This will be the path to the admin site, locally it can also be 'admin'

PINNED_IPCOUNTRY = "IL" #change if you want parashat hashavua to be diaspora.

MONGO_REPLICASET_NAME = None # If the below is a list, this should be set to something other than None.
# This can be either a string of one mongo host server or a list of `host:port` string pairs. So either e.g "localhost" of ["localhost:27017","localhost2:27017" ]
MONGO_HOST = "localhost"
MONGO_PORT = 27017 # Not used if the above is a list
# Name of the MongoDB database to use.
SEFARIA_DB = 'sefaria' # Change if you named your db something else
SEFARIA_DB_USER = '' # Leave user and password blank if not using Mongo Auth
SEFARIA_DB_PASSWORD = ''
APSCHEDULER_NAME = "apscheduler"


""" These are some examples of possible caches. more here: https://docs.djangoproject.com/en/1.11/topics/cache/"""
CACHES = {
"shared": {
'BACKEND': 'django.core.cache.backends.dummy.DummyCache',
},
"default": {
'BACKEND': 'django.core.cache.backends.dummy.DummyCache',
},
}
"""
CACHES = {
'shared': {
'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache',
'LOCATION': '/home/ephraim/www/sefaria/django_cache/',
},
'default': {
'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache',
'LOCATION': '/home/ephraim/www/sefaria/django_cache/',
}
}
"""

SESSION_CACHE_ALIAS = "default"
USER_AGENTS_CACHE = 'default'
SHARED_DATA_CACHE_ALIAS = 'shared'

"""THIS CACHE DEFINITION IS FOR USE WITH NODE AND SERVER SIDE RENDERING"""
"""
CACHES = {
"shared": {
"BACKEND": "django_redis.cache.RedisCache",
"LOCATION": "redis://127.0.0.1:6379/1", #The URI used to look like this "127.0.0.1:6379:0"
"OPTIONS": {
"CLIENT_CLASS": "django_redis.client.DefaultClient",
#"SERIALIZER": "django_redis.serializers.json.JSONSerializer", #this is the default, we override it to ensure_ascii=False
"SERIALIZER": "sefaria.system.serializers.JSONSerializer",
},
"TIMEOUT": None,
},
"default": {
"BACKEND": "django_redis.cache.RedisCache",
"LOCATION": "redis://127.0.0.1:6379/0", #The URI used to look like this "127.0.0.1:6379:0"
"OPTIONS": {
"CLIENT_CLASS": "django_redis.client.DefaultClient",
#"PASSWORD": "secretpassword", # Optional
},
"TIMEOUT": 60 * 60 * 24 * 30,
},
}
"""

SITE_PACKAGE = "sites.sefaria"







################ These are things you DO NOT NEED to touch unless you know what you are doing. ##############################
DEBUG = True
ALLOWED_HOSTS = ['localhost', '127.0.0.1', '[::1]']
OFFLINE = False
DOWN_FOR_MAINTENANCE = False
MAINTENANCE_MESSAGE = ""

# Location of Strapi CMS instance
# For local development, Strapi is located at http://localhost:1337 by default
STRAPI_LOCATION = None
STRAPI_PORT = None


MANAGERS = ADMINS

SECRET_KEY = 'insert your long random secret key here !'


EMAIL_HOST = 'localhost'
EMAIL_PORT = 1025
EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend'

# Example using anymail, replaces block above
# EMAIL_BACKEND = 'anymail.backends.mandrill.EmailBackend'
# DEFAULT_FROM_EMAIL = "Sefaria <[email protected]>"
# ANYMAIL = {
# "MANDRILL_API_KEY": "your api key",
# }


# ElasticSearch server
# URL to connect to ES server.
# Set this to https://sefaria.org/api/search to connect to production search.
# If ElasticSearch server has a password use the following format: http(s)://{username}:{password}@{base_url}
SEARCH_URL = "http://localhost:9200"

SEARCH_INDEX_ON_SAVE = False # Whether to send texts and source sheet to Search Host for indexing after save
SEARCH_INDEX_NAME_TEXT = 'text' # name of the ElasticSearch index to use
SEARCH_INDEX_NAME_SHEET = 'sheet'

# Node Server
USE_NODE = True
NODE_HOST = "http://localhost:3000"
NODE_TIMEOUT = 10

SEFARIA_DATA_PATH = '/path/to/your/Sefaria-Data' # used for Data
SEFARIA_EXPORT_PATH = '/path/to/your/Sefaria-Data/export' # used for exporting texts


GOOGLE_GTAG = 'your gtag id here'
GOOGLE_TAG_MANAGER_CODE = 'you tag manager code here'

HOTJAR_ID = None

# Determine which CRM connection implementations to use
CRM_TYPE = "NONE" # "SALESFORCE" || "NATIONBUILDER" || "NONE"

# Integration with a NationBuilder list
NATIONBUILDER_SLUG = ""
NATIONBUILDER_TOKEN = ""
NATIONBUILDER_CLIENT_ID = ""
NATIONBUILDER_CLIENT_SECRET = ""

# Integration with Salesforce
SALESFORCE_BASE_URL = ""
SALESFORCE_CLIENT_ID = ""
SALESFORCE_CLIENT_SECRET = ""

# Issue bans to Varnish on update.
USE_VARNISH = False
FRONT_END_URL = "http://localhost:8000" # This one wants the http://
VARNISH_ADM_ADDR = "localhost:6082" # And this one doesn't
VARNISH_HOST = "localhost"
VARNISH_FRNT_PORT = 8040
VARNISH_SECRET = "/etc/varnish/secret"
# Use ESI for user box in header.
USE_VARNISH_ESI = False

# Prevent modification of Index records
DISABLE_INDEX_SAVE = False

# Turns off search autocomplete suggestions, which are reinitialized on every server reload
# which can be annoying for local development.
DISABLE_AUTOCOMPLETER = False

# Turns on loading of machine learning models to run linker
ENABLE_LINKER = False

# Caching with Cloudflare
CLOUDFLARE_ZONE = ""
CLOUDFLARE_EMAIL = ""
CLOUDFLARE_TOKEN = ""

# Multiserver
MULTISERVER_ENABLED = False
MULTISERVER_REDIS_SERVER = "127.0.0.1"
MULTISERVER_REDIS_PORT = 6379
MULTISERVER_REDIS_DB = 0
MULTISERVER_REDIS_EVENT_CHANNEL = "msync" # Message queue on Redis
MULTISERVER_REDIS_CONFIRM_CHANNEL = "mconfirm" # Message queue on Redis

# OAUTH these fields dont need to be filled in. they are only required for oauth2client to __init__ successfully
GOOGLE_OAUTH2_CLIENT_ID = ""
GOOGLE_OAUTH2_CLIENT_SECRET = ""
# This is the field that is actually used
GOOGLE_OAUTH2_CLIENT_SECRET_FILEPATH = ""

GOOGLE_APPLICATION_CREDENTIALS_FILEPATH = ""

GEOIP_DATABASE = 'data/geoip/GeoLiteCity.dat'
GEOIPV6_DATABASE = 'data/geoip/GeoLiteCityv6.dat'

RAW_REF_MODEL_BY_LANG_FILEPATH = {
"en": None,
"he": None
}

RAW_REF_PART_MODEL_BY_LANG_FILEPATH = {
"en": None,
"he": None
}

# Simple JWT
SIMPLE_JWT = {
'ACCESS_TOKEN_LIFETIME': timedelta(days=1),
'REFRESH_TOKEN_LIFETIME': timedelta(days=90),
'ROTATE_REFRESH_TOKENS': True,
'SIGNING_KEY': 'a signing key: at least 256 bits',
}

# Celery - the following section defines variables to connect to the celery broker. This can be either redis or redis sentinel
# Either define SENTINEL_HEADLESS_URL if using sentinel or REDIS_URL for a simple redis instance
# If using sentinel, also pass any variables prefixed with SENTINEL. Otherwise, they can be left as default.
# All other variables need to be defined if connecting to either redis or redis sentinel
REDIS_URL = "redis://127.0.0.1"
REDIS_PORT = 6379 # the port exposed on either redis or redis sentinel. default for sentinel is 26379
REDIS_PASSWORD = None
SENTINEL_HEADLESS_URL = None
SENTINEL_PASSWORD = None
SENTINEL_TRANSPORT_OPTS = {}
CELERY_REDIS_BROKER_DB_NUM = 2
CELERY_REDIS_RESULT_BACKEND_DB_NUM = 3
CELERY_QUEUES = {}
# END Celery

# Key which identifies the Sefaria app as opposed to a user
# using our API outside of the app. Mainly for registration
MOBILE_APP_KEY = "MOBILE_APP_KEY"

LOGGING = {
'version': 1,
'disable_existing_loggers': False,
'formatters': {
"json_formatter": {
"()": structlog.stdlib.ProcessorFormatter,
"processor": structlog.processors.JSONRenderer(),
},
},
'handlers': {
'default': {
"class": "logging.StreamHandler",
"formatter": "json_formatter",
},
},
'loggers': {
'': {
'handlers': ['default'],
'propagate': False,
},
'django': {
'handlers': ['default'],
'propagate': False,
},
'django.request': {
'handlers': ['default'],
'propagate': False,
},
}
}

structlog.configure(
processors=[
structlog.stdlib.filter_by_level,
structlog.processors.TimeStamper(fmt="iso"),
structlog.stdlib.add_logger_name,
sefaria_logging.add_severity,
structlog.stdlib.PositionalArgumentsFormatter(),
structlog.processors.StackInfoRenderer(),
sefaria_logging.log_exception_info,
structlog.processors.UnicodeDecoder(),
sefaria_logging.decompose_request_info,
structlog.stdlib.ProcessorFormatter.wrap_for_formatter,
],
context_class=structlog.threadlocal.wrap_dict(dict),
logger_factory=structlog.stdlib.LoggerFactory(),
wrapper_class=structlog.stdlib.BoundLogger,
cache_logger_on_first_use=True,
)

SENTRY_DSN = None
CLIENT_SENTRY_DSN = None

# Fail gracefully when decorator conditional_graceful_exception on function. This should be set to True on production
# Example: If a text or ref cannot be properly loaded, fail gracefully and let the server continue to run
FAIL_GRACEFULLY = False
if "pytest" in sys.modules:
FAIL_GRACEFULLY = False
10 changes: 10 additions & 0 deletions .devcontainer/postCreate.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
#!/bin/bash

mongod --fork --logpath /var/log/mongodb.log --dbpath /data/db
redis-server --daemonize yes
cp .devcontainer/local_settings.py sefaria
npm install
npm run setup
npm run build-client
npm run build
/python3.8 manage.py migrate
3 changes: 3 additions & 0 deletions .devcontainer/postStart.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
#!/bin/bash

/python3.8 manage.py runserver

0 comments on commit b2b78ba

Please sign in to comment.