JavaScript получает пустой ответ или пустой responseXML от Python SimpleXMLRPCServer

Я нашел ответ на этот вопрос. Это решено.

Повествование: я получаю доступ к Python API, набору methodCalls поверх SimpleXMLRPCServer. Сервер отвечает на запрос GET браузера с помощью html-страницы "web_interface.html". HTML-страница - это очень простой скрипт, который отправляет XHR POST-запрос параметров xml на сервер XMLRPC. Сервер отвечает на XHR POST с заголовками, но пустым документом. Сервер отвечает на cURL с правильными данными. Почему JavaScript не получает читаемых данных в ответ от сервера?


| web_interface.html |

<!DOCTYPE html>
<html>
<head>

<script>

var xrequest = '<?xml version="1.0?"><methodCall><methodName>helloWorld<methodName><params><param><firstWord><string>hello</string><firstWord></param><param><secondWord><string>world</string></secondWord></param></params></methodCall>';

function hello() {
var xhr = new XMLHttpRequest();
xhr.onreadystatechange = function() {
    if (this.readyState == XMLHttpRequest.DONE) {
        alert(this.responseText);
        alert(this.status);
        alert(this.response);
    }
}
xhr.open('POST', '/', true);
xhr.setRequestHeader("Authorization", "Basic " + "aGVsbG8=" + ":" + "dGVzdA==");
xhr.send(xrequest);    
}    

</script>

</head>
<body>

<div>
  <h2 id="msgoutput">HelloWorld API Test</h2>
  <button type="button" onclick="hello(); return false;">SAY HELLO!</button>
</div>
</body>
</html>

Примечание. Нажатие кнопки приводит к появлению диалоговых окон с предупреждениями. В диалоговом окне состояния отображается "200", а в диалоговых окнах "Текст" и "Ответ" отсутствуют значения.


| Данные и заголовки инспектора Mozilla |

POST необработанные данные:

<?xml version="1.0?"><methodCall><methodName>helloWorld<methodName><params><param><firstWord><string>hello</string><firstWord></param><param><secondWord><string>world</string></secondWord></param></params></methodCall>

Заголовки ответа:

Content-Length 332
Content-Type text/html
Date Sat, 10 Dec 2016 23:51:21 GMT
Server BaseHTTP/0.3 Python/2.7.12

Заголовки запроса:

Accept text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Encoding gzip, deflate
Accept-Language en-US,en;q=0.5
AuthorizationBasic aGVsbG8=:dGVzdA== (not my actual creds, swapped fakes)
Connection keep-alive
Content-Length 218
Content-Type text/plain;charset=UTF-8
DNT1
Hostlocalhost:8442
Referer http://localhost:8442/
User-Agent Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:50.0) Gecko/20100101 Firefox/50.0

| Тест с cURL |

:~$ curl -i --data '<?xml version="1.0"?><methodCall><methodName>helloWorld</methodName><params><param><firstWord><string>hello</string></firstWord></param><param><secondWord><string>world</string></secondWord></param></params></methodCall>' http://username:password@localhost:8442
HTTP/1.0 200 OK
Server: BaseHTTP/0.3 Python/2.7.12
Date: Sun, 11 Dec 2016 00:00:13 GMT
Content-type: text/html
Content-length: 137

<?xml version='1.0'?>
<methodResponse>
<params>
<param>
<value><string>hello-world</string></value>
</param>
</params>
</methodResponse>

Примечание: нет проблем, cURL возвращает XML-ответ в виде текста. Я указал cURL на сокет netcat, чтобы увидеть, что именно он отправляет на сервер XMLRPC. Вот что показывает netcat при попадании cURL:


| CURL POST Данные |

POST / HTTP/1.1
Host: localhost:8442
Authorization: Basic YWRtaW46Z2liYmVyc2g=
User-Agent: curl/7.47.0
Accept: */*
Content-Length: 220
Content-Type: application/x-www-form-urlencoded

<?xml version="1.0"?><methodCall><methodName>helloWorld</methodName><params><param><firstWord><string>hello</string></firstWord></param><param><secondWord><string>world</string></secondWord></param></params></methodCall>

Это не CORS. Уже протестировал GET-запрос к xhr.responseText с тем же браузером на той же машине. Программа установки использует один и тот же хост, один и тот же порт, один и тот же каталог как для страницы GET, так и для запроса XHR POST XMLRPC.

Что мне не хватает?

1 ответ

В Python SimpleXMLRPCServer был взломан некоторый код для обработки куки. Проблема в том, что код не может обрабатывать объекты cookie, когда браузер устанавливает соединение. Этот код выдавал ошибку при каждом вызове. Он мешал серверу отправлять текст ответа в браузер. Я узнал об этом после написания фрагмента Python для вывода отладочной информации в файл. Я удалил код cookie, а затем XML-ответ был успешно отправлен с сервера в браузер.

Я также обнаружил, что отправка имени пользователя / пароля из оператора XHR Send приведет к ошибке аутентификации с сервера. Сервер ожидает, что учетные данные будут в заголовке POST как Аутентификация: Basic base64 [btoa(имя пользователя: пароль)].

Заголовок типа контента не был виновником в этом случае. Теперь для content-type установлено значение text/xml как на клиенте, так и на сервере.

Вот модифицированный код JavaScript, который работает:

<!DOCTYPE html>
<html>
<head>

<script>

var xrequest = '<?xml version="1.0"?><methodCall><methodName>helloWorld</methodName><params><param><firstWord><string>hello</string></firstWord></param><param><secondWord><string>world</string></secondWord></param></params></methodCall>';

function hello() {
var xhr = new XMLHttpRequest();
xhr.onreadystatechange = function() {
    if (this.readyState == XMLHttpRequest.DONE) {

    }
}
xhr.open("POST", "/", true);
xhr.setRequestHeader("Authorization", "Basic " + btoa("admin" + ":" + "1234"));
xhr.setRequestHeader("Content-Type", "text/xml")
xhr.send(xrequest);    
}    

</script>

</head>
<body>

<div>
  <h2 id="msgoutput">HelloWorld API Test</h2>
  <button type="button" onclick="hello(); return false;">SAY HELLO!</button>
</div>
</body>
</html>
Другие вопросы по тегам