UPSTREAM PR #26908: java: limit numeric string length before BigDecimal parsing#141
Open
loci-dev wants to merge 2 commits into
Open
UPSTREAM PR #26908: java: limit numeric string length before BigDecimal parsing#141loci-dev wants to merge 2 commits into
loci-dev wants to merge 2 commits into
Conversation
BigDecimal(String) has O(N^2) time complexity for N-digit strings on JDK versions before 18 (JDK-8291514). The JSON parser passes user-controlled strings directly to BigDecimal in parseInt32, parseInt64, parseUint32, parseUint64, and parseDouble, allowing a single long numeric value to consume seconds of CPU. Add a parseBigDecimal helper that rejects strings longer than 1000 characters before constructing BigDecimal. This is generous — valid protobuf numeric values never exceed ~350 characters.
|
No meaningful performance changes were detected across 10167 analyzed functions in the following binaries: build.protoc-stable. 💬 Questions? Tag @loci-dev |
2b89f87 to
2f65a6e
Compare
String.repeat() was added in Java 11. Use a StringBuilder loop instead to maintain Java 8 compatibility per project requirements.
|
The analysis encountered an error. Please review the Processing Details for more information. |
fa0e9d9 to
a96792a
Compare
d1d44f5 to
f292971
Compare
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.This suggestion is invalid because no changes were made to the code.Suggestions cannot be applied while the pull request is closed.Suggestions cannot be applied while viewing a subset of changes.Only one suggestion per line can be applied in a batch.Add this suggestion to a batch that can be applied as a single commit.Applying suggestions on deleted lines is not supported.You must change the existing code in this line in order to create a valid suggestion.Outdated suggestions cannot be applied.This suggestion has been applied or marked resolved.Suggestions cannot be applied from pending reviews.Suggestions cannot be applied on multi-line comments.Suggestions cannot be applied while the pull request is queued to merge.Suggestion cannot be applied right now. Please check back later.
Note
Source pull request: protocolbuffers/protobuf#26908
Summary
BigDecimal(String)has O(N²) time complexity for N-digit strings on JDK versions before 18 (JDK-8291514). Five JSON parser methods —parseInt32,parseInt64,parseUint32,parseUint64, andparseDouble— pass user-controlled strings directly tonew BigDecimal()without length validation.A single JSON numeric value with 1,000,000 digits takes ~13 seconds to parse on JDK 17. This can be used to DoS any service that parses protobuf JSON messages with numeric fields from untrusted input.
Benchmark (JDK 17, x86-64 Linux)
Fix
Added a
parseBigDecimal()helper that rejects strings longer than 1000 characters before constructingBigDecimal. This is generous — valid protobuf numeric values never exceed ~350 characters (Double.MAX_VALUE in non-scientific notation is ~309 digits).Affected JDK versions
Test
Added
testParserRejectOverlyLongNumericStringscovering all 5 affected field types.