Можете ли вы помочь мне решить эту проблему SUDS / SOAP?
поэтому я пытаюсь получить доступ к этому 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 and SUDS, но я слышал, что SUDS-Лучшая библиотека мыла для использования отсюда:какой клиент мыла библиотеки существуют для Python, и где документация для них?
Итак, мой вопрос просто в том, какие важные части, которые отличаются и которые делают запрос неудачным, и как я могу настроить SUDS для отправки правильно отформатированного запроса?
3 ответов
на первый взгляд кажется, что проблема у вас есть с SSL. Вы получаете доступ к URL https и обработчику транспорта для suds.клиент говорит 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>
если вы делаете http получить на этот URL, вы получите сообщение об ошибке вы получено:
<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"
 }
победа!
Pro совет для будущих целей отладки: включите полную отладку журнала.  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:
это не должно быть проблемой, связанной с подключением к службе через HTTPS. Я использую пену, чтобы сделать то же самое. Я пробовал несколько подходов к вашему файлу WSDL (не эксперт сам) и столкнулся с той же ошибкой. Что вы должны делать, как практика с пеной, хотя это использовать завод метод, например,
login = client.factory.create('doLogin')
login.username = 'username'
etc...
где все, что отправляется в функцию create, является одним из типов, определенных в файле WSDL. Если вы создадите этот тип в оболочке, вы можете запустить "print login" чтобы увидеть его дополнительные свойства.
надеюсь, что это, по крайней мере, скажет вам, где проблема не (с HTTPS). Кроме того, я заметил, что заголовки soapAction не установлены в файле WSDL, не уверен, как suds или служба обрабатывает запросы без этого.
используя suds-jurko https://pypi.python.org/pypi/suds-jurko который является поддерживаемой вилкой пены. Вы можете передать параметр __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})
я чувствую, что должен документировать все способы, которыми можно проверить сырое мыло, которое генерирует здесь suds.
- 
использование флага nosend при построении клиента. Обратите внимание, что с флагом True suds будет просто генерировать мыло, но не отправить его. client =Client(url, nosend=True)res = client.service.example()print res.envelope#печать сырого мыла
- 
использование ведения журнала. Здесь мы только добываем мыльную пену.транспорт.http, поэтому она будет выводить только то, что отправлено/получено. import loggingimport syshandler = logging.StreamHandler(sys.stderr)logger = logging.getLogger('suds.transport.http')logger.setLevel(logging.DEBUG), handler.setLevel(logging.DEBUG)logger.addHandler(handler)
- 
использование MessagePlugin from suds.plugin import MessagePluginclass 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
