#include "nix_api_store.h" #include "nix_api_store_internal.h" #include "nix_api_util.h" #include "nix_api_util_internal.h" #include "nix_api_expr.h" #include "nix_api_value.h" #include "tests/nix_api_expr.hh" #include "tests/string_callback.hh" #include "gmock/gmock.h" #include #include namespace nixC { TEST_F(nix_api_expr_test, nix_value_set_get_int) { ASSERT_EQ(0, nix_get_int(ctx, nullptr)); ASSERT_DEATH(nix_get_int(ctx, value), ""); int myInt = 1; nix_init_int(ctx, value, myInt); ASSERT_EQ(myInt, nix_get_int(ctx, value)); ASSERT_STREQ("an integer", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_INT, nix_get_type(ctx, value)); } TEST_F(nix_api_expr_test, nix_value_set_get_float) { ASSERT_FLOAT_EQ(0.0, nix_get_float(ctx, nullptr)); ASSERT_DEATH(nix_get_float(ctx, value), ""); float myDouble = 1.0; nix_init_float(ctx, value, myDouble); ASSERT_FLOAT_EQ(myDouble, nix_get_float(ctx, value)); ASSERT_STREQ("a float", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_FLOAT, nix_get_type(ctx, value)); } TEST_F(nix_api_expr_test, nix_value_set_get_bool) { ASSERT_EQ(false, nix_get_bool(ctx, nullptr)); ASSERT_DEATH(nix_get_bool(ctx, value), ""); bool myBool = true; nix_init_bool(ctx, value, myBool); ASSERT_EQ(myBool, nix_get_bool(ctx, value)); ASSERT_STREQ("a Boolean", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_BOOL, nix_get_type(ctx, value)); } TEST_F(nix_api_expr_test, nix_value_set_get_string) { std::string string_value; ASSERT_EQ(NIX_ERR_UNKNOWN, nix_get_string(ctx, nullptr, OBSERVE_STRING(string_value))); ASSERT_DEATH(nix_get_string(ctx, value, OBSERVE_STRING(string_value)), ""); const char * myString = "some string"; nix_init_string(ctx, value, myString); nix_get_string(ctx, value, OBSERVE_STRING(string_value)); ASSERT_STREQ(myString, string_value.c_str()); ASSERT_STREQ("a string", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_STRING, nix_get_type(ctx, value)); } TEST_F(nix_api_expr_test, nix_value_set_get_null) { ASSERT_DEATH(nix_get_typename(ctx, value), ""); nix_init_null(ctx, value); ASSERT_STREQ("null", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_NULL, nix_get_type(ctx, value)); } TEST_F(nix_api_expr_test, nix_value_set_get_path) { ASSERT_EQ(nullptr, nix_get_path_string(ctx, nullptr)); ASSERT_DEATH(nix_get_path_string(ctx, value), ""); const char * p = "/nix/store/40s0qmrfb45vlh6610rk29ym318dswdr-myname"; nix_init_path_string(ctx, state, value, p); ASSERT_STREQ(p, nix_get_path_string(ctx, value)); ASSERT_STREQ("a path", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_PATH, nix_get_type(ctx, value)); } TEST_F(nix_api_expr_test, nix_build_and_init_list) { ASSERT_EQ(nullptr, nix_get_list_byidx(ctx, nullptr, state, 0)); ASSERT_EQ(0, nix_get_list_size(ctx, nullptr)); ASSERT_DEATH(nix_get_list_byidx(ctx, value, state, 0), ""); ASSERT_DEATH(nix_get_list_size(ctx, value), ""); int size = 10; ListBuilder * builder = nix_make_list_builder(ctx, state, size); Value * intValue = nix_alloc_value(ctx, state); nix_init_int(ctx, intValue, 42); nix_list_builder_insert(ctx, builder, 0, intValue); nix_make_list(ctx, builder, value); nix_list_builder_free(builder); ASSERT_EQ(42, nix_get_int(ctx, nix_get_list_byidx(ctx, value, state, 0))); ASSERT_EQ(nullptr, nix_get_list_byidx(ctx, value, state, 1)); ASSERT_EQ(10, nix_get_list_size(ctx, value)); ASSERT_STREQ("a list", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_LIST, nix_get_type(ctx, value)); // Clean up nix_gc_decref(ctx, intValue); } TEST_F(nix_api_expr_test, nix_build_and_init_attr) { ASSERT_EQ(nullptr, nix_get_attr_byname(ctx, nullptr, state, 0)); ASSERT_EQ(nullptr, nix_get_attr_byidx(ctx, nullptr, state, 0, nullptr)); ASSERT_EQ(nullptr, nix_get_attr_name_byidx(ctx, nullptr, state, 0)); ASSERT_EQ(0, nix_get_attrs_size(ctx, nullptr)); ASSERT_EQ(false, nix_has_attr_byname(ctx, nullptr, state, "no-value")); ASSERT_DEATH(nix_get_attr_byname(ctx, value, state, 0), ""); ASSERT_DEATH(nix_get_attr_byidx(ctx, value, state, 0, nullptr), ""); ASSERT_DEATH(nix_get_attr_name_byidx(ctx, value, state, 0), ""); ASSERT_DEATH(nix_get_attrs_size(ctx, value), ""); ASSERT_DEATH(nix_has_attr_byname(ctx, value, state, "no-value"), ""); int size = 10; const char ** out_name = (const char **) malloc(sizeof(char *)); BindingsBuilder * builder = nix_make_bindings_builder(ctx, state, size); Value * intValue = nix_alloc_value(ctx, state); nix_init_int(ctx, intValue, 42); Value * stringValue = nix_alloc_value(ctx, state); nix_init_string(ctx, stringValue, "foo"); nix_bindings_builder_insert(ctx, builder, "a", intValue); nix_bindings_builder_insert(ctx, builder, "b", stringValue); nix_make_attrs(ctx, value, builder); nix_bindings_builder_free(builder); ASSERT_EQ(2, nix_get_attrs_size(ctx, value)); Value * out_value = nix_get_attr_byname(ctx, value, state, "a"); ASSERT_EQ(42, nix_get_int(ctx, out_value)); nix_gc_decref(ctx, out_value); out_value = nix_get_attr_byidx(ctx, value, state, 0, out_name); ASSERT_EQ(42, nix_get_int(ctx, out_value)); ASSERT_STREQ("a", *out_name); nix_gc_decref(ctx, out_value); ASSERT_STREQ("a", nix_get_attr_name_byidx(ctx, value, state, 0)); ASSERT_EQ(true, nix_has_attr_byname(ctx, value, state, "b")); ASSERT_EQ(false, nix_has_attr_byname(ctx, value, state, "no-value")); out_value = nix_get_attr_byname(ctx, value, state, "b"); std::string string_value; nix_get_string(ctx, out_value, OBSERVE_STRING(string_value)); ASSERT_STREQ("foo", string_value.c_str()); nix_gc_decref(nullptr, out_value); out_value = nix_get_attr_byidx(ctx, value, state, 1, out_name); nix_get_string(ctx, out_value, OBSERVE_STRING(string_value)); ASSERT_STREQ("foo", string_value.c_str()); ASSERT_STREQ("b", *out_name); nix_gc_decref(nullptr, out_value); ASSERT_STREQ("b", nix_get_attr_name_byidx(ctx, value, state, 1)); ASSERT_STREQ("a set", nix_get_typename(ctx, value)); ASSERT_EQ(NIX_TYPE_ATTRS, nix_get_type(ctx, value)); // Clean up nix_gc_decref(ctx, intValue); nix_gc_decref(ctx, stringValue); free(out_name); } TEST_F(nix_api_expr_test, nix_value_init) { // Setup // two = 2; // f = a: a * a; Value * two = nix_alloc_value(ctx, state); nix_init_int(ctx, two, 2); Value * f = nix_alloc_value(ctx, state); nix_expr_eval_from_string( ctx, state, R"( a: a * a )", "", f); // Test // r = f two; Value * r = nix_alloc_value(ctx, state); nix_init_apply(ctx, r, f, two); assert_ctx_ok(); ValueType t = nix_get_type(ctx, r); assert_ctx_ok(); ASSERT_EQ(t, NIX_TYPE_THUNK); nix_value_force(ctx, state, r); t = nix_get_type(ctx, r); assert_ctx_ok(); ASSERT_EQ(t, NIX_TYPE_INT); int n = nix_get_int(ctx, r); assert_ctx_ok(); ASSERT_EQ(n, 4); // Clean up nix_gc_decref(ctx, two); nix_gc_decref(ctx, f); nix_gc_decref(ctx, r); } TEST_F(nix_api_expr_test, nix_value_init_apply_error) { Value * some_string = nix_alloc_value(ctx, state); nix_init_string(ctx, some_string, "some string"); assert_ctx_ok(); Value * v = nix_alloc_value(ctx, state); nix_init_apply(ctx, v, some_string, some_string); assert_ctx_ok(); // All ok. Call has not been evaluated yet. // Evaluate it nix_value_force(ctx, state, v); ASSERT_EQ(ctx->last_err_code, NIX_ERR_NIX_ERROR); ASSERT_THAT(ctx->last_err.value(), testing::HasSubstr("attempt to call something which is not a function but")); // Clean up nix_gc_decref(ctx, some_string); nix_gc_decref(ctx, v); } TEST_F(nix_api_expr_test, nix_value_init_apply_lazy_arg) { // f is a lazy function: it does not evaluate its argument before returning its return value // g is a helper to produce e // e is a thunk that throws an exception // // r = f e // r should not throw an exception, because e is not evaluated Value * f = nix_alloc_value(ctx, state); nix_expr_eval_from_string( ctx, state, R"( a: { foo = a; } )", "", f); assert_ctx_ok(); Value * e = nix_alloc_value(ctx, state); { Value * g = nix_alloc_value(ctx, state); nix_expr_eval_from_string( ctx, state, R"( _ignore: throw "error message for test case nix_value_init_apply_lazy_arg" )", "", g); assert_ctx_ok(); nix_init_apply(ctx, e, g, g); assert_ctx_ok(); nix_gc_decref(ctx, g); } Value * r = nix_alloc_value(ctx, state); nix_init_apply(ctx, r, f, e); assert_ctx_ok(); nix_value_force(ctx, state, r); assert_ctx_ok(); auto n = nix_get_attrs_size(ctx, r); assert_ctx_ok(); ASSERT_EQ(1, n); // nix_get_attr_byname isn't lazy (it could have been) so it will throw the exception Value * foo = nix_get_attr_byname(ctx, r, state, "foo"); ASSERT_EQ(nullptr, foo); ASSERT_THAT(ctx->last_err.value(), testing::HasSubstr("error message for test case nix_value_init_apply_lazy_arg")); // Clean up nix_gc_decref(ctx, f); nix_gc_decref(ctx, e); } }