'use strict'; var nconf = require('nconf'); var winston = require('winston'); var path = require('path'); var async = require('async'); var express = require('express'); var meta = require('../meta'); var controllers = require('../controllers'); var plugins = require('../plugins'); var user = require('../user'); var accountRoutes = require('./accounts'); var metaRoutes = require('./meta'); var apiRoutes = require('./api'); var adminRoutes = require('./admin'); var feedRoutes = require('./feeds'); var authRoutes = require('./authentication'); var helpers = require('./helpers'); var setupPageRoute = helpers.setupPageRoute; function mainRoutes(app, middleware, controllers) { var loginRegisterMiddleware = [middleware.redirectToAccountIfLoggedIn]; setupPageRoute(app, '/login', middleware, loginRegisterMiddleware, controllers.login); setupPageRoute(app, '/register', middleware, loginRegisterMiddleware, controllers.register); setupPageRoute(app, '/register/complete', middleware, [], controllers.registerInterstitial); setupPageRoute(app, '/compose', middleware, [], controllers.composer.get); setupPageRoute(app, '/confirm/:code', middleware, [], controllers.confirmEmail); setupPageRoute(app, '/outgoing', middleware, [], controllers.outgoing); setupPageRoute(app, '/search', middleware, [], controllers.search.search); setupPageRoute(app, '/reset/:code?', middleware, [middleware.delayLoading], controllers.reset); setupPageRoute(app, '/tos', middleware, [], controllers.termsOfUse); app.post('/compose', middleware.applyCSRF, controllers.composer.post); } function modRoutes(app, middleware, controllers) { setupPageRoute(app, '/flags', middleware, [], controllers.mods.flags.list); setupPageRoute(app, '/flags/:flagId', middleware, [], controllers.mods.flags.detail); setupPageRoute(app, '/post-queue', middleware, [], controllers.mods.postQueue); } function globalModRoutes(app, middleware, controllers) { setupPageRoute(app, '/ip-blacklist', middleware, [], controllers.globalMods.ipBlacklist); } function topicRoutes(app, middleware, controllers) { setupPageRoute(app, '/topic/:topic_id/:slug/:post_index?', middleware, [], controllers.topics.get); setupPageRoute(app, '/topic/:topic_id/:slug?', middleware, [], controllers.topics.get); } function postRoutes(app, middleware, controllers) { setupPageRoute(app, '/post/:pid', middleware, [], controllers.posts.redirectToPost); } function tagRoutes(app, middleware, controllers) { setupPageRoute(app, '/tags/:tag', middleware, [middleware.privateTagListing], controllers.tags.getTag); setupPageRoute(app, '/tags', middleware, [middleware.privateTagListing], controllers.tags.getTags); } function categoryRoutes(app, middleware, controllers) { setupPageRoute(app, '/categories', middleware, [], controllers.categories.list); setupPageRoute(app, '/popular', middleware, [], controllers.popular.get); setupPageRoute(app, '/recent', middleware, [], controllers.recent.get); setupPageRoute(app, '/top', middleware, [], controllers.top.get); setupPageRoute(app, '/unread', middleware, [middleware.authenticate], controllers.unread.get); setupPageRoute(app, '/category/:category_id/:slug/:topic_index', middleware, [], controllers.category.get); setupPageRoute(app, '/category/:category_id/:slug?', middleware, [], controllers.category.get); } function userRoutes(app, middleware, controllers) { var middlewares = [middleware.checkGlobalPrivacySettings]; setupPageRoute(app, '/users', middleware, middlewares, controllers.users.index); } function groupRoutes(app, middleware, controllers) { var middlewares = [middleware.checkGlobalPrivacySettings]; setupPageRoute(app, '/groups', middleware, middlewares, controllers.groups.list); setupPageRoute(app, '/groups/:slug', middleware, middlewares, controllers.groups.details); setupPageRoute(app, '/groups/:slug/members', middleware, middlewares, controllers.groups.members); } module.exports = function (app, middleware, hotswapIds, callback) { var routers = [ express.Router(), // plugin router express.Router(), // main app router express.Router(), // auth router ]; var router = routers[1]; var pluginRouter = routers[0]; var authRouter = routers[2]; var relativePath = nconf.get('relative_path'); var ensureLoggedIn = require('connect-ensure-login'); var idx; var x; if (Array.isArray(hotswapIds) && hotswapIds.length) { for (x = 0; x < hotswapIds.length; x += 1) { idx = routers.push(express.Router()) - 1; routers[idx].hotswapId = hotswapIds[x]; } } pluginRouter.render = function () { app.render.apply(app, arguments); }; // Set-up for hotswapping (when NodeBB reloads) pluginRouter.hotswapId = 'plugins'; authRouter.hotswapId = 'auth'; app.all(relativePath + '(/+api|/+api/*?)', middleware.prepareAPI); app.all(relativePath + '(/+api/admin|/+api/admin/*?)', middleware.isAdmin); app.all(relativePath + '(/+admin|/+admin/*?)', ensureLoggedIn.ensureLoggedIn(nconf.get('relative_path') + '/login?local=1'), middleware.applyCSRF, middleware.isAdmin); app.use(middleware.stripLeadingSlashes); // handle custom homepage routes app.use(relativePath, controllers.home.rewrite); // homepage handled by `action:homepage.get:[route]` setupPageRoute(app, '/', middleware, [], controllers.home.pluginHook); adminRoutes(router, middleware, controllers); metaRoutes(router, middleware, controllers); apiRoutes(router, middleware, controllers); feedRoutes(router, middleware, controllers); mainRoutes(router, middleware, controllers); topicRoutes(router, middleware, controllers); postRoutes(router, middleware, controllers); modRoutes(router, middleware, controllers); globalModRoutes(router, middleware, controllers); tagRoutes(router, middleware, controllers); categoryRoutes(router, middleware, controllers); accountRoutes(router, middleware, controllers); userRoutes(router, middleware, controllers); groupRoutes(router, middleware, controllers); for (x = 0; x < routers.length; x += 1) { app.use(relativePath || '/', routers[x]); } if (process.env.NODE_ENV === 'development') { require('./debug')(app, middleware, controllers); } app.use(middleware.privateUploads); app.use(relativePath + '/assets/templates', middleware.templatesOnDemand); var statics = [ { route: '/assets', path: path.join(__dirname, '../../build/public') }, { route: '/assets', path: path.join(__dirname, '../../public') }, { route: '/plugins', path: path.join(__dirname, '../../build/public/plugins') }, ]; var staticOptions = { maxAge: app.enabled('cache') ? 5184000000 : 0, }; if (path.resolve(__dirname, '../../public/uploads') !== nconf.get('upload_path')) { statics.unshift({ route: '/assets/uploads', path: nconf.get('upload_path') }); } statics.forEach(function (obj) { app.use(relativePath + obj.route, express.static(obj.path, staticOptions)); }); app.use(relativePath + '/uploads', function (req, res) { res.redirect(relativePath + '/assets/uploads' + req.path + '?' + meta.config['cache-buster']); }); // only warn once var warned = new Set(); // DEPRECATED var deprecatedPaths = [ '/nodebb.min.js', '/acp.min.js', '/stylesheet.css', '/js-enabled.css', '/admin.css', '/logo.png', '/favicon.ico', '/vendor/', '/templates/', '/src/', '/images/', '/language/', '/sounds/', ]; app.use(relativePath, function (req, res, next) { if (deprecatedPaths.some(function (path) { return req.path.startsWith(path); })) { if (!warned.has(req.path)) { winston.warn('[deprecated] Accessing `' + req.path.slice(1) + '` from `/` is deprecated to be REMOVED in NodeBB v1.7.0. ' + 'Use `/assets' + req.path + '` to access this file.'); warned.add(req.path); } res.redirect(relativePath + '/assets' + req.path + '?' + meta.config['cache-buster']); } else { next(); } }); // DEPRECATED app.use(relativePath + '/api/language', function (req, res) { if (!warned.has(req.path)) { winston.warn('[deprecated] Accessing language files from `/api/language` is deprecated to be REMOVED in NodeBB v1.7.0. ' + 'Use `/assets/language' + req.path + '.json` for prefetch paths.'); warned.add(req.path); } res.redirect(relativePath + '/assets/language' + req.path + '.json?' + meta.config['cache-buster']); }); app.use(relativePath + '/assets/vendor/jquery/timeago/locales', middleware.processTimeagoLocales); app.use(controllers['404'].handle404); app.use(controllers.errors.handleURIErrors); app.use(controllers.errors.handleErrors); // Add plugin routes async.series([ async.apply(plugins.reloadRoutes), async.apply(authRoutes.reloadRoutes), async.apply(user.addInterstitials), function (next) { winston.info('Routes added'); next(); }, ], function (err) { callback(err); }); };