-
Notifications
You must be signed in to change notification settings - Fork 14.1k
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
Kafka 17792 header parsing times out processing and using large quantities of memory if the string looks like a number #17510
base: trunk
Are you sure you want to change the base?
Conversation
Thanks for the PR. Please make sure it only contains changes related to the issue this is addressing. Currently I see a lot of other changes including some non-Apache Kafka files. |
dd47ebc
to
3720951
Compare
I'm not quite sure what I've done wrong, I cloned the repo and applied my change again and forced pushed it to my fork and it still looks wrong The change is here msillence@3720951 |
Oh now I see the problem I've mixed up repos this change is against github.com:confluentinc/kafka |
3720951
to
4191d5a
Compare
actually there is a simpler solution https://stackoverflow.com/a/12748321 |
f8af0e1
to
4b4a8f1
Compare
I feel like the correct fix is the stackoverflow test: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@msillence Thanks for the PR!
I did not know that setScale with very large numbers was so costly. I see that the old logic (before #15469) used RoundingMode.ROUND_UNNECESSARY, which probably had some guards in place to avoid consuming resources for an impossible operation.
sadly there are quite a few tests that expect the perhaps incorrect behavior where Float.MAX_VALUE which is 3.4028235E38 looks like a integer to me but the ceil/floor calls failed due to the number of zeros so it fell though to another part of the parsing and returned a float.
That number is indeed an integer, but it's too large to fit in a LONG64, which has ~19 decimals of precision. The tests are correct that this should be parsed as a FLOAT32.
While I understand that having mostly STRINGs with an occasional number-look-alike makes it natural to suggest changing this into a STRING, I don't think that's backwards compatible or safe. Another user could currently have a flow which parses all Decimals, and the current implementation would have some of their decimals fall back to STRING.
To avoid accidental Decimal parsing, I would recommend either an alternative HeaderConverter implementation, or avoiding using IDs that can be confused for numbers, such as with an alphabetic prefix. Even with this patch, you will still get some accidental Decimal, Float, or Double parsing when the e
lands in some part of the value.
@@ -1041,6 +1043,10 @@ private static SchemaAndValue parseAsNumber(String token) { | |||
} | |||
|
|||
private static SchemaAndValue parseAsExactDecimal(BigDecimal decimal) { | |||
BigDecimal abs = decimal.abs(); | |||
if (abs.compareTo(TOO_BIG) > 0 || (abs.compareTo(TOO_SMALL) < 0 && BigDecimal.ZERO.compareTo(abs) != 0)) { | |||
throw new NumberFormatException("outside efficient parsing range"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
At this stage in parsing, we've successfully parsed a BigDecimal. This method tries to losslessly cast the BigDecimal to a primitive integer type. If the cast would be lossy, it returns null and lets the caller figure out whether it's a float32, float64, or has to be an arbitrary precision decimal.
Rather than throwing NumberFormatException and making this a STRING, we should return null, and let the other fallback logic happen, so this shows up as an arbitrary precision Decimal.
I also suspect that the comparison could be a little tighter in that case: TOO_BIG could be LONG_MAX, and TOO_SMALL could be ONE, eliminating calling setScale for 1e20
which cannot possibly fit in a LONG.
@gharris1727 , do you have any further comments for this? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@msillence Thanks for changing this to fallback to the Float/Double/Decimal logic! We're interested in getting this into the upcoming release, so please let us know if you're able to update the PR soon.
@showuon This PR is currently not mergeable and would cause build/test failures. But the changes needed to get it to mergeable seem minor.
private static BigDecimal TOO_BIG = new BigDecimal("1e1000000"); | ||
private static BigDecimal TOO_SMALL = new BigDecimal("1e-1000000"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These constants are unused.
public void avoidCpuAndMemoryIssuesConvertingExtremeBigDecimals() { | ||
long s = System.currentTimeMillis(); | ||
String veryBig = "1e+100000000"; // new BigDecimal().setScale(0, RoundingMode.FLOOR) takes around two minutes and uses 3GB; | ||
System.out.println("time taken" + System.currentTimeMillis() - s) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we should remove this debug print, and have a Timeout annotation on this method to cause fails on regressions.
|
||
String verySmall = "1e-100000000"; | ||
assertEquals(new SchemaAndValue(Schema.STRING_SCHEMA, verySmall), Values.parseString(verySmall)); | ||
assertFalse(true); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This will always fail.
|
||
private static SchemaAndValue parseAsExactDecimal(BigDecimal decimal) { | ||
BigDecimal abs = decimal.abs(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This value is unused.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I've addressed this now
@@ -1039,30 +1041,34 @@ private static SchemaAndValue parseAsNumber(String token) { | |||
return new SchemaAndValue(schema, decimal); | |||
} | |||
} | |||
|
|||
public static boolean isBigInteger(BigDecimal bd) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This uses tabs, which is disallowed by our linter.
long s = System.currentTimeMillis(); | ||
String veryBig = "1e+100000000"; // new BigDecimal().setScale(0, RoundingMode.FLOOR) takes around two minutes and uses 3GB; | ||
System.out.println("time taken" + System.currentTimeMillis() - s) | ||
assertEquals(new SchemaAndValue(Schema.BYTES_SCHEMA, veryBig), Values.parseString(veryBig)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This shouldn't be a plain BYTES schema, it should be a connect Decimal
type. And veryBig needs to be parsed into a BigDecimal.
|
||
String verySmall = "1e-100000000"; | ||
assertEquals(new SchemaAndValue(Schema.STRING_SCHEMA, verySmall), Values.parseString(verySmall)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This actually gets parsed as a FLOAT32. Arguably that's a bug and we're losing precision, but it's the legacy behavior and we shouldn't change it, or at least have a separate change for this. For now, we should just update this test to match the behavior.
@@ -45,6 +41,8 @@ | |||
import static org.junit.jupiter.api.Assertions.assertThrows; | |||
import static org.junit.jupiter.api.Assertions.assertTrue; | |||
import static org.junit.jupiter.api.Assertions.fail; | |||
import org.junit.jupiter.api.Test; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If you run into problems with the imports, run ./gradlew spotlessApply
@@ -1039,30 +1041,34 @@ private static SchemaAndValue parseAsNumber(String token) { | |||
return new SchemaAndValue(schema, decimal); | |||
} | |||
} | |||
|
|||
public static boolean isBigInteger(BigDecimal bd) { | |||
return bd.signum() == 0 || bd.scale() <= 0 || bd.stripTrailingZeros().scale() <= 0; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This stripTrailingZeros covers the cases that the ceil/floor comparison was supposed to cover, great find!
@@ -1154,7 +1152,17 @@ public void shouldParseFractionalPartsAsIntegerWhenNoFractionalPart() { | |||
assertEquals(new SchemaAndValue(Schema.INT32_SCHEMA, 66000), Values.parseString("66000.0")); | |||
assertEquals(new SchemaAndValue(Schema.FLOAT32_SCHEMA, 66000.0008f), Values.parseString("66000.0008")); | |||
} | |||
@Test |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: missing a newline above this test.
Sorry I'd been busy with other things. I took another look and I think I've worked out the issues with the long durations now and eliminated all the ceil/floor calls |
…ities of memory if the string looks like a number
…ities of memory if the string looks like a number
ca0750d
to
b407e58
Compare
I rebased and the build is now clean |
Jira KAFKA-17792
We have trace headers such as:
"X-B3-SpanId": "74320e6e26adc8f8"
if however the value happens to be: "407127e212797209"
This is then treated as a numeric value and it tries to convert this as a numeric representation and an exact value using BigDecimal
Specifically the calls to setScale with floor and ceiling seem to cause the problem
testing this the behaviour of biginteger and setScale calls shows there are already limits on the number range possible sadly my particular number falls in the range of about 25 minutes and 4 gig of memory rather than quick failure
1e+1 time 0.0 totalMemory 532676608
1e+10 time 0.0 totalMemory 532676608
1e+100 time 0.0 totalMemory 532676608
1e+1000 time 0.0 totalMemory 532676608
1e+10000 time 0.005 totalMemory 532676608
1e+100000 time 0.035 totalMemory 532676608
1e+1000000 time 0.228 totalMemory 532676608
1e+10000000 time 4.308 totalMemory 926941184
1e+100000000 time 117.119 totalMemory 3221225472
1e+1000000000 time 0.0 totalMemory 3221225472 BigInteger would overflow supported range
1e+10000000000 time 0.001 totalMemory 3221225472 Too many nonzero exponent digits.
1e+100000000000 time 0.0 totalMemory 3221225472 Too many nonzero exponent digits.
Committer Checklist (excluded from commit message)