/* +----------------------------------------------------------------------+ | Swoole | +----------------------------------------------------------------------+ | This source file is subject to version 2.0 of the Apache 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.apache.org/licenses/LICENSE-2.0.html | | If you did not receive a copy of the Apache2.0 license and are unable| | to obtain it through the world-wide-web, please send a note to | | license@swoole.com so we can mail you a copy immediately. | +----------------------------------------------------------------------+ | Author: Tianfeng Han | +----------------------------------------------------------------------+ */ #include "php_swoole.h" #include "swoole_http.h" #ifdef SW_USE_HTTP2 #include "swoole_http_v2_client.h" static zend_class_entry swoole_http2_client_ce; static zend_class_entry *swoole_http2_client_class_entry_ptr; static zend_class_entry swoole_http2_response_ce; zend_class_entry *swoole_http2_response_class_entry_ptr; swString *cookie_buffer = NULL; enum { HTTP2_CLIENT_PROPERTY_INDEX = 3, }; typedef struct { char *uri; uint32_t uri_len; uint32_t stream_id; uint8_t type; zval *callback; zval *data; #if PHP_MAJOR_VERSION >= 7 zval _callback; zval _data; #endif } http2_client_request; static PHP_METHOD(swoole_http2_client, __construct); static PHP_METHOD(swoole_http2_client, __destruct); static PHP_METHOD(swoole_http2_client, onConnect); static PHP_METHOD(swoole_http2_client, onError); static PHP_METHOD(swoole_http2_client, onReceive); static PHP_METHOD(swoole_http2_client, onClose); static PHP_METHOD(swoole_http2_client, setHeaders); static PHP_METHOD(swoole_http2_client, setCookies); static PHP_METHOD(swoole_http2_client, get); static PHP_METHOD(swoole_http2_client, post); static PHP_METHOD(swoole_http2_client, openStream); static PHP_METHOD(swoole_http2_client, push); static PHP_METHOD(swoole_http2_client, closeStream); static void http2_client_send_request(zval *zobject, http2_client_request *req TSRMLS_DC); static void http2_client_send_stream_request(zval *zobject, http2_client_request *req TSRMLS_DC); static void http2_client_send_all_requests(zval *zobject TSRMLS_DC); static void http2_client_request_free(void *ptr); static void http2_client_stream_free(void *ptr); static const zend_function_entry swoole_http2_client_methods[] = { PHP_ME(swoole_http2_client, __construct, NULL, ZEND_ACC_PUBLIC | ZEND_ACC_CTOR) PHP_ME(swoole_http2_client, __destruct, NULL, ZEND_ACC_PUBLIC | ZEND_ACC_DTOR) PHP_ME(swoole_http2_client, setHeaders, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, setCookies, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, get, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, post, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, onConnect, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, onError, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, onReceive, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, onClose, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, openStream, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, push, NULL, ZEND_ACC_PUBLIC) PHP_ME(swoole_http2_client, closeStream, NULL, ZEND_ACC_PUBLIC) PHP_FE_END }; void swoole_http2_client_init(int module_number TSRMLS_DC) { SWOOLE_INIT_CLASS_ENTRY(swoole_http2_client_ce, "swoole_http2_client", "Swoole\\Http2\\Client", swoole_http2_client_methods); swoole_http2_client_class_entry_ptr = sw_zend_register_internal_class_ex(&swoole_http2_client_ce, swoole_client_class_entry_ptr, "swoole_client" TSRMLS_CC); SWOOLE_CLASS_ALIAS(swoole_http2_client, "Swoole\\Http2\\Client"); zend_declare_property_null(swoole_http2_client_class_entry_ptr, SW_STRL("requestHeaders")-1, ZEND_ACC_PUBLIC TSRMLS_CC); zend_declare_property_null(swoole_http2_client_class_entry_ptr, SW_STRL("cookies")-1, ZEND_ACC_PUBLIC TSRMLS_CC); SWOOLE_INIT_CLASS_ENTRY(swoole_http2_response_ce, "swoole_http2_response", "Swoole\\Http2\\Response", NULL); swoole_http2_response_class_entry_ptr = zend_register_internal_class(&swoole_http2_response_ce TSRMLS_CC); SWOOLE_CLASS_ALIAS(swoole_http2_response, "Swoole\\Http2\\Response"); zend_declare_property_long(swoole_http2_response_class_entry_ptr, SW_STRL("errCode")-1, 0, ZEND_ACC_PUBLIC TSRMLS_CC); zend_declare_property_long(swoole_http2_response_class_entry_ptr, SW_STRL("statusCode")-1, 0, ZEND_ACC_PUBLIC TSRMLS_CC); zend_declare_property_null(swoole_http2_response_class_entry_ptr, SW_STRL("body")-1, ZEND_ACC_PUBLIC TSRMLS_CC); zend_declare_property_null(swoole_http2_response_class_entry_ptr, SW_STRL("streamId")-1, ZEND_ACC_PUBLIC TSRMLS_CC); if (cookie_buffer == NULL) { cookie_buffer = swString_new(8192); } } static PHP_METHOD(swoole_http2_client, __construct) { char *host; zend_size_t host_len; long port = 80; zend_bool ssl = SW_FALSE; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|lb", &host, &host_len, &port, &ssl) == FAILURE) { return; } if (host_len <= 0) { zend_throw_exception(swoole_exception_class_entry_ptr, "host is empty.", SW_ERROR_INVALID_PARAMS TSRMLS_CC); RETURN_FALSE; } http2_client_property *hcc; hcc = (http2_client_property*) emalloc(sizeof(http2_client_property)); bzero(hcc, sizeof(http2_client_property)); swoole_set_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX, hcc); hcc->requests = swLinkedList_new(0, http2_client_request_free); hcc->stream_requests = swLinkedList_new(0, http2_client_request_free); hcc->streams = swHashMap_new(8, http2_client_stream_free); hcc->stream_id = 1; zval *ztype; SW_MAKE_STD_ZVAL(ztype); long type = SW_FLAG_ASYNC | SW_SOCK_TCP; if (ssl) { type |= SW_SOCK_SSL; hcc->ssl = 1; } ZVAL_LONG(ztype, type); zval *zobject = getThis(); zval *retval = NULL; sw_zend_call_method_with_1_params(&zobject, swoole_client_class_entry_ptr, NULL, "__construct", &retval, ztype); if (retval) { sw_zval_ptr_dtor(&retval); } sw_zval_ptr_dtor(&ztype); hcc->host = estrndup(host, host_len); hcc->host_len = host_len; hcc->port = port; } static PHP_METHOD(swoole_http2_client, setHeaders) { zval *headers; if (zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC, "z", &headers) == FAILURE) { return; } zend_update_property(swoole_http2_client_class_entry_ptr, getThis(), ZEND_STRL("requestHeaders"), headers TSRMLS_CC); } static PHP_METHOD(swoole_http2_client, setCookies) { zval *cookies; if (zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC, "z", &cookies) == FAILURE) { return; } zend_update_property(swoole_http2_client_class_entry_ptr, getThis(), ZEND_STRL("cookies"), cookies TSRMLS_CC); } static int http2_client_build_header(zval *zobject, http2_client_request *req, char *buffer, int buffer_len TSRMLS_DC) { char *date_str = NULL; int ret; zval *zheader = sw_zend_read_property(swoole_http2_client_class_entry_ptr, zobject, ZEND_STRL("requestHeaders"), 1 TSRMLS_CC); int index = 0; int find_host = 0; nghttp2_nv nv[1024]; http2_client_property *hcc = swoole_get_property(zobject, HTTP2_CLIENT_PROPERTY_INDEX); if (req->type == HTTP_GET) { http2_add_header(&nv[index++], ZEND_STRL(":method"), ZEND_STRL("GET")); } else { http2_add_header(&nv[index++], ZEND_STRL(":method"), ZEND_STRL("POST")); } http2_add_header(&nv[index++], ZEND_STRL(":path"), req->uri, req->uri_len); if (hcc->ssl) { http2_add_header(&nv[index++], ZEND_STRL(":scheme"), ZEND_STRL("https")); } else { http2_add_header(&nv[index++], ZEND_STRL(":scheme"), ZEND_STRL("http")); } //Host index++; if (zheader && !ZVAL_IS_NULL(zheader)) { HashTable *ht = Z_ARRVAL_P(zheader); zval *value = NULL; char *key = NULL; uint32_t keylen = 0; int type; SW_HASHTABLE_FOREACH_START2(ht, key, keylen, type, value) { if (!key) { break; } if (*key == ':') { continue; } if (strncasecmp("Host", key, keylen) == 0) { http2_add_header(&nv[HTTP2_CLIENT_HOST_HEADER_INDEX], ZEND_STRL(":authority"), Z_STRVAL_P(value), Z_STRLEN_P(value)); find_host = 1; } else { http2_add_header(&nv[index++], key, keylen, Z_STRVAL_P(value), Z_STRLEN_P(value)); } } SW_HASHTABLE_FOREACH_END(); (void)type; } if (!find_host) { http2_add_header(&nv[HTTP2_CLIENT_HOST_HEADER_INDEX], ZEND_STRL(":authority"), hcc->host, hcc->host_len); } zval *zcookie = sw_zend_read_property(swoole_http2_client_class_entry_ptr, zobject, ZEND_STRL("cookies"), 1 TSRMLS_CC); //http cookies if (zcookie && !ZVAL_IS_NULL(zcookie)) { http2_add_cookie(nv, &index, zcookie TSRMLS_CC); } ssize_t rv; size_t buflen; size_t i; size_t sum = 0; #if 0 for (i = 0; i < index; ++i) { swTraceLog(SW_TRACE_HTTP2, "Header[%d]: "SW_ECHO_CYAN_BLUE"=%s", i, nv[i].name, nv[i].value); } #endif nghttp2_hd_deflater *deflater; ret = nghttp2_hd_deflate_new(&deflater, 4096); if (ret != 0) { swoole_php_error(E_WARNING, "nghttp2_hd_deflate_init failed with error: %s\n", nghttp2_strerror(ret)); return SW_ERR; } for (i = 0; i < index; ++i) { sum += nv[i].namelen + nv[i].valuelen; } buflen = nghttp2_hd_deflate_bound(deflater, nv, index); if (buflen > buffer_len) { swoole_php_error(E_WARNING, "header is too large."); return SW_ERR; } rv = nghttp2_hd_deflate_hd(deflater, (uchar *) buffer, buflen, nv, index); if (rv < 0) { swoole_php_error(E_WARNING, "nghttp2_hd_deflate_hd() failed with error: %s\n", nghttp2_strerror((int ) rv)); return SW_ERR; } for (i = 0; i < index; ++i) { efree(nv[i].name); // free lower header name copy } if (date_str) { efree(date_str); } nghttp2_hd_deflate_del(deflater); return rv; } void http2_add_cookie(nghttp2_nv *nv, int *index, zval *cookies TSRMLS_DC) { char *key; uint32_t keylen; int keytype; zval *value = NULL; char *encoded_value; uint32_t offest = 0; swString_clear(cookie_buffer); SW_HASHTABLE_FOREACH_START2(Z_ARRVAL_P(cookies), key, keylen, keytype, value) if (HASH_KEY_IS_STRING != keytype) { continue; } convert_to_string(value); if (Z_STRLEN_P(value) == 0) { continue; } swString_append_ptr(cookie_buffer, key, keylen); swString_append_ptr(cookie_buffer, "=", 1); int encoded_value_len; encoded_value = sw_php_url_encode(Z_STRVAL_P(value), Z_STRLEN_P(value), &encoded_value_len); if (encoded_value) { swString_append_ptr(cookie_buffer, encoded_value, encoded_value_len); efree(encoded_value); http2_add_header(&nv[(*index)++], ZEND_STRL("cookie"), cookie_buffer->str + offest, keylen + 1 + encoded_value_len); offest += keylen + 1 + encoded_value_len; } SW_HASHTABLE_FOREACH_END(); } int http2_client_parse_header(http2_client_property *hcc, http2_client_stream *stream , int flags, char *in, size_t inlen) { #if PHP_MAJOR_VERSION < 7 TSRMLS_FETCH_FROM_CTX(sw_thread_ctx ? sw_thread_ctx : NULL); #endif nghttp2_hd_inflater *inflater = hcc->inflater; zval *zresponse = stream->response_object; if (!inflater) { int ret = nghttp2_hd_inflate_new(&inflater); if (ret != 0) { swoole_php_error(E_WARNING, "nghttp2_hd_inflate_init() failed, Error: %s[%d].", nghttp2_strerror(ret), ret); return SW_ERR; } hcc->inflater = inflater; } if (flags & SW_HTTP2_FLAG_PRIORITY) { //int stream_deps = ntohl(*(int *) (in)); //uint8_t weight = in[4]; in += 5; inlen -= 5; } zval *zheader; SW_MAKE_STD_ZVAL(zheader); array_init(zheader); ssize_t rv; for (;;) { nghttp2_nv nv; int inflate_flags = 0; size_t proclen; rv = nghttp2_hd_inflate_hd(inflater, &nv, &inflate_flags, (uchar *) in, inlen, 1); if (rv < 0) { swoole_php_error(E_WARNING, "inflate failed, Error: %s[%zd].", nghttp2_strerror(rv), rv); return -1; } proclen = (size_t) rv; in += proclen; inlen -= proclen; //swTraceLog(SW_TRACE_HTTP2, "Header: %s[%d]: %s[%d]", nv.name, nv.namelen, nv.value, nv.valuelen); if (inflate_flags & NGHTTP2_HD_INFLATE_EMIT) { if (nv.name[0] == ':') { if (strncasecmp((char *) nv.name + 1, "status", nv.namelen -1) == 0) { zend_update_property_long(swoole_http2_response_class_entry_ptr, zresponse, ZEND_STRL("statusCode"), atoi((char *) nv.value) TSRMLS_CC); continue; } } #ifdef SW_HAVE_ZLIB else if (strncasecmp((char *) nv.name, "content-encoding", nv.namelen) == 0 && strncasecmp((char *) nv.value, "gzip", nv.valuelen) == 0) { http2_client_init_gzip_stream(stream); if (Z_OK != inflateInit2(&stream->gzip_stream, MAX_WBITS + 16)) { swWarn("inflateInit2() failed."); return SW_ERR; } } #endif sw_add_assoc_stringl_ex(zheader, (char *) nv.name, nv.namelen + 1, (char *) nv.value, nv.valuelen, 1); } if (inflate_flags & NGHTTP2_HD_INFLATE_FINAL) { nghttp2_hd_inflate_end_headers(inflater); break; } if ((inflate_flags & NGHTTP2_HD_INFLATE_EMIT) == 0 && inlen == 0) { break; } } zend_update_property(swoole_http2_response_class_entry_ptr, zresponse, ZEND_STRL("header"), zheader TSRMLS_CC); sw_zval_ptr_dtor(&zheader); rv = nghttp2_hd_inflate_change_table_size(inflater, 4096); if (rv != 0) { return rv; } return SW_OK; } /** * Http2 */ static int http2_client_onFrame(zval *zobject, zval *zdata TSRMLS_DC) { char *buf = Z_STRVAL_P(zdata); int type = buf[3]; int flags = buf[4]; int stream_id = ntohl((*(int *) (buf + 5))) & 0x7fffffff; uint32_t length = swHttp2_get_length(buf); buf += SW_HTTP2_FRAME_HEADER_SIZE; char frame[SW_HTTP2_FRAME_HEADER_SIZE + SW_HTTP2_FRAME_PING_PAYLOAD_SIZE]; http2_client_property *hcc = swoole_get_property(zobject, HTTP2_CLIENT_PROPERTY_INDEX); swClient *cli = swoole_get_object(zobject); uint16_t id; uint32_t value; swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_YELLOW"]\tflags=%d, stream_id=%d, length=%d", swHttp2_get_type(type), flags, stream_id, length); if (type == SW_HTTP2_TYPE_SETTINGS) { if (flags & SW_HTTP2_FLAG_ACK) { return SW_OK; } while(length > 0) { id = ntohs(*(uint16_t *) (buf)); value = ntohl(*(uint32_t *) (buf + sizeof(uint16_t))); switch (id) { case SW_HTTP2_SETTINGS_MAX_CONCURRENT_STREAMS: hcc->max_concurrent_streams = value; swTraceLog(SW_TRACE_HTTP2, "setting: max_concurrent_streams=%d.", value); break; case SW_HTTP2_SETTINGS_INIT_WINDOW_SIZE: hcc->window_size = value; swTraceLog(SW_TRACE_HTTP2, "setting: init_window_size=%d.", value); break; case SW_HTTP2_SETTINGS_MAX_FRAME_SIZE: hcc->max_frame_size = value; swTraceLog(SW_TRACE_HTTP2, "setting: max_frame_size=%d.", value); break; case SW_HTTP2_SETTINGS_MAX_HEADER_LIST_SIZE: hcc->max_header_list_size = value; swTraceLog(SW_TRACE_HTTP2, "setting: max_header_list_size=%d.", value); break; default: swWarn("unknown option[%d].", id); break; } buf += sizeof(id) + sizeof(value); length -= sizeof(id) + sizeof(value); } swHttp2_set_frame_header(frame, SW_HTTP2_TYPE_SETTINGS, 0, SW_HTTP2_FLAG_ACK, stream_id); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", ACK, STREAM#%d]\t[length=%d]", swHttp2_get_type(SW_HTTP2_TYPE_SETTINGS), stream_id, length); cli->send(cli, frame, SW_HTTP2_FRAME_HEADER_SIZE, 0); return SW_OK; } else if (type == SW_HTTP2_TYPE_WINDOW_UPDATE) { hcc->window_size = ntohl(*(int *) buf); swTraceLog(SW_TRACE_HTTP2, "update: window_size=%d.", hcc->window_size); return SW_OK; } else if (type == SW_HTTP2_TYPE_PING) { swHttp2_set_frame_header(frame, SW_HTTP2_TYPE_PING, SW_HTTP2_FRAME_PING_PAYLOAD_SIZE, SW_HTTP2_FLAG_ACK, stream_id); memcpy(frame + SW_HTTP2_FRAME_HEADER_SIZE, buf + SW_HTTP2_FRAME_HEADER_SIZE, SW_HTTP2_FRAME_PING_PAYLOAD_SIZE); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", STREAM#%d]", swHttp2_get_type(SW_HTTP2_FRAME_PING_PAYLOAD_SIZE), stream_id); cli->send(cli, frame, SW_HTTP2_FRAME_HEADER_SIZE + SW_HTTP2_FRAME_PING_PAYLOAD_SIZE, 0); return SW_OK; } else if (type == SW_HTTP2_TYPE_GOAWAY) { int last_stream_id = htonl(*(int *) (buf)); buf += 4; int error_code = htonl(*(int *) (buf)); swWarn("["SW_ECHO_RED"] last_stream_id=%d, error_code=%d.", "GOAWAY", last_stream_id, error_code); zval* retval; sw_zend_call_method_with_0_params(&zobject, swoole_client_class_entry_ptr, NULL, "close", &retval); if (retval) { sw_zval_ptr_dtor(&retval); } return SW_OK; } http2_client_stream *stream = swHashMap_find_int(hcc->streams, stream_id); // stream has closed if (stream == NULL) { return SW_OK; } if (type == SW_HTTP2_TYPE_HEADERS) { http2_client_parse_header(hcc, stream, flags, buf, length); } else if (type == SW_HTTP2_TYPE_DATA) { if (!stream->buffer) { stream->buffer = swString_new(8192); } #ifdef SW_HAVE_ZLIB if (stream->gzip) { if (http_response_uncompress(&stream->gzip_stream, stream->gzip_buffer, buf, length) == SW_ERR) { return -1; } swString_append_ptr(stream->buffer, stream->gzip_buffer->str, stream->gzip_buffer->length); } else #endif { swString_append_ptr(stream->buffer, buf, length); } } else { swWarn("unknown frame, type=%d, stream_id=%d, length=%d.", type, stream_id, length); return SW_OK; } if ((type == SW_HTTP2_TYPE_DATA && stream->type == SW_HTTP2_STREAM_PIPELINE) || (stream->type == SW_HTTP2_STREAM_NORMAL && (flags & SW_HTTP2_FLAG_END_STREAM))) { zval *retval = NULL; zval *zcallback = stream->callback; zval *zresponse = stream->response_object; if (stream->buffer) { zend_update_property_stringl(swoole_http2_response_class_entry_ptr, stream->response_object, ZEND_STRL("body"), stream->buffer->str, stream->buffer->length TSRMLS_CC); } zval **args[1]; args[0] = &zresponse; if (sw_call_user_function_ex(EG(function_table), NULL, zcallback, &retval, 1, args, 0, NULL TSRMLS_CC) == FAILURE) { swoole_php_fatal_error(E_WARNING, "swoole_http2_client handler error."); } if (EG(exception)) { zend_exception_error(EG(exception), E_ERROR TSRMLS_CC); } if (retval) { sw_zval_ptr_dtor(&retval); } if (stream->type == SW_HTTP2_STREAM_NORMAL) { swHashMap_del_int(hcc->streams, stream_id); } else { swString_clear(stream->buffer); } } return SW_OK; } static void http2_client_request_free(void *ptr) { http2_client_request *req = ptr; if (req->callback) { sw_zval_ptr_dtor(&req->callback); } if (req->data) { sw_zval_ptr_dtor(&req->data); } efree(req->uri); efree(req); } static void http2_client_stream_free(void *ptr) { http2_client_stream *stream = ptr; sw_zval_ptr_dtor(&stream->callback); sw_zval_ptr_dtor(&stream->response_object); if (stream->buffer) { swString_free(stream->buffer); } #ifdef SW_HAVE_ZLIB if (stream->gzip) { inflateEnd(&stream->gzip_stream); swString_free(stream->gzip_buffer); } #endif efree(stream); } static void http2_client_set_callback(zval *zobject, const char *callback_name, const char *method_name TSRMLS_DC) { zval *retval = NULL; zval *zcallback; SW_MAKE_STD_ZVAL(zcallback); array_init(zcallback); zval *zname; SW_MAKE_STD_ZVAL(zname); zval *zmethod_name; SW_MAKE_STD_ZVAL(zmethod_name); SW_ZVAL_STRING(zname, callback_name, 1); SW_ZVAL_STRING(zmethod_name, method_name, 1); #if PHP_MAJOR_VERSION < 7 sw_zval_add_ref(&zobject); #endif add_next_index_zval(zcallback, zobject); add_next_index_zval(zcallback, zmethod_name); sw_zend_call_method_with_2_params(&zobject, swoole_http2_client_class_entry_ptr, NULL, "on", &retval, zname, zcallback); if (retval) { sw_zval_ptr_dtor(&retval); } sw_zval_ptr_dtor(&zname); sw_zval_ptr_dtor(&zcallback); } static void http2_client_send_all_requests(zval *zobject TSRMLS_DC) { http2_client_property *hcc = swoole_get_property(zobject, HTTP2_CLIENT_PROPERTY_INDEX); swLinkedList *requests = hcc->requests; swLinkedList_node *node = requests->head; http2_client_request *request; while(node) { request = node->data; http2_client_send_request(zobject, request TSRMLS_CC); node = node->next; } swLinkedList_free(requests); hcc->requests = NULL; requests = hcc->stream_requests; node = requests->head; while(node) { request = node->data; http2_client_send_stream_request(zobject, request TSRMLS_CC); node = node->next; } swLinkedList_free(requests); hcc->stream_requests = NULL; } static void http2_client_send_stream_request(zval *zobject, http2_client_request *req TSRMLS_DC) { swClient *cli = swoole_get_object(zobject); http2_client_property *hcc = swoole_get_property(zobject, HTTP2_CLIENT_PROPERTY_INDEX); char buffer[8192]; /** * create stream */ if (req->stream_id == 0) { /** * send header */ int n = http2_client_build_header(zobject, req, buffer + SW_HTTP2_FRAME_HEADER_SIZE, sizeof(buffer) - SW_HTTP2_FRAME_HEADER_SIZE TSRMLS_CC); if (n <= 0) { swWarn("http2_client_build_header() failed."); return; } swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_HEADERS, n, SW_HTTP2_FLAG_END_HEADERS, hcc->stream_id); http2_client_stream *stream = emalloc(sizeof(http2_client_stream)); memset(stream, 0, sizeof(http2_client_stream)); zval *response_object; SW_MAKE_STD_ZVAL(response_object); object_init_ex(response_object, swoole_http2_response_class_entry_ptr); stream->stream_id = hcc->stream_id; stream->response_object = response_object; stream->callback = req->callback; stream->type = SW_HTTP2_STREAM_PIPELINE; sw_copy_to_stack(stream->callback, stream->_callback); sw_zval_add_ref(&stream->callback); sw_copy_to_stack(stream->response_object, stream->_response_object); zend_update_property_long(swoole_http2_response_class_entry_ptr, response_object, ZEND_STRL("streamId"), stream->stream_id TSRMLS_CC); swHashMap_add_int(hcc->streams, hcc->stream_id, stream); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", STREAM#%d] length=%d", swHttp2_get_type(SW_HTTP2_TYPE_HEADERS), hcc->stream_id, n); cli->send(cli, buffer, n + SW_HTTP2_FRAME_HEADER_SIZE, 0); hcc->stream_id += 2; return; } else { int stream_id = req->stream_id; zval *post_data = req->data; /** * send body */ if (post_data) { if (Z_TYPE_P(post_data) == IS_ARRAY) { zend_size_t len; smart_str formstr_s = { 0 }; char *formstr = sw_http_build_query(post_data, &len, &formstr_s TSRMLS_CC); if (formstr == NULL) { swoole_php_error(E_WARNING, "http_build_query failed."); return; } memset(buffer, 0, SW_HTTP2_FRAME_HEADER_SIZE); swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_DATA, len, 0, stream_id); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", END, STREAM#%d] length=%d", swHttp2_get_type(SW_HTTP2_TYPE_DATA), stream_id, len); cli->send(cli, buffer, SW_HTTP2_FRAME_HEADER_SIZE, 0); cli->send(cli, formstr, len, 0); smart_str_free(&formstr_s); } else { swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_DATA, Z_STRLEN_P(post_data), 0, stream_id); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", END, STREAM#%d] length=%d", swHttp2_get_type(SW_HTTP2_TYPE_DATA), stream_id, Z_STRLEN_P(post_data)); cli->send(cli, buffer, SW_HTTP2_FRAME_HEADER_SIZE, 0); cli->send(cli, Z_STRVAL_P(post_data), Z_STRLEN_P(post_data), 0); } } return; } } static void http2_client_send_request(zval *zobject, http2_client_request *req TSRMLS_DC) { swClient *cli = swoole_get_object(zobject); http2_client_property *hcc = swoole_get_property(zobject, HTTP2_CLIENT_PROPERTY_INDEX); zval *post_data = req->data; if (post_data) { zval *zheader = sw_zend_read_property(swoole_http2_client_class_entry_ptr, zobject, ZEND_STRL("requestHeaders"), 1 TSRMLS_CC); if (Z_TYPE_P(post_data) == IS_ARRAY) { sw_add_assoc_stringl_ex(zheader, ZEND_STRS("content-type"), ZEND_STRL("application/x-www-form-urlencoded"), 1); } } /** * send header */ char buffer[8192]; int n = http2_client_build_header(zobject, req, buffer + SW_HTTP2_FRAME_HEADER_SIZE, sizeof(buffer) - SW_HTTP2_FRAME_HEADER_SIZE TSRMLS_CC); if (n <= 0) { swWarn("http2_client_build_header() failed."); return; } if (post_data == NULL) { swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_HEADERS, n, SW_HTTP2_FLAG_END_STREAM | SW_HTTP2_FLAG_END_HEADERS, hcc->stream_id); } else { swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_HEADERS, n, SW_HTTP2_FLAG_END_HEADERS, hcc->stream_id); } http2_client_stream *stream = emalloc(sizeof(http2_client_stream)); memset(stream, 0, sizeof(http2_client_stream)); zval *response_object; SW_MAKE_STD_ZVAL(response_object); object_init_ex(response_object, swoole_http2_response_class_entry_ptr); stream->stream_id = hcc->stream_id; stream->response_object = response_object; stream->callback = req->callback; stream->type = SW_HTTP2_STREAM_NORMAL; sw_copy_to_stack(stream->callback, stream->_callback); sw_zval_add_ref(&stream->callback); sw_copy_to_stack(stream->response_object, stream->_response_object); zend_update_property_long(swoole_http2_response_class_entry_ptr, response_object, ZEND_STRL("streamId"), stream->stream_id TSRMLS_CC); swHashMap_add_int(hcc->streams, hcc->stream_id, stream); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", STREAM#%d] length=%d", swHttp2_get_type(SW_HTTP2_TYPE_HEADERS), hcc->stream_id, n); cli->send(cli, buffer, n + SW_HTTP2_FRAME_HEADER_SIZE, 0); /** * send body */ if (post_data) { if (Z_TYPE_P(post_data) == IS_ARRAY) { zend_size_t len; smart_str formstr_s = { 0 }; char *formstr = sw_http_build_query(post_data, &len, &formstr_s TSRMLS_CC); if (formstr == NULL) { swoole_php_error(E_WARNING, "http_build_query failed."); return; } memset(buffer, 0, SW_HTTP2_FRAME_HEADER_SIZE); swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_DATA, len, SW_HTTP2_FLAG_END_STREAM, hcc->stream_id); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", END, STREAM#%d] length=%d", swHttp2_get_type(SW_HTTP2_TYPE_DATA), hcc->stream_id, len); cli->send(cli, buffer, SW_HTTP2_FRAME_HEADER_SIZE, 0); cli->send(cli, formstr, len, 0); smart_str_free(&formstr_s); } else { swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_DATA, Z_STRLEN_P(req->data), SW_HTTP2_FLAG_END_STREAM, hcc->stream_id); swTraceLog(SW_TRACE_HTTP2, "["SW_ECHO_GREEN", END, STREAM#%d] length=%d", swHttp2_get_type(SW_HTTP2_TYPE_DATA), hcc->stream_id, Z_STRLEN_P(req->data)); cli->send(cli, buffer, SW_HTTP2_FRAME_HEADER_SIZE, 0); cli->send(cli, Z_STRVAL_P(post_data), Z_STRLEN_P(post_data), 0); } } hcc->stream_id += 2; return; } static void http2_client_connect(zval *zobject TSRMLS_DC) { http2_client_property *hcc = swoole_get_property(zobject, HTTP2_CLIENT_PROPERTY_INDEX); zval *retval = NULL; zval *zhost; SW_MAKE_STD_ZVAL(zhost); SW_ZVAL_STRINGL(zhost, hcc->host, hcc->host_len, 1); zval *zport; SW_MAKE_STD_ZVAL(zport); ZVAL_LONG(zport, hcc->port); http2_client_set_callback(zobject, "Connect", "onConnect" TSRMLS_CC); http2_client_set_callback(zobject, "Receive", "onReceive" TSRMLS_CC); if (!php_swoole_client_isset_callback(zobject, SW_CLIENT_CB_onClose TSRMLS_CC)) { http2_client_set_callback(zobject, "Close", "onClose" TSRMLS_CC); } if (!php_swoole_client_isset_callback(zobject, SW_CLIENT_CB_onError TSRMLS_CC)) { http2_client_set_callback(zobject, "Error", "onError" TSRMLS_CC); } sw_zend_call_method_with_2_params(&zobject, swoole_http2_client_class_entry_ptr, NULL, "connect", &retval, zhost, zport); if (retval) { sw_zval_ptr_dtor(&retval); } sw_zval_ptr_dtor(&zhost); sw_zval_ptr_dtor(&zport); swClient *cli = swoole_get_object(zobject); cli->http2 = 1; } static PHP_METHOD(swoole_http2_client, get) { zval *uri; zval *callback; http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); swClient *cli = swoole_get_object(getThis()); if (!cli && hcc->connecting == 1) { swoole_php_error(E_WARNING, "The connection is closed."); RETURN_FALSE; } if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz", &uri, &callback) == FAILURE) { return; } char *func_name = NULL; if (!sw_zend_is_callable(callback, 0, &func_name TSRMLS_CC)) { swoole_php_fatal_error(E_WARNING, "Function '%s' is not callable", func_name); efree(func_name); RETURN_FALSE; } efree(func_name); if (Z_TYPE_P(uri) != IS_STRING) { swoole_php_fatal_error(E_WARNING, "uri is not string."); RETURN_FALSE; } if (cli && cli->socket && cli->socket->active == 1) { http2_client_request _req; _req.uri = estrndup(Z_STRVAL_P(uri), Z_STRLEN_P(uri)); _req.uri_len = Z_STRLEN_P(uri); _req.type = HTTP_GET; _req.callback = callback; _req.data = NULL; http2_client_send_request(getThis(), &_req TSRMLS_CC); } else { swLinkedList *requests = hcc->requests; http2_client_request *req = emalloc(sizeof(http2_client_request)); req->uri = estrndup(Z_STRVAL_P(uri), Z_STRLEN_P(uri)); req->uri_len = Z_STRLEN_P(uri); req->type = HTTP_GET; req->callback = callback; req->data = NULL; sw_copy_to_stack(req->callback, req->_callback); sw_zval_add_ref(&req->callback); swLinkedList_append(requests, req); if (!hcc->connecting) { http2_client_connect(getThis() TSRMLS_CC); hcc->connecting = 1; } } RETURN_TRUE; } static PHP_METHOD(swoole_http2_client, post) { zval *uri; zval *callback; zval *data; http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); swClient *cli = swoole_get_object(getThis()); if (!cli && hcc->connecting == 1) { swoole_php_error(E_WARNING, "The connection is closed."); RETURN_FALSE; } if (zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC, "zzz", &uri, &data, &callback) == FAILURE) { return; } char *func_name = NULL; if (!sw_zend_is_callable(callback, 0, &func_name TSRMLS_CC)) { swoole_php_fatal_error(E_WARNING, "Function '%s' is not callable", func_name); efree(func_name); RETURN_FALSE; } efree(func_name); if (Z_TYPE_P(uri) != IS_STRING) { swoole_php_fatal_error(E_WARNING, "uri is not string."); RETURN_FALSE; } if (cli && cli->socket && cli->socket->active == 1) { http2_client_request _req; _req.uri = estrndup(Z_STRVAL_P(uri), Z_STRLEN_P(uri)); _req.uri_len = Z_STRLEN_P(uri); _req.type = HTTP_POST; _req.callback = callback; _req.data = data; http2_client_send_request(getThis(), &_req TSRMLS_CC); } else { swLinkedList *requests = hcc->requests; http2_client_request *req = emalloc(sizeof(http2_client_request)); req->uri = estrndup(Z_STRVAL_P(uri), Z_STRLEN_P(uri)); req->uri_len = Z_STRLEN_P(uri); req->type = HTTP_POST; req->data = data; req->callback = callback; sw_copy_to_stack(req->data, req->_data); sw_zval_add_ref(&req->data); sw_copy_to_stack(req->callback, req->_callback); sw_zval_add_ref(&req->callback); swLinkedList_append(requests, req); if (!hcc->connecting) { http2_client_connect(getThis() TSRMLS_CC); hcc->connecting = 1; } } RETURN_TRUE; } static PHP_METHOD(swoole_http2_client, openStream) { zval *uri; zval *callback; http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); swClient *cli = swoole_get_object(getThis()); if (!cli && hcc->connecting == 1) { swoole_php_error(E_WARNING, "The connection is closed."); RETURN_FALSE; } if (zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC, "zz", &uri, &callback) == FAILURE) { return; } char *func_name = NULL; if (!sw_zend_is_callable(callback, 0, &func_name TSRMLS_CC)) { swoole_php_fatal_error(E_WARNING, "Function '%s' is not callable", func_name); efree(func_name); RETURN_FALSE; } efree(func_name); if (Z_TYPE_P(uri) != IS_STRING) { swoole_php_fatal_error(E_WARNING, "uri is not string."); RETURN_FALSE; } if (cli && cli->socket && cli->socket->active == 1) { http2_client_request _req; _req.uri = estrndup(Z_STRVAL_P(uri), Z_STRLEN_P(uri)); _req.uri_len = Z_STRLEN_P(uri); _req.type = HTTP_POST; _req.callback = callback; _req.stream_id = 0; http2_client_send_stream_request(getThis(), &_req TSRMLS_CC); } else { swLinkedList *requests = hcc->stream_requests; http2_client_request *req = emalloc(sizeof(http2_client_request)); req->uri = estrndup(Z_STRVAL_P(uri), Z_STRLEN_P(uri)); req->uri_len = Z_STRLEN_P(uri); req->type = HTTP_POST; req->callback = callback; req->data = NULL; req->stream_id = 0; sw_copy_to_stack(req->callback, req->_callback); sw_zval_add_ref(&req->callback); swLinkedList_append(requests, req); if (!hcc->connecting) { http2_client_connect(getThis() TSRMLS_CC); hcc->connecting = 1; } } RETURN_LONG(hcc->stream_id); } static PHP_METHOD(swoole_http2_client, push) { long stream_id; zval *data; if (zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC, "lz", &stream_id, &data) == FAILURE) { return; } http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); swClient *cli = swoole_get_object(getThis()); if (!cli && hcc->connecting == 1) { swoole_php_error(E_WARNING, "The connection is closed."); RETURN_FALSE; } if (cli && cli->socket && cli->socket->active == 1) { http2_client_request _req; _req.uri = NULL; _req.uri_len = 0; _req.data = data; _req.stream_id = stream_id; _req.callback = NULL; http2_client_send_stream_request(getThis(), &_req TSRMLS_CC); } else { swLinkedList *requests = hcc->stream_requests; http2_client_request *req = emalloc(sizeof(http2_client_request)); req->uri = NULL; req->uri_len = 0; req->data = data; req->stream_id = stream_id; req->callback = NULL; sw_copy_to_stack(req->data, req->_data); sw_zval_add_ref(&req->data); swLinkedList_append(requests, req); if (!hcc->connecting) { http2_client_connect(getThis() TSRMLS_CC); hcc->connecting = 1; } } RETURN_TRUE; } static PHP_METHOD(swoole_http2_client, closeStream) { http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); swClient *cli = swoole_get_object(getThis()); if (!cli && hcc->connecting == 1) { swoole_php_error(E_WARNING, "The connection is closed."); RETURN_FALSE; } char buffer[8192]; long stream_id; if (zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC, "l", &stream_id) == FAILURE) { return; } swHttp2_set_frame_header(buffer, SW_HTTP2_TYPE_SETTINGS, 0, SW_HTTP2_FLAG_END_STREAM,hcc->stream_id); cli->send(cli, buffer, SW_HTTP2_FRAME_HEADER_SIZE, 0); swHashMap_del_int(hcc->streams, stream_id); RETURN_TRUE; } static PHP_METHOD(swoole_http2_client, onConnect) { swClient *cli = swoole_get_object(getThis()); cli->send(cli, ZEND_STRL("PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n"), 0); cli->open_length_check = 1; cli->protocol.get_package_length = swHttp2_get_frame_length; cli->protocol.package_length_size = SW_HTTP2_FRAME_HEADER_SIZE; http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); hcc->ready = 1; hcc->stream_id = 1; hcc->send_setting = 1; if (hcc->send_setting) { http2_client_send_setting(cli); } http2_client_send_all_requests(getThis() TSRMLS_CC); } static PHP_METHOD(swoole_http2_client, onError) { } static PHP_METHOD(swoole_http2_client, onClose) { } static PHP_METHOD(swoole_http2_client, onReceive) { zval *zobject; zval *zdata; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "zz", &zobject, &zdata) == FAILURE) { return; } http2_client_onFrame(zobject, zdata TSRMLS_CC); } static PHP_METHOD(swoole_http2_client, __destruct) { http2_client_property *hcc = swoole_get_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX); if (hcc) { if (hcc->requests) { swLinkedList_free(hcc->requests); } if (hcc->stream_requests) { swLinkedList_free(hcc->stream_requests); } if (hcc->inflater) { nghttp2_hd_inflate_del(hcc->inflater); hcc->inflater = NULL; } if (hcc->host) { efree(hcc->host); hcc->host = NULL; } swHashMap_free(hcc->streams); efree(hcc); swoole_set_property(getThis(), HTTP2_CLIENT_PROPERTY_INDEX, NULL); } zval *zobject = getThis(); zval *retval = NULL; sw_zend_call_method_with_0_params(&zobject, swoole_client_class_entry_ptr, NULL, "__destruct", &retval); if (retval) { sw_zval_ptr_dtor(&retval); } } #endif