marshal.c: Support dump and load of a Hash with the ruby2_keywords flag
It is useful for a program that dumps and load arguments (like drb). In future, they should deal with both positional arguments and keyword ones explicitly, but until ruby2_keywords is deprecated, it is good to support the flag in marshal. The implementation is similar to String's encoding; it is dumped as a hidden instance variable. [Feature #16501]
This commit is contained in:
parent
c98c492578
commit
b23fd59cbb
Notes:
git
2020-01-17 17:20:47 +09:00
48
marshal.c
48
marshal.c
@ -94,7 +94,7 @@ shortlen(size_t len, BDIGIT *ds)
|
|||||||
static ID s_dump, s_load, s_mdump, s_mload;
|
static ID s_dump, s_load, s_mdump, s_mload;
|
||||||
static ID s_dump_data, s_load_data, s_alloc, s_call;
|
static ID s_dump_data, s_load_data, s_alloc, s_call;
|
||||||
static ID s_getbyte, s_read, s_write, s_binmode;
|
static ID s_getbyte, s_read, s_write, s_binmode;
|
||||||
static ID s_encoding_short;
|
static ID s_encoding_short, s_ruby2_keywords_flag;
|
||||||
|
|
||||||
#define name_s_dump "_dump"
|
#define name_s_dump "_dump"
|
||||||
#define name_s_load "_load"
|
#define name_s_load "_load"
|
||||||
@ -109,6 +109,7 @@ static ID s_encoding_short;
|
|||||||
#define name_s_write "write"
|
#define name_s_write "write"
|
||||||
#define name_s_binmode "binmode"
|
#define name_s_binmode "binmode"
|
||||||
#define name_s_encoding_short "E"
|
#define name_s_encoding_short "E"
|
||||||
|
#define name_s_ruby2_keywords_flag "K"
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
VALUE newclass;
|
VALUE newclass;
|
||||||
@ -568,7 +569,7 @@ w_uclass(VALUE obj, VALUE super, struct dump_arg *arg)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define to_be_skipped_id(id) (id == rb_id_encoding() || id == s_encoding_short || !rb_id2str(id))
|
#define to_be_skipped_id(id) (id == rb_id_encoding() || id == s_encoding_short || id == s_ruby2_keywords_flag || !rb_id2str(id))
|
||||||
|
|
||||||
struct w_ivar_arg {
|
struct w_ivar_arg {
|
||||||
struct dump_call_arg *dump;
|
struct dump_call_arg *dump;
|
||||||
@ -588,6 +589,10 @@ w_obj_each(st_data_t key, st_data_t val, st_data_t a)
|
|||||||
rb_warn("instance variable `"name_s_encoding_short"' on class %"PRIsVALUE" is not dumped",
|
rb_warn("instance variable `"name_s_encoding_short"' on class %"PRIsVALUE" is not dumped",
|
||||||
CLASS_OF(arg->obj));
|
CLASS_OF(arg->obj));
|
||||||
}
|
}
|
||||||
|
if (id == s_ruby2_keywords_flag) {
|
||||||
|
rb_warn("instance variable `"name_s_ruby2_keywords_flag"' on class %"PRIsVALUE" is not dumped",
|
||||||
|
CLASS_OF(arg->obj));
|
||||||
|
}
|
||||||
return ST_CONTINUE;
|
return ST_CONTINUE;
|
||||||
}
|
}
|
||||||
if (!ivarg->num_ivar) {
|
if (!ivarg->num_ivar) {
|
||||||
@ -665,6 +670,7 @@ has_ivars(VALUE obj, VALUE encname, VALUE *ivobj)
|
|||||||
{
|
{
|
||||||
st_index_t enc = !NIL_P(encname);
|
st_index_t enc = !NIL_P(encname);
|
||||||
st_index_t num = 0;
|
st_index_t num = 0;
|
||||||
|
st_index_t ruby2_keywords_flag = 0;
|
||||||
|
|
||||||
if (SPECIAL_CONST_P(obj)) goto generic;
|
if (SPECIAL_CONST_P(obj)) goto generic;
|
||||||
switch (BUILTIN_TYPE(obj)) {
|
switch (BUILTIN_TYPE(obj)) {
|
||||||
@ -672,13 +678,16 @@ has_ivars(VALUE obj, VALUE encname, VALUE *ivobj)
|
|||||||
case T_CLASS:
|
case T_CLASS:
|
||||||
case T_MODULE:
|
case T_MODULE:
|
||||||
break; /* counted elsewhere */
|
break; /* counted elsewhere */
|
||||||
|
case T_HASH:
|
||||||
|
ruby2_keywords_flag = RHASH(obj)->basic.flags & RHASH_PASS_AS_KEYWORDS ? 1 : 0;
|
||||||
|
/* fall through */
|
||||||
default:
|
default:
|
||||||
generic:
|
generic:
|
||||||
rb_ivar_foreach(obj, obj_count_ivars, (st_data_t)&num);
|
rb_ivar_foreach(obj, obj_count_ivars, (st_data_t)&num);
|
||||||
if (num) *ivobj = obj;
|
if (ruby2_keywords_flag || num) *ivobj = obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
return num + enc;
|
return num + enc + ruby2_keywords_flag;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -698,7 +707,14 @@ w_ivar(st_index_t num, VALUE ivobj, VALUE encname, struct dump_call_arg *arg)
|
|||||||
{
|
{
|
||||||
w_long(num, arg->arg);
|
w_long(num, arg->arg);
|
||||||
num -= w_encoding(encname, arg);
|
num -= w_encoding(encname, arg);
|
||||||
if (ivobj != Qundef) {
|
if (RB_TYPE_P(ivobj, T_HASH) && (RHASH(ivobj)->basic.flags & RHASH_PASS_AS_KEYWORDS)) {
|
||||||
|
int limit = arg->limit;
|
||||||
|
if (limit >= 0) ++limit;
|
||||||
|
w_symbol(ID2SYM(s_ruby2_keywords_flag), arg->arg);
|
||||||
|
w_object(Qtrue, arg->arg, limit);
|
||||||
|
num--;
|
||||||
|
}
|
||||||
|
if (ivobj != Qundef && num) {
|
||||||
w_ivar_each(ivobj, num, arg);
|
w_ivar_each(ivobj, num, arg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1399,6 +1415,19 @@ sym2encidx(VALUE sym, VALUE val)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
ruby2_keywords_flag_check(VALUE sym)
|
||||||
|
{
|
||||||
|
const char *p;
|
||||||
|
long l;
|
||||||
|
RSTRING_GETMEM(sym, p, l);
|
||||||
|
if (l <= 0) return 0;
|
||||||
|
if (name_equal(name_s_ruby2_keywords_flag, rb_strlen_lit(name_s_ruby2_keywords_flag), p, 1)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static VALUE
|
static VALUE
|
||||||
r_symlink(struct load_arg *arg)
|
r_symlink(struct load_arg *arg)
|
||||||
{
|
{
|
||||||
@ -1552,6 +1581,14 @@ r_ivar(VALUE obj, int *has_encoding, struct load_arg *arg)
|
|||||||
}
|
}
|
||||||
if (has_encoding) *has_encoding = TRUE;
|
if (has_encoding) *has_encoding = TRUE;
|
||||||
}
|
}
|
||||||
|
else if (ruby2_keywords_flag_check(sym)) {
|
||||||
|
if (RB_TYPE_P(obj, T_HASH)) {
|
||||||
|
RHASH(obj)->basic.flags |= RHASH_PASS_AS_KEYWORDS;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
rb_raise(rb_eArgError, "ruby2_keywords flag is given but %"PRIsVALUE" is not a Hash", obj);
|
||||||
|
}
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
rb_ivar_set(obj, rb_intern_str(sym), val);
|
rb_ivar_set(obj, rb_intern_str(sym), val);
|
||||||
}
|
}
|
||||||
@ -2299,6 +2336,7 @@ Init_marshal(void)
|
|||||||
set_id(s_write);
|
set_id(s_write);
|
||||||
set_id(s_binmode);
|
set_id(s_binmode);
|
||||||
set_id(s_encoding_short);
|
set_id(s_encoding_short);
|
||||||
|
set_id(s_ruby2_keywords_flag);
|
||||||
|
|
||||||
rb_define_module_function(rb_mMarshal, "dump", marshal_dump, -1);
|
rb_define_module_function(rb_mMarshal, "dump", marshal_dump, -1);
|
||||||
rb_define_module_function(rb_mMarshal, "load", marshal_load, -1);
|
rb_define_module_function(rb_mMarshal, "load", marshal_load, -1);
|
||||||
|
@ -754,4 +754,18 @@ class TestMarshal < Test::Unit::TestCase
|
|||||||
Marshal.dump(obj)
|
Marshal.dump(obj)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
ruby2_keywords def ruby2_keywords_hash(*a)
|
||||||
|
a.last
|
||||||
|
end
|
||||||
|
|
||||||
|
def ruby2_keywords_test(key: 1)
|
||||||
|
key
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_marshal_with_ruby2_keywords_hash
|
||||||
|
flagged_hash = ruby2_keywords_hash(key: 42)
|
||||||
|
hash = Marshal.load(Marshal.dump(flagged_hash))
|
||||||
|
assert_equal(42, ruby2_keywords_test(*[hash]))
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
Loading…
x
Reference in New Issue
Block a user