| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 | 
							- const _ = require('lodash')
 
- const autoload = require('auto-load')
 
- const path = require('path')
 
- const Promise = require('bluebird')
 
- const Knex = require('knex')
 
- const fs = require('fs')
 
- const Objection = require('objection')
 
- const migrationSource = require('../db/migrator-source')
 
- const migrateFromBeta = require('../db/beta')
 
- /* global WIKI */
 
- /**
 
-  * ORM DB module
 
-  */
 
- module.exports = {
 
-   Objection,
 
-   knex: null,
 
-   listener: null,
 
-   /**
 
-    * Initialize DB
 
-    *
 
-    * @return     {Object}  DB instance
 
-    */
 
-   init() {
 
-     let self = this
 
-     // Fetch DB Config
 
-     let dbClient = null
 
-     let dbConfig = (!_.isEmpty(process.env.DATABASE_URL)) ? process.env.DATABASE_URL : {
 
-       host: WIKI.config.db.host.toString(),
 
-       user: WIKI.config.db.user.toString(),
 
-       password: WIKI.config.db.pass.toString(),
 
-       database: WIKI.config.db.db.toString(),
 
-       port: WIKI.config.db.port
 
-     }
 
-     // Handle SSL Options
 
-     let dbUseSSL = (WIKI.config.db.ssl === true || WIKI.config.db.ssl === 'true' || WIKI.config.db.ssl === 1 || WIKI.config.db.ssl === '1')
 
-     let sslOptions = null
 
-     if (dbUseSSL && _.isPlainObject(dbConfig) && _.get(WIKI.config.db, 'sslOptions.auto', null) === false) {
 
-       sslOptions = WIKI.config.db.sslOptions
 
-       sslOptions.rejectUnauthorized = sslOptions.rejectUnauthorized !== false
 
-       if (sslOptions.ca && sslOptions.ca.indexOf('-----') !== 0) {
 
-         sslOptions.ca = fs.readFileSync(path.resolve(WIKI.ROOTPATH, sslOptions.ca))
 
-       }
 
-       if (sslOptions.cert) {
 
-         sslOptions.cert = fs.readFileSync(path.resolve(WIKI.ROOTPATH, sslOptions.cert))
 
-       }
 
-       if (sslOptions.key) {
 
-         sslOptions.key = fs.readFileSync(path.resolve(WIKI.ROOTPATH, sslOptions.key))
 
-       }
 
-       if (sslOptions.pfx) {
 
-         sslOptions.pfx = fs.readFileSync(path.resolve(WIKI.ROOTPATH, sslOptions.pfx))
 
-       }
 
-     } else {
 
-       sslOptions = true
 
-     }
 
-     // Handle inline SSL CA Certificate mode
 
-     if (!_.isEmpty(process.env.DB_SSL_CA)) {
 
-       const chunks = []
 
-       for (let i = 0, charsLength = process.env.DB_SSL_CA.length; i < charsLength; i += 64) {
 
-         chunks.push(process.env.DB_SSL_CA.substring(i, i + 64))
 
-       }
 
-       dbUseSSL = true
 
-       sslOptions = {
 
-         rejectUnauthorized: true,
 
-         ca: '-----BEGIN CERTIFICATE-----\n' + chunks.join('\n') + '\n-----END CERTIFICATE-----\n'
 
-       }
 
-     }
 
-     // Engine-specific config
 
-     switch (WIKI.config.db.type) {
 
-       case 'postgres':
 
-         dbClient = 'pg'
 
-         if (dbUseSSL && _.isPlainObject(dbConfig)) {
 
-           dbConfig.ssl = (sslOptions === true) ? { rejectUnauthorized: true } : sslOptions
 
-         }
 
-         break
 
-       case 'mariadb':
 
-       case 'mysql':
 
-         dbClient = 'mysql2'
 
-         if (dbUseSSL && _.isPlainObject(dbConfig)) {
 
-           dbConfig.ssl = sslOptions
 
-         }
 
-         // Fix mysql boolean handling...
 
-         dbConfig.typeCast = (field, next) => {
 
-           if (field.type === 'TINY' && field.length === 1) {
 
-             let value = field.string()
 
-             return value ? (value === '1') : null
 
-           }
 
-           return next()
 
-         }
 
-         break
 
-       case 'mssql':
 
-         dbClient = 'mssql'
 
-         if (_.isPlainObject(dbConfig)) {
 
-           dbConfig.appName = 'Wiki.js'
 
-           _.set(dbConfig, 'options.appName', 'Wiki.js')
 
-           dbConfig.enableArithAbort = true
 
-           _.set(dbConfig, 'options.enableArithAbort', true)
 
-           if (dbUseSSL) {
 
-             dbConfig.encrypt = true
 
-             _.set(dbConfig, 'options.encrypt', true)
 
-           }
 
-         }
 
-         break
 
-       case 'sqlite':
 
-         dbClient = 'sqlite3'
 
-         dbConfig = { filename: WIKI.config.db.storage }
 
-         break
 
-       default:
 
-         WIKI.logger.error('Invalid DB Type')
 
-         process.exit(1)
 
-     }
 
-     // Initialize Knex
 
-     this.knex = Knex({
 
-       client: dbClient,
 
-       useNullAsDefault: true,
 
-       asyncStackTraces: WIKI.IS_DEBUG,
 
-       connection: dbConfig,
 
-       pool: {
 
-         ...WIKI.config.pool,
 
-         async afterCreate(conn, done) {
 
-           // -> Set Connection App Name
 
-           switch (WIKI.config.db.type) {
 
-             case 'postgres':
 
-               await conn.query(`set application_name = 'Wiki.js'`)
 
-               done()
 
-               break
 
-             case 'mysql':
 
-               await conn.promise().query(`set autocommit = 1`)
 
-               done()
 
-               break
 
-             default:
 
-               done()
 
-               break
 
-           }
 
-         }
 
-       },
 
-       debug: WIKI.IS_DEBUG
 
-     })
 
-     Objection.Model.knex(this.knex)
 
-     // Load DB Models
 
-     const models = autoload(path.join(WIKI.SERVERPATH, 'models'))
 
-     // Set init tasks
 
-     let conAttempts = 0
 
-     let initTasks = {
 
-       // -> Attempt initial connection
 
-       async connect () {
 
-         try {
 
-           WIKI.logger.info('Connecting to database...')
 
-           await self.knex.raw('SELECT 1 + 1;')
 
-           WIKI.logger.info('Database Connection Successful [ OK ]')
 
-         } catch (err) {
 
-           if (conAttempts < 10) {
 
-             if (err.code) {
 
-               WIKI.logger.error(`Database Connection Error: ${err.code} ${err.address}:${err.port}`)
 
-             } else {
 
-               WIKI.logger.error(`Database Connection Error: ${err.message}`)
 
-             }
 
-             WIKI.logger.warn(`Will retry in 3 seconds... [Attempt ${++conAttempts} of 10]`)
 
-             await new Promise(resolve => setTimeout(resolve, 3000))
 
-             await initTasks.connect()
 
-           } else {
 
-             throw err
 
-           }
 
-         }
 
-       },
 
-       // -> Migrate DB Schemas
 
-       async syncSchemas () {
 
-         return self.knex.migrate.latest({
 
-           tableName: 'migrations',
 
-           migrationSource
 
-         })
 
-       },
 
-       // -> Migrate DB Schemas from beta
 
-       async migrateFromBeta () {
 
-         return migrateFromBeta.migrate(self.knex)
 
-       }
 
-     }
 
-     let initTasksQueue = (WIKI.IS_MASTER) ? [
 
-       initTasks.connect,
 
-       initTasks.migrateFromBeta,
 
-       initTasks.syncSchemas
 
-     ] : [
 
-       () => { return Promise.resolve() }
 
-     ]
 
-     // Perform init tasks
 
-     WIKI.logger.info(`Using database driver ${dbClient} for ${WIKI.config.db.type} [ OK ]`)
 
-     this.onReady = Promise.each(initTasksQueue, t => t()).return(true)
 
-     return {
 
-       ...this,
 
-       ...models
 
-     }
 
-   },
 
-   /**
 
-    * Subscribe to database LISTEN / NOTIFY for multi-instances events
 
-    */
 
-   async subscribeToNotifications () {
 
-     const useHA = (WIKI.config.ha === true || WIKI.config.ha === 'true' || WIKI.config.ha === 1 || WIKI.config.ha === '1')
 
-     if (!useHA) {
 
-       return
 
-     } else if (WIKI.config.db.type !== 'postgres') {
 
-       WIKI.logger.warn(`Database engine doesn't support pub/sub. Will not handle concurrent instances: [ DISABLED ]`)
 
-       return
 
-     }
 
-     const PGPubSub = require('pg-pubsub')
 
-     this.listener = new PGPubSub(this.knex.client.connectionSettings, {
 
-       log (ev) {
 
-         WIKI.logger.debug(ev)
 
-       }
 
-     })
 
-     // -> Outbound events handling
 
-     this.listener.addChannel('wiki', payload => {
 
-       if (_.has(payload, 'event') && payload.source !== WIKI.INSTANCE_ID) {
 
-         WIKI.logger.info(`Received event ${payload.event} from instance ${payload.source}: [ OK ]`)
 
-         WIKI.events.inbound.emit(payload.event, payload.value)
 
-       }
 
-     })
 
-     WIKI.events.outbound.onAny(this.notifyViaDB)
 
-     // -> Listen to inbound events
 
-     WIKI.auth.subscribeToEvents()
 
-     WIKI.configSvc.subscribeToEvents()
 
-     WIKI.models.pages.subscribeToEvents()
 
-     WIKI.logger.info(`High-Availability Listener initialized successfully: [ OK ]`)
 
-   },
 
-   /**
 
-    * Unsubscribe from database LISTEN / NOTIFY
 
-    */
 
-   async unsubscribeToNotifications () {
 
-     if (this.listener) {
 
-       WIKI.events.outbound.offAny(this.notifyViaDB)
 
-       WIKI.events.inbound.removeAllListeners()
 
-       this.listener.close()
 
-     }
 
-   },
 
-   /**
 
-    * Publish event via database NOTIFY
 
-    *
 
-    * @param {string} event Event fired
 
-    * @param {object} value Payload of the event
 
-    */
 
-   notifyViaDB (event, value) {
 
-     WIKI.models.listener.publish('wiki', {
 
-       source: WIKI.INSTANCE_ID,
 
-       event,
 
-       value
 
-     })
 
-   }
 
- }
 
 
  |