2015-10-01 00:30:16 +02:00
|
|
|
'use strict';
|
|
|
|
|
2017-07-25 17:12:18 +02:00
|
|
|
const http = require('http');
|
|
|
|
const path = require('path');
|
2019-06-14 18:13:25 +02:00
|
|
|
const fse = require('fs-extra');
|
2018-05-04 17:02:27 +02:00
|
|
|
const Koa = require('koa');
|
2019-08-21 11:05:33 +02:00
|
|
|
const Router = require('koa-router');
|
2019-04-05 16:11:09 +02:00
|
|
|
const _ = require('lodash');
|
2019-10-15 16:21:59 +02:00
|
|
|
const chalk = require('chalk');
|
|
|
|
const CLITable = require('cli-table3');
|
2021-05-20 08:55:18 +02:00
|
|
|
const { models, getAbsoluteAdminUrl, getAbsoluteServerUrl } = require('@strapi/utils');
|
|
|
|
const { createLogger } = require('@strapi/logger');
|
2021-05-10 15:36:09 +02:00
|
|
|
const { Database } = require('@strapi/database');
|
2020-10-27 11:27:17 +01:00
|
|
|
const loadConfiguration = require('./core/app-configuration');
|
2019-10-15 16:21:59 +02:00
|
|
|
|
2018-05-04 17:02:27 +02:00
|
|
|
const utils = require('./utils');
|
2020-01-16 11:01:33 +01:00
|
|
|
const loadModules = require('./core/load-modules');
|
|
|
|
const bootstrap = require('./core/bootstrap');
|
2017-07-24 19:58:03 +02:00
|
|
|
const initializeMiddlewares = require('./middlewares');
|
|
|
|
const initializeHooks = require('./hooks');
|
2019-04-10 18:11:55 +02:00
|
|
|
const createStrapiFs = require('./core/fs');
|
2019-12-17 10:35:04 +01:00
|
|
|
const createEventHub = require('./services/event-hub');
|
2019-12-17 11:24:14 +01:00
|
|
|
const createWebhookRunner = require('./services/webhook-runner');
|
2020-03-02 15:18:08 +01:00
|
|
|
const { webhookModel, createWebhookStore } = require('./services/webhook-store');
|
2019-12-17 11:24:14 +01:00
|
|
|
const { createCoreStore, coreStoreModel } = require('./services/core-store');
|
2020-01-08 11:12:41 +01:00
|
|
|
const createEntityService = require('./services/entity-service');
|
2020-09-22 12:31:26 +02:00
|
|
|
const entityValidator = require('./services/entity-validator');
|
2020-03-27 15:30:03 +01:00
|
|
|
const createTelemetry = require('./services/metrics');
|
2020-11-04 18:26:26 +01:00
|
|
|
const createUpdateNotifier = require('./utils/update-notifier');
|
2020-07-01 09:43:28 +02:00
|
|
|
const ee = require('./utils/ee');
|
2019-12-17 10:35:04 +01:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
const LIFECYCLES = {
|
|
|
|
REGISTER: 'register',
|
2021-02-16 15:34:10 +01:00
|
|
|
BOOTSTRAP: 'bootstrap',
|
2021-02-16 12:23:51 +01:00
|
|
|
};
|
|
|
|
|
2020-03-27 10:30:43 +01:00
|
|
|
class Strapi {
|
2020-01-10 12:25:41 +01:00
|
|
|
constructor(opts = {}) {
|
2017-08-02 13:17:40 +02:00
|
|
|
this.reload = this.reload();
|
|
|
|
|
2016-07-06 15:51:52 +02:00
|
|
|
// Expose `koa`.
|
2016-11-04 16:00:19 +01:00
|
|
|
this.app = new Koa();
|
2019-08-21 11:05:33 +02:00
|
|
|
this.router = new Router();
|
2016-07-06 15:51:52 +02:00
|
|
|
|
2020-09-01 14:59:08 +02:00
|
|
|
this.initServer();
|
2020-07-27 23:26:07 +02:00
|
|
|
|
2017-09-12 17:58:31 +02:00
|
|
|
// Utils.
|
|
|
|
this.utils = {
|
2018-05-18 14:22:24 +02:00
|
|
|
models,
|
2017-09-12 17:58:31 +02:00
|
|
|
};
|
|
|
|
|
2019-12-17 10:35:04 +01:00
|
|
|
this.dir = opts.dir || process.cwd();
|
2020-07-01 09:43:28 +02:00
|
|
|
|
2017-07-28 18:34:13 +02:00
|
|
|
this.admin = {};
|
|
|
|
this.plugins = {};
|
2020-04-03 22:31:20 +02:00
|
|
|
this.config = loadConfiguration(this.dir, opts);
|
2020-08-03 17:34:05 +02:00
|
|
|
this.app.proxy = this.config.get('server.proxy');
|
|
|
|
|
2021-05-20 08:55:18 +02:00
|
|
|
// Logger.
|
|
|
|
const loggerUserConfiguration = this.config.get('logger', {});
|
|
|
|
this.log = createLogger(loggerUserConfiguration);
|
|
|
|
|
2020-05-05 14:21:23 +02:00
|
|
|
this.isLoaded = false;
|
2020-04-03 22:31:20 +02:00
|
|
|
|
2019-12-17 10:35:04 +01:00
|
|
|
// internal services.
|
|
|
|
this.fs = createStrapiFs(this);
|
|
|
|
this.eventHub = createEventHub();
|
|
|
|
|
|
|
|
this.requireProjectBootstrap();
|
2020-10-30 17:16:09 +01:00
|
|
|
|
2020-11-04 18:26:26 +01:00
|
|
|
createUpdateNotifier(this).notify();
|
2019-12-17 10:35:04 +01:00
|
|
|
}
|
2017-07-28 18:34:13 +02:00
|
|
|
|
2020-07-03 17:59:06 +02:00
|
|
|
get EE() {
|
2021-05-20 08:55:18 +02:00
|
|
|
return ee({ dir: this.dir, logger: this.log });
|
2020-07-03 17:59:06 +02:00
|
|
|
}
|
|
|
|
|
2020-07-27 23:26:07 +02:00
|
|
|
handleRequest(req, res) {
|
|
|
|
if (!this.requestHandler) {
|
|
|
|
this.requestHandler = this.app.callback();
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.requestHandler(req, res);
|
|
|
|
}
|
|
|
|
|
2019-06-14 18:13:25 +02:00
|
|
|
requireProjectBootstrap() {
|
2020-03-02 15:18:08 +01:00
|
|
|
const bootstrapPath = path.resolve(this.dir, 'config/functions/bootstrap.js');
|
2019-06-14 18:13:25 +02:00
|
|
|
|
|
|
|
if (fse.existsSync(bootstrapPath)) {
|
|
|
|
require(bootstrapPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-15 16:21:59 +02:00
|
|
|
logStats() {
|
|
|
|
const columns = Math.min(process.stderr.columns, 80) - 2;
|
|
|
|
console.log();
|
|
|
|
console.log(chalk.black.bgWhite(_.padEnd(' Project information', columns)));
|
|
|
|
console.log();
|
|
|
|
|
|
|
|
const infoTable = new CLITable({
|
|
|
|
colWidths: [20, 50],
|
|
|
|
chars: { mid: '', 'left-mid': '', 'mid-mid': '', 'right-mid': '' },
|
|
|
|
});
|
|
|
|
|
2020-07-08 09:10:37 +02:00
|
|
|
const isEE = strapi.EE === true && ee.isEE === true;
|
|
|
|
|
2019-10-15 16:21:59 +02:00
|
|
|
infoTable.push(
|
|
|
|
[chalk.blue('Time'), `${new Date()}`],
|
|
|
|
[chalk.blue('Launched in'), Date.now() - this.config.launchedAt + ' ms'],
|
|
|
|
[chalk.blue('Environment'), this.config.environment],
|
|
|
|
[chalk.blue('Process PID'), process.pid],
|
2020-07-08 09:10:37 +02:00
|
|
|
[chalk.blue('Version'), `${this.config.info.strapi} (node ${process.version})`],
|
|
|
|
[chalk.blue('Edition'), isEE ? 'Enterprise' : 'Community']
|
2019-10-15 16:21:59 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
console.log(infoTable.toString());
|
|
|
|
console.log();
|
|
|
|
console.log(chalk.black.bgWhite(_.padEnd(' Actions available', columns)));
|
|
|
|
console.log();
|
|
|
|
}
|
|
|
|
|
|
|
|
logFirstStartupMessage() {
|
|
|
|
this.logStats();
|
|
|
|
|
|
|
|
console.log(chalk.bold('One more thing...'));
|
|
|
|
console.log(
|
2020-03-02 15:18:08 +01:00
|
|
|
chalk.grey('Create your first administrator 💻 by going to the administration panel at:')
|
2019-10-15 16:21:59 +02:00
|
|
|
);
|
|
|
|
console.log();
|
|
|
|
|
|
|
|
const addressTable = new CLITable();
|
2020-05-28 11:17:59 +02:00
|
|
|
|
|
|
|
const adminUrl = getAbsoluteAdminUrl(strapi.config);
|
|
|
|
addressTable.push([chalk.bold(adminUrl)]);
|
|
|
|
|
2019-10-15 16:21:59 +02:00
|
|
|
console.log(`${addressTable.toString()}`);
|
|
|
|
console.log();
|
|
|
|
}
|
|
|
|
|
|
|
|
logStartupMessage() {
|
|
|
|
this.logStats();
|
|
|
|
|
|
|
|
console.log(chalk.bold('Welcome back!'));
|
|
|
|
|
|
|
|
if (this.config.serveAdminPanel === true) {
|
2020-03-02 15:18:08 +01:00
|
|
|
console.log(chalk.grey('To manage your project 🚀, go to the administration panel at:'));
|
2020-05-28 11:17:59 +02:00
|
|
|
const adminUrl = getAbsoluteAdminUrl(strapi.config);
|
|
|
|
console.log(chalk.bold(adminUrl));
|
2019-10-15 16:21:59 +02:00
|
|
|
console.log();
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log(chalk.grey('To access the server ⚡️, go to:'));
|
2020-05-28 11:17:59 +02:00
|
|
|
const serverUrl = getAbsoluteServerUrl(strapi.config);
|
|
|
|
console.log(chalk.bold(serverUrl));
|
2019-10-15 16:21:59 +02:00
|
|
|
console.log();
|
|
|
|
}
|
|
|
|
|
2020-09-01 14:59:08 +02:00
|
|
|
initServer() {
|
|
|
|
this.server = http.createServer(this.handleRequest.bind(this));
|
2019-04-05 16:11:09 +02:00
|
|
|
// handle port in use cleanly
|
|
|
|
this.server.on('error', err => {
|
|
|
|
if (err.code === 'EADDRINUSE') {
|
2020-03-02 15:18:08 +01:00
|
|
|
return this.stopWithError(`The port ${err.port} is already used by another application.`);
|
2019-04-05 16:11:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
this.log.error(err);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Close current connections to fully destroy the server
|
2017-07-25 17:12:18 +02:00
|
|
|
const connections = {};
|
2017-07-28 16:14:50 +02:00
|
|
|
|
2017-07-25 17:12:18 +02:00
|
|
|
this.server.on('connection', conn => {
|
|
|
|
const key = conn.remoteAddress + ':' + conn.remotePort;
|
|
|
|
connections[key] = conn;
|
2016-07-26 11:57:50 +02:00
|
|
|
|
2019-04-09 12:09:03 +02:00
|
|
|
conn.on('close', function() {
|
2018-05-04 17:02:27 +02:00
|
|
|
delete connections[key];
|
2017-07-25 17:12:18 +02:00
|
|
|
});
|
|
|
|
});
|
2016-07-26 11:57:50 +02:00
|
|
|
|
2017-07-25 17:12:18 +02:00
|
|
|
this.server.destroy = cb => {
|
|
|
|
this.server.close(cb);
|
2016-07-26 11:57:50 +02:00
|
|
|
|
2017-07-25 17:12:18 +02:00
|
|
|
for (let key in connections) {
|
|
|
|
connections[key].destroy();
|
2018-05-04 17:02:27 +02:00
|
|
|
}
|
2017-07-25 17:12:18 +02:00
|
|
|
};
|
2020-09-01 14:59:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async start(cb) {
|
|
|
|
try {
|
|
|
|
if (!this.isLoaded) {
|
|
|
|
await this.load();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.app.use(this.router.routes()).use(this.router.allowedMethods());
|
|
|
|
|
|
|
|
// Launch server.
|
|
|
|
this.listen(cb);
|
|
|
|
} catch (err) {
|
|
|
|
this.stopWithError(err);
|
|
|
|
}
|
|
|
|
}
|
2020-03-14 17:35:25 +01:00
|
|
|
|
2020-11-10 14:15:31 +01:00
|
|
|
async destroy() {
|
2020-11-27 15:58:50 +01:00
|
|
|
if (_.has(this, 'server.destroy')) {
|
2021-03-29 12:33:08 +02:00
|
|
|
await new Promise(res => this.server.destroy(res));
|
2020-11-27 15:58:50 +01:00
|
|
|
}
|
2020-11-25 17:22:41 +01:00
|
|
|
|
2020-12-29 17:44:14 +01:00
|
|
|
await Promise.all(
|
|
|
|
Object.values(this.plugins).map(plugin => {
|
|
|
|
if (_.has(plugin, 'destroy') && typeof plugin.destroy === 'function') {
|
|
|
|
return plugin.destroy();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
);
|
2020-11-25 17:22:41 +01:00
|
|
|
|
2020-11-25 18:27:01 +01:00
|
|
|
if (_.has(this, 'admin')) {
|
|
|
|
await this.admin.destroy();
|
|
|
|
}
|
|
|
|
|
2020-11-25 17:22:41 +01:00
|
|
|
this.eventHub.removeAllListeners();
|
|
|
|
|
2020-11-30 14:23:03 +01:00
|
|
|
if (_.has(this, 'db')) {
|
|
|
|
await this.db.destroy();
|
|
|
|
}
|
|
|
|
|
2021-03-29 12:33:08 +02:00
|
|
|
this.telemetry.destroy();
|
|
|
|
|
2020-11-25 17:22:41 +01:00
|
|
|
delete global.strapi;
|
2020-11-10 14:15:31 +01:00
|
|
|
}
|
|
|
|
|
2020-09-01 14:59:08 +02:00
|
|
|
/**
|
|
|
|
* Add behaviors to the server
|
|
|
|
*/
|
|
|
|
async listen(cb) {
|
2020-03-14 17:35:25 +01:00
|
|
|
const onListen = async err => {
|
|
|
|
if (err) return this.stopWithError(err);
|
|
|
|
|
|
|
|
// Is the project initialised?
|
|
|
|
const isInitialised = await utils.isInitialised(this);
|
|
|
|
|
2020-07-16 09:01:08 +02:00
|
|
|
// Should the startup message be displayed?
|
|
|
|
const hideStartupMessage = process.env.STRAPI_HIDE_STARTUP_MESSAGE
|
|
|
|
? process.env.STRAPI_HIDE_STARTUP_MESSAGE === 'true'
|
|
|
|
: false;
|
|
|
|
|
|
|
|
if (hideStartupMessage === false) {
|
|
|
|
if (!isInitialised) {
|
|
|
|
this.logFirstStartupMessage();
|
|
|
|
} else {
|
|
|
|
this.logStartupMessage();
|
|
|
|
}
|
2020-03-14 17:35:25 +01:00
|
|
|
}
|
|
|
|
|
2021-04-19 14:44:35 +02:00
|
|
|
// Get database clients
|
2021-01-12 18:20:10 +01:00
|
|
|
const databaseClients = _.map(this.config.get('connections'), _.property('settings.client'));
|
2021-04-19 14:44:35 +02:00
|
|
|
|
|
|
|
// Emit started event.
|
|
|
|
await this.telemetry.send('didStartServer', {
|
|
|
|
database: databaseClients,
|
|
|
|
plugins: this.config.installedPlugins,
|
|
|
|
providers: this.config.installedProviders,
|
|
|
|
});
|
2020-03-14 17:35:25 +01:00
|
|
|
|
|
|
|
if (cb && typeof cb === 'function') {
|
|
|
|
cb();
|
|
|
|
}
|
|
|
|
|
2021-06-28 12:34:29 +02:00
|
|
|
// if (
|
|
|
|
// (this.config.environment === 'development' &&
|
|
|
|
// this.config.get('server.admin.autoOpen', true) !== false) ||
|
|
|
|
// !isInitialised
|
|
|
|
// ) {
|
|
|
|
// await utils.openBrowser.call(this);
|
|
|
|
// }
|
2020-03-14 17:35:25 +01:00
|
|
|
};
|
|
|
|
|
2020-09-23 05:59:31 -04:00
|
|
|
const listenSocket = this.config.get('server.socket');
|
|
|
|
const listenErrHandler = err => onListen(err).catch(err => this.stopWithError(err));
|
|
|
|
|
|
|
|
if (listenSocket) {
|
|
|
|
this.server.listen(listenSocket, listenErrHandler);
|
|
|
|
} else {
|
2020-10-01 16:23:32 +02:00
|
|
|
this.server.listen(
|
|
|
|
this.config.get('server.port'),
|
|
|
|
this.config.get('server.host'),
|
|
|
|
listenErrHandler
|
|
|
|
);
|
2020-09-23 05:59:31 -04:00
|
|
|
}
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
|
|
|
|
2020-03-12 16:05:39 +01:00
|
|
|
stopWithError(err, customMessage) {
|
2021-06-17 19:51:35 +02:00
|
|
|
console.log(err);
|
2019-04-05 16:11:09 +02:00
|
|
|
this.log.debug(`⛔️ Server wasn't able to start properly.`);
|
2020-03-12 16:05:39 +01:00
|
|
|
if (customMessage) {
|
|
|
|
this.log.error(customMessage);
|
|
|
|
}
|
2021-06-17 16:17:15 +02:00
|
|
|
|
2019-04-05 16:11:09 +02:00
|
|
|
this.log.error(err);
|
|
|
|
return this.stop();
|
|
|
|
}
|
|
|
|
|
2019-12-28 17:09:00 +01:00
|
|
|
stop(exitCode = 1) {
|
2017-07-25 17:12:18 +02:00
|
|
|
// Destroy server and available connections.
|
2020-03-19 16:46:27 +01:00
|
|
|
if (_.has(this, 'server.destroy')) {
|
|
|
|
this.server.destroy();
|
|
|
|
}
|
2017-09-04 15:38:29 +02:00
|
|
|
|
2019-05-13 17:32:52 +02:00
|
|
|
if (this.config.autoReload) {
|
2017-10-02 14:17:44 +02:00
|
|
|
process.send('stop');
|
|
|
|
}
|
2017-09-04 15:38:29 +02:00
|
|
|
|
2021-03-29 12:33:08 +02:00
|
|
|
// Kill process
|
2021-07-05 18:35:16 +02:00
|
|
|
// process.exit(exitCode);
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
|
|
|
|
2017-07-24 19:58:03 +02:00
|
|
|
async load() {
|
2017-10-02 14:17:44 +02:00
|
|
|
this.app.use(async (ctx, next) => {
|
2020-10-08 00:33:32 -07:00
|
|
|
if (ctx.request.url === '/_health' && ['HEAD', 'GET'].includes(ctx.request.method)) {
|
2018-03-14 13:18:05 +01:00
|
|
|
ctx.set('strapi', 'You are so French!');
|
2017-10-02 14:17:44 +02:00
|
|
|
ctx.status = 204;
|
2017-09-06 11:06:18 +02:00
|
|
|
} else {
|
|
|
|
await next();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2020-01-16 11:01:33 +01:00
|
|
|
const modules = await loadModules(this);
|
|
|
|
|
|
|
|
this.api = modules.api;
|
|
|
|
this.admin = modules.admin;
|
|
|
|
this.components = modules.components;
|
|
|
|
this.plugins = modules.plugins;
|
|
|
|
this.middleware = modules.middlewares;
|
|
|
|
this.hook = modules.hook;
|
2019-05-02 17:51:58 +02:00
|
|
|
|
2019-04-11 09:32:16 +02:00
|
|
|
await bootstrap(this);
|
2019-05-02 17:51:58 +02:00
|
|
|
|
2020-01-10 12:42:57 +01:00
|
|
|
// init webhook runner
|
|
|
|
this.webhookRunner = createWebhookRunner({
|
|
|
|
eventHub: this.eventHub,
|
|
|
|
logger: this.log,
|
2020-04-03 22:31:20 +02:00
|
|
|
configuration: this.config.get('server.webhooks', {}),
|
2020-01-10 12:42:57 +01:00
|
|
|
});
|
|
|
|
|
2018-04-24 12:30:43 +02:00
|
|
|
// Init core store
|
2019-05-02 17:51:58 +02:00
|
|
|
|
2021-06-17 16:17:15 +02:00
|
|
|
const contentTypes = [
|
2021-06-22 17:13:11 +02:00
|
|
|
// todo: move corestore and webhook to real models instead of content types to avoid adding extra attributes
|
2021-05-17 16:34:19 +02:00
|
|
|
coreStoreModel,
|
|
|
|
webhookModel,
|
|
|
|
...Object.values(strapi.models),
|
|
|
|
...Object.values(strapi.components),
|
|
|
|
...Object.values(strapi.admin.models),
|
|
|
|
...Object.values(strapi.plugins).flatMap(plugin => Object.values(plugin.models)),
|
|
|
|
...Object.values(strapi.api).flatMap(api => Object.values(api.models)),
|
|
|
|
];
|
|
|
|
|
2021-06-28 12:34:29 +02:00
|
|
|
// TODO: create in RootProvider
|
|
|
|
this.db = await Database.init({
|
2021-05-17 16:34:19 +02:00
|
|
|
...this.config.get('database'),
|
2021-06-17 16:17:15 +02:00
|
|
|
models: Database.transformContentTypes(contentTypes),
|
2021-05-17 16:34:19 +02:00
|
|
|
});
|
2021-02-12 12:52:29 +01:00
|
|
|
|
2021-07-05 18:35:16 +02:00
|
|
|
await this.db.schema.sync();
|
2021-05-17 16:34:19 +02:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
await this.runLifecyclesFunctions(LIFECYCLES.REGISTER);
|
2021-06-17 16:17:15 +02:00
|
|
|
// await this.db.initialize();
|
2019-09-20 12:44:24 +02:00
|
|
|
|
2019-12-17 11:24:14 +01:00
|
|
|
this.store = createCoreStore({
|
|
|
|
environment: this.config.environment,
|
|
|
|
db: this.db,
|
|
|
|
});
|
|
|
|
|
|
|
|
this.webhookStore = createWebhookStore({ db: this.db });
|
|
|
|
|
2019-12-17 20:59:57 +01:00
|
|
|
await this.startWebhooks();
|
|
|
|
|
2020-09-22 12:31:26 +02:00
|
|
|
this.entityValidator = entityValidator;
|
2020-02-17 11:31:34 +01:00
|
|
|
|
2020-01-08 11:12:41 +01:00
|
|
|
this.entityService = createEntityService({
|
|
|
|
db: this.db,
|
|
|
|
eventHub: this.eventHub,
|
2020-02-17 11:31:34 +01:00
|
|
|
entityValidator: this.entityValidator,
|
2020-01-08 11:12:41 +01:00
|
|
|
});
|
|
|
|
|
2020-03-27 15:30:03 +01:00
|
|
|
this.telemetry = createTelemetry(this);
|
2020-03-27 10:30:43 +01:00
|
|
|
|
2017-07-25 17:12:18 +02:00
|
|
|
// Initialize hooks and middlewares.
|
2019-08-21 11:05:33 +02:00
|
|
|
await initializeMiddlewares.call(this);
|
|
|
|
await initializeHooks.call(this);
|
2020-05-05 14:21:23 +02:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
await this.runLifecyclesFunctions(LIFECYCLES.BOOTSTRAP);
|
2020-05-05 14:21:23 +02:00
|
|
|
await this.freeze();
|
2020-05-12 12:47:23 +02:00
|
|
|
|
|
|
|
this.isLoaded = true;
|
2020-05-07 18:49:44 +02:00
|
|
|
return this;
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
|
|
|
|
2019-12-17 20:59:57 +01:00
|
|
|
async startWebhooks() {
|
|
|
|
const webhooks = await this.webhookStore.findWebhooks();
|
2019-12-18 16:10:42 +01:00
|
|
|
webhooks.forEach(webhook => this.webhookRunner.add(webhook));
|
2019-12-17 20:59:57 +01:00
|
|
|
}
|
|
|
|
|
2017-08-02 11:25:18 +02:00
|
|
|
reload() {
|
2018-03-28 20:13:09 +02:00
|
|
|
const state = {
|
2019-04-09 12:09:03 +02:00
|
|
|
shouldReload: 0,
|
2018-03-28 20:13:09 +02:00
|
|
|
};
|
|
|
|
|
2019-04-09 12:09:03 +02:00
|
|
|
const reload = function() {
|
2018-08-31 13:47:10 +02:00
|
|
|
if (state.shouldReload > 0) {
|
2018-07-15 15:51:30 +02:00
|
|
|
// Reset the reloading state
|
2018-08-31 13:47:10 +02:00
|
|
|
state.shouldReload -= 1;
|
2018-07-15 15:51:30 +02:00
|
|
|
reload.isReloading = false;
|
2018-03-28 20:13:09 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-05-13 17:32:52 +02:00
|
|
|
if (this.config.autoReload) {
|
2019-03-11 10:42:43 +01:00
|
|
|
this.server.close();
|
2018-01-04 16:03:34 +01:00
|
|
|
process.send('reload');
|
|
|
|
}
|
2017-08-02 11:25:18 +02:00
|
|
|
};
|
|
|
|
|
2018-03-28 20:13:09 +02:00
|
|
|
Object.defineProperty(reload, 'isWatching', {
|
|
|
|
configurable: true,
|
|
|
|
enumerable: true,
|
2018-05-18 14:22:24 +02:00
|
|
|
set: value => {
|
2018-03-28 20:13:09 +02:00
|
|
|
// Special state when the reloader is disabled temporarly (see GraphQL plugin example).
|
2018-08-31 13:47:10 +02:00
|
|
|
if (state.isWatching === false && value === true) {
|
|
|
|
state.shouldReload += 1;
|
|
|
|
}
|
2018-03-28 20:13:09 +02:00
|
|
|
state.isWatching = value;
|
2018-05-02 22:16:13 +02:00
|
|
|
},
|
|
|
|
get: () => {
|
|
|
|
return state.isWatching;
|
2018-05-18 14:22:24 +02:00
|
|
|
},
|
2018-03-28 20:13:09 +02:00
|
|
|
});
|
|
|
|
|
2017-08-02 11:25:18 +02:00
|
|
|
reload.isReloading = false;
|
|
|
|
reload.isWatching = true;
|
|
|
|
|
|
|
|
return reload;
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
async runLifecyclesFunctions(lifecycleName) {
|
|
|
|
const execLifecycle = async fn => {
|
|
|
|
if (!fn) {
|
|
|
|
return;
|
|
|
|
}
|
2019-08-12 15:35:40 +02:00
|
|
|
|
2020-07-01 19:11:04 +02:00
|
|
|
return fn();
|
2020-02-26 18:34:45 +01:00
|
|
|
};
|
2019-08-12 15:35:40 +02:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
const configPath = `functions.${lifecycleName}`;
|
2021-02-15 11:38:50 +01:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
// plugins
|
|
|
|
await Promise.all(
|
|
|
|
Object.keys(this.plugins).map(plugin => {
|
|
|
|
const pluginFunc = _.get(this.plugins[plugin], `config.${configPath}`);
|
|
|
|
|
|
|
|
return execLifecycle(pluginFunc).catch(err => {
|
|
|
|
strapi.log.error(`${lifecycleName} function in plugin "${plugin}" failed`);
|
2021-06-17 19:51:35 +02:00
|
|
|
console.error(err);
|
2021-02-16 12:23:51 +01:00
|
|
|
strapi.stop();
|
|
|
|
});
|
|
|
|
})
|
|
|
|
);
|
2019-04-05 16:11:09 +02:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
// user
|
|
|
|
await execLifecycle(_.get(this.config, configPath));
|
2020-06-08 15:13:26 +02:00
|
|
|
|
2021-02-16 12:23:51 +01:00
|
|
|
// admin
|
2021-02-15 11:38:50 +01:00
|
|
|
const adminFunc = _.get(this.admin.config, configPath);
|
2021-02-16 12:23:51 +01:00
|
|
|
return execLifecycle(adminFunc).catch(err => {
|
|
|
|
strapi.log.error(`${lifecycleName} function in admin failed`);
|
2021-06-17 19:51:35 +02:00
|
|
|
console.error(err);
|
2020-11-27 15:58:50 +01:00
|
|
|
strapi.stop();
|
2020-06-08 15:13:26 +02:00
|
|
|
});
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
|
|
|
|
2017-07-31 11:35:57 +02:00
|
|
|
async freeze() {
|
2019-12-18 16:10:42 +01:00
|
|
|
Object.freeze(this.config);
|
|
|
|
Object.freeze(this.dir);
|
|
|
|
Object.freeze(this.admin);
|
|
|
|
Object.freeze(this.plugins);
|
|
|
|
Object.freeze(this.api);
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
2017-08-29 19:34:34 +02:00
|
|
|
|
2021-06-28 22:37:19 +02:00
|
|
|
getModel(uid) {
|
2021-07-01 13:52:01 +02:00
|
|
|
return this.contentTypes[uid] || this.components[uid];
|
2019-07-22 12:15:54 +02:00
|
|
|
}
|
|
|
|
|
2019-05-16 21:37:45 +02:00
|
|
|
/**
|
|
|
|
* Binds queries with a specific model
|
2021-06-30 20:00:03 +02:00
|
|
|
* @param {string} uid
|
|
|
|
* @returns {}
|
2019-05-16 21:37:45 +02:00
|
|
|
*/
|
2021-06-30 20:00:03 +02:00
|
|
|
query(uid) {
|
|
|
|
return this.db.query(uid);
|
2017-08-29 19:34:34 +02:00
|
|
|
}
|
2016-07-26 11:57:50 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 09:32:16 +02:00
|
|
|
module.exports = options => {
|
|
|
|
const strapi = new Strapi(options);
|
|
|
|
global.strapi = strapi;
|
|
|
|
return strapi;
|
|
|
|
};
|
2021-06-30 20:00:03 +02:00
|
|
|
|
|
|
|
module.exports.Strapi = Strapi;
|