index.js 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. 'use strict';
  2. var debug = require('debug')('crowi:crowi')
  3. , pkg = require('../../package.json')
  4. , path = require('path')
  5. , fs = require('fs')
  6. , sep = path.sep
  7. , Promise = require('bluebird')
  8. , mongoose = require('mongoose')
  9. , models = require('../models')
  10. ;
  11. function Crowi (rootdir, env)
  12. {
  13. var self = this;
  14. this.version = pkg.version;
  15. this.rootDir = rootdir;
  16. this.pluginDir = path.join(this.rootDir, 'node_modules') + sep;
  17. this.publicDir = path.join(this.rootDir, 'public') + sep;
  18. this.libDir = path.join(this.rootDir, 'lib') + sep;
  19. this.eventsDir = path.join(this.libDir, 'events') + sep;
  20. this.resourceDir = path.join(this.rootDir, 'resource') + sep;
  21. this.viewsDir = path.join(this.libDir, 'views') + sep;
  22. this.mailDir = path.join(this.viewsDir, 'mail') + sep;
  23. this.tmpDir = path.join(this.rootDir, 'tmp') + sep;
  24. this.cacheDir = path.join(this.tmpDir, 'cache');
  25. this.assets = {};
  26. try {
  27. this.assets = require(this.publicDir + '/js/manifest.json') || {};
  28. var isEnablePlugin = true; // TODO configurable
  29. if (isEnablePlugin) {
  30. var pluginAssets = require(this.publicDir + '/js/manifest-plugin.json') || {};
  31. this.assets = Object.assign(this.assets, pluginAssets);
  32. }
  33. } catch (e) {
  34. // ignore
  35. }
  36. this.config = {};
  37. this.searcher = null;
  38. this.mailer = {};
  39. this.tokens = null;
  40. this.models = {};
  41. this.env = env;
  42. this.node_env = this.env.NODE_ENV || 'development';
  43. this.port = this.env.PORT || 3000;
  44. this.events = {
  45. user: new (require(self.eventsDir + 'user'))(this),
  46. page: new (require(self.eventsDir + 'page'))(this),
  47. };
  48. if (this.node_env == 'development') {
  49. Promise.longStackTraces();
  50. }
  51. //time.tzset('Asia/Tokyo');
  52. };
  53. Crowi.prototype.init = function() {
  54. var self = this;
  55. return new Promise.resolve()
  56. .then(function() {
  57. // setup database server and load all modesl
  58. return self.setupDatabase();
  59. }).then(function() {
  60. return self.setupModels();
  61. }).then(function() {
  62. return self.setupSessionConfig();
  63. }).then(function() {
  64. return new Promise(function(resolve, reject) {
  65. self.model('Config', require('../models/config')(self));
  66. var Config = self.model('Config');
  67. Config.loadAllConfig(function(err, doc) {
  68. if (err) {
  69. return reject();
  70. }
  71. self.setConfig(doc);
  72. return resolve();
  73. });
  74. });
  75. }).then(function() {
  76. return self.setupSearcher();
  77. }).then(function() {
  78. return self.setupMailer();
  79. }).then(function() {
  80. return self.setupSlack();
  81. }).then(function() {
  82. return self.setupCsrf();
  83. }).then(function() {
  84. return self.buildServer();
  85. });
  86. }
  87. Crowi.prototype.setConfig = function(config) {
  88. this.config = config;
  89. };
  90. Crowi.prototype.getConfig = function() {
  91. return this.config;
  92. };
  93. Crowi.prototype.getAssetList = function() {
  94. if (this.node_env !== 'development') {
  95. return this.assets;
  96. }
  97. // reload manifest
  98. try {
  99. this.assets = JSON.parse(fs.readFileSync(this.publicDir + '/js/manifest.json'))|| {};
  100. var isEnablePlugin = true; // TODO configurable
  101. if (isEnablePlugin) {
  102. var pluginAssets = require(this.publicDir + '/js/manifest-plugin.json') || {};
  103. this.assets = Object.assign(this.assets, pluginAssets);
  104. }
  105. } catch (e) {
  106. // ignore
  107. debug('Failed to reload assets on development', e);
  108. }
  109. return this.assets;
  110. };
  111. // getter/setter of model instance
  112. //
  113. Crowi.prototype.model = function(name, model) {
  114. if (model) {
  115. return this.models[name] = model;
  116. }
  117. return this.models[name];
  118. };
  119. // getter/setter of event instance
  120. Crowi.prototype.event = function(name, event) {
  121. if (event) {
  122. return this.events[name] = event;
  123. }
  124. return this.events[name];
  125. };
  126. Crowi.prototype.setupDatabase = function() {
  127. // mongoUri = mongodb://user:password@host/dbname
  128. var mongoUri = this.env.MONGOLAB_URI || // for B.C.
  129. this.env.MONGODB_URI || // MONGOLAB changes their env name
  130. this.env.MONGOHQ_URL ||
  131. this.env.MONGO_URI ||
  132. 'mongodb://localhost/crowi'
  133. ;
  134. return new Promise(function(resolve, reject) {
  135. mongoose.connect(mongoUri, function(e) {
  136. if (e) {
  137. debug('DB Connect Error: ', e);
  138. debug('DB Connect Error: ', mongoUri);
  139. return reject(new Error('Cann\'t connect to Database Server.'));
  140. }
  141. return resolve();
  142. });
  143. });
  144. };
  145. Crowi.prototype.setupSessionConfig = function() {
  146. var self = this
  147. , session = require('express-session')
  148. , sessionConfig
  149. , sessionAge = (1000*3600*24*30)
  150. , redisUrl = this.env.REDISTOGO_URL || this.env.REDIS_URL || null
  151. , RedisStore
  152. ;
  153. return new Promise(function(resolve, reject) {
  154. sessionConfig = {
  155. rolling: true,
  156. secret: self.env.SECRET_TOKEN || 'this is default session secret',
  157. resave: false,
  158. saveUninitialized: true,
  159. cookie: {
  160. maxAge: sessionAge,
  161. },
  162. };
  163. if (redisUrl) {
  164. var ru = require('url').parse(redisUrl);
  165. var redis = require('redis');
  166. var redisClient = redis.createClient(ru.port, ru.hostname);
  167. if (ru.auth) {
  168. redisClient.auth(ru.auth.split(':')[1]);
  169. }
  170. RedisStore = require('connect-redis')(session);
  171. sessionConfig.store = new RedisStore({
  172. prefix: 'crowi:sess:',
  173. client: redisClient,
  174. });
  175. }
  176. self.sessionConfig = sessionConfig;
  177. resolve();
  178. });
  179. };
  180. Crowi.prototype.setupModels = function() {
  181. var self = this
  182. ;
  183. return new Promise(function(resolve, reject) {
  184. Object.keys(models).forEach(function(key) {
  185. self.model(key, models[key](self));
  186. });
  187. resolve();
  188. });
  189. };
  190. Crowi.prototype.getIo = function() {
  191. return this.io;
  192. };
  193. Crowi.prototype.getSearcher = function() {
  194. return this.searcher;
  195. };
  196. Crowi.prototype.getMailer = function() {
  197. return this.mailer;
  198. };
  199. Crowi.prototype.setupSearcher = function() {
  200. var self = this;
  201. var searcherUri = this.env.ELASTICSEARCH_URI
  202. || this.env.BONSAI_URL
  203. || null
  204. ;
  205. return new Promise(function(resolve, reject) {
  206. if (searcherUri) {
  207. try {
  208. self.searcher = new (require(path.join(self.libDir, 'util', 'search')))(self, searcherUri);
  209. } catch (e) {
  210. debug('Error on setup searcher', e);
  211. self.searcher = null;
  212. }
  213. }
  214. resolve();
  215. });
  216. };
  217. Crowi.prototype.setupMailer = function() {
  218. var self = this;
  219. return new Promise(function(resolve, reject) {
  220. self.mailer = require('../util/mailer')(self);
  221. resolve();
  222. });
  223. };
  224. Crowi.prototype.setupSlack = function() {
  225. var self = this;
  226. var config = this.getConfig();
  227. var Config = this.model('Config');
  228. return new Promise(function(resolve, reject) {
  229. if (!Config.hasSlackConfig(config)) {
  230. self.slack = {};
  231. } else {
  232. self.slack = require('../util/slack')(self);
  233. }
  234. resolve();
  235. });
  236. };
  237. Crowi.prototype.setupCsrf = function() {
  238. var Tokens = require('csrf');
  239. var tokens = this.tokens = new Tokens();
  240. return Promise.resolve();
  241. };
  242. Crowi.prototype.getTokens = function() {
  243. return this.tokens;
  244. };
  245. Crowi.prototype.start = function() {
  246. var self = this
  247. , http = require('http')
  248. , server
  249. , io;
  250. return self.buildServer()
  251. .then(function(app) {
  252. server = http.createServer(app).listen(self.port, function() {
  253. console.log('[' + self.node_env + '] Express server listening on port ' + self.port);
  254. });
  255. io = require('socket.io')(server);
  256. io.sockets.on('connection', function (socket) {
  257. });
  258. self.io = io;
  259. return app;
  260. });
  261. };
  262. Crowi.prototype.buildServer = function() {
  263. var express = require('express')
  264. , errorHandler = require('errorhandler')
  265. , morgan = require('morgan')
  266. , app = express()
  267. , env = this.node_env
  268. ;
  269. require('./express-init')(this, app);
  270. require('../routes')(this, app);
  271. if (env == 'development') {
  272. //swig.setDefaults({ cache: false });
  273. app.use(errorHandler({ dumpExceptions: true, showStack: true }));
  274. app.use(morgan('dev'));
  275. }
  276. if (env == 'production') {
  277. var oneYear = 31557600000;
  278. app.use(morgan('combined'));
  279. app.use(function (err, req, res, next) {
  280. res.status(500);
  281. res.render('500', { error: err });
  282. });
  283. }
  284. return new Promise.resolve(app);
  285. };
  286. Crowi.prototype.exitOnError = function(err) {
  287. debug('Critical error occured.');
  288. console.error(err);
  289. console.error(err.stack);
  290. process.exit(1);
  291. };
  292. module.exports = Crowi;