最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

java - Why @Encode annotation isn't applied to multiple parameters at Controller level? - Stack Overflow

programmeradmin1浏览0评论

I have a problem with OpenAPI curl generation.

I have to create POST method with three parameters: application/json, text/plain and multipart/form-data medium types. So curl that is going to be generated by OpenAPI is supposed to be like:

curl -X 'POST' \
  'http://localhost:8080/test/third' \
  -H 'accept: */*' \
  -H 'Content-Type: multipart/form-data' \
  -F 'file=@test_file.docx;type=multipart/form-data' \
  -F 'dto={"name":"string","description":"string"};type=application/json' \
  -F 'str=some_test_value;type=text/plain' \

But it doesn't work.

Controller:

package .example.springbased.web;

import io.swagger.v3.oas.annotations.Parameter;
import io.swagger.v3.oas.annotations.media.Content;
import io.swagger.v3.oas.annotations.media.Encoding;
import io.swagger.v3.oas.annotations.parameters.RequestBody;
import .springframework.http.MediaType;
import .springframework.http.ResponseEntity;
import .springframework.web.bind.annotation.PostMapping;
import .springframework.web.bind.annotation.RequestMapping;
import .springframework.web.bind.annotation.RequestPart;
import .springframework.web.bind.annotation.RestController;
import .springframework.web.multipart.MultipartFile;

@RestController
@RequestMapping("/test")
public class TestController {

    @PostMapping(value = "/second", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
    public ResponseEntity<String> test(
        @RequestBody(content = @Content(encoding = @Encoding(name = "dto", contentType = MediaType.APPLICATION_JSON_VALUE)))
        @Parameter(description = "An extra JSON payload sent with file")
        @RequestPart("dto")
        DTO dto,
        @RequestBody(content = @Content(encoding = @Encoding(name = "file", contentType = MediaType.MULTIPART_FORM_DATA_VALUE)))
        @Parameter(required = true)
        @RequestPart("file") MultipartFile file){
        return ResponseEntity.ok("Request Accepted: " + dto.toString());
    }

    @PostMapping(value ="/first", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
    public ResponseEntity<String> uploadMultipartWithBody(
        @RequestBody(content = @Content(encoding = @Encoding(name = "file", contentType = MediaType.MULTIPART_FORM_DATA_VALUE)))
        @Parameter(required = true)
        @RequestPart("file") MultipartFile file,
        @RequestBody(content = @Content(encoding = @Encoding(name = "dto", contentType = MediaType.APPLICATION_JSON_VALUE)))
        @Parameter(description = "An extra JSON payload sent with file")
        @RequestPart("dto")
        DTO dto){
        return ResponseEntity.ok("Request Accepted: " + dto.toString());
    }

    @PostMapping(value ="/third", consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
    public ResponseEntity<String> threeParams(
        @RequestBody(content = @Content(encoding = @Encoding(name = "file", contentType = MediaType.MULTIPART_FORM_DATA_VALUE)))
        @Parameter(required = true)
        @RequestPart("file") MultipartFile file,
        @RequestBody(content = @Content(encoding = @Encoding(name = "dto", contentType = MediaType.APPLICATION_JSON_VALUE)))
        @Parameter(description = "An extra JSON payload sent with file")
        @RequestPart("dto")
        DTO dto,
        @RequestBody(content = @Content(encoding = @Encoding(name = "str", contentType = MediaType.TEXT_PLAIN_VALUE)))
        @RequestPart("str")
        String str){
        return ResponseEntity.ok("Request Accepted: " + dto.toString());
    }
}

Generated OpenAPI:

{
  "openapi": "3.1.0",
  "info": {
    "title": "OpenAPI definition",
    "version": "v0"
  },
  "servers": [
    {
      "url": "http://localhost:8080",
      "description": "Generated server url"
    }
  ],
  "paths": {
    "/test/third": {
      "post": {
        "tags": [
          "test-controller"
        ],
        "operationId": "threeParams",
        "requestBody": {
          "content": {
            "multipart/form-data": {
              "schema": {
                "type": "object",
                "properties": {
                  "file": {
                    "type": "string",
                    "format": "binary"
                  },
                  "dto": {
                    "$ref": "#/components/schemas/DTO"
                  },
                  "str": {
                    "type": "string"
                  }
                },
                "required": [
                  "dto",
                  "file",
                  "str"
                ]
              },
              "encoding": {
                "file": {
                  "contentType": "multipart/form-data"
                }
              }
            }
          }
        },
        "responses": {
          "200": {
            "description": "OK",
            "content": {
              "*/*": {
                "schema": {
                  "type": "string"
                }
              }
            }
          }
        }
      }
    },
    "/test/second": {
      "post": {
        "tags": [
          "test-controller"
        ],
        "operationId": "test",
        "requestBody": {
          "content": {
            "multipart/form-data": {
              "schema": {
                "type": "object",
                "properties": {
                  "dto": {
                    "$ref": "#/components/schemas/DTO"
                  },
                  "file": {
                    "type": "string",
                    "format": "binary"
                  }
                },
                "required": [
                  "dto",
                  "file"
                ]
              },
              "encoding": {
                "dto": {
                  "contentType": "application/json"
                }
              }
            }
          }
        },
        "responses": {
          "200": {
            "description": "OK",
            "content": {
              "*/*": {
                "schema": {
                  "type": "string"
                }
              }
            }
          }
        }
      }
    },
    "/test/first": {
      "post": {
        "tags": [
          "test-controller"
        ],
        "operationId": "uploadMultipartWithBody",
        "requestBody": {
          "content": {
            "multipart/form-data": {
              "schema": {
                "type": "object",
                "properties": {
                  "file": {
                    "type": "string",
                    "format": "binary"
                  },
                  "dto": {
                    "$ref": "#/components/schemas/DTO"
                  }
                },
                "required": [
                  "dto",
                  "file"
                ]
              },
              "encoding": {
                "file": {
                  "contentType": "multipart/form-data"
                }
              }
            }
          }
        },
        "responses": {
          "200": {
            "description": "OK",
            "content": {
              "*/*": {
                "schema": {
                  "type": "string"
                }
              }
            }
          }
        }
      }
    }
  },
  "components": {
    "schemas": {
      "DTO": {
        "type": "object",
        "description": "An extra JSON payload sent with file",
        "properties": {
          "name": {
            "type": "string"
          },
          "description": {
            "type": "string"
          }
        }
      }
    }
  }
}

As you can see even if I specify

@RequestBody(content = @Content(encoding = @Encoding(name = "...", contentType = MediaType....))

for each parameter it takes only the first one that's specified for the first parameter.

Why? What I need to do to make it works?

Dependencies that I use with gradle + java 21:

dependencies {
    implementation '.springframework.boot:spring-boot-starter-web'
    implementation '.springdoc:springdoc-openapi-starter-webmvc-ui:2.8.6'
    implementation '.springframework.boot:spring-boot-devtools'
}

Documentation that I followed:

发布评论

评论列表(0)

  1. 暂无评论