Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Use text-encoding npm package instead of integrated text-encoding dep. #50

Merged
merged 26 commits into from
Dec 4, 2021

Conversation

Uzlopak
Copy link
Contributor

@Uzlopak Uzlopak commented Nov 30, 2021

Replace encodings with external package, which provide the same features.

We should first check to get reliable benchmarks. The implementor of the external package claims, that it is a reference implementation and no focus on performance. But maybe it makes sense, to use a reference implementation to avoid some security issues?

Anyway... we should maybe not give up on the idea of caching the utf-8 TextDecoder and using it from node directly and just use the external TextDecoder as fallback.

Checklist

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

@kibertoad

What I realized is, that mscdex took the text encoding from https://github.com/inexorabletash/text-encoding and modified it for better node integration. (according to his remarks in the License Header).

The package text-decoding forked from that project and made it to be on spec and wrote more unit tests for it.

According to this benchmarks:
https://github.com/samthor/fast-text-encoding/tree/master/bench
the native TextDecoder is the fastest.

So what I did is:

  • added caching for the TextDecoder instances
  • Always instantiate the native TextDecoder for utf-8/utf8 as node always has utf-8 encoding.
  • If we get a "new" encoding, we try to instantiate a TextDecoder with the specified destEncoding, and if we are not having the encoding installed in the node instance we fall back to the external textdecoder.

So we get rid of the responsibility to keep the encodings up to date or the need to write tests for covering the code.

=============================== Coverage summary ===============================
Statements   : 93.24% ( 800/858 )
Branches     : 86.12% ( 447/519 )
Functions    : 95.38% ( 62/65 )
Lines        : 96.11% ( 718/747 )
================================================================================
uzlopak@uzlopak-Lenovo-Legion-5-17ARH05H:~/Workspace/fastify/busboy$ npm run bench:busboy

> @fastify/[email protected] bench:busboy
> node bench/fastify-busboy-bench.js

7142.86 mb/sec
9090.91 mb/sec
9090.91 mb/sec
9090.91 mb/sec
10000.00 mb/sec
10000.00 mb/sec
10000.00 mb/sec
9090.91 mb/sec
10000.00 mb/sec
10000.00 mb/sec
uzlopak@uzlopak-Lenovo-Legion-5-17ARH05H:~/Workspace/fastify/busboy$ npm run bench:dicer

> @fastify/[email protected] bench:dicer
> node bench/dicer/dicer-bench-multipart-parser.js

9090.91 mb/sec
12500.00 mb/sec
11111.11 mb/sec
11111.11 mb/sec
10000.00 mb/sec
11111.11 mb/sec
12500.00 mb/sec
12500.00 mb/sec
12500.00 mb/sec
11111.11 mb/sec

@Uzlopak Uzlopak changed the title replace encodings Use text-encoding npm package instead of integrated text-encoding dep. Dec 1, 2021
package.json Outdated Show resolved Hide resolved
@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

@kibertoad is the caching in an Object Ok? Or do you have a specific fastify way?

@kibertoad
Copy link
Member

Will work on benchmarks today, then we can measure :)

@kibertoad
Copy link
Member

Will also benchmark different ways to cache, see if Object has same performance as Map here

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

I think for benchmarking, we actually need to create a new benchmark, were we send a form with alot of fields (like a registration form).

The current benchmark is missing the point for this case, as it ist just about sending a big chunk of data as file. So there is no real necessity to call decodeText, what is touched in this PR.

@kibertoad
Copy link
Member

good point. Can you provide some samples of such forms? I could build a benchmark around that

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

@kibertoad added the necessary benchmarks

:)

This PR:

node ./bench/fastify-busboy-form-bench.js 
62.25 mb/sec
node ./bench/fastify-busboy-form-bench-utf8-only.js 
72.89 mb/sec

Current Master:

node ./bench/fastify-busboy-form-bench.js 
22.88 mb/sec
node ./bench/fastify-busboy-form-bench-utf8-only.js 
23.20 mb/sec

Old Busboy:

node ./bench/busboy-form-bench.js 
24.23 mb/sec
node ./bench/busboy-form-bench-utf8-only.js 
25.10 mb/sec

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

Ok, if the content of the part is also encoded specifically we get some interesting results :D

This PR:

node ./bench/fastify-busboy-form-bench-utf8.js 
64.95 mb/sec
node ./bench/fastify-busboy-form-bench-latin1.js 
58.19 mb/sec

master:

node ./bench/fastify-busboy-form-bench-latin1.js 

<--- Last few GCs --->

[27407:0x4deab50]    26983 ms: Mark-sweep (reduce) 4095.4 (4104.8) -> 4095.3 (4105.5) MB, 2090.1 / 0.0 ms  (+ 69.6 ms in 16 steps since start of marking, biggest step 17.3 ms, walltime since start of marking 2163 ms) (average mu = 0.088, current mu = 0.00[27407:0x4deab50]    29134 ms: Mark-sweep (reduce) 4096.3 (4102.5) -> 4096.1 (4104.0) MB, 2080.3 / 0.0 ms  (+ 67.0 ms in 15 steps since start of marking, biggest step 17.6 ms, walltime since start of marking 2151 ms) (average mu = 0.046, current mu = 0.00

<--- JS stacktrace --->

FATAL ERROR: Ineffective mark-compacts near heap limit Allocation failed - JavaScript heap out of memory
 1: 0xa25510 node::Abort() [node]
 2: 0x9664d3 node::FatalError(char const*, char const*) [node]
 3: 0xb9a8be v8::Utils::ReportOOMFailure(v8::internal::Isolate*, char const*, bool) [node]
 4: 0xb9ac37 v8::internal::V8::FatalProcessOutOfMemory(v8::internal::Isolate*, char const*, bool) [node]
 5: 0xd56ca5  [node]
 6: 0xd5782f  [node]
 7: 0xd6566b v8::internal::Heap::CollectGarbage(v8::internal::AllocationSpace, v8::internal::GarbageCollectionReason, v8::GCCallbackFlags) [node]
 8: 0xd6922c v8::internal::Heap::AllocateRawWithRetryOrFailSlowPath(int, v8::internal::AllocationType, v8::internal::AllocationOrigin, v8::internal::AllocationAlignment) [node]
 9: 0xd3790b v8::internal::Factory::NewFillerObject(int, bool, v8::internal::AllocationType, v8::internal::AllocationOrigin) [node]
10: 0x107fbef v8::internal::Runtime_AllocateInYoungGeneration(int, unsigned long*, v8::internal::Isolate*) [node]
11: 0x1426919  [node]
Aborted (core dumped)
node ./bench/fastify-busboy-form-bench-utf8.js 
23.43 mb/sec

Old Busboy:

node ./bench/busboy-form-bench-latin1.js 

<--- Last few GCs --->
00[26881:0x48ecb40]    29350 ms: Mark-sweep (reduce) 4098.4 (4107.0) -> 4098.2 (4107.8) MB, 2134.1 / 0.0 ms  (+ 74.7 ms in 16 steps since start of marking, biggest step 18.8 ms, walltime since start of marking 2212 ms) (average mu = 0.088, current mu = 0.00[26881:0x48ecb40]    31602 ms: Mark-sweep (reduce) 4099.2 (4104.8) -> 4099.1 (4106.5) MB, 2173.8 / 0.0 ms  (+ 74.5 ms in 16 steps since start of marking, biggest step 21.4 ms, walltime since start of marking 2252 ms) (average mu = 0.045, current mu = 0.00

<--- JS stacktrace --->

FATAL ERROR: Ineffective mark-compacts near heap limit Allocation failed - JavaScript heap out of memory
 1: 0xa25510 node::Abort() [node]
 2: 0x9664d3 node::FatalError(char const*, char const*) [node]
 3: 0xb9a8be v8::Utils::ReportOOMFailure(v8::internal::Isolate*, char const*, bool) [node]
 4: 0xb9ac37 v8::internal::V8::FatalProcessOutOfMemory(v8::internal::Isolate*, char const*, bool) [node]
 5: 0xd56ca5  [node]
 6: 0xd5782f  [node]
 7: 0xd6566b v8::internal::Heap::CollectGarbage(v8::internal::AllocationSpace, v8::internal::GarbageCollectionReason, v8::GCCallbackFlags) [node]
 8: 0xd6922c v8::internal::Heap::AllocateRawWithRetryOrFailSlowPath(int, v8::internal::AllocationType, v8::internal::AllocationOrigin, v8::internal::AllocationAlignment) [node]
 9: 0xd3790b v8::internal::Factory::NewFillerObject(int, bool, v8::internal::AllocationType, v8::internal::AllocationOrigin) [node]
10: 0x107fbef v8::internal::Runtime_AllocateInYoungGeneration(int, unsigned long*, v8::internal::Isolate*) [node]
11: 0x1426919  [node]
Aborted (core dumped)
node ./bench/busboy-form-bench-utf8.js 
24.43 mb/sec

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

I tested with node 14.17.6

So this PR makes busboy even more resilient, yeah :)

@kibertoad
Copy link
Member

Can we also add tests based on same payloads?

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

But these are based on the same payload?

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

@kibertoad
extracted the createMultipartBufferForFormStream

As long as the parameters are the same they generate the same Lorem Ipsum payload.

@kibertoad
Copy link
Member

But these are based on the same payload?

I don't mean using same payload for all benchmarks (although extracting it was a nice touch), I was just wondering if we have code coverage for same kind of payloads in our unit tests, because you produced them specifically for benchmarks, and I'm not sure if we had similar ones in unit tests.

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 1, 2021

@kibertoad
It is actually a modification of parse-params.spec "Multiple extended parameters (RFC 5987) with mixed charsets"

Fair enough, We dont have a unit test, which does the same. Actually the current test is kind of crazy, as I create about 100000 fields :D.

I will create a unit test, like that benchmark.

@kibertoad
Copy link
Member

@Uzlopak I think it's correct now, can you check?

@kibertoad
Copy link
Member

@Uzlopak After (and if) this lands and I do #60, I'm thinking of starting to prep the 1.0.0 release. Anything else you want in prior to that?

@Fdawgs
Copy link
Member

Fdawgs commented Dec 3, 2021

@Uzlopak After (and if) this lands and I do #60, I'm thinking of starting to prep the 1.0.0 release. Anything else you want in prior to that?

Is it worth dropping support for node 10 and 12 now before v1.0.0 release, considering Fastify v4 will most likely do the same?

@kibertoad
Copy link
Member

@Fdawgs What is the advantage of dropping version support early if it doesn't affect our code or dependencies? Can't we just release a semver major later if we ever need to?
multer is going to be another major user of this, and they are 12+, so I think we can drop 10 already if there is a good reason to.

@Fdawgs
Copy link
Member

Fdawgs commented Dec 3, 2021

@kibertoad Cool beans, good point. Was just to avoid a future major release really.

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 3, 2021

@kibertoad
Are these benchmarks ok?

I reduced the size of the payload, as nodejs could not handle the backpressure and clogged the event loop so OOM was inevitable.

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 3, 2021

@kibertoad
Strange there is still a memory leak.

@kibertoad
Copy link
Member

I'll take a look in the evening.

@kibertoad
Copy link
Member

@LinusU Any preferences on Node 10 vs Node 12 baseline? I assume you don't intend to ever raise the Node version requirement for old api Multer, so it can't use fastify-busboy either way, and new api will be Node 12 from the get go, right?

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 3, 2021

@kibertoad
I think the memory leak is a common issue and was introduced before. I never get a busboy finish event.

see
mscdex/busboy#229

Nothing to do with this, i think

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 3, 2021

@kibertoad
I guess, this happens because backpressure is not handled correctly by busboy. This is a whole different can of worms. :/

@LinusU
Copy link
Contributor

LinusU commented Dec 3, 2021

@LinusU Any preferences on Node 10 vs Node 12 baseline? I assume you don't intend to ever raise the Node version requirement for old api Multer, so it can't use fastify-busboy either way, and new api will be Node 12 from the get go, right?

Multer 2.x currently only supports ^12.20.0 || ^14.13.1 || >=16.0.0 since we have ESM only dependencies, so 12 sounds good 👍

Multer 1.x supports Node.js 0.10 so we won't be able to upgrade there 😅

@Uzlopak
Copy link
Contributor Author

Uzlopak commented Dec 4, 2021

@kibertoad

anything you need from me to approve this PR :)?

@kibertoad
Copy link
Member

just want to try it out hands-on a bit, will approve after :).
hit my head on corpo party yesterday, so wasn't able to be productive yesterday, and today is a date night, but I hope to wrap up everything on Sunday and hopefully release 1.0.0 already :)

@Uzlopak Uzlopak mentioned this pull request Dec 4, 2021
2 tasks
Copy link
Member

@kibertoad kibertoad left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM, but we need to look into the memory leak problem eventually

@Uzlopak Uzlopak merged commit 640f5ab into fastify:master Dec 4, 2021
@KhafraDev KhafraDev mentioned this pull request Aug 13, 2023
@Uzlopak Uzlopak deleted the replace-encoding branch August 13, 2023 19:57
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants