File: command.cpp

package info (click to toggle)
pytango 10.1.4-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 8,304 kB
  • sloc: python: 27,795; cpp: 16,150; sql: 252; sh: 152; makefile: 43
file content (90 lines) | stat: -rw-r--r-- 3,159 bytes parent folder | download
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
88
89
90
/*
 * SPDX-FileCopyrightText: All Contributors to the PyTango project
 *
 * SPDX-License-Identifier: LGPL-3.0-or-later
 */

#include "common_header.h"
#include "function_call_macros.h"

#include "convertors/type_casters.h"
#include "convertors/commands/cpp_to_python.h"
#include "convertors/commands/python_to_cpp.h"

#include "base_types_structures/exception.h"
#include "server/command.h"

//+-------------------------------------------------------------------------
//
// method : 		PyCmd::is_allowed
//
// description : 	Decide if it is allowed to execute the command
//
// argin : - dev : The device on which the command has to be executed
//	   - any : The input data
//
// This method returns a boolean set to True if it is allowed to execute
// the command. Otherwise, returns false
//
//--------------------------------------------------------------------------
bool PyCmd::is_allowed(Tango::DeviceImpl *dev, [[maybe_unused]] const CORBA::Any &any) {
    if(py_allowed_defined) {
        GET_DEVICE
        try {
            return py_dev.attr(py_allowed_name.c_str())().cast<bool>();
        }
        CATCH_PY_EXCEPTION
    }
    return true;
}

void allocate_any(CORBA::Any *&any_ptr) {
    try {
        any_ptr = new CORBA::Any();
    } catch(std::bad_alloc &) {
        Tango::Except::throw_exception(
            "API_MemoryAllocation", "Can't allocate memory in server", "PyCmd::allocate_any()");
    }
}

CORBA::Any *PyCmd::execute(Tango::DeviceImpl *dev, const CORBA::Any &param_any) {
    GET_DEVICE
    try {
        CORBA::Any param_in = const_cast<CORBA::Any &>(param_any);
        // This call extracts the CORBA any into a python object.
        // So, the result is that param_py = param_in.
        // It is done with some template magic.

        py::object param_py;

        TANGO_DO_ON_DEVICE_DATA_TYPE_ID(in_type,
                                        (scalar_cpp_data_to_python<CORBA::Any, tangoTypeConst>(param_in, param_py));
                                        ,
                                        (array_cpp_data_to_python<tangoTypeConst>(param_in, param_py)););

        // Execute the python call for the command
        py::object ret_py_obj;

        if(in_type == Tango::DEV_VOID) {
            ret_py_obj = py_dev.attr(name.c_str())();
        } else {
            ret_py_obj = py_dev.attr(name.c_str())(param_py);
        }

        CORBA::Any *ret_any;
        allocate_any(ret_any);
        std::unique_ptr<CORBA::Any> ret_any_guard(ret_any);

        // It does: ret_any = ret_py_obj
        TANGO_DO_ON_DEVICE_DATA_TYPE_ID(out_type,
                                        (scalar_python_data_to_cpp<CORBA::Any, tangoTypeConst>(*ret_any, ret_py_obj));
                                        ,
                                        (array_python_data_to_cpp<CORBA::Any, tangoTypeConst>(*ret_any, ret_py_obj)););

        return ret_any_guard.release();
    } catch(py::error_already_set &eas) {
        TangoSys_OMemStream origin;
        origin << TANGO_EXCEPTION_ORIGIN << std::ends;
        handle_python_exception(eas, "PyDs_UnexpectedFailure", "Cannot execute command", origin.str());
    }
}