Skip to content

Commit e5f4716

Browse files
committed
Implement GH-15711: Allow SoapClient to use the backing value during response serialization (int enums)
1 parent b9b83b9 commit e5f4716

File tree

3 files changed

+62
-14
lines changed

3 files changed

+62
-14
lines changed

ext/soap/php_encoding.c

Lines changed: 25 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -850,6 +850,30 @@ static zend_string *get_serialization_string_from_zval(zval *data)
850850
}
851851
}
852852

853+
static zend_long get_serialization_long_from_zval(zval *data)
854+
{
855+
switch (Z_TYPE_P(data)) {
856+
case IS_OBJECT:
857+
if (Z_OBJCE_P(data)->ce_flags & ZEND_ACC_ENUM) {
858+
if (UNEXPECTED(Z_OBJCE_P(data)->enum_backing_type != IS_LONG)) {
859+
if (Z_OBJCE_P(data)->enum_backing_type == IS_UNDEF) {
860+
zend_value_error("Non-backed enums have no default serialization");
861+
} else {
862+
zend_value_error("String-backed enum cannot be serialized as int");
863+
}
864+
return 0;
865+
} else {
866+
zval *value = zend_enum_fetch_case_value(Z_OBJ_P(data));
867+
ZEND_ASSERT(Z_TYPE_P(value) == IS_LONG);
868+
return Z_LVAL_P(value);
869+
}
870+
}
871+
ZEND_FALLTHROUGH;
872+
default:
873+
return zval_get_long(data);
874+
}
875+
}
876+
853877
static xmlNodePtr to_xml_string(encodeTypePtr type, zval *data, int style, xmlNodePtr parent)
854878
{
855879
xmlNodePtr ret, text;
@@ -1075,7 +1099,7 @@ static xmlNodePtr to_xml_long(encodeTypePtr type, zval *data, int style, xmlNode
10751099
snprintf(s, sizeof(s), "%0.0F",floor(Z_DVAL_P(data)));
10761100
xmlNodeSetContent(ret, BAD_CAST(s));
10771101
} else {
1078-
zend_string *str = zend_long_to_str(zval_get_long(data));
1102+
zend_string *str = zend_long_to_str(get_serialization_long_from_zval(data));
10791103
xmlNodeSetContentLen(ret, BAD_CAST(ZSTR_VAL(str)), ZSTR_LEN(str));
10801104
zend_string_release_ex(str, 0);
10811105
}

ext/soap/tests/gh15711.phpt

Lines changed: 35 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ soap.wsdl_cache_enabled=0
77
--FILE--
88
<?php
99

10-
enum TestBackedEnum: string
10+
enum StringBackedEnum: string
1111
{
1212
case First = 'BackingValue1';
1313
case Second = 'BackingValue2';
@@ -16,7 +16,13 @@ enum TestBackedEnum: string
1616
case Fifth = 'BackingValue5';
1717
}
1818

19-
enum TestNonBackedEnum
19+
enum IntBackedEnum: int
20+
{
21+
case First = 1;
22+
case Second = 2;
23+
}
24+
25+
enum NonBackedEnum
2026
{
2127
case First;
2228
}
@@ -32,11 +38,13 @@ $client = new TestSoapClient('ext/soap/tests/gh15711.wsdl', ['classmap' => ['boo
3238
echo "--- Test with backed enum ---\n";
3339

3440
$book = new stdClass();
35-
$book->base64 = TestBackedEnum::First;
36-
$book->string = TestBackedEnum::Second;
37-
$book->any = TestBackedEnum::Third;
38-
$book->hexbin = TestBackedEnum::Fourth;
39-
$book->nmtokens = TestBackedEnum::Fifth;
41+
$book->base64 = StringBackedEnum::First;
42+
$book->string = StringBackedEnum::Second;
43+
$book->any = StringBackedEnum::Third;
44+
$book->hexbin = StringBackedEnum::Fourth;
45+
$book->nmtokens = StringBackedEnum::Fifth;
46+
$book->integer = IntBackedEnum::First;
47+
$book->short = IntBackedEnum::Second;
4048

4149
try {
4250
$client->dotest($book);
@@ -45,12 +53,24 @@ try {
4553
echo "--- Test with non-backed enum ---\n";
4654

4755
$book = new stdClass();
48-
$book->base64 = TestNonBackedEnum::First;
49-
$book->string = TestNonBackedEnum::First;
50-
$book->any = TestNonBackedEnum::First;
51-
$book->hexbin = TestNonBackedEnum::First;
52-
$book->nmtokens = TestNonBackedEnum::First;
56+
$book->base64 = NonBackedEnum::First;
57+
$book->string = NonBackedEnum::First;
58+
$book->any = NonBackedEnum::First;
59+
$book->hexbin = NonBackedEnum::First;
60+
$book->nmtokens = NonBackedEnum::First;
61+
$book->integer = NonBackedEnum::First;
62+
$book->short = NonBackedEnum::First;
63+
64+
try {
65+
$client->dotest($book);
66+
} catch (ValueError $e) {
67+
echo "ValueError: ", $e->getMessage(), "\n";
68+
}
69+
70+
echo "--- Test with mismatched enum backing type ---\n";
5371

72+
$book->integer = StringBackedEnum::First;
73+
$book->short = StringBackedEnum::First;
5474
try {
5575
$client->dotest($book);
5676
} catch (ValueError $e) {
@@ -61,6 +81,8 @@ try {
6181
--EXPECT--
6282
--- Test with backed enum ---
6383
<?xml version="1.0" encoding="UTF-8"?>
64-
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/" xmlns:ns1="http://schemas.nothing.com" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/" SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"><SOAP-ENV:Body><ns1:dotest><dotestReturn xsi:type="ns1:book"><base64 xsi:type="xsd:base64Binary">QmFja2luZ1ZhbHVlMQ==</base64><string xsi:type="xsd:string">BackingValue2</string><any xsi:type="xsd:any"><name xsi:type="xsd:string">Third</name><value xsi:type="xsd:string">BackingValue3</value></any><hexbin xsi:type="xsd:hexBinary">4261636B696E6756616C756534</hexbin><nmtokens>BackingValue5</nmtokens></dotestReturn></ns1:dotest></SOAP-ENV:Body></SOAP-ENV:Envelope>
84+
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/" xmlns:ns1="http://schemas.nothing.com" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/" SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"><SOAP-ENV:Body><ns1:dotest><dotestReturn xsi:type="ns1:book"><base64 xsi:type="xsd:base64Binary">QmFja2luZ1ZhbHVlMQ==</base64><string xsi:type="xsd:string">BackingValue2</string><any xsi:type="xsd:any"><name xsi:type="xsd:string">Third</name><value xsi:type="xsd:string">BackingValue3</value></any><hexbin xsi:type="xsd:hexBinary">4261636B696E6756616C756534</hexbin><nmtokens>BackingValue5</nmtokens><integer xsi:type="xsd:integer">1</integer><short xsi:type="xsd:short">2</short></dotestReturn></ns1:dotest></SOAP-ENV:Body></SOAP-ENV:Envelope>
6585
--- Test with non-backed enum ---
6686
ValueError: Non-backed enums have no default serialization
87+
--- Test with mismatched enum backing type ---
88+
ValueError: String-backed enum cannot be serialized as int

ext/soap/tests/gh15711.wsdl

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
<xsd:element name="any" type="xsd:any"/>
99
<xsd:element name="hexbin" type="xsd:hexBinary"/>
1010
<xsd:element name="nmtokens" type="xsd:NMTOKENS"/>
11+
<xsd:element name="integer" type="xsd:integer"/>
12+
<xsd:element name="short" type="xsd:short"/>
1113
</xsd:all>
1214
</xsd:complexType>
1315
</xsd:schema>

0 commit comments

Comments
 (0)