Код ошибки 706 при подписании PDF с помощью веб-агента в Java

При тестировании примера веб-агента на Java я получаю ответ об ошибке

<?xml version="1.0" encoding="utf-8"?>
<response xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" type="error">
  <Error>
    <returnCode>706</returnCode>
    <errorMessage>Value cannot be null.
Parameter name: s</errorMessage>
  </Error>
</response>

Я следовал примеру Ruby в примерах веб-агента CoSign и документации

Я использовал файл demo.pdf, представленный в примере.

Это XML (из тестового приложения), отправленный в запросе POST (<content></content> имеет закодированный в Base64 PDF, но опущен из-за длины).

<?xml version="1.0" encoding="utf-8" ?>
<request>
  <Logic>
    <allowAdHoc>true</allowAdHoc>
    <workingMode>pull</workingMode>
    <enforceReason>false</enforceReason>
  </Logic>
  <Url>
    <finishURL>http://localhost:64956/retrieveSignedFile.aspx</finishURL>
  </Url>
  <Document>
    <fileID>1234567890</fileID>
    <contentType>pdf</contentType>
    <content>{BASE64 encoded pdf content}</content>
  </Document>
</request>

Ниже приведен код Java, который я использовал:

public class CoSignTest {
    private static final String INPUT = "D:\\tmp\\demo.pdf";
    private static final String PRECONTENT = "<?xml version=\"1.0\" encoding=\"utf-8\" ?>\n" +
            "<request>\n" +
            "  <Logic>\n" +
            "    <allowAdHoc>true</allowAdHoc>\n" +
            "    <workingMode>pull</workingMode>\n" +
            "    <enforceReason>false</enforceReason>\n" +
            "  </Logic>\n" +
            "  <Url>\n" +
            "    <finishURL>http://localhost:64956/retrieveSignedFile.aspx</finishURL>\n" +
            "  </Url>\n" +
            "  <Document>\n" +
            "    <fileID>1234567890</fileID>\n" +
            "    <contentType>pdf</contentType>\n" +
            "    <content>";
    private static final String POSTCONTENT = "</content>\n" +
            "  </Document>\n" +
            "</request>";
    private static final String POST_URL = "https://webagentdev.arx.com/Sign/UploadFileToSign";
    private static final String PULL_URL = "https://webagentdev.arx.com/Sign/DownloadSignedFileG";
    public static final int TIMEOUT = 300000;

    public static void main(String[] args) throws Exception {
        InputStream is = new FileInputStream(INPUT);
        String content = PRECONTENT + new String(Base64.encodeBase64(loadResource(is)), "UTF-8") + POSTCONTENT;
        System.out.println(content);
        String reply = new String(sendDocForProcessing(URLEncoder.encode(content, "UTF-8")));
        System.out.println(reply);
        System.out.println("DONE");
    }

    private static String sendDocForProcessing(String content) throws Exception {
        HttpClient client = null;
        HttpMethodBase method = null;
        SimpleHttpConnectionManager mgr = new SimpleHttpConnectionManager();
        String reply = "";
        try {
            mgr.getParams().setConnectionTimeout(TIMEOUT);
            mgr.getParams().setSoTimeout(TIMEOUT);
            client = new HttpClient(mgr);
            method = new PostMethod(POST_URL);
            method.getParams().setParameter(HttpMethodParams.RETRY_HANDLER, new DefaultHttpMethodRetryHandler(1, false));
            method.getParams().setParameter("http.socket.timeout", TIMEOUT);
            client.getHttpConnectionManager().getParams().setConnectionTimeout(TIMEOUT);
            client.getParams().setCookiePolicy(CookiePolicy.BROWSER_COMPATIBILITY);
            method.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
            method.getParams().setParameter("inputXML", content);
            client.executeMethod(method);
            reply = new String(method.getResponseBody());
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if(method != null) {
                method.releaseConnection();
            }
            client = null;
            mgr.shutdown();
        }
        if (isSigningSuccessful(reply)) {
            return reply;
        } else {
            throw new Exception("Failed in signing the document. Error: " + reply);
        }
    }

    private static boolean isSigningSuccessful(String reply) throws ParserConfigurationException, IOException, SAXException {
        DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
        DocumentBuilder db = dbf.newDocumentBuilder();
        Document doc = db.parse(new ByteArrayInputStream(reply.getBytes()));
        Element elem = doc.getDocumentElement();
        String type = elem.getAttribute("type");
        return !"error".equals(type);
    }


    public static byte[] loadResource(InputStream in) {
        if (in == null) {
            return new byte[0];
        }
        try {
            int indice, tempIndice;
            byte[] tempArr;
            byte[] mainArr = new byte[0];
            byte[] byteArr = new byte[65535];
            for (indice = 0; (indice = in.read(byteArr)) > 0;) {
                tempIndice = mainArr.length + indice;
                tempArr = new byte[tempIndice];
                System.arraycopy(mainArr, 0, tempArr, 0, mainArr.length);
                System.arraycopy(byteArr, 0, tempArr, mainArr.length, indice);
                mainArr = tempArr;
            }
            in.close();
            return mainArr;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return new byte[0];
    }
}

2 ответа

Решение

Спасибо за добавление вашего кода Java. Обратите внимание, что HttpClient Экземпляр настроен неправильно, и в результате запрос http-post отправляется пустым. Посмотрите на изменения, которые я сделал в вашем sendDocForProcessing Функция для правильного размещения содержимого XML:

private static String sendDocForProcessing(String content) throws Exception {
    HttpClient client = null;
    PostMethod method = null;
    String reply = "";
    try {
        client = new HttpClient();
        method = new PostMethod(POST_URL);
        method.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
        NameValuePair[] data = { new NameValuePair("inputXML", content) };
        method.setRequestBody(data);
        client.executeMethod(method);
        reply = method.getResponseBodyAsString();
    } catch (Exception e) {
        e.printStackTrace();
    } finally {
        if(method != null) {
            method.releaseConnection();
        }
    }
    if (isSigningSuccessful(reply)) {
        return reply;
    } else {
        throw new Exception("Failed in signing the document. Error: " + reply);
    }
}

Содержимое, передаваемое вышеупомянутой функции, не должно быть закодировано в URL, как это уже сделано HttpClient библиотека.

Кроме того, при анализе ответа я предлагаю вам проверить значение returnCode элемент, а не type имущество. Ответ всегда имеет тип "ошибка". Также обратите внимание, что имя функции isSigningSuccessful вводит в заблуждение, так как этот этап еще до подписания.

Элементы XML чувствительны к регистру и должны быть переданы, как показано в документации (например, Document вместо document, Auth вместо auth и так далее). Кроме того, в вашем XML-запросе отсутствует finishURL параметр, который является обязательным.

Также обратите внимание, что некоторые параметры в вашем XML-запросе устарели. Смотрите обновленный список параметров запроса по ссылке выше. Образец XML доступен здесь.

Другие вопросы по тегам