grpc 0.15.0-universal-darwin → 1.0.0.pre1-universal-darwin

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of grpc might be problematic. Click here for more details.

Files changed (37) hide show
  1. checksums.yaml +4 -4
  2. data/etc/roots.pem +784 -509
  3. data/src/ruby/ext/grpc/rb_byte_buffer.c +4 -1
  4. data/src/ruby/ext/grpc/rb_call.c +87 -54
  5. data/src/ruby/ext/grpc/rb_call.h +1 -1
  6. data/src/ruby/ext/grpc/rb_call_credentials.c +1 -30
  7. data/src/ruby/ext/grpc/rb_channel.c +25 -50
  8. data/src/ruby/ext/grpc/rb_channel_credentials.c +1 -31
  9. data/src/ruby/ext/grpc/rb_completion_queue.c +15 -134
  10. data/src/ruby/ext/grpc/rb_completion_queue.h +3 -7
  11. data/src/ruby/ext/grpc/rb_grpc.c +2 -4
  12. data/src/ruby/ext/grpc/rb_grpc_imports.generated.c +2 -0
  13. data/src/ruby/ext/grpc/rb_grpc_imports.generated.h +4 -1
  14. data/src/ruby/ext/grpc/rb_server.c +81 -133
  15. data/src/ruby/ext/grpc/rb_server_credentials.c +4 -33
  16. data/src/ruby/lib/grpc/2.0/grpc_c.bundle +0 -0
  17. data/src/ruby/lib/grpc/2.1/grpc_c.bundle +0 -0
  18. data/src/ruby/lib/grpc/2.2/grpc_c.bundle +0 -0
  19. data/src/ruby/lib/grpc/2.3/grpc_c.bundle +0 -0
  20. data/src/ruby/lib/grpc/generic/active_call.rb +40 -55
  21. data/src/ruby/lib/grpc/generic/bidi_call.rb +21 -23
  22. data/src/ruby/lib/grpc/generic/client_stub.rb +20 -15
  23. data/src/ruby/lib/grpc/generic/rpc_server.rb +15 -37
  24. data/src/ruby/lib/grpc/generic/service.rb +1 -1
  25. data/src/ruby/lib/grpc/version.rb +1 -1
  26. data/src/ruby/pb/test/client.rb +25 -7
  27. data/src/ruby/pb/test/server.rb +7 -5
  28. data/src/ruby/spec/call_spec.rb +1 -2
  29. data/src/ruby/spec/channel_spec.rb +2 -3
  30. data/src/ruby/spec/client_server_spec.rb +74 -59
  31. data/src/ruby/spec/generic/active_call_spec.rb +66 -86
  32. data/src/ruby/spec/generic/client_stub_spec.rb +27 -48
  33. data/src/ruby/spec/generic/rpc_server_spec.rb +4 -34
  34. data/src/ruby/spec/pb/health/checker_spec.rb +0 -2
  35. data/src/ruby/spec/server_spec.rb +20 -24
  36. metadata +4 -6
  37. data/src/ruby/spec/completion_queue_spec.rb +0 -42
@@ -126,36 +126,6 @@ VALUE grpc_rb_wrap_channel_credentials(grpc_channel_credentials *c, VALUE mark)
126
126
  return rb_wrapper;
127
127
  }
128
128
 
129
- /* Clones ChannelCredentials instances.
130
- Gives ChannelCredentials a consistent implementation of Ruby's object copy/dup
131
- protocol. */
132
- static VALUE grpc_rb_channel_credentials_init_copy(VALUE copy, VALUE orig) {
133
- grpc_rb_channel_credentials *orig_cred = NULL;
134
- grpc_rb_channel_credentials *copy_cred = NULL;
135
-
136
- if (copy == orig) {
137
- return copy;
138
- }
139
-
140
- /* Raise an error if orig is not a credentials object or a subclass. */
141
- if (TYPE(orig) != T_DATA ||
142
- RDATA(orig)->dfree != (RUBY_DATA_FUNC)grpc_rb_channel_credentials_free) {
143
- rb_raise(rb_eTypeError, "not a %s",
144
- rb_obj_classname(grpc_rb_cChannelCredentials));
145
- }
146
-
147
- TypedData_Get_Struct(orig, grpc_rb_channel_credentials,
148
- &grpc_rb_channel_credentials_data_type, orig_cred);
149
- TypedData_Get_Struct(copy, grpc_rb_channel_credentials,
150
- &grpc_rb_channel_credentials_data_type, copy_cred);
151
-
152
- /* use ruby's MEMCPY to make a byte-for-byte copy of the credentials
153
- * wrapper object. */
154
- MEMCPY(copy_cred, orig_cred, grpc_rb_channel_credentials, 1);
155
- return copy;
156
- }
157
-
158
-
159
129
  /* The attribute used on the mark object to hold the pem_root_certs. */
160
130
  static ID id_pem_root_certs;
161
131
 
@@ -271,7 +241,7 @@ void Init_grpc_channel_credentials() {
271
241
  rb_define_method(grpc_rb_cChannelCredentials, "initialize",
272
242
  grpc_rb_channel_credentials_init, -1);
273
243
  rb_define_method(grpc_rb_cChannelCredentials, "initialize_copy",
274
- grpc_rb_channel_credentials_init_copy, 1);
244
+ grpc_rb_cannot_init_copy, 1);
275
245
  rb_define_method(grpc_rb_cChannelCredentials, "compose",
276
246
  grpc_rb_channel_credentials_compose, -1);
277
247
  rb_define_module_function(grpc_rb_cChannelCredentials,
@@ -40,12 +40,9 @@
40
40
 
41
41
  #include <grpc/grpc.h>
42
42
  #include <grpc/support/time.h>
43
+ #include <grpc/support/log.h>
43
44
  #include "rb_grpc.h"
44
45
 
45
- /* grpc_rb_cCompletionQueue is the ruby class that proxies
46
- * grpc_completion_queue. */
47
- static VALUE grpc_rb_cCompletionQueue = Qnil;
48
-
49
46
  /* Used to allow grpc_completion_queue_next call to release the GIL */
50
47
  typedef struct next_call_stack {
51
48
  grpc_completion_queue *cq;
@@ -55,23 +52,6 @@ typedef struct next_call_stack {
55
52
  volatile int interrupted;
56
53
  } next_call_stack;
57
54
 
58
- /* Calls grpc_completion_queue_next without holding the ruby GIL */
59
- static void *grpc_rb_completion_queue_next_no_gil(void *param) {
60
- next_call_stack *const next_call = (next_call_stack*)param;
61
- gpr_timespec increment = gpr_time_from_millis(20, GPR_TIMESPAN);
62
- gpr_timespec deadline;
63
- do {
64
- deadline = gpr_time_add(gpr_now(GPR_CLOCK_REALTIME), increment);
65
- next_call->event = grpc_completion_queue_next(next_call->cq,
66
- deadline, NULL);
67
- if (next_call->event.type != GRPC_QUEUE_TIMEOUT ||
68
- gpr_time_cmp(deadline, next_call->timeout) > 0) {
69
- break;
70
- }
71
- } while (!next_call->interrupted);
72
- return NULL;
73
- }
74
-
75
55
  /* Calls grpc_completion_queue_pluck without holding the ruby GIL */
76
56
  static void *grpc_rb_completion_queue_pluck_no_gil(void *param) {
77
57
  next_call_stack *const next_call = (next_call_stack*)param;
@@ -90,107 +70,32 @@ static void *grpc_rb_completion_queue_pluck_no_gil(void *param) {
90
70
  return NULL;
91
71
  }
92
72
 
93
- /* Shuts down and drains the completion queue if necessary.
94
- *
95
- * This is done when the ruby completion queue object is about to be GCed.
96
- */
97
- static void grpc_rb_completion_queue_shutdown_drain(grpc_completion_queue *cq) {
98
- next_call_stack next_call;
99
- grpc_completion_type type;
100
- int drained = 0;
101
- MEMZERO(&next_call, next_call_stack, 1);
102
-
103
- grpc_completion_queue_shutdown(cq);
104
- next_call.cq = cq;
105
- next_call.event.type = GRPC_QUEUE_TIMEOUT;
106
- /* TODO: the timeout should be a module level constant that defaults
107
- * to gpr_inf_future(GPR_CLOCK_REALTIME).
108
- *
109
- * - at the moment this does not work, it stalls. Using a small timeout like
110
- * this one works, and leads to fast test run times; a longer timeout was
111
- * causing unnecessary delays in the test runs.
112
- *
113
- * - investigate further, this is probably another example of C-level cleanup
114
- * not working consistently in all cases.
115
- */
116
- next_call.timeout = gpr_time_add(gpr_now(GPR_CLOCK_REALTIME),
117
- gpr_time_from_micros(5e3, GPR_TIMESPAN));
118
- do {
119
- rb_thread_call_without_gvl(grpc_rb_completion_queue_next_no_gil,
120
- (void *)&next_call, NULL, NULL);
121
- type = next_call.event.type;
122
- if (type == GRPC_QUEUE_TIMEOUT) break;
123
- if (type != GRPC_QUEUE_SHUTDOWN) {
124
- ++drained;
125
- rb_warning("completion queue shutdown: %d undrained events", drained);
126
- }
127
- } while (type != GRPC_QUEUE_SHUTDOWN);
128
- }
129
-
130
73
  /* Helper function to free a completion queue. */
131
- static void grpc_rb_completion_queue_destroy(void *p) {
132
- grpc_completion_queue *cq = NULL;
133
- if (p == NULL) {
134
- return;
135
- }
136
- cq = (grpc_completion_queue *)p;
137
- grpc_rb_completion_queue_shutdown_drain(cq);
74
+ void grpc_rb_completion_queue_destroy(grpc_completion_queue *cq) {
75
+ /* Every function that adds an event to a queue also synchronously plucks
76
+ that event from the queue, and holds a reference to the Ruby object that
77
+ holds the queue, so we only get to this point if all of those functions
78
+ have completed, and the queue is empty */
79
+ grpc_completion_queue_shutdown(cq);
138
80
  grpc_completion_queue_destroy(cq);
139
81
  }
140
82
 
141
- static rb_data_type_t grpc_rb_completion_queue_data_type = {
142
- "grpc_completion_queue",
143
- {GRPC_RB_GC_NOT_MARKED, grpc_rb_completion_queue_destroy,
144
- GRPC_RB_MEMSIZE_UNAVAILABLE, {NULL, NULL}},
145
- NULL, NULL,
146
- #ifdef RUBY_TYPED_FREE_IMMEDIATELY
147
- /* cannot immediately free because grpc_rb_completion_queue_shutdown_drain
148
- * calls rb_thread_call_without_gvl. */
149
- 0,
150
- #endif
151
- };
152
-
153
- /* Releases the c-level resources associated with a completion queue */
154
- static VALUE grpc_rb_completion_queue_close(VALUE self) {
155
- grpc_completion_queue* cq = grpc_rb_get_wrapped_completion_queue(self);
156
- grpc_rb_completion_queue_destroy(cq);
157
- RTYPEDDATA_DATA(self) = NULL;
158
- return Qnil;
159
- }
160
-
161
- /* Allocates a completion queue. */
162
- static VALUE grpc_rb_completion_queue_alloc(VALUE cls) {
163
- grpc_completion_queue *cq = grpc_completion_queue_create(NULL);
164
- if (cq == NULL) {
165
- rb_raise(rb_eArgError, "could not create a completion queue: not sure why");
166
- }
167
- return TypedData_Wrap_Struct(cls, &grpc_rb_completion_queue_data_type, cq);
168
- }
169
-
170
83
  static void unblock_func(void *param) {
171
84
  next_call_stack *const next_call = (next_call_stack*)param;
172
85
  next_call->interrupted = 1;
173
86
  }
174
87
 
175
- /* Blocks until the next event for given tag is available, and returns the
176
- * event. */
177
- grpc_event grpc_rb_completion_queue_pluck_event(VALUE self, VALUE tag,
178
- VALUE timeout) {
88
+ /* Does the same thing as grpc_completion_queue_pluck, while properly releasing
89
+ the GVL and handling interrupts */
90
+ grpc_event rb_completion_queue_pluck(grpc_completion_queue *queue, void *tag,
91
+ gpr_timespec deadline, void *reserved) {
179
92
  next_call_stack next_call;
180
93
  MEMZERO(&next_call, next_call_stack, 1);
181
- TypedData_Get_Struct(self, grpc_completion_queue,
182
- &grpc_rb_completion_queue_data_type, next_call.cq);
183
- if (TYPE(timeout) == T_NIL) {
184
- next_call.timeout = gpr_inf_future(GPR_CLOCK_REALTIME);
185
- } else {
186
- next_call.timeout = grpc_rb_time_timeval(timeout, /* absolute time*/ 0);
187
- }
188
- if (TYPE(tag) == T_NIL) {
189
- next_call.tag = NULL;
190
- } else {
191
- next_call.tag = ROBJECT(tag);
192
- }
94
+ next_call.cq = queue;
95
+ next_call.timeout = deadline;
96
+ next_call.tag = tag;
193
97
  next_call.event.type = GRPC_QUEUE_TIMEOUT;
98
+ (void)reserved;
194
99
  /* Loop until we finish a pluck without an interruption. The internal
195
100
  pluck function runs either until it is interrupted or it gets an
196
101
  event, or time runs out.
@@ -210,27 +115,3 @@ grpc_event grpc_rb_completion_queue_pluck_event(VALUE self, VALUE tag,
210
115
  next_call.event.type == GRPC_QUEUE_TIMEOUT);
211
116
  return next_call.event;
212
117
  }
213
-
214
- void Init_grpc_completion_queue() {
215
- grpc_rb_cCompletionQueue =
216
- rb_define_class_under(grpc_rb_mGrpcCore, "CompletionQueue", rb_cObject);
217
-
218
- /* constructor: uses an alloc func without an initializer. Using a simple
219
- alloc func works here as the grpc header does not specify any args for
220
- this func, so no separate initialization step is necessary. */
221
- rb_define_alloc_func(grpc_rb_cCompletionQueue,
222
- grpc_rb_completion_queue_alloc);
223
-
224
- /* close: Provides a way to close the underlying file descriptor without
225
- waiting for ruby garbage collection. */
226
- rb_define_method(grpc_rb_cCompletionQueue, "close",
227
- grpc_rb_completion_queue_close, 0);
228
- }
229
-
230
- /* Gets the wrapped completion queue from the ruby wrapper */
231
- grpc_completion_queue *grpc_rb_get_wrapped_completion_queue(VALUE v) {
232
- grpc_completion_queue *cq = NULL;
233
- TypedData_Get_Struct(v, grpc_completion_queue,
234
- &grpc_rb_completion_queue_data_type, cq);
235
- return cq;
236
- }
@@ -38,18 +38,14 @@
38
38
 
39
39
  #include <grpc/grpc.h>
40
40
 
41
- /* Gets the wrapped completion queue from the ruby wrapper */
42
- grpc_completion_queue *grpc_rb_get_wrapped_completion_queue(VALUE v);
41
+ void grpc_rb_completion_queue_destroy(grpc_completion_queue *cq);
43
42
 
44
43
  /**
45
44
  * Makes the implementation of CompletionQueue#pluck available in other files
46
45
  *
47
46
  * This avoids having code that holds the GIL repeated at multiple sites.
48
47
  */
49
- grpc_event grpc_rb_completion_queue_pluck_event(VALUE self, VALUE tag,
50
- VALUE timeout);
51
-
52
- /* Initializes the CompletionQueue class. */
53
- void Init_grpc_completion_queue();
48
+ grpc_event rb_completion_queue_pluck(grpc_completion_queue *queue, void *tag,
49
+ gpr_timespec deadline, void *reserved);
54
50
 
55
51
  #endif /* GRPC_RB_COMPLETION_QUEUE_H_ */
@@ -46,7 +46,6 @@
46
46
  #include "rb_call_credentials.h"
47
47
  #include "rb_channel.h"
48
48
  #include "rb_channel_credentials.h"
49
- #include "rb_completion_queue.h"
50
49
  #include "rb_loader.h"
51
50
  #include "rb_server.h"
52
51
  #include "rb_server_credentials.h"
@@ -85,7 +84,7 @@ VALUE grpc_rb_cannot_init(VALUE self) {
85
84
  VALUE grpc_rb_cannot_init_copy(VALUE copy, VALUE self) {
86
85
  (void)self;
87
86
  rb_raise(rb_eTypeError,
88
- "initialization of %s only allowed from the gRPC native layer",
87
+ "Copy initialization of %s is not supported",
89
88
  rb_obj_classname(copy));
90
89
  return Qnil;
91
90
  }
@@ -318,7 +317,7 @@ void Init_grpc_c() {
318
317
  grpc_rb_mGrpcCore = rb_define_module_under(grpc_rb_mGRPC, "Core");
319
318
  grpc_rb_sNewServerRpc =
320
319
  rb_struct_define("NewServerRpc", "method", "host",
321
- "deadline", "metadata", "call", "cq", NULL);
320
+ "deadline", "metadata", "call", NULL);
322
321
  grpc_rb_sStatus =
323
322
  rb_struct_define("Status", "code", "details", "metadata", NULL);
324
323
  sym_code = ID2SYM(rb_intern("code"));
@@ -326,7 +325,6 @@ void Init_grpc_c() {
326
325
  sym_metadata = ID2SYM(rb_intern("metadata"));
327
326
 
328
327
  Init_grpc_channel();
329
- Init_grpc_completion_queue();
330
328
  Init_grpc_call();
331
329
  Init_grpc_call_credentials();
332
330
  Init_grpc_channel_credentials();
@@ -128,6 +128,7 @@ grpc_is_binary_header_type grpc_is_binary_header_import;
128
128
  grpc_call_error_to_string_type grpc_call_error_to_string_import;
129
129
  grpc_insecure_channel_create_from_fd_type grpc_insecure_channel_create_from_fd_import;
130
130
  grpc_server_add_insecure_channel_from_fd_type grpc_server_add_insecure_channel_from_fd_import;
131
+ grpc_use_signal_type grpc_use_signal_import;
131
132
  grpc_auth_property_iterator_next_type grpc_auth_property_iterator_next_import;
132
133
  grpc_auth_context_property_iterator_type grpc_auth_context_property_iterator_import;
133
134
  grpc_auth_context_peer_identity_type grpc_auth_context_peer_identity_import;
@@ -399,6 +400,7 @@ void grpc_rb_load_imports(HMODULE library) {
399
400
  grpc_call_error_to_string_import = (grpc_call_error_to_string_type) GetProcAddress(library, "grpc_call_error_to_string");
400
401
  grpc_insecure_channel_create_from_fd_import = (grpc_insecure_channel_create_from_fd_type) GetProcAddress(library, "grpc_insecure_channel_create_from_fd");
401
402
  grpc_server_add_insecure_channel_from_fd_import = (grpc_server_add_insecure_channel_from_fd_type) GetProcAddress(library, "grpc_server_add_insecure_channel_from_fd");
403
+ grpc_use_signal_import = (grpc_use_signal_type) GetProcAddress(library, "grpc_use_signal");
402
404
  grpc_auth_property_iterator_next_import = (grpc_auth_property_iterator_next_type) GetProcAddress(library, "grpc_auth_property_iterator_next");
403
405
  grpc_auth_context_property_iterator_import = (grpc_auth_context_property_iterator_type) GetProcAddress(library, "grpc_auth_context_property_iterator");
404
406
  grpc_auth_context_peer_identity_import = (grpc_auth_context_peer_identity_type) GetProcAddress(library, "grpc_auth_context_peer_identity");
@@ -335,6 +335,9 @@ extern grpc_insecure_channel_create_from_fd_type grpc_insecure_channel_create_fr
335
335
  typedef void(*grpc_server_add_insecure_channel_from_fd_type)(grpc_server *server, grpc_completion_queue *cq, int fd);
336
336
  extern grpc_server_add_insecure_channel_from_fd_type grpc_server_add_insecure_channel_from_fd_import;
337
337
  #define grpc_server_add_insecure_channel_from_fd grpc_server_add_insecure_channel_from_fd_import
338
+ typedef void(*grpc_use_signal_type)(int signum);
339
+ extern grpc_use_signal_type grpc_use_signal_import;
340
+ #define grpc_use_signal grpc_use_signal_import
338
341
  typedef const grpc_auth_property *(*grpc_auth_property_iterator_next_type)(grpc_auth_property_iterator *it);
339
342
  extern grpc_auth_property_iterator_next_type grpc_auth_property_iterator_next_import;
340
343
  #define grpc_auth_property_iterator_next grpc_auth_property_iterator_next_import
@@ -467,7 +470,7 @@ extern grpc_byte_buffer_length_type grpc_byte_buffer_length_import;
467
470
  typedef void(*grpc_byte_buffer_destroy_type)(grpc_byte_buffer *byte_buffer);
468
471
  extern grpc_byte_buffer_destroy_type grpc_byte_buffer_destroy_import;
469
472
  #define grpc_byte_buffer_destroy grpc_byte_buffer_destroy_import
470
- typedef void(*grpc_byte_buffer_reader_init_type)(grpc_byte_buffer_reader *reader, grpc_byte_buffer *buffer);
473
+ typedef int(*grpc_byte_buffer_reader_init_type)(grpc_byte_buffer_reader *reader, grpc_byte_buffer *buffer);
471
474
  extern grpc_byte_buffer_reader_init_type grpc_byte_buffer_reader_init_import;
472
475
  #define grpc_byte_buffer_reader_init grpc_byte_buffer_reader_init_import
473
476
  typedef void(*grpc_byte_buffer_reader_destroy_type)(grpc_byte_buffer_reader *reader);
@@ -38,6 +38,7 @@
38
38
 
39
39
  #include <grpc/grpc.h>
40
40
  #include <grpc/grpc_security.h>
41
+ #include <grpc/support/log.h>
41
42
  #include "rb_call.h"
42
43
  #include "rb_channel_args.h"
43
44
  #include "rb_completion_queue.h"
@@ -53,53 +54,51 @@ static ID id_at;
53
54
  /* id_insecure_server is used to indicate that a server is insecure */
54
55
  static VALUE id_insecure_server;
55
56
 
56
- /* grpc_rb_server wraps a grpc_server. It provides a peer ruby object,
57
- 'mark' to minimize copying when a server is created from ruby. */
57
+ /* grpc_rb_server wraps a grpc_server. */
58
58
  typedef struct grpc_rb_server {
59
- /* Holder of ruby objects involved in constructing the server */
60
- VALUE mark;
61
59
  /* The actual server */
62
60
  grpc_server *wrapped;
63
61
  grpc_completion_queue *queue;
64
62
  } grpc_rb_server;
65
63
 
64
+ static void destroy_server(grpc_rb_server *server, gpr_timespec deadline) {
65
+ grpc_event ev;
66
+ if (server->wrapped != NULL) {
67
+ grpc_server_shutdown_and_notify(server->wrapped, server->queue, NULL);
68
+ ev = rb_completion_queue_pluck(server->queue, NULL, deadline, NULL);
69
+ if (ev.type == GRPC_QUEUE_TIMEOUT) {
70
+ grpc_server_cancel_all_calls(server->wrapped);
71
+ rb_completion_queue_pluck(server->queue, NULL,
72
+ gpr_inf_future(GPR_CLOCK_REALTIME), NULL);
73
+ }
74
+ grpc_server_destroy(server->wrapped);
75
+ grpc_rb_completion_queue_destroy(server->queue);
76
+ server->wrapped = NULL;
77
+ server->queue = NULL;
78
+ }
79
+ }
80
+
66
81
  /* Destroys server instances. */
67
82
  static void grpc_rb_server_free(void *p) {
68
83
  grpc_rb_server *svr = NULL;
84
+ gpr_timespec deadline;
69
85
  if (p == NULL) {
70
86
  return;
71
87
  };
72
88
  svr = (grpc_rb_server *)p;
73
89
 
74
- /* Deletes the wrapped object if the mark object is Qnil, which indicates
75
- that no other object is the actual owner. */
76
- /* grpc_server_shutdown does not exist. Change this to something that does
77
- or delete it */
78
- if (svr->wrapped != NULL && svr->mark == Qnil) {
79
- // grpc_server_shutdown(svr->wrapped);
80
- // Aborting to indicate a bug
81
- abort();
82
- grpc_server_destroy(svr->wrapped);
83
- }
90
+ deadline = gpr_time_add(
91
+ gpr_now(GPR_CLOCK_REALTIME),
92
+ gpr_time_from_seconds(2, GPR_TIMESPAN));
84
93
 
85
- xfree(p);
86
- }
94
+ destroy_server(svr, deadline);
87
95
 
88
- /* Protects the mark object from GC */
89
- static void grpc_rb_server_mark(void *p) {
90
- grpc_rb_server *server = NULL;
91
- if (p == NULL) {
92
- return;
93
- }
94
- server = (grpc_rb_server *)p;
95
- if (server->mark != Qnil) {
96
- rb_gc_mark(server->mark);
97
- }
96
+ xfree(p);
98
97
  }
99
98
 
100
99
  static const rb_data_type_t grpc_rb_server_data_type = {
101
100
  "grpc_server",
102
- {grpc_rb_server_mark, grpc_rb_server_free, GRPC_RB_MEMSIZE_UNAVAILABLE,
101
+ {GRPC_RB_GC_NOT_MARKED, grpc_rb_server_free, GRPC_RB_MEMSIZE_UNAVAILABLE,
103
102
  {NULL, NULL}},
104
103
  NULL,
105
104
  NULL,
@@ -116,23 +115,20 @@ static const rb_data_type_t grpc_rb_server_data_type = {
116
115
  static VALUE grpc_rb_server_alloc(VALUE cls) {
117
116
  grpc_rb_server *wrapper = ALLOC(grpc_rb_server);
118
117
  wrapper->wrapped = NULL;
119
- wrapper->mark = Qnil;
120
118
  return TypedData_Wrap_Struct(cls, &grpc_rb_server_data_type, wrapper);
121
119
  }
122
120
 
123
121
  /*
124
122
  call-seq:
125
- cq = CompletionQueue.new
126
- server = Server.new(cq, {'arg1': 'value1'})
123
+ server = Server.new({'arg1': 'value1'})
127
124
 
128
125
  Initializes server instances. */
129
- static VALUE grpc_rb_server_init(VALUE self, VALUE cqueue, VALUE channel_args) {
130
- grpc_completion_queue *cq = NULL;
126
+ static VALUE grpc_rb_server_init(VALUE self, VALUE channel_args) {
127
+ grpc_completion_queue *cq = grpc_completion_queue_create(NULL);
131
128
  grpc_rb_server *wrapper = NULL;
132
129
  grpc_server *srv = NULL;
133
130
  grpc_channel_args args;
134
131
  MEMZERO(&args, grpc_channel_args, 1);
135
- cq = grpc_rb_get_wrapped_completion_queue(cqueue);
136
132
  TypedData_Get_Struct(self, grpc_rb_server, &grpc_rb_server_data_type,
137
133
  wrapper);
138
134
  grpc_rb_hash_convert_to_channel_args(channel_args, &args);
@@ -148,41 +144,9 @@ static VALUE grpc_rb_server_init(VALUE self, VALUE cqueue, VALUE channel_args) {
148
144
  wrapper->wrapped = srv;
149
145
  wrapper->queue = cq;
150
146
 
151
- /* Add the cq as the server's mark object. This ensures the ruby cq can't be
152
- GCed before the server */
153
- wrapper->mark = cqueue;
154
147
  return self;
155
148
  }
156
149
 
157
- /* Clones Server instances.
158
-
159
- Gives Server a consistent implementation of Ruby's object copy/dup
160
- protocol. */
161
- static VALUE grpc_rb_server_init_copy(VALUE copy, VALUE orig) {
162
- grpc_rb_server *orig_srv = NULL;
163
- grpc_rb_server *copy_srv = NULL;
164
-
165
- if (copy == orig) {
166
- return copy;
167
- }
168
-
169
- /* Raise an error if orig is not a server object or a subclass. */
170
- if (TYPE(orig) != T_DATA ||
171
- RDATA(orig)->dfree != (RUBY_DATA_FUNC)grpc_rb_server_free) {
172
- rb_raise(rb_eTypeError, "not a %s", rb_obj_classname(grpc_rb_cServer));
173
- }
174
-
175
- TypedData_Get_Struct(orig, grpc_rb_server, &grpc_rb_server_data_type,
176
- orig_srv);
177
- TypedData_Get_Struct(copy, grpc_rb_server, &grpc_rb_server_data_type,
178
- copy_srv);
179
-
180
- /* use ruby's MEMCPY to make a byte-for-byte copy of the server wrapper
181
- object. */
182
- MEMCPY(copy_srv, orig_srv, grpc_rb_server, 1);
183
- return copy;
184
- }
185
-
186
150
  /* request_call_stack holds various values used by the
187
151
  * grpc_rb_server_request_call function */
188
152
  typedef struct request_call_stack {
@@ -208,65 +172,57 @@ static void grpc_request_call_stack_cleanup(request_call_stack* st) {
208
172
  }
209
173
 
210
174
  /* call-seq:
211
- cq = CompletionQueue.new
212
- tag = Object.new
213
- timeout = 10
214
- server.request_call(cqueue, tag, timeout)
175
+ server.request_call
215
176
 
216
177
  Requests notification of a new call on a server. */
217
- static VALUE grpc_rb_server_request_call(VALUE self, VALUE cqueue,
218
- VALUE tag_new, VALUE timeout) {
178
+ static VALUE grpc_rb_server_request_call(VALUE self) {
219
179
  grpc_rb_server *s = NULL;
220
180
  grpc_call *call = NULL;
221
181
  grpc_event ev;
222
182
  grpc_call_error err;
223
183
  request_call_stack st;
224
184
  VALUE result;
185
+ void *tag = (void*)&st;
186
+ grpc_completion_queue *call_queue = grpc_completion_queue_create(NULL);
225
187
  gpr_timespec deadline;
226
188
  TypedData_Get_Struct(self, grpc_rb_server, &grpc_rb_server_data_type, s);
227
189
  if (s->wrapped == NULL) {
228
190
  rb_raise(rb_eRuntimeError, "destroyed!");
229
191
  return Qnil;
230
- } else {
231
- grpc_request_call_stack_init(&st);
232
- /* call grpc_server_request_call, then wait for it to complete using
233
- * pluck_event */
234
- err = grpc_server_request_call(
235
- s->wrapped, &call, &st.details, &st.md_ary,
236
- grpc_rb_get_wrapped_completion_queue(cqueue),
237
- grpc_rb_get_wrapped_completion_queue(s->mark),
238
- ROBJECT(tag_new));
239
- if (err != GRPC_CALL_OK) {
240
- grpc_request_call_stack_cleanup(&st);
241
- rb_raise(grpc_rb_eCallError,
242
- "grpc_server_request_call failed: %s (code=%d)",
243
- grpc_call_error_detail_of(err), err);
244
- return Qnil;
245
- }
246
-
247
- ev = grpc_rb_completion_queue_pluck_event(s->mark, tag_new, timeout);
248
- if (ev.type == GRPC_QUEUE_TIMEOUT) {
249
- grpc_request_call_stack_cleanup(&st);
250
- return Qnil;
251
- }
252
- if (!ev.success) {
253
- grpc_request_call_stack_cleanup(&st);
254
- rb_raise(grpc_rb_eCallError, "request_call completion failed");
255
- return Qnil;
256
- }
192
+ }
193
+ grpc_request_call_stack_init(&st);
194
+ /* call grpc_server_request_call, then wait for it to complete using
195
+ * pluck_event */
196
+ err = grpc_server_request_call(
197
+ s->wrapped, &call, &st.details, &st.md_ary,
198
+ call_queue, s->queue, tag);
199
+ if (err != GRPC_CALL_OK) {
200
+ grpc_request_call_stack_cleanup(&st);
201
+ rb_raise(grpc_rb_eCallError,
202
+ "grpc_server_request_call failed: %s (code=%d)",
203
+ grpc_call_error_detail_of(err), err);
204
+ return Qnil;
205
+ }
257
206
 
258
- /* build the NewServerRpc struct result */
259
- deadline = gpr_convert_clock_type(st.details.deadline, GPR_CLOCK_REALTIME);
260
- result = rb_struct_new(
261
- grpc_rb_sNewServerRpc, rb_str_new2(st.details.method),
262
- rb_str_new2(st.details.host),
263
- rb_funcall(rb_cTime, id_at, 2, INT2NUM(deadline.tv_sec),
264
- INT2NUM(deadline.tv_nsec)),
265
- grpc_rb_md_ary_to_h(&st.md_ary), grpc_rb_wrap_call(call), cqueue, NULL);
207
+ ev = rb_completion_queue_pluck(s->queue, tag,
208
+ gpr_inf_future(GPR_CLOCK_REALTIME), NULL);
209
+ if (!ev.success) {
266
210
  grpc_request_call_stack_cleanup(&st);
267
- return result;
211
+ rb_raise(grpc_rb_eCallError, "request_call completion failed");
212
+ return Qnil;
268
213
  }
269
- return Qnil;
214
+
215
+ /* build the NewServerRpc struct result */
216
+ deadline = gpr_convert_clock_type(st.details.deadline, GPR_CLOCK_REALTIME);
217
+ result = rb_struct_new(
218
+ grpc_rb_sNewServerRpc, rb_str_new2(st.details.method),
219
+ rb_str_new2(st.details.host),
220
+ rb_funcall(rb_cTime, id_at, 2, INT2NUM(deadline.tv_sec),
221
+ INT2NUM(deadline.tv_nsec)),
222
+ grpc_rb_md_ary_to_h(&st.md_ary), grpc_rb_wrap_call(call, call_queue),
223
+ NULL);
224
+ grpc_request_call_stack_cleanup(&st);
225
+ return result;
270
226
  }
271
227
 
272
228
  static VALUE grpc_rb_server_start(VALUE self) {
@@ -282,41 +238,33 @@ static VALUE grpc_rb_server_start(VALUE self) {
282
238
 
283
239
  /*
284
240
  call-seq:
285
- cq = CompletionQueue.new
286
- server = Server.new(cq, {'arg1': 'value1'})
241
+ server = Server.new({'arg1': 'value1'})
287
242
  ... // do stuff with server
288
243
  ...
289
244
  ... // to shutdown the server
290
- server.destroy(cq)
245
+ server.destroy()
291
246
 
292
247
  ... // to shutdown the server with a timeout
293
- server.destroy(cq, timeout)
248
+ server.destroy(timeout)
294
249
 
295
250
  Destroys server instances. */
296
251
  static VALUE grpc_rb_server_destroy(int argc, VALUE *argv, VALUE self) {
297
- VALUE cqueue = Qnil;
298
252
  VALUE timeout = Qnil;
299
- grpc_completion_queue *cq = NULL;
300
- grpc_event ev;
253
+ gpr_timespec deadline;
301
254
  grpc_rb_server *s = NULL;
302
255
 
303
- /* "11" == 1 mandatory args, 1 (timeout) is optional */
304
- rb_scan_args(argc, argv, "11", &cqueue, &timeout);
305
- cq = grpc_rb_get_wrapped_completion_queue(cqueue);
256
+ /* "01" == 0 mandatory args, 1 (timeout) is optional */
257
+ rb_scan_args(argc, argv, "01", &timeout);
306
258
  TypedData_Get_Struct(self, grpc_rb_server, &grpc_rb_server_data_type, s);
307
-
308
- if (s->wrapped != NULL) {
309
- grpc_server_shutdown_and_notify(s->wrapped, cq, NULL);
310
- ev = grpc_rb_completion_queue_pluck_event(cqueue, Qnil, timeout);
311
- if (!ev.success) {
312
- rb_warn("server shutdown failed, cancelling the calls, objects may leak");
313
- grpc_server_cancel_all_calls(s->wrapped);
314
- return Qfalse;
315
- }
316
- grpc_server_destroy(s->wrapped);
317
- s->wrapped = NULL;
259
+ if (TYPE(timeout) == T_NIL) {
260
+ deadline = gpr_inf_future(GPR_CLOCK_REALTIME);
261
+ } else {
262
+ deadline = grpc_rb_time_timeval(timeout, /* absolute time*/ 0);
318
263
  }
319
- return Qtrue;
264
+
265
+ destroy_server(s, deadline);
266
+
267
+ return Qnil;
320
268
  }
321
269
 
322
270
  /*
@@ -376,13 +324,13 @@ void Init_grpc_server() {
376
324
  rb_define_alloc_func(grpc_rb_cServer, grpc_rb_server_alloc);
377
325
 
378
326
  /* Provides a ruby constructor and support for dup/clone. */
379
- rb_define_method(grpc_rb_cServer, "initialize", grpc_rb_server_init, 2);
327
+ rb_define_method(grpc_rb_cServer, "initialize", grpc_rb_server_init, 1);
380
328
  rb_define_method(grpc_rb_cServer, "initialize_copy",
381
- grpc_rb_server_init_copy, 1);
329
+ grpc_rb_cannot_init_copy, 1);
382
330
 
383
331
  /* Add the server methods. */
384
332
  rb_define_method(grpc_rb_cServer, "request_call",
385
- grpc_rb_server_request_call, 3);
333
+ grpc_rb_server_request_call, 0);
386
334
  rb_define_method(grpc_rb_cServer, "start", grpc_rb_server_start, 0);
387
335
  rb_define_method(grpc_rb_cServer, "destroy", grpc_rb_server_destroy, -1);
388
336
  rb_define_alias(grpc_rb_cServer, "close", "destroy");