2021-05-28 18:50:16 +08:00
|
|
|
import { setTimeout } from 'timers'
|
|
|
|
import { promisify } from 'util'
|
|
|
|
|
2023-04-06 11:04:53 +08:00
|
|
|
import * as colors from 'colorette'
|
2021-05-28 18:50:16 +08:00
|
|
|
import Dockerode from 'dockerode'
|
|
|
|
import prettyBytes from 'pretty-bytes'
|
|
|
|
|
|
|
|
const sleep = promisify(setTimeout)
|
|
|
|
|
|
|
|
const client = new Dockerode()
|
|
|
|
|
|
|
|
export async function createSuite(testFile, maxMemoryUsage) {
|
2023-04-06 11:04:53 +08:00
|
|
|
console.info(colors.cyanBright(`Create container to test ${testFile}`))
|
2021-05-28 18:50:16 +08:00
|
|
|
|
|
|
|
const container = await client.createContainer({
|
|
|
|
Image: 'node:lts-slim',
|
2022-04-06 23:00:21 +08:00
|
|
|
Cmd: ['/bin/bash', '-c', `node --expose-gc memory-testing/${testFile}.mjs`],
|
2021-05-28 18:50:16 +08:00
|
|
|
AttachStdout: true,
|
|
|
|
AttachStderr: true,
|
|
|
|
Tty: true,
|
|
|
|
WorkingDir: '/napi-rs',
|
|
|
|
Env: ['MAX_OLD_SPACE_SIZE=256', 'FORCE_COLOR=1'],
|
|
|
|
HostConfig: {
|
|
|
|
Binds: [`${process.cwd()}:/napi-rs:rw`],
|
|
|
|
Memory: 256 * 1024 * 1024,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
2023-04-06 11:04:53 +08:00
|
|
|
console.info(colors.cyanBright('Container created, starting ...'))
|
2021-05-28 18:50:16 +08:00
|
|
|
|
|
|
|
await container.start()
|
|
|
|
|
|
|
|
container.attach(
|
|
|
|
{ stream: true, stdout: true, stderr: true },
|
|
|
|
function (err, stream) {
|
|
|
|
if (err) {
|
|
|
|
console.error(err)
|
|
|
|
process.exit(1)
|
|
|
|
}
|
|
|
|
stream.pipe(process.stdout)
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
const stats = await container.stats()
|
|
|
|
|
2021-05-29 23:36:49 +08:00
|
|
|
let shouldAssertMemoryUsage = false
|
2022-12-28 22:28:47 +08:00
|
|
|
let initialMemoryUsage
|
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
const initialDate = Date.now()
|
2021-05-28 18:50:16 +08:00
|
|
|
stats.on('data', (d) => {
|
|
|
|
const { memory_stats } = JSON.parse(d.toString('utf8'))
|
2022-12-28 22:28:47 +08:00
|
|
|
if (Date.now() - initialDate > 10000 && !shouldAssertMemoryUsage) {
|
|
|
|
initialMemoryUsage = memory_stats.usage
|
|
|
|
shouldAssertMemoryUsage = true
|
2023-03-21 18:12:52 +08:00
|
|
|
resolve()
|
2022-12-28 22:28:47 +08:00
|
|
|
}
|
2021-05-29 23:36:49 +08:00
|
|
|
if (shouldAssertMemoryUsage && memory_stats?.usage) {
|
2021-05-28 18:50:16 +08:00
|
|
|
const memoryGrowth = memory_stats.usage - initialMemoryUsage
|
2022-12-28 22:28:47 +08:00
|
|
|
if (memoryGrowth > (maxMemoryUsage ?? initialMemoryUsage)) {
|
2021-05-28 18:50:16 +08:00
|
|
|
console.info(
|
2023-04-06 11:04:53 +08:00
|
|
|
colors.redBright(
|
2021-05-28 18:50:16 +08:00
|
|
|
`Potential memory leak, memory growth: ${prettyBytes(
|
|
|
|
memoryGrowth,
|
2023-11-20 15:56:20 +08:00
|
|
|
)}, test file: ${testFile}`,
|
2021-05-28 18:50:16 +08:00
|
|
|
),
|
|
|
|
)
|
|
|
|
process.exit(1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
stats.on('error', reject)
|
|
|
|
})
|
|
|
|
|
|
|
|
console.info(
|
2023-04-06 11:04:53 +08:00
|
|
|
colors.red(`Initial memory usage: ${prettyBytes(initialMemoryUsage ?? 0)}`),
|
2021-05-28 18:50:16 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
await sleep(60000)
|
|
|
|
|
2022-10-02 11:14:52 +08:00
|
|
|
try {
|
|
|
|
await container.stop()
|
|
|
|
await container.remove()
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e)
|
|
|
|
}
|
2021-05-28 18:50:16 +08:00
|
|
|
}
|