aboutsummaryrefslogtreecommitdiffstats
path: root/yql/essentials/udfs/common/python/bindings/py_callable_ut.cpp
blob: 1c58d7b3714f3984e813416aff2f66ca0d8b06cd (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#include "ut3/py_test_engine.h"

#include <library/cpp/testing/unittest/registar.h>


using namespace NPython;

Y_UNIT_TEST_SUITE(TPyCallableTest) {
    struct TTestCallable: public NUdf::TBoxedValue {
        NUdf::TUnboxedValue Run(
                const NUdf::IValueBuilder* valueBuilder,
                const NUdf::TUnboxedValuePod* args) const override
        {
            Y_UNUSED(valueBuilder);
            return NUdf::TUnboxedValuePod(args[0].Get<ui32>() + 42);
        }
    };

    Y_UNIT_TEST(FromPyFunction) {
        TPythonTestEngine engine;
        const NUdf::IValueBuilder* vb = &engine.GetValueBuilder();

        engine.ToMiniKQL<char* (*)(char*, ui32)>(
                "def Test():\n"
                "   def test(str, count):\n"
                "       return str * count\n"
                "   return test",
                [vb](const NUdf::TUnboxedValuePod& value) {
                    UNIT_ASSERT(value);
                    UNIT_ASSERT(value.IsBoxed());
                    NUdf::TUnboxedValue args[2];
                    args[0] = vb->NewString("j");
                    args[1] = NUdf::TUnboxedValuePod((ui32) 5);
                    auto result = value.Run(vb, args);

                    UNIT_ASSERT(result);
                    UNIT_ASSERT(5 == result.AsStringRef().Size());
                    UNIT_ASSERT_STRINGS_EQUAL(result.AsStringRef(), "jjjjj");
                });
    }

    Y_UNIT_TEST(ToPython) {
        TPythonTestEngine engine;
        engine.ToPython<i32 (*)(i32)>(
                [](const TType* type, const NUdf::IValueBuilder& vb) {
                    Y_UNUSED(type); Y_UNUSED(vb);
                    return NUdf::TUnboxedValuePod(new TTestCallable);
                },
                "def Test(value):\n"
                "    assert type(value).__name__ == 'TCallable'\n"
                "    assert value.__call__ != None\n"
                "    assert value(-2) == 40\n"
                "    assert value(-1) == 41\n"
                "    assert value(0) == 42\n"
                "    assert value(1) == 43\n"
                "    assert value(2) == 44\n");
    }

    Y_UNIT_TEST(ToPythonAndBack) {
        struct TTestCallable: public NUdf::TBoxedValue {
            NUdf::TUnboxedValue Run(
                    const NUdf::IValueBuilder* valueBuilder,
                    const NUdf::TUnboxedValuePod* args) const override
            {
                Y_UNUSED(valueBuilder);
                return NUdf::TUnboxedValuePod(args[0].Get<ui32>() + 42);
            }
        };

        TPythonTestEngine engine;
        engine.ToPythonAndBack<i32 (*)(i32)>(
                [](const TType* type, const NUdf::IValueBuilder& vb) {
                    Y_UNUSED(type); Y_UNUSED(vb);
                    return NUdf::TUnboxedValuePod(new TTestCallable);
                },
                "def Test(value): return value",
                [](const NUdf::TUnboxedValuePod& value) {
                    UNIT_ASSERT(value);
                    UNIT_ASSERT(value.IsBoxed());
                    NUdf::TUnboxedValue arg = NUdf::TUnboxedValuePod((ui32) 5);
                    const auto result = value.Run(nullptr, &arg);

                    UNIT_ASSERT(result);
                    UNIT_ASSERT_VALUES_EQUAL(47, result.Get<ui32>());
                });
    }
}