Move cpulimits responsability

This commit is contained in:
Adria Navarro 2024-02-07 18:12:14 +01:00
parent 3b8b60aa03
commit 0d0171fa08
3 changed files with 22 additions and 22 deletions

View File

@ -5,13 +5,6 @@ import tracer from "dd-trace"
import { IsolatedVM } from "./vm"
import { context } from "@budibase/backend-core"
class ExecutionTimeoutError extends Error {
constructor(message: string) {
super(message)
this.name = "ExecutionTimeoutError"
}
}
export function init() {
setJSRunner((js: string, ctx: Record<string, any>) => {
return tracer.trace("runJS", {}, span => {
@ -28,16 +21,6 @@ export function init() {
bbCtx.vm = vm
}
const perRequestLimit = env.JS_PER_REQUEST_TIME_LIMIT_MS
if (perRequestLimit) {
const cpuMs = Number(vm.cpuTime) / 1e6
if (cpuMs > perRequestLimit) {
throw new ExecutionTimeoutError(
`CPU time limit exceeded (${cpuMs}ms > ${perRequestLimit}ms)`
)
}
}
const result = vm.execute(js)
return result

View File

@ -7,11 +7,19 @@ import querystring from "querystring"
import { BundleType, loadBundle } from "../bundles"
import { VM } from "@budibase/types"
class ExecutionTimeoutError extends Error {
constructor(message: string) {
super(message)
this.name = "ExecutionTimeoutError"
}
}
export class IsolatedVM implements VM {
#isolate: ivm.Isolate
#vm: ivm.Context
#jail: ivm.Reference
#timeout: number
#perRequestLimit?: number
#modules: Record<
string,
@ -26,9 +34,11 @@ export class IsolatedVM implements VM {
constructor({
memoryLimit,
timeout,
perRequestLimit,
}: {
memoryLimit: number
timeout: number
perRequestLimit?: number
}) {
this.#isolate = new ivm.Isolate({ memoryLimit })
this.#vm = this.#isolate.createContextSync()
@ -40,10 +50,7 @@ export class IsolatedVM implements VM {
})
this.#timeout = timeout
}
get cpuTime() {
return this.#isolate.cpuTime
this.#perRequestLimit = perRequestLimit
}
withHelpers() {
@ -115,6 +122,17 @@ export class IsolatedVM implements VM {
}
execute(code: string): string {
const perRequestLimit = this.#perRequestLimit
if (perRequestLimit) {
const cpuMs = Number(this.#isolate.cpuTime) / 1e6
if (cpuMs > perRequestLimit) {
throw new ExecutionTimeoutError(
`CPU time limit exceeded (${cpuMs}ms > ${perRequestLimit}ms)`
)
}
}
code = [
...Object.values(this.#modules).map(m => m.headCode),
`results.out=${code};`,

View File

@ -1,4 +1,3 @@
export interface VM {
cpuTime: bigint
execute(code: string): string
}