google-protobuf 3.25.6 → 4.30.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (61) hide show
  1. checksums.yaml +4 -4
  2. data/ext/google/protobuf_c/convert.c +33 -15
  3. data/ext/google/protobuf_c/defs.c +526 -29
  4. data/ext/google/protobuf_c/extconf.rb +13 -1
  5. data/ext/google/protobuf_c/glue.c +79 -0
  6. data/ext/google/protobuf_c/map.c +72 -27
  7. data/ext/google/protobuf_c/map.h +7 -3
  8. data/ext/google/protobuf_c/message.c +113 -116
  9. data/ext/google/protobuf_c/message.h +2 -6
  10. data/ext/google/protobuf_c/protobuf.c +32 -18
  11. data/ext/google/protobuf_c/protobuf.h +3 -7
  12. data/ext/google/protobuf_c/repeated_field.c +58 -23
  13. data/ext/google/protobuf_c/repeated_field.h +6 -2
  14. data/ext/google/protobuf_c/ruby-upb.c +14926 -11596
  15. data/ext/google/protobuf_c/ruby-upb.h +8701 -5832
  16. data/ext/google/protobuf_c/shared_convert.c +7 -2
  17. data/ext/google/protobuf_c/shared_message.c +3 -32
  18. data/ext/google/protobuf_c/shared_message.h +0 -4
  19. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range.c +207 -0
  20. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range.h +9 -8
  21. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range_neon.inc +117 -0
  22. data/ext/google/protobuf_c/third_party/utf8_range/utf8_range_sse.inc +272 -0
  23. data/lib/google/protobuf/any_pb.rb +1 -22
  24. data/lib/google/protobuf/api_pb.rb +1 -24
  25. data/lib/google/protobuf/descriptor_pb.rb +4 -23
  26. data/lib/google/protobuf/duration_pb.rb +1 -22
  27. data/lib/google/protobuf/empty_pb.rb +1 -22
  28. data/lib/google/protobuf/ffi/descriptor.rb +14 -4
  29. data/lib/google/protobuf/ffi/descriptor_pool.rb +3 -1
  30. data/lib/google/protobuf/ffi/enum_descriptor.rb +13 -1
  31. data/lib/google/protobuf/ffi/ffi.rb +7 -6
  32. data/lib/google/protobuf/ffi/field_descriptor.rb +23 -2
  33. data/lib/google/protobuf/ffi/file_descriptor.rb +13 -13
  34. data/lib/google/protobuf/ffi/internal/arena.rb +0 -6
  35. data/lib/google/protobuf/ffi/internal/convert.rb +17 -30
  36. data/lib/google/protobuf/ffi/map.rb +50 -24
  37. data/lib/google/protobuf/ffi/message.rb +188 -67
  38. data/lib/google/protobuf/ffi/method_descriptor.rb +124 -0
  39. data/lib/google/protobuf/ffi/object_cache.rb +3 -3
  40. data/lib/google/protobuf/ffi/oneof_descriptor.rb +13 -1
  41. data/lib/google/protobuf/ffi/repeated_field.rb +47 -19
  42. data/lib/google/protobuf/ffi/service_descriptor.rb +117 -0
  43. data/lib/google/protobuf/field_mask_pb.rb +1 -22
  44. data/lib/google/protobuf/internal/object_cache.rb +99 -0
  45. data/lib/google/protobuf/plugin_pb.rb +2 -24
  46. data/lib/google/protobuf/repeated_field.rb +4 -5
  47. data/lib/google/protobuf/source_context_pb.rb +1 -22
  48. data/lib/google/protobuf/struct_pb.rb +1 -22
  49. data/lib/google/protobuf/timestamp_pb.rb +1 -22
  50. data/lib/google/protobuf/type_pb.rb +1 -24
  51. data/lib/google/protobuf/wrappers_pb.rb +1 -22
  52. data/lib/google/protobuf.rb +1 -1
  53. data/lib/google/protobuf_ffi.rb +5 -3
  54. data/lib/google/protobuf_native.rb +0 -1
  55. data/lib/google/tasks/ffi.rake +1 -3
  56. metadata +28 -13
  57. data/ext/google/protobuf_c/third_party/utf8_range/naive.c +0 -92
  58. data/ext/google/protobuf_c/third_party/utf8_range/range2-neon.c +0 -157
  59. data/ext/google/protobuf_c/third_party/utf8_range/range2-sse.c +0 -170
  60. data/lib/google/protobuf/descriptor_dsl.rb +0 -465
  61. data/lib/google/protobuf/object_cache.rb +0 -97
@@ -7,7 +7,6 @@
7
7
 
8
8
  #include <ctype.h>
9
9
  #include <errno.h>
10
- #include <ruby/version.h>
11
10
 
12
11
  #include "convert.h"
13
12
  #include "message.h"
@@ -23,6 +22,9 @@ static VALUE get_enumdef_obj(VALUE descriptor_pool, const upb_EnumDef* def);
23
22
  static VALUE get_fielddef_obj(VALUE descriptor_pool, const upb_FieldDef* def);
24
23
  static VALUE get_filedef_obj(VALUE descriptor_pool, const upb_FileDef* def);
25
24
  static VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_OneofDef* def);
25
+ static VALUE get_servicedef_obj(VALUE descriptor_pool,
26
+ const upb_ServiceDef* def);
27
+ static VALUE get_methoddef_obj(VALUE descriptor_pool, const upb_MethodDef* def);
26
28
 
27
29
  // A distinct object that is not accessible from Ruby. We use this as a
28
30
  // constructor argument to enforce that certain objects cannot be created from
@@ -144,8 +146,8 @@ VALUE DescriptorPool_add_serialized_file(VALUE _self,
144
146
  * call-seq:
145
147
  * DescriptorPool.lookup(name) => descriptor
146
148
  *
147
- * Finds a Descriptor, EnumDescriptor or FieldDescriptor by name and returns it,
148
- * or nil if none exists with the given name.
149
+ * Finds a Descriptor, EnumDescriptor, FieldDescriptor or ServiceDescriptor by
150
+ * name and returns it, or nil if none exists with the given name.
149
151
  */
150
152
  static VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
151
153
  DescriptorPool* self = ruby_to_DescriptorPool(_self);
@@ -153,6 +155,7 @@ static VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
153
155
  const upb_MessageDef* msgdef;
154
156
  const upb_EnumDef* enumdef;
155
157
  const upb_FieldDef* fielddef;
158
+ const upb_ServiceDef* servicedef;
156
159
 
157
160
  msgdef = upb_DefPool_FindMessageByName(self->symtab, name_str);
158
161
  if (msgdef) {
@@ -169,6 +172,11 @@ static VALUE DescriptorPool_lookup(VALUE _self, VALUE name) {
169
172
  return get_enumdef_obj(_self, enumdef);
170
173
  }
171
174
 
175
+ servicedef = upb_DefPool_FindServiceByName(self->symtab, name_str);
176
+ if (servicedef) {
177
+ return get_servicedef_obj(_self, servicedef);
178
+ }
179
+
172
180
  return Qnil;
173
181
  }
174
182
 
@@ -257,7 +265,20 @@ static VALUE decode_options(VALUE self, const char* option_type, int size,
257
265
  VALUE desc_rb = get_msgdef_obj(descriptor_pool, msgdef);
258
266
  const Descriptor* desc = ruby_to_Descriptor(desc_rb);
259
267
 
260
- options_rb = Message_decode_bytes(size, bytes, 0, desc->klass, true);
268
+ options_rb = Message_decode_bytes(size, bytes, 0, desc->klass, false);
269
+
270
+ // Strip features from the options proto to keep it internal.
271
+ const upb_MessageDef* decoded_desc = NULL;
272
+ upb_Message* options = Message_GetMutable(options_rb, &decoded_desc);
273
+ PBRUBY_ASSERT(options != NULL);
274
+ PBRUBY_ASSERT(decoded_desc == msgdef);
275
+ const upb_FieldDef* field =
276
+ upb_MessageDef_FindFieldByName(decoded_desc, "features");
277
+ PBRUBY_ASSERT(field != NULL);
278
+ upb_Message_ClearFieldByDef(options, field);
279
+
280
+ Message_freeze(options_rb);
281
+
261
282
  rb_ivar_set(self, options_instancevar_interned, options_rb);
262
283
  return options_rb;
263
284
  }
@@ -430,6 +451,27 @@ static VALUE Descriptor_options(VALUE _self) {
430
451
  return message_options;
431
452
  }
432
453
 
454
+ /*
455
+ * call-seq:
456
+ * Descriptor.to_proto => DescriptorProto
457
+ *
458
+ * Returns the `DescriptorProto` of this `Descriptor`.
459
+ */
460
+ static VALUE Descriptor_to_proto(VALUE _self) {
461
+ Descriptor* self = ruby_to_Descriptor(_self);
462
+ upb_Arena* arena = upb_Arena_New();
463
+ google_protobuf_DescriptorProto* proto =
464
+ upb_MessageDef_ToProto(self->msgdef, arena);
465
+ size_t size;
466
+ const char* serialized =
467
+ google_protobuf_DescriptorProto_serialize(proto, arena, &size);
468
+ VALUE proto_class = rb_path2class("Google::Protobuf::DescriptorProto");
469
+ VALUE proto_rb =
470
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
471
+ upb_Arena_Free(arena);
472
+ return proto_rb;
473
+ }
474
+
433
475
  static void Descriptor_register(VALUE module) {
434
476
  VALUE klass = rb_define_class_under(module, "Descriptor", rb_cObject);
435
477
  rb_define_alloc_func(klass, Descriptor_alloc);
@@ -442,6 +484,7 @@ static void Descriptor_register(VALUE module) {
442
484
  rb_define_method(klass, "name", Descriptor_name, 0);
443
485
  rb_define_method(klass, "file_descriptor", Descriptor_file_descriptor, 0);
444
486
  rb_define_method(klass, "options", Descriptor_options, 0);
487
+ rb_define_method(klass, "to_proto", Descriptor_to_proto, 0);
445
488
  rb_include_module(klass, rb_mEnumerable);
446
489
  rb_gc_register_address(&cDescriptor);
447
490
  cDescriptor = klass;
@@ -489,7 +532,7 @@ static VALUE FileDescriptor_alloc(VALUE klass) {
489
532
  * call-seq:
490
533
  * FileDescriptor.new => file
491
534
  *
492
- * Returns a new file descriptor. The syntax must be set before it's passed
535
+ * Returns a new file descriptor. May
493
536
  * to a builder.
494
537
  */
495
538
  static VALUE FileDescriptor_initialize(VALUE _self, VALUE cookie,
@@ -519,28 +562,6 @@ static VALUE FileDescriptor_name(VALUE _self) {
519
562
  return name == NULL ? Qnil : rb_str_new2(name);
520
563
  }
521
564
 
522
- /*
523
- * call-seq:
524
- * FileDescriptor.syntax => syntax
525
- *
526
- * Returns this file descriptors syntax.
527
- *
528
- * Valid syntax versions are:
529
- * :proto2 or :proto3.
530
- */
531
- static VALUE FileDescriptor_syntax(VALUE _self) {
532
- FileDescriptor* self = ruby_to_FileDescriptor(_self);
533
-
534
- switch (upb_FileDef_Syntax(self->filedef)) {
535
- case kUpb_Syntax_Proto3:
536
- return ID2SYM(rb_intern("proto3"));
537
- case kUpb_Syntax_Proto2:
538
- return ID2SYM(rb_intern("proto2"));
539
- default:
540
- return Qnil;
541
- }
542
- }
543
-
544
565
  /*
545
566
  * call-seq:
546
567
  * FileDescriptor.options => options
@@ -559,13 +580,37 @@ static VALUE FileDescriptor_options(VALUE _self) {
559
580
  return file_options;
560
581
  }
561
582
 
583
+ /*
584
+ * call-seq:
585
+ * FileDescriptor.to_proto => FileDescriptorProto
586
+ *
587
+ * Returns the `FileDescriptorProto` of this `FileDescriptor`.
588
+ */
589
+ static VALUE FileDescriptor_to_proto(VALUE _self) {
590
+ FileDescriptor* self = ruby_to_FileDescriptor(_self);
591
+ upb_Arena* arena = upb_Arena_New();
592
+ google_protobuf_FileDescriptorProto* file_proto =
593
+ upb_FileDef_ToProto(self->filedef, arena);
594
+
595
+ size_t size;
596
+ const char* serialized =
597
+ google_protobuf_FileDescriptorProto_serialize(file_proto, arena, &size);
598
+
599
+ VALUE file_proto_class =
600
+ rb_path2class("Google::Protobuf::FileDescriptorProto");
601
+ VALUE proto_rb =
602
+ Message_decode_bytes(size, serialized, 0, file_proto_class, false);
603
+ upb_Arena_Free(arena);
604
+ return proto_rb;
605
+ }
606
+
562
607
  static void FileDescriptor_register(VALUE module) {
563
608
  VALUE klass = rb_define_class_under(module, "FileDescriptor", rb_cObject);
564
609
  rb_define_alloc_func(klass, FileDescriptor_alloc);
565
610
  rb_define_method(klass, "initialize", FileDescriptor_initialize, 3);
566
611
  rb_define_method(klass, "name", FileDescriptor_name, 0);
567
- rb_define_method(klass, "syntax", FileDescriptor_syntax, 0);
568
612
  rb_define_method(klass, "options", FileDescriptor_options, 0);
613
+ rb_define_method(klass, "to_proto", FileDescriptor_to_proto, 0);
569
614
  rb_gc_register_address(&cFileDescriptor);
570
615
  cFileDescriptor = klass;
571
616
  }
@@ -727,7 +772,7 @@ static VALUE FieldDescriptor__type(VALUE _self) {
727
772
  static VALUE FieldDescriptor_default(VALUE _self) {
728
773
  FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
729
774
  const upb_FieldDef* f = self->fielddef;
730
- upb_MessageValue default_val = {0};
775
+ upb_MessageValue default_val = upb_MessageValue_Zero();
731
776
  if (upb_FieldDef_IsSubMessage(f)) {
732
777
  return Qnil;
733
778
  } else if (!upb_FieldDef_IsRepeated(f)) {
@@ -736,6 +781,28 @@ static VALUE FieldDescriptor_default(VALUE _self) {
736
781
  return Convert_UpbToRuby(default_val, TypeInfo_get(self->fielddef), Qnil);
737
782
  }
738
783
 
784
+ /*
785
+ * call-seq:
786
+ * FieldDescriptor.has_presence? => bool
787
+ *
788
+ * Returns whether this field tracks presence.
789
+ */
790
+ static VALUE FieldDescriptor_has_presence(VALUE _self) {
791
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
792
+ return upb_FieldDef_HasPresence(self->fielddef) ? Qtrue : Qfalse;
793
+ }
794
+
795
+ /*
796
+ * call-seq:
797
+ * FieldDescriptor.is_packed? => bool
798
+ *
799
+ * Returns whether this is a repeated field that uses packed encoding.
800
+ */
801
+ static VALUE FieldDescriptor_is_packed(VALUE _self) {
802
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
803
+ return upb_FieldDef_IsPacked(self->fielddef) ? Qtrue : Qfalse;
804
+ }
805
+
739
806
  /*
740
807
  * call-seq:
741
808
  * FieldDescriptor.json_name => json_name
@@ -936,6 +1003,27 @@ static VALUE FieldDescriptor_options(VALUE _self) {
936
1003
  return field_options;
937
1004
  }
938
1005
 
1006
+ /*
1007
+ * call-seq:
1008
+ * FieldDescriptor.to_proto => FieldDescriptorProto
1009
+ *
1010
+ * Returns the `FieldDescriptorProto` of this `FieldDescriptor`.
1011
+ */
1012
+ static VALUE FieldDescriptor_to_proto(VALUE _self) {
1013
+ FieldDescriptor* self = ruby_to_FieldDescriptor(_self);
1014
+ upb_Arena* arena = upb_Arena_New();
1015
+ google_protobuf_FieldDescriptorProto* proto =
1016
+ upb_FieldDef_ToProto(self->fielddef, arena);
1017
+ size_t size;
1018
+ const char* serialized =
1019
+ google_protobuf_FieldDescriptorProto_serialize(proto, arena, &size);
1020
+ VALUE proto_class = rb_path2class("Google::Protobuf::FieldDescriptorProto");
1021
+ VALUE proto_rb =
1022
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1023
+ upb_Arena_Free(arena);
1024
+ return proto_rb;
1025
+ }
1026
+
939
1027
  static void FieldDescriptor_register(VALUE module) {
940
1028
  VALUE klass = rb_define_class_under(module, "FieldDescriptor", rb_cObject);
941
1029
  rb_define_alloc_func(klass, FieldDescriptor_alloc);
@@ -943,6 +1031,8 @@ static void FieldDescriptor_register(VALUE module) {
943
1031
  rb_define_method(klass, "name", FieldDescriptor_name, 0);
944
1032
  rb_define_method(klass, "type", FieldDescriptor__type, 0);
945
1033
  rb_define_method(klass, "default", FieldDescriptor_default, 0);
1034
+ rb_define_method(klass, "has_presence?", FieldDescriptor_has_presence, 0);
1035
+ rb_define_method(klass, "is_packed?", FieldDescriptor_is_packed, 0);
946
1036
  rb_define_method(klass, "json_name", FieldDescriptor_json_name, 0);
947
1037
  rb_define_method(klass, "label", FieldDescriptor_label, 0);
948
1038
  rb_define_method(klass, "number", FieldDescriptor_number, 0);
@@ -953,6 +1043,7 @@ static void FieldDescriptor_register(VALUE module) {
953
1043
  rb_define_method(klass, "get", FieldDescriptor_get, 1);
954
1044
  rb_define_method(klass, "set", FieldDescriptor_set, 2);
955
1045
  rb_define_method(klass, "options", FieldDescriptor_options, 0);
1046
+ rb_define_method(klass, "to_proto", FieldDescriptor_to_proto, 0);
956
1047
  rb_gc_register_address(&cFieldDescriptor);
957
1048
  cFieldDescriptor = klass;
958
1049
  }
@@ -1071,6 +1162,27 @@ static VALUE OneOfDescriptor_options(VALUE _self) {
1071
1162
  return oneof_options;
1072
1163
  }
1073
1164
 
1165
+ /*
1166
+ * call-seq:
1167
+ * OneofDescriptor.to_proto => OneofDescriptorProto
1168
+ *
1169
+ * Returns the `OneofDescriptorProto` of this `OneofDescriptor`.
1170
+ */
1171
+ static VALUE OneOfDescriptor_to_proto(VALUE _self) {
1172
+ OneofDescriptor* self = ruby_to_OneofDescriptor(_self);
1173
+ upb_Arena* arena = upb_Arena_New();
1174
+ google_protobuf_OneofDescriptorProto* proto =
1175
+ upb_OneofDef_ToProto(self->oneofdef, arena);
1176
+ size_t size;
1177
+ const char* serialized =
1178
+ google_protobuf_OneofDescriptorProto_serialize(proto, arena, &size);
1179
+ VALUE proto_class = rb_path2class("Google::Protobuf::OneofDescriptorProto");
1180
+ VALUE proto_rb =
1181
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1182
+ upb_Arena_Free(arena);
1183
+ return proto_rb;
1184
+ }
1185
+
1074
1186
  static void OneofDescriptor_register(VALUE module) {
1075
1187
  VALUE klass = rb_define_class_under(module, "OneofDescriptor", rb_cObject);
1076
1188
  rb_define_alloc_func(klass, OneofDescriptor_alloc);
@@ -1078,6 +1190,7 @@ static void OneofDescriptor_register(VALUE module) {
1078
1190
  rb_define_method(klass, "name", OneofDescriptor_name, 0);
1079
1191
  rb_define_method(klass, "each", OneofDescriptor_each, 0);
1080
1192
  rb_define_method(klass, "options", OneOfDescriptor_options, 0);
1193
+ rb_define_method(klass, "to_proto", OneOfDescriptor_to_proto, 0);
1081
1194
  rb_include_module(klass, rb_mEnumerable);
1082
1195
  rb_gc_register_address(&cOneofDescriptor);
1083
1196
  cOneofDescriptor = klass;
@@ -1163,6 +1276,17 @@ static VALUE EnumDescriptor_file_descriptor(VALUE _self) {
1163
1276
  upb_EnumDef_File(self->enumdef));
1164
1277
  }
1165
1278
 
1279
+ /*
1280
+ * call-seq:
1281
+ * EnumDescriptor.is_closed? => bool
1282
+ *
1283
+ * Returns whether this enum is open or closed.
1284
+ */
1285
+ static VALUE EnumDescriptor_is_closed(VALUE _self) {
1286
+ EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1287
+ return upb_EnumDef_IsClosed(self->enumdef) ? Qtrue : Qfalse;
1288
+ }
1289
+
1166
1290
  /*
1167
1291
  * call-seq:
1168
1292
  * EnumDescriptor.name => name
@@ -1265,6 +1389,29 @@ static VALUE EnumDescriptor_options(VALUE _self) {
1265
1389
  return enum_options;
1266
1390
  }
1267
1391
 
1392
+ /*
1393
+ * call-seq:
1394
+ * EnumDescriptor.to_proto => EnumDescriptorProto
1395
+ *
1396
+ * Returns the `EnumDescriptorProto` of this `EnumDescriptor`.
1397
+ */
1398
+ static VALUE EnumDescriptor_to_proto(VALUE _self) {
1399
+ EnumDescriptor* self = ruby_to_EnumDescriptor(_self);
1400
+ upb_Arena* arena = upb_Arena_New();
1401
+ google_protobuf_EnumDescriptorProto* proto =
1402
+ upb_EnumDef_ToProto(self->enumdef, arena);
1403
+
1404
+ size_t size;
1405
+ const char* serialized =
1406
+ google_protobuf_EnumDescriptorProto_serialize(proto, arena, &size);
1407
+
1408
+ VALUE proto_class = rb_path2class("Google::Protobuf::EnumDescriptorProto");
1409
+ VALUE proto_rb =
1410
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1411
+ upb_Arena_Free(arena);
1412
+ return proto_rb;
1413
+ }
1414
+
1268
1415
  static void EnumDescriptor_register(VALUE module) {
1269
1416
  VALUE klass = rb_define_class_under(module, "EnumDescriptor", rb_cObject);
1270
1417
  rb_define_alloc_func(klass, EnumDescriptor_alloc);
@@ -1275,12 +1422,350 @@ static void EnumDescriptor_register(VALUE module) {
1275
1422
  rb_define_method(klass, "each", EnumDescriptor_each, 0);
1276
1423
  rb_define_method(klass, "enummodule", EnumDescriptor_enummodule, 0);
1277
1424
  rb_define_method(klass, "file_descriptor", EnumDescriptor_file_descriptor, 0);
1425
+ rb_define_method(klass, "is_closed?", EnumDescriptor_is_closed, 0);
1278
1426
  rb_define_method(klass, "options", EnumDescriptor_options, 0);
1427
+ rb_define_method(klass, "to_proto", EnumDescriptor_to_proto, 0);
1279
1428
  rb_include_module(klass, rb_mEnumerable);
1280
1429
  rb_gc_register_address(&cEnumDescriptor);
1281
1430
  cEnumDescriptor = klass;
1282
1431
  }
1283
1432
 
1433
+ // -----------------------------------------------------------------------------
1434
+ // ServiceDescriptor
1435
+ // -----------------------------------------------------------------------------
1436
+
1437
+ typedef struct {
1438
+ const upb_ServiceDef* servicedef;
1439
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
1440
+ // macro to update VALUE references, as to trigger write barriers.
1441
+ VALUE module; // begins as nil
1442
+ VALUE descriptor_pool; // Owns the upb_ServiceDef.
1443
+ } ServiceDescriptor;
1444
+
1445
+ static VALUE cServiceDescriptor = Qnil;
1446
+
1447
+ static void ServiceDescriptor_mark(void* _self) {
1448
+ ServiceDescriptor* self = _self;
1449
+ rb_gc_mark(self->module);
1450
+ rb_gc_mark(self->descriptor_pool);
1451
+ }
1452
+
1453
+ static const rb_data_type_t ServiceDescriptor_type = {
1454
+ "Google::Protobuf::ServicDescriptor",
1455
+ {ServiceDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
1456
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
1457
+ };
1458
+
1459
+ static ServiceDescriptor* ruby_to_ServiceDescriptor(VALUE val) {
1460
+ ServiceDescriptor* ret;
1461
+ TypedData_Get_Struct(val, ServiceDescriptor, &ServiceDescriptor_type, ret);
1462
+ return ret;
1463
+ }
1464
+
1465
+ static VALUE ServiceDescriptor_alloc(VALUE klass) {
1466
+ ServiceDescriptor* self = ALLOC(ServiceDescriptor);
1467
+ VALUE ret = TypedData_Wrap_Struct(klass, &ServiceDescriptor_type, self);
1468
+ self->servicedef = NULL;
1469
+ self->module = Qnil;
1470
+ self->descriptor_pool = Qnil;
1471
+ return ret;
1472
+ }
1473
+
1474
+ /*
1475
+ * call-seq:
1476
+ * ServiceDescriptor.new(c_only_cookie, ptr) => ServiceDescriptor
1477
+ *
1478
+ * Creates a descriptor wrapper object. May only be called from C.
1479
+ */
1480
+ static VALUE ServiceDescriptor_initialize(VALUE _self, VALUE cookie,
1481
+ VALUE descriptor_pool, VALUE ptr) {
1482
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1483
+
1484
+ if (cookie != c_only_cookie) {
1485
+ rb_raise(rb_eRuntimeError,
1486
+ "Descriptor objects may not be created from Ruby.");
1487
+ }
1488
+
1489
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
1490
+ self->servicedef = (const upb_ServiceDef*)NUM2ULL(ptr);
1491
+
1492
+ return Qnil;
1493
+ }
1494
+
1495
+ /*
1496
+ * call-seq:
1497
+ * ServiceDescriptor.name => name
1498
+ *
1499
+ * Returns the name of this service.
1500
+ */
1501
+ static VALUE ServiceDescriptor_name(VALUE _self) {
1502
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1503
+ return rb_str_maybe_null(upb_ServiceDef_FullName(self->servicedef));
1504
+ }
1505
+
1506
+ /*
1507
+ * call-seq:
1508
+ * ServiceDescriptor.file_descriptor
1509
+ *
1510
+ * Returns the FileDescriptor object this service belongs to.
1511
+ */
1512
+ static VALUE ServiceDescriptor_file_descriptor(VALUE _self) {
1513
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1514
+ return get_filedef_obj(self->descriptor_pool,
1515
+ upb_ServiceDef_File(self->servicedef));
1516
+ }
1517
+
1518
+ /*
1519
+ * call-seq:
1520
+ * ServiceDescriptor.each(&block)
1521
+ *
1522
+ * Iterates over methods in this service, yielding to the block on each one.
1523
+ */
1524
+ static VALUE ServiceDescriptor_each(VALUE _self) {
1525
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1526
+
1527
+ int n = upb_ServiceDef_MethodCount(self->servicedef);
1528
+ for (int i = 0; i < n; i++) {
1529
+ const upb_MethodDef* method = upb_ServiceDef_Method(self->servicedef, i);
1530
+ VALUE obj = get_methoddef_obj(self->descriptor_pool, method);
1531
+ rb_yield(obj);
1532
+ }
1533
+ return Qnil;
1534
+ }
1535
+
1536
+ /*
1537
+ * call-seq:
1538
+ * ServiceDescriptor.options => options
1539
+ *
1540
+ * Returns the `ServiceOptions` for this `ServiceDescriptor`.
1541
+ */
1542
+ static VALUE ServiceDescriptor_options(VALUE _self) {
1543
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1544
+ const google_protobuf_ServiceOptions* opts =
1545
+ upb_ServiceDef_Options(self->servicedef);
1546
+ upb_Arena* arena = upb_Arena_New();
1547
+ size_t size;
1548
+ char* serialized =
1549
+ google_protobuf_ServiceOptions_serialize(opts, arena, &size);
1550
+ VALUE service_options = decode_options(_self, "ServiceOptions", size,
1551
+ serialized, self->descriptor_pool);
1552
+ upb_Arena_Free(arena);
1553
+ return service_options;
1554
+ }
1555
+
1556
+ /*
1557
+ * call-seq:
1558
+ * ServiceDescriptor.to_proto => ServiceDescriptorProto
1559
+ *
1560
+ * Returns the `ServiceDescriptorProto` of this `ServiceDescriptor`.
1561
+ */
1562
+ static VALUE ServiceDescriptor_to_proto(VALUE _self) {
1563
+ ServiceDescriptor* self = ruby_to_ServiceDescriptor(_self);
1564
+ upb_Arena* arena = upb_Arena_New();
1565
+ google_protobuf_ServiceDescriptorProto* proto =
1566
+ upb_ServiceDef_ToProto(self->servicedef, arena);
1567
+ size_t size;
1568
+ const char* serialized =
1569
+ google_protobuf_ServiceDescriptorProto_serialize(proto, arena, &size);
1570
+ VALUE proto_class = rb_path2class("Google::Protobuf::ServiceDescriptorProto");
1571
+ VALUE proto_rb =
1572
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1573
+ upb_Arena_Free(arena);
1574
+ return proto_rb;
1575
+ }
1576
+
1577
+ static void ServiceDescriptor_register(VALUE module) {
1578
+ VALUE klass = rb_define_class_under(module, "ServiceDescriptor", rb_cObject);
1579
+ rb_define_alloc_func(klass, ServiceDescriptor_alloc);
1580
+ rb_define_method(klass, "initialize", ServiceDescriptor_initialize, 3);
1581
+ rb_define_method(klass, "name", ServiceDescriptor_name, 0);
1582
+ rb_define_method(klass, "each", ServiceDescriptor_each, 0);
1583
+ rb_define_method(klass, "file_descriptor", ServiceDescriptor_file_descriptor,
1584
+ 0);
1585
+ rb_define_method(klass, "options", ServiceDescriptor_options, 0);
1586
+ rb_define_method(klass, "to_proto", ServiceDescriptor_to_proto, 0);
1587
+ rb_include_module(klass, rb_mEnumerable);
1588
+ rb_gc_register_address(&cServiceDescriptor);
1589
+ cServiceDescriptor = klass;
1590
+ }
1591
+
1592
+ // -----------------------------------------------------------------------------
1593
+ // MethodDescriptor
1594
+ // -----------------------------------------------------------------------------
1595
+
1596
+ typedef struct {
1597
+ const upb_MethodDef* methoddef;
1598
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
1599
+ // macro to update VALUE references, as to trigger write barriers.
1600
+ VALUE module; // begins as nil
1601
+ VALUE descriptor_pool; // Owns the upb_MethodDef.
1602
+ } MethodDescriptor;
1603
+
1604
+ static VALUE cMethodDescriptor = Qnil;
1605
+
1606
+ static void MethodDescriptor_mark(void* _self) {
1607
+ MethodDescriptor* self = _self;
1608
+ rb_gc_mark(self->module);
1609
+ rb_gc_mark(self->descriptor_pool);
1610
+ }
1611
+
1612
+ static const rb_data_type_t MethodDescriptor_type = {
1613
+ "Google::Protobuf::MethodDescriptor",
1614
+ {MethodDescriptor_mark, RUBY_DEFAULT_FREE, NULL},
1615
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED,
1616
+ };
1617
+
1618
+ static MethodDescriptor* ruby_to_MethodDescriptor(VALUE val) {
1619
+ MethodDescriptor* ret;
1620
+ TypedData_Get_Struct(val, MethodDescriptor, &MethodDescriptor_type, ret);
1621
+ return ret;
1622
+ }
1623
+
1624
+ static VALUE MethodDescriptor_alloc(VALUE klass) {
1625
+ MethodDescriptor* self = ALLOC(MethodDescriptor);
1626
+ VALUE ret = TypedData_Wrap_Struct(klass, &MethodDescriptor_type, self);
1627
+ self->methoddef = NULL;
1628
+ self->module = Qnil;
1629
+ self->descriptor_pool = Qnil;
1630
+ return ret;
1631
+ }
1632
+
1633
+ /*
1634
+ * call-seq:
1635
+ * MethodDescriptor.new(c_only_cookie, ptr) => MethodDescriptor
1636
+ *
1637
+ * Creates a descriptor wrapper object. May only be called from C.
1638
+ */
1639
+ static VALUE MethodDescriptor_initialize(VALUE _self, VALUE cookie,
1640
+ VALUE descriptor_pool, VALUE ptr) {
1641
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1642
+
1643
+ if (cookie != c_only_cookie) {
1644
+ rb_raise(rb_eRuntimeError,
1645
+ "Descriptor objects may not be created from Ruby.");
1646
+ }
1647
+
1648
+ RB_OBJ_WRITE(_self, &self->descriptor_pool, descriptor_pool);
1649
+ self->methoddef = (const upb_MethodDef*)NUM2ULL(ptr);
1650
+
1651
+ return Qnil;
1652
+ }
1653
+
1654
+ /*
1655
+ * call-seq:
1656
+ * MethodDescriptor.name => name
1657
+ *
1658
+ * Returns the name of this method
1659
+ */
1660
+ static VALUE MethodDescriptor_name(VALUE _self) {
1661
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1662
+ return rb_str_maybe_null(upb_MethodDef_Name(self->methoddef));
1663
+ }
1664
+
1665
+ /*
1666
+ * call-seq:
1667
+ * MethodDescriptor.options => options
1668
+ *
1669
+ * Returns the `MethodOptions` for this `MethodDescriptor`.
1670
+ */
1671
+ static VALUE MethodDescriptor_options(VALUE _self) {
1672
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1673
+ const google_protobuf_MethodOptions* opts =
1674
+ upb_MethodDef_Options(self->methoddef);
1675
+ upb_Arena* arena = upb_Arena_New();
1676
+ size_t size;
1677
+ char* serialized =
1678
+ google_protobuf_MethodOptions_serialize(opts, arena, &size);
1679
+ VALUE method_options = decode_options(_self, "MethodOptions", size,
1680
+ serialized, self->descriptor_pool);
1681
+ upb_Arena_Free(arena);
1682
+ return method_options;
1683
+ }
1684
+
1685
+ /*
1686
+ * call-seq:
1687
+ * MethodDescriptor.input_type => Descriptor
1688
+ *
1689
+ * Returns the `Descriptor` for the request message type of this method
1690
+ */
1691
+ static VALUE MethodDescriptor_input_type(VALUE _self) {
1692
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1693
+ const upb_MessageDef* type = upb_MethodDef_InputType(self->methoddef);
1694
+ return get_msgdef_obj(self->descriptor_pool, type);
1695
+ }
1696
+
1697
+ /*
1698
+ * call-seq:
1699
+ * MethodDescriptor.output_type => Descriptor
1700
+ *
1701
+ * Returns the `Descriptor` for the response message type of this method
1702
+ */
1703
+ static VALUE MethodDescriptor_output_type(VALUE _self) {
1704
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1705
+ const upb_MessageDef* type = upb_MethodDef_OutputType(self->methoddef);
1706
+ return get_msgdef_obj(self->descriptor_pool, type);
1707
+ }
1708
+
1709
+ /*
1710
+ * call-seq:
1711
+ * MethodDescriptor.client_streaming => bool
1712
+ *
1713
+ * Returns whether or not this is a streaming request method
1714
+ */
1715
+ static VALUE MethodDescriptor_client_streaming(VALUE _self) {
1716
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1717
+ return upb_MethodDef_ClientStreaming(self->methoddef) ? Qtrue : Qfalse;
1718
+ }
1719
+
1720
+ /*
1721
+ * call-seq:
1722
+ * MethodDescriptor.to_proto => MethodDescriptorProto
1723
+ *
1724
+ * Returns the `MethodDescriptorProto` of this `MethodDescriptor`.
1725
+ */
1726
+ static VALUE MethodDescriptor_to_proto(VALUE _self) {
1727
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1728
+ upb_Arena* arena = upb_Arena_New();
1729
+ google_protobuf_MethodDescriptorProto* proto =
1730
+ upb_MethodDef_ToProto(self->methoddef, arena);
1731
+ size_t size;
1732
+ const char* serialized =
1733
+ google_protobuf_MethodDescriptorProto_serialize(proto, arena, &size);
1734
+ VALUE proto_class = rb_path2class("Google::Protobuf::MethodDescriptorProto");
1735
+ VALUE proto_rb =
1736
+ Message_decode_bytes(size, serialized, 0, proto_class, false);
1737
+ upb_Arena_Free(arena);
1738
+ return proto_rb;
1739
+ }
1740
+
1741
+ /*
1742
+ * call-seq:
1743
+ * MethodDescriptor.server_streaming => bool
1744
+ *
1745
+ * Returns whether or not this is a streaming response method
1746
+ */
1747
+ static VALUE MethodDescriptor_server_streaming(VALUE _self) {
1748
+ MethodDescriptor* self = ruby_to_MethodDescriptor(_self);
1749
+ return upb_MethodDef_ServerStreaming(self->methoddef) ? Qtrue : Qfalse;
1750
+ }
1751
+
1752
+ static void MethodDescriptor_register(VALUE module) {
1753
+ VALUE klass = rb_define_class_under(module, "MethodDescriptor", rb_cObject);
1754
+ rb_define_alloc_func(klass, MethodDescriptor_alloc);
1755
+ rb_define_method(klass, "initialize", MethodDescriptor_initialize, 3);
1756
+ rb_define_method(klass, "name", MethodDescriptor_name, 0);
1757
+ rb_define_method(klass, "options", MethodDescriptor_options, 0);
1758
+ rb_define_method(klass, "input_type", MethodDescriptor_input_type, 0);
1759
+ rb_define_method(klass, "output_type", MethodDescriptor_output_type, 0);
1760
+ rb_define_method(klass, "client_streaming", MethodDescriptor_client_streaming,
1761
+ 0);
1762
+ rb_define_method(klass, "server_streaming", MethodDescriptor_server_streaming,
1763
+ 0);
1764
+ rb_define_method(klass, "to_proto", MethodDescriptor_to_proto, 0);
1765
+ rb_gc_register_address(&cMethodDescriptor);
1766
+ cMethodDescriptor = klass;
1767
+ }
1768
+
1284
1769
  static VALUE get_def_obj(VALUE _descriptor_pool, const void* ptr, VALUE klass) {
1285
1770
  DescriptorPool* descriptor_pool = ruby_to_DescriptorPool(_descriptor_pool);
1286
1771
  VALUE key = ULL2NUM((intptr_t)ptr);
@@ -1322,6 +1807,16 @@ static VALUE get_oneofdef_obj(VALUE descriptor_pool, const upb_OneofDef* def) {
1322
1807
  return get_def_obj(descriptor_pool, def, cOneofDescriptor);
1323
1808
  }
1324
1809
 
1810
+ static VALUE get_servicedef_obj(VALUE descriptor_pool,
1811
+ const upb_ServiceDef* def) {
1812
+ return get_def_obj(descriptor_pool, def, cServiceDescriptor);
1813
+ }
1814
+
1815
+ static VALUE get_methoddef_obj(VALUE descriptor_pool,
1816
+ const upb_MethodDef* def) {
1817
+ return get_def_obj(descriptor_pool, def, cMethodDescriptor);
1818
+ }
1819
+
1325
1820
  // -----------------------------------------------------------------------------
1326
1821
  // Shared functions
1327
1822
  // -----------------------------------------------------------------------------
@@ -1397,6 +1892,8 @@ void Defs_register(VALUE module) {
1397
1892
  FieldDescriptor_register(module);
1398
1893
  OneofDescriptor_register(module);
1399
1894
  EnumDescriptor_register(module);
1895
+ ServiceDescriptor_register(module);
1896
+ MethodDescriptor_register(module);
1400
1897
 
1401
1898
  rb_gc_register_address(&c_only_cookie);
1402
1899
  c_only_cookie = rb_class_new_instance(0, NULL, rb_cObject);