14

所以我正在尝试使用 SUDS访问这个 api https://www.clarityaccounting.com/api-docs/ 。这是应该工作的代码:

from suds.client import Client
client = Client('https://www.clarityaccounting.com/api/v1?wsdl')
token = client.service.doLogin('demo', 'demo', 'www.kashoo.com', 'en_US', 300000)

但我得到这个错误:

WebFault: Server raised fault: 'No such operation:  (HTTP GET PATH_INFO: /api/v1)'

他们的支持人员说请求应该如下所示:

<SOAP-ENV:Envelope
SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"
xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:api="http://api.service.books/">
  <SOAP-ENV:Body>
     <api:doLogin>
        <username>demo</username>
        <password>demo</password>
        <siteName>www.kashoo.com</siteName>
        <locale>en_US</locale>
        <duration>300000</duration>
     </api:doLogin>
  </SOAP-ENV:Body>
</SOAP-ENV:Envelope>

但 SUDS 看起来像这样:

<?xml version="1.0" encoding="UTF-8"?>
<SOAP-ENV:Envelope 
xmlns:ns0="http://api.service.books/" 
xmlns:ns1="http://schemas.xmlsoap.org/soap/envelope/" 
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/">
   <SOAP-ENV:Header/>
   <ns1:Body>
      <ns0:doLogin>
         <username>demo</username>
         <password>demo</password>
         <siteName>www.kashoo.com</siteName>
         <locale>en_US</locale>
         <duration>300000</duration>
      </ns0:doLogin>
   </ns1:Body>
</SOAP-ENV:Envelope>

我是一个真正的 SOAP 和 SUDS 新手,但我听说 SUDS 是从这里使用的最佳 SOAP 库:Python 存在哪些 SOAP 客户端库,它们的文档在哪里?

所以我的问题只是有哪些不同的关键部分导致请求失败,我如何配置 SUDS 以发送格式正确的请求?

4

3 回答 3

37

乍一看,您遇到的问题似乎与 SSL 有关。您正在访问 https URL,并且 suds.client 的传输处理程序默认使用 http。

问题
如果您查看 WSDL 的底部,它会将默认位置指定为http://www.clarityaccounting.com/api/v1,这是一个 http URL,但 WSDL 是 SSL。

 <wsdl:service name="v1">
    <wsdl:port binding="tns:v1SoapBinding" name="BooksApiV1Port">
      <soap:address location="http://www.clarityaccounting.com/api/v1"/>
    </wsdl:port>
 </wsdl:service>

如果您对该 URL 执行 http GET,您会收到收到的错误消息:

<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
    <soap:Body>
        <soap:Fault>
            <faultcode>soap:Server</faultcode>
            <faultstring>No such operation:  (HTTP GET PATH_INFO: /api/v1)</faultstring>
        </soap:Fault>
    </soap:Body>
</soap:Envelope>

解决方案
要解决此问题,您需要在调用Client构造函数时覆盖默认位置以使其与 https 保持一致:

>>> url
'https://www.clarityaccounting.com/api/v1?wsdl'
>>> client = Client(url, location='https://www.clarityaccounting.com/api/v1')
>>> token = client.service.doLogin('demo', 'demo', 'www.kashoo.com', 'en_US', 300000)
>>> token
(authToken){
   authenticationCode = "ObaicdMJZY6UM8xZ2wzGjicT0jQ="
   expiryDate = 2010-03-05 12:31:41.000698
   locale = "en_US"
   myUserId = 4163
   site = "www.kashoo.com"
 }

胜利!

未来调试的专业提示:打开完整的日志记录调试。SUDS 使用标准logging库,因此它为您提供了很多控制权。所以我把它全部提高到DEBUG

import logging
logging.basicConfig(level=logging.INFO)
logging.getLogger('suds.client').setLevel(logging.DEBUG)
logging.getLogger('suds.transport').setLevel(logging.DEBUG)
logging.getLogger('suds.xsd.schema').setLevel(logging.DEBUG)
logging.getLogger('suds.wsdl').setLevel(logging.DEBUG)

这就是帮助我缩小范围的原因,因为它清楚地表明它是通过 http 发送的:

DEBUG:suds.transport.http:sending:
URL:http://www.clarityaccounting.com/api/v1
(xml output omitted)

然后回复也是这么说的:

DEBUG:suds.client:http failed:
于 2010-03-05T17:19:10.743 回答
2

使用 suds-jurko https://pypi.python.org/pypi/suds-jurko这是一个维护的 suds 分支。你可以传入一个 __inject 选项,你可以在其中给它你想要发送的原始 xml。

from suds.client import Client

username, password, sitename, locale, duration = 'demo', 'demo', 'www.kashoo.com', 'en_US', 300000

raw_xml = """<SOAP-ENV:Envelope
SOAP-ENV:encodingStyle="http://schemas.xmlsoap.org/soap/encoding/"
xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/"
xmlns:SOAP-ENC="http://schemas.xmlsoap.org/soap/encoding/"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:xsd="http://www.w3.org/2001/XMLSchema"
xmlns:api="http://api.service.books/">
  <SOAP-ENV:Body>
     <api:doLogin>
        <username>{0}</username>
        <password>{1}</password>
        <siteName>{2}</siteName>
        <locale>{3}</locale>
        <duration>{4}</duration>
     </api:doLogin>
  </SOAP-ENV:Body>
</SOAP-ENV:Envelope>""".format(username, password, sitename, locale, duration)

client = Client(url, location)
result = client.service.doLogin(__inject={'msg':raw_xml})

我觉得我应该记录所有可以检查此处产生的肥皂水的方法。

  1. 在构建客户端时使用鼻子标志。请注意,将标志设置为 True,suds 只会生成肥皂,但不会发送。

    client =Client(url, nosend=True)
    res = client.service.example()
    print res.envelope#打印生肥皂

  2. 使用日志记录。这里我们只记录 suds.transport.http,所以它只会输出发送/接收的任何内容。

    import logging
    import sys
    handler = logging.StreamHandler(sys.stderr)
    logger = logging.getLogger('suds.transport.http')
    logger.setLevel(logging.DEBUG), handler.setLevel(logging.DEBUG)
    logger.addHandler(handler)

  3. 使用消息插件

    from suds.plugin import MessagePlugin
    class MyPlugin(MessagePlugin):
    def marshalled(self, context):
    #import pdb; pdb.set_trace()
    print context.envelope.str()

    client = Client(url, plugins=[MyPlugin()])

MessagePlugin 不仅使您能够检查生成的肥皂,而且您还可以在发送之前对其进行修改,请参阅-> https://jortel.fedorapeople.org/suds/doc/suds.plugin.MessagePlugin-class.html

于 2014-10-30T10:00:26.657 回答
1

通过 HTTPS 连接到服务应该不是问题。我正在使用泡沫做同样的事情。我已经尝试了几种方法来处理您的 WSDL 文件(我自己不是专家)并且遇到了同样的错误。不过,您应该使用工厂方法来练习 suds,例如

login = client.factory.create('doLogin')
login.username = 'username'
etc...

发送到 create 函数的任何内容都是 WSDL 文件中定义的类型之一。如果您在 shell 中创建该类型,则可以运行“打印登录”以查看其附加属性。

希望这至少可以告诉您问题出在哪里(使用 HTTPS)。另外,我注意到soapAction 标头没有在WSDL 文件中设置,不确定suds 或服务如何处理没有它的请求。

于 2010-03-05T20:36:30.657 回答