Получить XML в виде простого текста - PullRequest
0 голосов
/ 23 ноября 2018

У меня есть конечная точка для Spring Rest API:

@PostMapping(value = "/v1/", consumes = { MediaType.APPLICATION_XML_VALUE,
            MediaType.APPLICATION_JSON_VALUE }, produces = { MediaType.APPLICATION_XML_VALUE,
                    MediaType.APPLICATION_JSON_VALUE })
    public PaymentResponse handleMessage(@RequestBody PaymentTransaction transaction, HttpServletRequest request) throws Exception {

    // get here plain XML  

}

модель XML.

@XmlRootElement(name = "payment_transaction")
@XmlAccessorType(XmlAccessType.FIELD)
public class PaymentTransaction {
    public enum Response {
        failed_response, successful_response
    }

    @XmlElement(name = "transaction_type")
    public String transactionType;
    .........
}

Как получить запрос XML в виде простого текста XML?

Я также пытался использовать перехватчик Spring: я пробовал этот код:

@SpringBootApplication
@EntityScan("org.plugin.entity")
public class Application extends SpringBootServletInitializer implements WebMvcConfigurer {

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
        return application.sources(Application.class);
    }

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
    ........

    @Bean
    public RestTemplate rsestTemplate() {
        List<ClientHttpRequestInterceptor> interceptors = new ArrayList<>();
        RestTemplate restTemplate = new RestTemplate(
                new BufferingClientHttpRequestFactory(new SimpleClientHttpRequestFactory()));
        restTemplate.setInterceptors(interceptors);
        return restTemplate;
    } 
}

Компонент для ведения журнала:

@Component
public class RestTemplateHeaderModifierInterceptor implements ClientHttpRequestInterceptor {

    @Override
    public ClientHttpResponse intercept(HttpRequest request, byte[] body, ClientHttpRequestExecution execution)
            throws IOException {

        StringBuilder sb = new StringBuilder();
        sb.append("[ ");
        for (byte b : body) {
            sb.append(String.format("0x%02X ", b));
        }
        sb.append("]");

        System.out.println("!!!!!!!!!!!!!!!");
        System.out.println(sb.toString());      

        ClientHttpResponse response = execution.execute(request, body);

        InputStream inputStream = response.getBody();

        String result = IOUtils.toString(inputStream, StandardCharsets.UTF_8);

        System.out.println("!!!!!!!!!!!!!!!");
        System.out.println(result);

        return response;
    }
}

Но в консоль ничего не выводится.Есть идеи, где я не прав?Возможно этот компонент не зарегистрирован?

Ответы [ 5 ]

0 голосов
/ 02 декабря 2018

Вы создали List<ClientHttpRequestInterceptor> interceptors = new ArrayList<>();, но не добавили к нему объект RestTemplateHeaderModifierInterceptor.

Вы можете автоматически подключить его в Application, как показано ниже:

@Autowired 
ClientHttpRequestInterceptor clientHttpRequestInterceptor;

и

interceptors.add(clientHttpRequestInterceptor);

Код выглядит следующим образом:

class Application {
...
@Autowired 
ClientHttpRequestInterceptor clientHttpRequestInterceptor;
@Bean
    public RestTemplate rsestTemplate() {
        List<ClientHttpRequestInterceptor> interceptors = new ArrayList<>();
        RestTemplate restTemplate = new RestTemplate(
                new BufferingClientHttpRequestFactory(new SimpleClientHttpRequestFactory()));
interceptors.add(clientHttpRequestInterceptor);
        restTemplate.setInterceptors(interceptors);
        return restTemplate;
    } 
 ...
}

Надеюсь, это поможет

0 голосов
/ 01 декабря 2018

Мы уже давно используем в производстве spring-mvc-logger .Он написан в виде фильтра сервлета, поэтому может быть добавлен в качестве независимой оболочки для конечной точки MVC.

Наша установка почти такая же, как описано в readme.md, хотя мы ограничиваем <url-pattern> под<filter-mapping> только к полезным конечным точкам.

Даже если это не совсем то, что вам нужно, база кода там представляет собой довольно хороший маленький пример.Особо обратите внимание на упаковку запроса / ответа, которая необходима в фильтре.(Это сделано для того, чтобы избежать IllegalStateException: getReader(), getInputStream() already called, которое могло бы произойти, если бы getReader() было вызвано дважды).

0 голосов
/ 26 ноября 2018

У нас была такая же проблема, и мы использовали это решение в производстве.Который не зависит от фреймворка (всегда в моей книге это хорошо) и прост.

Просто используйте его без указания XML.Затем прочитайте строки запроса и присоедините их к \n, если вы хотите, чтобы в вашем xml были новые строки.Если нет, присоединяйтесь к ним по номеру "" или как вам угодно.Это предполагает, что вы используете javax.servlet.http.HttpServletRequest

Пример:

@PostMapping(value = "/v1")
    public PaymentResponse handleMessage(HttpServletRequest request) throws Exception {

    final InputStream xml = request.getInputStream();
    final String xmlString = new BufferedReader(new InputStreamReader(xml))
          .lines()
          .collect(Collectors.joining("\n"));
   // do whatever you please with it

}

И у вас есть простая строка XML.

0 голосов
/ 29 ноября 2018

Чтобы ваш контроллер получил тело запроса в виде простой XML-строки, вам нужно только изменить тип параметра @RequestBody на String:

@PostMapping(value = "/v1/", consumes = { MediaType.APPLICATION_XML_VALUE, MediaType.APPLICATION_JSON_VALUE }, produces = { MediaType.APPLICATION_XML_VALUE, MediaType.APPLICATION_JSON_VALUE })
public PaymentResponse handleMessage(@RequestBody String xmlOrJson, HttpServletRequest request) throws Exception {
    ...

С помощью приведенного выше сопоставления, если клиент отправил XML,вы увидите сырой XML.В противном случае, если клиент отправил json, вы увидите необработанный JSON.Убедитесь, что вы проверили заголовок «Content-Type» запроса, чтобы узнать, с каким типом вы имеете дело.

См. https://docs.spring.io/spring/docs/current/spring-framework-reference/web.html#mvc-ann-requestbody

0 голосов
/ 25 ноября 2018

Разве не так просто, как показано ниже, получить его из HttpServletRequest, если я что-то не упустил.Я не думаю, что нужно использовать перехватчик и т. Д.

@PostMapping(value = "/v1/", consumes = { MediaType.APPLICATION_XML_VALUE,
            MediaType.APPLICATION_JSON_VALUE }, produces = { MediaType.APPLICATION_XML_VALUE,
                    MediaType.APPLICATION_JSON_VALUE })
    public PaymentResponse handleMessage(HttpServletRequest request) throws Exception {

    String str, wholeXML = "";
    try {
        BufferedReader br = request.getReader();
        while ((str = br.readLine()) != null) {
            wholeXML += str;
        }
    System.out.println(wholeXML);
    //Here goes comment question, to convert it into PaymentTransaction
   JAXBContext jaxbContext = JAXBContext.newInstance(PaymentTransaction.class);
    Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();

    StringReader reader = new StringReader(wholeXML);
    PaymentTransaction paymentTransaction = (PaymentTransaction) unmarshaller.unmarshal(reader);
}
...