mirror of
https://github.com/leon-ai/leon.git
synced 2024-11-09 11:44:54 +03:00
605 lines
18 KiB
JavaScript
605 lines
18 KiB
JavaScript
import fs from 'node:fs'
|
|
import path from 'node:path'
|
|
import os from 'node:os'
|
|
import { spawn } from 'node:child_process'
|
|
|
|
import dotenv from 'dotenv'
|
|
import { command } from 'execa'
|
|
import semver from 'semver'
|
|
import kill from 'tree-kill'
|
|
import axios from 'axios'
|
|
import osName from 'os-name'
|
|
import getos from 'getos'
|
|
|
|
import { LogHelper } from '@/helpers/log-helper'
|
|
import { SystemHelper } from '@/helpers/system-helper'
|
|
import { shouldIgnoreTCPServerError } from '@/utilities'
|
|
import {
|
|
MINIMUM_REQUIRED_RAM,
|
|
LEON_VERSION,
|
|
NODEJS_BRIDGE_BIN_PATH,
|
|
PYTHON_BRIDGE_BIN_PATH,
|
|
PYTHON_TCP_SERVER_BIN_PATH,
|
|
PYTHON_TCP_SERVER_VERSION,
|
|
NODEJS_BRIDGE_VERSION,
|
|
PYTHON_BRIDGE_VERSION,
|
|
INSTANCE_ID,
|
|
SKILLS_RESOLVERS_NLP_MODEL_PATH,
|
|
GLOBAL_RESOLVERS_NLP_MODEL_PATH,
|
|
MAIN_NLP_MODEL_PATH
|
|
} from '@/constants'
|
|
|
|
dotenv.config()
|
|
|
|
/**
|
|
* Checking script
|
|
* Help to figure out the setup state
|
|
*/
|
|
;(async () => {
|
|
try {
|
|
const nodeMinRequiredVersion = '16'
|
|
const npmMinRequiredVersion = '8'
|
|
const flitePath = 'bin/flite/flite'
|
|
const coquiLanguageModelPath = 'bin/coqui/huge-vocabulary.scorer'
|
|
const amazonPath = 'core/config/voice/amazon.json'
|
|
const googleCloudPath = 'core/config/voice/google-cloud.json'
|
|
const watsonSttPath = 'core/config/voice/watson-stt.json'
|
|
const watsonTtsPath = 'core/config/voice/watson-tts.json'
|
|
const report = {
|
|
can_run: { title: 'Run', type: 'error', v: true },
|
|
can_run_skill: { title: 'Run skills', type: 'error', v: true },
|
|
can_text: { title: 'Reply you by texting', type: 'error', v: true },
|
|
can_start_python_tcp_server: {
|
|
title: 'Start the Python TCP server',
|
|
type: 'error',
|
|
v: true
|
|
},
|
|
can_amazon_polly_tts: {
|
|
title: 'Amazon Polly text-to-speech',
|
|
type: 'warning',
|
|
v: true
|
|
},
|
|
can_google_cloud_tts: {
|
|
title: 'Google Cloud text-to-speech',
|
|
type: 'warning',
|
|
v: true
|
|
},
|
|
can_watson_tts: {
|
|
title: 'Watson text-to-speech',
|
|
type: 'warning',
|
|
v: true
|
|
},
|
|
can_offline_tts: {
|
|
title: 'Offline text-to-speech',
|
|
type: 'warning',
|
|
v: true
|
|
},
|
|
can_google_cloud_stt: {
|
|
title: 'Google Cloud speech-to-text',
|
|
type: 'warning',
|
|
v: true
|
|
},
|
|
can_watson_stt: {
|
|
title: 'Watson speech-to-text',
|
|
type: 'warning',
|
|
v: true
|
|
},
|
|
can_offline_stt: {
|
|
title: 'Offline speech-to-text',
|
|
type: 'warning',
|
|
v: true
|
|
}
|
|
}
|
|
let reportDataInput = {
|
|
leonVersion: null,
|
|
instanceID: INSTANCE_ID || null,
|
|
environment: {
|
|
osDetails: null,
|
|
nodeVersion: null,
|
|
npmVersion: null
|
|
},
|
|
nlpModels: {
|
|
globalResolversModelState: null,
|
|
skillsResolversModelState: null,
|
|
mainModelState: null
|
|
},
|
|
nodeJSBridge: {
|
|
version: null,
|
|
executionTime: null,
|
|
command: null,
|
|
output: null,
|
|
error: null
|
|
},
|
|
pythonBridge: {
|
|
version: null,
|
|
executionTime: null,
|
|
command: null,
|
|
output: null,
|
|
error: null
|
|
},
|
|
pythonTCPServer: {
|
|
version: null,
|
|
startTime: null,
|
|
command: null,
|
|
output: null,
|
|
error: null
|
|
},
|
|
report: null
|
|
}
|
|
|
|
LogHelper.title('Checking')
|
|
|
|
/**
|
|
* Leon version checking
|
|
*/
|
|
|
|
LogHelper.info('Leon version')
|
|
LogHelper.success(`${LEON_VERSION}\n`)
|
|
reportDataInput.leonVersion = LEON_VERSION
|
|
|
|
/**
|
|
* Environment checking
|
|
*/
|
|
|
|
LogHelper.info('Environment')
|
|
|
|
const osInfo = {
|
|
type: os.type(),
|
|
platform: os.platform(),
|
|
arch: os.arch(),
|
|
cpus: os.cpus().length,
|
|
release: os.release(),
|
|
osName: osName(),
|
|
distro: null
|
|
}
|
|
const totalRAMInGB = SystemHelper.getTotalRAM()
|
|
const freeRAMInGB = SystemHelper.getFreeRAM()
|
|
|
|
if (Math.round(freeRAMInGB) < MINIMUM_REQUIRED_RAM) {
|
|
report.can_run.v = false
|
|
LogHelper.error(
|
|
`Free RAM: ${freeRAMInGB} GB | Total RAM: ${totalRAMInGB} GB. Leon needs at least ${MINIMUM_REQUIRED_RAM} GB of RAM`
|
|
)
|
|
} else {
|
|
LogHelper.success(
|
|
`Free RAM: ${freeRAMInGB} GB | Total RAM: ${totalRAMInGB} GB`
|
|
)
|
|
}
|
|
|
|
if (osInfo.platform === 'linux') {
|
|
getos((e, os) => {
|
|
osInfo.distro = os
|
|
LogHelper.success(`${JSON.stringify(osInfo)}\n`)
|
|
})
|
|
} else {
|
|
LogHelper.success(`${JSON.stringify(osInfo)}\n`)
|
|
}
|
|
|
|
reportDataInput.environment.osDetails = osInfo
|
|
reportDataInput.environment.totalRAMInGB = totalRAMInGB
|
|
reportDataInput.environment.freeRAMInGB = freeRAMInGB
|
|
;(
|
|
await Promise.all([
|
|
command('node --version', { shell: true }),
|
|
command('npm --version', { shell: true })
|
|
])
|
|
).forEach((p) => {
|
|
LogHelper.info(p.command)
|
|
|
|
if (
|
|
p.command.indexOf('node --version') !== -1 &&
|
|
!semver.satisfies(semver.clean(p.stdout), `>=${nodeMinRequiredVersion}`)
|
|
) {
|
|
Object.keys(report).forEach((item) => {
|
|
if (report[item].type === 'error') report[item].v = false
|
|
})
|
|
LogHelper.error(
|
|
`${p.stdout}\nThe Node.js version must be >=${nodeMinRequiredVersion}. Please install it: https://nodejs.org (or use nvm)\n`
|
|
)
|
|
} else if (
|
|
p.command.indexOf('npm --version') !== -1 &&
|
|
!semver.satisfies(semver.clean(p.stdout), `>=${npmMinRequiredVersion}`)
|
|
) {
|
|
Object.keys(report).forEach((item) => {
|
|
if (report[item].type === 'error') report[item].v = false
|
|
})
|
|
LogHelper.error(
|
|
`${p.stdout}\nThe npm version must be >=${npmMinRequiredVersion}. Please install it: https://www.npmjs.com/get-npm (or use nvm)\n`
|
|
)
|
|
} else {
|
|
LogHelper.success(`${p.stdout}\n`)
|
|
if (p.command.includes('node --version')) {
|
|
reportDataInput.environment.nodeVersion = p.stdout
|
|
} else if (p.command.includes('npm --version')) {
|
|
reportDataInput.environment.npmVersion = p.stdout
|
|
}
|
|
}
|
|
})
|
|
|
|
/**
|
|
* Skill execution checking with Node.js bridge
|
|
*/
|
|
|
|
LogHelper.success(`Node.js bridge version: ${NODEJS_BRIDGE_VERSION}`)
|
|
reportDataInput.nodeJSBridge.version = NODEJS_BRIDGE_VERSION
|
|
LogHelper.info('Executing a skill...')
|
|
|
|
try {
|
|
const executionStart = Date.now()
|
|
const p = await command(
|
|
`${NODEJS_BRIDGE_BIN_PATH} "${path.join(
|
|
process.cwd(),
|
|
'scripts',
|
|
'assets',
|
|
'nodejs-bridge-intent-object.json'
|
|
)}"`,
|
|
{ shell: true }
|
|
)
|
|
const executionEnd = Date.now()
|
|
const executionTime = executionEnd - executionStart
|
|
LogHelper.info(p.command)
|
|
reportDataInput.nodeJSBridge.command = p.command
|
|
LogHelper.success(p.stdout)
|
|
reportDataInput.nodeJSBridge.output = p.stdout
|
|
LogHelper.info(`Skill execution time: ${executionTime}ms\n`)
|
|
reportDataInput.nodeJSBridge.executionTime = `${executionTime}ms`
|
|
} catch (e) {
|
|
LogHelper.info(e.command)
|
|
report.can_run_skill.v = false
|
|
LogHelper.error(`${e}\n`)
|
|
reportDataInput.nodeJSBridge.error = JSON.stringify(e)
|
|
}
|
|
|
|
/**
|
|
* Skill execution checking with Python bridge
|
|
*/
|
|
|
|
LogHelper.success(`Python bridge version: ${PYTHON_BRIDGE_VERSION}`)
|
|
reportDataInput.pythonBridge.version = PYTHON_BRIDGE_VERSION
|
|
LogHelper.info('Executing a skill...')
|
|
|
|
try {
|
|
const executionStart = Date.now()
|
|
const p = await command(
|
|
`${PYTHON_BRIDGE_BIN_PATH} "${path.join(
|
|
process.cwd(),
|
|
'scripts',
|
|
'assets',
|
|
'python-bridge-intent-object.json'
|
|
)}"`,
|
|
{ shell: true }
|
|
)
|
|
const executionEnd = Date.now()
|
|
const executionTime = executionEnd - executionStart
|
|
LogHelper.info(p.command)
|
|
reportDataInput.pythonBridge.command = p.command
|
|
LogHelper.success(p.stdout)
|
|
reportDataInput.pythonBridge.output = p.stdout
|
|
LogHelper.info(`Skill execution time: ${executionTime}ms\n`)
|
|
reportDataInput.pythonBridge.executionTime = `${executionTime}ms`
|
|
} catch (e) {
|
|
LogHelper.info(e.command)
|
|
report.can_run_skill.v = false
|
|
LogHelper.error(`${e}\n`)
|
|
reportDataInput.pythonBridge.error = JSON.stringify(e)
|
|
}
|
|
|
|
/**
|
|
* Python TCP server startup checking
|
|
*/
|
|
|
|
LogHelper.success(`Python TCP server version: ${PYTHON_TCP_SERVER_VERSION}`)
|
|
reportDataInput.pythonTCPServer.version = PYTHON_TCP_SERVER_VERSION
|
|
|
|
LogHelper.info('Starting the Python TCP server...')
|
|
|
|
const pythonTCPServerCommand = `${PYTHON_TCP_SERVER_BIN_PATH} en`
|
|
const pythonTCPServerStart = Date.now()
|
|
const p = spawn(pythonTCPServerCommand, { shell: true })
|
|
|
|
LogHelper.info(pythonTCPServerCommand)
|
|
reportDataInput.pythonTCPServer.command = pythonTCPServerCommand
|
|
|
|
if (osInfo.platform === 'darwin') {
|
|
LogHelper.info(
|
|
'For the first start, it may take a few minutes to cold start the Python TCP server on macOS. No worries it is a one-time thing'
|
|
)
|
|
}
|
|
|
|
let pythonTCPServerOutput = ''
|
|
|
|
p.stdout.on('data', (data) => {
|
|
const newData = data.toString()
|
|
pythonTCPServerOutput += newData
|
|
|
|
if (newData?.toLowerCase().includes('waiting for')) {
|
|
kill(p.pid)
|
|
LogHelper.success('The Python TCP server can successfully start')
|
|
}
|
|
})
|
|
|
|
p.stderr.on('data', (data) => {
|
|
const newData = data.toString()
|
|
const shouldIgnore = shouldIgnoreTCPServerError(newData)
|
|
|
|
// Ignore given warnings on stderr output
|
|
if (!shouldIgnore) {
|
|
pythonTCPServerOutput += newData
|
|
report.can_start_python_tcp_server.v = false
|
|
reportDataInput.pythonTCPServer.error = newData
|
|
LogHelper.error(`Cannot start the Python TCP server: ${newData}`)
|
|
}
|
|
})
|
|
|
|
const timeout = 3 * 60_000
|
|
// In case it takes too long, force kill
|
|
setTimeout(() => {
|
|
kill(p.pid)
|
|
|
|
const error = `The Python TCP server timed out after ${timeout}ms`
|
|
LogHelper.error(error)
|
|
reportDataInput.pythonTCPServer.error = error
|
|
report.can_start_python_tcp_server.v = false
|
|
}, timeout)
|
|
|
|
p.stdout.on('end', async () => {
|
|
const pythonTCPServerEnd = Date.now()
|
|
reportDataInput.pythonTCPServer.output = pythonTCPServerOutput
|
|
reportDataInput.pythonTCPServer.startTime = `${
|
|
pythonTCPServerEnd - pythonTCPServerStart
|
|
}ms`
|
|
LogHelper.info(
|
|
`Python TCP server startup time: ${reportDataInput.pythonTCPServer.startTime}\n`
|
|
)
|
|
|
|
/**
|
|
* Global resolvers NLP model checking
|
|
*/
|
|
|
|
LogHelper.info('Global resolvers NLP model state')
|
|
|
|
if (
|
|
!fs.existsSync(GLOBAL_RESOLVERS_NLP_MODEL_PATH) ||
|
|
!Object.keys(
|
|
await fs.promises.readFile(GLOBAL_RESOLVERS_NLP_MODEL_PATH)
|
|
).length
|
|
) {
|
|
const state = 'Global resolvers NLP model not found or broken'
|
|
|
|
report.can_text.v = false
|
|
Object.keys(report).forEach((item) => {
|
|
if (item.indexOf('stt') !== -1 || item.indexOf('tts') !== -1)
|
|
report[item].v = false
|
|
})
|
|
LogHelper.error(
|
|
`${state}. Try to generate a new one: "npm run train"\n`
|
|
)
|
|
reportDataInput.nlpModels.globalResolversModelState = state
|
|
} else {
|
|
const state = 'Found and valid'
|
|
|
|
LogHelper.success(`${state}\n`)
|
|
reportDataInput.nlpModels.globalResolversModelState = state
|
|
}
|
|
|
|
/**
|
|
* Skills resolvers NLP model checking
|
|
*/
|
|
|
|
LogHelper.info('Skills resolvers NLP model state')
|
|
|
|
if (
|
|
!fs.existsSync(SKILLS_RESOLVERS_NLP_MODEL_PATH) ||
|
|
!Object.keys(
|
|
await fs.promises.readFile(SKILLS_RESOLVERS_NLP_MODEL_PATH)
|
|
).length
|
|
) {
|
|
const state = 'Skills resolvers NLP model not found or broken'
|
|
|
|
report.can_text.v = false
|
|
Object.keys(report).forEach((item) => {
|
|
if (item.indexOf('stt') !== -1 || item.indexOf('tts') !== -1)
|
|
report[item].v = false
|
|
})
|
|
LogHelper.error(
|
|
`${state}. Try to generate a new one: "npm run train"\n`
|
|
)
|
|
reportDataInput.nlpModels.skillsResolversModelState = state
|
|
} else {
|
|
const state = 'Found and valid'
|
|
|
|
LogHelper.success(`${state}\n`)
|
|
reportDataInput.nlpModels.skillsResolversModelState = state
|
|
}
|
|
|
|
/**
|
|
* Main NLP model checking
|
|
*/
|
|
|
|
LogHelper.info('Main NLP model state')
|
|
|
|
if (
|
|
!fs.existsSync(MAIN_NLP_MODEL_PATH) ||
|
|
!Object.keys(await fs.promises.readFile(MAIN_NLP_MODEL_PATH)).length
|
|
) {
|
|
const state = 'Main NLP model not found or broken'
|
|
|
|
report.can_text.v = false
|
|
Object.keys(report).forEach((item) => {
|
|
if (item.indexOf('stt') !== -1 || item.indexOf('tts') !== -1)
|
|
report[item].v = false
|
|
})
|
|
LogHelper.error(
|
|
`${state}. Try to generate a new one: "npm run train"\n`
|
|
)
|
|
reportDataInput.nlpModels.mainModelState = state
|
|
} else {
|
|
const state = 'Found and valid'
|
|
|
|
LogHelper.success(`${state}\n`)
|
|
reportDataInput.nlpModels.mainModelState = state
|
|
}
|
|
|
|
/**
|
|
* TTS/STT checking
|
|
*/
|
|
|
|
LogHelper.info('Amazon Polly TTS')
|
|
|
|
try {
|
|
const json = JSON.parse(await fs.promises.readFile(amazonPath))
|
|
if (
|
|
json.credentials.accessKeyId === '' ||
|
|
json.credentials.secretAccessKey === ''
|
|
) {
|
|
report.can_amazon_polly_tts.v = false
|
|
LogHelper.warning('Amazon Polly TTS is not yet configured\n')
|
|
} else {
|
|
LogHelper.success('Configured\n')
|
|
}
|
|
} catch (e) {
|
|
report.can_amazon_polly_tts.v = false
|
|
LogHelper.warning(`Amazon Polly TTS is not yet configured: ${e}\n`)
|
|
}
|
|
|
|
LogHelper.info('Google Cloud TTS/STT')
|
|
|
|
try {
|
|
const json = JSON.parse(await fs.promises.readFile(googleCloudPath))
|
|
const results = []
|
|
Object.keys(json).forEach((item) => {
|
|
if (json[item] === '') results.push(false)
|
|
})
|
|
if (results.includes(false)) {
|
|
report.can_google_cloud_tts.v = false
|
|
report.can_google_cloud_stt.v = false
|
|
LogHelper.warning('Google Cloud TTS/STT is not yet configured\n')
|
|
} else {
|
|
LogHelper.success('Configured\n')
|
|
}
|
|
} catch (e) {
|
|
report.can_google_cloud_tts.v = false
|
|
report.can_google_cloud_stt.v = false
|
|
LogHelper.warning(`Google Cloud TTS/STT is not yet configured: ${e}\n`)
|
|
}
|
|
|
|
LogHelper.info('Watson TTS')
|
|
|
|
try {
|
|
const json = JSON.parse(await fs.promises.readFile(watsonTtsPath))
|
|
const results = []
|
|
Object.keys(json).forEach((item) => {
|
|
if (json[item] === '') results.push(false)
|
|
})
|
|
if (results.includes(false)) {
|
|
report.can_watson_tts.v = false
|
|
LogHelper.warning('Watson TTS is not yet configured\n')
|
|
} else {
|
|
LogHelper.success('Configured\n')
|
|
}
|
|
} catch (e) {
|
|
report.can_watson_tts.v = false
|
|
LogHelper.warning(`Watson TTS is not yet configured: ${e}\n`)
|
|
}
|
|
|
|
LogHelper.info('Offline TTS')
|
|
|
|
if (!fs.existsSync(flitePath)) {
|
|
report.can_offline_tts.v = false
|
|
LogHelper.warning(
|
|
`Cannot find ${flitePath}. You can set up the offline TTS by running: "npm run setup:offline-tts"\n`
|
|
)
|
|
} else {
|
|
LogHelper.success(`Found Flite at ${flitePath}\n`)
|
|
}
|
|
|
|
LogHelper.info('Watson STT')
|
|
|
|
try {
|
|
const json = JSON.parse(await fs.promises.readFile(watsonSttPath))
|
|
const results = []
|
|
Object.keys(json).forEach((item) => {
|
|
if (json[item] === '') results.push(false)
|
|
})
|
|
if (results.includes(false)) {
|
|
report.can_watson_stt.v = false
|
|
LogHelper.warning('Watson STT is not yet configured\n')
|
|
} else {
|
|
LogHelper.success('Configured\n')
|
|
}
|
|
} catch (e) {
|
|
report.can_watson_stt.v = false
|
|
LogHelper.warning(`Watson STT is not yet configured: ${e}`)
|
|
}
|
|
|
|
LogHelper.info('Offline STT')
|
|
|
|
if (!fs.existsSync(coquiLanguageModelPath)) {
|
|
report.can_offline_stt.v = false
|
|
LogHelper.warning(
|
|
`Cannot find ${coquiLanguageModelPath}. You can setup the offline STT by running: "npm run setup:offline-stt"`
|
|
)
|
|
} else {
|
|
LogHelper.success(
|
|
`Found Coqui language model at ${coquiLanguageModelPath}`
|
|
)
|
|
}
|
|
|
|
/**
|
|
* Report
|
|
*/
|
|
|
|
LogHelper.title('Report')
|
|
|
|
LogHelper.info('Here is the diagnosis about your current setup')
|
|
Object.keys(report).forEach((item) => {
|
|
if (report[item].v === true) {
|
|
LogHelper.success(report[item].title)
|
|
} else {
|
|
LogHelper[report[item].type](report[item].title)
|
|
}
|
|
})
|
|
|
|
LogHelper.default('')
|
|
if (
|
|
report.can_run.v &&
|
|
report.can_run_skill.v &&
|
|
report.can_text.v &&
|
|
report.can_start_python_tcp_server.v
|
|
) {
|
|
LogHelper.success('Hooray! Leon can run correctly')
|
|
LogHelper.info(
|
|
'If you have some yellow warnings, it is all good. It means some entities are not yet configured'
|
|
)
|
|
} else {
|
|
LogHelper.error('Please fix the errors above')
|
|
}
|
|
|
|
reportDataInput.report = report
|
|
|
|
reportDataInput = JSON.parse(
|
|
SystemHelper.sanitizeUsername(JSON.stringify(reportDataInput))
|
|
)
|
|
|
|
LogHelper.title('REPORT URL')
|
|
|
|
LogHelper.info('Sending report...')
|
|
|
|
try {
|
|
const { data } = await axios.post('https://getleon.ai/api/report', {
|
|
report: reportDataInput
|
|
})
|
|
const { data: responseReportData } = data
|
|
|
|
LogHelper.success(`Report URL: ${responseReportData.reportUrl}`)
|
|
} catch (e) {
|
|
LogHelper.error(`Failed to send report: ${e}`)
|
|
}
|
|
|
|
process.exit(0)
|
|
})
|
|
} catch (e) {
|
|
LogHelper.error(e)
|
|
}
|
|
})()
|