2 +--------------------------------------------------------------------+
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 +--------------------------------------------------------------------+
13 #include "php_http_api.h"
15 static php_http_params_token_t def_param_sep
= {",", 1}, *def_param_sep_ptr
[] = {&def_param_sep
, NULL
};
16 static php_http_params_token_t def_arg_sep
= {";", 1}, *def_arg_sep_ptr
[] = {&def_arg_sep
, NULL
};
17 static php_http_params_token_t def_val_sep
= {"=", 1}, *def_val_sep_ptr
[] = {&def_val_sep
, NULL
};
18 static php_http_params_opts_t def_opts
= {
24 PHP_HTTP_PARAMS_DEFAULT
27 php_http_params_opts_t
*php_http_params_opts_default_get(php_http_params_opts_t
*opts
)
30 opts
= emalloc(sizeof(*opts
));
33 memcpy(opts
, &def_opts
, sizeof(def_opts
));
38 typedef struct php_http_params_state
{
39 php_http_params_token_t input
;
40 php_http_params_token_t param
;
41 php_http_params_token_t arg
;
42 php_http_params_token_t val
;
51 } php_http_params_state_t
;
53 static inline void sanitize_escaped(zval
*zv TSRMLS_DC
)
55 if (Z_STRVAL_P(zv
)[0] == '"' && Z_STRVAL_P(zv
)[Z_STRLEN_P(zv
) - 1] == '"') {
56 size_t deq_len
= Z_STRLEN_P(zv
) - 2;
57 char *deq
= estrndup(Z_STRVAL_P(zv
) + 1, deq_len
);
60 ZVAL_STRINGL(zv
, deq
, deq_len
, 0);
63 php_stripcslashes(Z_STRVAL_P(zv
), &Z_STRLEN_P(zv
) TSRMLS_CC
);
66 static inline void prepare_escaped(zval
*zv TSRMLS_DC
)
68 if (Z_TYPE_P(zv
) == IS_STRING
) {
69 int len
= Z_STRLEN_P(zv
);
71 Z_STRVAL_P(zv
) = php_addcslashes(Z_STRVAL_P(zv
), Z_STRLEN_P(zv
), &Z_STRLEN_P(zv
), 1,
72 ZEND_STRL("\0..\37\173\\\"") TSRMLS_CC
);
74 if (len
!= Z_STRLEN_P(zv
) || strpbrk(Z_STRVAL_P(zv
), "()<>@,;:\"/[]?={} ")) {
76 int len
= Z_STRLEN_P(zv
) + 2;
77 char *str
= emalloc(len
+ 1);
80 memcpy(&str
[1], Z_STRVAL_P(zv
), Z_STRLEN_P(zv
));
85 ZVAL_STRINGL(zv
, str
, len
, 0);
89 ZVAL_EMPTY_STRING(zv
);
93 static inline void sanitize_urlencoded(zval
*zv TSRMLS_DC
)
95 Z_STRLEN_P(zv
) = php_raw_url_decode(Z_STRVAL_P(zv
), Z_STRLEN_P(zv
));
98 static inline void prepare_urlencoded(zval
*zv TSRMLS_DC
)
101 char *str
= php_raw_url_encode(Z_STRVAL_P(zv
), Z_STRLEN_P(zv
), &len
);
104 ZVAL_STRINGL(zv
, str
, len
, 0);
107 static void sanitize_dimension(zval
*zv TSRMLS_DC
)
109 zval
*arr
= NULL
, *tmp
= NULL
, **cur
= NULL
;
110 char *var
= NULL
, *ptr
= Z_STRVAL_P(zv
), *end
= Z_STRVAL_P(zv
) + Z_STRLEN_P(zv
);
124 if (++level
> PG(max_input_nesting_level
)) {
126 php_error_docref(NULL TSRMLS_CC
, E_WARNING
, "Max input nesting level of %ld exceeded", (long) PG(max_input_nesting_level
));
129 if (ptr
- var
== 0) {
139 convert_to_array(*cur
);
144 zend_symtable_update(Z_ARRVAL_PP(cur
), var
, ptr
- var
+ 1, (void *) &tmp
, sizeof(zval
*), (void *) &cur
);
147 zend_hash_next_index_insert(Z_ARRVAL_PP(cur
), (void *) &tmp
, sizeof(zval
*), (void *) &cur
);
157 if (zend_hash_num_elements(Z_ARRVAL_P(arr
))) {
159 #if PHP_VERSION_ID >= 50400
160 ZVAL_COPY_VALUE(zv
, arr
);
162 zv
->value
= arr
->value
;
163 Z_TYPE_P(zv
) = Z_TYPE_P(arr
);
171 static inline void shift_key(php_http_buffer_t
*buf
, char *key_str
, size_t key_len
, const char *ass
, size_t asl
, unsigned flags TSRMLS_DC
);
172 static inline void shift_val(php_http_buffer_t
*buf
, zval
*zvalue
, const char *vss
, size_t vsl
, unsigned flags TSRMLS_DC
);
174 static void prepare_dimension(php_http_buffer_t
*buf
, php_http_buffer_t
*keybuf
, zval
*zvalue
, const char *pss
, size_t psl
, const char *vss
, size_t vsl
, unsigned flags TSRMLS_DC
)
176 HashTable
*ht
= HASH_OF(zvalue
);
178 php_http_array_hashkey_t key
= php_http_array_hashkey_init(0);
180 php_http_buffer_t prefix
;
182 if (!ht
->nApplyCount
++) {
183 php_http_buffer_init(&prefix
);
184 php_http_buffer_append(&prefix
, keybuf
->data
, keybuf
->used
);
186 FOREACH_HASH_KEYVAL(pos
, ht
, key
, val
) {
187 if (key
.type
== HASH_KEY_IS_STRING
&& !*key
.str
) {
188 /* only public properties */
192 php_http_buffer_appends(&prefix
, "[");
193 if (key
.type
== HASH_KEY_IS_STRING
) {
194 php_http_buffer_append(&prefix
, key
.str
, key
.len
- 1);
196 php_http_buffer_appendf(&prefix
, "%lu", key
.num
);
198 php_http_buffer_appends(&prefix
, "]");
200 if (Z_TYPE_PP(val
) == IS_ARRAY
|| Z_TYPE_PP(val
) == IS_OBJECT
) {
201 prepare_dimension(buf
, &prefix
, *val
, pss
, psl
, vss
, vsl
, flags TSRMLS_CC
);
203 zval
*cpy
= php_http_ztyp(IS_STRING
, *val
);
205 shift_key(buf
, prefix
.data
, prefix
.used
, pss
, psl
, flags TSRMLS_CC
);
206 shift_val(buf
, cpy
, vss
, vsl
, flags TSRMLS_CC
);
210 php_http_buffer_cut(&prefix
, keybuf
->used
, prefix
.used
- keybuf
->used
);
212 php_http_buffer_dtor(&prefix
);
217 static inline void sanitize_key(unsigned flags
, char *str
, size_t len
, zval
*zv
, zend_bool
*rfc5987 TSRMLS_DC
)
222 php_trim(str
, len
, NULL
, 0, zv
, 3 TSRMLS_CC
);
224 if (flags
& PHP_HTTP_PARAMS_ESCAPED
) {
225 sanitize_escaped(zv TSRMLS_CC
);
228 eos
= &Z_STRVAL_P(zv
)[Z_STRLEN_P(zv
)-1];
235 if (flags
& PHP_HTTP_PARAMS_URLENCODED
) {
236 sanitize_urlencoded(zv TSRMLS_CC
);
239 if (flags
& PHP_HTTP_PARAMS_DIMENSION
) {
240 sanitize_dimension(zv TSRMLS_CC
);
244 static inline void sanitize_rfc5987(zval
*zv
, char **language
, zend_bool
*latin1 TSRMLS_DC
)
249 * iso-8850-1'de'bl%f6der%20schei%df%21
250 * utf-8'de-DE'bl%c3%b6der%20schei%c3%9f%21
253 switch (Z_STRVAL_P(zv
)[0]) {
256 if (!strncasecmp(Z_STRVAL_P(zv
), ZEND_STRL("iso-8859-1"))) {
258 ptr
= Z_STRVAL_P(zv
) + lenof("iso-8859-1");
264 if (!strncasecmp(Z_STRVAL_P(zv
), ZEND_STRL("utf-8"))) {
266 ptr
= Z_STRVAL_P(zv
) + lenof("utf-8");
274 /* extract language */
276 for (*language
= ++ptr
; *ptr
&& *ptr
!= '\''; ++ptr
);
281 *language
= estrndup(*language
, ptr
- *language
);
284 ptr
= estrdup(++ptr
);
286 ZVAL_STRING(zv
, ptr
, 0);
290 static void utf8encode(zval
*zv
)
293 unsigned char *ptr
= (unsigned char *) Z_STRVAL_P(zv
);
296 if (*ptr
++ >= 0x80) {
302 ptr
= safe_emalloc(1, len
, 1);
303 for (len
= 0, pos
= 0; len
<= Z_STRLEN_P(zv
); ++len
, ++pos
) {
304 ptr
[pos
] = Z_STRVAL_P(zv
)[len
];
305 if ((ptr
[pos
]) >= 0x80) {
306 ptr
[pos
+ 1] = 0x80 | (ptr
[pos
] & 0x3f);
307 ptr
[pos
] = 0xc0 | ((ptr
[pos
] >> 6) & 0x1f);
312 ZVAL_STRINGL(zv
, (char *) ptr
, pos
-1, 0);
315 static inline void sanitize_value(unsigned flags
, char *str
, size_t len
, zval
*zv
, zend_bool rfc5987 TSRMLS_DC
)
317 char *language
= NULL
;
318 zend_bool latin1
= 0;
321 php_trim(str
, len
, NULL
, 0, zv
, 3 TSRMLS_CC
);
324 sanitize_rfc5987(zv
, &language
, &latin1 TSRMLS_CC
);
327 if (flags
& PHP_HTTP_PARAMS_ESCAPED
) {
328 sanitize_escaped(zv TSRMLS_CC
);
331 if ((flags
& PHP_HTTP_PARAMS_URLENCODED
) || (rfc5987
&& language
)) {
332 sanitize_urlencoded(zv TSRMLS_CC
);
335 if (rfc5987
&& language
) {
343 ZVAL_COPY_VALUE(tmp
, zv
);
345 add_assoc_zval(zv
, language
, tmp
);
350 static inline void prepare_key(unsigned flags
, char *old_key
, size_t old_len
, char **new_key
, size_t *new_len TSRMLS_DC
)
355 ZVAL_STRINGL(&zv
, old_key
, old_len
, 1);
357 if (flags
& PHP_HTTP_PARAMS_URLENCODED
) {
358 prepare_urlencoded(&zv TSRMLS_CC
);
361 if (flags
& PHP_HTTP_PARAMS_ESCAPED
) {
362 prepare_escaped(&zv TSRMLS_CC
);
365 *new_key
= Z_STRVAL(zv
);
366 *new_len
= Z_STRLEN(zv
);
369 static inline void prepare_value(unsigned flags
, zval
*zv TSRMLS_DC
)
371 if (flags
& PHP_HTTP_PARAMS_URLENCODED
) {
372 prepare_urlencoded(zv TSRMLS_CC
);
375 if (flags
& PHP_HTTP_PARAMS_ESCAPED
) {
376 prepare_escaped(zv TSRMLS_CC
);
380 static void merge_param(HashTable
*params
, zval
*zdata
, zval
***current_param
, zval
***current_args TSRMLS_DC
)
382 zval
**ptr
, **zdata_ptr
;
383 php_http_array_hashkey_t hkey
= php_http_array_hashkey_init(0);
388 INIT_PZVAL_ARRAY(&tmp
, params
);
389 fprintf(stderr
, "params = ");
390 zend_print_zval_r(&tmp
, 1 TSRMLS_CC
);
391 fprintf(stderr
, "\n");
395 hkey
.type
= zend_hash_get_current_key_ex(Z_ARRVAL_P(zdata
), &hkey
.str
, &hkey
.len
, &hkey
.num
, hkey
.dup
, NULL
);
397 if ((hkey
.type
== HASH_KEY_IS_STRING
&& !zend_hash_exists(params
, hkey
.str
, hkey
.len
))
398 || (hkey
.type
== HASH_KEY_IS_LONG
&& !zend_hash_index_exists(params
, hkey
.num
))
400 zval
*tmp
, *arg
, **args
;
402 /* create the entry if it doesn't exist */
403 zend_hash_get_current_data(Z_ARRVAL_P(zdata
), (void *) &ptr
);
407 add_assoc_zval_ex(tmp
, ZEND_STRS("value"), *ptr
);
411 zend_hash_update(Z_ARRVAL_P(tmp
), "arguments", sizeof("arguments"), (void *) &arg
, sizeof(zval
*), (void *) &args
);
412 *current_args
= args
;
414 if (hkey
.type
== HASH_KEY_IS_STRING
) {
415 zend_hash_update(params
, hkey
.str
, hkey
.len
, (void *) &tmp
, sizeof(zval
*), (void *) &ptr
);
417 zend_hash_index_update(params
, hkey
.num
, (void *) &tmp
, sizeof(zval
*), (void *) &ptr
);
421 if (hkey
.type
== HASH_KEY_IS_STRING
) {
422 zend_hash_find(params
, hkey
.str
, hkey
.len
, (void *) &ptr
);
424 zend_hash_index_find(params
, hkey
.num
, (void *) &ptr
);
429 if (Z_TYPE_PP(ptr
) == IS_ARRAY
430 && SUCCESS
== zend_hash_find(Z_ARRVAL_PP(ptr
), "value", sizeof("value"), (void *) &ptr
)
431 && SUCCESS
== zend_hash_get_current_data(Z_ARRVAL_PP(zdata_ptr
), (void *) &zdata_ptr
)
434 * params = [arr => [value => [0 => 1]]]
436 * zdata = [arr => [0 => NULL]]
441 while (Z_TYPE_PP(zdata_ptr
) == IS_ARRAY
442 && SUCCESS
== zend_hash_get_current_data(Z_ARRVAL_PP(zdata_ptr
), (void *) &test_ptr
)
444 if (Z_TYPE_PP(test_ptr
) == IS_ARRAY
) {
446 /* now find key in ptr */
447 if (HASH_KEY_IS_STRING
== zend_hash_get_current_key_ex(Z_ARRVAL_PP(zdata_ptr
), &hkey
.str
, &hkey
.len
, &hkey
.num
, hkey
.dup
, NULL
)) {
448 if (SUCCESS
== zend_hash_find(Z_ARRVAL_PP(ptr
), hkey
.str
, hkey
.len
, (void *) &ptr
)) {
449 zdata_ptr
= test_ptr
;
451 Z_ADDREF_PP(test_ptr
);
452 zend_hash_update(Z_ARRVAL_PP(ptr
), hkey
.str
, hkey
.len
, (void *) test_ptr
, sizeof(zval
*), (void *) &ptr
);
456 if (SUCCESS
== zend_hash_index_find(Z_ARRVAL_PP(ptr
), hkey
.num
, (void *) &ptr
)) {
457 zdata_ptr
= test_ptr
;
458 } else if (hkey
.num
) {
459 Z_ADDREF_PP(test_ptr
);
460 zend_hash_index_update(Z_ARRVAL_PP(ptr
), hkey
.num
, (void *) test_ptr
, sizeof(zval
*), (void *) &ptr
);
463 Z_ADDREF_PP(test_ptr
);
464 zend_hash_next_index_insert(Z_ARRVAL_PP(ptr
), (void *) test_ptr
, sizeof(zval
*), (void *) &ptr
);
469 /* this is the leaf */
470 Z_ADDREF_PP(test_ptr
);
471 if (Z_TYPE_PP(ptr
) != IS_ARRAY
) {
475 if (HASH_KEY_IS_STRING
== zend_hash_get_current_key_ex(Z_ARRVAL_PP(zdata_ptr
), &hkey
.str
, &hkey
.len
, &hkey
.num
, hkey
.dup
, NULL
)) {
476 zend_hash_update(Z_ARRVAL_PP(ptr
), hkey
.str
, hkey
.len
, (void *) test_ptr
, sizeof(zval
*), (void *) &ptr
);
477 } else if (hkey
.num
) {
478 zend_hash_index_update(Z_ARRVAL_PP(ptr
), hkey
.num
, (void *) test_ptr
, sizeof(zval
*), (void *) &ptr
);
480 zend_hash_next_index_insert(Z_ARRVAL_PP(ptr
), (void *) test_ptr
, sizeof(zval
*), (void *) &ptr
);
490 while (Z_TYPE_PP(ptr
) == IS_ARRAY
&& SUCCESS
== zend_hash_get_current_data(Z_ARRVAL_PP(ptr
), (void *) &ptr
));
491 *current_param
= ptr
;
494 static void push_param(HashTable
*params
, php_http_params_state_t
*state
, const php_http_params_opts_t
*opts TSRMLS_DC
)
496 if (state
->val
.str
) {
497 if (0 < (state
->val
.len
= state
->input
.str
- state
->val
.str
)) {
498 sanitize_value(opts
->flags
, state
->val
.str
, state
->val
.len
, *(state
->current
.val
), state
->rfc5987 TSRMLS_CC
);
501 } else if (state
->arg
.str
) {
502 if (0 < (state
->arg
.len
= state
->input
.str
- state
->arg
.str
)) {
504 zend_bool rfc5987
= 0;
508 sanitize_key(opts
->flags
, state
->arg
.str
, state
->arg
.len
, &key
, &rfc5987 TSRMLS_CC
);
509 state
->rfc5987
= rfc5987
;
510 if (Z_TYPE(key
) == IS_STRING
&& Z_STRLEN(key
)) {
517 if (SUCCESS
== zend_hash_find(Z_ARRVAL_PP(state
->current
.args
), ZEND_STRS("*rfc5987*"), (void *) &rfc
)) {
518 zend_symtable_update(Z_ARRVAL_PP(rfc
), Z_STRVAL(key
), Z_STRLEN(key
) + 1, (void *) &val
, sizeof(zval
*), (void *) &state
->current
.val
);
523 array_init_size(tmp
, 1);
524 zend_symtable_update(Z_ARRVAL_P(tmp
), Z_STRVAL(key
), Z_STRLEN(key
) + 1, (void *) &val
, sizeof(zval
*), (void *) &state
->current
.val
);
525 zend_symtable_update(Z_ARRVAL_PP(state
->current
.args
), ZEND_STRS("*rfc5987*"), (void *) &tmp
, sizeof(zval
*), NULL
);
528 zend_symtable_update(Z_ARRVAL_PP(state
->current
.args
), Z_STRVAL(key
), Z_STRLEN(key
) + 1, (void *) &val
, sizeof(zval
*), (void *) &state
->current
.val
);
533 } else if (state
->param
.str
) {
534 if (0 < (state
->param
.len
= state
->input
.str
- state
->param
.str
)) {
535 zval
*prm
, *arg
, *val
, *key
;
536 zend_bool rfc5987
= 0;
540 sanitize_key(opts
->flags
, state
->param
.str
, state
->param
.len
, key
, &rfc5987 TSRMLS_CC
);
541 state
->rfc5987
= rfc5987
;
542 if (Z_TYPE_P(key
) != IS_STRING
) {
543 merge_param(params
, key
, &state
->current
.val
, &state
->current
.args TSRMLS_CC
);
544 } else if (Z_STRLEN_P(key
)) {
546 array_init_size(prm
, 2);
550 ZVAL_COPY_VALUE(val
, opts
->defval
);
555 if (rfc5987
&& (opts
->flags
& PHP_HTTP_PARAMS_RFC5987
)) {
556 zend_hash_update(Z_ARRVAL_P(prm
), "*rfc5987*", sizeof("*rfc5987*"), (void *) &val
, sizeof(zval
*), (void *) &state
->current
.val
);
558 zend_hash_update(Z_ARRVAL_P(prm
), "value", sizeof("value"), (void *) &val
, sizeof(zval
*), (void *) &state
->current
.val
);
562 array_init_size(arg
, 3);
563 zend_hash_update(Z_ARRVAL_P(prm
), "arguments", sizeof("arguments"), (void *) &arg
, sizeof(zval
*), (void *) &state
->current
.args
);
565 zend_symtable_update(params
, Z_STRVAL_P(key
), Z_STRLEN_P(key
) + 1, (void *) &prm
, sizeof(zval
*), (void *) &state
->current
.param
);
572 static inline zend_bool
check_str(const char *chk_str
, size_t chk_len
, const char *sep_str
, size_t sep_len
) {
573 return 0 < sep_len
&& chk_len
>= sep_len
&& !memcmp(chk_str
, sep_str
, sep_len
);
576 static size_t check_sep(php_http_params_state_t
*state
, php_http_params_token_t
**separators
)
578 php_http_params_token_t
**sep
= separators
;
580 if (state
->quotes
|| state
->escape
) {
584 if (sep
) while (*sep
) {
585 if (check_str(state
->input
.str
, state
->input
.len
, (*sep
)->str
, (*sep
)->len
)) {
593 static void skip_sep(size_t skip
, php_http_params_state_t
*state
, php_http_params_token_t
**param
, php_http_params_token_t
**arg
, php_http_params_token_t
**val TSRMLS_DC
)
597 state
->input
.str
+= skip
;
598 state
->input
.len
-= skip
;
600 while ( (param
&& (sep_len
= check_sep(state
, param
)))
601 || (arg
&& (sep_len
= check_sep(state
, arg
)))
602 || (val
&& (sep_len
= check_sep(state
, val
)))
604 state
->input
.str
+= sep_len
;
605 state
->input
.len
-= sep_len
;
609 HashTable
*php_http_params_parse(HashTable
*params
, const php_http_params_opts_t
*opts TSRMLS_DC
)
611 php_http_params_state_t state
= {{NULL
,0}, {NULL
,0}, {NULL
,0}, {NULL
,0}, {NULL
,NULL
,NULL
}, 0, 0};
613 state
.input
.str
= opts
->input
.str
;
614 state
.input
.len
= opts
->input
.len
;
617 ALLOC_HASHTABLE(params
);
618 ZEND_INIT_SYMTABLE(params
);
621 while (state
.input
.len
) {
622 if (*state
.input
.str
== '"' && !state
.escape
) {
623 state
.quotes
= !state
.quotes
;
625 state
.escape
= (*state
.input
.str
== '\\');
628 if (!state
.param
.str
) {
630 skip_sep(0, &state
, opts
->param
, opts
->arg
, opts
->val TSRMLS_CC
);
631 state
.param
.str
= state
.input
.str
;
634 /* are we at a param separator? */
635 if (0 < (sep_len
= check_sep(&state
, opts
->param
))) {
636 push_param(params
, &state
, opts TSRMLS_CC
);
638 skip_sep(sep_len
, &state
, opts
->param
, opts
->arg
, opts
->val TSRMLS_CC
);
640 /* start off with a new param */
641 state
.param
.str
= state
.input
.str
;
643 state
.arg
.str
= NULL
;
645 state
.val
.str
= NULL
;
651 /* are we at an arg separator? */
652 if (0 < (sep_len
= check_sep(&state
, opts
->arg
))) {
653 push_param(params
, &state
, opts TSRMLS_CC
);
655 skip_sep(sep_len
, &state
, NULL
, opts
->arg
, opts
->val TSRMLS_CC
);
657 /* continue with a new arg */
658 state
.arg
.str
= state
.input
.str
;
660 state
.val
.str
= NULL
;
666 /* are we at a val separator? */
667 if (0 < (sep_len
= check_sep(&state
, opts
->val
))) {
668 /* only handle separator if we're not already reading in a val */
669 if (!state
.val
.str
) {
670 push_param(params
, &state
, opts TSRMLS_CC
);
672 skip_sep(sep_len
, &state
, NULL
, NULL
, opts
->val TSRMLS_CC
);
674 state
.val
.str
= state
.input
.str
;
682 if (state
.input
.len
) {
688 push_param(params
, &state
, opts TSRMLS_CC
);
693 static inline void shift_key(php_http_buffer_t
*buf
, char *key_str
, size_t key_len
, const char *ass
, size_t asl
, unsigned flags TSRMLS_DC
)
699 php_http_buffer_append(buf
, ass
, asl
);
702 prepare_key(flags
, key_str
, key_len
, &str
, &len TSRMLS_CC
);
703 php_http_buffer_append(buf
, str
, len
);
707 static inline void shift_rfc5987(php_http_buffer_t
*buf
, zval
*zvalue
, const char *vss
, size_t vsl
, unsigned flags TSRMLS_DC
)
709 HashTable
*ht
= HASH_OF(zvalue
);
711 php_http_array_hashkey_t key
= php_http_array_hashkey_init(0);
713 if (SUCCESS
== zend_hash_get_current_data(ht
, (void *) &zdata
)
714 && HASH_KEY_NON_EXISTENT
!= (key
.type
= zend_hash_get_current_key_ex(ht
, &key
.str
, &key
.len
, &key
.num
, key
.dup
, NULL
))
716 php_http_array_hashkey_stringify(&key
);
717 php_http_buffer_appendf(buf
, "*%.*sutf-8'%.*s'",
718 (int) (vsl
> INT_MAX
? INT_MAX
: vsl
), vss
,
719 (int) (key
.len
> INT_MAX
? INT_MAX
: key
.len
), key
.str
);
720 php_http_array_hashkey_stringfree(&key
);
722 tmp
= php_http_zsep(1, IS_STRING
, *zdata
);
723 prepare_value(flags
| PHP_HTTP_PARAMS_URLENCODED
, tmp TSRMLS_CC
);
724 php_http_buffer_append(buf
, Z_STRVAL_P(tmp
), Z_STRLEN_P(tmp
));
729 static inline void shift_val(php_http_buffer_t
*buf
, zval
*zvalue
, const char *vss
, size_t vsl
, unsigned flags TSRMLS_DC
)
731 if (Z_TYPE_P(zvalue
) != IS_BOOL
) {
732 zval
*tmp
= php_http_zsep(1, IS_STRING
, zvalue
);
734 prepare_value(flags
, tmp TSRMLS_CC
);
735 php_http_buffer_append(buf
, vss
, vsl
);
736 php_http_buffer_append(buf
, Z_STRVAL_P(tmp
), Z_STRLEN_P(tmp
));
739 } else if (!Z_BVAL_P(zvalue
)) {
740 php_http_buffer_append(buf
, vss
, vsl
);
741 php_http_buffer_appends(buf
, "0");
745 static void shift_arg(php_http_buffer_t
*buf
, char *key_str
, size_t key_len
, zval
*zvalue
, const char *ass
, size_t asl
, const char *vss
, size_t vsl
, unsigned flags TSRMLS_DC
)
747 if (Z_TYPE_P(zvalue
) == IS_ARRAY
|| Z_TYPE_P(zvalue
) == IS_OBJECT
) {
749 php_http_array_hashkey_t key
= php_http_array_hashkey_init(0);
751 zend_bool rfc5987
= !strcmp(key_str
, "*rfc5987*");
754 shift_key(buf
, key_str
, key_len
, ass
, asl
, flags TSRMLS_CC
);
756 FOREACH_KEYVAL(pos
, zvalue
, key
, val
) {
757 /* did you mean recursion? */
758 php_http_array_hashkey_stringify(&key
);
759 if (rfc5987
&& (Z_TYPE_PP(val
) == IS_ARRAY
|| Z_TYPE_PP(val
) == IS_OBJECT
)) {
760 shift_key(buf
, key
.str
, key
.len
-1, ass
, asl
, flags TSRMLS_CC
);
761 shift_rfc5987(buf
, *val
, vss
, vsl
, flags TSRMLS_CC
);
763 shift_arg(buf
, key
.str
, key
.len
-1, *val
, ass
, asl
, vss
, vsl
, flags TSRMLS_CC
);
765 php_http_array_hashkey_stringfree(&key
);
768 shift_key(buf
, key_str
, key_len
, ass
, asl
, flags TSRMLS_CC
);
769 shift_val(buf
, zvalue
, vss
, vsl
, flags TSRMLS_CC
);
773 static void shift_param(php_http_buffer_t
*buf
, char *key_str
, size_t key_len
, zval
*zvalue
, const char *pss
, size_t psl
, const char *ass
, size_t asl
, const char *vss
, size_t vsl
, unsigned flags
, zend_bool rfc5987 TSRMLS_DC
)
775 if (Z_TYPE_P(zvalue
) == IS_ARRAY
|| Z_TYPE_P(zvalue
) == IS_OBJECT
) {
776 /* treat as arguments, unless we care for dimensions or rfc5987 */
777 if (flags
& PHP_HTTP_PARAMS_DIMENSION
) {
778 php_http_buffer_t
*keybuf
= php_http_buffer_from_string(key_str
, key_len
);
779 prepare_dimension(buf
, keybuf
, zvalue
, pss
, psl
, vss
, vsl
, flags TSRMLS_CC
);
780 php_http_buffer_free(&keybuf
);
781 } else if (rfc5987
) {
782 shift_key(buf
, key_str
, key_len
, pss
, psl
, flags TSRMLS_CC
);
783 shift_rfc5987(buf
, zvalue
, vss
, vsl
, flags TSRMLS_CC
);
785 shift_arg(buf
, key_str
, key_len
, zvalue
, ass
, asl
, vss
, vsl
, flags TSRMLS_CC
);
788 shift_key(buf
, key_str
, key_len
, pss
, psl
, flags TSRMLS_CC
);
789 shift_val(buf
, zvalue
, vss
, vsl
, flags TSRMLS_CC
);
793 php_http_buffer_t
*php_http_params_to_string(php_http_buffer_t
*buf
, HashTable
*params
, const char *pss
, size_t psl
, const char *ass
, size_t asl
, const char *vss
, size_t vsl
, unsigned flags TSRMLS_DC
)
796 HashPosition pos
, pos1
;
797 php_http_array_hashkey_t key
= php_http_array_hashkey_init(0), key1
= php_http_array_hashkey_init(0);
798 zend_bool rfc5987
= 0;
801 buf
= php_http_buffer_init(NULL
);
804 FOREACH_HASH_KEYVAL(pos
, params
, key
, zparam
) {
805 zval
**zvalue
, **zargs
;
807 if (Z_TYPE_PP(zparam
) != IS_ARRAY
) {
810 if (SUCCESS
!= zend_hash_find(Z_ARRVAL_PP(zparam
), ZEND_STRS("value"), (void *) &zvalue
)) {
811 if (SUCCESS
!= zend_hash_find(Z_ARRVAL_PP(zparam
), ZEND_STRS("*rfc5987*"), (void *) &zvalue
)) {
819 php_http_array_hashkey_stringify(&key
);
820 shift_param(buf
, key
.str
, key
.len
- 1, *zvalue
, pss
, psl
, ass
, asl
, vss
, vsl
, flags
, rfc5987 TSRMLS_CC
);
821 php_http_array_hashkey_stringfree(&key
);
823 if (Z_TYPE_PP(zparam
) == IS_ARRAY
&& SUCCESS
!= zend_hash_find(Z_ARRVAL_PP(zparam
), ZEND_STRS("arguments"), (void *) &zvalue
)) {
824 if (zvalue
== zparam
) {
830 if (Z_TYPE_PP(zvalue
) == IS_ARRAY
) {
831 FOREACH_KEYVAL(pos1
, *zvalue
, key1
, zargs
) {
832 if (zvalue
== zparam
&& key1
.type
== HASH_KEY_IS_STRING
&& !strcmp(key1
.str
, "value")) {
836 php_http_array_hashkey_stringify(&key1
);
837 shift_arg(buf
, key1
.str
, key1
.len
- 1, *zargs
, ass
, asl
, vss
, vsl
, flags TSRMLS_CC
);
838 php_http_array_hashkey_stringfree(&key1
);
843 php_http_buffer_shrink(buf
);
844 php_http_buffer_fix(buf
);
849 php_http_params_token_t
**php_http_params_separator_init(zval
*zv TSRMLS_DC
)
853 php_http_params_token_t
**ret
, **tmp
;
859 zv
= php_http_ztyp(IS_ARRAY
, zv
);
860 ret
= ecalloc(zend_hash_num_elements(Z_ARRVAL_P(zv
)) + 1, sizeof(*ret
));
863 FOREACH_VAL(pos
, zv
, sep
) {
864 zval
*zt
= php_http_ztyp(IS_STRING
, *sep
);
866 if (Z_STRLEN_P(zt
)) {
867 *tmp
= emalloc(sizeof(**tmp
));
868 (*tmp
)->str
= estrndup(Z_STRVAL_P(zt
), (*tmp
)->len
= Z_STRLEN_P(zt
));
879 void php_http_params_separator_free(php_http_params_token_t
**separator
)
881 php_http_params_token_t
**sep
= separator
;
884 STR_FREE((*sep
)->str
);
892 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams___construct
, 0, 0, 0)
893 ZEND_ARG_INFO(0, params
)
894 ZEND_ARG_INFO(0, param_sep
)
895 ZEND_ARG_INFO(0, arg_sep
)
896 ZEND_ARG_INFO(0, val_sep
)
897 ZEND_ARG_INFO(0, flags
)
899 PHP_METHOD(HttpParams
, __construct
)
901 zval
*zcopy
, *zparams
= NULL
, *param_sep
= NULL
, *arg_sep
= NULL
, *val_sep
= NULL
;
902 long flags
= PHP_HTTP_PARAMS_DEFAULT
;
903 zend_error_handling zeh
;
905 php_http_expect(SUCCESS
== zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "|z!/z/z/z/l", &zparams
, ¶m_sep
, &arg_sep
, &val_sep
, &flags
), invalid_arg
, return);
907 zend_replace_error_handling(EH_THROW
, php_http_exception_runtime_class_entry
, &zeh TSRMLS_CC
);
909 switch (ZEND_NUM_ARGS()) {
911 zend_update_property_long(php_http_params_class_entry
, getThis(), ZEND_STRL("flags"), flags TSRMLS_CC
);
914 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("val_sep"), val_sep TSRMLS_CC
);
917 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("arg_sep"), arg_sep TSRMLS_CC
);
920 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("param_sep"), param_sep TSRMLS_CC
);
925 switch (Z_TYPE_P(zparams
)) {
928 zcopy
= php_http_zsep(1, IS_ARRAY
, zparams
);
929 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), zcopy TSRMLS_CC
);
930 zval_ptr_dtor(&zcopy
);
933 zcopy
= php_http_ztyp(IS_STRING
, zparams
);
934 if (Z_STRLEN_P(zcopy
)) {
935 php_http_params_opts_t opts
= {
936 {Z_STRVAL_P(zcopy
), Z_STRLEN_P(zcopy
)},
937 php_http_params_separator_init(zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("param_sep"), 0 TSRMLS_CC
) TSRMLS_CC
),
938 php_http_params_separator_init(zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("arg_sep"), 0 TSRMLS_CC
) TSRMLS_CC
),
939 php_http_params_separator_init(zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("val_sep"), 0 TSRMLS_CC
) TSRMLS_CC
),
943 MAKE_STD_ZVAL(zparams
);
945 php_http_params_parse(Z_ARRVAL_P(zparams
), &opts TSRMLS_CC
);
946 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), zparams TSRMLS_CC
);
947 zval_ptr_dtor(&zparams
);
949 php_http_params_separator_free(opts
.param
);
950 php_http_params_separator_free(opts
.arg
);
951 php_http_params_separator_free(opts
.val
);
953 zval_ptr_dtor(&zcopy
);
957 MAKE_STD_ZVAL(zparams
);
959 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), zparams TSRMLS_CC
);
960 zval_ptr_dtor(&zparams
);
963 zend_restore_error_handling(&zeh TSRMLS_CC
);
966 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams_toArray
, 0, 0, 0)
968 PHP_METHOD(HttpParams
, toArray
)
972 if (SUCCESS
!= zend_parse_parameters_none()) {
975 zparams
= zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), 0 TSRMLS_CC
);
976 RETURN_ZVAL(zparams
, 1, 0);
979 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams_toString
, 0, 0, 0)
981 PHP_METHOD(HttpParams
, toString
)
983 zval
**tmp
, *zparams
, *zpsep
, *zasep
, *zvsep
, *zflags
;
984 php_http_buffer_t buf
;
986 zparams
= php_http_zsep(1, IS_ARRAY
, zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), 0 TSRMLS_CC
));
987 zflags
= php_http_ztyp(IS_LONG
, zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("flags"), 0 TSRMLS_CC
));
989 zpsep
= zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("param_sep"), 0 TSRMLS_CC
);
990 if (Z_TYPE_P(zpsep
) == IS_ARRAY
&& SUCCESS
== zend_hash_get_current_data(Z_ARRVAL_P(zpsep
), (void *) &tmp
)) {
991 zpsep
= php_http_ztyp(IS_STRING
, *tmp
);
993 zpsep
= php_http_ztyp(IS_STRING
, zpsep
);
995 zasep
= zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("arg_sep"), 0 TSRMLS_CC
);
996 if (Z_TYPE_P(zasep
) == IS_ARRAY
&& SUCCESS
== zend_hash_get_current_data(Z_ARRVAL_P(zasep
), (void *) &tmp
)) {
997 zasep
= php_http_ztyp(IS_STRING
, *tmp
);
999 zasep
= php_http_ztyp(IS_STRING
, zasep
);
1001 zvsep
= zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("val_sep"), 0 TSRMLS_CC
);
1002 if (Z_TYPE_P(zvsep
) == IS_ARRAY
&& SUCCESS
== zend_hash_get_current_data(Z_ARRVAL_P(zvsep
), (void *) &tmp
)) {
1003 zvsep
= php_http_ztyp(IS_STRING
, *tmp
);
1005 zvsep
= php_http_ztyp(IS_STRING
, zvsep
);
1008 php_http_buffer_init(&buf
);
1009 php_http_params_to_string(&buf
, Z_ARRVAL_P(zparams
), Z_STRVAL_P(zpsep
), Z_STRLEN_P(zpsep
), Z_STRVAL_P(zasep
), Z_STRLEN_P(zasep
), Z_STRVAL_P(zvsep
), Z_STRLEN_P(zvsep
), Z_LVAL_P(zflags
) TSRMLS_CC
);
1011 zval_ptr_dtor(&zparams
);
1012 zval_ptr_dtor(&zpsep
);
1013 zval_ptr_dtor(&zasep
);
1014 zval_ptr_dtor(&zvsep
);
1015 zval_ptr_dtor(&zflags
);
1017 RETVAL_PHP_HTTP_BUFFER_VAL(&buf
);
1020 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams_offsetExists
, 0, 0, 1)
1021 ZEND_ARG_INFO(0, name
)
1022 ZEND_END_ARG_INFO();
1023 PHP_METHOD(HttpParams
, offsetExists
)
1027 zval
**zparam
, *zparams
;
1029 if (SUCCESS
!= zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "s", &name_str
, &name_len
)) {
1033 zparams
= php_http_ztyp(IS_ARRAY
, zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), 0 TSRMLS_CC
));
1035 if (SUCCESS
== zend_symtable_find(Z_ARRVAL_P(zparams
), name_str
, name_len
+ 1, (void *) &zparam
)) {
1036 RETVAL_BOOL(Z_TYPE_PP(zparam
) != IS_NULL
);
1040 zval_ptr_dtor(&zparams
);
1043 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams_offsetGet
, 0, 0, 1)
1044 ZEND_ARG_INFO(0, name
)
1045 ZEND_END_ARG_INFO();
1046 PHP_METHOD(HttpParams
, offsetGet
)
1050 zval
**zparam
, *zparams
;
1052 if (SUCCESS
!= zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "s", &name_str
, &name_len
)) {
1056 zparams
= php_http_ztyp(IS_ARRAY
, zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), 0 TSRMLS_CC
));
1058 if (SUCCESS
== zend_symtable_find(Z_ARRVAL_P(zparams
), name_str
, name_len
+ 1, (void *) &zparam
)) {
1059 RETVAL_ZVAL(*zparam
, 1, 0);
1062 zval_ptr_dtor(&zparams
);
1065 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams_offsetUnset
, 0, 0, 1)
1066 ZEND_ARG_INFO(0, name
)
1067 ZEND_END_ARG_INFO();
1068 PHP_METHOD(HttpParams
, offsetUnset
)
1074 if (SUCCESS
!= zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "s", &name_str
, &name_len
)) {
1078 zparams
= php_http_zsep(1, IS_ARRAY
, zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), 0 TSRMLS_CC
));
1080 zend_symtable_del(Z_ARRVAL_P(zparams
), name_str
, name_len
+ 1);
1081 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), zparams TSRMLS_CC
);
1083 zval_ptr_dtor(&zparams
);
1086 ZEND_BEGIN_ARG_INFO_EX(ai_HttpParams_offsetSet
, 0, 0, 2)
1087 ZEND_ARG_INFO(0, name
)
1088 ZEND_ARG_INFO(0, value
)
1089 ZEND_END_ARG_INFO();
1090 PHP_METHOD(HttpParams
, offsetSet
)
1095 zval
**zparam
, *zparams
;
1097 if (SUCCESS
!= zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC
, "sz", &name_str
, &name_len
, &nvalue
)) {
1101 zparams
= php_http_zsep(1, IS_ARRAY
, zend_read_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), 0 TSRMLS_CC
));
1104 if (Z_TYPE_P(nvalue
) == IS_ARRAY
) {
1107 if (SUCCESS
== zend_symtable_find(Z_ARRVAL_P(zparams
), name_str
, name_len
+ 1, (void *) &zparam
)) {
1108 new_zparam
= php_http_zsep(1, IS_ARRAY
, *zparam
);
1109 array_join(Z_ARRVAL_P(nvalue
), Z_ARRVAL_P(new_zparam
), 0, 0);
1111 new_zparam
= nvalue
;
1112 Z_ADDREF_P(new_zparam
);
1114 add_assoc_zval_ex(zparams
, name_str
, name_len
+ 1, new_zparam
);
1118 if (SUCCESS
== zend_symtable_find(Z_ARRVAL_P(zparams
), name_str
, name_len
+ 1, (void *) &zparam
)) {
1119 tmp
= php_http_zsep(1, IS_ARRAY
, *zparam
);
1126 add_assoc_zval_ex(tmp
, ZEND_STRS("value"), nvalue
);
1127 add_assoc_zval_ex(zparams
, name_str
, name_len
+ 1, tmp
);
1130 zval
*tmp
= php_http_ztyp(IS_STRING
, nvalue
), *arr
;
1134 add_assoc_bool_ex(arr
, ZEND_STRS("value"), 1);
1135 add_assoc_zval_ex(zparams
, Z_STRVAL_P(tmp
), Z_STRLEN_P(tmp
) + 1, arr
);
1136 zval_ptr_dtor(&tmp
);
1139 zend_update_property(php_http_params_class_entry
, getThis(), ZEND_STRL("params"), zparams TSRMLS_CC
);
1140 zval_ptr_dtor(&zparams
);
1143 static zend_function_entry php_http_params_methods
[] = {
1144 PHP_ME(HttpParams
, __construct
, ai_HttpParams___construct
, ZEND_ACC_PUBLIC
|ZEND_ACC_CTOR
|ZEND_ACC_FINAL
)
1146 PHP_ME(HttpParams
, toArray
, ai_HttpParams_toArray
, ZEND_ACC_PUBLIC
)
1147 PHP_ME(HttpParams
, toString
, ai_HttpParams_toString
, ZEND_ACC_PUBLIC
)
1148 ZEND_MALIAS(HttpParams
, __toString
, toString
, ai_HttpParams_toString
, ZEND_ACC_PUBLIC
)
1150 PHP_ME(HttpParams
, offsetExists
, ai_HttpParams_offsetExists
, ZEND_ACC_PUBLIC
)
1151 PHP_ME(HttpParams
, offsetUnset
, ai_HttpParams_offsetUnset
, ZEND_ACC_PUBLIC
)
1152 PHP_ME(HttpParams
, offsetSet
, ai_HttpParams_offsetSet
, ZEND_ACC_PUBLIC
)
1153 PHP_ME(HttpParams
, offsetGet
, ai_HttpParams_offsetGet
, ZEND_ACC_PUBLIC
)
1155 EMPTY_FUNCTION_ENTRY
1158 zend_class_entry
*php_http_params_class_entry
;
1160 PHP_MINIT_FUNCTION(http_params
)
1162 zend_class_entry ce
= {0};
1164 INIT_NS_CLASS_ENTRY(ce
, "http", "Params", php_http_params_methods
);
1165 php_http_params_class_entry
= zend_register_internal_class(&ce TSRMLS_CC
);
1166 php_http_params_class_entry
->create_object
= php_http_params_object_new
;
1167 zend_class_implements(php_http_params_class_entry TSRMLS_CC
, 1, zend_ce_arrayaccess
);
1169 zend_declare_class_constant_stringl(php_http_params_class_entry
, ZEND_STRL("DEF_PARAM_SEP"), ZEND_STRL(",") TSRMLS_CC
);
1170 zend_declare_class_constant_stringl(php_http_params_class_entry
, ZEND_STRL("DEF_ARG_SEP"), ZEND_STRL(";") TSRMLS_CC
);
1171 zend_declare_class_constant_stringl(php_http_params_class_entry
, ZEND_STRL("DEF_VAL_SEP"), ZEND_STRL("=") TSRMLS_CC
);
1172 zend_declare_class_constant_stringl(php_http_params_class_entry
, ZEND_STRL("COOKIE_PARAM_SEP"), ZEND_STRL("") TSRMLS_CC
);
1174 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_RAW"), PHP_HTTP_PARAMS_RAW TSRMLS_CC
);
1175 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_ESCAPED"), PHP_HTTP_PARAMS_ESCAPED TSRMLS_CC
);
1176 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_URLENCODED"), PHP_HTTP_PARAMS_URLENCODED TSRMLS_CC
);
1177 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_DIMENSION"), PHP_HTTP_PARAMS_DIMENSION TSRMLS_CC
);
1178 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_RFC5987"), PHP_HTTP_PARAMS_RFC5987 TSRMLS_CC
);
1179 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_DEFAULT"), PHP_HTTP_PARAMS_DEFAULT TSRMLS_CC
);
1180 zend_declare_class_constant_long(php_http_params_class_entry
, ZEND_STRL("PARSE_QUERY"), PHP_HTTP_PARAMS_QUERY TSRMLS_CC
);
1182 zend_declare_property_null(php_http_params_class_entry
, ZEND_STRL("params"), ZEND_ACC_PUBLIC TSRMLS_CC
);
1183 zend_declare_property_stringl(php_http_params_class_entry
, ZEND_STRL("param_sep"), ZEND_STRL(","), ZEND_ACC_PUBLIC TSRMLS_CC
);
1184 zend_declare_property_stringl(php_http_params_class_entry
, ZEND_STRL("arg_sep"), ZEND_STRL(";"), ZEND_ACC_PUBLIC TSRMLS_CC
);
1185 zend_declare_property_stringl(php_http_params_class_entry
, ZEND_STRL("val_sep"), ZEND_STRL("="), ZEND_ACC_PUBLIC TSRMLS_CC
);
1186 zend_declare_property_long(php_http_params_class_entry
, ZEND_STRL("flags"), PHP_HTTP_PARAMS_DEFAULT
, ZEND_ACC_PUBLIC TSRMLS_CC
);
1196 * vim600: noet sw=4 ts=4 fdm=marker
1197 * vim<600: noet sw=4 ts=4