jsonnet 0.1.1 → 0.2.0
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.
- checksums.yaml +4 -4
- data/.clang-format +11 -0
- data/ext/jsonnet/callbacks.c +241 -0
- data/ext/jsonnet/helpers.c +74 -0
- data/ext/jsonnet/jsonnet.c +4 -348
- data/ext/jsonnet/jsonnet_values.c +206 -0
- data/ext/jsonnet/ruby_jsonnet.h +39 -0
- data/ext/jsonnet/vm.c +351 -0
- data/lib/jsonnet/version.rb +1 -1
- data/lib/jsonnet/vm.rb +36 -1
- data/test/fixtures/jpath.libsonnet +3 -0
- data/test/test_vm.rb +218 -1
- metadata +10 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 683ffa45384b3ea064634f162faea0450cdbb2ac
|
4
|
+
data.tar.gz: 74539b2a0e88253d86fad6ed532bc48240a74df2
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d1a6f533c001414eacd2b776f2a7cbdb53e9dae4a0c53e2b83e43e90a53fe4d4eb888a96eb76623ad1007ed2d0fd45cc49737795cc6b5adeb5524922bf5d0d00
|
7
|
+
data.tar.gz: 175c210648ad1a5fe49d2a5c5846039d5ffb9c62146880925ead54dfd69ded603b0cbf5b8f14db04fa87133f3867052af1a374f185bbb9dedf81853cc3424c37
|
data/.clang-format
ADDED
@@ -0,0 +1,241 @@
|
|
1
|
+
#include <string.h>
|
2
|
+
|
3
|
+
#include <libjsonnet.h>
|
4
|
+
#include <ruby/ruby.h>
|
5
|
+
|
6
|
+
#include "ruby_jsonnet.h"
|
7
|
+
|
8
|
+
/* Magic prefix which distinguishes Ruby-level non-exception global escapes
|
9
|
+
* from other errors in Jsonnet.
|
10
|
+
* Any other errors in Jsonnet evaluation cannot contain this fragment of message.
|
11
|
+
*/
|
12
|
+
#define RUBYJSONNET_GLOBAL_ESCAPE_MAGIC "\x07\x03\x0c:rubytag:\x07\x03\x0c:"
|
13
|
+
|
14
|
+
/*
|
15
|
+
* callback support in VM
|
16
|
+
*/
|
17
|
+
|
18
|
+
static ID id_call;
|
19
|
+
|
20
|
+
/*
|
21
|
+
* Just invokes a callback with arguments, but also adapts the invocation to rb_protect.
|
22
|
+
* @param[in] args list of arguments whose first value is the callable object itself.
|
23
|
+
* @return result of the callback
|
24
|
+
*/
|
25
|
+
static VALUE
|
26
|
+
invoke_callback(VALUE args)
|
27
|
+
{
|
28
|
+
long len = RARRAY_LEN(args);
|
29
|
+
VALUE callback = rb_ary_entry(args, 0);
|
30
|
+
return rb_funcall2(callback, id_call, len - 1, RARRAY_PTR(args) + 1);
|
31
|
+
}
|
32
|
+
|
33
|
+
/*
|
34
|
+
* Handles long jump caught by rb_protect() in a callback function.
|
35
|
+
*
|
36
|
+
* Returns an error message which represents rb_errinfo().
|
37
|
+
* It is the caller's responsibility to return the message to the Jsonnet VM
|
38
|
+
* in the right way.
|
39
|
+
* Also the caller of the VM must handle evaluation failure caused by the
|
40
|
+
* error message.
|
41
|
+
* \sa rubyjsonnet_jump_tag
|
42
|
+
* \sa raise_eval_error
|
43
|
+
*/
|
44
|
+
static VALUE
|
45
|
+
rescue_callback(int state, const char *fmt, ...)
|
46
|
+
{
|
47
|
+
VALUE err = rb_errinfo();
|
48
|
+
if (rb_obj_is_kind_of(err, rb_eException)) {
|
49
|
+
VALUE msg = rb_protect(rubyjsonnet_format_exception, rb_errinfo(), NULL);
|
50
|
+
if (msg == Qnil) {
|
51
|
+
va_list ap;
|
52
|
+
va_start(ap, fmt);
|
53
|
+
msg = rb_vsprintf(fmt, ap);
|
54
|
+
va_end(ap);
|
55
|
+
}
|
56
|
+
rb_set_errinfo(Qnil);
|
57
|
+
return msg;
|
58
|
+
}
|
59
|
+
|
60
|
+
/*
|
61
|
+
* Other types of global escape.
|
62
|
+
* Here, we'll report the state as an evaluation error to let
|
63
|
+
* the Jsonnet VM clean up its internal resources.
|
64
|
+
* But we'll translate the error into an non-exception global escape
|
65
|
+
* in Ruby again in raise_eval_error().
|
66
|
+
*/
|
67
|
+
return rb_sprintf("%s%d%s", RUBYJSONNET_GLOBAL_ESCAPE_MAGIC, state,
|
68
|
+
RUBYJSONNET_GLOBAL_ESCAPE_MAGIC);
|
69
|
+
}
|
70
|
+
|
71
|
+
/*
|
72
|
+
* Tries to extract a jump tag from a string representation encoded by
|
73
|
+
* rescue_callback.
|
74
|
+
* @retval zero if \a exc_mesg is not such a string representation.
|
75
|
+
* @retval non-zero the extracted tag
|
76
|
+
*/
|
77
|
+
int
|
78
|
+
rubyjsonnet_jump_tag(const char *exc_mesg)
|
79
|
+
{
|
80
|
+
const char *tag;
|
81
|
+
#define JSONNET_RUNTIME_ERROR_PREFIX "RUNTIME ERROR: "
|
82
|
+
if (strncmp(exc_mesg, JSONNET_RUNTIME_ERROR_PREFIX, strlen(JSONNET_RUNTIME_ERROR_PREFIX))) {
|
83
|
+
return 0;
|
84
|
+
}
|
85
|
+
tag = strstr(exc_mesg + strlen(JSONNET_RUNTIME_ERROR_PREFIX), RUBYJSONNET_GLOBAL_ESCAPE_MAGIC);
|
86
|
+
if (tag) {
|
87
|
+
const char *const body = tag + strlen(RUBYJSONNET_GLOBAL_ESCAPE_MAGIC);
|
88
|
+
char *last;
|
89
|
+
long state = strtol(body, &last, 10);
|
90
|
+
if (!strncmp(last, RUBYJSONNET_GLOBAL_ESCAPE_MAGIC,
|
91
|
+
strlen(RUBYJSONNET_GLOBAL_ESCAPE_MAGIC)) &&
|
92
|
+
INT_MIN <= state && state <= INT_MAX) {
|
93
|
+
return (int)state;
|
94
|
+
}
|
95
|
+
}
|
96
|
+
return 0;
|
97
|
+
}
|
98
|
+
|
99
|
+
static char *
|
100
|
+
import_callback_entrypoint(void *ctx, const char *base, const char *rel, char **found_here,
|
101
|
+
int *success)
|
102
|
+
{
|
103
|
+
struct jsonnet_vm_wrap *const vm = (struct jsonnet_vm_wrap *)ctx;
|
104
|
+
int state;
|
105
|
+
VALUE result, args;
|
106
|
+
|
107
|
+
args = rb_ary_tmp_new(3);
|
108
|
+
|
109
|
+
rb_ary_push(args, vm->import_callback);
|
110
|
+
rb_ary_push(args, rb_enc_str_new_cstr(base, rb_filesystem_encoding()));
|
111
|
+
rb_ary_push(args, rb_enc_str_new_cstr(rel, rb_filesystem_encoding()));
|
112
|
+
result = rb_protect(invoke_callback, args, &state);
|
113
|
+
|
114
|
+
rb_ary_free(args);
|
115
|
+
|
116
|
+
if (state) {
|
117
|
+
VALUE msg = rescue_callback(state, "cannot import %s from %s", rel, base);
|
118
|
+
*success = 0;
|
119
|
+
return rubyjsonnet_str_to_cstr(vm->vm, msg);
|
120
|
+
}
|
121
|
+
|
122
|
+
result = rb_Array(result);
|
123
|
+
*success = 1;
|
124
|
+
*found_here = rubyjsonnet_str_to_cstr(vm->vm, rb_ary_entry(result, 1));
|
125
|
+
return rubyjsonnet_str_to_cstr(vm->vm, rb_ary_entry(result, 0));
|
126
|
+
}
|
127
|
+
|
128
|
+
/*
|
129
|
+
* Sets a custom way to resolve "import" expression.
|
130
|
+
* @param [#call] callback receives two parameters and returns two values.
|
131
|
+
* The first parameter "base" is a base directory to resolve
|
132
|
+
* "rel" from.
|
133
|
+
* The second parameter "rel" is an absolute or a relative
|
134
|
+
* path to the file to import.
|
135
|
+
* The first return value is the content of the imported file.
|
136
|
+
* The second return value is the resolved path of the imported file.
|
137
|
+
*/
|
138
|
+
static VALUE
|
139
|
+
vm_set_import_callback(VALUE self, VALUE callback)
|
140
|
+
{
|
141
|
+
struct jsonnet_vm_wrap *const vm = rubyjsonnet_obj_to_vm(self);
|
142
|
+
|
143
|
+
vm->import_callback = callback;
|
144
|
+
jsonnet_import_callback(vm->vm, import_callback_entrypoint, vm);
|
145
|
+
|
146
|
+
return callback;
|
147
|
+
}
|
148
|
+
|
149
|
+
/**
|
150
|
+
* Generic entrypoint of native callbacks which adapts callable objects in Ruby to \c
|
151
|
+
* JsonnetNativeCallback.
|
152
|
+
*
|
153
|
+
* @param[in] data pointer to a {\c struct native_callback_ctx}
|
154
|
+
* @param[in] argv NULL-terminated array of arguments
|
155
|
+
* @param[out] success set to 1 on success, or 0 if otherwise.
|
156
|
+
* @returns the result of the callback on success, an error message on failure.
|
157
|
+
*/
|
158
|
+
static struct JsonnetJsonValue *
|
159
|
+
native_callback_entrypoint(void *data, const struct JsonnetJsonValue *const *argv, int *success)
|
160
|
+
{
|
161
|
+
long i;
|
162
|
+
int state = 0;
|
163
|
+
|
164
|
+
struct native_callback_ctx *const ctx = (struct native_callback_ctx *)data;
|
165
|
+
struct JsonnetVm *const vm = rubyjsonnet_obj_to_vm(ctx->vm)->vm;
|
166
|
+
VALUE result, args = rb_ary_tmp_new(ctx->arity + 1);
|
167
|
+
|
168
|
+
rb_ary_push(args, ctx->callback);
|
169
|
+
for (i = 0; i < ctx->arity; ++i) {
|
170
|
+
rb_ary_push(args, rubyjsonnet_json_to_obj(vm, argv[i]));
|
171
|
+
}
|
172
|
+
|
173
|
+
result = rb_protect(invoke_callback, args, &state);
|
174
|
+
|
175
|
+
rb_ary_free(args);
|
176
|
+
|
177
|
+
if (state) {
|
178
|
+
VALUE msg = rescue_callback(state, "something wrong in %" PRIsVALUE, ctx->callback);
|
179
|
+
*success = 0;
|
180
|
+
return rubyjsonnet_obj_to_json(vm, msg, &state);
|
181
|
+
}
|
182
|
+
|
183
|
+
return rubyjsonnet_obj_to_json(vm, result, success);
|
184
|
+
}
|
185
|
+
|
186
|
+
/*
|
187
|
+
* Registers a native extension written in Ruby.
|
188
|
+
* @param callback [#call] a PURE callable object
|
189
|
+
* @param params [Array] names of the parameters of the function
|
190
|
+
*/
|
191
|
+
static VALUE
|
192
|
+
vm_register_native_callback(VALUE self, VALUE name, VALUE callback, VALUE params)
|
193
|
+
{
|
194
|
+
struct {
|
195
|
+
volatile VALUE store;
|
196
|
+
long len;
|
197
|
+
const char **buf;
|
198
|
+
} cstr_params;
|
199
|
+
struct native_callback_ctx *ctx;
|
200
|
+
struct jsonnet_vm_wrap *vm = rubyjsonnet_obj_to_vm(self);
|
201
|
+
long i;
|
202
|
+
|
203
|
+
name = rb_to_symbol(name);
|
204
|
+
rubyjsonnet_assert_asciicompat(name);
|
205
|
+
|
206
|
+
params = rb_Array(params);
|
207
|
+
cstr_params.len = RARRAY_LEN(params);
|
208
|
+
cstr_params.buf = (const char **)rb_alloc_tmp_buffer(
|
209
|
+
&cstr_params.store, sizeof(const char *const) * (cstr_params.len + 1));
|
210
|
+
for (i = 0; i < cstr_params.len; ++i) {
|
211
|
+
const VALUE pname = rb_to_symbol(RARRAY_AREF(params, i));
|
212
|
+
rubyjsonnet_assert_asciicompat(pname);
|
213
|
+
cstr_params.buf[i] = rb_id2name(RB_SYM2ID(pname));
|
214
|
+
}
|
215
|
+
cstr_params.buf[cstr_params.len] = NULL;
|
216
|
+
|
217
|
+
ctx = RB_ALLOC_N(struct native_callback_ctx, 1);
|
218
|
+
ctx->callback = callback;
|
219
|
+
ctx->arity = cstr_params.len;
|
220
|
+
ctx->vm = self;
|
221
|
+
jsonnet_native_callback(vm->vm, rb_id2name(RB_SYM2ID(name)), native_callback_entrypoint, ctx,
|
222
|
+
cstr_params.buf);
|
223
|
+
|
224
|
+
rb_free_tmp_buffer(&cstr_params.store);
|
225
|
+
|
226
|
+
RB_REALLOC_N(vm->native_callbacks.contexts, struct native_callback_ctx *,
|
227
|
+
vm->native_callbacks.len + 1);
|
228
|
+
vm->native_callbacks.contexts[vm->native_callbacks.len] = ctx;
|
229
|
+
vm->native_callbacks.len++;
|
230
|
+
|
231
|
+
return name;
|
232
|
+
}
|
233
|
+
|
234
|
+
void
|
235
|
+
rubyjsonnet_init_callbacks(VALUE cVM)
|
236
|
+
{
|
237
|
+
id_call = rb_intern("call");
|
238
|
+
|
239
|
+
rb_define_method(cVM, "import_callback=", vm_set_import_callback, 1);
|
240
|
+
rb_define_private_method(cVM, "register_native_callback", vm_register_native_callback, 3);
|
241
|
+
}
|
@@ -0,0 +1,74 @@
|
|
1
|
+
#include <string.h>
|
2
|
+
|
3
|
+
#include <libjsonnet.h>
|
4
|
+
#include <ruby/ruby.h>
|
5
|
+
#include <ruby/encoding.h>
|
6
|
+
|
7
|
+
#include "ruby_jsonnet.h"
|
8
|
+
|
9
|
+
static ID id_message;
|
10
|
+
|
11
|
+
/*
|
12
|
+
* Indicates that the encoding of the given string is not allowed by
|
13
|
+
* the C++ implementation of Jsonnet.
|
14
|
+
*/
|
15
|
+
static VALUE eUnsupportedEncodingError;
|
16
|
+
|
17
|
+
/**
|
18
|
+
* Asserts that the given string-like object is in an
|
19
|
+
* ASCII-compatible encoding.
|
20
|
+
*
|
21
|
+
* @param[in] str a String-like object
|
22
|
+
* @throw UnsupportedEncodingError on assertion failure.
|
23
|
+
*/
|
24
|
+
rb_encoding *
|
25
|
+
rubyjsonnet_assert_asciicompat(VALUE str)
|
26
|
+
{
|
27
|
+
rb_encoding *enc = rb_enc_get(str);
|
28
|
+
if (!rb_enc_asciicompat(enc)) {
|
29
|
+
rb_raise(eUnsupportedEncodingError, "jsonnet encoding must be ASCII-compatible but got %s",
|
30
|
+
rb_enc_name(enc));
|
31
|
+
}
|
32
|
+
return enc;
|
33
|
+
}
|
34
|
+
|
35
|
+
/**
|
36
|
+
* Allocates a C string whose content is equal to \c str with jsonnet_realloc.
|
37
|
+
*/
|
38
|
+
char *
|
39
|
+
rubyjsonnet_str_to_cstr(struct JsonnetVm *vm, VALUE str)
|
40
|
+
{
|
41
|
+
const char *const cstr = StringValueCStr(str);
|
42
|
+
char *const buf = jsonnet_realloc(vm, NULL, strlen(cstr));
|
43
|
+
strcpy(buf, cstr);
|
44
|
+
return buf;
|
45
|
+
}
|
46
|
+
|
47
|
+
/**
|
48
|
+
* @return a human readable string which contains the class name of the
|
49
|
+
* exception and its message. It might be nil on failure
|
50
|
+
*/
|
51
|
+
VALUE
|
52
|
+
rubyjsonnet_format_exception(VALUE exc)
|
53
|
+
{
|
54
|
+
VALUE name = rb_class_name(rb_obj_class(exc));
|
55
|
+
VALUE msg = rb_funcall(exc, id_message, 0);
|
56
|
+
if (RB_TYPE_P(name, RUBY_T_STRING) && rb_str_strlen(name)) {
|
57
|
+
if (RB_TYPE_P(msg, RUBY_T_STRING) && rb_str_strlen(msg)) {
|
58
|
+
return rb_str_concat(rb_str_cat_cstr(name, " : "), msg);
|
59
|
+
} else {
|
60
|
+
return name;
|
61
|
+
}
|
62
|
+
} else if (RB_TYPE_P(msg, RUBY_T_STRING) && rb_str_strlen(msg)) {
|
63
|
+
return msg;
|
64
|
+
}
|
65
|
+
return Qnil;
|
66
|
+
}
|
67
|
+
|
68
|
+
void
|
69
|
+
rubyjsonnet_init_helpers(VALUE mJsonnet)
|
70
|
+
{
|
71
|
+
id_message = rb_intern("message");
|
72
|
+
eUnsupportedEncodingError =
|
73
|
+
rb_define_class_under(mJsonnet, "UnsupportedEncodingError", rb_eEncodingError);
|
74
|
+
}
|
data/ext/jsonnet/jsonnet.c
CHANGED
@@ -1,127 +1,8 @@
|
|
1
|
-
#include <
|
1
|
+
#include <libjsonnet.h>
|
2
2
|
#include <ruby/ruby.h>
|
3
|
-
#include <ruby/intern.h>
|
4
3
|
#include <ruby/encoding.h>
|
5
|
-
#include <libjsonnet.h>
|
6
4
|
|
7
|
-
|
8
|
-
static ID id_message;
|
9
|
-
|
10
|
-
/*
|
11
|
-
* Jsonnet evaluator
|
12
|
-
*/
|
13
|
-
static VALUE cVM;
|
14
|
-
static VALUE eEvaluationError;
|
15
|
-
/**
|
16
|
-
* Indicates that the encoding of the given string is not allowed by
|
17
|
-
* the C++ implementation of Jsonnet.
|
18
|
-
*/
|
19
|
-
static VALUE eUnsupportedEncodingError;
|
20
|
-
|
21
|
-
struct jsonnet_vm_wrap {
|
22
|
-
struct JsonnetVm *vm;
|
23
|
-
VALUE callback;
|
24
|
-
};
|
25
|
-
|
26
|
-
static void vm_free(void *ptr);
|
27
|
-
static void vm_mark(void *ptr);
|
28
|
-
|
29
|
-
static const rb_data_type_t jsonnet_vm_type = {
|
30
|
-
"JsonnetVm",
|
31
|
-
{
|
32
|
-
/* dmark = */ vm_mark,
|
33
|
-
/* dfree = */ vm_free,
|
34
|
-
/* dsize = */ 0,
|
35
|
-
},
|
36
|
-
/* parent = */ 0,
|
37
|
-
/* data = */ 0,
|
38
|
-
/* flags = */ RUBY_TYPED_FREE_IMMEDIATELY
|
39
|
-
};
|
40
|
-
|
41
|
-
/**
|
42
|
-
* raises an EvaluationError whose message is \c msg.
|
43
|
-
* @param[in] vm a JsonnetVM
|
44
|
-
* @param[in] msg must be a NUL-terminated string returned by \c vm.
|
45
|
-
* @return never returns
|
46
|
-
* @throw EvaluationError
|
47
|
-
*/
|
48
|
-
static void
|
49
|
-
raise_eval_error(struct JsonnetVm *vm, char *msg, rb_encoding *enc)
|
50
|
-
{
|
51
|
-
VALUE ex = rb_exc_new3(eEvaluationError, rb_enc_str_new_cstr(msg, enc));
|
52
|
-
jsonnet_realloc(vm, msg, 0);
|
53
|
-
rb_exc_raise(ex);
|
54
|
-
}
|
55
|
-
|
56
|
-
/**
|
57
|
-
* Returns a String whose contents is equal to \c json.
|
58
|
-
* It automatically frees \c json just after constructing the return value.
|
59
|
-
*
|
60
|
-
* @param[in] vm a JsonnetVM
|
61
|
-
* @param[in] json must be a NUL-terminated string returned by \c vm.
|
62
|
-
* @return Ruby string equal to \c json.
|
63
|
-
*/
|
64
|
-
static VALUE
|
65
|
-
str_new_json(struct JsonnetVm *vm, char *json, rb_encoding *enc)
|
66
|
-
{
|
67
|
-
VALUE str = rb_enc_str_new_cstr(json, enc);
|
68
|
-
jsonnet_realloc(vm, json, 0);
|
69
|
-
return str;
|
70
|
-
}
|
71
|
-
|
72
|
-
/**
|
73
|
-
* Returns a Hash, whose keys are file names in the multi-mode of Jsonnet,
|
74
|
-
* and whose values are corresponding JSON values.
|
75
|
-
* It automatically frees \c json just after constructing the return value.
|
76
|
-
*
|
77
|
-
* @param[in] vm a JsonnetVM
|
78
|
-
* @param[in] buf NUL-separated and double-NUL-terminated sequence of strings returned by \c vm.
|
79
|
-
* @return Hash
|
80
|
-
*/
|
81
|
-
static VALUE
|
82
|
-
fileset_new(struct JsonnetVm *vm, char *buf, rb_encoding *enc)
|
83
|
-
{
|
84
|
-
VALUE fileset = rb_hash_new();
|
85
|
-
char *ptr, *json;
|
86
|
-
for (ptr = buf; *ptr; ptr = json + strlen(json)+1) {
|
87
|
-
json = ptr + strlen(ptr) + 1;
|
88
|
-
if (!*json) {
|
89
|
-
VALUE ex = rb_exc_new3(
|
90
|
-
eEvaluationError,
|
91
|
-
rb_enc_sprintf(enc, "output file %s without body", ptr));
|
92
|
-
jsonnet_realloc(vm, buf, 0);
|
93
|
-
rb_exc_raise(ex);
|
94
|
-
}
|
95
|
-
|
96
|
-
rb_hash_aset(fileset, rb_enc_str_new_cstr(ptr, enc), rb_enc_str_new_cstr(json, enc));
|
97
|
-
}
|
98
|
-
jsonnet_realloc(vm, buf, 0);
|
99
|
-
return fileset;
|
100
|
-
}
|
101
|
-
|
102
|
-
/**
|
103
|
-
* Allocates a C string whose content is equal to \c str with jsonnet_realloc.
|
104
|
-
*/
|
105
|
-
static char *
|
106
|
-
str_jsonnet_cstr(struct JsonnetVm *vm, VALUE str)
|
107
|
-
{
|
108
|
-
const char *const cstr = StringValueCStr(str);
|
109
|
-
char *const buf = jsonnet_realloc(vm, NULL, strlen(cstr));
|
110
|
-
strcpy(buf, cstr);
|
111
|
-
return buf;
|
112
|
-
}
|
113
|
-
|
114
|
-
static rb_encoding *
|
115
|
-
enc_assert_asciicompat(VALUE str) {
|
116
|
-
rb_encoding *enc = rb_enc_get(str);
|
117
|
-
if (!rb_enc_asciicompat(enc)) {
|
118
|
-
rb_raise(
|
119
|
-
eUnsupportedEncodingError,
|
120
|
-
"jsonnet encoding must be ASCII-compatible but got %s",
|
121
|
-
rb_enc_name(enc));
|
122
|
-
}
|
123
|
-
return enc;
|
124
|
-
}
|
5
|
+
#include "ruby_jsonnet.h"
|
125
6
|
|
126
7
|
/*
|
127
8
|
* call-seq:
|
@@ -135,237 +16,12 @@ jw_s_version(VALUE mod)
|
|
135
16
|
return rb_usascii_str_new_cstr(jsonnet_version());
|
136
17
|
}
|
137
18
|
|
138
|
-
static VALUE
|
139
|
-
vm_s_new(int argc, const VALUE *argv, VALUE klass)
|
140
|
-
{
|
141
|
-
struct jsonnet_vm_wrap *vm;
|
142
|
-
VALUE self = TypedData_Make_Struct(cVM, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
143
|
-
vm->vm = jsonnet_make();
|
144
|
-
vm->callback = Qnil;
|
145
|
-
rb_obj_call_init(self, argc, argv);
|
146
|
-
return self;
|
147
|
-
}
|
148
|
-
|
149
|
-
static void
|
150
|
-
vm_free(void *ptr)
|
151
|
-
{
|
152
|
-
struct jsonnet_vm_wrap *vm = (struct jsonnet_vm_wrap*)ptr;
|
153
|
-
jsonnet_destroy(vm->vm);
|
154
|
-
REALLOC_N(vm, struct jsonnet_vm_wrap, 0);
|
155
|
-
}
|
156
|
-
|
157
|
-
static void
|
158
|
-
vm_mark(void *ptr)
|
159
|
-
{
|
160
|
-
struct jsonnet_vm_wrap *vm = (struct jsonnet_vm_wrap*)ptr;
|
161
|
-
rb_gc_mark(vm->callback);
|
162
|
-
}
|
163
|
-
|
164
|
-
static VALUE
|
165
|
-
vm_evaluate_file(VALUE self, VALUE fname, VALUE encoding, VALUE multi_p)
|
166
|
-
{
|
167
|
-
struct jsonnet_vm_wrap *vm;
|
168
|
-
int error;
|
169
|
-
char *result;
|
170
|
-
rb_encoding *const enc = rb_to_encoding(encoding);
|
171
|
-
|
172
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
173
|
-
FilePathValue(fname);
|
174
|
-
if (RTEST(multi_p)) {
|
175
|
-
result = jsonnet_evaluate_file_multi(vm->vm, StringValueCStr(fname), &error);
|
176
|
-
}
|
177
|
-
else {
|
178
|
-
result = jsonnet_evaluate_file(vm->vm, StringValueCStr(fname), &error);
|
179
|
-
}
|
180
|
-
|
181
|
-
if (error) {
|
182
|
-
raise_eval_error(vm->vm, result, rb_enc_get(fname));
|
183
|
-
}
|
184
|
-
return RTEST(multi_p) ? fileset_new(vm->vm, result, enc) : str_new_json(vm->vm, result, enc);
|
185
|
-
}
|
186
|
-
|
187
|
-
static VALUE
|
188
|
-
vm_evaluate(VALUE self, VALUE snippet, VALUE fname, VALUE multi_p)
|
189
|
-
{
|
190
|
-
struct jsonnet_vm_wrap *vm;
|
191
|
-
int error;
|
192
|
-
char *result;
|
193
|
-
|
194
|
-
rb_encoding *enc = enc_assert_asciicompat(StringValue(snippet));
|
195
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
196
|
-
FilePathValue(fname);
|
197
|
-
if (RTEST(multi_p)) {
|
198
|
-
result = jsonnet_evaluate_snippet_multi(
|
199
|
-
vm->vm,
|
200
|
-
StringValueCStr(fname), StringValueCStr(snippet), &error);
|
201
|
-
}
|
202
|
-
else {
|
203
|
-
result = jsonnet_evaluate_snippet(
|
204
|
-
vm->vm,
|
205
|
-
StringValueCStr(fname), StringValueCStr(snippet), &error);
|
206
|
-
}
|
207
|
-
|
208
|
-
if (error) {
|
209
|
-
raise_eval_error(vm->vm, result, rb_enc_get(fname));
|
210
|
-
}
|
211
|
-
return RTEST(multi_p) ? fileset_new(vm->vm, result, enc) : str_new_json(vm->vm, result, enc);
|
212
|
-
}
|
213
|
-
|
214
|
-
static VALUE
|
215
|
-
import_callback_thunk0(VALUE args)
|
216
|
-
{
|
217
|
-
VALUE callback = rb_ary_entry(args, 0);
|
218
|
-
return rb_funcall(callback, id_call, 2, rb_ary_entry(args, 1), rb_ary_entry(args, 2));
|
219
|
-
}
|
220
|
-
|
221
|
-
static char *
|
222
|
-
import_callback_thunk(void *ctx, const char *base, const char *rel, char **found_here, int *success)
|
223
|
-
{
|
224
|
-
struct jsonnet_vm_wrap *const vm = (struct jsonnet_vm_wrap*)ctx;
|
225
|
-
int state;
|
226
|
-
VALUE result, args;
|
227
|
-
|
228
|
-
args = rb_ary_tmp_new(3);
|
229
|
-
rb_ary_push(args, vm->callback);
|
230
|
-
rb_ary_push(args, rb_enc_str_new_cstr(base, rb_filesystem_encoding()));
|
231
|
-
rb_ary_push(args, rb_enc_str_new_cstr(rel, rb_filesystem_encoding()));
|
232
|
-
result = rb_protect(import_callback_thunk0, args, &state);
|
233
|
-
|
234
|
-
if (state) {
|
235
|
-
VALUE err = rb_errinfo();
|
236
|
-
VALUE msg, name;
|
237
|
-
|
238
|
-
rb_set_errinfo(Qnil);
|
239
|
-
name = rb_class_name(rb_obj_class(err));
|
240
|
-
msg = rb_funcall(err, id_message, 0);
|
241
|
-
if (rb_str_strlen(name)) {
|
242
|
-
if (rb_str_strlen(msg)) {
|
243
|
-
msg = rb_str_concat(rb_str_cat_cstr(name, " : "), msg);
|
244
|
-
} else {
|
245
|
-
msg = name;
|
246
|
-
}
|
247
|
-
} else if (!rb_str_strlen(msg)) {
|
248
|
-
msg = rb_sprintf("cannot import %s from %s", rel, base);
|
249
|
-
}
|
250
|
-
*success = 0;
|
251
|
-
return str_jsonnet_cstr(vm->vm, msg);
|
252
|
-
}
|
253
|
-
|
254
|
-
result = rb_Array(result);
|
255
|
-
*success = 1;
|
256
|
-
*found_here = str_jsonnet_cstr(vm->vm, rb_ary_entry(result, 1));
|
257
|
-
return str_jsonnet_cstr(vm->vm, rb_ary_entry(result, 0));
|
258
|
-
}
|
259
|
-
|
260
|
-
/**
|
261
|
-
* Sets a custom way to resolve "import" expression.
|
262
|
-
* @param [#call] callback receives two parameters and returns two values.
|
263
|
-
* The first parameter "base" is a base directory to resolve
|
264
|
-
* "rel" from.
|
265
|
-
* The second parameter "rel" is an absolute or a relative
|
266
|
-
* path to the file to import.
|
267
|
-
* The first return value is the content of the imported file.
|
268
|
-
* The second return value is the resolved path of the imported file.
|
269
|
-
*/
|
270
|
-
static VALUE
|
271
|
-
vm_set_import_callback(VALUE self, VALUE callback)
|
272
|
-
{
|
273
|
-
struct jsonnet_vm_wrap *vm;
|
274
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
275
|
-
|
276
|
-
vm->callback = callback;
|
277
|
-
jsonnet_import_callback(vm->vm, import_callback_thunk, vm);
|
278
|
-
return callback;
|
279
|
-
}
|
280
|
-
|
281
|
-
/*
|
282
|
-
* Binds an external variable to a value.
|
283
|
-
* @param [String] key name of the variable
|
284
|
-
* @param [String] val the value
|
285
|
-
*/
|
286
|
-
static VALUE
|
287
|
-
vm_ext_var(VALUE self, VALUE key, VALUE val)
|
288
|
-
{
|
289
|
-
struct jsonnet_vm_wrap *vm;
|
290
|
-
|
291
|
-
enc_assert_asciicompat(StringValue(key));
|
292
|
-
enc_assert_asciicompat(StringValue(val));
|
293
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
294
|
-
jsonnet_ext_var(vm->vm, StringValueCStr(key), StringValueCStr(val));
|
295
|
-
return Qnil;
|
296
|
-
}
|
297
|
-
|
298
|
-
static VALUE
|
299
|
-
vm_set_max_stack(VALUE self, VALUE val)
|
300
|
-
{
|
301
|
-
struct jsonnet_vm_wrap *vm;
|
302
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
303
|
-
jsonnet_max_stack(vm->vm, NUM2UINT(val));
|
304
|
-
return Qnil;
|
305
|
-
}
|
306
|
-
|
307
|
-
static VALUE
|
308
|
-
vm_set_gc_min_objects(VALUE self, VALUE val)
|
309
|
-
{
|
310
|
-
struct jsonnet_vm_wrap *vm;
|
311
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
312
|
-
jsonnet_gc_min_objects(vm->vm, NUM2UINT(val));
|
313
|
-
return Qnil;
|
314
|
-
}
|
315
|
-
|
316
|
-
static VALUE
|
317
|
-
vm_set_gc_growth_trigger(VALUE self, VALUE val)
|
318
|
-
{
|
319
|
-
struct jsonnet_vm_wrap *vm;
|
320
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
321
|
-
jsonnet_gc_growth_trigger(vm->vm, NUM2DBL(val));
|
322
|
-
return Qnil;
|
323
|
-
}
|
324
|
-
|
325
|
-
/*
|
326
|
-
* Let #evaluate and #evaluate_file return a raw String instead of JSON-encoded string if val is true
|
327
|
-
* @param [Boolean] val
|
328
|
-
*/
|
329
|
-
static VALUE
|
330
|
-
vm_set_string_output(VALUE self, VALUE val)
|
331
|
-
{
|
332
|
-
struct jsonnet_vm_wrap *vm;
|
333
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
334
|
-
jsonnet_string_output(vm->vm, RTEST(val));
|
335
|
-
return Qnil;
|
336
|
-
}
|
337
|
-
|
338
|
-
static VALUE
|
339
|
-
vm_set_max_trace(VALUE self, VALUE val)
|
340
|
-
{
|
341
|
-
struct jsonnet_vm_wrap *vm;
|
342
|
-
TypedData_Get_Struct(self, struct jsonnet_vm_wrap, &jsonnet_vm_type, vm);
|
343
|
-
jsonnet_max_trace(vm->vm, NUM2UINT(val));
|
344
|
-
return Qnil;
|
345
|
-
}
|
346
|
-
|
347
19
|
void
|
348
20
|
Init_jsonnet_wrap(void)
|
349
21
|
{
|
350
|
-
id_call = rb_intern("call");
|
351
|
-
id_message = rb_intern("message");
|
352
|
-
|
353
22
|
VALUE mJsonnet = rb_define_module("Jsonnet");
|
354
23
|
rb_define_singleton_method(mJsonnet, "libversion", jw_s_version, 0);
|
355
24
|
|
356
|
-
|
357
|
-
|
358
|
-
rb_define_private_method(cVM, "eval_file", vm_evaluate_file, 3);
|
359
|
-
rb_define_private_method(cVM, "eval_snippet", vm_evaluate, 3);
|
360
|
-
rb_define_method(cVM, "ext_var", vm_ext_var, 2);
|
361
|
-
rb_define_method(cVM, "max_stack=", vm_set_max_stack, 1);
|
362
|
-
rb_define_method(cVM, "gc_min_objects=", vm_set_gc_min_objects, 1);
|
363
|
-
rb_define_method(cVM, "gc_growth_trigger=", vm_set_gc_growth_trigger, 1);
|
364
|
-
rb_define_method(cVM, "string_output=", vm_set_string_output, 1);
|
365
|
-
rb_define_method(cVM, "max_trace=", vm_set_max_trace, 1);
|
366
|
-
rb_define_method(cVM, "import_callback=", vm_set_import_callback, 1);
|
367
|
-
|
368
|
-
eEvaluationError = rb_define_class_under(mJsonnet, "EvaluationError", rb_eRuntimeError);
|
369
|
-
eUnsupportedEncodingError =
|
370
|
-
rb_define_class_under(mJsonnet, "UnsupportedEncodingError", rb_eEncodingError);
|
25
|
+
rubyjsonnet_init_helpers(mJsonnet);
|
26
|
+
rubyjsonnet_init_vm(mJsonnet);
|
371
27
|
}
|