Chen Jian

导航

 

运行openerp模块中的函数 有很多方式, 可以在视图中加个按钮然后点击它, 也可以在集成开发环境中强制执行它。 不过, 用python写个小脚本,xml-rpc调用直接执行它, 无疑是最简便的方法了

一 调用方法

示例:

import xmlrpclib 
from pprint import pprint
 
uid=1  #用户名admin所对应的id号 
pwd="admin"   #admin对应的密码
dbname="demo"
tcp_port="8069"
sock=xmlrpclib.ServerProxy("http://127.0.0.1:%s/xmlrpc/object" % tcp_port)
 
model="hr.employee.category"  ##要运行哪个类中的函数?
method="read"      ##要运行的函数
 
##def read (self, cr, uid, ids, fields=None, context=None)  #函数签名
ids=[1,2,3]  ##参数一,
fields=['name','complete_name','parent_id', 'child_ids']  ##参数二
result = sock.execute(dbname, uid, pwd, 
                      model,method , 
                      ids, fields )
print "\n==== %s of %s ====" % (model, method)
pprint(result)

可以看到, 我们在sock.execute中使用的参数, 与函数签名中给出的参数不完全相同,其对应关系是:

---函数签名中最前面的三个参数为 self, cr, uid

--socke.exec中最前面的五个参数为 dbname, uid, pwd,model,method

其后的参数相同。

二 调用日志

在openerp 7.0中, 我们在浏览器中做的操作,都会转换成xml-rpc调用, 我们可以通过查看xml-rpc的调用日志, 来了解openerp是使用什么参数调用xml-rpc的。 我们可以在函数disptach下增加如下语句来生成日志

params = params[3:] ##加在这一行下面
from pprint import pformat  ##这一行最后加在文件顶部
if ( params[0][0:2] =='op.' or params[0][1:3] =='op.'): #最关心的模块
    s=method+'!'+pformat(params[0])+pformat(params[1:],depth=9) 
elif ( params[1] =='get_needaction_data'):  #最不关心的调用
    s=method+'!'+pformat(params[0])+pformat(params[1:],depth=1)
else :  #其他模块
    s=method+'!'+pformat(params[0])+pformat(params[1:],depth=3

在7.0中,这段语句需要加在文件 service\web_services.py 的class objects_proxy中

在8.0中,这段语句需要加在文件 service\model.py中。 需要指出的是, 在默认情况下, oe8的浏览器操作已经不再使用xml-rpc了。

openerp提供了--log-request, 但是它记录的参数信息不够详细, 在oe7中, 生成的日志条数又太多, 因此我们不采用它。


三 调用以下划线开头的函数

出于安全的考虑, openerp不允许从模块外部调用下划线开头的函数。 在试验环境中, 这对于我们探索openerp的奥秘是个障碍, 我们注释如下这两行的源码就行了 (oe8在文件model.py中,oe7在文件osv.py中

if method.startswith('_'):

#   raise except_orm('Access Denied', 'Private methods (such as %s) cannot be called remotely.' % (method,))

运行这些以下划线开头的函数(以及browse函数),返回结果可能无法打包成xml格式, 这就会出现如下错误,

xmlrpclib.Fault: objects:

我们对python库文件xmlrpclib.py中的dump_struct函数稍加修改, 就可以运行除了broswe之外几乎所有的函数了

for k, v in value.items():
            write("\n")
            if type(k) is IntType:  ##添加此行和下一行
                k = "%d" % k        ##将diction的键值由数字转为字符  
            if type(k) is not StringType:
                if unicode and type(k) is UnicodeType:
                    k = k.encode(self.encoding)
                else:
                    raise TypeError, "dictionary key must be string"
posted on 2016-03-30 11:56  Chen Jian  阅读(829)  评论(0编辑  收藏  举报