commit
48bbd000eb
139 changed files with 2839 additions and 2022 deletions
@ -0,0 +1,121 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#include "src/core/surface/call.h" |
||||
|
||||
#include "src/core/support/string.h" |
||||
#include <grpc/support/alloc.h> |
||||
|
||||
int grpc_trace_batch = 0; |
||||
|
||||
static void add_metadata(gpr_strvec *b, const grpc_metadata *md, size_t count) { |
||||
size_t i; |
||||
for(i = 0; i < count; i++) { |
||||
gpr_strvec_add(b, gpr_strdup("\nkey=")); |
||||
gpr_strvec_add(b, gpr_strdup(md[i].key)); |
||||
|
||||
gpr_strvec_add(b, gpr_strdup(" value=")); |
||||
gpr_strvec_add(b, gpr_hexdump(md[i].value, md[i].value_length, |
||||
GPR_HEXDUMP_PLAINTEXT)); |
||||
} |
||||
} |
||||
|
||||
char *grpc_op_string(const grpc_op *op) { |
||||
char *tmp; |
||||
char *out; |
||||
|
||||
gpr_strvec b; |
||||
gpr_strvec_init(&b); |
||||
|
||||
switch (op->op) { |
||||
case GRPC_OP_SEND_INITIAL_METADATA: |
||||
gpr_strvec_add(&b, gpr_strdup("SEND_INITIAL_METADATA")); |
||||
add_metadata(&b, op->data.send_initial_metadata.metadata, |
||||
op->data.send_initial_metadata.count); |
||||
break; |
||||
case GRPC_OP_SEND_MESSAGE: |
||||
gpr_asprintf(&tmp, "SEND_MESSAGE ptr=%p", op->data.send_message); |
||||
gpr_strvec_add(&b, tmp); |
||||
break; |
||||
case GRPC_OP_SEND_CLOSE_FROM_CLIENT: |
||||
gpr_strvec_add(&b, gpr_strdup("SEND_CLOSE_FROM_CLIENT")); |
||||
break; |
||||
case GRPC_OP_SEND_STATUS_FROM_SERVER: |
||||
gpr_asprintf(&tmp, "SEND_STATUS_FROM_SERVER status=%d details=%s", |
||||
op->data.send_status_from_server.status, |
||||
op->data.send_status_from_server.status_details); |
||||
gpr_strvec_add(&b, tmp); |
||||
add_metadata(&b, op->data.send_status_from_server.trailing_metadata, |
||||
op->data.send_status_from_server.trailing_metadata_count); |
||||
break; |
||||
case GRPC_OP_RECV_INITIAL_METADATA: |
||||
gpr_asprintf(&tmp, "RECV_INITIAL_METADATA ptr=%p", |
||||
op->data.recv_initial_metadata); |
||||
gpr_strvec_add(&b, tmp); |
||||
break; |
||||
case GRPC_OP_RECV_MESSAGE: |
||||
gpr_asprintf(&tmp, "RECV_MESSAGE ptr=%p", op->data.recv_message); |
||||
gpr_strvec_add(&b, tmp); |
||||
break; |
||||
case GRPC_OP_RECV_STATUS_ON_CLIENT: |
||||
gpr_asprintf(&tmp, |
||||
"RECV_STATUS_ON_CLIENT metadata=%p status=%p details=%p", |
||||
op->data.recv_status_on_client.trailing_metadata, |
||||
op->data.recv_status_on_client.status, |
||||
op->data.recv_status_on_client.status_details); |
||||
gpr_strvec_add(&b, tmp); |
||||
break; |
||||
case GRPC_OP_RECV_CLOSE_ON_SERVER: |
||||
gpr_asprintf(&tmp, "RECV_CLOSE_ON_SERVER cancelled=%p", |
||||
op->data.recv_close_on_server.cancelled); |
||||
gpr_strvec_add(&b, tmp); |
||||
} |
||||
out = gpr_strvec_flatten(&b, NULL); |
||||
gpr_strvec_destroy(&b); |
||||
|
||||
return out; |
||||
} |
||||
|
||||
void grpc_call_log_batch(char *file, int line, gpr_log_severity severity, |
||||
grpc_call *call, const grpc_op *ops, size_t nops, |
||||
void *tag) { |
||||
char *tmp; |
||||
size_t i; |
||||
gpr_log(file, line, severity, |
||||
"grpc_call_start_batch(%p, %p, %d, 0x%x)", call, ops, nops, tag); |
||||
for(i = 0; i < nops; i++) { |
||||
tmp = grpc_op_string(&ops[i]); |
||||
gpr_log(file, line, severity, "ops[%d]: %s", i, tmp); |
||||
gpr_free(tmp); |
||||
} |
||||
} |
@ -0,0 +1,136 @@ |
||||
/* |
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
/** |
||||
* This script runs a QPS test. It sends requests for a specified length of time |
||||
* with a specified number pending at any one time. It then outputs the measured |
||||
* QPS. Usage: |
||||
* node qps_test.js [--concurrent=count] [--time=seconds] |
||||
* concurrent defaults to 100 and time defaults to 10 |
||||
*/ |
||||
|
||||
'use strict'; |
||||
|
||||
var async = require('async'); |
||||
var parseArgs = require('minimist'); |
||||
|
||||
var grpc = require('..'); |
||||
var testProto = grpc.load(__dirname + '/../interop/test.proto').grpc.testing; |
||||
var interop_server = require('../interop/interop_server.js'); |
||||
|
||||
/** |
||||
* Runs the QPS test. Sends requests constantly for the given number of seconds, |
||||
* and keeps concurrent_calls requests pending at all times. When the test ends, |
||||
* the callback is called with the number of calls that completed within the |
||||
* time limit. |
||||
* @param {number} concurrent_calls The number of calls to have pending |
||||
* simultaneously |
||||
* @param {number} seconds The number of seconds to run the test for |
||||
* @param {function(Error, number)} callback Callback for test completion |
||||
*/ |
||||
function runTest(concurrent_calls, seconds, callback) { |
||||
var testServer = interop_server.getServer(0, false); |
||||
testServer.server.listen(); |
||||
var client = new testProto.TestService('localhost:' + testServer.port); |
||||
|
||||
var warmup_num = 100; |
||||
|
||||
/** |
||||
* Warms up the client to avoid counting startup time in the test result |
||||
* @param {function(Error)} callback Called when warmup is complete |
||||
*/ |
||||
function warmUp(callback) { |
||||
var pending = warmup_num; |
||||
function startCall() { |
||||
client.emptyCall({}, function(err, resp) { |
||||
if (err) { |
||||
callback(err); |
||||
return; |
||||
} |
||||
pending--; |
||||
if (pending === 0) { |
||||
callback(null); |
||||
} |
||||
}); |
||||
} |
||||
for (var i = 0; i < warmup_num; i++) { |
||||
startCall(); |
||||
} |
||||
} |
||||
/** |
||||
* Run the QPS test. Starts concurrent_calls requests, then starts a new |
||||
* request whenever one completes until time runs out. |
||||
* @param {function(Error, number)} callback Called when the test is complete. |
||||
* The second argument is the number of calls that finished within the |
||||
* time limit |
||||
*/ |
||||
function run(callback) { |
||||
var running = 0; |
||||
var count = 0; |
||||
var start = process.hrtime(); |
||||
function responseCallback(err, resp) { |
||||
if (process.hrtime(start)[0] < seconds) { |
||||
count += 1; |
||||
client.emptyCall({}, responseCallback); |
||||
} else { |
||||
running -= 1; |
||||
if (running <= 0) { |
||||
callback(null, count); |
||||
} |
||||
} |
||||
} |
||||
for (var i = 0; i < concurrent_calls; i++) { |
||||
running += 1; |
||||
client.emptyCall({}, responseCallback); |
||||
} |
||||
} |
||||
async.waterfall([warmUp, run], function(err, count) { |
||||
testServer.server.shutdown(); |
||||
callback(err, count); |
||||
}); |
||||
} |
||||
|
||||
if (require.main === module) { |
||||
var argv = parseArgs(process.argv.slice(2), { |
||||
default: {'concurrent': 100, |
||||
'time': 10} |
||||
}); |
||||
runTest(argv.concurrent, argv.time, function(err, count) { |
||||
if (err) { |
||||
throw err; |
||||
} |
||||
console.log('Concurrent calls:', argv.concurrent); |
||||
console.log('Time:', argv.time, 'seconds'); |
||||
console.log('QPS:', (count/argv.time)); |
||||
}); |
||||
} |
@ -1,170 +0,0 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#include "completion_queue.h" |
||||
|
||||
#ifdef HAVE_CONFIG_H |
||||
#include "config.h" |
||||
#endif |
||||
|
||||
#include "php.h" |
||||
#include "php_ini.h" |
||||
#include "ext/standard/info.h" |
||||
#include "ext/spl/spl_exceptions.h" |
||||
#include "php_grpc.h" |
||||
|
||||
#include "zend_exceptions.h" |
||||
|
||||
#include <stdbool.h> |
||||
|
||||
#include "grpc/grpc.h" |
||||
|
||||
#include "event.h" |
||||
#include "timeval.h" |
||||
|
||||
zend_class_entry *grpc_ce_completion_queue; |
||||
|
||||
/* Frees and destroys a wrapped instance of grpc_completion_queue */ |
||||
void free_wrapped_grpc_completion_queue(void *object TSRMLS_DC) { |
||||
wrapped_grpc_completion_queue *queue = NULL; |
||||
grpc_event *event; |
||||
queue = (wrapped_grpc_completion_queue *)object; |
||||
if (queue->wrapped != NULL) { |
||||
grpc_completion_queue_shutdown(queue->wrapped); |
||||
event = grpc_completion_queue_next(queue->wrapped, gpr_inf_future); |
||||
while (event != NULL) { |
||||
if (event->type == GRPC_QUEUE_SHUTDOWN) { |
||||
break; |
||||
} |
||||
event = grpc_completion_queue_next(queue->wrapped, gpr_inf_future); |
||||
} |
||||
grpc_completion_queue_destroy(queue->wrapped); |
||||
} |
||||
efree(queue); |
||||
} |
||||
|
||||
/* Initializes an instance of wrapped_grpc_channel to be associated with an
|
||||
* object of a class specified by class_type */ |
||||
zend_object_value create_wrapped_grpc_completion_queue( |
||||
zend_class_entry *class_type TSRMLS_DC) { |
||||
zend_object_value retval; |
||||
wrapped_grpc_completion_queue *intern; |
||||
|
||||
intern = (wrapped_grpc_completion_queue *)emalloc( |
||||
sizeof(wrapped_grpc_completion_queue)); |
||||
memset(intern, 0, sizeof(wrapped_grpc_completion_queue)); |
||||
|
||||
zend_object_std_init(&intern->std, class_type TSRMLS_CC); |
||||
object_properties_init(&intern->std, class_type); |
||||
retval.handle = zend_objects_store_put( |
||||
intern, (zend_objects_store_dtor_t)zend_objects_destroy_object, |
||||
free_wrapped_grpc_completion_queue, NULL TSRMLS_CC); |
||||
retval.handlers = zend_get_std_object_handlers(); |
||||
return retval; |
||||
} |
||||
|
||||
/**
|
||||
* Construct an instance of CompletionQueue |
||||
*/ |
||||
PHP_METHOD(CompletionQueue, __construct) { |
||||
wrapped_grpc_completion_queue *queue = |
||||
(wrapped_grpc_completion_queue *)zend_object_store_get_object(getThis() |
||||
TSRMLS_CC); |
||||
queue->wrapped = grpc_completion_queue_create(); |
||||
} |
||||
|
||||
/**
|
||||
* Blocks until an event is available, the completion queue is being shutdown, |
||||
* or timeout is reached. Returns NULL on timeout, otherwise the event that |
||||
* occurred. Callers should call event.finish once they have processed the |
||||
* event. |
||||
* @param Timeval $timeout The timeout for the event |
||||
* @return Event The event that occurred |
||||
*/ |
||||
PHP_METHOD(CompletionQueue, next) { |
||||
zval *timeout; |
||||
/* "O" == 1 Object */ |
||||
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "O", &timeout, |
||||
grpc_ce_timeval) == FAILURE) { |
||||
zend_throw_exception(spl_ce_InvalidArgumentException, |
||||
"next needs a Timeval", 1 TSRMLS_CC); |
||||
return; |
||||
} |
||||
wrapped_grpc_completion_queue *completion_queue = |
||||
(wrapped_grpc_completion_queue *)zend_object_store_get_object(getThis() |
||||
TSRMLS_CC); |
||||
wrapped_grpc_timeval *wrapped_timeout = |
||||
(wrapped_grpc_timeval *)zend_object_store_get_object(timeout TSRMLS_CC); |
||||
grpc_event *event = grpc_completion_queue_next(completion_queue->wrapped, |
||||
wrapped_timeout->wrapped); |
||||
if (event == NULL) { |
||||
RETURN_NULL(); |
||||
} |
||||
zval *wrapped_event = grpc_php_convert_event(event); |
||||
RETURN_DESTROY_ZVAL(wrapped_event); |
||||
} |
||||
|
||||
PHP_METHOD(CompletionQueue, pluck) { |
||||
long tag; |
||||
zval *timeout; |
||||
/* "lO" == 1 long, 1 Object */ |
||||
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "lO", &tag, &timeout, |
||||
grpc_ce_timeval) == FAILURE) { |
||||
zend_throw_exception(spl_ce_InvalidArgumentException, |
||||
"pluck needs a long and a Timeval", 1 TSRMLS_CC); |
||||
} |
||||
wrapped_grpc_completion_queue *completion_queue = |
||||
(wrapped_grpc_completion_queue *)zend_object_store_get_object(getThis() |
||||
TSRMLS_CC); |
||||
wrapped_grpc_timeval *wrapped_timeout = |
||||
(wrapped_grpc_timeval *)zend_object_store_get_object(timeout TSRMLS_CC); |
||||
grpc_event *event = grpc_completion_queue_pluck( |
||||
completion_queue->wrapped, (void *)tag, wrapped_timeout->wrapped); |
||||
if (event == NULL) { |
||||
RETURN_NULL(); |
||||
} |
||||
zval *wrapped_event = grpc_php_convert_event(event); |
||||
RETURN_DESTROY_ZVAL(wrapped_event); |
||||
} |
||||
|
||||
static zend_function_entry completion_queue_methods[] = { |
||||
PHP_ME(CompletionQueue, __construct, NULL, ZEND_ACC_PUBLIC | ZEND_ACC_CTOR) |
||||
PHP_ME(CompletionQueue, next, NULL, ZEND_ACC_PUBLIC) |
||||
PHP_ME(CompletionQueue, pluck, NULL, ZEND_ACC_PUBLIC) PHP_FE_END}; |
||||
|
||||
void grpc_init_completion_queue(TSRMLS_D) { |
||||
zend_class_entry ce; |
||||
INIT_CLASS_ENTRY(ce, "Grpc\\CompletionQueue", completion_queue_methods); |
||||
ce.create_object = create_wrapped_grpc_completion_queue; |
||||
grpc_ce_completion_queue = zend_register_internal_class(&ce TSRMLS_CC); |
||||
} |
@ -1,62 +0,0 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#ifndef NET_GRPC_PHP_GRPC_COMPLETION_QUEUE_H_ |
||||
#define NET_GRPC_PHP_GRPC_COMPLETION_QUEUE_H_ |
||||
|
||||
#ifdef HAVE_CONFIG_H |
||||
#include "config.h" |
||||
#endif |
||||
|
||||
#include "php.h" |
||||
#include "php_ini.h" |
||||
#include "ext/standard/info.h" |
||||
#include "php_grpc.h" |
||||
|
||||
#include "grpc/grpc.h" |
||||
|
||||
/* Class entry for the PHP CompletionQueue class */ |
||||
extern zend_class_entry *grpc_ce_completion_queue; |
||||
|
||||
/* Wrapper class for grpc_completion_queue that can be associated with a
|
||||
PHP object */ |
||||
typedef struct wrapped_grpc_completion_queue { |
||||
zend_object std; |
||||
|
||||
grpc_completion_queue *wrapped; |
||||
} wrapped_grpc_completion_queue; |
||||
|
||||
/* Initialize the CompletionQueue class */ |
||||
void grpc_init_completion_queue(TSRMLS_D); |
||||
|
||||
#endif /* NET_GRPC_PHP_GRPC_COMPLETION_QUEUE_H_ */ |
@ -1,150 +0,0 @@ |
||||
/*
|
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
#include "event.h" |
||||
|
||||
#ifdef HAVE_CONFIG_H |
||||
#include "config.h" |
||||
#endif |
||||
|
||||
#include "php.h" |
||||
#include "php_ini.h" |
||||
#include "ext/standard/info.h" |
||||
#include "php_grpc.h" |
||||
|
||||
#include <stdbool.h> |
||||
|
||||
#include "grpc/grpc.h" |
||||
|
||||
#include "byte_buffer.h" |
||||
#include "call.h" |
||||
#include "timeval.h" |
||||
|
||||
/* Create a new PHP object containing the event data in the event struct.
|
||||
event must not be used after this function is called */ |
||||
zval *grpc_php_convert_event(grpc_event *event) { |
||||
zval *data_object; |
||||
char *detail_string; |
||||
size_t detail_len; |
||||
char *method_string; |
||||
size_t method_len; |
||||
char *host_string; |
||||
size_t host_len; |
||||
char *read_string; |
||||
size_t read_len; |
||||
|
||||
zval *event_object; |
||||
|
||||
if (event == NULL) { |
||||
return NULL; |
||||
} |
||||
|
||||
MAKE_STD_ZVAL(event_object); |
||||
object_init(event_object); |
||||
|
||||
add_property_zval( |
||||
event_object, "call", |
||||
grpc_php_wrap_call(event->call, event->type == GRPC_SERVER_RPC_NEW)); |
||||
add_property_long(event_object, "type", event->type); |
||||
add_property_long(event_object, "tag", (long)event->tag); |
||||
|
||||
switch (event->type) { |
||||
case GRPC_QUEUE_SHUTDOWN: |
||||
add_property_null(event_object, "data"); |
||||
break; |
||||
case GRPC_READ: |
||||
if (event->data.read == NULL) { |
||||
add_property_null(event_object, "data"); |
||||
} else { |
||||
byte_buffer_to_string(event->data.read, &read_string, &read_len); |
||||
add_property_stringl(event_object, "data", read_string, read_len, true); |
||||
} |
||||
break; |
||||
case GRPC_WRITE_ACCEPTED: |
||||
add_property_long(event_object, "data", (long)event->data.write_accepted); |
||||
break; |
||||
case GRPC_FINISH_ACCEPTED: |
||||
add_property_long(event_object, "data", |
||||
(long)event->data.finish_accepted); |
||||
break; |
||||
case GRPC_CLIENT_METADATA_READ: |
||||
data_object = grpc_call_create_metadata_array( |
||||
event->data.client_metadata_read.count, |
||||
event->data.client_metadata_read.elements); |
||||
add_property_zval(event_object, "data", data_object); |
||||
break; |
||||
case GRPC_FINISHED: |
||||
MAKE_STD_ZVAL(data_object); |
||||
object_init(data_object); |
||||
add_property_long(data_object, "code", event->data.finished.status); |
||||
if (event->data.finished.details == NULL) { |
||||
add_property_null(data_object, "details"); |
||||
} else { |
||||
detail_len = strlen(event->data.finished.details); |
||||
detail_string = ecalloc(detail_len + 1, sizeof(char)); |
||||
memcpy(detail_string, event->data.finished.details, detail_len); |
||||
add_property_string(data_object, "details", detail_string, true); |
||||
} |
||||
add_property_zval(data_object, "metadata", |
||||
grpc_call_create_metadata_array( |
||||
event->data.finished.metadata_count, |
||||
event->data.finished.metadata_elements)); |
||||
add_property_zval(event_object, "data", data_object); |
||||
break; |
||||
case GRPC_SERVER_RPC_NEW: |
||||
MAKE_STD_ZVAL(data_object); |
||||
object_init(data_object); |
||||
method_len = strlen(event->data.server_rpc_new.method); |
||||
method_string = ecalloc(method_len + 1, sizeof(char)); |
||||
memcpy(method_string, event->data.server_rpc_new.method, method_len); |
||||
add_property_string(data_object, "method", method_string, false); |
||||
host_len = strlen(event->data.server_rpc_new.host); |
||||
host_string = ecalloc(host_len + 1, sizeof(char)); |
||||
memcpy(host_string, event->data.server_rpc_new.host, host_len); |
||||
add_property_string(data_object, "host", host_string, false); |
||||
add_property_zval( |
||||
data_object, "absolute_timeout", |
||||
grpc_php_wrap_timeval(event->data.server_rpc_new.deadline)); |
||||
add_property_zval(data_object, "metadata", |
||||
grpc_call_create_metadata_array( |
||||
event->data.server_rpc_new.metadata_count, |
||||
event->data.server_rpc_new.metadata_elements)); |
||||
add_property_zval(event_object, "data", data_object); |
||||
break; |
||||
default: |
||||
add_property_null(event_object, "data"); |
||||
break; |
||||
} |
||||
grpc_event_finish(event); |
||||
return event_object; |
||||
} |
@ -0,0 +1,79 @@ |
||||
<?php |
||||
/* |
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
namespace Grpc; |
||||
|
||||
require_once realpath(dirname(__FILE__) . '/../autoload.php'); |
||||
|
||||
abstract class AbstractCall { |
||||
|
||||
protected $call; |
||||
protected $deserialize; |
||||
protected $metadata; |
||||
|
||||
/** |
||||
* Create a new Call wrapper object. |
||||
* @param Channel $channel The channel to communicate on |
||||
* @param string $method The method to call on the remote server |
||||
*/ |
||||
public function __construct(Channel $channel, $method, $deserialize) { |
||||
$this->call = new Call($channel, $method, Timeval::inf_future()); |
||||
$this->deserialize = $deserialize; |
||||
} |
||||
|
||||
/** |
||||
* @return The metadata sent by the server. |
||||
*/ |
||||
public function getMetadata() { |
||||
return $this->metadata; |
||||
} |
||||
|
||||
/** |
||||
* Cancels the call |
||||
*/ |
||||
public function cancel() { |
||||
$this->call->cancel(); |
||||
} |
||||
|
||||
/** |
||||
* Deserialize a response value to an object. |
||||
* @param string $value The binary value to deserialize |
||||
* @return The deserialized value |
||||
*/ |
||||
protected function deserializeResponse($value) { |
||||
if ($value === null) { |
||||
return null; |
||||
} |
||||
return call_user_func($this->deserialize, $value); |
||||
} |
||||
} |
@ -1,98 +0,0 @@ |
||||
<?php |
||||
|
||||
/* |
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
|
||||
namespace Grpc; |
||||
|
||||
require_once realpath(dirname(__FILE__) . '/../autoload.php'); |
||||
|
||||
/** |
||||
* Represents an active call that allows sending and recieving messages. |
||||
* Subclasses restrict how data can be sent and recieved. |
||||
*/ |
||||
abstract class AbstractSurfaceActiveCall { |
||||
private $active_call; |
||||
private $deserialize; |
||||
|
||||
/** |
||||
* Create a new surface active call. |
||||
* @param Channel $channel The channel to communicate on |
||||
* @param string $method The method to call on the remote server |
||||
* @param callable $deserialize The function to deserialize a value |
||||
* @param array $metadata Metadata to send with the call, if applicable |
||||
* @param long $flags Write flags to use with this call |
||||
*/ |
||||
public function __construct(Channel $channel, |
||||
$method, |
||||
callable $deserialize, |
||||
$metadata = array(), |
||||
$flags = 0) { |
||||
$this->active_call = new ActiveCall($channel, $method, $metadata, $flags); |
||||
$this->deserialize = $deserialize; |
||||
} |
||||
|
||||
/** |
||||
* @return The metadata sent by the server |
||||
*/ |
||||
public function getMetadata() { |
||||
return $this->metadata(); |
||||
} |
||||
|
||||
/** |
||||
* Cancels the call |
||||
*/ |
||||
public function cancel() { |
||||
$this->active_call->cancel(); |
||||
} |
||||
|
||||
protected function _read() { |
||||
$response = $this->active_call->read(); |
||||
if ($response === null) { |
||||
return null; |
||||
} |
||||
return call_user_func($this->deserialize, $response); |
||||
} |
||||
|
||||
protected function _write($value) { |
||||
return $this->active_call->write($value->serialize()); |
||||
} |
||||
|
||||
protected function _writesDone() { |
||||
$this->active_call->writesDone(); |
||||
} |
||||
|
||||
protected function _getStatus() { |
||||
return $this->active_call->getStatus(); |
||||
} |
||||
} |
@ -1,123 +0,0 @@ |
||||
<?php |
||||
/* |
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
namespace Grpc; |
||||
require_once realpath(dirname(__FILE__) . '/../autoload.php'); |
||||
|
||||
/** |
||||
* Represents an active call that allows sending and recieving binary data |
||||
*/ |
||||
class ActiveCall { |
||||
private $completion_queue; |
||||
private $call; |
||||
private $flags; |
||||
private $metadata; |
||||
|
||||
/** |
||||
* Create a new active call. |
||||
* @param Channel $channel The channel to communicate on |
||||
* @param string $method The method to call on the remote server |
||||
* @param array $metadata Metadata to send with the call, if applicable |
||||
* @param long $flags Write flags to use with this call |
||||
*/ |
||||
public function __construct(Channel $channel, |
||||
$method, |
||||
$metadata = array(), |
||||
$flags = 0) { |
||||
$this->completion_queue = new CompletionQueue(); |
||||
$this->call = new Call($channel, $method, Timeval::inf_future()); |
||||
$this->call->add_metadata($metadata, 0); |
||||
$this->flags = $flags; |
||||
|
||||
// Invoke the call. |
||||
$this->call->invoke($this->completion_queue, |
||||
CLIENT_METADATA_READ, |
||||
FINISHED, 0); |
||||
$metadata_event = $this->completion_queue->pluck(CLIENT_METADATA_READ, |
||||
Timeval::inf_future()); |
||||
$this->metadata = $metadata_event->data; |
||||
} |
||||
|
||||
/** |
||||
* @return The metadata sent by the server. |
||||
*/ |
||||
public function getMetadata() { |
||||
return $this->metadata; |
||||
} |
||||
|
||||
/** |
||||
* Cancels the call |
||||
*/ |
||||
public function cancel() { |
||||
$this->call->cancel(); |
||||
} |
||||
|
||||
/** |
||||
* Read a single message from the server. |
||||
* @return The next message from the server, or null if there is none. |
||||
*/ |
||||
public function read() { |
||||
$this->call->start_read(READ); |
||||
$read_event = $this->completion_queue->pluck(READ, Timeval::inf_future()); |
||||
return $read_event->data; |
||||
} |
||||
|
||||
/** |
||||
* Write a single message to the server. This cannot be called after |
||||
* writesDone is called. |
||||
* @param ByteBuffer $data The data to write |
||||
*/ |
||||
public function write($data) { |
||||
$this->call->start_write($data, WRITE_ACCEPTED, $this->flags); |
||||
$this->completion_queue->pluck(WRITE_ACCEPTED, Timeval::inf_future()); |
||||
} |
||||
|
||||
/** |
||||
* Indicate that no more writes will be sent. |
||||
*/ |
||||
public function writesDone() { |
||||
$this->call->writes_done(FINISH_ACCEPTED); |
||||
$this->completion_queue->pluck(FINISH_ACCEPTED, Timeval::inf_future()); |
||||
} |
||||
|
||||
/** |
||||
* Wait for the server to send the status, and return it. |
||||
* @return object The status object, with integer $code, string $details, |
||||
* and array $metadata members |
||||
*/ |
||||
public function getStatus() { |
||||
$status_event = $this->completion_queue->pluck(FINISHED, |
||||
Timeval::inf_future()); |
||||
return $status_event->data; |
||||
} |
||||
} |
@ -1,46 +0,0 @@ |
||||
<?php |
||||
/* |
||||
* |
||||
* Copyright 2015, Google Inc. |
||||
* All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions are |
||||
* met: |
||||
* |
||||
* * Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* * Redistributions in binary form must reproduce the above |
||||
* copyright notice, this list of conditions and the following disclaimer |
||||
* in the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* * Neither the name of Google Inc. nor the names of its |
||||
* contributors may be used to endorse or promote products derived from |
||||
* this software without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
||||
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
||||
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
||||
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
||||
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
||||
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
||||
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
||||
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
*/ |
||||
class CompletionQueueTest extends PHPUnit_Framework_TestCase{ |
||||
public function testNextReturnsNullWithNoCall() { |
||||
$cq = new Grpc\CompletionQueue(); |
||||
$event = $cq->next(Grpc\Timeval::zero()); |
||||
$this->assertNull($event); |
||||
} |
||||
|
||||
public function testPluckReturnsNullWithNoCall() { |
||||
$cq = new Grpc\CompletionQueue(); |
||||
$event = $cq->pluck(0, Grpc\Timeval::zero()); |
||||
$this->assertNull($event); |
||||
} |
||||
} |
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in new issue