2017-11-28 17:56:53 +00:00
"use strict" ;
const path = require ( "path" ) ;
const fsextra = require ( "fs-extra" ) ;
2018-04-17 08:06:08 +00:00
const Helper = require ( "../../helper" ) ;
const Msg = require ( "../../models/msg" ) ;
2017-11-28 17:56:53 +00:00
2018-04-26 09:11:38 +00:00
let sqlite3 ;
try {
sqlite3 = require ( "sqlite3" ) ;
} catch ( e ) {
Helper . config . messageStorage = Helper . config . messageStorage . filter ( ( item ) => item !== "sqlite" ) ;
log . error ( "Unable to load sqlite3 module. You might need to install it manually." ) ;
}
2018-03-02 08:42:12 +00:00
const currentSchemaVersion = 1520239200 ;
2017-11-28 17:56:53 +00:00
const schema = [
// Schema version #1
"CREATE TABLE IF NOT EXISTS options (name TEXT, value TEXT, CONSTRAINT name_unique UNIQUE (name))" ,
"CREATE TABLE IF NOT EXISTS messages (network TEXT, channel TEXT, time INTEGER, type TEXT, msg TEXT)" ,
"CREATE INDEX IF NOT EXISTS network_channel ON messages (network, channel)" ,
"CREATE INDEX IF NOT EXISTS time ON messages (time)" ,
] ;
class MessageStorage {
2018-04-27 10:16:23 +00:00
constructor ( client ) {
this . client = client ;
2017-11-28 17:56:53 +00:00
this . isEnabled = false ;
}
2018-04-17 08:06:08 +00:00
enable ( ) {
const logsPath = Helper . getUserLogsPath ( ) ;
const sqlitePath = path . join ( logsPath , ` ${ this . client . name } .sqlite3 ` ) ;
2017-11-28 17:56:53 +00:00
try {
fsextra . ensureDirSync ( logsPath ) ;
} catch ( e ) {
log . error ( "Unable to create logs directory" , e ) ;
return ;
}
this . isEnabled = true ;
2018-03-10 16:49:16 +00:00
this . database = new sqlite3 . Database ( sqlitePath ) ;
2017-11-28 17:56:53 +00:00
this . database . serialize ( ( ) => {
schema . forEach ( ( line ) => this . database . run ( line ) ) ;
this . database . get ( "SELECT value FROM options WHERE name = 'schema_version'" , ( err , row ) => {
if ( err ) {
return log . error ( ` Failed to retrieve schema version: ${ err } ` ) ;
}
// New table
if ( row === undefined ) {
this . database . serialize ( ( ) => this . database . run ( "INSERT INTO options (name, value) VALUES ('schema_version', ?)" , currentSchemaVersion ) ) ;
return ;
}
const storedSchemaVersion = parseInt ( row . value , 10 ) ;
if ( storedSchemaVersion === currentSchemaVersion ) {
return ;
}
if ( storedSchemaVersion > currentSchemaVersion ) {
return log . error ( ` sqlite messages schema version is higher than expected ( ${ storedSchemaVersion } > ${ currentSchemaVersion } ). Is The Lounge out of date? ` ) ;
}
log . info ( ` sqlite messages schema version is out of date ( ${ storedSchemaVersion } < ${ currentSchemaVersion } ). Running migrations if any. ` ) ;
this . database . serialize ( ( ) => this . database . run ( "UPDATE options SET value = ? WHERE name = 'schema_version'" , currentSchemaVersion ) ) ;
} ) ;
} ) ;
}
2018-03-10 16:49:16 +00:00
close ( callback ) {
if ( ! this . isEnabled ) {
return ;
}
2018-03-14 06:45:52 +00:00
this . isEnabled = false ;
2018-03-10 16:49:16 +00:00
this . database . close ( ( err ) => {
if ( err ) {
log . error ( ` Failed to close sqlite database: ${ err } ` ) ;
}
if ( callback ) {
callback ( err ) ;
}
} ) ;
}
2017-11-28 17:56:53 +00:00
index ( network , channel , msg ) {
if ( ! this . isEnabled ) {
return ;
}
const clonedMsg = Object . keys ( msg ) . reduce ( ( newMsg , prop ) => {
// id is regenerated when messages are retrieved
// previews are not stored because storage is cleared on lounge restart
// type and time are stored in a separate column
if ( prop !== "id" && prop !== "previews" && prop !== "type" && prop !== "time" ) {
newMsg [ prop ] = msg [ prop ] ;
}
return newMsg ;
} , { } ) ;
this . database . serialize ( ( ) => this . database . run (
"INSERT INTO messages(network, channel, time, type, msg) VALUES(?, ?, ?, ?, ?)" ,
2018-04-17 08:06:08 +00:00
network . uuid , channel . name . toLowerCase ( ) , msg . time . getTime ( ) , msg . type , JSON . stringify ( clonedMsg )
2017-11-28 17:56:53 +00:00
) ) ;
}
/ * *
* Load messages for given channel on a given network and resolve a promise with loaded messages .
*
* @ param Network network - Network object where the channel is
* @ param Chan channel - Channel object for which to load messages for
* /
getMessages ( network , channel ) {
2018-06-11 11:08:05 +00:00
if ( ! this . isEnabled || Helper . config . maxHistory === 0 ) {
2017-11-28 17:56:53 +00:00
return Promise . resolve ( [ ] ) ;
}
2018-06-11 11:08:05 +00:00
// If unlimited history is specified, load 100k messages
const limit = Helper . config . maxHistory < 0 ? 100000 : Helper . config . maxHistory ;
2017-11-28 17:56:53 +00:00
return new Promise ( ( resolve , reject ) => {
this . database . parallelize ( ( ) => this . database . all (
"SELECT msg, type, time FROM messages WHERE network = ? AND channel = ? ORDER BY time DESC LIMIT ?" ,
2018-06-11 11:08:05 +00:00
[ network . uuid , channel . name . toLowerCase ( ) , limit ] ,
2017-11-28 17:56:53 +00:00
( err , rows ) => {
if ( err ) {
return reject ( err ) ;
}
resolve ( rows . map ( ( row ) => {
const msg = JSON . parse ( row . msg ) ;
msg . time = row . time ;
msg . type = row . type ;
2018-04-27 10:16:23 +00:00
const newMsg = new Msg ( msg ) ;
newMsg . id = this . client . idMsg ++ ;
return newMsg ;
2017-11-28 17:56:53 +00:00
} ) . reverse ( ) ) ;
}
) ) ;
} ) ;
}
2018-04-17 08:06:08 +00:00
canProvideMessages ( ) {
return this . isEnabled ;
}
2017-11-28 17:56:53 +00:00
}
module . exports = MessageStorage ;