2011-05-14 18:42:04 +02:00
/ * *
2013-11-26 09:20:59 +01:00
* The Settings Modul reads the settings out of settings . json and provides
2011-05-30 16:53:11 +02:00
* this information to the other modules
* /
/ *
2011-08-11 16:26:41 +02:00
* 2011 Peter 'Pita' Martischka ( Primary Technology Ltd )
2011-05-14 18:42:04 +02:00
*
* Licensed under the Apache License , Version 2.0 ( the "License" ) ;
* you may not use this file except in compliance with the License .
* You may obtain a copy of the License at
*
* http : //www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing , software
* distributed under the License is distributed on an "AS-IS" BASIS ,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND , either express or implied .
* See the License for the specific language governing permissions and
* limitations under the License .
* /
2011-05-14 18:22:25 +02:00
var fs = require ( "fs" ) ;
2011-12-18 06:18:35 +01:00
var os = require ( "os" ) ;
2012-01-05 11:24:33 +01:00
var path = require ( 'path' ) ;
2012-02-21 20:20:45 +01:00
var argv = require ( './Cli' ) . argv ;
2012-02-26 13:07:51 +01:00
var npm = require ( "npm/lib/npm.js" ) ;
2013-11-26 09:20:59 +01:00
var jsonminify = require ( "jsonminify" ) ;
2013-01-13 12:20:49 +01:00
var log4js = require ( "log4js" ) ;
2014-01-15 20:25:47 +01:00
var randomString = require ( "./randomstring" ) ;
2015-04-06 01:13:38 +02:00
var suppressDisableMsg = " -- To suppress these warning messages change suppressErrorsInPadText to true in your settings.json\n" ;
2015-04-11 23:01:46 +02:00
var _ = require ( "underscore" ) ;
2012-02-26 13:07:51 +01:00
/* Root path of the installation */
exports . root = path . normalize ( path . join ( npm . dir , ".." ) ) ;
2011-05-14 18:22:25 +02:00
2012-11-02 12:30:57 +01:00
/ * *
* The app title , visible e . g . in the browser window
* /
2013-08-23 00:32:25 +02:00
exports . title = "Etherpad" ;
2012-11-02 12:30:57 +01:00
2012-11-04 11:26:17 +01:00
/ * *
* The app favicon fully specified url , visible e . g . in the browser window
* /
exports . favicon = "favicon.ico" ;
2012-11-30 23:50:54 +01:00
exports . faviconPad = "../" + exports . favicon ;
exports . faviconTimeslider = "../../" + exports . favicon ;
2012-11-04 11:26:17 +01:00
2011-07-30 17:39:53 +02:00
/ * *
* The IP ep - lite should listen to
* /
exports . ip = "0.0.0.0" ;
2013-11-26 09:20:59 +01:00
2011-05-30 16:53:11 +02:00
/ * *
* The Port ep - lite should listen to
* /
2012-10-25 19:21:34 +02:00
exports . port = process . env . PORT || 9001 ;
2012-11-22 10:12:58 +01:00
2015-04-06 01:13:38 +02:00
/ * *
* Should we suppress Error messages from being in Pad Contents
* /
exports . suppressErrorsInPadText = false ;
2012-11-22 10:12:58 +01:00
/ * *
* The SSL signed server key and the Certificate Authority ' s own certificate
* default case : ep - lite does * not * use SSL . A signed server key is not required in this case .
* /
exports . ssl = false ;
2012-12-02 18:28:28 +01:00
/ * *
2012-12-02 18:44:39 +01:00
* socket . io transport methods
2012-12-02 18:28:28 +01:00
* * /
exports . socketTransportProtocols = [ 'xhr-polling' , 'jsonp-polling' , 'htmlfile' ] ;
2011-05-30 16:53:11 +02:00
/ *
* The Type of the database
* /
2011-08-17 20:26:56 +02:00
exports . dbType = "dirty" ;
2011-05-30 16:53:11 +02:00
/ * *
* This setting is passed with dbType to ueberDB to set up the database
* /
2012-04-20 22:39:16 +02:00
exports . dbSettings = { "filename" : path . join ( exports . root , "dirty.db" ) } ;
2012-07-08 18:59:46 +02:00
2011-05-30 16:53:11 +02:00
/ * *
* The default Text of a new pad
* /
2015-08-04 04:53:54 +02:00
exports . defaultPadText = "Welcome to Etherpad!\n\nThis pad text is synchronized as you type, so that everyone viewing this page sees the same text. This allows you to collaborate seamlessly on documents!\n\nEtherpad on Github: https:\/\/github.com\/ether\/etherpad-lite\n" ;
2011-11-21 07:45:37 +01:00
2015-04-11 22:22:00 +02:00
/ * *
* The default Pad Settings for a user ( Can be overridden by changing the setting
* /
exports . padOptions = {
"noColors" : false ,
"showControls" : true ,
"showChat" : true ,
"showLineNumbers" : true ,
"useMonospaceFont" : false ,
"userName" : false ,
"userColor" : false ,
"rtl" : false ,
"alwaysShowChat" : false ,
"chatAndUsers" : false ,
"lang" : "en-gb"
2016-01-21 13:38:41 +01:00
} ,
/ * *
* Whether certain shortcut keys are enabled for a user in the pad
* /
exports . padShortcutEnabled = {
"altF9" : true ,
"altC" : true ,
"delete" : true ,
"cmdShift2" : true ,
"return" : true ,
"cmdS" : true ,
"tab" : true ,
"cmdZ" : true ,
"cmdY" : true ,
"cmdB" : true ,
"cmdI" : true ,
"cmdU" : true ,
"cmd5" : true ,
"cmdShiftL" : true ,
"cmdShiftN" : true ,
"cmdShiftC" : true ,
"cmdH" : true ,
"ctrlHome" : true ,
"pageUp" : true ,
"pageDown" : true ,
} ,
2015-04-11 22:22:00 +02:00
2013-03-09 23:57:42 +01:00
/ * *
* The toolbar buttons and order .
* /
exports . toolbar = {
left : [
[ "bold" , "italic" , "underline" , "strikethrough" ] ,
[ "orderedlist" , "unorderedlist" , "indent" , "outdent" ] ,
[ "undo" , "redo" ] ,
[ "clearauthorship" ]
] ,
right : [
[ "importexport" , "timeslider" , "savedrevision" ] ,
[ "settings" , "embed" ] ,
[ "showusers" ]
2014-03-30 13:02:41 +02:00
] ,
timeslider : [
2015-04-05 16:18:36 +02:00
[ "timeslider_export" , "timeslider_settings" , "timeslider_returnToPad" ]
2013-03-09 23:57:42 +01:00
]
}
2011-11-21 07:45:37 +01:00
/ * *
* A flag that requires any user to have a valid session ( via the api ) before accessing a pad
* /
exports . requireSession = false ;
2011-11-21 18:44:33 +01:00
/ * *
* A flag that prevents users from creating new pads
* /
exports . editOnly = false ;
2013-05-29 16:53:14 +02:00
/ * *
2013-05-29 16:54:39 +02:00
* A flag that bypasses password prompts for users with valid sessions
2013-05-29 16:53:14 +02:00
* /
exports . sessionNoPassword = false ;
2012-02-07 08:04:02 +01:00
/ * *
* Max age that responses will have ( affects caching layer ) .
* /
exports . maxAge = 1000 * 60 * 60 * 6 ; // 6 hours
2011-05-30 16:53:11 +02:00
/ * *
* A flag that shows if minification is enabled or not
* /
2011-05-28 19:09:17 +02:00
exports . minify = true ;
2011-05-14 18:22:25 +02:00
2011-07-19 20:48:11 +02:00
/ * *
* The path of the abiword executable
* /
exports . abiword = null ;
2015-10-20 20:46:08 +02:00
/ * *
* The path of the libreoffice executable
* /
exports . soffice = null ;
2015-05-18 17:24:41 +02:00
/ * *
* The path of the tidy executable
* /
exports . tidyHtml = null ;
2014-07-22 16:46:31 +02:00
/ * *
* Should we support none natively supported file types on import ?
* /
exports . allowUnknownFileEnds = true ;
2011-08-17 18:45:47 +02:00
/ * *
* The log level of log4js
* /
exports . loglevel = "INFO" ;
2013-10-19 21:37:11 +02:00
/ * *
* Disable IP logging
* /
exports . disableIPlogging = false ;
2017-04-04 16:09:24 +02:00
/ * *
* Number of seconds to automatically reconnect pad
* /
exports . automaticReconnectionTimeout = 0 ;
2015-05-18 17:24:41 +02:00
/ * *
2015-02-17 00:02:19 +01:00
* Disable Load Testing
* /
exports . loadTest = false ;
2015-10-13 23:39:23 +02:00
/ * *
* Enable indentation on new lines
* /
exports . indentationOnNewLine = true ;
2013-01-13 12:20:49 +01:00
/ *
* log4js appender configuration
* /
exports . logconfig = { appenders : [ { type : "console" } ] } ;
2013-02-13 22:51:09 +01:00
/ *
* Session Key , do not sure this .
* /
exports . sessionKey = false ;
2013-04-24 12:19:41 +02:00
/ *
* Trust Proxy , whether or not trust the x - forwarded - for header .
* /
exports . trustProxy = false ;
2012-04-19 14:25:12 +02:00
/ * T h i s s e t t i n g i s u s e d i f y o u n e e d a u t h e n t i c a t i o n a n d / o r
* authorization . Note : / a d m i n a l w a y s r e q u i r e s a u t h e n t i c a t i o n , a n d
* either authorization by a module , or a user with is _admin set * /
exports . requireAuthentication = false ;
exports . requireAuthorization = false ;
exports . users = { } ;
2012-04-02 18:45:37 +02:00
2016-05-22 17:28:51 +02:00
/ *
* Show settings in admin page , by default it is true
* /
exports . showSettingsInAdminPage = true ;
2011-12-18 06:18:35 +01:00
//checks if abiword is avaiable
exports . abiwordAvailable = function ( )
{
if ( exports . abiword != null )
{
return os . type ( ) . indexOf ( "Windows" ) != - 1 ? "withoutPDF" : "yes" ;
}
else
{
return "no" ;
}
2013-03-24 01:18:44 +01:00
} ;
2011-12-18 06:18:35 +01:00
2015-12-18 04:54:04 +01:00
exports . sofficeAvailable = function ( ) {
if ( exports . soffice != null ) {
return os . type ( ) . indexOf ( "Windows" ) != - 1 ? "withoutPDF" : "yes" ;
} else {
return "no" ;
}
} ;
exports . exportAvailable = function ( ) {
var abiword = exports . abiwordAvailable ( ) ;
var soffice = exports . sofficeAvailable ( ) ;
if ( abiword == "no" && soffice == "no" ) {
return "no" ;
} else if ( ( abiword == "withoutPDF" && soffice == "no" ) || ( abiword == "no" && soffice == "withoutPDF" ) ) {
return "withoutPDF" ;
} else {
return "yes" ;
}
} ;
2015-02-11 18:59:05 +01:00
// Provide git version if available
exports . getGitCommit = function ( ) {
var version = "" ;
try
{
var rootPath = path . resolve ( npm . dir , '..' ) ;
2015-10-07 15:43:29 +02:00
if ( fs . lstatSync ( rootPath + '/.git' ) . isFile ( ) ) {
rootPath = fs . readFileSync ( rootPath + '/.git' , "utf8" ) ;
rootPath = rootPath . split ( ' ' ) . pop ( ) . trim ( ) ;
} else {
rootPath += '/.git' ;
}
var ref = fs . readFileSync ( rootPath + "/HEAD" , "utf-8" ) ;
var refPath = rootPath + "/" + ref . substring ( 5 , ref . indexOf ( "\n" ) ) ;
2015-02-11 18:59:05 +01:00
version = fs . readFileSync ( refPath , "utf-8" ) ;
version = version . substring ( 0 , 7 ) ;
}
catch ( e )
{
console . warn ( "Can't get git version for server header\n" + e . message )
}
return version ;
}
2015-04-11 00:13:04 +02:00
// Return etherpad version from package.json
exports . getEpVersion = function ( ) {
return require ( 'ep_etherpad-lite/package.json' ) . version ;
}
2012-11-06 17:35:05 +01:00
exports . reloadSettings = function reloadSettings ( ) {
// Discover where the settings file lives
var settingsFilename = argv . settings || "settings.json" ;
2014-06-22 20:56:21 +02:00
2015-12-02 12:54:37 +01:00
// Discover if a credential file exists
var credentialsFilename = argv . credentials || "credentials.json" ;
2014-07-24 12:06:52 +02:00
if ( path . resolve ( settingsFilename ) === settingsFilename ) {
2014-06-22 20:56:21 +02:00
settingsFilename = path . resolve ( settingsFilename ) ;
2014-07-24 12:06:52 +02:00
} else {
2014-06-22 20:56:21 +02:00
settingsFilename = path . resolve ( path . join ( exports . root , settingsFilename ) ) ;
2014-07-24 12:06:52 +02:00
}
2015-05-18 17:24:41 +02:00
2015-12-02 12:54:37 +01:00
if ( path . resolve ( credentialsFilename ) === credentialsFilename ) {
credentialsFilename = path . resolve ( credentialsFilename ) ;
}
var settingsStr , credentialsStr ;
2012-11-06 17:35:05 +01:00
try {
//read the settings sync
settingsStr = fs . readFileSync ( settingsFilename ) . toString ( ) ;
} catch ( e ) {
console . warn ( 'No settings file found. Continuing using defaults!' ) ;
2012-07-08 18:59:46 +02:00
}
2011-05-14 18:22:25 +02:00
2015-12-02 12:54:37 +01:00
try {
//read the credentials sync
credentialsStr = fs . readFileSync ( credentialsFilename ) . toString ( ) ;
} catch ( e ) {
// Doesn't matter if no credentials file found..
}
2012-11-06 17:35:05 +01:00
// try to parse the settings
var settings ;
2015-12-02 12:54:37 +01:00
var credentials ;
2012-11-06 17:35:05 +01:00
try {
if ( settingsStr ) {
2013-11-26 09:20:59 +01:00
settingsStr = jsonminify ( settingsStr ) . replace ( ",]" , "]" ) . replace ( ",}" , "}" ) ;
settings = JSON . parse ( settingsStr ) ;
2012-11-06 17:35:05 +01:00
}
} catch ( e ) {
console . error ( 'There was an error processing your settings.json file: ' + e . message ) ;
process . exit ( 1 ) ;
2011-05-14 18:22:25 +02:00
}
2015-12-02 12:54:37 +01:00
if ( credentialsStr ) {
credentialsStr = jsonminify ( credentialsStr ) . replace ( ",]" , "]" ) . replace ( ",}" , "}" ) ;
credentials = JSON . parse ( credentialsStr ) ;
}
2012-11-06 17:35:05 +01:00
//loop trough the settings
for ( var i in settings )
2011-05-14 18:22:25 +02:00
{
2012-11-06 17:35:05 +01:00
//test if the setting start with a low character
if ( i . charAt ( 0 ) . search ( "[a-z]" ) !== 0 )
{
console . warn ( "Settings should start with a low character: '" + i + "'" ) ;
}
//we know this setting, so we overwrite it
//or it's a settings hash, specific to a plugin
if ( exports [ i ] !== undefined || i . indexOf ( 'ep_' ) == 0 )
{
2015-04-11 23:01:46 +02:00
if ( _ . isObject ( settings [ i ] ) && ! _ . isArray ( settings [ i ] ) ) {
exports [ i ] = _ . defaults ( settings [ i ] , exports [ i ] ) ;
} else {
exports [ i ] = settings [ i ] ;
}
2012-11-06 17:35:05 +01:00
}
//this setting is unkown, output a warning and throw it away
else
{
console . warn ( "Unknown Setting: '" + i + "'. This setting doesn't exist or it was removed" ) ;
}
2011-05-14 18:22:25 +02:00
}
2013-11-26 09:20:59 +01:00
2015-12-02 12:54:37 +01:00
//loop trough the settings
for ( var i in credentials )
{
//test if the setting start with a low character
if ( i . charAt ( 0 ) . search ( "[a-z]" ) !== 0 )
{
console . warn ( "Settings should start with a low character: '" + i + "'" ) ;
}
//we know this setting, so we overwrite it
//or it's a settings hash, specific to a plugin
if ( exports [ i ] !== undefined || i . indexOf ( 'ep_' ) == 0 )
{
if ( _ . isObject ( credentials [ i ] ) && ! _ . isArray ( credentials [ i ] ) ) {
exports [ i ] = _ . defaults ( credentials [ i ] , exports [ i ] ) ;
} else {
exports [ i ] = credentials [ i ] ;
}
}
//this setting is unkown, output a warning and throw it away
else
{
console . warn ( "Unknown Setting: '" + i + "'. This setting doesn't exist or it was removed" ) ;
}
}
2013-01-13 12:20:49 +01:00
log4js . configure ( exports . logconfig ) ; //Configure the logging appenders
log4js . setGlobalLogLevel ( exports . loglevel ) ; //set loglevel
2014-11-04 19:32:26 +01:00
process . env [ 'DEBUG' ] = 'socket.io:' + exports . loglevel ; // Used by SocketIO for Debug
2013-01-13 12:20:49 +01:00
log4js . replaceConsole ( ) ;
2012-11-06 17:35:05 +01:00
2015-01-04 15:47:08 +01:00
if ( exports . abiword ) {
// Check abiword actually exists
if ( exports . abiword != null )
{
fs . exists ( exports . abiword , function ( exists ) {
if ( ! exists ) {
2015-04-06 01:13:38 +02:00
var abiwordError = "Abiword does not exist at this path, check your settings file" ;
if ( ! exports . suppressErrorsInPadText ) {
exports . defaultPadText = exports . defaultPadText + "\nError: " + abiwordError + suppressDisableMsg ;
}
console . error ( abiwordError ) ;
2015-01-04 15:47:08 +01:00
exports . abiword = null ;
}
} ) ;
}
}
2015-12-18 04:54:04 +01:00
if ( exports . soffice ) {
fs . exists ( exports . soffice , function ( exists ) {
if ( ! exists ) {
var sofficeError = "SOffice does not exist at this path, check your settings file" ;
if ( ! exports . suppressErrorsInPadText ) {
exports . defaultPadText = exports . defaultPadText + "\nError: " + sofficeError + suppressDisableMsg ;
}
console . error ( sofficeError ) ;
exports . soffice = null ;
}
} ) ;
}
2015-04-11 18:45:14 +02:00
if ( ! exports . sessionKey ) {
try {
exports . sessionKey = fs . readFileSync ( "./SESSIONKEY.txt" , "utf8" ) ;
} catch ( e ) {
exports . sessionKey = randomString ( 32 ) ;
fs . writeFileSync ( "./SESSIONKEY.txt" , exports . sessionKey , "utf8" ) ;
2015-04-06 01:13:38 +02:00
}
2015-04-11 18:45:14 +02:00
} else {
console . warn ( "Declaring the sessionKey in the settings.json is deprecated. This value is auto-generated now. Please remove the setting from the file." ) ;
2013-02-13 22:51:09 +01:00
}
2012-11-06 17:35:05 +01:00
if ( exports . dbType === "dirty" ) {
2015-04-06 01:13:38 +02:00
var dirtyWarning = "DirtyDB is used. This is fine for testing but not recommended for production." ;
if ( ! exports . suppressErrorsInPadText ) {
exports . defaultPadText = exports . defaultPadText + "\nWarning: " + dirtyWarning + suppressDisableMsg ;
}
console . warn ( dirtyWarning ) ;
2011-05-14 18:22:25 +02:00
}
2013-03-24 01:18:44 +01:00
} ;
2012-04-20 17:03:37 +02:00
2012-11-06 17:35:05 +01:00
// initially load settings
exports . reloadSettings ( ) ;
2015-02-11 18:59:05 +01:00