2021-07-07 07:39:45 -04:00
|
|
|
/* eslint-disable
|
|
|
|
camelcase,
|
|
|
|
no-cond-assign,
|
|
|
|
no-dupe-keys,
|
|
|
|
no-unused-vars,
|
|
|
|
*/
|
|
|
|
// TODO: This file was created by bulk-decaffeinate.
|
|
|
|
// Fix any style issues and re-enable lint.
|
2021-07-07 07:39:42 -04:00
|
|
|
/*
|
|
|
|
* decaffeinate suggestions:
|
|
|
|
* DS205: Consider reworking code to avoid use of IIFEs
|
|
|
|
* DS207: Consider shorter variations of null checks
|
|
|
|
* Full docs: https://github.com/decaffeinate/decaffeinate/blob/master/docs/suggestions.md
|
|
|
|
*/
|
2022-01-31 09:00:15 -05:00
|
|
|
let redisConfig, siteUrl
|
2021-07-07 07:43:33 -04:00
|
|
|
let e
|
|
|
|
const Path = require('path')
|
2021-07-07 07:39:42 -04:00
|
|
|
|
|
|
|
// These credentials are used for authenticating api requests
|
|
|
|
// between services that may need to go over public channels
|
2021-07-07 07:43:33 -04:00
|
|
|
const httpAuthUser = 'sharelatex'
|
|
|
|
const httpAuthPass = process.env.WEB_API_PASSWORD
|
|
|
|
const httpAuthUsers = {}
|
|
|
|
httpAuthUsers[httpAuthUser] = httpAuthPass
|
|
|
|
|
|
|
|
const parse = function (option) {
|
|
|
|
if (option != null) {
|
|
|
|
try {
|
|
|
|
const opt = JSON.parse(option)
|
|
|
|
return opt
|
|
|
|
} catch (err) {
|
|
|
|
throw new Error(`problem parsing ${option}, invalid JSON`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
|
2021-07-07 07:43:33 -04:00
|
|
|
const parseIntOrFail = function (value) {
|
|
|
|
const parsedValue = parseInt(value, 10)
|
|
|
|
if (isNaN(parsedValue)) {
|
|
|
|
throw new Error(`'${value}' is an invalid integer`)
|
|
|
|
}
|
|
|
|
return parsedValue
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
|
2021-07-07 07:43:33 -04:00
|
|
|
const DATA_DIR = '/var/lib/sharelatex/data'
|
|
|
|
const TMP_DIR = '/var/lib/sharelatex/tmp'
|
|
|
|
|
|
|
|
const settings = {
|
|
|
|
clsi: {
|
|
|
|
optimiseInDocker: process.env.OPTIMISE_PDF === 'true',
|
|
|
|
},
|
|
|
|
|
|
|
|
brandPrefix: '',
|
|
|
|
|
|
|
|
allowAnonymousReadAndWriteSharing:
|
|
|
|
process.env.SHARELATEX_ALLOW_ANONYMOUS_READ_AND_WRITE_SHARING === 'true',
|
|
|
|
|
|
|
|
// Databases
|
|
|
|
// ---------
|
|
|
|
|
|
|
|
// ShareLaTeX's main persistent data store is MongoDB (http://www.mongodb.org/)
|
|
|
|
// Documentation about the URL connection string format can be found at:
|
|
|
|
//
|
|
|
|
// http://docs.mongodb.org/manual/reference/connection-string/
|
|
|
|
//
|
|
|
|
// The following works out of the box with Mongo's default settings:
|
|
|
|
mongo: {
|
|
|
|
url: process.env.SHARELATEX_MONGO_URL || 'mongodb://dockerhost/sharelatex',
|
|
|
|
},
|
|
|
|
|
|
|
|
// Redis is used in ShareLaTeX for high volume queries, like real-time
|
|
|
|
// editing, and session management.
|
|
|
|
//
|
|
|
|
// The following config will work with Redis's default settings:
|
|
|
|
redis: {
|
|
|
|
web: (redisConfig = {
|
|
|
|
host: process.env.SHARELATEX_REDIS_HOST || 'dockerhost',
|
|
|
|
port: process.env.SHARELATEX_REDIS_PORT || '6379',
|
|
|
|
password: process.env.SHARELATEX_REDIS_PASS || undefined,
|
|
|
|
key_schema: {
|
|
|
|
// document-updater
|
|
|
|
blockingKey({ doc_id }) {
|
|
|
|
return `Blocking:${doc_id}`
|
|
|
|
},
|
|
|
|
docLines({ doc_id }) {
|
|
|
|
return `doclines:${doc_id}`
|
|
|
|
},
|
|
|
|
docOps({ doc_id }) {
|
|
|
|
return `DocOps:${doc_id}`
|
|
|
|
},
|
|
|
|
docVersion({ doc_id }) {
|
|
|
|
return `DocVersion:${doc_id}`
|
|
|
|
},
|
|
|
|
docHash({ doc_id }) {
|
|
|
|
return `DocHash:${doc_id}`
|
|
|
|
},
|
|
|
|
projectKey({ doc_id }) {
|
|
|
|
return `ProjectId:${doc_id}`
|
|
|
|
},
|
|
|
|
docsInProject({ project_id }) {
|
|
|
|
return `DocsIn:${project_id}`
|
|
|
|
},
|
|
|
|
ranges({ doc_id }) {
|
|
|
|
return `Ranges:${doc_id}`
|
|
|
|
},
|
|
|
|
// document-updater:realtime
|
|
|
|
pendingUpdates({ doc_id }) {
|
|
|
|
return `PendingUpdates:${doc_id}`
|
|
|
|
},
|
|
|
|
// document-updater:history
|
|
|
|
uncompressedHistoryOps({ doc_id }) {
|
|
|
|
return `UncompressedHistoryOps:${doc_id}`
|
|
|
|
},
|
|
|
|
docsWithHistoryOps({ project_id }) {
|
|
|
|
return `DocsWithHistoryOps:${project_id}`
|
|
|
|
},
|
|
|
|
// document-updater:lock
|
|
|
|
blockingKey({ doc_id }) {
|
|
|
|
return `Blocking:${doc_id}`
|
|
|
|
},
|
|
|
|
// realtime
|
|
|
|
clientsInProject({ project_id }) {
|
|
|
|
return `clients_in_project:${project_id}`
|
|
|
|
},
|
|
|
|
connectedUser({ project_id, client_id }) {
|
|
|
|
return `connected_user:${project_id}:${client_id}`
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
fairy: redisConfig,
|
2023-05-25 04:32:59 -04:00
|
|
|
// document-updater
|
2021-07-07 07:43:33 -04:00
|
|
|
realtime: redisConfig,
|
|
|
|
documentupdater: redisConfig,
|
|
|
|
lock: redisConfig,
|
|
|
|
history: redisConfig,
|
|
|
|
websessions: redisConfig,
|
|
|
|
api: redisConfig,
|
|
|
|
pubsub: redisConfig,
|
|
|
|
project_history: redisConfig,
|
2023-01-20 06:06:26 -05:00
|
|
|
|
|
|
|
project_history_migration: {
|
|
|
|
host: redisConfig.host,
|
|
|
|
port: redisConfig.port,
|
|
|
|
password: redisConfig.password,
|
|
|
|
maxRetriesPerRequest: parseInt(
|
|
|
|
process.env.REDIS_MAX_RETRIES_PER_REQUEST || '20'
|
|
|
|
),
|
|
|
|
key_schema: {
|
|
|
|
projectHistoryOps({ projectId }) {
|
|
|
|
return `ProjectHistory:Ops:{${projectId}}` // NOTE: the extra braces are intentional
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2021-07-07 07:43:33 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
// Local disk caching
|
|
|
|
// ------------------
|
|
|
|
path: {
|
|
|
|
// If we ever need to write something to disk (e.g. incoming requests
|
|
|
|
// that need processing but may be too big for memory), then write
|
|
|
|
// them to disk here:
|
|
|
|
dumpFolder: Path.join(TMP_DIR, 'dumpFolder'),
|
|
|
|
// Where to write uploads before they are processed
|
|
|
|
uploadFolder: Path.join(TMP_DIR, 'uploads'),
|
2023-02-02 10:12:52 -05:00
|
|
|
// Where to write intermediate file for full project history migration
|
|
|
|
projectHistories: Path.join(TMP_DIR, 'projectHistories'),
|
2021-07-07 07:43:33 -04:00
|
|
|
// Where to write the project to disk before running LaTeX on it
|
|
|
|
compilesDir: Path.join(DATA_DIR, 'compiles'),
|
|
|
|
// Where to cache downloaded URLs for the CLSI
|
|
|
|
clsiCacheDir: Path.join(DATA_DIR, 'cache'),
|
|
|
|
// Where to write the output files to disk after running LaTeX
|
|
|
|
outputDir: Path.join(DATA_DIR, 'output'),
|
|
|
|
},
|
|
|
|
|
|
|
|
// Server Config
|
|
|
|
// -------------
|
|
|
|
|
|
|
|
// Where your instance of ShareLaTeX can be found publicly. This is used
|
|
|
|
// when emails are sent out and in generated links:
|
|
|
|
siteUrl: (siteUrl = process.env.SHARELATEX_SITE_URL || 'http://localhost'),
|
|
|
|
|
2022-10-18 06:41:19 -04:00
|
|
|
// Status page URL as displayed on the maintenance/500 pages.
|
|
|
|
statusPageUrl: process.env.SHARELATEX_STATUS_PAGE_URL,
|
|
|
|
|
2021-07-07 07:43:33 -04:00
|
|
|
// The name this is used to describe your ShareLaTeX Installation
|
|
|
|
appName: process.env.SHARELATEX_APP_NAME || 'ShareLaTeX (Community Edition)',
|
|
|
|
|
|
|
|
restrictInvitesToExistingAccounts:
|
|
|
|
process.env.SHARELATEX_RESTRICT_INVITES_TO_EXISTING_ACCOUNTS === 'true',
|
|
|
|
|
|
|
|
nav: {
|
|
|
|
title:
|
|
|
|
process.env.SHARELATEX_NAV_TITLE ||
|
|
|
|
process.env.SHARELATEX_APP_NAME ||
|
|
|
|
'ShareLaTeX Community Edition',
|
|
|
|
},
|
|
|
|
|
|
|
|
// The email address which users will be directed to as the main point of
|
|
|
|
// contact for this installation of ShareLaTeX.
|
|
|
|
adminEmail: process.env.SHARELATEX_ADMIN_EMAIL || 'placeholder@example.com',
|
|
|
|
|
|
|
|
// If provided, a sessionSecret is used to sign cookies so that they cannot be
|
|
|
|
// spoofed. This is recommended.
|
|
|
|
security: {
|
|
|
|
sessionSecret:
|
|
|
|
process.env.SHARELATEX_SESSION_SECRET || process.env.CRYPTO_RANDOM,
|
|
|
|
},
|
|
|
|
|
|
|
|
// These credentials are used for authenticating api requests
|
|
|
|
// between services that may need to go over public channels
|
|
|
|
httpAuthUsers,
|
|
|
|
|
|
|
|
// Should javascript assets be served minified or not.
|
|
|
|
useMinifiedJs: true,
|
|
|
|
|
|
|
|
// Should static assets be sent with a header to tell the browser to cache
|
|
|
|
// them. This should be false in development where changes are being made,
|
|
|
|
// but should be set to true in production.
|
|
|
|
cacheStaticAssets: true,
|
|
|
|
|
|
|
|
// If you are running ShareLaTeX over https, set this to true to send the
|
|
|
|
// cookie with a secure flag (recommended).
|
|
|
|
secureCookie: process.env.SHARELATEX_SECURE_COOKIE != null,
|
|
|
|
|
|
|
|
// If you are running ShareLaTeX behind a proxy (like Apache, Nginx, etc)
|
|
|
|
// then set this to true to allow it to correctly detect the forwarded IP
|
|
|
|
// address and http/https protocol information.
|
|
|
|
|
|
|
|
behindProxy: process.env.SHARELATEX_BEHIND_PROXY || false,
|
2023-05-09 06:39:13 -04:00
|
|
|
trustedProxyIps: process.env.SHARELATEX_TRUSTED_PROXY_IPS,
|
2021-07-07 07:43:33 -04:00
|
|
|
|
2023-09-28 05:38:24 -04:00
|
|
|
// The amount of time, in milliseconds, until the (rolling) cookie session expires
|
|
|
|
cookieSessionLength: parseInt(
|
|
|
|
process.env.SHARELATEX_COOKIE_SESSION_LENGTH || 5 * 24 * 60 * 60 * 1000, // default 5 days
|
|
|
|
10
|
|
|
|
),
|
|
|
|
|
2021-07-07 07:43:33 -04:00
|
|
|
i18n: {
|
|
|
|
subdomainLang: {
|
|
|
|
www: {
|
|
|
|
lngCode: process.env.SHARELATEX_SITE_LANGUAGE || 'en',
|
|
|
|
url: siteUrl,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
defaultLng: process.env.SHARELATEX_SITE_LANGUAGE || 'en',
|
|
|
|
},
|
|
|
|
|
|
|
|
currentImageName: process.env.TEX_LIVE_DOCKER_IMAGE,
|
|
|
|
|
|
|
|
apis: {
|
|
|
|
web: {
|
|
|
|
url: 'http://localhost:3000',
|
|
|
|
user: httpAuthUser,
|
|
|
|
pass: httpAuthPass,
|
|
|
|
},
|
2023-01-20 06:06:26 -05:00
|
|
|
project_history: {
|
|
|
|
sendProjectStructureOps: true,
|
|
|
|
url: 'http://localhost:3054',
|
|
|
|
},
|
|
|
|
v1_history: {
|
2023-04-04 08:14:56 -04:00
|
|
|
url: process.env.V1_HISTORY_URL || 'http://localhost:3100/api',
|
2023-01-20 06:06:26 -05:00
|
|
|
user: 'staging',
|
|
|
|
pass: process.env.STAGING_PASSWORD,
|
2023-10-24 09:28:37 -04:00
|
|
|
requestTimeout: parseInt(
|
|
|
|
process.env.SHARELATEX_HISTORY_V1_HTTP_REQUEST_TIMEOUT || '300000', // default is 5min
|
|
|
|
10
|
|
|
|
),
|
2023-02-10 04:07:10 -05:00
|
|
|
},
|
2021-07-07 07:43:33 -04:00
|
|
|
},
|
|
|
|
references: {},
|
|
|
|
notifications: undefined,
|
|
|
|
|
|
|
|
defaultFeatures: {
|
|
|
|
collaborators: -1,
|
|
|
|
dropbox: true,
|
|
|
|
versioning: true,
|
|
|
|
compileTimeout: parseIntOrFail(process.env.COMPILE_TIMEOUT || 180),
|
|
|
|
compileGroup: 'standard',
|
|
|
|
trackChanges: true,
|
|
|
|
templates: true,
|
|
|
|
references: true,
|
|
|
|
},
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
// # OPTIONAL CONFIGURABLE SETTINGS
|
2021-07-07 07:39:42 -04:00
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_LEFT_FOOTER != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
try {
|
|
|
|
settings.nav.left_footer = JSON.parse(process.env.SHARELATEX_LEFT_FOOTER)
|
|
|
|
} catch (error) {
|
|
|
|
e = error
|
|
|
|
console.error('could not parse SHARELATEX_LEFT_FOOTER, not valid JSON')
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_RIGHT_FOOTER != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.nav.right_footer = process.env.SHARELATEX_RIGHT_FOOTER
|
|
|
|
try {
|
|
|
|
settings.nav.right_footer = JSON.parse(process.env.SHARELATEX_RIGHT_FOOTER)
|
|
|
|
} catch (error1) {
|
|
|
|
e = error1
|
|
|
|
console.error('could not parse SHARELATEX_RIGHT_FOOTER, not valid JSON')
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_HEADER_IMAGE_URL != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.nav.custom_logo = process.env.SHARELATEX_HEADER_IMAGE_URL
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_HEADER_NAV_LINKS != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
console.error(`\
|
2017-01-12 08:51:10 -05:00
|
|
|
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
|
|
|
|
#
|
|
|
|
# WARNING: SHARELATEX_HEADER_NAV_LINKS is no longer supported
|
|
|
|
# See https://github.com/sharelatex/sharelatex/wiki/Configuring-Headers,-Footers-&-Logo
|
|
|
|
#
|
2021-07-07 07:39:42 -04:00
|
|
|
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #\
|
2021-07-07 07:43:33 -04:00
|
|
|
`)
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_HEADER_EXTRAS != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
try {
|
|
|
|
settings.nav.header_extras = JSON.parse(
|
|
|
|
process.env.SHARELATEX_HEADER_EXTRAS
|
|
|
|
)
|
|
|
|
} catch (error2) {
|
|
|
|
e = error2
|
|
|
|
console.error('could not parse SHARELATEX_HEADER_EXTRAS, not valid JSON')
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sending Email
|
|
|
|
// -------------
|
|
|
|
//
|
|
|
|
// You must configure a mail server to be able to send invite emails from
|
|
|
|
// ShareLaTeX. The config settings are passed to nodemailer. See the nodemailer
|
|
|
|
// documentation for available options:
|
|
|
|
//
|
|
|
|
// http://www.nodemailer.com/docs/transports
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_EMAIL_FROM_ADDRESS != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.email = {
|
|
|
|
fromAddress: process.env.SHARELATEX_EMAIL_FROM_ADDRESS,
|
|
|
|
replyTo: process.env.SHARELATEX_EMAIL_REPLY_TO || '',
|
|
|
|
driver: process.env.SHARELATEX_EMAIL_DRIVER,
|
|
|
|
parameters: {
|
|
|
|
// AWS Creds
|
|
|
|
AWSAccessKeyID: process.env.SHARELATEX_EMAIL_AWS_SES_ACCESS_KEY_ID,
|
|
|
|
AWSSecretKey: process.env.SHARELATEX_EMAIL_AWS_SES_SECRET_KEY,
|
|
|
|
|
|
|
|
// SMTP Creds
|
|
|
|
host: process.env.SHARELATEX_EMAIL_SMTP_HOST,
|
|
|
|
port: process.env.SHARELATEX_EMAIL_SMTP_PORT,
|
|
|
|
secure: parse(process.env.SHARELATEX_EMAIL_SMTP_SECURE),
|
|
|
|
ignoreTLS: parse(process.env.SHARELATEX_EMAIL_SMTP_IGNORE_TLS),
|
|
|
|
name: process.env.SHARELATEX_EMAIL_SMTP_NAME,
|
|
|
|
logger: process.env.SHARELATEX_EMAIL_SMTP_LOGGER === 'true',
|
|
|
|
},
|
|
|
|
|
|
|
|
textEncoding: process.env.SHARELATEX_EMAIL_TEXT_ENCODING,
|
|
|
|
template: {
|
|
|
|
customFooter: process.env.SHARELATEX_CUSTOM_EMAIL_FOOTER,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process.env.SHARELATEX_EMAIL_AWS_SES_REGION != null) {
|
|
|
|
settings.email.parameters.region =
|
|
|
|
process.env.SHARELATEX_EMAIL_AWS_SES_REGION
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
process.env.SHARELATEX_EMAIL_SMTP_USER != null ||
|
|
|
|
process.env.SHARELATEX_EMAIL_SMTP_PASS != null
|
|
|
|
) {
|
|
|
|
settings.email.parameters.auth = {
|
|
|
|
user: process.env.SHARELATEX_EMAIL_SMTP_USER,
|
|
|
|
pass: process.env.SHARELATEX_EMAIL_SMTP_PASS,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process.env.SHARELATEX_EMAIL_SMTP_TLS_REJECT_UNAUTH != null) {
|
|
|
|
settings.email.parameters.tls = {
|
|
|
|
rejectUnauthorized: parse(
|
|
|
|
process.env.SHARELATEX_EMAIL_SMTP_TLS_REJECT_UNAUTH
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
2017-01-06 06:28:32 -05:00
|
|
|
|
2021-07-07 07:39:42 -04:00
|
|
|
// i18n
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_LANG_DOMAIN_MAPPING != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.i18n.subdomainLang = parse(
|
|
|
|
process.env.SHARELATEX_LANG_DOMAIN_MAPPING
|
|
|
|
)
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
2016-04-06 12:07:59 -04:00
|
|
|
|
2021-07-07 07:39:42 -04:00
|
|
|
// Password Settings
|
|
|
|
// -----------
|
|
|
|
// These restrict the passwords users can use when registering
|
|
|
|
// opts are from http://antelle.github.io/passfield
|
2021-07-07 07:43:33 -04:00
|
|
|
if (
|
|
|
|
process.env.SHARELATEX_PASSWORD_VALIDATION_PATTERN ||
|
|
|
|
process.env.SHARELATEX_PASSWORD_VALIDATION_MIN_LENGTH ||
|
|
|
|
process.env.SHARELATEX_PASSWORD_VALIDATION_MAX_LENGTH
|
|
|
|
) {
|
|
|
|
settings.passwordStrengthOptions = {
|
|
|
|
pattern: process.env.SHARELATEX_PASSWORD_VALIDATION_PATTERN || 'aA$3',
|
|
|
|
length: {
|
|
|
|
min: process.env.SHARELATEX_PASSWORD_VALIDATION_MIN_LENGTH || 8,
|
2023-07-27 08:50:52 -04:00
|
|
|
max: process.env.SHARELATEX_PASSWORD_VALIDATION_MAX_LENGTH || 72,
|
2021-07-07 07:43:33 -04:00
|
|
|
},
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
2016-04-06 12:07:59 -04:00
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
// ######################
|
2021-07-07 07:39:42 -04:00
|
|
|
// ShareLaTeX Server Pro
|
2021-07-07 07:39:45 -04:00
|
|
|
// ######################
|
2016-05-23 06:30:25 -04:00
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (parse(process.env.SHARELATEX_IS_SERVER_PRO) === true) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.bypassPercentageRollouts = true
|
|
|
|
settings.apis.references = { url: 'http://localhost:3040' }
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
2016-05-23 06:30:25 -04:00
|
|
|
|
2021-07-07 07:39:42 -04:00
|
|
|
// LDAP - SERVER PRO ONLY
|
|
|
|
// ----------
|
2016-12-09 04:22:31 -05:00
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_LDAP_HOST) {
|
2021-07-07 07:43:33 -04:00
|
|
|
console.error(`\
|
2016-12-09 04:22:31 -05:00
|
|
|
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #
|
|
|
|
#
|
|
|
|
# WARNING: The LDAP configuration format has changed in version 0.5.1
|
|
|
|
# See https://github.com/sharelatex/sharelatex/wiki/Server-Pro:-LDAP-Config
|
|
|
|
#
|
2021-07-07 07:39:42 -04:00
|
|
|
# # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # #\
|
2021-07-07 07:43:33 -04:00
|
|
|
`)
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_LDAP_URL) {
|
2021-07-07 07:43:33 -04:00
|
|
|
let _ldap_connect_timeout,
|
|
|
|
_ldap_group_search_attribs,
|
|
|
|
_ldap_search_attribs,
|
|
|
|
_ldap_timeout
|
|
|
|
settings.externalAuth = true
|
|
|
|
settings.ldap = {
|
|
|
|
emailAtt: process.env.SHARELATEX_LDAP_EMAIL_ATT,
|
|
|
|
nameAtt: process.env.SHARELATEX_LDAP_NAME_ATT,
|
|
|
|
lastNameAtt: process.env.SHARELATEX_LDAP_LAST_NAME_ATT,
|
|
|
|
updateUserDetailsOnLogin:
|
|
|
|
process.env.SHARELATEX_LDAP_UPDATE_USER_DETAILS_ON_LOGIN === 'true',
|
|
|
|
placeholder: process.env.SHARELATEX_LDAP_PLACEHOLDER,
|
|
|
|
server: {
|
|
|
|
url: process.env.SHARELATEX_LDAP_URL,
|
|
|
|
bindDn: process.env.SHARELATEX_LDAP_BIND_DN,
|
|
|
|
bindCredentials: process.env.SHARELATEX_LDAP_BIND_CREDENTIALS,
|
|
|
|
bindProperty: process.env.SHARELATEX_LDAP_BIND_PROPERTY,
|
|
|
|
searchBase: process.env.SHARELATEX_LDAP_SEARCH_BASE,
|
|
|
|
searchScope: process.env.SHARELATEX_LDAP_SEARCH_SCOPE,
|
|
|
|
searchFilter: process.env.SHARELATEX_LDAP_SEARCH_FILTER,
|
|
|
|
searchAttributes: (_ldap_search_attribs =
|
|
|
|
process.env.SHARELATEX_LDAP_SEARCH_ATTRIBUTES)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return JSON.parse(_ldap_search_attribs)
|
|
|
|
} catch (error3) {
|
|
|
|
e = error3
|
|
|
|
return console.error(
|
|
|
|
'could not parse SHARELATEX_LDAP_SEARCH_ATTRIBUTES'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
groupDnProperty: process.env.SHARELATEX_LDAP_GROUP_DN_PROPERTY,
|
|
|
|
groupSearchBase: process.env.SHARELATEX_LDAP_GROUP_SEARCH_BASE,
|
|
|
|
groupSearchScope: process.env.SHARELATEX_LDAP_GROUP_SEARCH_SCOPE,
|
|
|
|
groupSearchFilter: process.env.SHARELATEX_LDAP_GROUP_SEARCH_FILTER,
|
|
|
|
groupSearchAttributes: (_ldap_group_search_attribs =
|
|
|
|
process.env.SHARELATEX_LDAP_GROUP_SEARCH_ATTRIBUTES)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return JSON.parse(_ldap_group_search_attribs)
|
|
|
|
} catch (error4) {
|
|
|
|
e = error4
|
|
|
|
return console.error(
|
|
|
|
'could not parse SHARELATEX_LDAP_GROUP_SEARCH_ATTRIBUTES'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
cache: process.env.SHARELATEX_LDAP_CACHE === 'true',
|
|
|
|
timeout: (_ldap_timeout = process.env.SHARELATEX_LDAP_TIMEOUT)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return parseIntOrFail(_ldap_timeout)
|
|
|
|
} catch (error5) {
|
|
|
|
e = error5
|
|
|
|
return console.error('Cannot parse SHARELATEX_LDAP_TIMEOUT')
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
connectTimeout: (_ldap_connect_timeout =
|
|
|
|
process.env.SHARELATEX_LDAP_CONNECT_TIMEOUT)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return parseIntOrFail(_ldap_connect_timeout)
|
|
|
|
} catch (error6) {
|
|
|
|
e = error6
|
|
|
|
return console.error(
|
|
|
|
'Cannot parse SHARELATEX_LDAP_CONNECT_TIMEOUT'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process.env.SHARELATEX_LDAP_TLS_OPTS_CA_PATH) {
|
|
|
|
let ca, ca_paths
|
|
|
|
try {
|
|
|
|
ca = JSON.parse(process.env.SHARELATEX_LDAP_TLS_OPTS_CA_PATH)
|
|
|
|
} catch (error7) {
|
|
|
|
e = error7
|
|
|
|
console.error(
|
|
|
|
'could not parse SHARELATEX_LDAP_TLS_OPTS_CA_PATH, invalid JSON'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof ca === 'string') {
|
|
|
|
ca_paths = [ca]
|
|
|
|
} else if (
|
|
|
|
typeof ca === 'object' &&
|
|
|
|
(ca != null ? ca.length : undefined) != null
|
|
|
|
) {
|
|
|
|
ca_paths = ca
|
|
|
|
} else {
|
|
|
|
console.error('problem parsing SHARELATEX_LDAP_TLS_OPTS_CA_PATH')
|
|
|
|
}
|
|
|
|
|
|
|
|
settings.ldap.server.tlsOptions = {
|
|
|
|
rejectUnauthorized:
|
|
|
|
process.env.SHARELATEX_LDAP_TLS_OPTS_REJECT_UNAUTH === 'true',
|
|
|
|
ca: ca_paths, // e.g.'/etc/ldap/ca_certs.pem'
|
|
|
|
}
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_SAML_ENTRYPOINT) {
|
2021-07-07 07:43:33 -04:00
|
|
|
// NOTE: see https://github.com/node-saml/passport-saml/blob/master/README.md for docs of `server` options
|
|
|
|
let _saml_additionalAuthorizeParams,
|
|
|
|
_saml_additionalLogoutParams,
|
|
|
|
_saml_additionalParams,
|
|
|
|
_saml_expiration,
|
|
|
|
_saml_skew
|
|
|
|
settings.externalAuth = true
|
|
|
|
settings.saml = {
|
|
|
|
updateUserDetailsOnLogin:
|
|
|
|
process.env.SHARELATEX_SAML_UPDATE_USER_DETAILS_ON_LOGIN === 'true',
|
|
|
|
identityServiceName: process.env.SHARELATEX_SAML_IDENTITY_SERVICE_NAME,
|
|
|
|
emailField:
|
|
|
|
process.env.SHARELATEX_SAML_EMAIL_FIELD ||
|
|
|
|
process.env.SHARELATEX_SAML_EMAIL_FIELD_NAME,
|
|
|
|
firstNameField: process.env.SHARELATEX_SAML_FIRST_NAME_FIELD,
|
|
|
|
lastNameField: process.env.SHARELATEX_SAML_LAST_NAME_FIELD,
|
|
|
|
server: {
|
|
|
|
// strings
|
|
|
|
entryPoint: process.env.SHARELATEX_SAML_ENTRYPOINT,
|
|
|
|
callbackUrl: process.env.SHARELATEX_SAML_CALLBACK_URL,
|
|
|
|
issuer: process.env.SHARELATEX_SAML_ISSUER,
|
|
|
|
decryptionPvk: process.env.SHARELATEX_SAML_DECRYPTION_PVK,
|
|
|
|
decryptionCert: process.env.SHARELATEX_SAML_DECRYPTION_CERT,
|
2022-03-24 07:36:49 -04:00
|
|
|
signingCert: process.env.SHARELATEX_SAML_SIGNING_CERT,
|
2021-07-07 07:43:33 -04:00
|
|
|
signatureAlgorithm: process.env.SHARELATEX_SAML_SIGNATURE_ALGORITHM,
|
|
|
|
identifierFormat: process.env.SHARELATEX_SAML_IDENTIFIER_FORMAT,
|
|
|
|
attributeConsumingServiceIndex:
|
|
|
|
process.env.SHARELATEX_SAML_ATTRIBUTE_CONSUMING_SERVICE_INDEX,
|
2022-12-12 05:58:20 -05:00
|
|
|
authnContext:
|
|
|
|
process.env.SHARELATEX_SAML_AUTHN_CONTEXT &&
|
|
|
|
process.env.SHARELATEX_SAML_AUTHN_CONTEXT.split(','),
|
2021-07-07 07:43:33 -04:00
|
|
|
authnRequestBinding: process.env.SHARELATEX_SAML_AUTHN_REQUEST_BINDING,
|
|
|
|
validateInResponseTo: process.env.SHARELATEX_SAML_VALIDATE_IN_RESPONSE_TO,
|
|
|
|
cacheProvider: process.env.SHARELATEX_SAML_CACHE_PROVIDER,
|
|
|
|
logoutUrl: process.env.SHARELATEX_SAML_LOGOUT_URL,
|
|
|
|
logoutCallbackUrl: process.env.SHARELATEX_SAML_LOGOUT_CALLBACK_URL,
|
|
|
|
disableRequestedAuthnContext:
|
|
|
|
process.env.SHARELATEX_SAML_DISABLE_REQUESTED_AUTHN_CONTEXT === 'true',
|
|
|
|
forceAuthn: process.env.SHARELATEX_SAML_FORCE_AUTHN === 'true',
|
|
|
|
skipRequestCompression:
|
|
|
|
process.env.SHARELATEX_SAML_SKIP_REQUEST_COMPRESSION === 'true',
|
|
|
|
acceptedClockSkewMs: (_saml_skew =
|
|
|
|
process.env.SHARELATEX_SAML_ACCEPTED_CLOCK_SKEW_MS)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return parseIntOrFail(_saml_skew)
|
|
|
|
} catch (error8) {
|
|
|
|
e = error8
|
|
|
|
return console.error(
|
|
|
|
'Cannot parse SHARELATEX_SAML_ACCEPTED_CLOCK_SKEW_MS'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
requestIdExpirationPeriodMs: (_saml_expiration =
|
|
|
|
process.env.SHARELATEX_SAML_REQUEST_ID_EXPIRATION_PERIOD_MS)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return parseIntOrFail(_saml_expiration)
|
|
|
|
} catch (error9) {
|
|
|
|
e = error9
|
|
|
|
return console.error(
|
|
|
|
'Cannot parse SHARELATEX_SAML_REQUEST_ID_EXPIRATION_PERIOD_MS'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
additionalParams: (_saml_additionalParams =
|
|
|
|
process.env.SHARELATEX_SAML_ADDITIONAL_PARAMS)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return JSON.parse(_saml_additionalParams)
|
|
|
|
} catch (error10) {
|
|
|
|
e = error10
|
|
|
|
return console.error(
|
|
|
|
'Cannot parse SHARELATEX_SAML_ADDITIONAL_PARAMS'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
additionalAuthorizeParams: (_saml_additionalAuthorizeParams =
|
|
|
|
process.env.SHARELATEX_SAML_ADDITIONAL_AUTHORIZE_PARAMS)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return JSON.parse(_saml_additionalAuthorizeParams)
|
|
|
|
} catch (error11) {
|
|
|
|
e = error11
|
|
|
|
return console.error(
|
|
|
|
'Cannot parse SHARELATEX_SAML_ADDITIONAL_AUTHORIZE_PARAMS'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
additionalLogoutParams: (_saml_additionalLogoutParams =
|
|
|
|
process.env.SHARELATEX_SAML_ADDITIONAL_LOGOUT_PARAMS)
|
|
|
|
? (() => {
|
|
|
|
try {
|
|
|
|
return JSON.parse(_saml_additionalLogoutParams)
|
|
|
|
} catch (error12) {
|
|
|
|
e = error12
|
|
|
|
return console.error(
|
|
|
|
'Cannot parse SHARELATEX_SAML_ADDITIONAL_LOGOUT_PARAMS'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
: undefined,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// SHARELATEX_SAML_CERT cannot be empty
|
|
|
|
// https://github.com/node-saml/passport-saml/commit/f6b1c885c0717f1083c664345556b535f217c102
|
|
|
|
if (process.env.SHARELATEX_SAML_CERT) {
|
|
|
|
settings.saml.server.cert = process.env.SHARELATEX_SAML_CERT
|
2021-07-09 10:52:26 -04:00
|
|
|
settings.saml.server.privateKey = process.env.SHARELATEX_SAML_PRIVATE_CERT
|
2021-07-07 07:43:33 -04:00
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compiler
|
|
|
|
// --------
|
2021-07-07 07:43:33 -04:00
|
|
|
if (process.env.SANDBOXED_COMPILES === 'true') {
|
|
|
|
settings.clsi = {
|
|
|
|
dockerRunner: true,
|
|
|
|
docker: {
|
|
|
|
image: process.env.TEX_LIVE_DOCKER_IMAGE,
|
|
|
|
env: {
|
|
|
|
HOME: '/tmp',
|
|
|
|
PATH:
|
|
|
|
process.env.COMPILER_PATH ||
|
|
|
|
'/usr/local/texlive/2015/bin/x86_64-linux:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin',
|
|
|
|
},
|
|
|
|
user: 'www-data',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if (settings.path == null) {
|
|
|
|
settings.path = {}
|
|
|
|
}
|
|
|
|
settings.path.synctexBaseDir = () => '/compile'
|
|
|
|
if (process.env.SANDBOXED_COMPILES_SIBLING_CONTAINERS === 'true') {
|
|
|
|
console.log('Using sibling containers for sandboxed compiles')
|
|
|
|
if (process.env.SANDBOXED_COMPILES_HOST_DIR) {
|
|
|
|
settings.path.sandboxedCompilesHostDir =
|
|
|
|
process.env.SANDBOXED_COMPILES_HOST_DIR
|
|
|
|
} else {
|
|
|
|
console.error(
|
|
|
|
'Sibling containers, but SANDBOXED_COMPILES_HOST_DIR not set'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Templates
|
|
|
|
// ---------
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_TEMPLATES_USER_ID) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.templates = {
|
|
|
|
mountPointUrl: '/templates',
|
|
|
|
user_id: process.env.SHARELATEX_TEMPLATES_USER_ID,
|
|
|
|
}
|
|
|
|
|
|
|
|
settings.templateLinks = parse(
|
|
|
|
process.env.SHARELATEX_NEW_PROJECT_TEMPLATE_LINKS
|
|
|
|
)
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
2016-05-23 06:30:25 -04:00
|
|
|
|
2021-07-07 07:39:42 -04:00
|
|
|
// /Learn
|
|
|
|
// -------
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_PROXY_LEARN != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.proxyLearn = parse(process.env.SHARELATEX_PROXY_LEARN)
|
2022-04-07 05:50:33 -04:00
|
|
|
if (settings.proxyLearn) {
|
2023-02-10 04:07:10 -05:00
|
|
|
settings.nav.header_extras = [
|
|
|
|
{
|
|
|
|
url: '/learn',
|
|
|
|
text: 'documentation',
|
|
|
|
},
|
|
|
|
].concat(settings.nav.header_extras || [])
|
2022-04-07 05:50:33 -04:00
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
2016-04-06 12:07:59 -04:00
|
|
|
|
2021-07-07 07:39:42 -04:00
|
|
|
// /References
|
|
|
|
// -----------
|
2021-07-07 07:39:45 -04:00
|
|
|
if (process.env.SHARELATEX_ELASTICSEARCH_URL != null) {
|
2021-07-07 07:43:33 -04:00
|
|
|
settings.references.elasticsearch = {
|
|
|
|
host: process.env.SHARELATEX_ELASTICSEARCH_URL,
|
|
|
|
}
|
2021-07-07 07:39:42 -04:00
|
|
|
}
|
|
|
|
|
2023-02-10 04:07:10 -05:00
|
|
|
// filestore
|
|
|
|
switch (process.env.SHARELATEX_FILESTORE_BACKEND) {
|
|
|
|
case 's3':
|
|
|
|
settings.filestore = {
|
|
|
|
backend: 's3',
|
|
|
|
stores: {
|
|
|
|
user_files: process.env.SHARELATEX_FILESTORE_USER_FILES_BUCKET_NAME,
|
|
|
|
template_files:
|
|
|
|
process.env.SHARELATEX_FILESTORE_TEMPLATE_FILES_BUCKET_NAME,
|
|
|
|
},
|
|
|
|
s3: {
|
|
|
|
key:
|
|
|
|
process.env.SHARELATEX_FILESTORE_S3_ACCESS_KEY_ID ||
|
|
|
|
process.env.AWS_ACCESS_KEY_ID,
|
|
|
|
secret:
|
|
|
|
process.env.SHARELATEX_FILESTORE_S3_SECRET_ACCESS_KEY ||
|
2023-02-10 10:57:34 -05:00
|
|
|
process.env.AWS_SECRET_ACCESS_KEY,
|
2023-02-10 04:07:10 -05:00
|
|
|
endpoint: process.env.SHARELATEX_FILESTORE_S3_ENDPOINT,
|
|
|
|
pathStyle: process.env.SHARELATEX_FILESTORE_S3_PATH_STYLE === 'true',
|
|
|
|
region:
|
|
|
|
process.env.SHARELATEX_FILESTORE_S3_REGION ||
|
|
|
|
process.env.AWS_DEFAULT_REGION,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
settings.filestore = {
|
|
|
|
backend: 'fs',
|
|
|
|
stores: {
|
|
|
|
user_files: Path.join(DATA_DIR, 'user_files'),
|
|
|
|
template_files: Path.join(DATA_DIR, 'template_files'),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-07 07:39:42 -04:00
|
|
|
// With lots of incoming and outgoing HTTP connections to different services,
|
|
|
|
// sometimes long running, it is a good idea to increase the default number
|
|
|
|
// of sockets that Node will hold open.
|
2021-07-07 07:43:33 -04:00
|
|
|
const http = require('http')
|
|
|
|
http.globalAgent.maxSockets = 300
|
|
|
|
const https = require('https')
|
|
|
|
https.globalAgent.maxSockets = 300
|
2021-07-07 07:39:42 -04:00
|
|
|
|
2021-07-07 07:43:33 -04:00
|
|
|
module.exports = settings
|