|
|
|
@ -122,7 +122,7 @@ void DescriptorPool_register(VALUE module) { |
|
|
|
|
module, "DescriptorPool", rb_cObject); |
|
|
|
|
rb_define_alloc_func(klass, DescriptorPool_alloc); |
|
|
|
|
rb_define_method(klass, "add", DescriptorPool_add, 1); |
|
|
|
|
rb_define_method(klass, "build", DescriptorPool_build, 0); |
|
|
|
|
rb_define_method(klass, "build", DescriptorPool_build, -1); |
|
|
|
|
rb_define_method(klass, "lookup", DescriptorPool_lookup, 1); |
|
|
|
|
rb_define_singleton_method(klass, "generated_pool", |
|
|
|
|
DescriptorPool_generated_pool, 0); |
|
|
|
@ -181,7 +181,7 @@ VALUE DescriptorPool_add(VALUE _self, VALUE def) { |
|
|
|
|
* Builder#add_enum within the block as appropriate. This is the recommended, |
|
|
|
|
* idiomatic way to define new message and enum types. |
|
|
|
|
*/ |
|
|
|
|
VALUE DescriptorPool_build(VALUE _self) { |
|
|
|
|
VALUE DescriptorPool_build(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
VALUE ctx = rb_class_new_instance(0, NULL, cBuilder); |
|
|
|
|
VALUE block = rb_block_proc(); |
|
|
|
|
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block); |
|
|
|
@ -289,6 +289,7 @@ void Descriptor_register(VALUE module) { |
|
|
|
|
VALUE klass = rb_define_class_under( |
|
|
|
|
module, "Descriptor", rb_cObject); |
|
|
|
|
rb_define_alloc_func(klass, Descriptor_alloc); |
|
|
|
|
rb_define_method(klass, "initialize", Descriptor_initialize, 1); |
|
|
|
|
rb_define_method(klass, "each", Descriptor_each, 0); |
|
|
|
|
rb_define_method(klass, "lookup", Descriptor_lookup, 1); |
|
|
|
|
rb_define_method(klass, "add_field", Descriptor_add_field, 1); |
|
|
|
@ -298,11 +299,42 @@ void Descriptor_register(VALUE module) { |
|
|
|
|
rb_define_method(klass, "msgclass", Descriptor_msgclass, 0); |
|
|
|
|
rb_define_method(klass, "name", Descriptor_name, 0); |
|
|
|
|
rb_define_method(klass, "name=", Descriptor_name_set, 1); |
|
|
|
|
rb_define_method(klass, "file_descriptor", Descriptor_file_descriptor, 0); |
|
|
|
|
rb_include_module(klass, rb_mEnumerable); |
|
|
|
|
rb_gc_register_address(&cDescriptor); |
|
|
|
|
cDescriptor = klass; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Descriptor.new(file_descriptor) |
|
|
|
|
* |
|
|
|
|
* Initializes a new descriptor and assigns a file descriptor to it. |
|
|
|
|
*/ |
|
|
|
|
VALUE Descriptor_initialize(VALUE _self, VALUE file_descriptor_rb) { |
|
|
|
|
DEFINE_SELF(Descriptor, self, _self); |
|
|
|
|
|
|
|
|
|
FileDescriptor* file_descriptor = ruby_to_FileDescriptor(file_descriptor_rb); |
|
|
|
|
|
|
|
|
|
CHECK_UPB( |
|
|
|
|
upb_filedef_addmsg(file_descriptor->filedef, self->msgdef, NULL, &status), |
|
|
|
|
"Failed to associate message to file descriptor."); |
|
|
|
|
add_def_obj(file_descriptor->filedef, file_descriptor_rb); |
|
|
|
|
|
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Descriptor.file_descriptor |
|
|
|
|
* |
|
|
|
|
* Returns the FileDescriptor object this message belongs to. |
|
|
|
|
*/ |
|
|
|
|
VALUE Descriptor_file_descriptor(VALUE _self) { |
|
|
|
|
DEFINE_SELF(Descriptor, self, _self); |
|
|
|
|
return get_def_obj(upb_def_file(self->msgdef)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Descriptor.name => name |
|
|
|
@ -470,6 +502,142 @@ VALUE Descriptor_msgclass(VALUE _self) { |
|
|
|
|
return self->klass; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
// FileDescriptor.
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
DEFINE_CLASS(FileDescriptor, "Google::Protobuf::FileDescriptor"); |
|
|
|
|
|
|
|
|
|
void FileDescriptor_mark(void* _self) { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void FileDescriptor_free(void* _self) { |
|
|
|
|
FileDescriptor* self = _self; |
|
|
|
|
upb_filedef_unref(self->filedef, &self->filedef); |
|
|
|
|
xfree(self); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileDescriptor.new => file |
|
|
|
|
* |
|
|
|
|
* Returns a new file descriptor. The syntax must be set before it's passed |
|
|
|
|
* to a builder. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileDescriptor_alloc(VALUE klass) { |
|
|
|
|
FileDescriptor* self = ALLOC(FileDescriptor); |
|
|
|
|
VALUE ret = TypedData_Wrap_Struct(klass, &_FileDescriptor_type, self); |
|
|
|
|
upb_filedef* filedef = upb_filedef_new(&self->filedef); |
|
|
|
|
self->filedef = filedef; |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void FileDescriptor_register(VALUE module) { |
|
|
|
|
VALUE klass = rb_define_class_under( |
|
|
|
|
module, "FileDescriptor", rb_cObject); |
|
|
|
|
rb_define_alloc_func(klass, FileDescriptor_alloc); |
|
|
|
|
rb_define_method(klass, "initialize", FileDescriptor_initialize, -1); |
|
|
|
|
rb_define_method(klass, "name", FileDescriptor_name, 0); |
|
|
|
|
rb_define_method(klass, "syntax", FileDescriptor_syntax, 0); |
|
|
|
|
rb_define_method(klass, "syntax=", FileDescriptor_syntax_set, 1); |
|
|
|
|
cFileDescriptor = klass; |
|
|
|
|
rb_gc_register_address(&cFileDescriptor); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileDescriptor.new(name, options = nil) => file |
|
|
|
|
* |
|
|
|
|
* Initializes a new file descriptor with the given file name. |
|
|
|
|
* Also accepts an optional "options" hash, specifying other optional |
|
|
|
|
* metadata about the file. The options hash currently accepts the following |
|
|
|
|
* * "syntax": :proto2 or :proto3 (default: :proto3) |
|
|
|
|
*/ |
|
|
|
|
VALUE FileDescriptor_initialize(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
DEFINE_SELF(FileDescriptor, self, _self); |
|
|
|
|
|
|
|
|
|
VALUE name_rb; |
|
|
|
|
VALUE options = Qnil; |
|
|
|
|
rb_scan_args(argc, argv, "11", &name_rb, &options); |
|
|
|
|
|
|
|
|
|
if (name_rb != Qnil) { |
|
|
|
|
Check_Type(name_rb, T_STRING); |
|
|
|
|
const char* name = get_str(name_rb); |
|
|
|
|
CHECK_UPB(upb_filedef_setname(self->filedef, name, &status), |
|
|
|
|
"Error setting file name"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Default syntax is proto3.
|
|
|
|
|
VALUE syntax = ID2SYM(rb_intern("proto3")); |
|
|
|
|
if (options != Qnil) { |
|
|
|
|
Check_Type(options, T_HASH); |
|
|
|
|
|
|
|
|
|
if (rb_funcall(options, rb_intern("key?"), 1, |
|
|
|
|
ID2SYM(rb_intern("syntax"))) == Qtrue) { |
|
|
|
|
syntax = rb_hash_lookup(options, ID2SYM(rb_intern("syntax"))); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
FileDescriptor_syntax_set(_self, syntax); |
|
|
|
|
|
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileDescriptor.name => name |
|
|
|
|
* |
|
|
|
|
* Returns the name of the file. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileDescriptor_name(VALUE _self) { |
|
|
|
|
DEFINE_SELF(FileDescriptor, self, _self); |
|
|
|
|
const char* name = upb_filedef_name(self->filedef); |
|
|
|
|
return name == NULL ? Qnil : rb_str_new2(name); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileDescriptor.syntax => syntax |
|
|
|
|
* |
|
|
|
|
* Returns this file descriptors syntax. |
|
|
|
|
* |
|
|
|
|
* Valid syntax versions are: |
|
|
|
|
* :proto2 or :proto3. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileDescriptor_syntax(VALUE _self) { |
|
|
|
|
DEFINE_SELF(FileDescriptor, self, _self); |
|
|
|
|
|
|
|
|
|
switch (upb_filedef_syntax(self->filedef)) { |
|
|
|
|
case UPB_SYNTAX_PROTO3: return ID2SYM(rb_intern("proto3")); |
|
|
|
|
case UPB_SYNTAX_PROTO2: return ID2SYM(rb_intern("proto2")); |
|
|
|
|
default: return Qnil; |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileDescriptor.syntax = version |
|
|
|
|
* |
|
|
|
|
* Sets this file descriptor's syntax, can be :proto3 or :proto2. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileDescriptor_syntax_set(VALUE _self, VALUE syntax_rb) { |
|
|
|
|
DEFINE_SELF(FileDescriptor, self, _self); |
|
|
|
|
Check_Type(syntax_rb, T_SYMBOL); |
|
|
|
|
|
|
|
|
|
upb_syntax_t syntax; |
|
|
|
|
if (SYM2ID(syntax_rb) == rb_intern("proto3")) { |
|
|
|
|
syntax = UPB_SYNTAX_PROTO3; |
|
|
|
|
} else if (SYM2ID(syntax_rb) == rb_intern("proto2")) { |
|
|
|
|
syntax = UPB_SYNTAX_PROTO2; |
|
|
|
|
} else { |
|
|
|
|
rb_raise(rb_eArgError, "Expected :proto3 or :proto3, received '%s'", |
|
|
|
|
rb_id2name(SYM2ID(syntax_rb))); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
CHECK_UPB(upb_filedef_setsyntax(self->filedef, syntax, &status), |
|
|
|
|
"Error setting file syntax for proto"); |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
// FieldDescriptor.
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
@ -509,6 +677,8 @@ void FieldDescriptor_register(VALUE module) { |
|
|
|
|
rb_define_method(klass, "name=", FieldDescriptor_name_set, 1); |
|
|
|
|
rb_define_method(klass, "type", FieldDescriptor_type, 0); |
|
|
|
|
rb_define_method(klass, "type=", FieldDescriptor_type_set, 1); |
|
|
|
|
rb_define_method(klass, "default", FieldDescriptor_default, 0); |
|
|
|
|
rb_define_method(klass, "default=", FieldDescriptor_default_set, 1); |
|
|
|
|
rb_define_method(klass, "label", FieldDescriptor_label, 0); |
|
|
|
|
rb_define_method(klass, "label=", FieldDescriptor_label_set, 1); |
|
|
|
|
rb_define_method(klass, "number", FieldDescriptor_number, 0); |
|
|
|
@ -516,6 +686,8 @@ void FieldDescriptor_register(VALUE module) { |
|
|
|
|
rb_define_method(klass, "submsg_name", FieldDescriptor_submsg_name, 0); |
|
|
|
|
rb_define_method(klass, "submsg_name=", FieldDescriptor_submsg_name_set, 1); |
|
|
|
|
rb_define_method(klass, "subtype", FieldDescriptor_subtype, 0); |
|
|
|
|
rb_define_method(klass, "has?", FieldDescriptor_has, 1); |
|
|
|
|
rb_define_method(klass, "clear", FieldDescriptor_clear, 1); |
|
|
|
|
rb_define_method(klass, "get", FieldDescriptor_get, 1); |
|
|
|
|
rb_define_method(klass, "set", FieldDescriptor_set, 2); |
|
|
|
|
rb_gc_register_address(&cFieldDescriptor); |
|
|
|
@ -691,6 +863,71 @@ VALUE FieldDescriptor_type_set(VALUE _self, VALUE type) { |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FieldDescriptor.default => default |
|
|
|
|
* |
|
|
|
|
* Returns this field's default, as a Ruby object, or nil if not yet set. |
|
|
|
|
*/ |
|
|
|
|
VALUE FieldDescriptor_default(VALUE _self) { |
|
|
|
|
DEFINE_SELF(FieldDescriptor, self, _self); |
|
|
|
|
return layout_get_default(self->fielddef); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FieldDescriptor.default = default |
|
|
|
|
* |
|
|
|
|
* Sets this field's default value. Raises an exception when calling with |
|
|
|
|
* proto syntax 3. |
|
|
|
|
*/ |
|
|
|
|
VALUE FieldDescriptor_default_set(VALUE _self, VALUE default_value) { |
|
|
|
|
DEFINE_SELF(FieldDescriptor, self, _self); |
|
|
|
|
upb_fielddef* mut_def = check_field_notfrozen(self->fielddef); |
|
|
|
|
|
|
|
|
|
switch (upb_fielddef_type(mut_def)) { |
|
|
|
|
case UPB_TYPE_FLOAT:
|
|
|
|
|
upb_fielddef_setdefaultfloat(mut_def, NUM2DBL(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_DOUBLE: |
|
|
|
|
upb_fielddef_setdefaultdouble(mut_def, NUM2DBL(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_BOOL: |
|
|
|
|
if (!RB_TYPE_P(default_value, T_TRUE) && |
|
|
|
|
!RB_TYPE_P(default_value, T_FALSE) && |
|
|
|
|
!RB_TYPE_P(default_value, T_NIL)) { |
|
|
|
|
rb_raise(cTypeError, "Expected boolean for default value."); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
upb_fielddef_setdefaultbool(mut_def, RTEST(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_ENUM: |
|
|
|
|
case UPB_TYPE_INT32:
|
|
|
|
|
upb_fielddef_setdefaultint32(mut_def, NUM2INT(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_INT64:
|
|
|
|
|
upb_fielddef_setdefaultint64(mut_def, NUM2INT(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_UINT32:
|
|
|
|
|
upb_fielddef_setdefaultuint32(mut_def, NUM2UINT(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_UINT64:
|
|
|
|
|
upb_fielddef_setdefaultuint64(mut_def, NUM2UINT(default_value)); |
|
|
|
|
break; |
|
|
|
|
case UPB_TYPE_STRING: |
|
|
|
|
case UPB_TYPE_BYTES: |
|
|
|
|
CHECK_UPB(upb_fielddef_setdefaultcstr(mut_def, StringValuePtr(default_value), |
|
|
|
|
&status), |
|
|
|
|
"Error setting default string"); |
|
|
|
|
break; |
|
|
|
|
default: |
|
|
|
|
rb_raise(rb_eArgError, "Defaults not supported on field %s.%s", |
|
|
|
|
upb_fielddef_fullname(mut_def), upb_fielddef_name(mut_def)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FieldDescriptor.label => label |
|
|
|
@ -859,6 +1096,44 @@ VALUE FieldDescriptor_get(VALUE _self, VALUE msg_rb) { |
|
|
|
|
return layout_get(msg->descriptor->layout, Message_data(msg), self->fielddef); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FieldDescriptor.has?(message) => boolean |
|
|
|
|
* |
|
|
|
|
* Returns whether the value is set on the given message. Raises an |
|
|
|
|
* exception when calling with proto syntax 3. |
|
|
|
|
*/ |
|
|
|
|
VALUE FieldDescriptor_has(VALUE _self, VALUE msg_rb) { |
|
|
|
|
DEFINE_SELF(FieldDescriptor, self, _self); |
|
|
|
|
MessageHeader* msg; |
|
|
|
|
TypedData_Get_Struct(msg_rb, MessageHeader, &Message_type, msg); |
|
|
|
|
if (msg->descriptor->msgdef != upb_fielddef_containingtype(self->fielddef)) { |
|
|
|
|
rb_raise(cTypeError, "has method called on wrong message type"); |
|
|
|
|
} else if (!upb_fielddef_haspresence(self->fielddef)) { |
|
|
|
|
rb_raise(rb_eArgError, "does not track presence"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return layout_has(msg->descriptor->layout, Message_data(msg), self->fielddef); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FieldDescriptor.clear(message) |
|
|
|
|
* |
|
|
|
|
* Clears the field from the message if it's set. |
|
|
|
|
*/ |
|
|
|
|
VALUE FieldDescriptor_clear(VALUE _self, VALUE msg_rb) { |
|
|
|
|
DEFINE_SELF(FieldDescriptor, self, _self); |
|
|
|
|
MessageHeader* msg; |
|
|
|
|
TypedData_Get_Struct(msg_rb, MessageHeader, &Message_type, msg); |
|
|
|
|
if (msg->descriptor->msgdef != upb_fielddef_containingtype(self->fielddef)) { |
|
|
|
|
rb_raise(cTypeError, "has method called on wrong message type"); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
layout_clear(msg->descriptor->layout, Message_data(msg), self->fielddef); |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FieldDescriptor.set(message, value) |
|
|
|
@ -1029,6 +1304,7 @@ void EnumDescriptor_register(VALUE module) { |
|
|
|
|
VALUE klass = rb_define_class_under( |
|
|
|
|
module, "EnumDescriptor", rb_cObject); |
|
|
|
|
rb_define_alloc_func(klass, EnumDescriptor_alloc); |
|
|
|
|
rb_define_method(klass, "initialize", EnumDescriptor_initialize, 1); |
|
|
|
|
rb_define_method(klass, "name", EnumDescriptor_name, 0); |
|
|
|
|
rb_define_method(klass, "name=", EnumDescriptor_name_set, 1); |
|
|
|
|
rb_define_method(klass, "add_value", EnumDescriptor_add_value, 2); |
|
|
|
@ -1036,11 +1312,41 @@ void EnumDescriptor_register(VALUE module) { |
|
|
|
|
rb_define_method(klass, "lookup_value", EnumDescriptor_lookup_value, 1); |
|
|
|
|
rb_define_method(klass, "each", EnumDescriptor_each, 0); |
|
|
|
|
rb_define_method(klass, "enummodule", EnumDescriptor_enummodule, 0); |
|
|
|
|
rb_define_method(klass, "file_descriptor", EnumDescriptor_file_descriptor, 0); |
|
|
|
|
rb_include_module(klass, rb_mEnumerable); |
|
|
|
|
rb_gc_register_address(&cEnumDescriptor); |
|
|
|
|
cEnumDescriptor = klass; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Descriptor.new(file_descriptor) |
|
|
|
|
* |
|
|
|
|
* Initializes a new descriptor and assigns a file descriptor to it. |
|
|
|
|
*/ |
|
|
|
|
VALUE EnumDescriptor_initialize(VALUE _self, VALUE file_descriptor_rb) { |
|
|
|
|
DEFINE_SELF(EnumDescriptor, self, _self); |
|
|
|
|
FileDescriptor* file_descriptor = ruby_to_FileDescriptor(file_descriptor_rb); |
|
|
|
|
CHECK_UPB( |
|
|
|
|
upb_filedef_addenum(file_descriptor->filedef, self->enumdef, |
|
|
|
|
NULL, &status), |
|
|
|
|
"Failed to associate enum to file descriptor."); |
|
|
|
|
add_def_obj(file_descriptor->filedef, file_descriptor_rb); |
|
|
|
|
|
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Descriptor.file_descriptor |
|
|
|
|
* |
|
|
|
|
* Returns the FileDescriptor object this enum belongs to. |
|
|
|
|
*/ |
|
|
|
|
VALUE EnumDescriptor_file_descriptor(VALUE _self) { |
|
|
|
|
DEFINE_SELF(EnumDescriptor, self, _self); |
|
|
|
|
return get_def_obj(upb_def_file(self->enumdef)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* EnumDescriptor.name => name |
|
|
|
@ -1223,34 +1529,56 @@ VALUE MessageBuilderContext_initialize(VALUE _self, |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static VALUE msgdef_add_field(VALUE msgdef, |
|
|
|
|
static VALUE msgdef_add_field(VALUE msgdef_rb, |
|
|
|
|
const char* label, VALUE name, |
|
|
|
|
VALUE type, VALUE number, |
|
|
|
|
VALUE type_class) { |
|
|
|
|
VALUE fielddef = rb_class_new_instance(0, NULL, cFieldDescriptor); |
|
|
|
|
VALUE type_class, |
|
|
|
|
VALUE options) { |
|
|
|
|
VALUE fielddef_rb = rb_class_new_instance(0, NULL, cFieldDescriptor); |
|
|
|
|
VALUE name_str = rb_str_new2(rb_id2name(SYM2ID(name))); |
|
|
|
|
|
|
|
|
|
rb_funcall(fielddef, rb_intern("label="), 1, ID2SYM(rb_intern(label))); |
|
|
|
|
rb_funcall(fielddef, rb_intern("name="), 1, name_str); |
|
|
|
|
rb_funcall(fielddef, rb_intern("type="), 1, type); |
|
|
|
|
rb_funcall(fielddef, rb_intern("number="), 1, number); |
|
|
|
|
rb_funcall(fielddef_rb, rb_intern("label="), 1, ID2SYM(rb_intern(label))); |
|
|
|
|
rb_funcall(fielddef_rb, rb_intern("name="), 1, name_str); |
|
|
|
|
rb_funcall(fielddef_rb, rb_intern("type="), 1, type); |
|
|
|
|
rb_funcall(fielddef_rb, rb_intern("number="), 1, number); |
|
|
|
|
|
|
|
|
|
if (type_class != Qnil) { |
|
|
|
|
if (TYPE(type_class) != T_STRING) { |
|
|
|
|
rb_raise(rb_eArgError, "Expected string for type class"); |
|
|
|
|
} |
|
|
|
|
Check_Type(type_class, T_STRING); |
|
|
|
|
|
|
|
|
|
// Make it an absolute type name by prepending a dot.
|
|
|
|
|
type_class = rb_str_append(rb_str_new2("."), type_class); |
|
|
|
|
rb_funcall(fielddef, rb_intern("submsg_name="), 1, type_class); |
|
|
|
|
rb_funcall(fielddef_rb, rb_intern("submsg_name="), 1, type_class); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
rb_funcall(msgdef, rb_intern("add_field"), 1, fielddef); |
|
|
|
|
return fielddef; |
|
|
|
|
if (options != Qnil) { |
|
|
|
|
Check_Type(options, T_HASH); |
|
|
|
|
|
|
|
|
|
if (rb_funcall(options, rb_intern("key?"), 1, |
|
|
|
|
ID2SYM(rb_intern("default"))) == Qtrue) { |
|
|
|
|
Descriptor* msgdef = ruby_to_Descriptor(msgdef_rb); |
|
|
|
|
if (upb_msgdef_syntax((upb_msgdef*)msgdef->msgdef) == UPB_SYNTAX_PROTO3) { |
|
|
|
|
rb_raise(rb_eArgError, "Cannot set :default when using proto3 syntax."); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
FieldDescriptor* fielddef = ruby_to_FieldDescriptor(fielddef_rb); |
|
|
|
|
if (!upb_fielddef_haspresence((upb_fielddef*)fielddef->fielddef) || |
|
|
|
|
upb_fielddef_issubmsg((upb_fielddef*)fielddef->fielddef)) { |
|
|
|
|
rb_raise(rb_eArgError, "Cannot set :default on this kind of field."); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
rb_funcall(fielddef_rb, rb_intern("default="), 1, |
|
|
|
|
rb_hash_lookup(options, ID2SYM(rb_intern("default")))); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
rb_funcall(msgdef_rb, rb_intern("add_field"), 1, fielddef_rb); |
|
|
|
|
return fielddef_rb; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* MessageBuilderContext.optional(name, type, number, type_class = nil) |
|
|
|
|
* MessageBuilderContext.optional(name, type, number, type_class = nil, |
|
|
|
|
* options = nil) |
|
|
|
|
* |
|
|
|
|
* Defines a new optional field on this message type with the given type, tag |
|
|
|
|
* number, and type class (for message and enum fields). The type must be a Ruby |
|
|
|
@ -1259,23 +1587,26 @@ static VALUE msgdef_add_field(VALUE msgdef, |
|
|
|
|
*/ |
|
|
|
|
VALUE MessageBuilderContext_optional(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
DEFINE_SELF(MessageBuilderContext, self, _self); |
|
|
|
|
VALUE name, type, number, type_class; |
|
|
|
|
VALUE name, type, number; |
|
|
|
|
VALUE type_class, options = Qnil; |
|
|
|
|
|
|
|
|
|
if (argc < 3) { |
|
|
|
|
rb_raise(rb_eArgError, "Expected at least 3 arguments."); |
|
|
|
|
rb_scan_args(argc, argv, "32", &name, &type, &number, &type_class, &options); |
|
|
|
|
|
|
|
|
|
// Allow passing (name, type, number, options) or
|
|
|
|
|
// (name, type, number, type_class, options)
|
|
|
|
|
if (argc == 4 && RB_TYPE_P(type_class, T_HASH)) { |
|
|
|
|
options = type_class; |
|
|
|
|
type_class = Qnil; |
|
|
|
|
} |
|
|
|
|
name = argv[0]; |
|
|
|
|
type = argv[1]; |
|
|
|
|
number = argv[2]; |
|
|
|
|
type_class = (argc > 3) ? argv[3] : Qnil; |
|
|
|
|
|
|
|
|
|
return msgdef_add_field(self->descriptor, "optional", |
|
|
|
|
name, type, number, type_class); |
|
|
|
|
name, type, number, type_class, options); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* MessageBuilderContext.required(name, type, number, type_class = nil) |
|
|
|
|
* MessageBuilderContext.required(name, type, number, type_class = nil, |
|
|
|
|
* options = nil) |
|
|
|
|
* |
|
|
|
|
* Defines a new required field on this message type with the given type, tag |
|
|
|
|
* number, and type class (for message and enum fields). The type must be a Ruby |
|
|
|
@ -1288,18 +1619,20 @@ VALUE MessageBuilderContext_optional(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
*/ |
|
|
|
|
VALUE MessageBuilderContext_required(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
DEFINE_SELF(MessageBuilderContext, self, _self); |
|
|
|
|
VALUE name, type, number, type_class; |
|
|
|
|
VALUE name, type, number; |
|
|
|
|
VALUE type_class, options = Qnil; |
|
|
|
|
|
|
|
|
|
if (argc < 3) { |
|
|
|
|
rb_raise(rb_eArgError, "Expected at least 3 arguments."); |
|
|
|
|
rb_scan_args(argc, argv, "32", &name, &type, &number, &type_class, &options); |
|
|
|
|
|
|
|
|
|
// Allow passing (name, type, number, options) or
|
|
|
|
|
// (name, type, number, type_class, options)
|
|
|
|
|
if (argc == 4 && RB_TYPE_P(type_class, T_HASH)) { |
|
|
|
|
options = type_class; |
|
|
|
|
type_class = Qnil; |
|
|
|
|
} |
|
|
|
|
name = argv[0]; |
|
|
|
|
type = argv[1]; |
|
|
|
|
number = argv[2]; |
|
|
|
|
type_class = (argc > 3) ? argv[3] : Qnil; |
|
|
|
|
|
|
|
|
|
return msgdef_add_field(self->descriptor, "required", |
|
|
|
|
name, type, number, type_class); |
|
|
|
|
name, type, number, type_class, options); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
@ -1324,7 +1657,7 @@ VALUE MessageBuilderContext_repeated(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
type_class = (argc > 3) ? argv[3] : Qnil; |
|
|
|
|
|
|
|
|
|
return msgdef_add_field(self->descriptor, "repeated", |
|
|
|
|
name, type, number, type_class); |
|
|
|
|
name, type, number, type_class, Qnil); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
@ -1365,9 +1698,17 @@ VALUE MessageBuilderContext_map(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
"type."); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
Descriptor* descriptor = ruby_to_Descriptor(self->descriptor); |
|
|
|
|
if (upb_msgdef_syntax(descriptor->msgdef) == UPB_SYNTAX_PROTO2) { |
|
|
|
|
rb_raise(rb_eArgError, |
|
|
|
|
"Cannot add a native map field using proto2 syntax."); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Create a new message descriptor for the map entry message, and create a
|
|
|
|
|
// repeated submessage field here with that type.
|
|
|
|
|
mapentry_desc = rb_class_new_instance(0, NULL, cDescriptor); |
|
|
|
|
VALUE file_descriptor_rb = |
|
|
|
|
rb_funcall(self->descriptor, rb_intern("file_descriptor"), 0); |
|
|
|
|
mapentry_desc = rb_class_new_instance(1, &file_descriptor_rb, cDescriptor); |
|
|
|
|
mapentry_desc_name = rb_funcall(self->descriptor, rb_intern("name"), 0); |
|
|
|
|
mapentry_desc_name = rb_str_cat2(mapentry_desc_name, "_MapEntry_"); |
|
|
|
|
mapentry_desc_name = rb_str_cat2(mapentry_desc_name, |
|
|
|
@ -1410,8 +1751,8 @@ VALUE MessageBuilderContext_map(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
{ |
|
|
|
|
// Add the map-entry message type to the current builder, and use the type
|
|
|
|
|
// to create the map field itself.
|
|
|
|
|
Builder* builder_self = ruby_to_Builder(self->builder); |
|
|
|
|
rb_ary_push(builder_self->pending_list, mapentry_desc); |
|
|
|
|
Builder* builder = ruby_to_Builder(self->builder); |
|
|
|
|
rb_ary_push(builder->pending_list, mapentry_desc); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
{ |
|
|
|
@ -1514,7 +1855,8 @@ VALUE OneofBuilderContext_initialize(VALUE _self, |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* OneofBuilderContext.optional(name, type, number, type_class = nil) |
|
|
|
|
* OneofBuilderContext.optional(name, type, number, type_class = nil, |
|
|
|
|
* default_value = nil) |
|
|
|
|
* |
|
|
|
|
* Defines a new optional field in this oneof with the given type, tag number, |
|
|
|
|
* and type class (for message and enum fields). The type must be a Ruby symbol |
|
|
|
@ -1523,18 +1865,13 @@ VALUE OneofBuilderContext_initialize(VALUE _self, |
|
|
|
|
*/ |
|
|
|
|
VALUE OneofBuilderContext_optional(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
DEFINE_SELF(OneofBuilderContext, self, _self); |
|
|
|
|
VALUE name, type, number, type_class; |
|
|
|
|
VALUE name, type, number; |
|
|
|
|
VALUE type_class, options = Qnil; |
|
|
|
|
|
|
|
|
|
if (argc < 3) { |
|
|
|
|
rb_raise(rb_eArgError, "Expected at least 3 arguments."); |
|
|
|
|
} |
|
|
|
|
name = argv[0]; |
|
|
|
|
type = argv[1]; |
|
|
|
|
number = argv[2]; |
|
|
|
|
type_class = (argc > 3) ? argv[3] : Qnil; |
|
|
|
|
rb_scan_args(argc, argv, "32", &name, &type, &number, &type_class, &options); |
|
|
|
|
|
|
|
|
|
return msgdef_add_field(self->descriptor, "optional", |
|
|
|
|
name, type, number, type_class); |
|
|
|
|
name, type, number, type_class, options); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
@ -1604,6 +1941,112 @@ VALUE EnumBuilderContext_value(VALUE _self, VALUE name, VALUE number) { |
|
|
|
|
return enumdef_add_value(self->enumdesc, name, number); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
// FileBuilderContext.
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
DEFINE_CLASS(FileBuilderContext, |
|
|
|
|
"Google::Protobuf::Internal::FileBuilderContext"); |
|
|
|
|
|
|
|
|
|
void FileBuilderContext_mark(void* _self) { |
|
|
|
|
FileBuilderContext* self = _self; |
|
|
|
|
rb_gc_mark(self->pending_list); |
|
|
|
|
rb_gc_mark(self->file_descriptor); |
|
|
|
|
rb_gc_mark(self->builder); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void FileBuilderContext_free(void* _self) { |
|
|
|
|
FileBuilderContext* self = _self; |
|
|
|
|
xfree(self); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
VALUE FileBuilderContext_alloc(VALUE klass) { |
|
|
|
|
FileBuilderContext* self = ALLOC(FileBuilderContext); |
|
|
|
|
VALUE ret = TypedData_Wrap_Struct(klass, &_FileBuilderContext_type, self); |
|
|
|
|
self->pending_list = Qnil; |
|
|
|
|
self->file_descriptor = Qnil; |
|
|
|
|
self->builder = Qnil; |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void FileBuilderContext_register(VALUE module) { |
|
|
|
|
VALUE klass = rb_define_class_under(module, "FileBuilderContext", rb_cObject); |
|
|
|
|
rb_define_alloc_func(klass, FileBuilderContext_alloc); |
|
|
|
|
rb_define_method(klass, "initialize", FileBuilderContext_initialize, 2); |
|
|
|
|
rb_define_method(klass, "add_message", FileBuilderContext_add_message, 1); |
|
|
|
|
rb_define_method(klass, "add_enum", FileBuilderContext_add_enum, 1); |
|
|
|
|
rb_gc_register_address(&cFileBuilderContext); |
|
|
|
|
cFileBuilderContext = klass; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileBuilderContext.new(file_descriptor, builder) => context |
|
|
|
|
* |
|
|
|
|
* Create a new file builder context for the given file descriptor and |
|
|
|
|
* builder context. This class is intended to serve as a DSL context to be used |
|
|
|
|
* with #instance_eval. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileBuilderContext_initialize(VALUE _self, VALUE file_descriptor, |
|
|
|
|
VALUE builder) { |
|
|
|
|
DEFINE_SELF(FileBuilderContext, self, _self); |
|
|
|
|
self->pending_list = rb_ary_new(); |
|
|
|
|
self->file_descriptor = file_descriptor; |
|
|
|
|
self->builder = builder; |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileBuilderContext.add_message(name, &block) |
|
|
|
|
* |
|
|
|
|
* Creates a new, empty descriptor with the given name, and invokes the block in |
|
|
|
|
* the context of a MessageBuilderContext on that descriptor. The block can then |
|
|
|
|
* call, e.g., MessageBuilderContext#optional and MessageBuilderContext#repeated |
|
|
|
|
* methods to define the message fields. |
|
|
|
|
* |
|
|
|
|
* This is the recommended, idiomatic way to build message definitions. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileBuilderContext_add_message(VALUE _self, VALUE name) { |
|
|
|
|
DEFINE_SELF(FileBuilderContext, self, _self); |
|
|
|
|
VALUE msgdef = rb_class_new_instance(1, &self->file_descriptor, cDescriptor); |
|
|
|
|
VALUE args[2] = { msgdef, self->builder }; |
|
|
|
|
VALUE ctx = rb_class_new_instance(2, args, cMessageBuilderContext); |
|
|
|
|
VALUE block = rb_block_proc(); |
|
|
|
|
rb_funcall(msgdef, rb_intern("name="), 1, name); |
|
|
|
|
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block); |
|
|
|
|
rb_ary_push(self->pending_list, msgdef); |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* FileBuilderContext.add_enum(name, &block) |
|
|
|
|
* |
|
|
|
|
* Creates a new, empty enum descriptor with the given name, and invokes the |
|
|
|
|
* block in the context of an EnumBuilderContext on that descriptor. The block |
|
|
|
|
* can then call EnumBuilderContext#add_value to define the enum values. |
|
|
|
|
* |
|
|
|
|
* This is the recommended, idiomatic way to build enum definitions. |
|
|
|
|
*/ |
|
|
|
|
VALUE FileBuilderContext_add_enum(VALUE _self, VALUE name) { |
|
|
|
|
DEFINE_SELF(FileBuilderContext, self, _self); |
|
|
|
|
VALUE enumdef = |
|
|
|
|
rb_class_new_instance(1, &self->file_descriptor, cEnumDescriptor); |
|
|
|
|
VALUE ctx = rb_class_new_instance(1, &enumdef, cEnumBuilderContext); |
|
|
|
|
VALUE block = rb_block_proc(); |
|
|
|
|
rb_funcall(enumdef, rb_intern("name="), 1, name); |
|
|
|
|
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block); |
|
|
|
|
rb_ary_push(self->pending_list, enumdef); |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
VALUE FileBuilderContext_pending_descriptors(VALUE _self) { |
|
|
|
|
DEFINE_SELF(FileBuilderContext, self, _self); |
|
|
|
|
return self->pending_list; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
|
// Builder.
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
@ -1613,6 +2056,7 @@ DEFINE_CLASS(Builder, "Google::Protobuf::Internal::Builder"); |
|
|
|
|
void Builder_mark(void* _self) { |
|
|
|
|
Builder* self = _self; |
|
|
|
|
rb_gc_mark(self->pending_list); |
|
|
|
|
rb_gc_mark(self->default_file_descriptor); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void Builder_free(void* _self) { |
|
|
|
@ -1635,15 +2079,17 @@ VALUE Builder_alloc(VALUE klass) { |
|
|
|
|
klass, &_Builder_type, self); |
|
|
|
|
self->pending_list = Qnil; |
|
|
|
|
self->defs = NULL; |
|
|
|
|
self->default_file_descriptor = Qnil; |
|
|
|
|
return ret; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void Builder_register(VALUE module) { |
|
|
|
|
VALUE klass = rb_define_class_under(module, "Builder", rb_cObject); |
|
|
|
|
rb_define_alloc_func(klass, Builder_alloc); |
|
|
|
|
rb_define_alloc_func(klass, Builder_alloc);
|
|
|
|
|
rb_define_method(klass, "initialize", Builder_initialize, 0); |
|
|
|
|
rb_define_method(klass, "add_file", Builder_add_file, -1); |
|
|
|
|
rb_define_method(klass, "add_message", Builder_add_message, 1); |
|
|
|
|
rb_define_method(klass, "add_enum", Builder_add_enum, 1); |
|
|
|
|
rb_define_method(klass, "initialize", Builder_initialize, 0); |
|
|
|
|
rb_define_method(klass, "finalize_to_pool", Builder_finalize_to_pool, 1); |
|
|
|
|
rb_gc_register_address(&cBuilder); |
|
|
|
|
cBuilder = klass; |
|
|
|
@ -1651,13 +2097,40 @@ void Builder_register(VALUE module) { |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Builder.new(d) => builder |
|
|
|
|
* Builder.new |
|
|
|
|
* |
|
|
|
|
* Create a new message builder. |
|
|
|
|
* Initializes a new builder. |
|
|
|
|
*/ |
|
|
|
|
VALUE Builder_initialize(VALUE _self) { |
|
|
|
|
DEFINE_SELF(Builder, self, _self); |
|
|
|
|
self->pending_list = rb_ary_new(); |
|
|
|
|
VALUE file_name = Qnil; |
|
|
|
|
self->default_file_descriptor = |
|
|
|
|
rb_class_new_instance(1, &file_name, cFileDescriptor); |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* call-seq: |
|
|
|
|
* Builder.add_file(name, options = nil, &block) |
|
|
|
|
* |
|
|
|
|
* Creates a new, file descriptor with the given name and options and invokes |
|
|
|
|
* the block in the context of a FileBuilderContext on that descriptor. The |
|
|
|
|
* block can then call FileBuilderContext#add_message or |
|
|
|
|
* FileBuilderContext#add_enum to define new messages or enums, respectively. |
|
|
|
|
* |
|
|
|
|
* This is the recommended, idiomatic way to build file descriptors. |
|
|
|
|
*/ |
|
|
|
|
VALUE Builder_add_file(int argc, VALUE* argv, VALUE _self) { |
|
|
|
|
DEFINE_SELF(Builder, self, _self); |
|
|
|
|
VALUE file_descriptor = rb_class_new_instance(argc, argv, cFileDescriptor); |
|
|
|
|
VALUE args[2] = { file_descriptor, _self }; |
|
|
|
|
VALUE ctx = rb_class_new_instance(2, args, cFileBuilderContext); |
|
|
|
|
VALUE block = rb_block_proc(); |
|
|
|
|
rb_funcall_with_block(ctx, rb_intern("instance_eval"), 0, NULL, block); |
|
|
|
|
|
|
|
|
|
rb_ary_concat(self->pending_list, |
|
|
|
|
FileBuilderContext_pending_descriptors(ctx)); |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -1665,16 +2138,17 @@ VALUE Builder_initialize(VALUE _self) { |
|
|
|
|
* call-seq: |
|
|
|
|
* Builder.add_message(name, &block) |
|
|
|
|
* |
|
|
|
|
* Creates a new, empty descriptor with the given name, and invokes the block in |
|
|
|
|
* the context of a MessageBuilderContext on that descriptor. The block can then |
|
|
|
|
* call, e.g., MessageBuilderContext#optional and MessageBuilderContext#repeated |
|
|
|
|
* methods to define the message fields. |
|
|
|
|
* Old and deprecated way to create a new descriptor. |
|
|
|
|
* See FileBuilderContext.add_message for the recommended way. |
|
|
|
|
* |
|
|
|
|
* This is the recommended, idiomatic way to build message definitions. |
|
|
|
|
* Exists for backwards compatibility to allow building descriptor pool for |
|
|
|
|
* files generated by protoc which don't add messages within "add_file" block. |
|
|
|
|
* Descriptors created this way get assigned to a default empty FileDescriptor. |
|
|
|
|
*/ |
|
|
|
|
VALUE Builder_add_message(VALUE _self, VALUE name) { |
|
|
|
|
DEFINE_SELF(Builder, self, _self); |
|
|
|
|
VALUE msgdef = rb_class_new_instance(0, NULL, cDescriptor); |
|
|
|
|
VALUE msgdef = |
|
|
|
|
rb_class_new_instance(1, &self->default_file_descriptor, cDescriptor); |
|
|
|
|
VALUE args[2] = { msgdef, _self }; |
|
|
|
|
VALUE ctx = rb_class_new_instance(2, args, cMessageBuilderContext); |
|
|
|
|
VALUE block = rb_block_proc(); |
|
|
|
@ -1688,15 +2162,18 @@ VALUE Builder_add_message(VALUE _self, VALUE name) { |
|
|
|
|
* call-seq: |
|
|
|
|
* Builder.add_enum(name, &block) |
|
|
|
|
* |
|
|
|
|
* Creates a new, empty enum descriptor with the given name, and invokes the |
|
|
|
|
* block in the context of an EnumBuilderContext on that descriptor. The block |
|
|
|
|
* can then call EnumBuilderContext#add_value to define the enum values. |
|
|
|
|
* Old and deprecated way to create a new enum descriptor. |
|
|
|
|
* See FileBuilderContext.add_enum for the recommended way. |
|
|
|
|
* |
|
|
|
|
* This is the recommended, idiomatic way to build enum definitions. |
|
|
|
|
* Exists for backwards compatibility to allow building descriptor pool for |
|
|
|
|
* files generated by protoc which don't add enums within "add_file" block. |
|
|
|
|
* Enum descriptors created this way get assigned to a default empty |
|
|
|
|
* FileDescriptor. |
|
|
|
|
*/ |
|
|
|
|
VALUE Builder_add_enum(VALUE _self, VALUE name) { |
|
|
|
|
DEFINE_SELF(Builder, self, _self); |
|
|
|
|
VALUE enumdef = rb_class_new_instance(0, NULL, cEnumDescriptor); |
|
|
|
|
VALUE enumdef = |
|
|
|
|
rb_class_new_instance(1, &self->default_file_descriptor, cEnumDescriptor); |
|
|
|
|
VALUE ctx = rb_class_new_instance(1, &enumdef, cEnumBuilderContext); |
|
|
|
|
VALUE block = rb_block_proc(); |
|
|
|
|
rb_funcall(enumdef, rb_intern("name="), 1, name); |
|
|
|
@ -1705,7 +2182,7 @@ VALUE Builder_add_enum(VALUE _self, VALUE name) { |
|
|
|
|
return Qnil; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void validate_msgdef(const upb_msgdef* msgdef) { |
|
|
|
|
static void proto3_validate_msgdef(const upb_msgdef* msgdef) { |
|
|
|
|
// Verify that no required fields exist. proto3 does not support these.
|
|
|
|
|
upb_msg_field_iter it; |
|
|
|
|
for (upb_msg_field_begin(&it, msgdef); |
|
|
|
@ -1718,7 +2195,7 @@ static void validate_msgdef(const upb_msgdef* msgdef) { |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static void validate_enumdef(const upb_enumdef* enumdef) { |
|
|
|
|
static void proto3_validate_enumdef(const upb_enumdef* enumdef) { |
|
|
|
|
// Verify that an entry exists with integer value 0. (This is the default
|
|
|
|
|
// value.)
|
|
|
|
|
const char* lookup = upb_enumdef_iton(enumdef, 0); |
|
|
|
@ -1753,10 +2230,16 @@ VALUE Builder_finalize_to_pool(VALUE _self, VALUE pool_rb) { |
|
|
|
|
VALUE def_rb = rb_ary_entry(self->pending_list, i); |
|
|
|
|
if (CLASS_OF(def_rb) == cDescriptor) { |
|
|
|
|
self->defs[i] = (upb_def*)ruby_to_Descriptor(def_rb)->msgdef; |
|
|
|
|
validate_msgdef((const upb_msgdef*)self->defs[i]); |
|
|
|
|
|
|
|
|
|
if (upb_filedef_syntax(upb_def_file(self->defs[i])) == UPB_SYNTAX_PROTO3) { |
|
|
|
|
proto3_validate_msgdef((const upb_msgdef*)self->defs[i]); |
|
|
|
|
} |
|
|
|
|
} else if (CLASS_OF(def_rb) == cEnumDescriptor) { |
|
|
|
|
self->defs[i] = (upb_def*)ruby_to_EnumDescriptor(def_rb)->enumdef; |
|
|
|
|
validate_enumdef((const upb_enumdef*)self->defs[i]); |
|
|
|
|
|
|
|
|
|
if (upb_filedef_syntax(upb_def_file(self->defs[i])) == UPB_SYNTAX_PROTO3) { |
|
|
|
|
proto3_validate_enumdef((const upb_enumdef*)self->defs[i]); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|