2022-02-27 11:07:39 +09:00
import * as fs from 'node:fs' ;
import { fileURLToPath } from 'node:url' ;
import { dirname } from 'node:path' ;
import * as os from 'node:os' ;
import cluster from 'node:cluster' ;
import chalk from 'chalk' ;
import chalkTemplate from 'chalk-template' ;
2022-03-20 01:34:45 +09:00
import semver from 'semver' ;
2022-09-18 03:27:08 +09:00
import Logger from '@/logger.js' ;
import { loadConfig } from '@/config.js' ;
import type { Config } from '@/config.js' ;
2022-02-27 11:07:39 +09:00
import { showMachineInfo } from '@/misc/show-machine-info.js' ;
2023-01-23 20:07:48 +09:00
import { envOption } from '@/env.js' ;
import { jobQueue , server } from './common.js' ;
2021-08-20 21:34:56 +09:00
2022-02-27 11:07:39 +09:00
const _filename = fileURLToPath ( import . meta . url ) ;
2021-08-20 21:34:56 +09:00
const _dirname = dirname ( _filename ) ;
2021-11-12 02:02:25 +09:00
const meta = JSON . parse ( fs . readFileSync ( ` ${ _dirname } /../../../../built/meta.json ` , 'utf-8' ) ) ;
2019-04-07 21:50:36 +09:00
const logger = new Logger ( 'core' , 'cyan' ) ;
const bootLogger = logger . createSubLogger ( 'boot' , 'magenta' , false ) ;
2022-08-31 22:53:59 +09:00
const themeColor = chalk . hex ( '#ffa9c3' ) ;
2022-02-27 11:07:39 +09:00
2019-11-24 17:11:53 +09:00
function greet() {
2021-10-08 21:24:05 +09:00
if ( ! envOption . quiet ) {
2022-08-31 22:53:59 +09:00
//#region CherryPick logo
2019-11-24 17:11:53 +09:00
const v = ` v ${ meta . version } ` ;
2022-08-31 22:53:59 +09:00
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' _________ .__ ' ) + chalk . hex ( '#95e3e8' ) . bold ( '__________.__ __ ' ) ) ;
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' \\_ ___ \\| |__ __________________ ___.__.' ) + chalk . hex ( '#95e3e8' ) . bold ( '\\______ \\__| ____ | | __' ) ) ;
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' / \\ \\/| | \\_/ __ \\_ __ \\_ __ < | |' ) + chalk . hex ( '#95e3e8' ) . bold ( ' | ___/ |/ ___\\| |/ /' ) ) ;
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' \\ \\___| Y \\ ___/| | \\/| | \\/\\___ |' ) + chalk . hex ( '#95e3e8' ) . bold ( ' | | | \\ \\___| < ' ) ) ;
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' \\______ /___| /\\___ >__| |__| / ____|' ) + chalk . hex ( '#95e3e8' ) . bold ( ' |____| |__|\\___ >__|_ \\' ) ) ;
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' \\/ \\/ \\/ \\/ ' ) + chalk . hex ( '#95e3e8' ) . bold ( ' \\/ \\/' ) ) ;
2019-04-07 21:50:36 +09:00
//#endregion
2022-08-31 22:53:59 +09:00
console . log ( chalk . hex ( '#ffa9c3' ) . bold ( ' Cherry' ) + chalk . hex ( '#95e3e8' ) . bold ( 'Pick' ) + ( ' is an open-source decentralized microblogging platform based from' ) + ( chalk . hex ( '#9ec23f' ) . bold ( ' Misskey' ) + ( '.' ) ) ) ;
2022-09-15 20:32:25 +09:00
console . log ( chalk . hex ( '#ffbb00' ) ( ' If you like ' ) + chalk . hex ( '#ffa9c3' ) . bold ( 'Cherry' ) + chalk . hex ( '#95e3e8' ) . bold ( 'Pick' ) + chalk . hex ( '#ffbb00' ) ( ', please donate to support development. https://www.patreon.com/noridev & https://www.paypal.me/noridev & https://toss.me/noridev' ) ) ;
2022-08-31 22:53:59 +09:00
// console.log(chalk.hex('#ffa9c3').bold(' KOKO') + chalk.hex('#95e3e8').bold('NECT') + chalk.hex('#ffa9c3')(' with') + chalk.hex('#95e3e8').bold(' NoriDev.'));
2019-04-07 21:50:36 +09:00
console . log ( '' ) ;
2022-02-27 11:07:39 +09:00
console . log ( chalkTemplate ` --- ${ os . hostname ( ) } {gray (PID: ${ process . pid . toString ( ) } )} --- ` ) ;
2019-04-07 21:50:36 +09:00
}
2022-08-31 22:53:59 +09:00
bootLogger . info ( 'Welcome to CherryPick!' ) ;
bootLogger . info ( ` CherryPick v ${ meta . version } ` , null , true ) ;
2019-04-07 21:50:36 +09:00
}
/ * *
* Init master process
* /
export async function masterMain() {
2019-04-13 01:43:22 +09:00
let config ! : Config ;
2019-04-07 21:50:36 +09:00
2021-04-24 22:55:18 +09:00
// initialize app
2019-04-07 21:50:36 +09:00
try {
2019-11-24 17:11:53 +09:00
greet ( ) ;
2021-04-24 22:55:18 +09:00
showEnvironment ( ) ;
await showMachineInfo ( bootLogger ) ;
showNodejsVersion ( ) ;
config = loadConfigBoot ( ) ;
2022-09-18 03:27:08 +09:00
//await connectDb();
2019-04-07 21:50:36 +09:00
} catch ( e ) {
bootLogger . error ( 'Fatal error occurred during initialization' , null , true ) ;
process . exit ( 1 ) ;
}
2023-01-23 20:07:48 +09:00
if ( envOption . onlyServer ) {
await server ( ) ;
} else if ( envOption . onlyQueue ) {
await jobQueue ( ) ;
} else {
await server ( ) ;
}
2023-01-21 14:09:01 +09:00
2022-08-31 22:53:59 +09:00
bootLogger . succ ( chalk . hex ( '#ffa9c3' ) ( 'Cherry' ) + chalk . hex ( '#95e3e8' ) ( 'Pick' ) + ( ' initialized' ) ) ;
2019-04-07 21:50:36 +09:00
2021-10-08 21:24:05 +09:00
if ( ! envOption . disableClustering ) {
2019-04-07 21:50:36 +09:00
await spawnWorkers ( config . clusterLimit ) ;
}
2023-07-17 14:12:02 +09:00
bootLogger . succ ( config . socket ? ` Now listening on socket ${ config . socket } on ${ config . url } ` : ` Now listening on port ${ config . port } on ${ config . url } ` , null , true ) ;
2019-04-07 21:50:36 +09:00
}
function showEnvironment ( ) : void {
const env = process . env . NODE_ENV ;
const logger = bootLogger . createSubLogger ( 'env' ) ;
2020-04-04 08:46:54 +09:00
logger . info ( typeof env === 'undefined' ? 'NODE_ENV is not set' : ` NODE_ENV: ${ env } ` ) ;
2019-04-07 21:50:36 +09:00
if ( env !== 'production' ) {
logger . warn ( 'The environment is not in production mode.' ) ;
logger . warn ( 'DO NOT USE FOR PRODUCTION PURPOSE!' , null , true ) ;
}
}
2021-04-24 22:55:18 +09:00
function showNodejsVersion ( ) : void {
2019-04-07 21:50:36 +09:00
const nodejsLogger = bootLogger . createSubLogger ( 'nodejs' ) ;
2022-03-20 01:34:45 +09:00
nodejsLogger . info ( ` Version ${ process . version } detected. ` ) ;
2021-04-24 22:55:18 +09:00
}
2019-04-07 21:50:36 +09:00
2021-04-24 22:55:18 +09:00
function loadConfigBoot ( ) : Config {
2019-04-07 21:50:36 +09:00
const configLogger = bootLogger . createSubLogger ( 'config' ) ;
let config ;
try {
config = loadConfig ( ) ;
} catch ( exception ) {
if ( typeof exception === 'string' ) {
configLogger . error ( exception ) ;
process . exit ( 1 ) ;
2022-09-24 06:45:44 +09:00
} else if ( ( exception as any ) . code === 'ENOENT' ) {
2019-04-07 21:50:36 +09:00
configLogger . error ( 'Configuration file not found' , null , true ) ;
process . exit ( 1 ) ;
}
throw exception ;
}
configLogger . succ ( 'Loaded' ) ;
2021-04-24 22:55:18 +09:00
return config ;
}
2022-09-18 03:27:08 +09:00
/ *
2021-04-24 22:55:18 +09:00
async function connectDb ( ) : Promise < void > {
2020-04-26 11:39:15 +09:00
const dbLogger = bootLogger . createSubLogger ( 'db' ) ;
2019-04-07 21:50:36 +09:00
// Try to connect to DB
try {
2020-04-26 11:39:15 +09:00
dbLogger . info ( 'Connecting...' ) ;
2019-04-07 21:50:36 +09:00
await initDb ( ) ;
2022-03-26 15:34:00 +09:00
const v = await db . query ( 'SHOW server_version' ) . then ( x = > x [ 0 ] . server_version ) ;
2020-04-26 11:39:15 +09:00
dbLogger . succ ( ` Connected: v ${ v } ` ) ;
2022-09-18 03:27:08 +09:00
} catch ( err ) {
2020-04-26 11:39:15 +09:00
dbLogger . error ( 'Cannot connect' , null , true ) ;
2022-09-18 03:27:08 +09:00
dbLogger . error ( err ) ;
2019-04-07 21:50:36 +09:00
process . exit ( 1 ) ;
}
2021-04-24 22:55:18 +09:00
}
2022-09-18 03:27:08 +09:00
* /
2019-04-07 21:50:36 +09:00
2022-09-18 03:27:08 +09:00
async function spawnWorkers ( limit = 1 ) {
2019-04-07 21:50:36 +09:00
const workers = Math . min ( limit , os . cpus ( ) . length ) ;
bootLogger . info ( ` Starting ${ workers } worker ${ workers === 1 ? '' : 's' } ... ` ) ;
await Promise . all ( [ . . . Array ( workers ) ] . map ( spawnWorker ) ) ;
bootLogger . succ ( 'All workers started' ) ;
}
function spawnWorker ( ) : Promise < void > {
return new Promise ( res = > {
const worker = cluster . fork ( ) ;
worker . on ( 'message' , message = > {
2022-05-19 11:49:07 +09:00
if ( message === 'listenFailed' ) {
2022-09-18 03:27:08 +09:00
bootLogger . error ( 'The server Listen failed due to the previous error.' ) ;
2022-05-19 11:49:07 +09:00
process . exit ( 1 ) ;
}
2019-04-07 21:50:36 +09:00
if ( message !== 'ready' ) return ;
res ( ) ;
} ) ;
} ) ;
}