2022-10-19 14:23:28 +03:00
|
|
|
import fs from 'node:fs'
|
|
|
|
import path from 'node:path'
|
|
|
|
import stream from 'node:stream'
|
2022-10-22 20:00:55 +03:00
|
|
|
import readline from 'node:readline'
|
2022-10-19 14:23:28 +03:00
|
|
|
|
|
|
|
import axios from 'axios'
|
2023-05-12 19:30:15 +03:00
|
|
|
import { command } from 'execa'
|
2022-10-22 20:00:55 +03:00
|
|
|
import prettyBytes from 'pretty-bytes'
|
|
|
|
import prettyMilliseconds from 'pretty-ms'
|
2022-10-19 14:23:28 +03:00
|
|
|
import extractZip from 'extract-zip'
|
|
|
|
|
|
|
|
import {
|
|
|
|
BINARIES_FOLDER_NAME,
|
|
|
|
GITHUB_URL,
|
2023-05-12 19:30:15 +03:00
|
|
|
NODEJS_BRIDGE_ROOT_PATH,
|
2023-04-30 18:39:39 +03:00
|
|
|
NODEJS_BRIDGE_DIST_PATH,
|
2022-10-19 14:23:28 +03:00
|
|
|
PYTHON_BRIDGE_DIST_PATH,
|
|
|
|
TCP_SERVER_DIST_PATH,
|
2023-04-30 18:39:39 +03:00
|
|
|
NODEJS_BRIDGE_BIN_NAME,
|
2022-10-19 14:23:28 +03:00
|
|
|
PYTHON_BRIDGE_BIN_NAME,
|
|
|
|
TCP_SERVER_BIN_NAME,
|
2023-04-30 18:39:39 +03:00
|
|
|
NODEJS_BRIDGE_VERSION,
|
2022-10-19 14:23:28 +03:00
|
|
|
PYTHON_BRIDGE_VERSION,
|
|
|
|
TCP_SERVER_VERSION
|
|
|
|
} from '@/constants'
|
|
|
|
import { LogHelper } from '@/helpers/log-helper'
|
|
|
|
|
|
|
|
/**
|
2023-04-30 18:39:39 +03:00
|
|
|
* Set up binaries according to the given setup target
|
2022-10-22 18:22:50 +03:00
|
|
|
* 1. Delete the existing dist binaries if already exist
|
2023-04-30 18:39:39 +03:00
|
|
|
* 2. Download the latest binaries from GitHub releases
|
2022-10-22 18:22:50 +03:00
|
|
|
* 3. Extract the downloaded ZIP file to the dist folder
|
2022-10-19 14:23:28 +03:00
|
|
|
*/
|
|
|
|
|
2023-04-30 18:39:39 +03:00
|
|
|
const TARGETS = new Map()
|
2022-10-19 14:23:28 +03:00
|
|
|
|
2023-04-30 18:39:39 +03:00
|
|
|
TARGETS.set('nodejs-bridge', {
|
|
|
|
name: 'Node.js bridge',
|
|
|
|
distPath: NODEJS_BRIDGE_DIST_PATH,
|
|
|
|
manifestPath: path.join(NODEJS_BRIDGE_DIST_PATH, 'manifest.json'),
|
2023-04-30 19:36:03 +03:00
|
|
|
archiveName: `${NODEJS_BRIDGE_BIN_NAME.split('.')[0]}.zip`,
|
2023-04-30 18:39:39 +03:00
|
|
|
version: NODEJS_BRIDGE_VERSION,
|
|
|
|
isPlatformDependent: false // Need to be built for the target platform or not
|
|
|
|
})
|
|
|
|
TARGETS.set('python-bridge', {
|
2022-10-19 14:23:28 +03:00
|
|
|
name: 'Python bridge',
|
|
|
|
distPath: PYTHON_BRIDGE_DIST_PATH,
|
2023-04-29 15:58:10 +03:00
|
|
|
manifestPath: path.join(PYTHON_BRIDGE_DIST_PATH, 'manifest.json'),
|
2022-10-19 14:23:28 +03:00
|
|
|
archiveName: `${PYTHON_BRIDGE_BIN_NAME}-${BINARIES_FOLDER_NAME}.zip`,
|
2023-04-30 18:39:39 +03:00
|
|
|
version: PYTHON_BRIDGE_VERSION,
|
|
|
|
isPlatformDependent: true
|
2022-10-19 14:23:28 +03:00
|
|
|
})
|
2023-04-30 18:39:39 +03:00
|
|
|
TARGETS.set('tcp-server', {
|
2022-10-19 14:23:28 +03:00
|
|
|
name: 'TCP server',
|
|
|
|
distPath: TCP_SERVER_DIST_PATH,
|
2023-04-29 15:58:10 +03:00
|
|
|
manifestPath: path.join(TCP_SERVER_DIST_PATH, 'manifest.json'),
|
2022-10-19 14:23:28 +03:00
|
|
|
archiveName: `${TCP_SERVER_BIN_NAME}-${BINARIES_FOLDER_NAME}.zip`,
|
2023-04-30 18:39:39 +03:00
|
|
|
version: TCP_SERVER_VERSION,
|
|
|
|
isPlatformDependent: true
|
2022-10-19 14:23:28 +03:00
|
|
|
})
|
|
|
|
|
2023-04-29 15:58:10 +03:00
|
|
|
async function createManifestFile(manifestPath, name, version) {
|
|
|
|
const manifest = {
|
|
|
|
name,
|
|
|
|
version,
|
2023-04-30 18:39:39 +03:00
|
|
|
setupDate: Date.now()
|
2023-04-29 15:58:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
await fs.promises.writeFile(manifestPath, JSON.stringify(manifest, null, 2))
|
|
|
|
}
|
|
|
|
|
2023-04-30 18:39:39 +03:00
|
|
|
const setupBinaries = async (key) => {
|
|
|
|
const {
|
|
|
|
name,
|
|
|
|
distPath,
|
|
|
|
archiveName,
|
|
|
|
version,
|
|
|
|
manifestPath,
|
|
|
|
isPlatformDependent
|
|
|
|
} = TARGETS.get(key)
|
2023-04-29 15:58:10 +03:00
|
|
|
let manifest = null
|
|
|
|
|
|
|
|
if (fs.existsSync(manifestPath)) {
|
|
|
|
manifest = JSON.parse(await fs.promises.readFile(manifestPath, 'utf8'))
|
|
|
|
|
|
|
|
LogHelper.info(`Found ${name} ${manifest.version}`)
|
|
|
|
LogHelper.info(`Latest version is ${version}`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!manifest || manifest.version !== version) {
|
2023-04-30 18:39:39 +03:00
|
|
|
const buildPath = isPlatformDependent
|
|
|
|
? path.join(distPath, BINARIES_FOLDER_NAME)
|
2023-04-30 20:13:19 +03:00
|
|
|
: path.join(distPath, 'bin')
|
2023-04-29 15:58:10 +03:00
|
|
|
const archivePath = path.join(distPath, archiveName)
|
|
|
|
|
|
|
|
await Promise.all([
|
|
|
|
fs.promises.rm(buildPath, { recursive: true, force: true }),
|
|
|
|
fs.promises.rm(archivePath, { recursive: true, force: true })
|
|
|
|
])
|
|
|
|
|
2023-05-12 19:30:15 +03:00
|
|
|
if (key === 'nodejs-bridge') {
|
|
|
|
try {
|
|
|
|
LogHelper.info('Installing Node.js bridge npm packages...')
|
|
|
|
|
|
|
|
await command(
|
|
|
|
`npm install --package-lock=false --prefix ${NODEJS_BRIDGE_ROOT_PATH}`,
|
|
|
|
{
|
|
|
|
shell: true
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
LogHelper.success('Node.js bridge npm packages installed')
|
|
|
|
} catch (e) {
|
|
|
|
throw new Error(`Failed to install Node.js bridge npm packages: ${e}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-29 15:58:10 +03:00
|
|
|
try {
|
|
|
|
LogHelper.info(`Downloading ${name}...`)
|
|
|
|
|
|
|
|
const archiveWriter = fs.createWriteStream(archivePath)
|
|
|
|
const latestReleaseAssetURL = `${GITHUB_URL}/releases/download/${key}_v${version}/${archiveName}`
|
|
|
|
const { data } = await axios.get(latestReleaseAssetURL, {
|
|
|
|
responseType: 'stream',
|
|
|
|
onDownloadProgress: ({ loaded, total, progress, estimated, rate }) => {
|
|
|
|
const percentage = Math.floor(progress * 100)
|
|
|
|
const downloadedSize = prettyBytes(loaded)
|
|
|
|
const totalSize = prettyBytes(total)
|
|
|
|
const estimatedTime = !estimated
|
|
|
|
? 0
|
|
|
|
: prettyMilliseconds(estimated * 1_000, { secondsDecimalDigits: 0 })
|
|
|
|
const downloadRate = !rate ? 0 : prettyBytes(rate)
|
|
|
|
|
|
|
|
readline.clearLine(process.stdout, 0)
|
|
|
|
readline.cursorTo(process.stdout, 0, null)
|
|
|
|
process.stdout.write(
|
|
|
|
`Download progress: ${percentage}% (${downloadedSize}/${totalSize} | ${downloadRate}/s | ${estimatedTime} ETA)`
|
|
|
|
)
|
|
|
|
|
|
|
|
if (percentage === 100) {
|
|
|
|
process.stdout.write('\n')
|
|
|
|
}
|
2022-10-22 20:00:55 +03:00
|
|
|
}
|
2023-04-29 15:58:10 +03:00
|
|
|
})
|
2022-10-22 18:22:50 +03:00
|
|
|
|
2023-04-29 15:58:10 +03:00
|
|
|
data.pipe(archiveWriter)
|
|
|
|
await stream.promises.finished(archiveWriter)
|
2022-10-22 18:22:50 +03:00
|
|
|
|
2023-04-29 15:58:10 +03:00
|
|
|
LogHelper.success(`${name} downloaded`)
|
|
|
|
LogHelper.info(`Extracting ${name}...`)
|
2022-10-22 20:00:55 +03:00
|
|
|
|
2023-05-01 17:48:31 +03:00
|
|
|
const absoluteDistPath = path.resolve(distPath)
|
2023-04-29 15:58:10 +03:00
|
|
|
await extractZip(archivePath, { dir: absoluteDistPath })
|
2022-10-22 18:22:50 +03:00
|
|
|
|
2023-04-29 15:58:10 +03:00
|
|
|
LogHelper.success(`${name} extracted`)
|
2022-10-22 20:00:55 +03:00
|
|
|
|
2023-04-30 18:39:39 +03:00
|
|
|
await Promise.all([
|
2023-04-29 15:58:10 +03:00
|
|
|
fs.promises.rm(archivePath, { recursive: true, force: true }),
|
|
|
|
createManifestFile(manifestPath, name, version)
|
|
|
|
])
|
2022-10-22 20:00:55 +03:00
|
|
|
|
2023-04-29 15:58:10 +03:00
|
|
|
LogHelper.success(`${name} manifest file created`)
|
|
|
|
LogHelper.success(`${name} ${version} ready`)
|
2023-05-12 18:54:57 +03:00
|
|
|
} catch (e) {
|
|
|
|
throw new Error(`Failed to set up ${name}: ${e}`)
|
2023-04-29 15:58:10 +03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
LogHelper.success(`${name} is already at the latest version (${version})`)
|
2022-10-19 14:23:28 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default async () => {
|
2023-04-30 20:13:19 +03:00
|
|
|
await setupBinaries('nodejs-bridge')
|
2023-04-30 18:39:39 +03:00
|
|
|
await setupBinaries('python-bridge')
|
|
|
|
await setupBinaries('tcp-server')
|
2022-10-19 14:23:28 +03:00
|
|
|
}
|