Files
bun.sh/src/codegen/class-definitions.ts
2025-05-01 16:09:44 -07:00

280 lines
7.0 KiB
TypeScript

interface PropertyAttribute {
enumerable?: boolean;
configurable?: boolean;
/**
* The name for a private symbol to use as the property name. The value should
* be a private symbol from `BunBuiltinNames.h`. This will omit the property
* from the prototype hash table, instead setting it using `putDirect()`.
*/
privateSymbol?: string;
}
/**
* Specifies what happens when a method is called with `this` set to a value that is not an instance
* of the class.
*/
export enum InvalidThisBehavior {
/**
* Default. Throws a `TypeError`.
*/
Throw,
/**
* Do not call the native implementation; return `undefined`. Some Node.js methods are supposed to
* work like this.
*/
NoOp,
}
export type Field =
| ({
getter: string;
cache?: true | string;
/**
* Allow overriding the value of the property
*/
writable?: boolean;
this?: boolean;
} & PropertyAttribute)
| { value: string }
| ({ setter: string; this?: boolean } & PropertyAttribute)
| ({
accessor: { getter: string; setter: string };
cache?: true | string;
this?: boolean;
} & PropertyAttribute)
| ({
fn: string;
/**
* Mark it as an async function in the TypeScript definition.
*
* Does not do anything at runtime.
*/
async?: boolean;
/**
* Number of parameters accepted by the function.
*
* Sets [`function.length`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function/length).
*/
length?: number;
passThis?: boolean;
invalidThisBehavior?: InvalidThisBehavior;
DOMJIT?: {
returns: string;
args?: [string, string] | [string, string, string] | [string] | [];
pure?: boolean;
};
} & PropertyAttribute)
| { internal: true }
| {
/**
* The function is a builtin (its implementation is defined in
* src/js/builtins/), this value is the name of the code generator
* function: `camelCase(fileName + functionName + "CodeGenerator"`)
*/
builtin: string;
/**
* Number of parameters accepted by the function.
*
* Sets [`function.length`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function/length).
*/
length?: number;
};
export class ClassDefinition {
/**
* Class name.
*
* Used to find the proper struct and as the `.name` of the JS constructor
* function.
*/
name: string;
/**
* Class constructor is newable.
*/
construct?: boolean;
/**
* Class constructor is callable. In JS, ES6 class constructors are not
* callable.
*/
call?: boolean;
/**
* ## IMPORTANT
* You _must_ free the pointer to your native class!
*
* Example for pointers only owned by JavaScript classes:
* ```zig
* pub const NativeClass = struct {
*
* fn constructor(global: *JSC.JSGlobalObject, frame: *JSC.CallFrame) bun.JSError!*SocketAddress {
* // do stuff
* return bun.new(NativeClass, .{
* // ...
* });
* }
*
* fn finalize(this: *NativeClass) void {
* // free allocations owned by this class, then free the struct itself.
* bun.destroy(this);
* }
* };
* ```
* Example with ref counting:
* ```
* pub const RefCountedNativeClass = struct {
* const RefCount = bun.ptr.RefCount(@This(), "ref_count", deinit, .{});
* pub const ref = RefCount.ref;
* pub const deref = RefCount.deref;
*
* fn constructor(global: *JSC.JSGlobalObject, frame: *JSC.CallFrame) bun.JSError!*SocketAddress {
* // do stuff
* return bun.new(NativeClass, .{
* // ...
* });
* }
*
* fn deinit(this: *NativeClass) void {
* // free allocations owned by this class, then free the struct itself.
* bun.destroy(this);
* }
*
* pub const finalize = deref; // GC will deref, which can free if no references are left.
* };
* ```
* @todo remove this and require all classes to implement `finalize`.
*/
finalize?: boolean;
overridesToJS?: boolean;
/**
* Static properties and methods.
*/
klass: Record<string, Field>;
/**
* properties and methods on the prototype.
*/
proto: Record<string, Field>;
/**
* Properties and methods attached to the instance itself.
*/
own: Record<string, string>;
values?: string[];
/**
* Set this to `"0b11101110"`.
*/
JSType?: string;
noConstructor?: boolean;
final?: boolean;
// Do not try to track the `this` value in the constructor automatically.
// That is a memory leak.
wantsThis?: never;
/**
* Class has an `estimatedSize` function that reports external allocations to GC.
* Called from any thread.
*
* When `true`, classes should have a method with this signature:
* ```zig
* pub fn estimatedSize(this: *@This()) usize;
* ```
*
* Report `@sizeOf(@this())` as well as any external allocations.
*/
estimatedSize?: boolean;
/**
* Used in heap snapshots.
*
* If true, the class will have a `memoryCost` method that returns the size of the object in bytes.
*
* Unlike estimatedSize, this is always called on the main thread and not used for GC.
*
* If none is provided, we use the struct size.
*/
memoryCost?: boolean;
hasPendingActivity?: boolean;
isEventEmitter?: boolean;
supportsObjectCreate?: boolean;
getInternalProperties?: boolean;
custom?: Record<string, CustomField>;
configurable?: boolean;
enumerable?: boolean;
structuredClone?: { transferable: boolean; tag: number };
customInspect?: boolean;
callbacks?: Record<string, string>;
constructor(options: Partial<ClassDefinition>) {
this.name = options.name ?? "";
this.klass = options.klass ?? {};
this.proto = options.proto ?? {};
this.own = options.own ?? {};
Object.assign(this, options);
}
hasOwnProperties() {
for (const key in this.own) {
return true;
}
return false;
}
}
export interface CustomField {
header?: string;
extraHeaderIncludes?: string[];
impl?: string;
type?: string;
}
/**
* Define a native class written in ZIg. Bun's codegen step will create CPP wrappers
* for interacting with JSC.
*/
export function define(
{
klass = {},
proto = {},
own = {},
values = [],
overridesToJS = false,
estimatedSize = false,
call = false,
construct = false,
structuredClone,
...rest
} = {} as Partial<ClassDefinition>,
): ClassDefinition {
return new ClassDefinition({
...rest,
call,
overridesToJS,
construct,
estimatedSize,
structuredClone,
values,
own: own || {},
klass: Object.fromEntries(
Object.entries(klass)
.sort(([a], [b]) => a.localeCompare(b))
.map(([k, v]) => {
v.DOMJIT = undefined;
return [k, v];
}),
),
proto: Object.fromEntries(
Object.entries(proto)
.sort(([a], [b]) => a.localeCompare(b))
.map(([k, v]) => {
v.DOMJIT = undefined;
return [k, v];
}),
),
});
}