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.


Reply via email to