2019-10-15 22:42:47 -04:00
const express = require ( 'express' ) ;
2021-01-18 21:54:18 -05:00
const request = require ( 'request' ) ;
2019-11-06 23:07:31 -05:00
const compression = require ( 'compression' ) ;
2021-01-18 21:54:18 -05:00
const timeout = require ( 'connect-timeout' ) ;
2020-09-10 09:02:40 -04:00
const rateLimit = require ( "express-rate-limit" ) ;
2021-01-18 21:54:18 -05:00
const fs = require ( "fs" ) ;
2020-09-10 09:02:40 -04:00
const app = express ( ) ;
2020-11-07 19:20:44 -05:00
2021-01-18 21:54:18 -05:00
app . config = require ( './settings.js' )
app . servers = require ( './servers.json' )
2020-11-07 19:20:44 -05:00
let rlMessage = "Rate limited ¯\\_(ツ)_/¯<br><br>Please do not spam my servers with a crazy amount of requests. It slows things down on my end and stresses RobTop's servers just as much." +
" If you really want to send a zillion requests for whatever reason, please download the GDBrowser repository locally - or even just send the request directly to the GD servers.<br><br>" +
"This kind of spam usually leads to GDBrowser getting IP banned by RobTop, and every time that happens I have to start making the rate limit even stricter. Please don't be the reason for that.<br><br>" +
"(also, keep in mind that most endpoints have a ?count parameter that let you fetch a LOT more stuff in just one request)"
2020-09-10 09:02:40 -04:00
const RL = rateLimit ( {
2020-11-01 15:29:32 -05:00
windowMs : app . config . rateLimiting ? 5 * 60 * 1000 : 0 ,
max : app . config . rateLimiting ? 100 : 0 , // max requests per 5 minutes
2020-11-07 19:20:44 -05:00
message : rlMessage ,
keyGenerator : function ( req ) { return req . headers [ 'x-real-ip' ] || req . headers [ 'x-forwarded-for' ] } ,
2020-09-10 09:02:40 -04:00
skip : function ( req ) { return ( ( req . url . includes ( "api/level" ) && ! req . query . hasOwnProperty ( "download" ) ) ? true : false ) }
} )
2019-11-06 23:07:31 -05:00
2020-11-07 19:20:44 -05:00
const RL2 = rateLimit ( {
windowMs : app . config . rateLimiting ? 2 * 60 * 1000 : 0 ,
max : app . config . rateLimiting ? 200 : 0 , // max requests per 1 minute
message : rlMessage ,
keyGenerator : function ( req ) { return req . headers [ 'x-real-ip' ] || req . headers [ 'x-forwarded-for' ] }
} )
2021-01-18 21:54:18 -05:00
let forms = { "player" : "cube" , "bird" : "ufo" , "dart" : "wave" }
let colorOrder = [ 0 , 1 , 2 , 3 , 16 , 4 , 5 , 6 , 13 , 7 , 8 , 9 , 29 , 10 , 14 , 11 , 12 , 17 , 18 , 15 , 27 , 32 , 28 , 38 , 20 , 33 , 21 , 34 , 22 , 39 , 23 , 35 , 24 , 36 , 25 , 37 , 30 , 26 , 31 , 19 , 40 , 41 ]
let XOR = require ( './classes/XOR.js' ) ;
2020-09-24 22:07:53 -04:00
let sampleIcons = require ( './misc/sampleIcons.json' )
2021-01-11 16:00:21 -05:00
let achievements = require ( './misc/achievements.json' )
let achievementTypes = require ( './misc/achievementTypes.json' )
2021-01-14 12:02:38 -05:00
let shopIcons = require ( './misc/shops.json' )
2020-11-26 16:04:42 -05:00
let colorList = require ( './icons/colors.json' )
2021-01-18 21:54:18 -05:00
let gdIcons = fs . readdirSync ( './assets/previewicons' )
2021-01-04 10:21:58 -05:00
let assetPage = fs . readFileSync ( './html/assets.html' , 'utf8' )
2020-11-26 16:04:42 -05:00
let whiteIcons = fs . readdirSync ( './icons' ) . filter ( x => x . endsWith ( "extra_001.png" ) ) . map ( function ( x ) { let xh = x . split ( "_" ) ; return [ xh [ 1 ] == "ball" ? "ball" : forms [ xh [ 0 ] ] || xh [ 0 ] , + xh [ xh [ 1 ] == "ball" ? 2 : 1 ] ] } )
2020-09-24 22:07:53 -04:00
2021-01-18 21:54:18 -05:00
app . accountCache = { }
app . lastSuccess = { }
app . actuallyWorked = { }
app . servers . forEach ( x => {
app . accountCache [ x . id || "gd" ] = { }
app . lastSuccess [ x . id || "gd" ] = Date . now ( )
} )
app . set ( 'json spaces' , 2 )
2019-10-21 08:20:25 -07:00
app . use ( compression ( ) ) ;
2019-10-15 22:42:47 -04:00
app . use ( express . json ( ) ) ;
app . use ( express . urlencoded ( { extended : true } ) ) ;
2020-11-01 15:29:32 -05:00
app . use ( timeout ( '20s' ) ) ;
2019-10-15 22:42:47 -04:00
2021-01-18 21:54:18 -05:00
app . use ( async function ( req , res , next ) {
let subdomains = req . subdomains . map ( x => x . toLowerCase ( ) )
if ( ! subdomains . length ) subdomains = [ "" ]
req . server = app . servers . find ( x => subdomains . includes ( x . id . toLowerCase ( ) ) )
if ( subdomains . length > 1 || ! req . server ) return res . redirect ( "http://" + req . get ( 'host' ) . split ( "." ) . slice ( subdomains . length ) . join ( "." ) + req . originalUrl )
// literally just for convenience
req . offline = req . server . offline
req . endpoint = req . server . endpoint
req . onePointNine = req . server . onePointNine
req . id = req . server . id || "gd"
req . isGDPS = req . server . endpoint != "http://boomlings.com/database/"
if ( req . isGDPS ) res . set ( "gdps" , ( req . onePointNine ? "1.9/" : "" ) + req . id )
2021-01-17 00:05:06 -05:00
req . gdParams = function ( obj = { } , substitute = true ) {
2020-11-01 15:29:32 -05:00
Object . keys ( app . config . params ) . forEach ( x => { if ( ! obj [ x ] ) obj [ x ] = app . config . params [ x ] } )
2021-01-18 21:54:18 -05:00
Object . keys ( req . server . extraParams || { } ) . forEach ( x => { if ( ! obj [ x ] ) obj [ x ] = req . server . extraParams [ x ] } )
2020-11-01 15:29:32 -05:00
let ip = req . headers [ 'x-real-ip' ] || req . headers [ 'x-forwarded-for' ]
2021-01-14 18:18:19 -05:00
let params = { form : obj , headers : app . config . ipForwarding && ip ? { 'x-forwarded-for' : ip , 'x-real-ip' : ip } : { } }
2021-01-17 00:05:06 -05:00
if ( substitute ) { // GDPS substitutions in settings.js
2021-01-18 21:54:18 -05:00
for ( let ss in req . server . substitutions ) {
if ( params . form [ ss ] ) { params . form [ req . server . substitutions [ ss ] ] = params . form [ ss ] ; delete params . form [ ss ] }
2021-01-17 00:05:06 -05:00
}
2021-01-14 18:18:19 -05:00
}
return params
2020-11-01 15:29:32 -05:00
}
2021-01-18 21:54:18 -05:00
req . gdRequest = function ( target , params = { } , cb = function ( ) { } ) {
if ( ! target ) return cb ( true )
target = req . server . overrides ? ( req . server . overrides [ target ] || target ) : target
let parameters = params . headers ? params : req . gdParams ( params )
let endpoint = req . endpoint
if ( params . forceGD || ( params . form && params . form . forceGD ) ) endpoint = "http://boomlings.com/database/"
request . post ( endpoint + target + '.php' , parameters , function ( err , res , body ) {
return cb ( err , res , body )
} )
}
2020-11-01 15:29:32 -05:00
next ( )
} )
2020-10-28 21:25:13 -04:00
let directories = [ "" ]
fs . readdirSync ( './api' ) . filter ( x => ! x . includes ( "." ) ) . forEach ( x => directories . push ( x ) )
2019-12-25 16:20:32 -05:00
2021-01-18 21:54:18 -05:00
app . trackSuccess = function ( id ) {
app . lastSuccess [ id ] = Date . now ( )
if ( ! app . actuallyWorked [ id ] ) app . actuallyWorked [ id ] = true
2021-01-11 16:00:21 -05:00
}
2021-01-18 21:54:18 -05:00
app . timeSince = function ( id , time ) {
if ( ! time ) time = app . lastSuccess [ id ]
2021-01-11 16:00:21 -05:00
let secsPassed = Math . floor ( ( Date . now ( ) - time ) / 1000 )
let minsPassed = Math . floor ( secsPassed / 60 )
secsPassed -= 60 * minsPassed ;
2021-01-18 21:54:18 -05:00
return ` ${ app . actuallyWorked [ id ] ? "" : "~" } ${ minsPassed } m ${ secsPassed } s `
2021-01-11 16:00:21 -05:00
}
2019-12-25 16:20:32 -05:00
app . run = { }
directories . forEach ( d => {
fs . readdirSync ( './api/' + d ) . forEach ( x => { if ( x . includes ( '.' ) ) app . run [ x . split ( '.' ) [ 0 ] ] = require ( './api/' + d + "/" + x ) } )
2019-10-16 18:47:53 -04:00
} )
2019-10-15 22:42:47 -04:00
2019-12-21 22:52:09 -05:00
try {
const secrets = require ( "./misc/secretStuff.json" )
app . id = secrets . id
app . gjp = secrets . gjp
2020-04-24 15:26:29 -04:00
app . sheetsKey = secrets . sheetsKey
2020-02-20 20:09:40 -05:00
if ( app . id == "account id goes here" || app . gjp == "account gjp goes here" ) console . warn ( "Warning: No account ID and/or GJP has been provided in secretStuff.json! These are required for level leaderboards to work." )
2020-04-24 15:26:29 -04:00
if ( app . sheetsKey . startsWith ( "google sheets api key" ) ) app . sheetsKey = undefined
2019-12-21 22:52:09 -05:00
}
2019-10-15 22:42:47 -04:00
2019-12-25 16:20:32 -05:00
catch ( e ) {
2019-12-21 22:52:09 -05:00
app . id = 0
app . gjp = 0
2020-02-20 20:09:40 -05:00
console . warn ( "Warning: secretStuff.json has not been created! These are required for level leaderboards to work." )
2019-10-15 22:42:47 -04:00
}
app . parseResponse = function ( responseBody , splitter ) {
2019-10-23 14:48:01 -07:00
if ( ! responseBody || responseBody == "-1" ) return { } ;
2021-01-15 10:29:46 -05:00
if ( responseBody . startsWith ( "\nWarning:" ) ) responseBody = responseBody . split ( "\n" ) . slice ( 2 ) . join ( "\n" ) . trim ( ) // GDPS'es are wild
if ( responseBody . startsWith ( "<br />" ) ) responseBody = responseBody . split ( "<br />" ) . slice ( 2 ) . join ( "<br />" ) . trim ( ) // Seriously screw this
2019-10-15 22:42:47 -04:00
let response = responseBody . split ( '#' ) [ 0 ] . split ( splitter || ':' ) ;
let res = { } ;
for ( let i = 0 ; i < response . length ; i += 2 ) {
res [ response [ i ] ] = response [ i + 1 ] }
2021-01-15 10:29:46 -05:00
return res
}
2019-10-15 22:42:47 -04:00
2021-01-18 21:54:18 -05:00
app . xor = new XOR ( )
2019-10-16 18:47:53 -04:00
//xss bad
2019-10-25 17:22:29 -04:00
app . clean = function ( text ) { if ( ! text || typeof text != "string" ) return text ; else return text . replace ( /&/g , "&" ) . replace ( /</g , "<" ) . replace ( />/g , ">" ) . replace ( /=/g , "=" ) . replace ( /"/g , """ ) . replace ( /'/g , "'" ) }
2019-10-15 22:42:47 -04:00
2019-12-16 08:48:05 -05:00
// ASSETS
app . use ( '/assets' , express . static ( _ _dirname + '/assets' , { maxAge : "7d" } ) ) ;
2021-01-04 10:21:58 -05:00
app . use ( '/assets/css' , express . static ( _ _dirname + '/assets/css' ) ) ; // override maxAge
app . get ( "/sizecheck.js" , function ( req , res ) { res . sendFile ( _ _dirname + "/misc/sizecheck.js" ) } )
app . get ( "/dragscroll.js" , function ( req , res ) { res . sendFile ( _ _dirname + "/misc/dragscroll.js" ) } )
app . get ( "/assets/:dir*?" , function ( req , res ) {
let main = ( req . params . dir || "" ) . toLowerCase ( )
let dir = main + ( req . params [ 0 ] || "" ) . toLowerCase ( )
if ( dir . includes ( '.' ) || ! req . path . endsWith ( "/" ) ) {
if ( ! req . params [ 0 ] ) main = ""
2021-01-11 16:00:21 -05:00
if ( req . params . dir == "deatheffects" || req . params . dir == "trails" ) return res . sendFile ( _ _dirname + "/assets/deatheffects/0.png" )
2021-01-04 10:21:58 -05:00
return res . send ( ` <p style="font-size: 20px; font-family: aller, helvetica, arial">Looks like this file doesn't exist ¯ \\ _(ツ)_/¯<br><a href='/assets/ ${ main } '>View directory listing for <b>/assets/ ${ main } </b></a></p> ` )
}
let path = ` ./assets/ ${ dir } `
let files = [ ]
if ( fs . existsSync ( path ) ) { files = fs . readdirSync ( path ) }
assetPage = fs . readFileSync ( './html/assets.html' , 'utf8' )
let assetData = JSON . stringify ( { files : files . filter ( x => x . includes ( '.' ) ) , directories : files . filter ( x => ! x . includes ( '.' ) ) } )
res . send ( assetPage . replace ( '{NAME}' , dir || "assets" ) . replace ( '{DATA}' , assetData ) )
} )
2019-10-15 22:42:47 -04:00
2019-12-16 08:48:05 -05:00
// POST REQUESTS
2019-10-15 22:42:47 -04:00
2020-09-10 09:02:40 -04:00
app . post ( "/like" , RL , function ( req , res ) { app . run . like ( app , req , res ) } )
app . post ( "/postComment" , RL , function ( req , res ) { app . run . postComment ( app , req , res ) } )
app . post ( "/postProfileComment" , RL , function ( req , res ) { app . run . postProfileComment ( app , req , res ) } )
2019-12-25 16:20:32 -05:00
2021-01-18 21:54:18 -05:00
app . post ( "/messages" , RL , function ( req , res ) { app . run . getMessages ( app , req , res ) } )
app . post ( "/messages/:id" , RL , function ( req , res ) { app . run . fetchMessage ( app , req , res ) } )
2020-09-10 09:02:40 -04:00
app . post ( "/deleteMessage" , RL , function ( req , res ) { app . run . deleteMessage ( app , req , res ) } )
app . post ( "/sendMessage" , RL , function ( req , res ) { app . run . sendMessage ( app , req , res ) } )
2019-10-15 22:42:47 -04:00
2020-10-28 21:25:13 -04:00
app . post ( "/accurateLeaderboard" , function ( req , res ) { app . run . accurate ( app , req , res , true ) } )
2020-10-02 14:33:24 -04:00
2019-12-16 08:48:05 -05:00
// HTML
2019-10-15 22:42:47 -04:00
2021-01-18 21:54:18 -05:00
let onePointNineDisabled = [ 'daily' , 'weekly' , 'gauntlets' , 'messages' ]
let downloadDisabled = [ 'daily' , 'weekly' ]
let gdpsHide = [ 'achievements' , 'messages' ]
2020-09-10 09:02:40 -04:00
app . get ( "/" , function ( req , res ) {
2021-01-18 21:54:18 -05:00
if ( req . offline && ! req . query . hasOwnProperty ( "home" ) ) res . sendFile ( _ _dirname + "/html/offline.html" )
else {
fs . readFile ( './html/home.html' , 'utf8' , function ( err , data ) {
let html = data ;
if ( req . isGDPS ) {
html = html . replace ( '"levelBG"' , '"levelBG purpleBG"' )
. replace ( /Geometry Dash Browser!/g , req . server . name + " Browser!" )
. replace ( "/assets/gdlogo" , ` /assets/gdps/ ${ req . id } _logo ` )
gdpsHide . forEach ( x => { html = html . replace ( ` menu- ${ x } ` , 'changeDaWorld' ) } )
}
if ( req . onePointNine ) onePointNineDisabled . forEach ( x => { html = html . replace ( ` menu- ${ x } ` , 'menuDisabled' ) } )
if ( req . server . downloadsDisabled ) {
downloadDisabled . forEach ( x => { html = html . replace ( ` menu- ${ x } ` , 'menuDisabled' ) } )
html = html . replace ( 'id="dl" style="display: none' , 'style="display: block' )
}
return res . send ( html )
} )
}
2020-09-10 09:02:40 -04:00
} )
2021-01-11 16:00:21 -05:00
app . get ( "/achievements" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/achievements.html" ) } )
2021-01-18 21:54:18 -05:00
app . get ( "/analyze/:id" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/analyze.html" ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/api" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/api.html" ) } )
2020-11-16 19:28:24 -05:00
app . get ( "/boomlings" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/boomlings.html" ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/comments/:id" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/comments.html" ) } )
2020-10-28 21:25:13 -04:00
app . get ( "/demon/:id" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/demon.html" ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/gauntlets" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/gauntlets.html" ) } )
2021-01-18 21:54:18 -05:00
app . get ( "/gdps" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/gdps.html" ) } )
2019-12-16 15:01:46 -05:00
app . get ( "/iconkit" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/iconkit.html" ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/leaderboard" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/leaderboard.html" ) } )
app . get ( "/leaderboard/:text" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/levelboard.html" ) } )
app . get ( "/mappacks" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/mappacks.html" ) } )
2019-12-29 18:59:29 -05:00
app . get ( "/messages" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/messages.html" ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/search" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/filters.html" ) } )
app . get ( "/search/:text" , function ( req , res ) { res . sendFile ( _ _dirname + "/html/search.html" ) } )
2019-10-15 22:42:47 -04:00
2019-12-16 08:48:05 -05:00
// API
2019-10-20 23:33:01 -04:00
2021-01-18 21:54:18 -05:00
app . get ( "/api/analyze/:id" , RL , function ( req , res ) { app . run . level ( app , req , res , true , true ) } )
2020-11-16 19:28:24 -05:00
app . get ( "/api/boomlings" , function ( req , res ) { app . run . boomlings ( app , req , res ) } )
2020-11-07 19:20:44 -05:00
app . get ( "/api/comments/:id" , RL2 , function ( req , res ) { app . run . comments ( app , req , res ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/api/credits" , function ( req , res ) { res . send ( require ( './misc/credits.json' ) ) } )
2021-01-18 21:54:18 -05:00
app . get ( "/api/gauntlets" , function ( req , res ) { app . run . gauntlets ( app , req , res ) } )
2020-10-28 21:25:13 -04:00
app . get ( "/api/leaderboard" , function ( req , res ) { app . run [ req . query . hasOwnProperty ( "accurate" ) ? "accurate" : "scores" ] ( app , req , res ) } )
2020-11-07 19:20:44 -05:00
app . get ( "/api/leaderboardLevel/:id" , RL2 , function ( req , res ) { app . run . leaderboardLevel ( app , req , res ) } )
2021-01-18 21:54:18 -05:00
app . get ( "/api/level/:id" , RL , function ( req , res ) { app . run . level ( app , req , res , true ) } )
app . get ( "/api/mappacks" , function ( req , res ) { app . run . mappacks ( app , req , res ) } )
app . get ( "/api/profile/:id" , RL2 , function ( req , res ) { app . run . profile ( app , req , res , true ) } )
2020-11-07 19:20:44 -05:00
app . get ( "/api/search/:text" , RL2 , function ( req , res ) { app . run . search ( app , req , res ) } )
2021-01-05 11:03:04 -05:00
app . get ( "/api/song/:song" , function ( req , res ) { app . run . song ( app , req , res ) } )
2019-12-16 08:48:05 -05:00
2019-10-15 22:42:47 -04:00
2019-12-16 08:48:05 -05:00
// REDIRECTS
2019-10-15 22:42:47 -04:00
2019-12-16 08:48:05 -05:00
app . get ( "/icon" , function ( req , res ) { res . redirect ( '/iconkit' ) } )
app . get ( "/iconkit/:text" , function ( req , res ) { res . redirect ( '/icon/' + req . params . text ) } )
2020-10-13 10:25:07 -04:00
app . get ( "/obj/:text" , function ( req , res ) { res . redirect ( '/obj/' + req . params . text ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/leaderboards/:id" , function ( req , res ) { res . redirect ( '/leaderboard/' + req . params . id ) } )
2020-09-29 21:42:18 -04:00
app . get ( "/profile/:id" , function ( req , res ) { res . redirect ( '/u/' + req . params . id ) } )
app . get ( "/p/:id" , function ( req , res ) { res . redirect ( '/u/' + req . params . id ) } )
2019-12-16 08:48:05 -05:00
app . get ( "/l/:id" , function ( req , res ) { res . redirect ( '/leaderboard/' + req . params . id ) } )
app . get ( "/a/:id" , function ( req , res ) { res . redirect ( '/analyze/' + req . params . id ) } )
app . get ( "/c/:id" , function ( req , res ) { res . redirect ( '/comments/' + req . params . id ) } )
2020-10-28 21:25:13 -04:00
app . get ( "/d/:id" , function ( req , res ) { res . redirect ( '/demon/' + req . params . id ) } )
2019-10-15 22:42:47 -04:00
2020-09-24 22:07:53 -04:00
// API AND HTML
2020-09-29 21:42:18 -04:00
app . get ( "/u/:id" , function ( req , res ) { app . run . profile ( app , req , res ) } )
2020-09-24 22:07:53 -04:00
app . get ( "/:id" , function ( req , res ) { app . run . level ( app , req , res ) } )
2019-12-16 08:48:05 -05:00
// MISC
2019-10-15 22:42:47 -04:00
2019-12-25 16:20:32 -05:00
app . get ( "/icon/:text" , function ( req , res ) { app . run . icon ( app , req , res ) } )
2021-01-18 21:54:18 -05:00
app . get ( "/api/gdps" , function ( req , res ) { res . send ( app . servers ) } )
2021-01-14 12:02:38 -05:00
app . get ( "/api/achievements" , function ( req , res ) { res . send ( { achievements , types : achievementTypes , shopIcons , colors : colorList } ) } )
2020-09-24 22:07:53 -04:00
app . get ( '/api/icons' , function ( req , res ) {
let sample = [ JSON . stringify ( sampleIcons [ Math . floor ( Math . random ( ) * sampleIcons . length ) ] . slice ( 1 ) ) ]
2021-01-18 21:54:18 -05:00
let iconserver = req . isGDPS ? req . server . name : undefined
res . send ( { icons : gdIcons , colors : colorList , colorOrder , whiteIcons , server : iconserver , noCopy : req . onePointNine , sample } ) ;
2020-09-24 22:07:53 -04:00
} ) ;
2019-10-15 22:42:47 -04:00
app . get ( '*' , function ( req , res ) {
2019-12-16 08:48:05 -05:00
if ( req . path . startsWith ( '/api' ) ) res . send ( '-1' )
2019-12-15 19:11:35 -05:00
else res . redirect ( '/search/404%20' )
2019-10-15 22:42:47 -04:00
} ) ;
2020-11-01 15:29:32 -05:00
app . use ( function ( err , req , res , next ) {
if ( err && err . message == "Response timeout" ) res . status ( 500 ) . send ( 'Internal server error! (Timed out)' )
} )
2021-01-04 10:21:58 -05:00
app . listen ( app . config . port , ( ) => console . log ( ` Site online! (port ${ app . config . port } ) ` ) )