Можете ли вы помочь мне решить эту проблему с SUDS/SOAP?
Итак, я пытаюсь получить доступ к этому api https://www.clarityaccounting.com/api-docs/ с помощью SUDS. Вот код, который должен работать:
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, которую можно использовать здесь: Какие существуют клиентские библиотеки SOAP для Python и где находится документация для них?
Итак, мой вопрос заключается в том, какие ключевые моменты отличаются друг от друга, и что приводит к сбою запроса и как я могу настроить SUDS для отправки правильно отформатированного запроса?
Ответы
Ответ 1
На первый взгляд похоже, что проблема связана с SSL. Вы получаете доступ к URL-адресу https, а обработчик транспорта для suds.client говорит по умолчанию по умолчанию.
Проблема
Если вы посмотрите на нижнюю часть WSDL, он указывает местоположение по умолчанию как http://www.clarityaccounting.com/api/v1
, которое является URL-адресом http, но 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 GET на этом 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"
}
Победа!
Совет для будущих целей отладки: включите полную отладочную регистрацию. 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:
Ответ 2
Это не должно быть проблемой, связанной с подключением к службе через HTTPS. Я использую пенообразование, чтобы сделать то же самое. Я пробовал несколько подходов к вашему WSDL файлу (а не самому эксперту) и столкнулся с той же ошибкой. То, что вы должны делать, как практика с пеной, но использует метод factory, например.
login = client.factory.create('doLogin')
login.username = 'username'
etc...
Если все, что отправляется функции create, является одним из типов, определенных в файле WSDL. Если вы создаете этот тип в оболочке, вы можете запустить "login для печати", чтобы увидеть его дополнительные свойства.
Надеюсь, это, по крайней мере, говорит вам, где проблема (с HTTPS). Кроме того, я заметил, что заголовки soapAction не установлены в WSDL файле, не уверены, как бездействует или может обрабатывать запросы без этого.
Ответ 3
Использование 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})
Мне кажется, что я должен документировать все способы, которыми можно проверять сырое мыло, которое вызывается пеной.
-
Использование флага nosend при построении клиента. Обратите внимание, что с флагом, установленным на True, пена будет просто генерировать мыло, но не отправлять его.
client =Client(url, nosend=True)
res = client.service.example()
print res.envelope
# отпечатка необработанного мыла
-
Использование ведения журнала. Здесь мы только регистрируем 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)
-
Использование MessagePlugin
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 дает вам возможность проверять мыло, но вы также можете изменить его перед отправкой, see- > https://jortel.fedorapeople.org/suds/doc/suds.plugin.MessagePlugin-class.html