2017-09-29 23:21:48 +00:00
// Check for environmental variables.
require ( 'checkenv' ) . check ( ) ;
2017-08-03 01:48:16 +00:00
const discord = require ( 'discord.js' ) ;
const path = require ( 'path' ) ;
const schedule = require ( 'node-schedule' ) ;
2017-10-05 02:21:36 +00:00
const fs = require ( 'fs' ) ;
2016-12-08 03:52:37 +00:00
2017-08-03 01:48:16 +00:00
const logger = require ( './logging.js' ) ;
2017-10-05 02:21:36 +00:00
const state = require ( './state.js' ) ;
2017-08-03 01:48:16 +00:00
const data = require ( './data.js' ) ;
2016-12-08 03:52:37 +00:00
2018-01-20 04:29:33 +00:00
state . responses = require ( './responses.json' ) ;
2017-09-29 23:21:48 +00:00
2016-12-08 03:52:37 +00:00
var cachedModules = [ ] ;
2016-12-31 05:49:12 +00:00
var cachedTriggers = [ ] ;
2016-12-08 03:52:37 +00:00
var client = new discord . Client ( ) ;
2017-09-29 23:38:00 +00:00
process . on ( 'unhandledRejection' , function onError ( err ) {
2017-09-24 19:44:49 +00:00
logger . error ( err ) ;
} ) ;
2017-09-29 23:38:00 +00:00
function findArray ( haystack , arr ) {
return arr . some ( function ( v ) {
return haystack . indexOf ( v ) >= 0 ;
} ) ;
}
2016-12-08 03:52:37 +00:00
client . on ( 'ready' , ( ) => {
// Initalize app channels.
2017-10-05 02:21:36 +00:00
state . logChannel = client . channels . get ( process . env . DISCORD _LOG _CHANNEL ) ;
state . guild = state . logChannel . guild ;
2016-12-08 03:52:37 +00:00
2017-03-31 01:17:21 +00:00
logger . info ( 'Bot is now online and connected to server.' ) ;
2016-12-08 03:52:37 +00:00
} ) ;
2017-09-29 23:38:00 +00:00
client . on ( 'guildMemberAdd' , ( member ) => {
2017-10-05 02:21:36 +00:00
state . stats . joins += 1 ;
2017-07-20 01:45:58 +00:00
} ) ;
2017-09-29 23:38:00 +00:00
client . on ( 'guildMemberRemove' , ( member ) => {
2017-10-05 02:21:36 +00:00
state . stats . leaves += 1 ;
2017-07-20 01:45:58 +00:00
} ) ;
2017-10-05 02:21:36 +00:00
// Output the stats for state.stats every 24 hours.
2017-07-20 02:05:02 +00:00
// Server is in UTC mode, 11:30 EST would be 03:30 UTC.
2017-09-29 23:38:00 +00:00
schedule . scheduleJob ( { hour : 3 , minute : 30 } , function ( ) {
2017-10-05 02:21:36 +00:00
logger . info ( ` Here are today's stats for ${ ( new Date ( ) ) . toLocaleDateString ( ) } ! ${ state . stats . joins } users have joined, ${ state . stats . leaves } users have left, ${ state . stats . warnings } warnings have been issued. ` ) ;
state . logChannel . sendMessage ( ` Here are today's stats for ${ ( new Date ( ) ) . toLocaleDateString ( ) } ! ${ state . stats . joins } users have joined, ${ state . stats . leaves } users have left, ${ state . stats . warnings } warnings have been issued. ` ) ;
2017-07-20 01:45:58 +00:00
// Clear the stats for the day.
2017-10-05 02:21:36 +00:00
state . stats . joins = 0 ;
state . stats . leaves = 0 ;
state . stats . warnings = 0 ;
2017-07-20 01:45:58 +00:00
} ) ;
2016-12-08 03:52:37 +00:00
client . on ( 'message' , message => {
2017-09-29 23:38:00 +00:00
if ( message . author . bot && message . content . startsWith ( '.ban' ) === false ) { return ; }
2016-12-08 03:52:37 +00:00
2018-01-20 04:29:33 +00:00
if ( message . guild == null && state . responses . pmReply ) {
2016-12-08 03:52:37 +00:00
// We want to log PM attempts.
logger . info ( ` ${ message . author . username } ${ message . author } [PM]: ${ message . content } ` ) ;
2017-10-05 02:21:36 +00:00
state . logChannel . sendMessage ( ` ${ message . author } [PM]: ${ message . content } ` ) ;
2018-01-20 04:29:33 +00:00
message . reply ( state . responses . pmReply ) ;
2016-12-08 03:52:37 +00:00
return ;
}
2017-07-09 03:53:01 +00:00
logger . verbose ( ` ${ message . author . username } ${ message . author } [Channel: ${ message . channel . name } ${ message . channel } ]: ${ message . content } ` ) ;
2016-12-08 03:52:37 +00:00
2017-09-30 15:47:59 +00:00
// We want to make sure it's an actual command, not someone '...'-ing.
if ( message . content . startsWith ( '.' ) && message . content . startsWith ( '..' ) === false ) {
2016-12-08 03:52:37 +00:00
let cmd = message . content . split ( ' ' ) [ 0 ] . slice ( 1 ) ;
// Check by the name of the command.
let cachedModule = cachedModules [ ` ${ cmd } .js ` ] ;
let cachedModuleType = 'Command' ;
// Check by the quotes in the configuration.
2018-01-20 04:29:33 +00:00
if ( cachedModule == null ) { cachedModule = state . responses . quotes [ cmd ] ; cachedModuleType = 'Quote' ; }
2016-12-08 03:52:37 +00:00
if ( cachedModule ) {
// Check access permissions.
2017-09-29 23:38:00 +00:00
if ( cachedModule . roles !== undefined && findArray ( message . member . roles . map ( function ( x ) { return x . name ; } ) , cachedModule . roles ) === false ) {
2017-10-05 02:21:36 +00:00
state . logChannel . sendMessage ( ` ${ message . author } attempted to use admin command: ${ message . content } ` ) ;
2017-08-02 23:11:05 +00:00
logger . info ( ` ${ message . author . username } ${ message . author } attempted to use admin command: ${ message . content } ` ) ;
2016-12-08 03:52:37 +00:00
return false ;
}
2017-03-31 01:29:52 +00:00
logger . info ( ` ${ message . author . username } ${ message . author } [Channel: ${ message . channel } ] executed command: ${ message . content } ` ) ;
2016-12-08 03:52:37 +00:00
message . delete ( ) ;
2017-01-15 17:37:43 +00:00
try {
2017-09-29 23:38:00 +00:00
if ( cachedModuleType === 'Command' ) {
2017-01-15 17:37:43 +00:00
cachedModule . command ( message ) ;
2017-09-29 23:38:00 +00:00
} else if ( cachedModuleType === 'Quote' ) {
2017-01-15 17:37:43 +00:00
cachedModules [ 'quote.js' ] . command ( message , cachedModule . reply ) ;
}
} catch ( err ) { logger . error ( err ) ; }
2016-12-08 03:52:37 +00:00
2017-04-07 23:33:03 +00:00
// Warn after running command?
2017-01-15 17:37:43 +00:00
try {
// Check if the command requires a warning.
2017-09-29 23:38:00 +00:00
if ( cmd !== 'warn' && cachedModule . warn === true ) {
2017-04-07 23:33:03 +00:00
// Access check to see if the user has privilages to warn.
let warnCommand = cachedModules [ 'warn.js' ] ;
2017-09-29 23:38:00 +00:00
if ( findArray ( message . member . roles . map ( function ( x ) { return x . name ; } ) , warnCommand . roles ) ) {
2017-04-07 23:33:03 +00:00
// They are allowed to warn because they are in warn's roles.
warnCommand . command ( message ) ;
}
2017-01-15 17:37:43 +00:00
}
} catch ( err ) { logger . error ( err ) ; }
2016-12-08 03:52:37 +00:00
} else {
// Not a valid command.
}
2017-09-29 23:38:00 +00:00
} else if ( message . author . bot === false ) {
2016-12-31 05:49:12 +00:00
// This is a normal channel message.
2017-09-29 23:38:00 +00:00
cachedTriggers . forEach ( function ( trigger ) {
if ( trigger . roles === undefined || findArray ( message . member . roles . map ( function ( x ) { return x . name ; } ) , trigger . roles ) ) {
if ( trigger . trigger ( message ) === true ) {
logger . debug ( ` ${ message . author . username } ${ message . author } [Channel: ${ message . channel } ] triggered: ${ message . content } ` ) ;
try {
trigger . execute ( message ) ;
} catch ( err ) { logger . error ( err ) ; }
2016-12-31 05:49:12 +00:00
}
2017-09-29 23:38:00 +00:00
}
2016-12-31 05:49:12 +00:00
} ) ;
2016-12-08 03:52:37 +00:00
}
2017-03-31 01:17:21 +00:00
} ) ;
2016-12-08 03:52:37 +00:00
2017-03-31 01:17:21 +00:00
// Cache all command modules.
2017-03-31 01:18:23 +00:00
cachedModules = [ ] ;
2017-10-05 02:21:36 +00:00
fs . readdirSync ( './src/commands/' ) . forEach ( function ( file ) {
2017-03-31 01:17:21 +00:00
// Load the module if it's a script.
2017-09-29 23:38:00 +00:00
if ( path . extname ( file ) === '.js' ) {
2017-04-07 23:36:45 +00:00
if ( file . includes ( '.disabled' ) ) {
logger . info ( ` Did not load disabled module: ${ file } ` ) ;
} else {
logger . info ( ` Loaded module: ${ file } ` ) ;
cachedModules [ file ] = require ( ` ./commands/ ${ file } ` ) ;
}
2017-03-31 01:17:21 +00:00
}
2016-12-08 03:52:37 +00:00
} ) ;
2017-03-31 01:17:21 +00:00
// Cache all triggers.
2017-03-31 01:18:23 +00:00
cachedTriggers = [ ] ;
2017-10-05 02:21:36 +00:00
fs . readdirSync ( './src/triggers/' ) . forEach ( function ( file ) {
2017-03-31 01:17:21 +00:00
// Load the trigger if it's a script.
2017-09-29 23:38:00 +00:00
if ( path . extname ( file ) === '.js' ) {
2017-04-07 23:36:45 +00:00
if ( file . includes ( '.disabled' ) ) {
logger . info ( ` Did not load disabled trigger: ${ file } ` ) ;
} else {
logger . info ( ` Loaded trigger: ${ file } ` ) ;
cachedTriggers . push ( require ( ` ./triggers/ ${ file } ` ) ) ;
}
2017-03-31 01:17:21 +00:00
}
} ) ;
data . readWarnings ( ) ;
data . readBans ( ) ;
2018-01-20 04:29:33 +00:00
// Load custom responses
if ( process . env . DATA _CUSTOM _RESPONSES )
{
data . readCustomResponses ( ) ;
}
2017-09-29 23:21:48 +00:00
client . login ( process . env . DISCORD _LOGIN _TOKEN ) ;
2017-09-29 23:38:00 +00:00
logger . info ( 'Startup completed. Established connection to Discord.' ) ;