我正在尝试编写一个简单的肥皂服务器。我知道肥皂封套的外观如何(预定义)。
对于服务器将服务的每个请求,我都知道可选的soap标头,方法的名称和参数,并且我也知道soap响应是什么样的(换句话说,定义了WSDL)。
我想了解的是我的spyne服务(从ServiceBase继承)应该是什么样的,才能使用它。
这是我期望的肥皂请求示例:
<?xml version='1.0' encoding='UTF-8'?>
<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/" xmlns:ns="http://resources.mydomain.com/soap/1.1">
<soap:Header>
<ns:myinfo>someinfo</ns:myinfo>
</soap:Header>
<soap:Body>
<ns:MyMethod>
<ns:mymethodparam>somevalue</ns:mymethodparam>
</ns:MyMethod>
</soap:Body>
</soap:Envelope>
目前,我正在做的只是通过curl发送上述信封。我得到的答复是:
<?xml version='1.0' encoding='UTF-8'?>
<tns:Envelope xmlns:tns="http://schemas.xmlsoap.org/soap/envelope/">
<tns:Body>
<tns:Fault>
<faultcode>senv:Client.ResourceNotFound</faultcode>
<faultstring>Requested resource '{http://resources.mydomain.com/soap/1.1}MyMethod' not found</faultstring>
<faultactor></faultactor>
</tns:Fault>
</tns:Body>
</tns:Envelope>
根据Burak Arslan的建议,我使用的代码是:
example.py:
from spyne.model.primitive import Boolean, Unicode, Integer
from spyne.model.complex import ComplexModelBase
from spyne.model.complex import ComplexModel
from spyne.model.complex import ComplexModelMeta
from spyne.service import ServiceBase
from spyne.decorator import rpc
import logging
class MyHeader(ComplexModel):
myinfo = Unicode
class MyMethodRequest(ComplexModel):
mymethodparam = Unicode
class SomeService(ServiceBase):
__in_header__ = MyHeader
@rpc(MyMethodRequest, _body_style='bare')
def MyMethod(request):
print "I got:", request.mymethodparam
我用来启动服务器的代码是:
test_run.py:
from spyne.application import Application
from spyne.protocol.soap import Soap11
from spyne.server.wsgi import WsgiApplication
from wsgiref.simple_server import make_server
import logging
logging.basicConfig(level=logging.DEBUG)
logging.info('listening to http://127.0.0.1:8000')
application = Application([SomeService],
'http://schemas.xmlsoap.org/soap/envelope/',
in_protocol=Soap11(),
out_protocol=Soap11())
wsgi_application = WsgiApplication(application)
server = make_server('127.0.0.1', 8000, wsgi_application)
server.serve_forever()
我不确定的另一件事是,如果我正确地设置了WSDL(在项目中的正确位置),但是据我所知(而且我可能完全不了解,我是肥皂和斯潘妮的新手),只要我相应地构建API(使用spyne),这实际上就没有关系。我对么?
我正在用curl发出请求:
curl --header "content-type: text/soap+xml; charset=utf-8" --data @example.xml http://127.0.0.1:8000
关于我在做什么错的任何指示?非常感谢!
最佳答案
好的,我实际上运行了您的代码,结果发现它存在一些微妙的问题,而我早先的答案中却没有提到。您实际上不需要任何'bare'
东西。tns
实例的Application()
参数不能已经存在。我将确保在Spyne 2.11中抛出ValueError()
。我为此提出了一个问题:https://github.com/arskom/spyne/issues/317
如果需要,请阅读http://spyne.io/docs/2.10/manual/01_highlevel.html中“应用程序”部分中的注释。如果不是,只需将tns="gobbledygook"
传递给Application
并完成它。
您需要根据此更改调整您的请求文档。
您还需要在ctx
之前添加mymethodparam
,因为装饰@rpc
的函数的第一个参数始终是ctx
。
至于SOAP标头中<myinfo>
中的数据,由于Spyne中的错误,这还行不通:只有ComplexModel
子类作为__in_header__
的参数似乎可以工作。我已向Spyne的问题跟踪器(https://github.com/arskom/spyne/issues/316)提交了一个错误,并希望此问题很快得到解决。
解决方法是将ComplexModel
子类用作标头类型。这意味着您需要通过修改客户端或使用method_accept_document
事件在对象化之前修改请求文档来修改客户端发送的请求。
因此,这是工作服务器:
import logging
from spyne.model.primitive import Boolean, Unicode, Integer
from spyne.model.complex import ComplexModel
from spyne.service import ServiceBase
from spyne.decorator import rpc
class MyHeader(ComplexModel):
myinfo = Unicode
class SomeService(ServiceBase):
__in_header__ = MyHeader
@rpc(Unicode)
def MyMethod(ctx, mymethodparam):
print "I got:", mymethodparam
print "I also got:", ctx.in_header.myinfo
if __name__ == '__main__':
from spyne.application import Application
from spyne.protocol.soap import Soap11
from spyne.server.wsgi import WsgiApplication
from wsgiref.simple_server import make_server
logging.basicConfig(level=logging.DEBUG)
logging.info('listening to http://127.0.0.1:8000')
application = Application([SomeService],
'myapp',
in_protocol=Soap11(validator='lxml'),
out_protocol=Soap11())
wsgi_application = WsgiApplication(application)
server = make_server('127.0.0.1', 8000, wsgi_application)
server.serve_forever()
这是一个工作要求:
<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"
xmlns:ns="http://resources.mydomain.com/soap/1.1"
xmlns:ns0="myapp">
<soap:Header>
<ns0:MyHeader>
<ns0:myinfo>someinfo</ns0:myinfo>
</ns0:MyHeader>
</soap:Header>
<soap:Body>
<ns0:MyMethod>
<ns0:mymethodparam>somevalue</ns0:mymethodparam>
</ns0:MyMethod>
</soap:Body>
</soap:Envelope>