Skip to content

Commit

Permalink
Fixes #7827
Browse files Browse the repository at this point in the history
  • Loading branch information
Jarred-Sumner committed Dec 27, 2023
1 parent f8159f1 commit 8a5f5c6
Show file tree
Hide file tree
Showing 3 changed files with 79 additions and 28 deletions.
45 changes: 17 additions & 28 deletions src/bun.js/api/html_rewriter.zig
Original file line number Diff line number Diff line change
Expand Up @@ -16,38 +16,43 @@ pub const LOLHTMLContext = struct {
selectors: SelectorMap = .{},
element_handlers: std.ArrayListUnmanaged(*ElementHandler) = .{},
document_handlers: std.ArrayListUnmanaged(*DocumentHandler) = .{},
ref_count: u32 = 1,

pub fn deinit(this: *LOLHTMLContext, allocator: std.mem.Allocator) void {
pub usingnamespace bun.NewRefCounted(@This(), deinit);

fn deinit(this: *LOLHTMLContext) void {
for (this.selectors.items) |selector| {
selector.deinit();
}
this.selectors.deinit(allocator);
this.selectors.deinit(bun.default_allocator);
this.selectors = .{};

for (this.element_handlers.items) |handler| {
handler.deinit();
}
this.element_handlers.deinit(allocator);
this.element_handlers.deinit(bun.default_allocator);
this.element_handlers = .{};

for (this.document_handlers.items) |handler| {
handler.deinit();
}
this.document_handlers.deinit(allocator);
this.document_handlers.deinit(bun.default_allocator);
this.document_handlers = .{};

this.destroy();
}
};
pub const HTMLRewriter = struct {
builder: *LOLHTML.HTMLRewriter.Builder,
context: LOLHTMLContext,
context: *LOLHTMLContext,

pub usingnamespace JSC.Codegen.JSHTMLRewriter;

pub fn constructor(_: *JSGlobalObject, _: *JSC.CallFrame) callconv(.C) ?*HTMLRewriter {
const rewriter = bun.default_allocator.create(HTMLRewriter) catch unreachable;
rewriter.* = HTMLRewriter{
.builder = LOLHTML.HTMLRewriter.Builder.init(),
.context = .{},
.context = LOLHTMLContext.new(.{}),
};
return rewriter;
}
Expand Down Expand Up @@ -152,13 +157,13 @@ pub const HTMLRewriter = struct {
}

pub fn finalizeWithoutDestroy(this: *HTMLRewriter) void {
this.context.deinit(bun.default_allocator);
this.context.deref();
this.builder.deinit();
}

pub fn beginTransform(this: *HTMLRewriter, global: *JSGlobalObject, response: *Response) JSValue {
const new_context = this.context;
this.context = .{};
new_context.ref();
return BufferOutputSink.init(new_context, global, response, this.builder);
}

Expand Down Expand Up @@ -298,17 +303,10 @@ pub const HTMLRewriter = struct {
pub fn setup(
this: *HTMLRewriterLoader,
builder: *LOLHTML.HTMLRewriter.Builder,
context: LOLHTMLContext,
context: *LOLHTMLContext,
size_hint: ?usize,
output: JSC.WebCore.Sink,
) ?[]const u8 {
for (context.document_handlers.items) |doc| {
doc.ctx = this;
}
for (context.element_handlers.items) |doc| {
doc.ctx = this;
}

const chunk_size = @max(size_hint orelse 16384, 1024);
this.rewriter = builder.build(
.UTF8,
Expand Down Expand Up @@ -392,13 +390,13 @@ pub const HTMLRewriter = struct {
global: *JSGlobalObject,
bytes: bun.MutableString,
rewriter: ?*LOLHTML.HTMLRewriter = null,
context: LOLHTMLContext,
context: *LOLHTMLContext,
response: *Response,
response_value: JSC.Strong = .{},
bodyValueBufferer: ?JSC.WebCore.BodyValueBufferer = null,
tmp_sync_error: ?*JSC.JSValue = null,
// const log = bun.Output.scoped(.BufferOutputSink, false);
pub fn init(context: LOLHTMLContext, global: *JSGlobalObject, original: *Response, builder: *LOLHTML.HTMLRewriter.Builder) JSC.JSValue {
pub fn init(context: *LOLHTMLContext, global: *JSGlobalObject, original: *Response, builder: *LOLHTML.HTMLRewriter.Builder) JSC.JSValue {
var sink = bun.new(BufferOutputSink, BufferOutputSink{
.global = global,
.bytes = bun.MutableString.initEmpty(bun.default_allocator),
Expand All @@ -422,13 +420,6 @@ pub const HTMLRewriter = struct {

sink.response = result;

for (sink.context.document_handlers.items) |doc| {
doc.ctx = sink;
}
for (sink.context.element_handlers.items) |doc| {
doc.ctx = sink;
}

const input_size = original.body.len();
sink.rewriter = builder.build(
.UTF8,
Expand Down Expand Up @@ -616,7 +607,7 @@ pub const HTMLRewriter = struct {
bufferer.deinit();
}

this.context.deinit(bun.default_allocator);
this.context.deref();
this.response_value.deinit();
if (this.rewriter) |rewriter| {
rewriter.deinit();
Expand Down Expand Up @@ -752,7 +743,6 @@ const DocumentHandler = struct {
onEndCallback: ?JSValue = null,
thisObject: JSValue,
global: *JSGlobalObject,
ctx: ?*HTMLRewriter.BufferOutputSink = null,

pub const onDocType = HandlerCallback(
DocumentHandler,
Expand Down Expand Up @@ -928,7 +918,6 @@ const ElementHandler = struct {
onTextCallback: ?JSValue = null,
thisObject: JSValue,
global: *JSGlobalObject,
ctx: ?*HTMLRewriter.BufferOutputSink = null,

pub fn init(global: *JSGlobalObject, thisObject: JSValue) !ElementHandler {
var handler = ElementHandler{
Expand Down
40 changes: 40 additions & 0 deletions src/bun.zig
Original file line number Diff line number Diff line change
Expand Up @@ -2862,6 +2862,46 @@ pub fn New(comptime T: type) type {
};
}

pub fn NewRefCounted(comptime T: type, comptime deinit_fn: ?fn (self: *T) void) type {
return struct {
pub inline fn destroy(self: *T) void {
if (comptime is_heap_breakdown_enabled) {
HeapBreakdown.allocator(T).destroy(self);
} else {
default_allocator.destroy(self);
}
}

pub fn ref(self: *T) void {
self.ref_count += 1;
}

pub fn deref(self: *T) void {
self.ref_count -= 1;

if (self.ref_count == 0) {
if (comptime deinit_fn) |deinit| {
deinit(self);
} else {
self.destroy();
}
}
}

pub inline fn new(t: T) *T {
if (comptime is_heap_breakdown_enabled) {
const ptr = HeapBreakdown.allocator(T).create(T) catch outOfMemory();
ptr.* = t;
return ptr;
}

const ptr = default_allocator.create(T) catch outOfMemory();
ptr.* = t;
return ptr;
}
};
}

/// Free a globally-allocated a value.
///
/// Must have used `new` to allocate the value.
Expand Down
22 changes: 22 additions & 0 deletions test/regression/issue/07827.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
import { test, expect, jest } from "bun:test";

test("#7827", () => {
for (let i = 0; i < 10; i++)
(function () {
const element = jest.fn(element => {
element.tagName;
});
const rewriter = new HTMLRewriter().on("p", {
element,
});

const content = "<p>Lorem ipsum!</p>";

rewriter.transform(new Response(content));
rewriter.transform(new Response(content));

expect(element).toHaveBeenCalledTimes(2);
})();

Bun.gc(true);
});

0 comments on commit 8a5f5c6

Please sign in to comment.