mirror of
https://github.com/php/php-src.git
synced 2025-08-18 15:08:55 +02:00

This patch removes the so called local variables defined per file basis for certain editors to properly show tab width, and similar settings. These are mainly used by Vim and Emacs editors yet with recent changes the once working definitions don't work anymore in Vim without custom plugins or additional configuration. Neither are these settings synced across the PHP code base. A simpler and better approach is EditorConfig and fixing code using some code style fixing tools in the future instead. This patch also removes the so called modelines for Vim. Modelines allow Vim editor specifically to set some editor configuration such as syntax highlighting, indentation style and tab width to be set in the first line or the last 5 lines per file basis. Since the php test files have syntax highlighting already set in most editors properly and EditorConfig takes care of the indentation settings, this patch removes these as well for the Vim 6.0 and newer versions. With the removal of local variables for certain editors such as Emacs and Vim, the footer is also probably not needed anymore when creating extensions using ext_skel.php script. Additionally, Vim modelines for setting php syntax and some editor settings has been removed from some *.phpt files. All these are mostly not relevant for phpt files neither work properly in the middle of the file.
728 lines
18 KiB
C
728 lines
18 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| PHP Version 7 |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| http://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Author: Jim Winstead <jimw@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
#include <sys/types.h>
|
|
|
|
#include "php.h"
|
|
|
|
#include "url.h"
|
|
#include "file.h"
|
|
#ifdef _OSD_POSIX
|
|
#ifndef APACHE
|
|
#error On this EBCDIC platform, PHP is only supported as an Apache module.
|
|
#else /*APACHE*/
|
|
#ifndef CHARSET_EBCDIC
|
|
#define CHARSET_EBCDIC /* this machine uses EBCDIC, not ASCII! */
|
|
#endif
|
|
#include "ebcdic.h"
|
|
#endif /*APACHE*/
|
|
#endif /*_OSD_POSIX*/
|
|
|
|
/* {{{ free_url
|
|
*/
|
|
PHPAPI void php_url_free(php_url *theurl)
|
|
{
|
|
if (theurl->scheme)
|
|
zend_string_release_ex(theurl->scheme, 0);
|
|
if (theurl->user)
|
|
zend_string_release_ex(theurl->user, 0);
|
|
if (theurl->pass)
|
|
zend_string_release_ex(theurl->pass, 0);
|
|
if (theurl->host)
|
|
zend_string_release_ex(theurl->host, 0);
|
|
if (theurl->path)
|
|
zend_string_release_ex(theurl->path, 0);
|
|
if (theurl->query)
|
|
zend_string_release_ex(theurl->query, 0);
|
|
if (theurl->fragment)
|
|
zend_string_release_ex(theurl->fragment, 0);
|
|
efree(theurl);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ php_replace_controlchars
|
|
*/
|
|
PHPAPI char *php_replace_controlchars_ex(char *str, size_t len)
|
|
{
|
|
unsigned char *s = (unsigned char *)str;
|
|
unsigned char *e = (unsigned char *)str + len;
|
|
|
|
if (!str) {
|
|
return (NULL);
|
|
}
|
|
|
|
while (s < e) {
|
|
|
|
if (iscntrl(*s)) {
|
|
*s='_';
|
|
}
|
|
s++;
|
|
}
|
|
|
|
return (str);
|
|
}
|
|
/* }}} */
|
|
|
|
PHPAPI char *php_replace_controlchars(char *str)
|
|
{
|
|
return php_replace_controlchars_ex(str, strlen(str));
|
|
}
|
|
|
|
PHPAPI php_url *php_url_parse(char const *str)
|
|
{
|
|
return php_url_parse_ex(str, strlen(str));
|
|
}
|
|
|
|
/* {{{ php_url_parse
|
|
*/
|
|
PHPAPI php_url *php_url_parse_ex(char const *str, size_t length)
|
|
{
|
|
char port_buf[6];
|
|
php_url *ret = ecalloc(1, sizeof(php_url));
|
|
char const *s, *e, *p, *pp, *ue;
|
|
|
|
s = str;
|
|
ue = s + length;
|
|
|
|
/* parse scheme */
|
|
if ((e = memchr(s, ':', length)) && e != s) {
|
|
/* validate scheme */
|
|
p = s;
|
|
while (p < e) {
|
|
/* scheme = 1*[ lowalpha | digit | "+" | "-" | "." ] */
|
|
if (!isalpha(*p) && !isdigit(*p) && *p != '+' && *p != '.' && *p != '-') {
|
|
if (e + 1 < ue && e < s + strcspn(s, "?#")) {
|
|
goto parse_port;
|
|
} else if (s + 1 < ue && *s == '/' && *(s + 1) == '/') { /* relative-scheme URL */
|
|
s += 2;
|
|
e = 0;
|
|
goto parse_host;
|
|
} else {
|
|
goto just_path;
|
|
}
|
|
}
|
|
p++;
|
|
}
|
|
|
|
if (e + 1 == ue) { /* only scheme is available */
|
|
ret->scheme = zend_string_init(s, (e - s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->scheme), ZSTR_LEN(ret->scheme));
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* certain schemas like mailto: and zlib: may not have any / after them
|
|
* this check ensures we support those.
|
|
*/
|
|
if (*(e+1) != '/') {
|
|
/* check if the data we get is a port this allows us to
|
|
* correctly parse things like a.com:80
|
|
*/
|
|
p = e + 1;
|
|
while (p < ue && isdigit(*p)) {
|
|
p++;
|
|
}
|
|
|
|
if ((p == ue || *p == '/') && (p - e) < 7) {
|
|
goto parse_port;
|
|
}
|
|
|
|
ret->scheme = zend_string_init(s, (e-s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->scheme), ZSTR_LEN(ret->scheme));
|
|
|
|
s = e + 1;
|
|
goto just_path;
|
|
} else {
|
|
ret->scheme = zend_string_init(s, (e-s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->scheme), ZSTR_LEN(ret->scheme));
|
|
|
|
if (e + 2 < ue && *(e + 2) == '/') {
|
|
s = e + 3;
|
|
if (zend_string_equals_literal_ci(ret->scheme, "file")) {
|
|
if (e + 3 < ue && *(e + 3) == '/') {
|
|
/* support windows drive letters as in:
|
|
file:///c:/somedir/file.txt
|
|
*/
|
|
if (e + 5 < ue && *(e + 5) == ':') {
|
|
s = e + 4;
|
|
}
|
|
goto just_path;
|
|
}
|
|
}
|
|
} else {
|
|
s = e + 1;
|
|
goto just_path;
|
|
}
|
|
}
|
|
} else if (e) { /* no scheme; starts with colon: look for port */
|
|
parse_port:
|
|
p = e + 1;
|
|
pp = p;
|
|
|
|
while (pp < ue && pp - p < 6 && isdigit(*pp)) {
|
|
pp++;
|
|
}
|
|
|
|
if (pp - p > 0 && pp - p < 6 && (pp == ue || *pp == '/')) {
|
|
zend_long port;
|
|
memcpy(port_buf, p, (pp - p));
|
|
port_buf[pp - p] = '\0';
|
|
port = ZEND_STRTOL(port_buf, NULL, 10);
|
|
if (port > 0 && port <= 65535) {
|
|
ret->port = (unsigned short) port;
|
|
if (s + 1 < ue && *s == '/' && *(s + 1) == '/') { /* relative-scheme URL */
|
|
s += 2;
|
|
}
|
|
} else {
|
|
php_url_free(ret);
|
|
return NULL;
|
|
}
|
|
} else if (p == pp && pp == ue) {
|
|
php_url_free(ret);
|
|
return NULL;
|
|
} else if (s + 1 < ue && *s == '/' && *(s + 1) == '/') { /* relative-scheme URL */
|
|
s += 2;
|
|
} else {
|
|
goto just_path;
|
|
}
|
|
} else if (s + 1 < ue && *s == '/' && *(s + 1) == '/') { /* relative-scheme URL */
|
|
s += 2;
|
|
} else {
|
|
goto just_path;
|
|
}
|
|
|
|
parse_host:
|
|
/* Binary-safe strcspn(s, "/?#") */
|
|
e = ue;
|
|
if ((p = memchr(s, '/', e - s))) {
|
|
e = p;
|
|
}
|
|
if ((p = memchr(s, '?', e - s))) {
|
|
e = p;
|
|
}
|
|
if ((p = memchr(s, '#', e - s))) {
|
|
e = p;
|
|
}
|
|
|
|
/* check for login and password */
|
|
if ((p = zend_memrchr(s, '@', (e-s)))) {
|
|
if ((pp = memchr(s, ':', (p-s)))) {
|
|
ret->user = zend_string_init(s, (pp-s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->user), ZSTR_LEN(ret->user));
|
|
|
|
pp++;
|
|
ret->pass = zend_string_init(pp, (p-pp), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->pass), ZSTR_LEN(ret->pass));
|
|
} else {
|
|
ret->user = zend_string_init(s, (p-s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->user), ZSTR_LEN(ret->user));
|
|
}
|
|
|
|
s = p + 1;
|
|
}
|
|
|
|
/* check for port */
|
|
if (s < ue && *s == '[' && *(e-1) == ']') {
|
|
/* Short circuit portscan,
|
|
we're dealing with an
|
|
IPv6 embedded address */
|
|
p = NULL;
|
|
} else {
|
|
p = zend_memrchr(s, ':', (e-s));
|
|
}
|
|
|
|
if (p) {
|
|
if (!ret->port) {
|
|
p++;
|
|
if (e-p > 5) { /* port cannot be longer then 5 characters */
|
|
php_url_free(ret);
|
|
return NULL;
|
|
} else if (e - p > 0) {
|
|
zend_long port;
|
|
memcpy(port_buf, p, (e - p));
|
|
port_buf[e - p] = '\0';
|
|
port = ZEND_STRTOL(port_buf, NULL, 10);
|
|
if (port > 0 && port <= 65535) {
|
|
ret->port = (unsigned short)port;
|
|
} else {
|
|
php_url_free(ret);
|
|
return NULL;
|
|
}
|
|
}
|
|
p--;
|
|
}
|
|
} else {
|
|
p = e;
|
|
}
|
|
|
|
/* check if we have a valid host, if we don't reject the string as url */
|
|
if ((p-s) < 1) {
|
|
php_url_free(ret);
|
|
return NULL;
|
|
}
|
|
|
|
ret->host = zend_string_init(s, (p-s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->host), ZSTR_LEN(ret->host));
|
|
|
|
if (e == ue) {
|
|
return ret;
|
|
}
|
|
|
|
s = e;
|
|
|
|
just_path:
|
|
|
|
e = ue;
|
|
p = memchr(s, '#', (e - s));
|
|
if (p) {
|
|
p++;
|
|
if (p < e) {
|
|
ret->fragment = zend_string_init(p, (e - p), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->fragment), ZSTR_LEN(ret->fragment));
|
|
}
|
|
e = p-1;
|
|
}
|
|
|
|
p = memchr(s, '?', (e - s));
|
|
if (p) {
|
|
p++;
|
|
if (p < e) {
|
|
ret->query = zend_string_init(p, (e - p), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->query), ZSTR_LEN(ret->query));
|
|
}
|
|
e = p-1;
|
|
}
|
|
|
|
if (s < e || s == ue) {
|
|
ret->path = zend_string_init(s, (e - s), 0);
|
|
php_replace_controlchars_ex(ZSTR_VAL(ret->path), ZSTR_LEN(ret->path));
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto mixed parse_url(string url, [int url_component])
|
|
Parse a URL and return its components */
|
|
PHP_FUNCTION(parse_url)
|
|
{
|
|
char *str;
|
|
size_t str_len;
|
|
php_url *resource;
|
|
zend_long key = -1;
|
|
zval tmp;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 2)
|
|
Z_PARAM_STRING(str, str_len)
|
|
Z_PARAM_OPTIONAL
|
|
Z_PARAM_LONG(key)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
resource = php_url_parse_ex(str, str_len);
|
|
if (resource == NULL) {
|
|
/* @todo Find a method to determine why php_url_parse_ex() failed */
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (key > -1) {
|
|
switch (key) {
|
|
case PHP_URL_SCHEME:
|
|
if (resource->scheme != NULL) RETVAL_STR_COPY(resource->scheme);
|
|
break;
|
|
case PHP_URL_HOST:
|
|
if (resource->host != NULL) RETVAL_STR_COPY(resource->host);
|
|
break;
|
|
case PHP_URL_PORT:
|
|
if (resource->port != 0) RETVAL_LONG(resource->port);
|
|
break;
|
|
case PHP_URL_USER:
|
|
if (resource->user != NULL) RETVAL_STR_COPY(resource->user);
|
|
break;
|
|
case PHP_URL_PASS:
|
|
if (resource->pass != NULL) RETVAL_STR_COPY(resource->pass);
|
|
break;
|
|
case PHP_URL_PATH:
|
|
if (resource->path != NULL) RETVAL_STR_COPY(resource->path);
|
|
break;
|
|
case PHP_URL_QUERY:
|
|
if (resource->query != NULL) RETVAL_STR_COPY(resource->query);
|
|
break;
|
|
case PHP_URL_FRAGMENT:
|
|
if (resource->fragment != NULL) RETVAL_STR_COPY(resource->fragment);
|
|
break;
|
|
default:
|
|
php_error_docref(NULL, E_WARNING, "Invalid URL component identifier " ZEND_LONG_FMT, key);
|
|
RETVAL_FALSE;
|
|
}
|
|
goto done;
|
|
}
|
|
|
|
/* allocate an array for return */
|
|
array_init(return_value);
|
|
|
|
/* add the various elements to the array */
|
|
if (resource->scheme != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->scheme);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_SCHEME), &tmp);
|
|
}
|
|
if (resource->host != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->host);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_HOST), &tmp);
|
|
}
|
|
if (resource->port != 0) {
|
|
ZVAL_LONG(&tmp, resource->port);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_PORT), &tmp);
|
|
}
|
|
if (resource->user != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->user);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_USER), &tmp);
|
|
}
|
|
if (resource->pass != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->pass);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_PASS), &tmp);
|
|
}
|
|
if (resource->path != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->path);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_PATH), &tmp);
|
|
}
|
|
if (resource->query != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->query);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_QUERY), &tmp);
|
|
}
|
|
if (resource->fragment != NULL) {
|
|
ZVAL_STR_COPY(&tmp, resource->fragment);
|
|
zend_hash_add_new(Z_ARRVAL_P(return_value), ZSTR_KNOWN(ZEND_STR_FRAGMENT), &tmp);
|
|
}
|
|
done:
|
|
php_url_free(resource);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ php_htoi
|
|
*/
|
|
static int php_htoi(char *s)
|
|
{
|
|
int value;
|
|
int c;
|
|
|
|
c = ((unsigned char *)s)[0];
|
|
if (isupper(c))
|
|
c = tolower(c);
|
|
value = (c >= '0' && c <= '9' ? c - '0' : c - 'a' + 10) * 16;
|
|
|
|
c = ((unsigned char *)s)[1];
|
|
if (isupper(c))
|
|
c = tolower(c);
|
|
value += c >= '0' && c <= '9' ? c - '0' : c - 'a' + 10;
|
|
|
|
return (value);
|
|
}
|
|
/* }}} */
|
|
|
|
/* rfc1738:
|
|
|
|
...The characters ";",
|
|
"/", "?", ":", "@", "=" and "&" are the characters which may be
|
|
reserved for special meaning within a scheme...
|
|
|
|
...Thus, only alphanumerics, the special characters "$-_.+!*'(),", and
|
|
reserved characters used for their reserved purposes may be used
|
|
unencoded within a URL...
|
|
|
|
For added safety, we only leave -_. unencoded.
|
|
*/
|
|
|
|
static unsigned char hexchars[] = "0123456789ABCDEF";
|
|
|
|
/* {{{ php_url_encode
|
|
*/
|
|
PHPAPI zend_string *php_url_encode(char const *s, size_t len)
|
|
{
|
|
register unsigned char c;
|
|
unsigned char *to;
|
|
unsigned char const *from, *end;
|
|
zend_string *start;
|
|
|
|
from = (unsigned char *)s;
|
|
end = (unsigned char *)s + len;
|
|
start = zend_string_safe_alloc(3, len, 0, 0);
|
|
to = (unsigned char*)ZSTR_VAL(start);
|
|
|
|
while (from < end) {
|
|
c = *from++;
|
|
|
|
if (c == ' ') {
|
|
*to++ = '+';
|
|
#ifndef CHARSET_EBCDIC
|
|
} else if ((c < '0' && c != '-' && c != '.') ||
|
|
(c < 'A' && c > '9') ||
|
|
(c > 'Z' && c < 'a' && c != '_') ||
|
|
(c > 'z')) {
|
|
to[0] = '%';
|
|
to[1] = hexchars[c >> 4];
|
|
to[2] = hexchars[c & 15];
|
|
to += 3;
|
|
#else /*CHARSET_EBCDIC*/
|
|
} else if (!isalnum(c) && strchr("_-.", c) == NULL) {
|
|
/* Allow only alphanumeric chars and '_', '-', '.'; escape the rest */
|
|
to[0] = '%';
|
|
to[1] = hexchars[os_toascii[c] >> 4];
|
|
to[2] = hexchars[os_toascii[c] & 15];
|
|
to += 3;
|
|
#endif /*CHARSET_EBCDIC*/
|
|
} else {
|
|
*to++ = c;
|
|
}
|
|
}
|
|
*to = '\0';
|
|
|
|
start = zend_string_truncate(start, to - (unsigned char*)ZSTR_VAL(start), 0);
|
|
|
|
return start;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string urlencode(string str)
|
|
URL-encodes string */
|
|
PHP_FUNCTION(urlencode)
|
|
{
|
|
zend_string *in_str;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_STR(in_str)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
RETURN_STR(php_url_encode(ZSTR_VAL(in_str), ZSTR_LEN(in_str)));
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string urldecode(string str)
|
|
Decodes URL-encoded string */
|
|
PHP_FUNCTION(urldecode)
|
|
{
|
|
zend_string *in_str, *out_str;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_STR(in_str)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
out_str = zend_string_init(ZSTR_VAL(in_str), ZSTR_LEN(in_str), 0);
|
|
ZSTR_LEN(out_str) = php_url_decode(ZSTR_VAL(out_str), ZSTR_LEN(out_str));
|
|
|
|
RETURN_NEW_STR(out_str);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ php_url_decode
|
|
*/
|
|
PHPAPI size_t php_url_decode(char *str, size_t len)
|
|
{
|
|
char *dest = str;
|
|
char *data = str;
|
|
|
|
while (len--) {
|
|
if (*data == '+') {
|
|
*dest = ' ';
|
|
}
|
|
else if (*data == '%' && len >= 2 && isxdigit((int) *(data + 1))
|
|
&& isxdigit((int) *(data + 2))) {
|
|
#ifndef CHARSET_EBCDIC
|
|
*dest = (char) php_htoi(data + 1);
|
|
#else
|
|
*dest = os_toebcdic[(char) php_htoi(data + 1)];
|
|
#endif
|
|
data += 2;
|
|
len -= 2;
|
|
} else {
|
|
*dest = *data;
|
|
}
|
|
data++;
|
|
dest++;
|
|
}
|
|
*dest = '\0';
|
|
return dest - str;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ php_raw_url_encode
|
|
*/
|
|
PHPAPI zend_string *php_raw_url_encode(char const *s, size_t len)
|
|
{
|
|
register size_t x, y;
|
|
zend_string *str;
|
|
char *ret;
|
|
|
|
str = zend_string_safe_alloc(3, len, 0, 0);
|
|
ret = ZSTR_VAL(str);
|
|
for (x = 0, y = 0; len--; x++, y++) {
|
|
char c = s[x];
|
|
|
|
ret[y] = c;
|
|
#ifndef CHARSET_EBCDIC
|
|
if ((c < '0' && c != '-' && c != '.') ||
|
|
(c < 'A' && c > '9') ||
|
|
(c > 'Z' && c < 'a' && c != '_') ||
|
|
(c > 'z' && c != '~')) {
|
|
ret[y++] = '%';
|
|
ret[y++] = hexchars[(unsigned char) c >> 4];
|
|
ret[y] = hexchars[(unsigned char) c & 15];
|
|
#else /*CHARSET_EBCDIC*/
|
|
if (!isalnum(c) && strchr("_-.~", c) != NULL) {
|
|
ret[y++] = '%';
|
|
ret[y++] = hexchars[os_toascii[(unsigned char) c] >> 4];
|
|
ret[y] = hexchars[os_toascii[(unsigned char) c] & 15];
|
|
#endif /*CHARSET_EBCDIC*/
|
|
}
|
|
}
|
|
ret[y] = '\0';
|
|
str = zend_string_truncate(str, y, 0);
|
|
|
|
return str;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string rawurlencode(string str)
|
|
URL-encodes string */
|
|
PHP_FUNCTION(rawurlencode)
|
|
{
|
|
zend_string *in_str;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_STR(in_str)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
RETURN_STR(php_raw_url_encode(ZSTR_VAL(in_str), ZSTR_LEN(in_str)));
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto string rawurldecode(string str)
|
|
Decodes URL-encodes string */
|
|
PHP_FUNCTION(rawurldecode)
|
|
{
|
|
zend_string *in_str, *out_str;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 1)
|
|
Z_PARAM_STR(in_str)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
out_str = zend_string_init(ZSTR_VAL(in_str), ZSTR_LEN(in_str), 0);
|
|
ZSTR_LEN(out_str) = php_raw_url_decode(ZSTR_VAL(out_str), ZSTR_LEN(out_str));
|
|
|
|
RETURN_NEW_STR(out_str);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ php_raw_url_decode
|
|
*/
|
|
PHPAPI size_t php_raw_url_decode(char *str, size_t len)
|
|
{
|
|
char *dest = str;
|
|
char *data = str;
|
|
|
|
while (len--) {
|
|
if (*data == '%' && len >= 2 && isxdigit((int) *(data + 1))
|
|
&& isxdigit((int) *(data + 2))) {
|
|
#ifndef CHARSET_EBCDIC
|
|
*dest = (char) php_htoi(data + 1);
|
|
#else
|
|
*dest = os_toebcdic[(char) php_htoi(data + 1)];
|
|
#endif
|
|
data += 2;
|
|
len -= 2;
|
|
} else {
|
|
*dest = *data;
|
|
}
|
|
data++;
|
|
dest++;
|
|
}
|
|
*dest = '\0';
|
|
return dest - str;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ proto array get_headers(string url[, int format[, resource context]])
|
|
fetches all the headers sent by the server in response to a HTTP request */
|
|
PHP_FUNCTION(get_headers)
|
|
{
|
|
char *url;
|
|
size_t url_len;
|
|
php_stream *stream;
|
|
zval *prev_val, *hdr = NULL;
|
|
zend_long format = 0;
|
|
zval *zcontext = NULL;
|
|
php_stream_context *context;
|
|
|
|
ZEND_PARSE_PARAMETERS_START(1, 3)
|
|
Z_PARAM_STRING(url, url_len)
|
|
Z_PARAM_OPTIONAL
|
|
Z_PARAM_LONG(format)
|
|
Z_PARAM_RESOURCE_EX(zcontext, 1, 0)
|
|
ZEND_PARSE_PARAMETERS_END();
|
|
|
|
context = php_stream_context_from_zval(zcontext, 0);
|
|
|
|
if (!(stream = php_stream_open_wrapper_ex(url, "r", REPORT_ERRORS | STREAM_USE_URL | STREAM_ONLY_GET_HEADERS, NULL, context))) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (Z_TYPE(stream->wrapperdata) != IS_ARRAY) {
|
|
php_stream_close(stream);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
array_init(return_value);
|
|
|
|
ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(&stream->wrapperdata), hdr) {
|
|
if (Z_TYPE_P(hdr) != IS_STRING) {
|
|
continue;
|
|
}
|
|
if (!format) {
|
|
no_name_header:
|
|
add_next_index_str(return_value, zend_string_copy(Z_STR_P(hdr)));
|
|
} else {
|
|
char c;
|
|
char *s, *p;
|
|
|
|
if ((p = strchr(Z_STRVAL_P(hdr), ':'))) {
|
|
c = *p;
|
|
*p = '\0';
|
|
s = p + 1;
|
|
while (isspace((int)*(unsigned char *)s)) {
|
|
s++;
|
|
}
|
|
|
|
if ((prev_val = zend_hash_str_find(Z_ARRVAL_P(return_value), Z_STRVAL_P(hdr), (p - Z_STRVAL_P(hdr)))) == NULL) {
|
|
add_assoc_stringl_ex(return_value, Z_STRVAL_P(hdr), (p - Z_STRVAL_P(hdr)), s, (Z_STRLEN_P(hdr) - (s - Z_STRVAL_P(hdr))));
|
|
} else { /* some headers may occur more than once, therefor we need to remake the string into an array */
|
|
convert_to_array(prev_val);
|
|
add_next_index_stringl(prev_val, s, (Z_STRLEN_P(hdr) - (s - Z_STRVAL_P(hdr))));
|
|
}
|
|
|
|
*p = c;
|
|
} else {
|
|
goto no_name_header;
|
|
}
|
|
}
|
|
} ZEND_HASH_FOREACH_END();
|
|
|
|
php_stream_close(stream);
|
|
}
|
|
/* }}} */
|