Merge pull request #1495 from poolifier/combined-prs-branch
[poolifier.git] / src / pools / worker.ts
CommitLineData
85aeb3f3 1import type { MessageChannel } from 'node:worker_threads'
f06e48d8 2import type { CircularArray } from '../circular-array'
5c4d16da 3import type { Task } from '../utility-types'
f06e48d8 4
671d5154
JB
5/**
6 * Callback invoked when the worker has started successfully.
7 */
8export type OnlineHandler<Worker extends IWorker> = (this: Worker) => void
9
bdaf31cd
JB
10/**
11 * Callback invoked if the worker has received a message.
12 */
50e66724
JB
13export type MessageHandler<Worker extends IWorker> = (
14 this: Worker,
e102732c 15 message: unknown
50e66724 16) => void
bdaf31cd
JB
17
18/**
19 * Callback invoked if the worker raised an error.
20 */
50e66724
JB
21export type ErrorHandler<Worker extends IWorker> = (
22 this: Worker,
e102732c 23 error: Error
50e66724 24) => void
bdaf31cd 25
bdaf31cd
JB
26/**
27 * Callback invoked when the worker exits successfully.
28 */
50e66724
JB
29export type ExitHandler<Worker extends IWorker> = (
30 this: Worker,
e102732c 31 exitCode: number
50e66724 32) => void
bdaf31cd 33
f06e48d8 34/**
cd4d348a 35 * Measurement statistics.
f9b4bbf8
JB
36 *
37 * @internal
f06e48d8 38 */
cd4d348a 39export interface MeasurementStatistics {
02706357 40 /**
932fc8be 41 * Measurement aggregate.
02706357 42 */
71514351 43 aggregate?: number
f7510105
JB
44 /**
45 * Measurement minimum.
46 */
71514351 47 minimum?: number
f7510105
JB
48 /**
49 * Measurement maximum.
50 */
71514351 51 maximum?: number
02706357 52 /**
cd4d348a 53 * Measurement average.
02706357 54 */
71514351 55 average?: number
02706357 56 /**
cd4d348a 57 * Measurement median.
02706357 58 */
71514351 59 median?: number
02706357 60 /**
cd4d348a 61 * Measurement history.
02706357 62 */
eb7bf744 63 readonly history: CircularArray<number>
a4e07f72
JB
64}
65
5df69fab
JB
66/**
67 * Event loop utilization measurement statistics.
68 *
69 * @internal
70 */
71export interface EventLoopUtilizationMeasurementStatistics {
eb7bf744
JB
72 readonly idle: MeasurementStatistics
73 readonly active: MeasurementStatistics
f7510105 74 utilization?: number
5df69fab
JB
75}
76
a4e07f72
JB
77/**
78 * Task statistics.
79 *
80 * @internal
81 */
a4e07f72 82export interface TaskStatistics {
02706357 83 /**
9a0613e9 84 * Number of executed tasks.
02706357 85 */
a4e07f72 86 executed: number
02706357 87 /**
9a0613e9 88 * Number of executing tasks.
02706357 89 */
a4e07f72 90 executing: number
0567595a 91 /**
9a0613e9 92 * Number of queued tasks.
0567595a 93 */
8604aaab 94 readonly queued: number
df593701
JB
95 /**
96 * Maximum number of queued tasks.
97 */
b25a42cd 98 readonly maxQueued?: number
68cbdc84
JB
99 /**
100 * Number of stolen tasks.
101 */
102 stolen: number
0567595a 103 /**
9a0613e9 104 * Number of failed tasks.
0567595a 105 */
a4e07f72
JB
106 failed: number
107}
108
4b628b48
JB
109/**
110 * Enumeration of worker types.
111 */
112export const WorkerTypes = Object.freeze({
149fdbb9
JB
113 thread: 'thread',
114 cluster: 'cluster'
4b628b48
JB
115} as const)
116
117/**
118 * Worker type.
119 */
120export type WorkerType = keyof typeof WorkerTypes
121
f59e1027
JB
122/**
123 * Worker information.
124 *
125 * @internal
126 */
127export interface WorkerInfo {
128 /**
83fa0a36 129 * Worker id.
f59e1027 130 */
eb7bf744 131 readonly id: number | undefined
4b628b48
JB
132 /**
133 * Worker type.
134 */
5b49e864 135 readonly type: WorkerType
8a1260a3
JB
136 /**
137 * Dynamic flag.
138 */
139 dynamic: boolean
f59e1027 140 /**
7c8381eb 141 * Ready flag.
f59e1027 142 */
7c8381eb 143 ready: boolean
b558f6b5
JB
144 /**
145 * Task function names.
146 */
6703b9f4 147 taskFunctionNames?: string[]
f59e1027
JB
148}
149
a4e07f72
JB
150/**
151 * Worker usage statistics.
152 *
153 * @internal
154 */
155export interface WorkerUsage {
0567595a 156 /**
a4e07f72 157 * Tasks statistics.
0567595a 158 */
eb7bf744 159 readonly tasks: TaskStatistics
0567595a 160 /**
a4e07f72 161 * Tasks runtime statistics.
0567595a 162 */
eb7bf744 163 readonly runTime: MeasurementStatistics
02706357 164 /**
a4e07f72 165 * Tasks wait time statistics.
02706357 166 */
eb7bf744 167 readonly waitTime: MeasurementStatistics
62c15a68 168 /**
5df69fab 169 * Tasks event loop utilization statistics.
62c15a68 170 */
eb7bf744 171 readonly elu: EventLoopUtilizationMeasurementStatistics
f06e48d8
JB
172}
173
f3a91bac 174/**
9df180cb 175 * Worker choice strategy data.
57a29f75
JB
176 *
177 * @internal
f3a91bac
JB
178 */
179export interface StrategyData {
180 virtualTaskEndTimestamp?: number
181}
182
f06e48d8
JB
183/**
184 * Worker interface.
185 */
186export interface IWorker {
f59e1027 187 /**
83fa0a36 188 * Worker id.
f59e1027 189 */
aecc6e48
JB
190 readonly id?: number
191 readonly threadId?: number
bdaf31cd 192 /**
64383951 193 * Registers an event listener.
bdaf31cd 194 *
38e795c1 195 * @param event - The event.
48ef9107 196 * @param handler - The event handler.
bdaf31cd 197 */
fd04474e
JB
198 readonly on: ((event: 'online', handler: OnlineHandler<this>) => void) &
199 ((event: 'message', handler: MessageHandler<this>) => void) &
78cea37e 200 ((event: 'error', handler: ErrorHandler<this>) => void) &
78cea37e 201 ((event: 'exit', handler: ExitHandler<this>) => void)
bdaf31cd 202 /**
64383951 203 * Registers a listener to the exit event that will only be performed once.
bdaf31cd 204 *
9ece5893 205 * @param event - The `'exit'` event.
38e795c1 206 * @param handler - The exit handler.
bdaf31cd 207 */
4b628b48 208 readonly once: (event: 'exit', handler: ExitHandler<this>) => void
bdaf31cd 209}
f06e48d8 210
5b49e864 211/**
9f95d5eb 212 * Worker node event detail.
5b49e864 213 *
5b49e864
JB
214 * @internal
215 */
9f95d5eb
JB
216export interface WorkerNodeEventDetail {
217 workerId: number
218}
ec287edf 219
f06e48d8
JB
220/**
221 * Worker node interface.
c319c66b
JB
222 *
223 * @typeParam Worker - Type of worker.
e102732c 224 * @typeParam Data - Type of data sent to the worker. This can only be structured-cloneable data.
c319c66b 225 * @internal
f06e48d8 226 */
9f95d5eb
JB
227export interface IWorkerNode<Worker extends IWorker, Data = unknown>
228 extends EventTarget {
c319c66b 229 /**
ccd73e07 230 * Worker.
c319c66b 231 */
02706357 232 readonly worker: Worker
f59e1027 233 /**
ccd73e07 234 * Worker info.
f59e1027 235 */
eb7bf744 236 readonly info: WorkerInfo
c319c66b 237 /**
ccd73e07 238 * Worker usage statistics.
c319c66b 239 */
5b49e864 240 readonly usage: WorkerUsage
f3a91bac 241 /**
9df180cb 242 * Worker choice strategy data.
f3a91bac
JB
243 * This is used to store data that is specific to the worker choice strategy.
244 */
245 strategyData?: StrategyData
26fb3c18
JB
246 /**
247 * Message channel (worker_threads only).
248 */
249 readonly messageChannel?: MessageChannel
20c6f652
JB
250 /**
251 * Tasks queue back pressure size.
252 * This is the number of tasks that can be enqueued before the worker node has back pressure.
253 */
254 tasksQueueBackPressureSize: number
c319c66b 255 /**
ccd73e07 256 * Tasks queue size.
4b628b48
JB
257 *
258 * @returns The tasks queue size.
259 */
260 readonly tasksQueueSize: () => number
261 /**
ccd73e07 262 * Enqueue task.
4b628b48
JB
263 *
264 * @param task - The task to queue.
a1763c54 265 * @returns The tasks queue size.
4b628b48
JB
266 */
267 readonly enqueueTask: (task: Task<Data>) => number
72695f86
JB
268 /**
269 * Prepends a task to the tasks queue.
270 *
271 * @param task - The task to prepend.
272 * @returns The tasks queue size.
273 */
274 readonly unshiftTask: (task: Task<Data>) => number
4b628b48 275 /**
ccd73e07 276 * Dequeue task.
4b628b48
JB
277 *
278 * @returns The dequeued task.
279 */
280 readonly dequeueTask: () => Task<Data> | undefined
72695f86
JB
281 /**
282 * Pops a task from the tasks queue.
283 *
284 * @returns The popped task.
285 */
286 readonly popTask: () => Task<Data> | undefined
4b628b48 287 /**
ccd73e07 288 * Clears tasks queue.
4b628b48
JB
289 */
290 readonly clearTasksQueue: () => void
671d5154 291 /**
e2b31e32 292 * Whether the worker node has back pressure (i.e. its tasks queue is full).
671d5154
JB
293 *
294 * @returns `true` if the worker node has back pressure, `false` otherwise.
295 */
296 readonly hasBackPressure: () => boolean
4b628b48 297 /**
ff469b0e 298 * Resets usage statistics.
c319c66b 299 */
4b628b48 300 readonly resetUsage: () => void
3f09ed9f 301 /**
a5d15204 302 * Closes communication channel.
3f09ed9f
JB
303 */
304 readonly closeChannel: () => void
ff128cc9 305 /**
2809112e
JB
306 * Gets task function worker usage statistics.
307 *
308 * @param name - The task function name.
309 * @returns The task function worker usage statistics if the task function worker usage statistics are initialized, `undefined` otherwise.
ff128cc9 310 */
db0e38ee 311 readonly getTaskFunctionWorkerUsage: (name: string) => WorkerUsage | undefined
adee6053
JB
312 /**
313 * Deletes task function worker usage statistics.
314 *
315 * @param name - The task function name.
316 * @returns `true` if the task function worker usage statistics were deleted, `false` otherwise.
317 */
318 readonly deleteTaskFunctionWorkerUsage: (name: string) => boolean
f06e48d8 319}