Hi Andriy, Thanks for the swift response, but I could still use some clarifications on:
1) You mention that passing an Attachment object as service method parameter should work. My initial test setup did pass an Attachment object as input parameter as shown in "> 1)API interface declaration" in my mail. However when the client (see code below) tries to send a request with this signature, the JAXRSUtils.writeMessageBody(...) method that is called by the CXF stack throws an exception on the Attachment parameter saying: okt 03, 2024 9:46:54 AM org.apache.cxf.jaxrs.provider.MultipartProvider getHandlerForObject SEVERE: No message body writer found for class : class org.apache.cxf.jaxrs.ext.multipart.Attachment. okt 03, 2024 9:47:05 AM org.apache.cxf.jaxrs.utils.JAXRSUtils logMessageHandlerProblem SEVERE: Problem with writing the data, class java.util.ArrayList, ContentType: multipart/form-data okt 03, 2024 9:47:14 AM org.apache.cxf.phase.PhaseInterceptorChain doDefaultLogging WARNING: Interceptor for {http://api.documenten.magda.common.aeo.dvtm.be/}MessagesApi has thrown exception, unwinding now org.apache.cxf.interceptor.Fault: Problem with writing the data, class java.util.ArrayList, ContentType: multipart/form-data at org.apache.cxf.jaxrs.client.ClientProxyImpl$BodyWriter.doWriteBody(ClientProxyImpl.java:1142) at org.apache.cxf.jaxrs.client.AbstractClient$AbstractBodyWriter.handleMessage(AbstractClient.java:1223) The JAXRSUtils.writeMessageBody(...) method takes an 'entity' Object that is a List<Attachment>. The first Attachment in the list contains an object of type MessageToSend, while the second one contains an object of type Attachment for which 'no message body writer' could be found. The stack creates itself an Attachment object for each parameter of the multipart body, that is why I though that I can not pass it as a parameter to my service method. I guess I am not allowed to annotate an 'Attachment' parameter with @Multipart annotation as currently done in the method signature: @FormDataParam(value="upfile1") @Parameter(schema = @Schema(type = "string", format = "binary")) @Multipart(value = "upfile1", type="application/pdf", required = false) Attachment upfile1Detail However leaving the @Multipart annotation for the Attachment parameter away leads to the error: javax.ws.rs.ProcessingException: Resource method be.dvtm.aeo.common.magda.documenten.api.MessagesApi.createMessage2 has more than one parameter representing a request body I.e. now it is no longer clear that the Attachment parameter is part of the 'multipart'. That's why I switched to using an InputStream as parameter with @Multipart annotation but then I loose the Content-Disposition information. The @Multipart annotation doesn't allow to specify Content-Disposition information. Is there an alternative here? 2) Logging of Binary Data. I create my client with: private static MessagesApi getThreadsafeProxy(String baseAddress) { JacksonJsonProvider jjProvider = new JacksonJsonProvider(new CustomObjectMapper()); List<Object> providers = Arrays.asList(new MultipartProvider(), jjProvider); final JAXRSClientFactoryBean factory = new JAXRSClientFactoryBean(); factory.setAddress(baseAddress); factory.setServiceClass(MessagesApi.class); factory.setProviders(providers); factory.getOutInterceptors().add(new LoggingOutInterceptor()); factory.getInInterceptors().add(new LoggingInInterceptor()); factory.setThreadSafe(true); LoggingFeature feature = new LoggingFeature(); feature.setLogMultipart(true); feature.setLogBinary(true); feature.addBinaryContentMediaTypes(MediaType.APPLICATION_OCTET_STREAM); feature.addBinaryContentMediaTypes("application/pdf"); factory.setFeatures(Arrays.asList(feature)); MessagesApi api = factory.create(MessagesApi.class); ClientConfiguration config = WebClient.getConfig(api); addTLSClientParameters(config.getHttpConduit()); return api; } Here I do activate the logging for multipart and binary and also added some mediatypes (although couldn't find what it actually does). So I was expecting to see the whole message (attachments are rather small as it is a test). Well I used wireshark to get the full message and it doesn't show any Content-Disposition headers for the multipart elements. Regards, J.P. Urkens -----Original Message----- From: Andriy Redko <drr...@gmail.com> Sent: donderdag 3 oktober 2024 1:01 To: Jean Pierre URKENS <jean-pierre.urk...@devoteam.com>; dev@cxf.apache.org Subject: Re: CXF JAX-RS: working with multipart form-data Hi Jean, JPU> What is the correct way to annotate a file attachment as part of a JPU> mutlipart/form-data content body? We have many examples in our test suites over here [1], it really depends on the problem at hand. JPU> -> Question-01: Is this the appropriate way to pass files JPU> (PDF documents) as attachment in a mutlipart/form-data request, or are there better ways? You would probably better of with "multipart/mixed" [2] as in you case, you are sending different data types. But "multipart/form-data" should be fine as well. The right answer answer depends on how large the files are. In many cases you are better off using chunked transfer (no need to read the whole file in memory), like you do with InputStream. JPU> -> Question-02: When I activate logging, I can't see JPU> anything about the file attachment, not is content, nor the JPU> appropriate Content-Disposition settings? I get '--Content suppressed--', e.g.: Correct, by default the logging interceptors do not log binary data, you could checks the docs here [3]. JPU> -> Question-03: Don't I need to set anything regarding the JPU> Content-Disposition header? The example here is simplified in a JPU> sense that I used a test setup with just one file attachment. In JPU> the real interface up to JPU> 20 attachments can be passed. Somehow I need to know which part relates to JPU> which attachment or not? This is probably caused by the fact you are sending the InputStream and not an Attachment, if my understanding is correct. I am pretty sure passing the Attachment (as you did initially) should work. JPU> -> Question-04: At the server implemtation side, since JPU> upfile1Detail is passed as a method parameter, who is going to JPU> close this InputStream at the server side? The stream should be closed by the service implementation (since the stream is expected to be consumed). The Apache CXF runtime will try to close the stream in most cases as well but sometimes it may not be able to. Hope it answers your questions. Thanks! [1] https://github.com/apache/cxf/blob/main/systests/jaxrs/src/test/java/org/apache/cxf/systest/jaxrs/MultipartStore.java [2] https://learn.microsoft.com/en-us/exchange/troubleshoot/administration/multipart-mixed-mime-message-format [3] https://cxf.apache.org/docs/message-logging.html Best Regards, Andriy Redko JPU> Hi Andriy, JPU> What is the correct way to annotate a file attachment as part of a JPU> mutlipart/form-data content body? JPU> I currently have the following (only the relevant parts): JPU> 1)API interface declaration JPU> ====================== JPU> @POST JPU> @Path("/messages1") JPU> @Consumes("multipart/form-data") JPU> @Produces({ "application/json" }) JPU> @Operation(...) JPU> @ApiResponses(...) JPU> Response createMessage1( JPU> @HeaderParam("x-correlation-id") @NotNull JPU> @Size(min = 10, max = 36) @Parameter(description="ID of the JPU> transaction. Use this ID for log tracing and incident handling.") String xCorrelationId, JPU> @HeaderParam("Idempotency-Key") @NotNull JPU> @Size(min = 10, max = 36) @Parameter(description="When retrying a JPU> failed call, the retry call should have the same Idempotency Key.") String idempotencyKey, JPU> @FormDataParam(value="messageToSend") JPU> @Parameter(required=true,schema = JPU> @Schema(implementation=MessageToSend.class)) @Multipart(value = JPU> "messageToSend", type="application/json", required= true) JPU> MessageToSend messageToSend, JPU> @FormDataParam(value="upfile1") JPU> @Parameter(schema = @Schema(type = "string", format = "binary")) JPU> @Multipart(value = "upfile1", type="application/octet-stream", JPU> required = false) Attachment upfile1Detail); JPU> So I pass the file to upload as an Attachment object. JPU> 2)API client test code JPU> ================= JPU> String xCorrelationId = UUID.randomUUID().toString(); JPU> String idempotencyKey = JPU> UUID.randomUUID().toString(); JPU> //01. Get the attachment to include JPU> String fileName = "test.pdf"; JPU> try (InputStream is = JPU> this.getClass().getClassLoader().getResourceAsStream(fileName); JPU> BufferedReader reader = new JPU> BufferedReader(new InputStreamReader(is))) { JPU> if (is == null) { JPU> Assert.fail("Couldn't load test.pdf from classpath!"); JPU> } JPU> DataHandler dataHandler = new DataHandler(is, "application/pdf"); JPU> ContentDisposition cd = new JPU> ContentDisposition("attachment;name=upfile1;filename="+fileName); JPU> Attachment upfile1Detail = new AttachmentBuilder() JPU> .id("upfile1") JPU> .dataHandler(dataHandler) JPU> .contentDisposition(cd) JPU> .mediaType("application/pdf") JPU> .build(); JPU> //02. create the message to send JPU> MessageToSend mts = new MessageToSend(); JPU> //03. Call the server JPU> Response resp = JPU> apiClient.createMessage1(xCorrelationId, idempotencyKey, mts, JPU> upfile1Detail); JPU> When running the API client test code and getting at '03.' The method: JPU> JAXRSUtils.writeMessageBody(List<WriterInterceptor> JPU> writers,Object entity,Class<?> type, Type genericType,Annotation[] JPU> annotations,MediaType mediaType,MultivaluedMap<String, Object> JPU> httpHeaders,Message message) JPU> is called. The 'entity' object is a list of Attachment objects where: JPU> - the first Attachment object contains an object of type JPU> MessageToSend JPU> - the second Attachment object contains an object of type JPU> Attachment JPU> This second object leads to a fatal error within the JPU> JAXRSUtils.writeMessageBody(...) method: JPU> okt 02, 2024 1:36:02 PM JPU> org.apache.cxf.jaxrs.provider.MultipartProvider JPU> getHandlerForObject JPU> SEVERE: No message body writer found for class : class JPU> org.apache.cxf.jaxrs.ext.multipart.Attachment. JPU> okt 02, 2024 1:36:02 PM JPU> org.apache.cxf.jaxrs.utils.JAXRSUtils JPU> logMessageHandlerProblem JPU> SEVERE: Problem with writing the data, class JPU> java.util.ArrayList, JPU> ContentType: multipart/form-data JPU> I modified the interface and implementation classes to use JPU> 'InputStream upfile1Detail' as type for the input parameter of the JPU> service method. And in this case my 'dummy' server implementation JPU> can read the file and save it to disk. JPU> -> Question-01: Is this the appropriate way to pass files JPU> (PDF documents) as attachment in a mutlipart/form-data request, or are there better ways? JPU> -> Question-02: When I activate logging, I can't see JPU> anything about the file attachment, not is content, nor the JPU> appropriate Content-Disposition settings? I get '--Content suppressed--', e.g.: JPU> [MAGDADOC] 2024-10-02 14:29:08,911 [main] INFO JPU> $--$ JPU> (org.apache.cxf.ext.logging.slf4j.Slf4jEventSender:84) - REQ_OUT JPU> Address: JPU> http://localhost:8091/services/magdadoc/api/v1/messages/messages1 JPU> HttpMethod: POST JPU> Content-Type: multipart/form-data; JPU> boundary="uuid:3c3fa9c0-8470-4655-b026-3ed09f79e862" JPU> ExchangeId: a583a695-d881-4fa7-b65a-8961cdbbd412 JPU> Headers: {Authorization=Bearer JPU> f4262ccf-3250-4bcf-a1bc-7ee1bf9a56cf, JPU> Accept=application/json, JPU> Idempotency-Key=bd06c05d-9fe2-4b60-b8db-5ad1121b74dc, JPU> x-correlation-id=511c51ba-95fe-4f69-9443-f05c377cffab} JPU> Payload: JPU> --uuid:3c3fa9c0-8470-4655-b026-3ed09f79e862 JPU> Content-Type: application/json JPU> Content-Transfer-Encoding: binary JPU> Content-ID: <messageToSend> JPU> { JPU> "delivery" : "AUTOMATIC", JPU> "eboxDeliveryData" : { /* all fine */}, JPU> "paperDeliveryData" : {/* all fine */}, JPU> "emailDeliveryData" : null, JPU> "businessData" : [ ] JPU> } JPU> --uuid:3c3fa9c0-8470-4655-b026-3ed09f79e862 JPU> --- Content suppressed --- JPU> -> Question-03: Don't I need to set anything regarding the JPU> Content-Disposition header? The example here is simplified in a JPU> sense that I used a test setup with just one file attachment. In JPU> the real interface up to JPU> 20 attachments can be passed. Somehow I need to know which part relates to JPU> which attachment or not? JPU> -> Question-04: At the server implemtation side, since JPU> upfile1Detail is passed as a method parameter, who is going to JPU> close this InputStream at the server side? JPU> Regards, JPU> J.P. Urkens JPU> P.S.: Is there a migration document describing upgrading CXF-3.5.6 JPU> (my current version) to CXF-3.5.8 (latest JDK8 version). I'd like JPU> to know whether upgrading can happen without too much burden. JPU> -----Original Message----- JPU> From: Jean Pierre URKENS <jean-pierre.urk...@devoteam.com> JPU> Sent: vrijdag 5 juli 2024 13:04 JPU> To: 'Andriy Redko' <drr...@gmail.com>; 'dev@cxf.apache.org' JPU> <dev@cxf.apache.org> JPU> Subject: RE: CXF JAX-RS: working with multipart form-data JPU> Hi Andriy, JPU> When searching the net I came along this Jersey annotation but JPU> since I was not depending on 'Jersey' components I skipped it. So JPU> apparently swagger-core has processing for externally defined JPU> annotations (like this FormDataParam in Jersey), indeed inside know-how. JPU> I included it in my project as JPU> io.swagger.v3.oas.annotations.FormDataParam, JPU> since it should somehow be included in the supported swagger JPU> annotations (maybe we should submit a request for it to the JPU> swagger-core team) and this indeed generates an appropriate openapi.json spec. JPU> Thanks alot, JPU> J.P. Urkens JPU> -----Original Message----- JPU> From: Andriy Redko <drr...@gmail.com> JPU> Sent: vrijdag 5 juli 2024 2:16 JPU> To: Jean Pierre URKENS <jean-pierre.urk...@devoteam.com>; JPU> dev@cxf.apache.org JPU> Subject: Re: CXF JAX-RS: working with multipart form-data JPU> Hi Jean, JPU> Here is how you could make it work (there is some magic knowledge JPU> involved sadly). First of all, define such annotation anywhere in JPU> your codebase (where it dims appropriate): JPU> import java.lang.annotation.ElementType; import JPU> java.lang.annotation.Retention; import JPU> java.lang.annotation.RetentionPolicy; JPU> import java.lang.annotation.Target; JPU> @Target({ElementType.PARAMETER, ElementType.METHOD, JPU> ElementType.FIELD}) JPU> @Retention(RetentionPolicy.RUNTIME) JPU> public @interface FormDataParam { JPU> String value(); JPU> } JPU> Use this annotation on each Attachment parameter: JPU> /* Skipping other annotations as those are not important here */ JPU> public Response createMessage( JPU> @HeaderParam("x-correlation-id") @NotNull @Size(min = 10, JPU> max = 36) @Parameter(description="ID of the transaction. Use this JPU> ID for log tracing and incident handling.") String xCorrelationId, JPU> @HeaderParam("Idempotency-Key") @Size(min = 10, max = 36) JPU> @Parameter(description="When retrying a failed call, the retry call JPU> should have the same Idempotency Key.") String idempotencyKey, JPU> @FormDataParam("upfile1") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile1", JPU> type="application/octet-stream", required = false) InputStream JPU> upfile1Detail, JPU> @FormDataParam("upfile2") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile2", JPU> type="application/octet-stream", required = false) InputStream JPU> upfile2Detail, JPU> @FormDataParam("upfile3") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile3", JPU> type="application/octet-stream", required = false) Attachment upfile3Detail, JPU> @FormDataParam("upfile4") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile4", JPU> type="application/octet-stream", required = false) Attachment upfile4Detail, JPU> @FormDataParam("upfile5") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile5", JPU> type="application/octet-stream", required = false) Attachment upfile5Detail, JPU> @FormDataParam("upfile6") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile6", JPU> type="application/octet-stream", required = false) Attachment upfile6Detail, JPU> @FormDataParam("upfile7") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile7", JPU> type="application/octet-stream", required = false) Attachment upfile7Detail, JPU> @FormDataParam("upfile8") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile8", JPU> type="application/octet-stream", required = false) Attachment upfile8Detail, JPU> @FormDataParam("upfile9") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "upfile9", JPU> type="application/octet-stream", required = false) Attachment upfile9Detail, JPU> @FormDataParam("upfile10") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile10", type="application/octet-stream", required = false) JPU> Attachment upfile10Detail, JPU> @FormDataParam("upfile11") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile11", type="application/octet-stream", required = false) JPU> Attachment upfile11Detail, JPU> @FormDataParam("upfile12") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile12", type="application/octet-stream", required = false) JPU> Attachment upfile12Detail, JPU> @FormDataParam("upfile13") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile13", type="application/octet-stream", required = false) JPU> Attachment upfile13Detail, JPU> @FormDataParam("upfile14") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile14", type="application/octet-stream", required = false) JPU> Attachment upfile14Detail, JPU> @FormDataParam("upfile15") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile15", type="application/octet-stream", required = false) JPU> Attachment upfile15Detail, JPU> @FormDataParam("upfile16") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile16", type="application/octet-stream", required = false) JPU> Attachment upfile16Detail, JPU> @FormDataParam("upfile17") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile17", type="application/octet-stream", required = false) JPU> Attachment upfile17Detail, JPU> @FormDataParam("upfile18") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile18", type="application/octet-stream", required = false) JPU> Attachment upfile18Detail, JPU> @FormDataParam("upfile19") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile19", type="application/octet-stream", required = false) JPU> Attachment upfile19Detail, JPU> @FormDataParam("upfile20") @Parameter(schema = JPU> @Schema(type = "string", format = "binary")) @Multipart(value = JPU> "upfile20", type="application/octet-stream", required = false) JPU> Attachment upfile20Detail, JPU> @FormDataParam("qrfile") @Parameter(schema = @Schema(type JPU> = "string", format = "binary")) @Multipart(value = "qrfile", JPU> type="application/octet-stream", required = false) Attachment qrfileDetail JPU> ) { JPU> .... JPU> } JPU> With that, you will get a nice request body schema (publishing a JPU> bit large YAML snippet to preserve the context): JPU> paths: JPU> /sample/messages: JPU> post: JPU> tags: JPU> - messages JPU> summary: "Send a message, using a channel (email, paper mail, JPU> ebox) and delivery\ JPU> \ method (registered or normal) of your choice. More than 6 JPU> upfiles only supported\ JPU> \ for PAPER delivery." JPU> operationId: createMessage JPU> parameters: JPU> - name: x-correlation-id JPU> in: header JPU> description: ID of the transaction. Use this ID for log JPU> tracing and incident JPU> handling. JPU> required: true JPU> schema: JPU> maxLength: 36 JPU> minLength: 10 JPU> type: string JPU> - name: Idempotency-Key JPU> in: header JPU> description: "When retrying a failed call, the retry call JPU> should have the\ JPU> \ same Idempotency Key." JPU> schema: JPU> maxLength: 36 JPU> minLength: 10 JPU> type: string JPU> requestBody: JPU> content: JPU> multipart/form-data: JPU> schema: JPU> type: object JPU> properties: JPU> upfile1: JPU> type: string JPU> format: binary JPU> upfile2: JPU> type: string JPU> format: binary JPU> upfile3: JPU> type: string JPU> format: binary JPU> upfile4: JPU> type: string JPU> format: binary JPU> upfile5: JPU> type: string JPU> format: binary JPU> upfile6: JPU> type: string JPU> format: binary JPU> upfile7: JPU> type: string JPU> format: binary JPU> upfile8: JPU> type: string JPU> format: binary JPU> upfile9: JPU> type: string JPU> format: binary JPU> upfile10: JPU> type: string JPU> format: binary JPU> upfile11: JPU> type: string JPU> format: binary JPU> upfile12: JPU> type: string JPU> format: binary JPU> upfile13: JPU> type: string JPU> format: binary JPU> upfile14: JPU> type: string JPU> format: binary JPU> upfile15: JPU> type: string JPU> format: binary JPU> upfile16: JPU> type: string JPU> format: binary JPU> upfile17: JPU> type: string JPU> format: binary JPU> upfile18: JPU> type: string JPU> format: binary JPU> upfile19: JPU> type: string JPU> format: binary JPU> upfile20: JPU> type: string JPU> format: binary JPU> qrfile: JPU> type: string JPU> format: binary JPU> The key here is @FormDataParam annotation which (originally) comes JPU> from Jersey but has special treatment in Swagger Core (but, likely, JPU> no attribution to Jersey). JPU> Hope it helps! JPU> Thank you. JPU> Best Regards, JPU> Andriy Redko >> V2.2.22 (15/05/2024) is the latest version of io.swagger.core.v3 >> libraries. >> I upgrade to this version to make sure I had the latest swagger >> implementation. >> -----Original Message----- >> From: Andriy Redko <drr...@gmail.com> >> Sent: donderdag 4 juli 2024 4:44 >> To: Jean Pierre URKENS <jean-pierre.urk...@devoteam.com>; >> dev@cxf.apache.org >> Subject: Re: CXF JAX-RS: working with multipart form-data >> Hi Jean, >> Interesting, I was experimenting with different ways to express what >> you need, but no luck so far, I will try to spend a bit more time on >> that this week since OAS 3.x does support multipart [1] but we may >> indeed hit the >> limitation(s) of this particular Swagger Core version. Thank you. >> [1] >> https://swagger.io/docs/specification/describing-request-body/multipa >> r >> t-requests/ >> Best Regards, >> Andriy Redko >>> Hi Andriy, >>> I already tried this but it didn't work. E.g. for following API >>> interface >>> specification: >>> /** >>> * Send a message, using a channel (email, paper >>> mail, >>> ebox) and delivery method (registered or normal) of your choice. >>> More than >>> 6 upfiles only supported for PAPER delivery. >>> * >>> */ >>> @POST >>> @Path("/messages") >>> @Consumes("multipart/form-data") >>> @Produces({ "application/json" }) >>> @Operation( >>> summary >>> = "Send a message, using a channel (email, paper mail, ebox) and >>> delivery method (registered or normal) of your choice. More than 6 >>> upfiles only supported for PAPER delivery.", >>> tags = >>> {"messages" }, operationId="createMessage", >>> security=@SecurityRequirement(name="BearerAuthentication")) >>> @ApiResponses({ >>> @ApiResponse( >>> responseCode = "201", >>> description = "Created", >>> content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,array=@ArraySchema(sch >>> e m a=@Schema(implementation=SendStatusMessage.class))), >>> headers = {@Header( >>> name="X-Magda-Exceptions", >>> required=false, >>> description="Only used in the context of EBOX delivery and if there >>> was a problem with the consent of the receiver's ebox.", >>> schema=@Schema(implementation=MagdaExceptionList.class)) >>> }), >>> @ApiResponse( >>> responseCode = "400", >>> description = "Invalid data supplied", content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>> n >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "401", >>> description = "Invalid authorization", content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>> n >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "500", >>> description = "Unexpected Server Error", content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>> n >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "502", >>> description = "Bad Gateway", >>> content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>> n >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "503", >>> description = "Service unavailable", content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>> n >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "504", >>> description = "Gateway Timeout", >>> content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>> n >>> t >>> ation=ErrorMessage.class))) >>> }) >>> public Response createMessage( >>> @HeaderParam("x-correlation-id") @NotNull @Size(min = 10, max = 36) >>> @Parameter(description="ID of the transaction. Use this ID for log >>> tracing and incident handling.") String xCorrelationId, >>> @HeaderParam("Idempotency-Key") @Size(min = 10, max = 36) >>> @Parameter(description="When retrying a failed call, the retry call >>> should have the same Idempotency Key.") String idempotencyKey, >>> @Parameter(required=true,schema = >>> @Schema(implementation=MessageToSend.class)) @Multipart(value = >>> "messageToSend", type="application/json", required= true) >>> MessageToSend messageToSend, @Parameter(schema = @Schema(type = >>> "string", format = "binary")) @Multipart(value = "upfile1", >>> type="application/octet-stream", required = false) Attachment >>> upfile1Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile2", >>> type="application/octet-stream", required = false) Attachment >>> upfile2Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile3", >>> type="application/octet-stream", required = false) Attachment >>> upfile3Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile4", >>> type="application/octet-stream", required = false) Attachment >>> upfile4Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile5", >>> type="application/octet-stream", required = false) Attachment >>> upfile5Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile6", >>> type="application/octet-stream", required = false) Attachment >>> upfile6Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile7", >>> type="application/octet-stream", required = false) Attachment >>> upfile7Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile8", >>> type="application/octet-stream", required = false) Attachment >>> upfile8Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile9", >>> type="application/octet-stream", required = false) Attachment >>> upfile9Detail, @Parameter(schema = @Schema(type = "string", format = >>> "binary")) @Multipart(value = "upfile10", >>> type="application/octet-stream", required = false) Attachment >>> upfile10Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile11", >>> type="application/octet-stream", required = false) Attachment >>> upfile11Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile12", >>> type="application/octet-stream", required = false) Attachment >>> upfile12Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile13", >>> type="application/octet-stream", required = false) Attachment >>> upfile13Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile14", >>> type="application/octet-stream", required = false) Attachment >>> upfile14Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile15", >>> type="application/octet-stream", required = false) Attachment >>> upfile15Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile16", >>> type="application/octet-stream", required = false) Attachment >>> upfile16Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile17", >>> type="application/octet-stream", required = false) Attachment >>> upfile17Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile18", >>> type="application/octet-stream", required = false) Attachment >>> upfile18Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile19", >>> type="application/octet-stream", required = false) Attachment >>> upfile19Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "upfile20", >>> type="application/octet-stream", required = false) Attachment >>> upfile20Detail, @Parameter(schema = @Schema(type = "string", format >>> = >>> "binary")) @Multipart(value = "qrfile", >>> type="application/octet-stream", required = false) Attachment >>> qrfileDetail); I’ve attached the generated openapi specification. It >>> only contains the ‘messageToSend’ as part of the multipart/form-data >>> requestBody content, all attachments are ignored. >>> Below I’ve listed the libraries I’ve included in the project (cxf >>> v3.5.8 and swagger v2.2.2). Which of these libraries is acutal >>> responsible for generating the openapi.json specification from the >>> interface description? >>> * cxf-rt-rs-service-description-common-openapi:3.5.8 >>> * cxf-rt-rs-service-description-openapi:3.5.8 >>> * cxf-rt-rs-service-description-swagger-ui:3.5.8 >>> * swagger-core:2.2.2 >>> * swagger-annotations:2.2.2 >>> * swagger-integration:2.2.2 >>> * swagger-jaxrs2: 2.2.2 >>> * swagger-model: 2.2.2 >>> Note that I am still on JDK8, so I guess I can’t upgrade to a higher >>> version (currently our projects use cxf-v3.5.6 and swagger 2.1.13). >>> Regards, >>> J.P. Urkens >>> -----Original Message----- >>> From: Andriy Redko <drr...@gmail.com> >>> Sent: woensdag 3 juli 2024 5:57 >>> To: Jean Pierre URKENS <jean-pierre.urk...@devoteam.com>; >>> dev@cxf.apache.org >>> Subject: Re: CXF JAX-RS: working with multipart form-data Hi Jean >>> Pierre, I suspect the @Multipart annotation is coming from CXF >>> (org.apache.cxf.jaxrs.ext.multipart.Multipart), right? If yes, this >>> is not a part of JAX-RS specification but CXF specific extension. >>> You may need to add Swagger API annotation to the parameters in >>> question: >>> @Parameter(schema = @Schema(type = "string", format = "binary")) >>> Hope it helps. >>> Thank you. >>> Best Regards, >>> Andriy Redko >>> Monday, July 1, 2024, 12:09:17 PM, you wrote: >>>> Hi all, >>>> I am having problems to correctly annotate service methods which >>>> consumes multipart/form-data that contains attachments next to >>>> other model objects. >>>> I’ve an openapi specification that contains following requestBody >>>> definition: >>>> /messages: >>>> post: >>>> tags: >>>> - "messages" >>>> summary: "Send a message, using a channel (email, paper mail, >>>> ebox) and delivery method (registered or normal) of your choice. >>>> More than 6 upfiles only supported for PAPER delivery." >>>> operationId: createMessage >>>> parameters: >>>> - $ref: '#/components/parameters/CorrelationId' >>>> - $ref: '#/components/parameters/Idempotency-Key' >>>> requestBody: >>>> content: >>>> multipart/form-data: >>>> schema: >>>> type: object >>>> required: >>>> - messageToSend >>>> properties: >>>> messageToSend: >>>> $ref: '#/components/schemas/MessageToSend' >>>> upfile1: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile2: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile3: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile4: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile5: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile6: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile7: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile8: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile9: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile10: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile11: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile12: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile13: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile14: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile15: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile16: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile17: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile18: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile19: >>>> type: string >>>> format: binary >>>> nullable: true >>>> upfile20: >>>> type: string >>>> format: binary >>>> nullable: true >>>> qrfile: >>>> type: string >>>> format: binary >>>> nullable: true >>>> required: true >>>> When using the openapi-generator-maven-plugin v7.6.0 it generates >>>> following method signature: >>>> @POST >>>> @Path("/messages") >>>> @Consumes("multipart/form-data") >>>> @Produces({ "application/json" }) >>>> @Operation( >>>> summary = "Send a message, using a channel >>>> (email, paper mail, ebox) and delivery method (registered or >>>> normal) of your choice. More than 6 upfiles only supported for >>>> PAPER delivery.", >>>> tags = {"messages" }, >>>> operationId="createMessage", >>>> security=@SecurityRequirement(name="BearerAuthentication"), >>>> responses= { >>>> @ApiResponse( >>>> >>>> responseCode = "201", >>>> description >>>> = "Created", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,array=@ArraySchema(sc >>>> h em a=@Schema(implementation=SendStatusMessage.class))), >>>> headers = >>>> {@Header( name="X-Magda-Exceptions", required=false, >>>> description="Only used in the context of EBOX delivery and if there >>>> was a problem with the consent of the receiver's ebox.", >>>> schema=@Schema(implementation=MagdaExceptionList.class)) >>>> }), >>>> @ApiResponse( >>>> >>>> responseCode = "400", >>>> description >>>> = "Invalid data supplied", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implem >>>> e >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> >>>> responseCode = "401", >>>> description >>>> = "Invalid authorization", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implem >>>> e >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> >>>> responseCode = "500", >>>> description >>>> = "Unexpected Server Error", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implem >>>> e >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> >>>> responseCode = "502", >>>> description >>>> = "Bad Gateway", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implem >>>> e >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> >>>> responseCode = "503", >>>> description >>>> = "Service unavailable", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implem >>>> e >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> >>>> responseCode = "504", >>>> description >>>> = "Gateway Timeout", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implem >>>> e >>>> nt >>>> ation=ErrorMessage.class))) >>>> }) >>>> public Response createMessage( >>>> @HeaderParam("x-correlation-id") @NotNull >>>> @Size(min = 10, max = 36) @Parameter(description="ID of the >>>> transaction. Use this ID for log tracing and incident handling.") >>>> String xCorrelationId, >>>> @HeaderParam("Idempotency-Key") @Size(min = >>>> 10, max = 36) @Parameter(description="When retrying a failed call, >>>> the retry call should have the same Idempotency Key.") String >>>> idempotencyKey, >>>> @Multipart(value = "messageToSend”, >>>> required= >>>> true) MessageToSend messageToSend, >>>> @Multipart(value = "upfile1", required = >>>> false) Attachment upfile1Detail, >>>> @Multipart(value = "upfile2", required = >>>> false) Attachment upfile2Detail, >>>> @Multipart(value = "upfile3", required = >>>> false) Attachment upfile3Detail, >>>> @Multipart(value = "upfile4", required = >>>> false) Attachment upfile4Detail, >>>> @Multipart(value = "upfile5", required = >>>> false) Attachment upfile5Detail, >>>> @Multipart(value = "upfile6", required = >>>> false) Attachment upfile6Detail, >>>> @Multipart(value = "upfile7", required = >>>> false) Attachment upfile7Detail, >>>> @Multipart(value = "upfile8", required = >>>> false) Attachment upfile8Detail, >>>> @Multipart(value = "upfile9", required = >>>> false) Attachment upfile9Detail, >>>> @Multipart(value = "upfile10", required = >>>> false) Attachment upfile10Detail, >>>> @Multipart(value = "upfile11", required = >>>> false) Attachment upfile11Detail, >>>> @Multipart(value = "upfile12", required = >>>> false) Attachment upfile12Detail, >>>> @Multipart(value = "upfile13", required = >>>> false) Attachment upfile13Detail, >>>> @Multipart(value = "upfile14", required = >>>> false) Attachment upfile14Detail, >>>> @Multipart(value = "upfile15", required = >>>> false) Attachment upfile15Detail, >>>> @Multipart(value = "upfile16", required = >>>> false) Attachment upfile16Detail, >>>> @Multipart(value = "upfile17", required = >>>> false) Attachment upfile17Detail, >>>> @Multipart(value = "upfile18", required = >>>> false) Attachment upfile18Detail, >>>> @Multipart(value = "upfile19", required = >>>> false) Attachment upfile19Detail, >>>> @Multipart(value = "upfile20", required = >>>> false) Attachment upfile20Detail, >>>> @Multipart(value = "qrfile", required = >>>> false) Attachment qrfileDetail); If I now generate the swagger >>>> from this code (I modified the annotations in the generated code >>>> for using OAS v3 annotations through swagger-jaxrs2 v2.1.13 and I >>>> am using cxf-v3.5.6 having swagger-ui v4.18.2 generate the user >>>> interface) none of the upload files appears as request parameter, >>>> only the messageToSend is shown. >>>> Is the above signature for the method createMessage(...) incorrect? >>>> If I look at the generated openapi.json all the Attachment upFiles >>>> are missing from the specification? So is it a >>>> problem/short-coming(?) of the used software libraries, which then: >>>> · cxf-rt-rs-service-description-common-openapi v3.5.6 -> >>>> this library references swagger-jaxrs2 v2.1.13 · swagger-jaxrs2 >>>> v2.1.13 -> can I upgradethis >>>> to >>>> e.g. swagger-jaxrs2 v2.2.22 (latest) while retaining cxf v3.5.6? >>>> · ...another? >>>> Regards, >>>> J.P.