1 | /* -*- c-basic-offset: 2 -*- */ |
2 | /* |
3 | Copyright(C) 2013-2015 Brazil |
4 | |
5 | This library is free software; you can redistribute it and/or |
6 | modify it under the terms of the GNU Lesser General Public |
7 | License version 2.1 as published by the Free Software Foundation. |
8 | |
9 | This library is distributed in the hope that it will be useful, |
10 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
11 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
12 | Lesser General Public License for more details. |
13 | |
14 | You should have received a copy of the GNU Lesser General Public |
15 | License along with this library; if not, write to the Free Software |
16 | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
17 | */ |
18 | |
19 | #include "../grn_ctx_impl.h" |
20 | #include "../grn_db.h" |
21 | #include <string.h> |
22 | |
23 | #ifdef GRN_WITH_MRUBY |
24 | #include <mruby.h> |
25 | #include <mruby/class.h> |
26 | #include <mruby/data.h> |
27 | #include <mruby/string.h> |
28 | |
29 | #include "mrb_converter.h" |
30 | #include "mrb_bulk.h" |
31 | |
32 | void |
33 | grn_mrb_value_to_raw_data_buffer_init(mrb_state *mrb, |
34 | grn_mrb_value_to_raw_data_buffer *buffer) |
35 | { |
36 | GRN_VOID_INIT(&(buffer->from)); |
37 | GRN_VOID_INIT(&(buffer->to)); |
38 | } |
39 | |
40 | void |
41 | grn_mrb_value_to_raw_data_buffer_fin(mrb_state *mrb, |
42 | grn_mrb_value_to_raw_data_buffer *buffer) |
43 | { |
44 | grn_ctx *ctx = (grn_ctx *)mrb->ud; |
45 | |
46 | GRN_OBJ_FIN(ctx, &(buffer->from)); |
47 | GRN_OBJ_FIN(ctx, &(buffer->to)); |
48 | } |
49 | |
50 | void |
51 | grn_mrb_value_to_raw_data(mrb_state *mrb, |
52 | const char *context, |
53 | mrb_value mrb_value_, |
54 | grn_id domain_id, |
55 | grn_mrb_value_to_raw_data_buffer *buffer, |
56 | void **raw_value, |
57 | unsigned int *raw_value_size) |
58 | { |
59 | grn_ctx *ctx = (grn_ctx *)mrb->ud; |
60 | enum mrb_vtype mrb_value_type; |
61 | grn_bool try_cast = GRN_FALSE; |
62 | grn_obj *from_bulk = NULL; |
63 | |
64 | if (mrb_nil_p(mrb_value_)) { |
65 | *raw_value = NULL; |
66 | *raw_value_size = 0; |
67 | return; |
68 | } |
69 | |
70 | mrb_value_type = mrb_type(mrb_value_); |
71 | |
72 | switch (mrb_value_type) { |
73 | case MRB_TT_STRING : |
74 | switch (domain_id) { |
75 | case GRN_DB_SHORT_TEXT : |
76 | case GRN_DB_TEXT : |
77 | case GRN_DB_LONG_TEXT : |
78 | *raw_value = RSTRING_PTR(mrb_value_); |
79 | *raw_value_size = RSTRING_LEN(mrb_value_); |
80 | break; |
81 | default : |
82 | try_cast = GRN_TRUE; |
83 | break; |
84 | } |
85 | break; |
86 | default : |
87 | { |
88 | struct RClass *klass; |
89 | grn_mrb_data *data = &(ctx->impl->mrb); |
90 | |
91 | klass = mrb_class(mrb, mrb_value_); |
92 | if (domain_id == GRN_DB_TIME && |
93 | klass == data->builtin.time_class) { |
94 | mrb_value mrb_sec; |
95 | mrb_value mrb_usec; |
96 | |
97 | mrb_sec = mrb_funcall(mrb, mrb_value_, "to_i" , 0); |
98 | mrb_usec = mrb_funcall(mrb, mrb_value_, "usec" , 0); |
99 | buffer->value.time_value = GRN_TIME_PACK(mrb_fixnum(mrb_sec), |
100 | mrb_fixnum(mrb_usec)); |
101 | *raw_value = &(buffer->value.time_value); |
102 | *raw_value_size = sizeof(buffer->value.time_value); |
103 | } else { |
104 | try_cast = GRN_TRUE; |
105 | if (mrb_value_type == MRB_TT_DATA && |
106 | klass == mrb_class_get_under(mrb, data->module, "Bulk" )) { |
107 | from_bulk = DATA_PTR(mrb_value_); |
108 | } |
109 | } |
110 | } |
111 | break; |
112 | } |
113 | |
114 | if (!try_cast) { |
115 | return; |
116 | } |
117 | |
118 | if (!from_bulk) { |
119 | from_bulk = &(buffer->from); |
120 | grn_mrb_value_to_bulk(mrb, mrb_value_, from_bulk); |
121 | } |
122 | if (!grn_mrb_bulk_cast(mrb, from_bulk, &(buffer->to), domain_id)) { |
123 | grn_obj *domain; |
124 | char domain_name[GRN_TABLE_MAX_KEY_SIZE]; |
125 | int domain_name_size; |
126 | |
127 | domain = grn_ctx_at(ctx, domain_id); |
128 | domain_name_size = grn_obj_name(ctx, domain, domain_name, |
129 | GRN_TABLE_MAX_KEY_SIZE); |
130 | mrb_raisef(mrb, E_ARGUMENT_ERROR, |
131 | "%S: failed to convert to %S: %S" , |
132 | mrb_str_new_static(mrb, context, strlen(context)), |
133 | mrb_str_new_static(mrb, domain_name, domain_name_size), |
134 | mrb_funcall(mrb, mrb_value_, "inspect" , 0)); |
135 | } |
136 | *raw_value = GRN_BULK_HEAD(&(buffer->to)); |
137 | *raw_value_size = GRN_BULK_VSIZE(&(buffer->to)); |
138 | } |
139 | |
140 | mrb_value |
141 | grn_mrb_value_from_raw_data(mrb_state *mrb, |
142 | grn_id domain, |
143 | void *raw_value, |
144 | unsigned int raw_value_size) |
145 | { |
146 | grn_ctx *ctx = (grn_ctx *)mrb->ud; |
147 | mrb_value mrb_value_; |
148 | |
149 | switch (domain) { |
150 | case GRN_DB_INT32 : |
151 | if (raw_value_size == 0) { |
152 | mrb_value_ = mrb_fixnum_value(0); |
153 | } else { |
154 | int32_t value; |
155 | value = *((int32_t *)raw_value); |
156 | mrb_value_ = mrb_fixnum_value(value); |
157 | } |
158 | break; |
159 | case GRN_DB_SHORT_TEXT : |
160 | case GRN_DB_TEXT : |
161 | case GRN_DB_LONG_TEXT : |
162 | mrb_value_ = mrb_str_new(mrb, |
163 | raw_value, |
164 | raw_value_size); |
165 | break; |
166 | default : |
167 | { |
168 | grn_obj *domain_object; |
169 | #define MESSAGE_SIZE 4096 |
170 | char message[MESSAGE_SIZE]; |
171 | char domain_name[GRN_TABLE_MAX_KEY_SIZE]; |
172 | int domain_name_size; |
173 | |
174 | domain_object = grn_ctx_at(ctx, domain); |
175 | if (domain_object) { |
176 | domain_name_size = grn_obj_name(ctx, domain_object, |
177 | domain_name, GRN_TABLE_MAX_KEY_SIZE); |
178 | grn_obj_unlink(ctx, domain_object); |
179 | } else { |
180 | grn_strcpy(domain_name, GRN_TABLE_MAX_KEY_SIZE, "unknown" ); |
181 | domain_name_size = strlen(domain_name); |
182 | } |
183 | grn_snprintf(message, MESSAGE_SIZE, MESSAGE_SIZE, |
184 | "unsupported raw value type: <%d>(%.*s)" , |
185 | domain, |
186 | domain_name_size, |
187 | domain_name); |
188 | mrb_raise(mrb, E_RANGE_ERROR, message); |
189 | } |
190 | #undef MESSAGE_SIZE |
191 | break; |
192 | } |
193 | |
194 | return mrb_value_; |
195 | } |
196 | |
197 | struct RClass * |
198 | grn_mrb_class_from_grn_obj(mrb_state *mrb, grn_obj *object) |
199 | { |
200 | grn_ctx *ctx = (grn_ctx *)mrb->ud; |
201 | grn_mrb_data *data; |
202 | struct RClass *klass = NULL; |
203 | |
204 | data = &(ctx->impl->mrb); |
205 | switch (object->header.type) { |
206 | case GRN_BULK : |
207 | klass = mrb_class_get_under(mrb, data->module, "Bulk" ); |
208 | break; |
209 | case GRN_PTR : |
210 | klass = mrb_class_get_under(mrb, data->module, "Pointer" ); |
211 | break; |
212 | case GRN_ACCESSOR : |
213 | klass = mrb_class_get_under(mrb, data->module, "Accessor" ); |
214 | break; |
215 | case GRN_COLUMN_FIX_SIZE : |
216 | klass = mrb_class_get_under(mrb, data->module, "FixedSizeColumn" ); |
217 | break; |
218 | case GRN_COLUMN_VAR_SIZE : |
219 | klass = mrb_class_get_under(mrb, data->module, "VariableSizeColumn" ); |
220 | break; |
221 | case GRN_COLUMN_INDEX : |
222 | klass = mrb_class_get_under(mrb, data->module, "IndexColumn" ); |
223 | break; |
224 | case GRN_TYPE : |
225 | klass = mrb_class_get_under(mrb, data->module, "Type" ); |
226 | break; |
227 | case GRN_PROC : |
228 | klass = mrb_class_get_under(mrb, data->module, "Procedure" ); |
229 | break; |
230 | case GRN_EXPR : |
231 | klass = mrb_class_get_under(mrb, data->module, "Expression" ); |
232 | break; |
233 | case GRN_TABLE_NO_KEY : |
234 | klass = mrb_class_get_under(mrb, data->module, "Array" ); |
235 | break; |
236 | case GRN_TABLE_HASH_KEY : |
237 | klass = mrb_class_get_under(mrb, data->module, "HashTable" ); |
238 | break; |
239 | case GRN_TABLE_PAT_KEY : |
240 | klass = mrb_class_get_under(mrb, data->module, "PatriciaTrie" ); |
241 | break; |
242 | case GRN_TABLE_DAT_KEY : |
243 | klass = mrb_class_get_under(mrb, data->module, "DoubleArrayTrie" ); |
244 | break; |
245 | case GRN_DB : |
246 | klass = mrb_class_get_under(mrb, data->module, "Database" ); |
247 | break; |
248 | case GRN_VOID : |
249 | klass = mrb_class_get_under(mrb, data->module, "Void" ); |
250 | break; |
251 | default : |
252 | break; |
253 | } |
254 | |
255 | if (!klass) { |
256 | #define BUFFER_SIZE 1024 |
257 | char buffer[BUFFER_SIZE]; |
258 | grn_snprintf(buffer, BUFFER_SIZE, BUFFER_SIZE, |
259 | "can't find class for object type: %#x" , object->header.type); |
260 | mrb_raise(mrb, E_ARGUMENT_ERROR, buffer); |
261 | #undef BUFFER_SIZE |
262 | } |
263 | |
264 | return klass; |
265 | } |
266 | |
267 | mrb_value |
268 | grn_mrb_value_from_grn_obj(mrb_state *mrb, grn_obj *object) |
269 | { |
270 | struct RClass *mrb_class; |
271 | mrb_value mrb_new_arguments[1]; |
272 | mrb_value mrb_object; |
273 | |
274 | if (!object) { |
275 | return mrb_nil_value(); |
276 | } |
277 | |
278 | mrb_class = grn_mrb_class_from_grn_obj(mrb, object); |
279 | mrb_new_arguments[0] = mrb_cptr_value(mrb, object); |
280 | mrb_object = mrb_obj_new(mrb, mrb_class, 1, mrb_new_arguments); |
281 | return mrb_object; |
282 | } |
283 | |
284 | grn_id |
285 | grn_mrb_class_to_type(mrb_state *mrb, struct RClass *klass) |
286 | { |
287 | grn_ctx *ctx = (grn_ctx *)mrb->ud; |
288 | grn_id type = GRN_DB_VOID; |
289 | |
290 | if (klass == mrb->nil_class) { |
291 | type = GRN_DB_VOID; |
292 | } else if (klass == mrb->true_class || |
293 | klass == mrb->false_class) { |
294 | type = GRN_DB_BOOL; |
295 | } else if (klass == mrb->symbol_class) { |
296 | type = GRN_DB_TEXT; |
297 | } else if (klass == mrb->fixnum_class) { |
298 | type = GRN_DB_INT64; |
299 | } else if (klass == mrb->float_class) { |
300 | type = GRN_DB_FLOAT; |
301 | } else if (klass == mrb->string_class) { |
302 | type = GRN_DB_TEXT; |
303 | } else if (klass == ctx->impl->mrb.builtin.time_class) { |
304 | type = GRN_DB_TIME; |
305 | } else { |
306 | mrb_raisef(mrb, E_ARGUMENT_ERROR, |
307 | "unsupported class: %S" , mrb_obj_value(klass)); |
308 | } |
309 | |
310 | return type; |
311 | } |
312 | |
313 | static mrb_value |
314 | mrb_grn_converter_class_convert(mrb_state *mrb, mrb_value klass) |
315 | { |
316 | grn_ctx *ctx = (grn_ctx *)mrb->ud; |
317 | grn_obj *from = &(ctx->impl->mrb.buffer.from); |
318 | grn_obj *to = &(ctx->impl->mrb.buffer.to); |
319 | mrb_value mrb_from; |
320 | mrb_value mrb_to_class; |
321 | grn_id to_type; |
322 | |
323 | mrb_get_args(mrb, "oC" , &mrb_from, &mrb_to_class); |
324 | |
325 | grn_mrb_value_to_bulk(mrb, mrb_from, from); |
326 | to_type = grn_mrb_class_to_type(mrb, mrb_class_ptr(mrb_to_class)); |
327 | if (!grn_mrb_bulk_cast(mrb, from, to, to_type)) { |
328 | mrb_raisef(mrb, E_ARGUMENT_ERROR, |
329 | "failed to convert to %S: %S" , |
330 | mrb_to_class, |
331 | mrb_from); |
332 | } |
333 | |
334 | return grn_mrb_value_from_bulk(mrb, to); |
335 | } |
336 | |
337 | void |
338 | grn_mrb_converter_init(grn_ctx *ctx) |
339 | { |
340 | grn_mrb_data *data = &(ctx->impl->mrb); |
341 | mrb_state *mrb = data->state; |
342 | struct RClass *module; |
343 | |
344 | module = mrb_define_module_under(mrb, data->module, "Converter" ); |
345 | |
346 | mrb_define_class_method(mrb, module, "convert" , |
347 | mrb_grn_converter_class_convert, |
348 | MRB_ARGS_REQ(2)); |
349 | } |
350 | #endif |
351 | |