一、背景
今年3月份辞职,进入了一家新公司,公司是做网络存储和数据智能的。进去不到一个月,老大给我叫我把ipmi集成到项目中。
以前一直未做过将开源项目集成到项目中,对自己来说是一种挑战。
二、探索
ipmi是inter出的一个协议,用于监测服务器硬件状态等,如电源、CPU、风扇、温度等
刚接受任务,满怀信心开始做,首先先了解一下ipmi是啥东东。
之后发现现在有很多ipmi工具已经封装了ipmi,即:ipmitool,ipmiutils、freeipmi,OpenIPMI等,直接使用工具就可以监测硬件。
于是,我就开始安装了ipmitool,然后用C的系统函数system调用ipmitool命令,返回一些结果,然后对结果进行处理。
其实这样很简单,当我把demo拿个老大看得时候。老大说可以这样做,但是这样做不好,直接封装api更好,其实ipmitool也是封装api做的。
所以,我就开始尝试去看api,看能不能封装api。
于是乎,漫长的痛苦之旅启程了。完全看不懂,而且资料少之又少。最重要是不知道程序处理流程是啥。各位博友,如果有人做过这方面(开源代码封装)的开发,能否详细交流一下,互相学习,感激不尽。我这个人技术虽然不是太好,普通的程序猿,但是我很爱这行,爱交知心朋友,希望各位同行,多多联系,多多交流学习。
我就给老大说这个搞不定,老大说要不你就用命令吧,用Python写最好。
当听到python,感觉还是很友好的,因为之前写过一些python,感觉还是很不错。
于是我就尝试python封装ipmitool命令,并用c去调用python。可能咋一看,各位会感觉多此一举,python调用c,c又调用python。
之所以用python,是因为,python正则处理非常强大。
三、难题
我简单看了一下c调用python的代码(网上很多),就写了一个demo,用c调用python类中的函数,运行起来,答应了一条hello world,感觉还不错。
下面c_python_utils.h是处理工具函数,test.cpp是测试程序,hello.py是python类
可是当我集成到项目中的时候,PyImport_Import总是返回为空,起初我以为是init()中设置目录的问题,但是目录设置的是当前目录,hello.py也放在当前目录。
然而,然后让我发现项目是部署在服务器上的,而服务器上可执行文件在某个目录内,这个目录下根本就没有hello.py(因为我没有放进去)
所以,根本就在当前目录下找不到,就是这个小小的问题,都把我折腾了好久,现在想起来真是心伤啊。不过还是解决了。
/*************************************************************************************************** c_python_utils.h C++ Network Library, Copyright (c) Datatom Software, Inc.(2015) Author: liu.pan (liu.pan@datatom.com) Creating Time: 2015-5-4 ***************************************************************************************************/ #ifndef _DTCORE_C_PYTHON_UTILS_H_ #define _DTCORE_C_PYTHON_UTILS_H_ #include <Python.h> #include <stdio.h> #ifdef __cplusplus extern "C" { #endif /* C++ */ /** * 调用python类中的成员函数 * @param module python脚本名称,不含扩展 * @param class_name python类名称 * @param function python类成员函数 * @param format python类函数参数格式 * @return 返回字符串 */ char* py_call( const char* module, const char* class_name, char* function, char* format, ... ) { PyObject* pName = NULL; PyObject* pMod = NULL; PyObject* pDict = NULL; PyObject* pClass = NULL; PyObject* pInstance = NULL; PyObject* pParam = NULL; PyObject* pResult = NULL; // 导入模块 pName = PyString_FromString(module); pMod = PyImport_Import(pName); if( !pMod ) { return ""; } // 获取模块字典属性 pDict = PyModule_GetDict(pMod); if ( !pDict ) { return ""; } // 通过字典获取模块中的类 pClass = PyDict_GetItemString(pDict, class_name); if ( !pClass ) { return ""; } pInstance = PyInstance_New(pClass, NULL, NULL); if ( !pInstance ) { return ""; } pResult = PyObject_CallMethod(pInstance, function, format); char *rlt_ch = NULL; PyArg_Parse( pResult, "s", &rlt_ch ); return rlt_ch; } /** * 一些环境的初始化 * */ void init() { Py_Initialize(); PyRun_SimpleString("import sys"); PyRun_SimpleString("sys.path.append('./')"); } /** * 逆初始化 */ void finit() { Py_Finalize(); } #ifdef __cplusplus } /* extern "C" */ #endif /* C++ */ #endif //_DTCORE_C_PYTHON_UTILS_H_
// test.cpp #include "c_python_utils.h" #include <stdio.h> int main(int argc, char const *argv[]) { init(); char* rlt_char = py_call("hello", "power", "liupan", "()"); finit(); printf("%s ", rlt_char); return 0; }
# hello.py class power(): def liupan(self): return "hello world"
四、总结
所以,如果PyImport_Import总是返回为空,一定是查询目录的问题,要么目录设置错误,要么python代码根本没有在这个目录内。
因为C调用python是运行时执行的,而不是编译时链接的。