zend_read_property API update
[m6w6/ext-http] / php_http_header.c
1 /*
2 +--------------------------------------------------------------------+
3 | PECL :: http |
4 +--------------------------------------------------------------------+
5 | Redistribution and use in source and binary forms, with or without |
6 | modification, are permitted provided that the conditions mentioned |
7 | in the accompanying LICENSE file are met. |
8 +--------------------------------------------------------------------+
9 | Copyright (c) 2004-2014, Michael Wallner <mike@php.net> |
10 +--------------------------------------------------------------------+
11 */
12
13 #include "php_http_api.h"
14
15 ZEND_RESULT_CODE php_http_header_parse(const char *header, size_t length, HashTable *headers, php_http_info_callback_t callback_func, void **callback_data)
16 {
17 php_http_header_parser_t ctx;
18 php_http_buffer_t buf;
19 php_http_header_parser_state_t rs;
20
21 if (!php_http_buffer_from_string_ex(&buf, header, length)) {
22 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Could not allocate buffer");
23 return FAILURE;
24 }
25
26 if (!php_http_header_parser_init(&ctx TSRMLS_CC)) {
27 php_http_buffer_dtor(&buf);
28 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Could not initialize header parser");
29 return FAILURE;
30 }
31
32 rs = php_http_header_parser_parse(&ctx, &buf, PHP_HTTP_HEADER_PARSER_CLEANUP, headers, callback_func, callback_data);
33 php_http_header_parser_dtor(&ctx);
34 php_http_buffer_dtor(&buf);
35
36 if (rs == PHP_HTTP_HEADER_PARSER_STATE_FAILURE) {
37 php_error_docref(NULL TSRMLS_CC, E_WARNING, "Could not parse headers");
38 return FAILURE;
39 }
40
41 return SUCCESS;
42 }
43
44 void php_http_header_to_callback(HashTable *headers, zend_bool crlf, php_http_pass_format_callback_t cb, void *cb_arg TSRMLS_DC)
45 {
46 php_http_arrkey_t key;
47 zval *header, *single_header;
48
49 ZEND_HASH_FOREACH_KEY_VAL(headers, key.h, key.key, header)
50 {
51 if (key.key) {
52 if (zend_string_equals_literal(key.key, "Set-Cookie") && Z_TYPE_P(header) == IS_ARRAY) {
53 ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(header), single_header)
54 {
55 if (Z_TYPE_P(single_header) == IS_ARRAY) {
56 php_http_cookie_list_t *cookie = php_http_cookie_list_from_struct(NULL, single_header);
57
58 if (cookie) {
59 char *buf;
60 size_t len;
61
62 php_http_cookie_list_to_string(cookie, &buf, &len);
63 cb(cb_arg, crlf ? "Set-Cookie: %s" PHP_HTTP_CRLF : "Set-Cookie: %s", buf);
64 php_http_cookie_list_free(&cookie);
65 efree(buf);
66 }
67 } else {
68 zend_string *zs = php_http_header_value_to_string(single_header);
69
70 cb(cb_arg, crlf ? "Set-Cookie: %s" PHP_HTTP_CRLF : "Set-Cookie: %s", zs->val);
71 zend_string_release(zs);
72 }
73 }
74 ZEND_HASH_FOREACH_END();
75 } else {
76 zend_string *zs = php_http_header_value_to_string(header);
77
78 cb(cb_arg, crlf ? "%s: %s" PHP_HTTP_CRLF : "%s: %s", key.key->val, zs->val);
79 zend_string_release(zs);
80 }
81 }
82 }
83 ZEND_HASH_FOREACH_END();
84 }
85
86 void php_http_header_to_string(php_http_buffer_t *str, HashTable *headers TSRMLS_DC)
87 {
88 php_http_header_to_callback(headers, 1, (php_http_pass_format_callback_t) php_http_buffer_appendf, str TSRMLS_CC);
89 }
90
91 zend_string *php_http_header_value_array_to_string(zval *header)
92 {
93 zval *val;
94 php_http_buffer_t str;
95
96 php_http_buffer_init(&str);
97 ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(header), val)
98 {
99 zend_string *zs = php_http_header_value_to_string(val);
100
101 php_http_buffer_appendf(&str, str.used ? ", %s":"%s", zs->val);
102 zend_string_release(zs);
103 }
104 ZEND_HASH_FOREACH_END();
105 php_http_buffer_fix(&str);
106
107 return php_http_cs2zs(str.data, str.used);
108 }
109
110 zend_string *php_http_header_value_to_string(zval *header)
111 {
112 switch (Z_TYPE_P(header)) {
113 case IS_TRUE:
114 return zend_string_init(ZEND_STRL("true"), 0);
115 case IS_FALSE:
116 return zend_string_init(ZEND_STRL("false"), 0);
117 case IS_ARRAY:
118 return php_http_header_value_array_to_string(header);
119 default:
120 return zval_get_string(header);
121 }
122 }
123
124 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader___construct, 0, 0, 0)
125 ZEND_ARG_INFO(0, name)
126 ZEND_ARG_INFO(0, value)
127 ZEND_END_ARG_INFO();
128 PHP_METHOD(HttpHeader, __construct)
129 {
130 char *name_str = NULL, *value_str = NULL;
131 size_t name_len = 0, value_len = 0;
132
133 php_http_expect(SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS(), "|s!s!", &name_str, &name_len, &value_str, &value_len), invalid_arg, return);
134
135 if (name_str && name_len) {
136 char *pretty_str = estrndup(name_str, name_len);
137 zend_update_property_stringl(php_http_header_class_entry, getThis(), ZEND_STRL("name"), php_http_pretty_key(pretty_str, name_len, 1, 1), name_len);
138 efree(pretty_str);
139 }
140 if (value_str && value_len) {
141 zend_update_property_stringl(php_http_header_class_entry, getThis(), ZEND_STRL("value"), value_str, value_len);
142 }
143 }
144
145 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader_serialize, 0, 0, 0)
146 ZEND_END_ARG_INFO();
147 PHP_METHOD(HttpHeader, serialize)
148 {
149 if (SUCCESS == zend_parse_parameters_none()) {
150 php_http_buffer_t buf;
151 zend_string *zs;
152 zval name_tmp, value_tmp;
153
154 php_http_buffer_init(&buf);
155 zs = zval_get_string(zend_read_property(php_http_header_class_entry, getThis(), ZEND_STRL("name"), 0, &name_tmp));
156 php_http_buffer_appendz(&buf, zs);
157 zend_string_release(zs);
158
159 zs = zval_get_string(zend_read_property(php_http_header_class_entry, getThis(), ZEND_STRL("value"), 0, &value_tmp));
160 if (zs->len) {
161 php_http_buffer_appends(&buf, ": ");
162 php_http_buffer_appendz(&buf, zs);
163 } else {
164 php_http_buffer_appends(&buf, ":");
165 }
166 zend_string_release(zs);
167
168 RETURN_STR(php_http_cs2zs(buf.data, buf.used));
169 }
170 RETURN_EMPTY_STRING();
171 }
172
173 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader_unserialize, 0, 0, 1)
174 ZEND_ARG_INFO(0, serialized)
175 ZEND_END_ARG_INFO();
176 PHP_METHOD(HttpHeader, unserialize)
177 {
178 char *serialized_str;
179 size_t serialized_len;
180
181 if (SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &serialized_str, &serialized_len)) {
182 HashTable ht;
183
184 zend_hash_init(&ht, 1, NULL, ZVAL_PTR_DTOR, 0);
185 if (SUCCESS == php_http_header_parse(serialized_str, serialized_len, &ht, NULL, NULL)) {
186 if (zend_hash_num_elements(&ht)) {
187 zend_string *zs, *key;
188 zend_ulong idx;
189
190 zend_hash_internal_pointer_reset(&ht);
191 switch (zend_hash_get_current_key(&ht, &key, &idx)) {
192 case HASH_KEY_IS_STRING:
193 zend_update_property_str(php_http_header_class_entry, getThis(), ZEND_STRL("name"), key);
194 break;
195 case HASH_KEY_IS_LONG:
196 zend_update_property_long(php_http_header_class_entry, getThis(), ZEND_STRL("name"), idx);
197 break;
198 default:
199 break;
200 }
201 zs = zval_get_string(zend_hash_get_current_data(&ht));
202 zend_update_property_str(php_http_header_class_entry, getThis(), ZEND_STRL("value"), zs);
203 zend_string_release(zs);
204 }
205 }
206 zend_hash_destroy(&ht);
207 }
208
209 }
210
211 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader_match, 0, 0, 1)
212 ZEND_ARG_INFO(0, value)
213 ZEND_ARG_INFO(0, flags)
214 ZEND_END_ARG_INFO();
215 PHP_METHOD(HttpHeader, match)
216 {
217 char *val_str;
218 size_t val_len;
219 zend_long flags = PHP_HTTP_MATCH_LOOSE;
220 zend_string *zs;
221 zval value_tmp;
222
223 if (SUCCESS != zend_parse_parameters(ZEND_NUM_ARGS(), "|sl", &val_str, &val_len, &flags)) {
224 return;
225 }
226
227 zs = zval_get_string(zend_read_property(php_http_header_class_entry, getThis(), ZEND_STRL("value"), 0, &value_tmp));
228 RETVAL_BOOL(php_http_match(zs->val, val_str, flags));
229 zend_string_release(zs);
230 }
231
232 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader_negotiate, 0, 0, 1)
233 ZEND_ARG_INFO(0, supported)
234 ZEND_ARG_INFO(1, result)
235 ZEND_END_ARG_INFO();
236 PHP_METHOD(HttpHeader, negotiate)
237 {
238 HashTable *supported, *rs;
239 zval name_tmp, value_tmp, *rs_array = NULL;
240 zend_string *zs;
241 char *sep_str = NULL;
242 size_t sep_len = 0;
243
244 if (SUCCESS != zend_parse_parameters(ZEND_NUM_ARGS(), "H|z", &supported, &rs_array)) {
245 return;
246 }
247 if (rs_array) {
248 ZVAL_DEREF(rs_array);
249 zval_dtor(rs_array);
250 array_init(rs_array);
251 }
252
253 zs = zval_get_string(zend_read_property(php_http_header_class_entry, getThis(), ZEND_STRL("name"), 0, &name_tmp));
254 if (zend_string_equals_literal(zs, "Accept")) {
255 sep_str = "/";
256 sep_len = 1;
257 } else if (zend_string_equals_literal(zs, "Accept-Language")) {
258 sep_str = "-";
259 sep_len = 1;
260 }
261 zend_string_release(zs);
262
263 zs = zval_get_string(zend_read_property(php_http_header_class_entry, getThis(), ZEND_STRL("value"), 0, &value_tmp));
264 if ((rs = php_http_negotiate(zs->val, zs->len, supported, sep_str, sep_len))) {
265 PHP_HTTP_DO_NEGOTIATE_HANDLE_RESULT(rs, supported, rs_array);
266 } else {
267 PHP_HTTP_DO_NEGOTIATE_HANDLE_DEFAULT(supported, rs_array);
268 }
269 zend_string_release(zs);
270 }
271
272 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader_getParams, 0, 0, 0)
273 ZEND_ARG_INFO(0, param_sep)
274 ZEND_ARG_INFO(0, arg_sep)
275 ZEND_ARG_INFO(0, val_sep)
276 ZEND_ARG_INFO(0, flags)
277 ZEND_END_ARG_INFO();
278 PHP_METHOD(HttpHeader, getParams)
279 {
280 zval value_tmp, zctor, zparams_obj, *zargs = NULL;
281
282 ZVAL_STRINGL(&zctor, "__construct", lenof("__construct"));
283
284 object_init_ex(&zparams_obj, php_http_params_class_entry);
285
286 zargs = (zval *) ecalloc(ZEND_NUM_ARGS()+1, sizeof(zval));
287 ZVAL_COPY_VALUE(&zargs[0], zend_read_property(php_http_header_class_entry, getThis(), ZEND_STRL("value"), 0, &value_tmp));
288 if (ZEND_NUM_ARGS()) {
289 zend_get_parameters_array(ZEND_NUM_ARGS(), ZEND_NUM_ARGS(), &zargs[1]);
290 }
291
292 if (SUCCESS == call_user_function(NULL, &zparams_obj, &zctor, return_value, ZEND_NUM_ARGS()+1, zargs)) {
293 RETVAL_ZVAL(&zparams_obj, 0, 1);
294 }
295
296 zval_ptr_dtor(&zctor);
297 if (zargs) {
298 efree(zargs);
299 }
300 }
301
302 ZEND_BEGIN_ARG_INFO_EX(ai_HttpHeader_parse, 0, 0, 1)
303 ZEND_ARG_INFO(0, string)
304 ZEND_ARG_INFO(0, header_class)
305 ZEND_END_ARG_INFO();
306 PHP_METHOD(HttpHeader, parse)
307 {
308 char *header_str;
309 size_t header_len;
310 zend_class_entry *ce = NULL;
311
312 if (SUCCESS == zend_parse_parameters(ZEND_NUM_ARGS(), "s|C", &header_str, &header_len, &ce)) {
313 array_init(return_value);
314
315 if (SUCCESS != php_http_header_parse(header_str, header_len, Z_ARRVAL_P(return_value), NULL, NULL)) {
316 zval_dtor(return_value);
317 RETURN_FALSE;
318 } else {
319 if (ce && instanceof_function(ce, php_http_header_class_entry TSRMLS_CC)) {
320 php_http_arrkey_t key;
321 zval *val;
322
323 ZEND_HASH_FOREACH_KEY_VAL(Z_ARRVAL_P(return_value), key.h, key.key, val)
324 {
325 zval zkey, zho;
326
327 if (key.key) {
328 ZVAL_STR_COPY(&zkey, key.key);
329 } else {
330 ZVAL_LONG(&zkey, key.h);
331 }
332
333 object_init_ex(&zho, ce);
334 Z_TRY_ADDREF_P(val);
335 zend_call_method_with_2_params(&zho, ce, NULL, "__construct", NULL, &zkey, val);
336 zval_ptr_dtor(val);
337 zval_ptr_dtor(&zkey);
338
339 if (key.key) {
340 add_assoc_zval_ex(return_value, key.key->val, key.key->len, &zho);
341 } else {
342 add_index_zval(return_value, key.h, &zho);
343 }
344 }
345 ZEND_HASH_FOREACH_END();
346 }
347 }
348 }
349 }
350
351 static zend_function_entry php_http_header_methods[] = {
352 PHP_ME(HttpHeader, __construct, ai_HttpHeader___construct, ZEND_ACC_PUBLIC|ZEND_ACC_CTOR)
353 PHP_ME(HttpHeader, serialize, ai_HttpHeader_serialize, ZEND_ACC_PUBLIC)
354 ZEND_MALIAS(HttpHeader, __toString, serialize, ai_HttpHeader_serialize, ZEND_ACC_PUBLIC)
355 ZEND_MALIAS(HttpHeader, toString, serialize, ai_HttpHeader_serialize, ZEND_ACC_PUBLIC)
356 PHP_ME(HttpHeader, unserialize, ai_HttpHeader_unserialize, ZEND_ACC_PUBLIC)
357 PHP_ME(HttpHeader, match, ai_HttpHeader_match, ZEND_ACC_PUBLIC)
358 PHP_ME(HttpHeader, negotiate, ai_HttpHeader_negotiate, ZEND_ACC_PUBLIC)
359 PHP_ME(HttpHeader, getParams, ai_HttpHeader_getParams, ZEND_ACC_PUBLIC)
360 PHP_ME(HttpHeader, parse, ai_HttpHeader_parse, ZEND_ACC_PUBLIC|ZEND_ACC_STATIC)
361 EMPTY_FUNCTION_ENTRY
362 };
363
364 zend_class_entry *php_http_header_class_entry;
365
366 PHP_MINIT_FUNCTION(http_header)
367 {
368 zend_class_entry ce = {0};
369
370 INIT_NS_CLASS_ENTRY(ce, "http", "Header", php_http_header_methods);
371 php_http_header_class_entry = zend_register_internal_class(&ce);
372 zend_class_implements(php_http_header_class_entry, 1, zend_ce_serializable);
373 zend_declare_class_constant_long(php_http_header_class_entry, ZEND_STRL("MATCH_LOOSE"), PHP_HTTP_MATCH_LOOSE);
374 zend_declare_class_constant_long(php_http_header_class_entry, ZEND_STRL("MATCH_CASE"), PHP_HTTP_MATCH_CASE);
375 zend_declare_class_constant_long(php_http_header_class_entry, ZEND_STRL("MATCH_WORD"), PHP_HTTP_MATCH_WORD);
376 zend_declare_class_constant_long(php_http_header_class_entry, ZEND_STRL("MATCH_FULL"), PHP_HTTP_MATCH_FULL);
377 zend_declare_class_constant_long(php_http_header_class_entry, ZEND_STRL("MATCH_STRICT"), PHP_HTTP_MATCH_STRICT);
378 zend_declare_property_null(php_http_header_class_entry, ZEND_STRL("name"), ZEND_ACC_PUBLIC);
379 zend_declare_property_null(php_http_header_class_entry, ZEND_STRL("value"), ZEND_ACC_PUBLIC);
380
381 return SUCCESS;
382 }
383
384 /*
385 * Local variables:
386 * tab-width: 4
387 * c-basic-offset: 4
388 * End:
389 * vim600: noet sw=4 ts=4 fdm=marker
390 * vim<600: noet sw=4 ts=4
391 */
392