-
Notifications
You must be signed in to change notification settings - Fork 736
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
json_decode() and integer overflow strike back #1003
Comments
@theDisco Which PHP version are you using? |
@ruflin here it is:
|
@theDisco Very interesting. I found some more information here which could be related: http://stackoverflow.com/questions/19520487/json-bigint-as-string-removed-in-php-5-5 |
@ruflin haha, that is one of the links I have provided in my initial comment :) Nevertheless it doesn't explain why 2) is failing and why 3) is implemented as it is? |
@theDisco Here a more detailed answer:
Thanks a lot for the detailed reporting and also the suggested solutions. Sorry again about my brief answer in the beginning. |
Interesting find, thanks for reporting this @theDisco.
if ($this->getJsonBigintConversion() && defined('JSON_BIGINT_AS_STRING')) {
$response = JSON::parse($response, false, 512, JSON_BIGINT_AS_STRING);
} else {
$response = JSON::parse($response);
}
|
The main reason I don't like the suggestion for 1) above is that the user doesn't get what he configured. Elastica as a library should make sure not to do magic and the the developer gets what he configured. If he doesn't he should get an exception. For 2) Thanks @oldskool for bringing that about. I remember I strongly suggested this change to have the default behaviour. What I'm asking myself now is what exactly goes wrong in 2, as this then should be unrelated to 1). It would also fail without JSON_BIGING_*? |
$config = [
'host' => '127.0.0.1',
'port' => 9200,
'bigintConversion' => true,
];
new \Elastica\Client($config); In this case it does: $config = [
'host' => '127.0.0.1',
'port' => 9200,
'config' => [
'bigintConversion' => true,
],
];
new \Elastica\Client($config); After debugging a little bit more, it seems bigintConversion has to be added here https://github.com/ruflin/Elastica/blob/master/lib/Elastica/Client.php#L121: /**
* Creates a Connection params array from a Client or server config array.
*
* @param array $config
*
* @return array
*/
protected function _prepareConnectionParams(array $config)
{
$params = array();
$params['config'] = array();
foreach ($config as $key => $value) {
if (in_array($key, array('curl', 'headers', 'url'))) {
$params['config'][$key] = $value;
} else {
$params[$key] = $value;
}
}
return $params;
} |
About 3, I think the confusing part is that the current implementation uses the transport(connection) object to set the variable for response object. This works but is ugly as it has to be set multiple times and is confusing as we see. The problem is: What is the better solution? I'm quite oposed to set global variables. So the question is how do we get it best from the client to the response object (which doesn't no about the client ...). |
I don't think it is that bad, because the client acts as a factory and uses configuration to set up the underlying ecosystem. Transport should be primitive and only responsible for transferring the data between two endpoints. The response creation can be moved to the request object and request object can be composed of serializers which take care of json transformation. This way you would completely decouple the static calls to JSON object. Nevertheless this implementation exceeds the scope of the problem we are dealing with in here. I also think all of that is not necessary if you are going to replace the communication layer with php-elasticsearch (as stated in #944 and #989). |
@theDisco You are correct, especially with your last statement. Going with #944 will move the problem one layer up. I'm sure @polyfractal likes that :-D |
Urgh. Json encoding is the bane of my existence in PHP. This issue came up at es-php recently too: elastic/elasticsearch-php#334 :'( I'm not really sure the best way forward. As mentioned here, Personally, this feels like something that should be handled outside the library, either by the user dealing with BigInts ahead of time (since ES doesn't support BigInts either), or by subclassing the Serializer and providing one that is BigInt-aware (because you know you have BigInt support). That said, I could add a BigIntSerializer, or ability to specify custom Open to suggestions :) EDIT: Realized this is for json_decode, not encode. Meaning you dont have a chance to escape yourself. Sigh. Will think about the best of the options :( |
I think detection, if BigInt is supported or not is a good thing. The goal of libraries like elastica or es-php is to hide the details of low or even high level implementations and let the user concentrate on the use case. I like the simplicity of just instantiating After saying that I think it is a good thing to have 2 serializers, one with BigInt support and one without. I would add a detection for BigInt support by default and let the user overwrite the serializer, if necessary. After all you want to solve things for the user and not make them more complicated as they already are. You should also think about aligning with the requirements, es-php is >= 5.3.9, elastica >= 5.4.0. Let me know, what you think about that? |
Regarding versions, ES-PHP 2.0 is explicitly PHP 5.4+, so that will be fine. I'm trying to move people off ES-PHP 1.x as soon as possible :) I'll mull over the options and work something out. I agree this is a less than ideal situation for the end-user, who doesn't really care about how numbers are encoded at all. And I agree this should probably be handled at the ES-PHP level, not Elastica :) |
@polyfractal Yep, fun problem. Until yesterday I thought JSON_BIGINT_AS_STRING is "solved" by going with PHP 5.4 but @theDisco proofed me wrong :-( |
…bigintConversion config parameter to the connection, provide tests for both
PR is ready, when can I expect it to become available through composer? |
@theDisco Just commented on it. What are you referencing to in composer? Master? |
@ruflin Sorry, let me clarify, when will you create a tag I will be able to configure in composer.json? I haven't found any information on how you deal with versioning... |
@ruflin I was rather thinking about 3.0.0-beta2 :) |
The other thing I'm thinking about is actually going with 3.0.0 to bring the adoption to discover more bugs. I think currently lots of user stick with the old version. |
@ruflin This works out really well for me, but you need to think about people, who are still using facets instead of aggregations. They will have to invest some time to refactor their code. You are completely right though, people should be encouraged to upgrade to newer versions. |
@theDisco Unfortunately it is much worse then people just using Facets. It's also about the PHP version and some people use even older ES versions like 0.90.0 ... I'm closing this issue as the PR was merged. Thanks again also for bringing up all the points and take the time to discuss it. |
Thanks for taking your time and merging it so quick! |
… bigintConversion config parameter to the connection, provide tests for both
… bigintConversion config parameter to the connection, provide tests for both
I have just installed version 3.0.0-beta1 after reading #717 and #941 in hope to get the integer overflow problem in json_decode() sorted out. After some debugging I have found following problems:
JSON_BIGINT_AS_STRING
is not implemented throwing a warning[Warning] json_decode(): option JSON_BIGINT_AS_STRING not implemented
. Below my version table:After upgrading to 1.3.9 json_decode() does not complain about
JSON_BIGINT_AS_STRING
anymore. Maybe you should look at how firebase/php-jwt (found here) has solved the problem and implement it similarly?After sorting out the problem with json_decode() and
JSON_BIGINT_AS_STRING
I get following error nowFatal error: Cannot use object of type stdClass as array in /srv/app/vendor/ruflin/elastica/lib/Elastica/Response.php on line 121
which can be traced to this line. Fix is very simple, just replace false with true, but what is more important, I couldn't find a test for it anywhere in the test directory. I am more than happy to provide a PR for that.A smaller thing, but it would save me a lot of time in debugging. The default definition of
bigintConversion
config variable from here should be actually listed underconfig
key, since it is obtained by the connection from the config in here. I can provide tests for that too.Can you please confirm my findings?
The text was updated successfully, but these errors were encountered: