2014-10-17 02:21:06 +00:00
/* jslint node: true */
'use strict' ;
2018-06-23 03:26:46 +00:00
// ENiGMA½
const Errors = require ( './enig_error.js' ) . Errors ;
2014-10-17 02:21:06 +00:00
2018-06-23 03:26:46 +00:00
// deps
const paths = require ( 'path' ) ;
const async = require ( 'async' ) ;
const _ = require ( 'lodash' ) ;
const assert = require ( 'assert' ) ;
2015-04-17 04:29:53 +00:00
2018-11-08 03:24:05 +00:00
exports . init = init ;
exports . getDefaultPath = getDefaultPath ;
exports . getDefaultConfig = getDefaultConfig ;
2015-04-19 08:13:13 +00:00
2018-06-21 01:57:06 +00:00
let currentConfiguration = { } ;
2016-02-03 04:35:59 +00:00
function hasMessageConferenceAndArea ( config ) {
2018-06-22 05:15:04 +00:00
assert ( _ . isObject ( config . messageConferences ) ) ; // we create one ourself!
2016-09-20 03:28:21 +00:00
2018-06-22 05:15:04 +00:00
const nonInternalConfs = Object . keys ( config . messageConferences ) . filter ( confTag => {
return 'system_internal' !== confTag ;
} ) ;
2016-09-20 03:28:21 +00:00
2018-06-22 05:15:04 +00:00
if ( 0 === nonInternalConfs . length ) {
return false ;
}
2016-09-20 03:28:21 +00:00
2018-06-22 05:15:04 +00:00
// :TODO: there is likely a better/cleaner way of doing this
2016-09-20 03:28:21 +00:00
2018-06-22 05:15:04 +00:00
let result = false ;
_ . forEach ( nonInternalConfs , confTag => {
if ( _ . has ( config . messageConferences [ confTag ] , 'areas' ) &&
2018-06-23 03:26:46 +00:00
Object . keys ( config . messageConferences [ confTag ] . areas ) > 0 )
2018-06-22 05:15:04 +00:00
{
result = true ;
return false ; // stop iteration
}
} ) ;
2016-09-20 03:28:21 +00:00
2018-06-22 05:15:04 +00:00
return result ;
2016-02-03 04:35:59 +00:00
}
2018-11-17 20:14:51 +00:00
const ArrayReplaceKeyPaths = [
'loginServers.ssh.algorithms.kex' ,
'loginServers.ssh.algorithms.cipher' ,
'loginServers.ssh.algorithms.hmac' ,
'loginServers.ssh.algorithms.compress' ,
] ;
const ArrayReplaceKeys = [
'args' ,
'sendArgs' , 'recvArgs' , 'recvArgsNonBatch' ,
] ;
2018-06-14 03:02:00 +00:00
function mergeValidateAndFinalize ( config , cb ) {
2018-11-17 20:14:51 +00:00
const defaultConfig = getDefaultConfig ( ) ;
const arrayReplaceKeyPathsMutable = _ . clone ( ArrayReplaceKeyPaths ) ;
const shouldReplaceArray = ( arr , key ) => {
if ( ArrayReplaceKeys . includes ( key ) ) {
return true ;
}
for ( let i = 0 ; i < arrayReplaceKeyPathsMutable . length ; ++ i ) {
const o = _ . get ( defaultConfig , arrayReplaceKeyPathsMutable [ i ] ) ;
if ( _ . isEqual ( o , arr ) ) {
arrayReplaceKeyPathsMutable . splice ( i , 1 ) ;
return true ;
}
}
return false ;
} ;
2018-06-22 05:15:04 +00:00
async . waterfall (
[
function mergeWithDefaultConfig ( callback ) {
const mergedConfig = _ . mergeWith (
2018-11-17 20:14:51 +00:00
defaultConfig ,
config ,
( defConfig , userConfig , key ) => {
if ( Array . isArray ( defConfig ) && Array . isArray ( userConfig ) ) {
//
// Arrays are special: Some we merge, while others
// we simply replace.
//
if ( shouldReplaceArray ( defConfig , key ) ) {
return userConfig ;
} else {
return _ . uniq ( defConfig . concat ( userConfig ) ) ;
}
2018-06-22 05:15:04 +00:00
}
}
) ;
return callback ( null , mergedConfig ) ;
} ,
function validate ( mergedConfig , callback ) {
//
2018-06-23 03:26:46 +00:00
// Various sections must now exist in config
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// :TODO: Logic is broken here:
2018-06-22 05:15:04 +00:00
if ( hasMessageConferenceAndArea ( mergedConfig ) ) {
return callback ( Errors . MissingConfig ( 'Please create at least one message conference and area!' ) ) ;
}
return callback ( null , mergedConfig ) ;
} ,
function setIt ( mergedConfig , callback ) {
2018-06-23 03:26:46 +00:00
// :TODO: .config property is to be deprecated once conversions are done
2018-06-22 05:15:04 +00:00
exports . config = currentConfiguration = mergedConfig ;
exports . get = ( ) => currentConfiguration ;
return callback ( null ) ;
}
] ,
err => {
if ( cb ) {
return cb ( err ) ;
}
}
) ;
2018-06-14 03:02:00 +00:00
}
function init ( configPath , options , cb ) {
2018-06-22 05:15:04 +00:00
if ( ! cb && _ . isFunction ( options ) ) {
cb = options ;
options = { } ;
}
const changed = ( { fileName , fileRoot } ) => {
const reCachedPath = paths . join ( fileRoot , fileName ) ;
ConfigCache . getConfig ( reCachedPath , ( err , config ) => {
if ( ! err ) {
2018-06-24 03:02:16 +00:00
mergeValidateAndFinalize ( config , err => {
if ( ! err ) {
const Events = require ( './events.js' ) ;
Events . emit ( Events . getSystemEvents ( ) . ConfigChanged ) ;
}
} ) ;
2018-07-03 01:32:27 +00:00
} else {
console . stdout ( ` Configuration ${ reCachedPath } is invalid: ${ err . message } ` ) ; // eslint-disable-line no-console
2018-06-22 05:15:04 +00:00
}
} ) ;
} ;
const ConfigCache = require ( './config_cache.js' ) ;
const getConfigOptions = {
2018-06-23 03:26:46 +00:00
filePath : configPath ,
noWatch : options . noWatch ,
2018-06-22 05:15:04 +00:00
} ;
if ( ! options . noWatch ) {
getConfigOptions . callback = changed ;
}
ConfigCache . getConfigWithOptions ( getConfigOptions , ( err , config ) => {
if ( err ) {
return cb ( err ) ;
}
return mergeValidateAndFinalize ( config , cb ) ;
} ) ;
2015-04-19 08:13:13 +00:00
}
2014-10-17 02:21:06 +00:00
2015-04-19 08:13:13 +00:00
function getDefaultPath ( ) {
2018-06-23 03:26:46 +00:00
// e.g. /enigma-bbs-install-path/config/
2018-06-22 05:15:04 +00:00
return './config/' ;
2015-04-19 08:13:13 +00:00
}
2014-10-17 02:21:06 +00:00
2015-04-19 08:13:13 +00:00
function getDefaultConfig ( ) {
2018-06-22 05:15:04 +00:00
return {
general : {
2018-06-23 03:26:46 +00:00
boardName : 'Another Fine ENiGMA½ BBS' ,
2015-10-22 04:51:35 +00:00
2018-11-08 03:24:05 +00:00
// :TODO: closedSystem and loginAttemps prob belong under users{}?
2018-06-23 03:26:46 +00:00
closedSystem : false , // is the system closed to new users?
2015-12-11 20:58:58 +00:00
2018-11-14 04:32:22 +00:00
menuFile : 'menu.hjson' , // 'oputil.js config new' will set this appropriately in config.hjson; may be full path
promptFile : 'prompt.hjson' , // 'oputil.js config new' will set this appropriately in config.hjson; may be full path
2018-06-22 05:15:04 +00:00
} ,
2014-10-17 02:21:06 +00:00
2018-06-22 05:15:04 +00:00
users : {
2018-06-23 03:26:46 +00:00
usernameMin : 2 ,
usernameMax : 16 , // Note that FidoNet wants 36 max
2018-07-21 20:32:06 +00:00
usernamePattern : '^[A-Za-z0-9~!@#$%^&*()\\-\\_+ .]+$' ,
2015-12-24 18:51:49 +00:00
2018-06-23 03:26:46 +00:00
passwordMin : 6 ,
passwordMax : 128 ,
2018-11-10 17:17:24 +00:00
//
// The bad password list is a text file containing a password per line.
// Entries in this list are not allowed to be used on the system as they
// are known to be too common.
//
// A great resource can be found at https://github.com/danielmiessler/SecLists
//
// Current list source: https://raw.githubusercontent.com/danielmiessler/SecLists/master/Passwords/probable-v2-top12000.txt
//
badPassFile : paths . join ( _ _dirname , '../misc/bad_passwords.txt' ) ,
2015-12-24 18:51:49 +00:00
2018-06-23 03:26:46 +00:00
realNameMax : 32 ,
locationMax : 32 ,
affilsMax : 32 ,
emailMax : 255 ,
webMax : 255 ,
2015-12-24 18:51:49 +00:00
2018-06-23 03:26:46 +00:00
requireActivation : false , // require SysOp activation? false = auto-activate
2015-08-21 04:29:16 +00:00
2018-06-23 03:26:46 +00:00
groups : [ 'users' , 'sysops' ] , // built in groups
defaultGroups : [ 'users' ] , // default groups new users belong to
2015-10-22 17:04:50 +00:00
2018-06-23 03:26:46 +00:00
newUserNames : [ 'new' , 'apply' ] , // Names reserved for applying
2015-10-28 03:12:55 +00:00
2018-06-23 03:26:46 +00:00
badUserNames : [
2018-06-22 05:15:04 +00:00
'sysop' , 'admin' , 'administrator' , 'root' , 'all' ,
'areamgr' , 'filemgr' , 'filefix' , 'areafix' , 'allfix'
] ,
2018-11-11 08:00:42 +00:00
preAuthIdleLogoutSeconds : 60 * 3 , // 3m
idleLogoutSeconds : 60 * 6 , // 6m
2018-11-23 06:07:37 +00:00
failedLogin : {
disconnect : 3 , // 0=disabled
lockAccount : 9 , // 0=disabled; Mark user status as "locked" if >= N
autoUnlockMinutes : 60 * 6 , // 0=disabled; Auto unlock after N minutes.
} ,
unlockAtEmailPwReset : true , // if true, password reset via email will unlock locked accounts
2018-06-22 05:15:04 +00:00
} ,
2015-04-17 04:29:53 +00:00
2018-11-08 01:33:07 +00:00
theme : {
default : 'luciano_blocktronics' ,
preLogin : 'luciano_blocktronics' ,
2018-11-08 03:24:05 +00:00
passwordChar : '*' ,
2018-06-23 03:26:46 +00:00
dateFormat : {
short : 'MM/DD/YYYY' ,
long : 'ddd, MMMM Do, YYYY' ,
2018-06-22 05:15:04 +00:00
} ,
timeFormat : {
2018-06-23 03:26:46 +00:00
short : 'h:mm a' ,
2018-06-22 05:15:04 +00:00
} ,
dateTimeFormat : {
2018-06-23 03:26:46 +00:00
short : 'MM/DD/YYYY h:mm a' ,
long : 'ddd, MMMM Do, YYYY, h:mm a' ,
2018-06-22 05:15:04 +00:00
}
} ,
menus : {
2018-06-23 03:26:46 +00:00
cls : true , // Clear screen before each menu by default?
2018-06-22 05:15:04 +00:00
} ,
2018-11-08 01:33:07 +00:00
paths : {
2018-06-23 03:26:46 +00:00
config : paths . join ( _ _dirname , './../config/' ) ,
mods : paths . join ( _ _dirname , './../mods/' ) ,
loginServers : paths . join ( _ _dirname , './servers/login/' ) ,
contentServers : paths . join ( _ _dirname , './servers/content/' ) ,
scannerTossers : paths . join ( _ _dirname , './scanner_tossers/' ) ,
mailers : paths . join ( _ _dirname , './mailers/' ) ,
art : paths . join ( _ _dirname , './../art/general/' ) ,
themes : paths . join ( _ _dirname , './../art/themes/' ) ,
logs : paths . join ( _ _dirname , './../logs/' ) , // :TODO: set up based on system, e.g. /var/logs/enigmabbs or such
db : paths . join ( _ _dirname , './../db/' ) ,
modsDb : paths . join ( _ _dirname , './../db/mods/' ) ,
2018-12-09 06:41:42 +00:00
dropFiles : paths . join ( _ _dirname , './../drop/' ) , // + "/node<x>/
2018-06-23 03:26:46 +00:00
misc : paths . join ( _ _dirname , './../misc/' ) ,
2018-06-22 05:15:04 +00:00
} ,
loginServers : {
telnet : {
2018-06-23 03:26:46 +00:00
port : 8888 ,
enabled : true ,
firstMenu : 'telnetConnected' ,
2018-06-22 05:15:04 +00:00
} ,
ssh : {
2018-06-23 03:26:46 +00:00
port : 8889 ,
enabled : false , // default to false as PK/pass in config.hjson are required
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// Private key in PEM format
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// Generating your PK:
// > openssl genrsa -des3 -out ./config/ssh_private_key.pem 2048
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// Then, set servers.ssh.privateKeyPass to the password you use above
// in your config.hjson
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
privateKeyPem : paths . join ( _ _dirname , './../config/ssh_private_key.pem' ) ,
firstMenu : 'sshConnected' ,
firstMenuNewUser : 'sshConnectedNewUser' ,
2018-11-05 02:29:51 +00:00
//
// SSH details that can affect security. Stronger ciphers are better for example,
// but terminals such as SyncTERM require KEX diffie-hellman-group14-sha1,
// cipher 3des-cbc, etc.
//
// See https://github.com/mscdex/ssh2-streams for the full list of supported
// algorithms.
//
algorithms : {
kex : [
'ecdh-sha2-nistp256' ,
'ecdh-sha2-nistp384' ,
'ecdh-sha2-nistp521' ,
'diffie-hellman-group-exchange-sha256' ,
'diffie-hellman-group14-sha1' ,
'diffie-hellman-group-exchange-sha1' ,
'diffie-hellman-group1-sha1' ,
] ,
cipher : [
'aes128-ctr' ,
'aes192-ctr' ,
'aes256-ctr' ,
'aes128-gcm' ,
'aes128-gcm@openssh.com' ,
'aes256-gcm' ,
'aes256-gcm@openssh.com' ,
'aes256-cbc' ,
'aes192-cbc' ,
'aes128-cbc' ,
'blowfish-cbc' ,
'3des-cbc' ,
'arcfour256' ,
'arcfour128' ,
'cast128-cbc' ,
'arcfour' ,
] ,
hmac : [
'hmac-sha2-256' ,
'hmac-sha2-512' ,
'hmac-sha1' ,
'hmac-md5' ,
'hmac-sha2-256-96' ,
'hmac-sha2-512-96' ,
'hmac-ripemd160' ,
'hmac-sha1-96' ,
'hmac-md5-96' ,
] ,
// note that we disable compression by default due to issues with many clients. YMMV.
compress : [ 'none' ]
} ,
2018-06-22 05:15:04 +00:00
} ,
webSocket : {
ws : {
2018-06-23 03:26:46 +00:00
// non-secure ws://
enabled : false ,
port : 8810 ,
2018-06-22 05:15:04 +00:00
} ,
wss : {
2018-06-23 03:26:46 +00:00
// secure ws://
// must provide valid certPem and keyPem
enabled : false ,
port : 8811 ,
certPem : paths . join ( _ _dirname , './../config/https_cert.pem' ) ,
keyPem : paths . join ( _ _dirname , './../config/https_cert_key.pem' ) ,
2018-06-22 05:15:04 +00:00
} ,
} ,
} ,
contentServers : {
web : {
domain : 'another-fine-enigma-bbs.org' ,
staticRoot : paths . join ( _ _dirname , './../www' ) ,
resetPassword : {
//
2018-06-23 03:26:46 +00:00
// The following templates have these variables available to them:
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// * %BOARDNAME% : Name of BBS
// * %USERNAME% : Username of whom to reset password
// * %TOKEN% : Reset token
// * %RESET_URL% : In case of email, the link to follow for reset. In case of landing page,
// URL to POST submit reset form.
2018-06-22 05:15:04 +00:00
2018-06-23 03:26:46 +00:00
// templates for pw reset *email*
resetPassEmailText : paths . join ( _ _dirname , '../misc/reset_password_email.template.txt' ) , // plain text version
resetPassEmailHtml : paths . join ( _ _dirname , '../misc/reset_password_email.template.html' ) , // HTML version
2018-06-22 05:15:04 +00:00
2018-06-23 03:26:46 +00:00
// tempalte for pw reset *landing page*
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
resetPageTemplate : paths . join ( _ _dirname , './../www/reset_password.template.html' ) ,
2018-06-22 05:15:04 +00:00
} ,
http : {
enabled : false ,
2018-06-23 03:26:46 +00:00
port : 8080 ,
2018-06-22 05:15:04 +00:00
} ,
https : {
2018-06-23 03:26:46 +00:00
enabled : false ,
port : 8443 ,
certPem : paths . join ( _ _dirname , './../config/https_cert.pem' ) ,
keyPem : paths . join ( _ _dirname , './../config/https_cert_key.pem' ) ,
2018-06-22 05:15:04 +00:00
}
2018-11-11 06:59:26 +00:00
} ,
gopher : {
enabled : false ,
port : 8070 ,
publicHostname : 'another-fine-enigma-bbs.org' ,
2018-12-15 09:39:57 +00:00
publicPort : 8070 , // adjust if behind NAT/etc.
2018-11-11 06:59:26 +00:00
bannerFile : 'gopher_banner.asc' ,
//
// Set messageConferences{} to maps of confTag -> [ areaTag1, areaTag2, ... ]
// to export message confs/areas
//
2018-12-15 05:21:57 +00:00
} ,
nntp : {
// internal caching of groups, message lists, etc.
cache : {
maxItems : 200 ,
maxAge : 1000 * 30 , // 30s
} ,
//
// Set publicMessageConferences{} to a map of confTag -> [ areaTag1, areaTag2, ... ]
// in order to export *public* conf/areas that are available to anonymous
// NNTP users. Other conf/areas: Standard ACS rules apply.
//
publicMessageConferences : { } ,
nntp : {
enabled : false ,
port : 8119 ,
} ,
nntps : {
enabled : false ,
port : 8563 ,
certPem : paths . join ( _ _dirname , './../config/nntps_cert.pem' ) ,
keyPem : paths . join ( _ _dirname , './../config/nntps_key.pem' ) ,
}
2018-06-22 05:15:04 +00:00
}
} ,
infoExtractUtils : {
Exiftool2Desc : {
2018-06-23 03:26:46 +00:00
cmd : ` ${ _ _dirname } /../util/exiftool2desc.js ` , // ensure chmod +x
2018-06-22 05:15:04 +00:00
} ,
Exiftool : {
2018-06-23 03:26:46 +00:00
cmd : 'exiftool' ,
args : [
2018-06-22 05:15:04 +00:00
'-charset' , 'utf8' , '{filePath}' ,
2018-06-23 03:26:46 +00:00
// exclude the following:
2018-06-22 05:15:04 +00:00
'--directory' , '--filepermissions' , '--exiftoolversion' , '--filename' , '--filesize' ,
'--filemodifydate' , '--fileaccessdate' , '--fileinodechangedate' , '--createdate' , '--modifydate' ,
'--metadatadate' , '--xmptoolkit'
]
} ,
XDMS2Desc : {
2018-06-23 03:26:46 +00:00
// http://manpages.ubuntu.com/manpages/trusty/man1/xdms.1.html
cmd : 'xdms' ,
args : [ 'd' , '{filePath}' ]
2018-06-22 05:15:04 +00:00
} ,
XDMS2LongDesc : {
2018-06-23 03:26:46 +00:00
// http://manpages.ubuntu.com/manpages/trusty/man1/xdms.1.html
cmd : 'xdms' ,
args : [ 'f' , '{filePath}' ]
2018-06-30 04:40:13 +00:00
} ,
2018-06-22 05:15:04 +00:00
} ,
fileTypes : {
//
2018-06-23 03:26:46 +00:00
// File types explicitly known to the system. Here we can configure
// information extraction, archive treatment, etc.
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// MIME types can be found in mime-db: https://github.com/jshttp/mime-db
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// Resources for signature/magic bytes:
// * http://www.garykessler.net/library/file_sigs.html
2018-06-22 05:15:04 +00:00
//
//
2018-06-23 03:26:46 +00:00
// :TODO: text/x-ansi -> SAUCE extraction for .ans uploads
// :TODO: textual : bool -- if text, we can view.
// :TODO: asText : { cmd, args[] } -> viewable text
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// Audio
2018-06-22 05:15:04 +00:00
//
'audio/mpeg' : {
2018-06-23 03:26:46 +00:00
desc : 'MP3 Audio' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
'application/pdf' : {
2018-06-23 03:26:46 +00:00
desc : 'Adobe PDF' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// Video
2018-06-22 05:15:04 +00:00
//
'video/mp4' : {
2018-06-23 03:26:46 +00:00
desc : 'MPEG Video' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
'video/x-matroska ' : {
2018-06-23 03:26:46 +00:00
desc : 'Matroska Video' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
'video/x-msvideo' : {
2018-06-23 03:26:46 +00:00
desc : 'Audio Video Interleave' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// Images
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
'image/jpeg' : {
desc : 'JPEG Image' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
'image/png' : {
desc : 'Portable Network Graphic Image' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
'image/gif' : {
2018-06-23 03:26:46 +00:00
desc : 'Graphics Interchange Format Image' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
'image/webp' : {
2018-06-23 03:26:46 +00:00
desc : 'WebP Image' ,
shortDescUtil : 'Exiftool2Desc' ,
longDescUtil : 'Exiftool' ,
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// Archives
2018-06-22 05:15:04 +00:00
//
'application/zip' : {
2018-06-23 03:26:46 +00:00
desc : 'ZIP Archive' ,
sig : '504b0304' ,
offset : 0 ,
archiveHandler : '7Zip' ,
2018-06-22 05:15:04 +00:00
} ,
/ *
2018-06-23 03:26:46 +00:00
'application/x-cbr' : {
desc : 'Comic Book Archive' ,
sig : '504b0304' ,
} ,
* /
2018-06-22 05:15:04 +00:00
'application/x-arj' : {
2018-06-23 03:26:46 +00:00
desc : 'ARJ Archive' ,
sig : '60ea' ,
offset : 0 ,
archiveHandler : 'Arj' ,
2018-06-22 05:15:04 +00:00
} ,
'application/x-rar-compressed' : {
2018-06-23 03:26:46 +00:00
desc : 'RAR Archive' ,
sig : '526172211a0700' ,
offset : 0 ,
archiveHandler : 'Rar' ,
2018-06-22 05:15:04 +00:00
} ,
'application/gzip' : {
2018-06-23 03:26:46 +00:00
desc : 'Gzip Archive' ,
sig : '1f8b' ,
offset : 0 ,
archiveHandler : 'TarGz' ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
// :TODO: application/x-bzip
2018-06-22 05:15:04 +00:00
'application/x-bzip2' : {
2018-06-23 03:26:46 +00:00
desc : 'BZip2 Archive' ,
sig : '425a68' ,
offset : 0 ,
archiveHandler : '7Zip' ,
2018-06-22 05:15:04 +00:00
} ,
'application/x-lzh-compressed' : {
2018-06-23 03:26:46 +00:00
desc : 'LHArc Archive' ,
sig : '2d6c68' ,
offset : 2 ,
archiveHandler : 'Lha' ,
2018-06-22 05:15:04 +00:00
} ,
'application/x-lzx' : {
2018-06-23 03:26:46 +00:00
desc : 'LZX Archive' ,
sig : '4c5a5800' ,
offset : 0 ,
archiveHandler : 'Lzx' ,
2018-06-22 05:15:04 +00:00
} ,
'application/x-7z-compressed' : {
2018-06-23 03:26:46 +00:00
desc : '7-Zip Archive' ,
sig : '377abcaf271c' ,
offset : 0 ,
archiveHandler : '7Zip' ,
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// Generics that need further mapping
2018-06-22 05:15:04 +00:00
//
'application/octet-stream' : [
{
2018-06-23 03:26:46 +00:00
desc : 'Amiga DISKMASHER' ,
sig : '444d5321' , // DMS!
ext : '.dms' ,
shortDescUtil : 'XDMS2Desc' ,
longDescUtil : 'XDMS2LongDesc' ,
2018-06-30 04:40:13 +00:00
} ,
{
desc : 'SIO2PC Atari Disk Image' ,
sig : '9602' , // 16bit sum of "NICKATARI"
ext : '.atr' ,
archiveHandler : 'Atr' ,
2018-06-22 05:15:04 +00:00
}
]
} ,
archives : {
archivers : {
'7Zip' : {
2018-06-23 03:26:46 +00:00
compress : {
cmd : '7za' ,
args : [ 'a' , '-tzip' , '{archivePath}' , '{fileList}' ] ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
decompress : {
cmd : '7za' ,
args : [ 'e' , '-o{extractPath}' , '{archivePath}' ] // :TODO: should be 'x'?
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
list : {
cmd : '7za' ,
args : [ 'l' , '{archivePath}' ] ,
entryMatch : '^[0-9]{4}-[0-9]{2}-[0-9]{2}\\s[0-9]{2}:[0-9]{2}:[0-9]{2}\\s[A-Za-z\\.]{5}\\s+([0-9]+)\\s+[0-9]+\\s+([^\\r\\n]+)$' ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
extract : {
cmd : '7za' ,
args : [ 'e' , '-o{extractPath}' , '{archivePath}' , '{fileList}' ] ,
2018-06-22 05:15:04 +00:00
} ,
} ,
Lha : {
//
2018-06-23 03:26:46 +00:00
// 'lha' command can be obtained from:
// * apt-get: lhasa
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// (compress not currently supported)
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
decompress : {
cmd : 'lha' ,
args : [ '-efw={extractPath}' , '{archivePath}' ] ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
list : {
cmd : 'lha' ,
args : [ '-l' , '{archivePath}' ] ,
entryMatch : '^[\\[a-z\\]]+(?:\\s+[0-9]+\\s+[0-9]+|\\s+)([0-9]+)\\s+[0-9]{2}\\.[0-9]\\%\\s+[A-Za-z]{3}\\s+[0-9]{1,2}\\s+[0-9]{4}\\s+([^\\r\\n]+)$' ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
extract : {
cmd : 'lha' ,
args : [ '-efw={extractPath}' , '{archivePath}' , '{fileList}' ]
2018-06-22 05:15:04 +00:00
}
} ,
Lzx : {
//
2018-06-23 03:26:46 +00:00
// 'unlzx' command can be obtained from:
// * Debian based: https://launchpad.net/~rzr/+archive/ubuntu/ppa/+build/2486127 (amd64/x86_64)
// * RedHat: https://fedora.pkgs.org/28/rpm-sphere/unlzx-1.1-4.1.x86_64.rpm.html
// * Source: http://xavprods.free.fr/lzx/
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
decompress : {
cmd : 'unlzx' ,
// unzlx doesn't have a output dir option, but we'll cwd to the temp output dir first
args : [ '-x' , '{archivePath}' ] ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
list : {
cmd : 'unlzx' ,
args : [ '-v' , '{archivePath}' ] ,
entryMatch : '^\\s+([0-9]+)\\s+[^\\s]+\\s+[0-9]{2}:[0-9]{2}:[0-9]{2}\\s+[0-9]{1,2}-[a-z]{3}-[0-9]{4}\\s+[a-z\\-]+\\s+\\"([^"]+)\\"$' ,
2018-06-22 05:15:04 +00:00
}
} ,
Arj : {
//
2018-06-23 03:26:46 +00:00
// 'arj' command can be obtained from:
// * apt-get: arj
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
decompress : {
cmd : 'arj' ,
args : [ 'x' , '{archivePath}' , '{extractPath}' ] ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
list : {
cmd : 'arj' ,
args : [ 'l' , '{archivePath}' ] ,
entryMatch : '^([^\\s]+)\\s+([0-9]+)\\s+[0-9]+\\s[0-9\\.]+\\s+[0-9]{2}\\-[0-9]{2}\\-[0-9]{2}\\s[0-9]{2}\\:[0-9]{2}\\:[0-9]{2}\\s+(?:[^\\r\\n]+)$' ,
entryGroupOrder : { // defaults to { byteSize : 1, fileName : 2 }
fileName : 1 ,
byteSize : 2 ,
2018-06-22 05:15:04 +00:00
}
} ,
2018-06-23 03:26:46 +00:00
extract : {
cmd : 'arj' ,
args : [ 'e' , '{archivePath}' , '{extractPath}' , '{fileList}' ] ,
2018-06-22 05:15:04 +00:00
}
} ,
Rar : {
2018-06-23 03:26:46 +00:00
decompress : {
cmd : 'unrar' ,
args : [ 'x' , '{archivePath}' , '{extractPath}' ] ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
list : {
cmd : 'unrar' ,
args : [ 'l' , '{archivePath}' ] ,
entryMatch : '^\\s+[\\.A-Z]+\\s+([\\d]+)\\s{2}[0-9]{2}\\-[0-9]{2}\\-[0-9]{2}\\s[0-9]{2}\\:[0-9]{2}\\s{2}([^\\r\\n]+)$' ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
extract : {
cmd : 'unrar' ,
args : [ 'e' , '{archivePath}' , '{extractPath}' , '{fileList}' ] ,
2018-06-22 05:15:04 +00:00
}
} ,
TarGz : {
2018-06-23 03:26:46 +00:00
decompress : {
cmd : 'tar' ,
args : [ '-xf' , '{archivePath}' , '-C' , '{extractPath}' , '--strip-components=1' ] ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
list : {
cmd : 'tar' ,
args : [ '-tvf' , '{archivePath}' ] ,
entryMatch : '^[drwx\\-]{10}\\s[A-Za-z0-9\\/]+\\s+([0-9]+)\\s[0-9]{4}\\-[0-9]{2}\\-[0-9]{2}\\s[0-9]{2}\\:[0-9]{2}\\s([^\\r\\n]+)$' ,
2018-06-22 05:15:04 +00:00
} ,
2018-06-23 03:26:46 +00:00
extract : {
cmd : 'tar' ,
args : [ '-xvf' , '{archivePath}' , '-C' , '{extractPath}' , '{fileList}' ] ,
2018-06-22 05:15:04 +00:00
}
2018-06-30 04:40:13 +00:00
} ,
Atr : {
decompress : {
cmd : 'atr' ,
args : [ '{archivePath}' , 'x' , '-a' , '-o' , '{extractPath}' ]
} ,
list : {
cmd : 'atr' ,
args : [ '{archivePath}' , 'ls' , '-la1' ] ,
entryMatch : '^[rwxs-]{5}\\s+([0-9]+)\\s\\([0-9\\s]+\\)\\s([^\\r\\n\\s]*)(?:[^\\r\\n]+)?$' ,
} ,
extract : {
cmd : 'atr' ,
// note: -l converts Atari 0x9b line feeds to 0x0a; not ideal if we're dealing with a binary of course.
2018-06-30 17:55:13 +00:00
args : [ '{archivePath}' , 'x' , '-a' , '-l' , '-o' , '{extractPath}' , '{fileList}' ]
2018-06-30 04:40:13 +00:00
}
2018-06-22 05:15:04 +00:00
}
} ,
} ,
fileTransferProtocols : {
//
2018-06-23 03:26:46 +00:00
// See http://www.synchro.net/docs/sexyz.txt for information on SEXYZ
2018-06-22 05:15:04 +00:00
//
zmodem8kSexyz : {
2018-06-23 03:26:46 +00:00
name : 'ZModem 8k (SEXYZ)' ,
type : 'external' ,
sort : 1 ,
external : {
// :TODO: Look into shipping sexyz binaries or at least hosting them somewhere for common systems
2018-11-05 04:00:54 +00:00
// Linux x86_64 binary: https://l33t.codes/outgoing/sexyz
2018-06-23 03:26:46 +00:00
sendCmd : 'sexyz' ,
sendArgs : [ '-telnet' , '-8' , 'sz' , '@{fileListPath}' ] ,
recvCmd : 'sexyz' ,
recvArgs : [ '-telnet' , '-8' , 'rz' , '{uploadDir}' ] ,
recvArgsNonBatch : [ '-telnet' , '-8' , 'rz' , '{fileName}' ] ,
2018-06-22 05:15:04 +00:00
}
} ,
xmodemSexyz : {
2018-06-23 03:26:46 +00:00
name : 'XModem (SEXYZ)' ,
type : 'external' ,
sort : 3 ,
external : {
sendCmd : 'sexyz' ,
sendArgs : [ '-telnet' , 'sX' , '@{fileListPath}' ] ,
recvCmd : 'sexyz' ,
recvArgsNonBatch : [ '-telnet' , 'rC' , '{fileName}' ]
2018-06-22 05:15:04 +00:00
}
} ,
ymodemSexyz : {
2018-06-23 03:26:46 +00:00
name : 'YModem (SEXYZ)' ,
type : 'external' ,
sort : 4 ,
external : {
sendCmd : 'sexyz' ,
sendArgs : [ '-telnet' , 'sY' , '@{fileListPath}' ] ,
recvCmd : 'sexyz' ,
recvArgs : [ '-telnet' , 'ry' , '{uploadDir}' ] ,
2018-06-22 05:15:04 +00:00
}
} ,
zmodem8kSz : {
2018-06-23 03:26:46 +00:00
name : 'ZModem 8k' ,
type : 'external' ,
sort : 2 ,
external : {
sendCmd : 'sz' , // Avail on Debian/Ubuntu based systems as the package "lrzsz"
sendArgs : [
// :TODO: try -q
2018-06-22 05:15:04 +00:00
'--zmodem' , '--try-8k' , '--binary' , '--restricted' , '{filePaths}'
] ,
2018-06-23 03:26:46 +00:00
recvCmd : 'rz' , // Avail on Debian/Ubuntu based systems as the package "lrzsz"
recvArgs : [
'--zmodem' , '--binary' , '--restricted' , '--keep-uppercase' , // dumps to CWD which is set to {uploadDir}
2018-06-22 05:15:04 +00:00
] ,
2018-06-23 03:26:46 +00:00
// :TODO: can we not just use --escape ?
escapeTelnet : true , // set to true to escape Telnet codes such as IAC
2018-06-22 05:15:04 +00:00
}
}
} ,
messageAreaDefaults : {
//
2018-06-23 03:26:46 +00:00
// The following can be override per-area as well
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
maxMessages : 1024 , // 0 = unlimited
maxAgeDays : 0 , // 0 = unlimited
2018-06-22 05:15:04 +00:00
} ,
messageConferences : {
system _internal : {
2018-06-23 03:26:46 +00:00
name : 'System Internal' ,
desc : 'Built in conference for private messages, bulletins, etc.' ,
2018-06-22 05:15:04 +00:00
areas : {
private _mail : {
2018-06-23 03:26:46 +00:00
name : 'Private Mail' ,
desc : 'Private user to user mail/email' ,
maxExternalSentAgeDays : 30 , // max external "outbox" item age
2018-06-22 05:15:04 +00:00
} ,
local _bulletin : {
2018-06-23 03:26:46 +00:00
name : 'System Bulletins' ,
desc : 'Bulletin messages for all users' ,
2018-06-22 05:15:04 +00:00
}
}
}
} ,
scannerTossers : {
ftn _bso : {
paths : {
2018-06-23 03:26:46 +00:00
outbound : paths . join ( _ _dirname , './../mail/ftn_out/' ) ,
inbound : paths . join ( _ _dirname , './../mail/ftn_in/' ) ,
secInbound : paths . join ( _ _dirname , './../mail/ftn_secin/' ) ,
reject : paths . join ( _ _dirname , './../mail/reject/' ) , // bad pkt, bundles, TIC attachments that fail any check, etc.
//outboundNetMail : paths.join(__dirname, './../mail/ftn_netmail_out/'),
// set 'retain' to a valid path to keep good pkt files
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// Packet and (ArcMail) bundle target sizes are just that: targets.
// Actual sizes may be slightly larger when we must place a full
// PKT contents *somewhere*
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
packetTargetByteSize : 512000 , // 512k, before placing messages in a new pkt
bundleTargetByteSize : 2048000 , // 2M, before creating another archive
packetMsgEncoding : 'utf8' , // default packet encoding. Override per node if desired.
packetAnsiMsgEncoding : 'cp437' , // packet encoding for *ANSI ART* messages
2018-06-22 05:15:04 +00:00
tic : {
2018-06-23 03:26:46 +00:00
secureInOnly : true , // only bring in from secure inbound (|secInbound| path, password protected)
uploadBy : 'ENiGMA TIC' , // default upload by username (override @ network)
allowReplace : false , // use "Replaces" TIC field
descPriority : 'diz' , // May be diz=.DIZ/etc., or tic=from TIC Ldesc
2018-06-22 05:15:04 +00:00
}
}
} ,
fileBase : {
2018-06-23 03:26:46 +00:00
// areas with an explicit |storageDir| will be stored relative to |areaStoragePrefix|:
areaStoragePrefix : paths . join ( _ _dirname , './../file_base/' ) ,
2018-06-22 05:15:04 +00:00
2018-06-23 03:26:46 +00:00
maxDescFileByteSize : 471859 , // ~1/4 MB
maxDescLongFileByteSize : 524288 , // 1/2 MB
2018-06-22 05:15:04 +00:00
fileNamePatterns : {
2018-06-23 03:26:46 +00:00
// These are NOT case sensitive
// FILE_ID.DIZ - https://en.wikipedia.org/wiki/FILE_ID.DIZ
// Some groups include a FILE_ID.ANS. We try to use that over FILE_ID.DIZ if available.
desc : [
'^[^/\]*FILE_ID\.ANS$' , '^[^/\]*FILE_ID\.DIZ$' , '^[^/\]*DESC\.SDI$' , '^[^/\]*DESCRIPT\.ION$' , '^[^/\]*FILE\.DES$' , '^[^/\]*FILE\.SDI$' , '^[^/\]*DISK\.ID$' // eslint-disable-line no-useless-escape
2018-06-22 05:15:04 +00:00
] ,
2018-06-23 03:26:46 +00:00
// common README filename - https://en.wikipedia.org/wiki/README
descLong : [
2018-07-16 01:18:44 +00:00
'^[^/\]*\.NFO$' , '^[^/\]*README\.1ST$' , '^[^/\]*README\.NOW$' , '^[^/\]*README\.TXT$' , '^[^/\]*READ\.ME$' , '^[^/\]*README$' , '^[^/\]*README\.md$' , // eslint-disable-line no-useless-escape
'^RELEASE-INFO.ASC$' // eslint-disable-line no-useless-escape
2018-06-22 05:15:04 +00:00
] ,
} ,
yearEstPatterns : [
//
2018-06-23 03:26:46 +00:00
// Patterns should produce the year in the first submatch.
// The extracted year may be YY or YYYY
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
'\\b((?:[1-2][0-9][0-9]{2}))[\\-\\/\\.][0-3]?[0-9][\\-\\/\\.][0-3]?[0-9]\\b' , // yyyy-mm-dd, yyyy/mm/dd, ...
'\\b[0-3]?[0-9][\\-\\/\\.][0-3]?[0-9][\\-\\/\\.]((?:[1-2][0-9][0-9]{2}))\\b' , // mm/dd/yyyy, mm.dd.yyyy, ...
'\\b((?:[1789][0-9]))[\\-\\/\\.][0-3]?[0-9][\\-\\/\\.][0-3]?[0-9]\\b' , // yy-mm-dd, yy-mm-dd, ...
'\\b[0-3]?[0-9][\\-\\/\\.][0-3]?[0-9][\\-\\/\\.]((?:[1789][0-9]))\\b' , // mm-dd-yy, mm/dd/yy, ...
//'\\b((?:[1-2][0-9][0-9]{2}))[\\-\\/\\.][0-3]?[0-9][\\-\\/\\.][0-3]?[0-9]|[0-3]?[0-9][\\-\\/\\.][0-3]?[0-9][\\-\\/\\.]((?:[0-9]{2})?[0-9]{2})\\b', // yyyy-mm-dd, m/d/yyyy, mm-dd-yyyy, etc.
//"\\b('[1789][0-9])\\b", // eslint-disable-line quotes
2018-06-22 05:15:04 +00:00
'\\b[0-3]?[0-9][\\-\\/\\.](?:jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec|january|february|march|april|may|june|july|august|september|october|november|december)[\\-\\/\\.]((?:[0-9]{2})?[0-9]{2})\\b' ,
2018-06-23 03:26:46 +00:00
'\\b(?:jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec|january|february|march|april|may|june|july|august|september|october|november|december),?\\s[0-9]+(?:st|nd|rd|th)?,?\\s((?:[0-9]{2})?[0-9]{2})\\b' , // November 29th, 1997
'\\(((?:19|20)[0-9]{2})\\)' , // (19xx) or (20xx) -- with parens -- do this before 19xx 20xx such that this has priority
'\\b((?:19|20)[0-9]{2})\\b' , // simple 19xx or 20xx with word boundaries
'\\b\'([17-9][0-9])\\b' , // '95, '17, ...
// :TODO: DD/MMM/YY, DD/MMMM/YY, DD/MMM/YYYY, etc.
2018-06-22 05:15:04 +00:00
] ,
web : {
2018-06-23 03:26:46 +00:00
path : '/f/' ,
routePath : '/f/[a-zA-Z0-9]+$' ,
expireMinutes : 1440 , // 1 day
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// File area storage location tag/value pairs.
// Non-absolute paths are relative to |areaStoragePrefix|.
2018-06-22 05:15:04 +00:00
//
storageTags : {
2018-06-23 03:26:46 +00:00
sys _msg _attach : 'sys_msg_attach' ,
sys _temp _download : 'sys_temp_download' ,
2018-06-22 05:15:04 +00:00
} ,
areas : {
system _message _attachment : {
2018-06-23 03:26:46 +00:00
name : 'System Message Attachments' ,
desc : 'File attachments to messages' ,
storageTags : [ 'sys_msg_attach' ] ,
2018-06-22 05:15:04 +00:00
} ,
system _temporary _download : {
2018-06-23 03:26:46 +00:00
name : 'System Temporary Downloads' ,
desc : 'Temporary downloadables' ,
storageTags : [ 'sys_temp_download' ] ,
2018-06-22 05:15:04 +00:00
}
}
} ,
eventScheduler : {
events : {
trimMessageAreas : {
2018-06-23 03:26:46 +00:00
// may optionally use [or ]@watch:/path/to/file
schedule : 'every 24 hours' ,
2018-06-22 05:15:04 +00:00
2018-06-23 03:26:46 +00:00
// action:
// - @method:path/to/module.js:theMethodName
// (path is relative to engima base dir)
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
// - @execute:/path/to/something/executable.sh
2018-06-22 05:15:04 +00:00
//
2018-06-23 03:26:46 +00:00
action : '@method:core/message_area.js:trimMessageAreasScheduledEvent' ,
2018-06-22 05:15:04 +00:00
} ,
updateFileAreaStats : {
2018-06-23 03:26:46 +00:00
schedule : 'every 1 hours' ,
action : '@method:core/file_base_area.js:updateAreaStatsScheduledEvent' ,
2018-06-22 05:15:04 +00:00
} ,
forgotPasswordMaintenance : {
2018-06-23 03:26:46 +00:00
schedule : 'every 24 hours' ,
action : '@method:core/web_password_reset.js:performMaintenanceTask' ,
args : [ '24 hours' ] // items older than this will be removed
2018-06-22 05:15:04 +00:00
} ,
//
2018-06-23 03:26:46 +00:00
// Enable the following entry in your config.hjson to periodically create/update
// DESCRIPT.ION files for your file base
2018-06-22 05:15:04 +00:00
//
/ *
2018-06-23 03:26:46 +00:00
updateDescriptIonFiles : {
schedule : 'on the last day of the week' ,
action : '@method:core/file_base_list_export.js:updateFileBaseDescFilesScheduledEvent' ,
}
* /
2018-06-22 05:15:04 +00:00
}
} ,
logging : {
2018-06-23 03:26:46 +00:00
rotatingFile : { // set to 'disabled' or false to disable
type : 'rotating-file' ,
fileName : 'enigma-bbs.log' ,
period : '1d' ,
count : 3 ,
level : 'debug' ,
2018-06-22 05:15:04 +00:00
}
2018-06-23 03:26:46 +00:00
// :TODO: syslog - https://github.com/mcavage/node-bunyan-syslog
2018-06-22 05:15:04 +00:00
} ,
debug : {
2018-06-23 03:26:46 +00:00
assertsEnabled : false ,
2018-11-11 08:55:38 +00:00
} ,
statLog : {
systemEvents : {
2018-11-28 02:45:36 +00:00
loginHistoryMax : - 1 , // set to -1 for forever
2018-11-11 08:55:38 +00:00
}
2018-06-22 05:15:04 +00:00
}
} ;
2015-04-19 08:13:13 +00:00
}