Skip to content

Commit 3d501f8

Browse files
authoredMar 4, 2025··
Fix json deserialization error (#320)
* Fix JSON deserialization error in SmartSerializer Signed-off-by: takuyanakazawa <t.nakazawa0222@gmail.com> * test: modify assert Signed-off-by: takuyanakazawa <t.nakazawa0222@gmail.com> * Fix checking for json content type when decoding Signed-off-by: takuyanakazawa <t.nakazawa0222@gmail.com> * Add a test for missing Content-Type header Signed-off-by: takuyanakazawa <t.nakazawa0222@gmail.com> * Fixed header case normalization and content type checks Signed-off-by: takuyanakazawa <t.nakazawa0222@gmail.com> --------- Signed-off-by: takuyanakazawa <t.nakazawa0222@gmail.com>
1 parent 877796e commit 3d501f8

File tree

3 files changed

+79
-9
lines changed

3 files changed

+79
-9
lines changed
 

‎CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ Inspired from [Keep a Changelog](https://keepachangelog.com/en/1.0.0/)
1111
### Deprecated
1212
### Removed
1313
### Fixed
14+
- Fixed checking for content type in JSON deserialization ([#318](https://github.com/opensearch-project/opensearch-php/issues/318))
1415
- Fixed mismatch in return types between `Client::performRequest()` and `Transport::sendRequest()` ([#307](https://github.com/opensearch-project/opensearch-php/issues/307))
1516
- Fixed legacy client options being passed as headers ([#301](https://github.com/opensearch-project/opensearch-php/issues/301))
1617
- Fixed endpoint options not being passed to legacy transport ([#296](https://github.com/opensearch-project/opensearch-php/issues/296))

‎src/OpenSearch/Serializers/SmartSerializer.php

+29-9
Original file line numberDiff line numberDiff line change
@@ -56,17 +56,10 @@ public function serialize($data): string
5656
*/
5757
public function deserialize(?string $data, array $headers)
5858
{
59-
if (isset($headers['content_type']) === true) {
60-
if (strpos($headers['content_type'], 'json') !== false) {
61-
return $this->decode($data);
62-
} else {
63-
//Not json, return as string
64-
return $data;
65-
}
66-
} else {
67-
//No content headers, assume json
59+
if ($this->isJson($headers)) {
6860
return $this->decode($data);
6961
}
62+
return $data;
7063
}
7164

7265
/**
@@ -86,4 +79,31 @@ private function decode(?string $data): array
8679
throw new JsonException($e->getCode(), $data, $e);
8780
}
8881
}
82+
83+
/**
84+
* Check the response content type to see if it is JSON.
85+
*
86+
* @param array<string,mixed> $headers
87+
*/
88+
private function isJson(array $headers): bool
89+
{
90+
// Legacy support for 'transfer_stats'.
91+
if (array_key_exists('content_type', $headers)) {
92+
return str_contains($headers['content_type'], 'json');
93+
}
94+
95+
// Check PSR-7 headers.
96+
$lowercaseHeaders = array_change_key_case($headers, CASE_LOWER);
97+
if (array_key_exists('content-type', $lowercaseHeaders)) {
98+
foreach ($lowercaseHeaders['content-type'] as $type) {
99+
if (str_contains($type, 'json')) {
100+
return true;
101+
}
102+
}
103+
return false;
104+
}
105+
106+
// No content type header, so assume it is JSON.
107+
return true;
108+
}
89109
}

‎tests/Serializers/SmartSerializerTest.php

+49
Original file line numberDiff line numberDiff line change
@@ -58,4 +58,53 @@ public function testThrowJsonException(): void
5858
}
5959
}
6060

61+
public function testDeserialize(): void
62+
{
63+
$data = '{ "foo" : "bar" }';
64+
$headers = ['Content-Type' => ['application/json']];
65+
66+
$result = $this->serializer->deserialize($data, $headers);
67+
68+
$this->assertEquals('bar', $result['foo']);
69+
}
70+
71+
public function testDeserializeLowercaseContentTypeHeader(): void
72+
{
73+
$data = '{ "foo" : "bar" }';
74+
$headers = ['content-type' => ['application/json']];
75+
76+
$result = $this->serializer->deserialize($data, $headers);
77+
78+
$this->assertEquals('bar', $result['foo']);
79+
}
80+
81+
public function testDeserializeWithLegacyContentTypeHeader(): void
82+
{
83+
$data = '{ "foo" : "bar" }';
84+
$headers = ['content_type' => 'application/json'];
85+
86+
$result = $this->serializer->deserialize($data, $headers);
87+
88+
$this->assertEquals('bar', $result['foo']);
89+
}
90+
91+
public function testDeserializeNonJsonData(): void
92+
{
93+
$data = 'plain text data';
94+
$headers = ['Content-Type' => ['text/plain']];
95+
96+
$result = $this->serializer->deserialize($data, $headers);
97+
98+
$this->assertEquals('plain text data', $result);
99+
}
100+
101+
public function testDeserializeWithNoContentTypeHeader(): void
102+
{
103+
$data = '{ "foo" : "bar" }';
104+
$headers = [];
105+
106+
$result = $this->serializer->deserialize($data, $headers);
107+
108+
$this->assertEquals('bar', $result['foo']);
109+
}
61110
}

0 commit comments

Comments
 (0)
Please sign in to comment.