参加新工作以来的一些吐槽

一、背景

今年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\n", rlt_char);

    return 0;
}
# hello.py
class power():
    def liupan(self):
        return "hello world"

四、总结

所以,如果PyImport_Import总是返回为空,一定是查询目录的问题,要么目录设置错误,要么python代码根本没有在这个目录内。

因为C调用python是运行时执行的,而不是编译时链接的。

posted @ 2015-05-07 15:53  silianpan  阅读(435)  评论(0编辑  收藏  举报