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 (PDF documents) JPU> 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 anything about the JPU> file attachment, not is content, nor the appropriate Content-Disposition JPU> 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 sense that I JPU> used a test setup with just one file attachment. In 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 upfile1Detail is JPU> passed as a method parameter, who is going to close this InputStream at the JPU> 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 @Size(min = 10, max = 36) JPU> @Parameter(description="ID of the transaction. Use this ID for log tracing JPU> and incident handling.") String xCorrelationId, JPU> @HeaderParam("Idempotency-Key") @NotNull @Size(min = 10, max = 36) JPU> @Parameter(description="When retrying a failed call, the retry call should JPU> have the same Idempotency Key.") String idempotencyKey, JPU> @FormDataParam(value="messageToSend") @Parameter(required=true,schema = JPU> @Schema(implementation=MessageToSend.class)) @Multipart(value = JPU> "messageToSend", type="application/json", required= true) MessageToSend JPU> messageToSend, JPU> @FormDataParam(value="upfile1") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile1", JPU> type="application/octet-stream", required = false) Attachment JPU> 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 = 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 BufferedReader(new InputStreamReader(is))) JPU> { 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 = apiClient.createMessage1(xCorrelationId, idempotencyKey, JPU> mts, upfile1Detail); JPU> When running the API client test code and getting at '03.' The method: JPU> JAXRSUtils.writeMessageBody(List<WriterInterceptor> writers,Object JPU> entity,Class<?> type, Type genericType,Annotation[] annotations,MediaType JPU> mediaType,MultivaluedMap<String, Object> 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 MessageToSend JPU> - the second Attachment object contains an object of type Attachment JPU> This second object leads to a fatal error within the JPU> JAXRSUtils.writeMessageBody(...) method: JPU> okt 02, 2024 1:36:02 PM 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 org.apache.cxf.jaxrs.utils.JAXRSUtils JPU> logMessageHandlerProblem JPU> SEVERE: Problem with writing the data, class java.util.ArrayList, JPU> ContentType: multipart/form-data JPU> I modified the interface and implementation classes to use 'InputStream JPU> upfile1Detail' as type for the input parameter of the service method. And in JPU> this case my 'dummy' server implementation can read the file and save it to JPU> disk. JPU> -> Question-01: Is this the appropriate way to pass files (PDF documents) JPU> as attachment in a mutlipart/form-data request, or are there better ways? JPU> -> Question-02: When I activate logging, I can't see anything about the JPU> file attachment, not is content, nor the appropriate Content-Disposition JPU> settings? I get '--Content suppressed--', e.g.: JPU> [MAGDADOC] 2024-10-02 14:29:08,911 [main] INFO $--$ 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 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 sense that I JPU> used a test setup with just one file attachment. In 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 upfile1Detail is JPU> passed as a method parameter, who is going to close this InputStream at the JPU> server side? JPU> Regards, JPU> J.P. Urkens JPU> P.S.: Is there a migration document describing upgrading CXF-3.5.6 (my JPU> current version) to CXF-3.5.8 (latest JDK8 version). I'd like to know JPU> 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 since I was JPU> not depending on 'Jersey' components I skipped it. So apparently JPU> swagger-core has processing for externally defined annotations (like this JPU> FormDataParam in Jersey), indeed inside know-how. JPU> I included it in my project as io.swagger.v3.oas.annotations.FormDataParam, JPU> since it should somehow be included in the supported swagger annotations JPU> (maybe we should submit a request for it to the swagger-core team) and this JPU> 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>; 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 involved JPU> sadly). First of all, define such annotation anywhere in your codebase JPU> (where it dims appropriate): JPU> import java.lang.annotation.ElementType; import JPU> java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; JPU> import java.lang.annotation.Target; JPU> @Target({ElementType.PARAMETER, ElementType.METHOD, 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 */ public JPU> Response createMessage( JPU> @HeaderParam("x-correlation-id") @NotNull @Size(min = 10, max = 36) JPU> @Parameter(description="ID of the transaction. Use this ID for log tracing JPU> 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 should JPU> 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 = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile10", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile10Detail, JPU> @FormDataParam("upfile11") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile11", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile11Detail, JPU> @FormDataParam("upfile12") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile12", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile12Detail, JPU> @FormDataParam("upfile13") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile13", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile13Detail, JPU> @FormDataParam("upfile14") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile14", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile14Detail, JPU> @FormDataParam("upfile15") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile15", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile15Detail, JPU> @FormDataParam("upfile16") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile16", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile16Detail, JPU> @FormDataParam("upfile17") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile17", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile17Detail, JPU> @FormDataParam("upfile18") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile18", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile18Detail, JPU> @FormDataParam("upfile19") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile19", JPU> type="application/octet-stream", required = false) Attachment JPU> upfile19Detail, JPU> @FormDataParam("upfile20") @Parameter(schema = @Schema(type = JPU> "string", format = "binary")) @Multipart(value = "upfile20", JPU> type="application/octet-stream", required = false) Attachment JPU> 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 bit large JPU> 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, ebox) JPU> and delivery\ JPU> \ method (registered or normal) of your choice. More than 6 upfiles JPU> 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 tracing and JPU> 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 should JPU> 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 from JPU> Jersey but has special treatment in Swagger Core (but, likely, no JPU> 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/multipar >> 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(sche >>> 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(implemen >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "401", >>> description = "Invalid authorization", content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implemen >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "500", >>> description = "Unexpected Server Error", content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implemen >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "502", >>> description = "Bad Gateway", >>> content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implemen >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "503", >>> description = "Service unavailable", >>> content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implemen >>> t >>> ation=ErrorMessage.class))), >>> @ApiResponse( >>> responseCode = "504", >>> description = "Gateway Timeout", >>> content = >>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(implemen >>> 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(sch >>>> 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(impleme >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> responseCode >>>> = "401", >>>> description >>>> = "Invalid authorization", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> responseCode >>>> = "500", >>>> description >>>> = "Unexpected Server Error", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> responseCode >>>> = "502", >>>> description >>>> = "Bad Gateway", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> responseCode >>>> = "503", >>>> description >>>> = "Service unavailable", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>>> nt >>>> ation=ErrorMessage.class))), >>>> @ApiResponse( >>>> responseCode >>>> = "504", >>>> description >>>> = "Gateway Timeout", >>>> content = >>>> @Content(mediaType=MediaType.APPLICATION_JSON,schema=@Schema(impleme >>>> 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.