IOS (симулятор) -> Локальное изображение POST Vapor/png: Abort.413: слишком большая полезная нагрузка
Среда:
Vapor (v4.77+)
MacOS v13.4.1
Сервер является локальным для моего Mac: http://localhost:8080/upload.
Он работает одновременно с моим симулятором Xcode iOS.
Результат:
Начальная полезная нагрузка: PayloadModel(имя файла: «ToiletOnTheSea», contentType: «image/png», данные: 466220 байт) Код состояния: 413
Вот модель данных:
struct PayloadModel: Codable {
let filename: String
let contentType: String
let data: Data
enum CodingKeys: String, CodingKey {
case filename
case contentType
case data
}
}
Вот клиент iOS:
func uploadImageToServer() async {
let sampleImage = "ToiletOnTheSea"
guard let image = UIImage(named: sampleImage), let imageData = image.pngData() else {
showAlert("No Image", "I couldn't access \(sampleImage)")
print("Error converting image to data")
return
}
let payload = PayloadModel(filename: sampleImage, contentType: "image/png", data: imageData)
var request = URLRequest(url: serverURL)
request.httpMethod = "POST"
request.setValue("application/json", forHTTPHeaderField: "Content-Type")
print("Initial Payload: ", payload)
do {
let jsonData = try JSONEncoder().encode(payload)
request.httpBody = jsonData
let (data, response) = try await URLSession.shared.data(for: request)
if let httpResponse = response as? HTTPURLResponse {
print("Status Code: \(httpResponse.statusCode)")
if let serverData = try? JSONDecoder().decode(ServerResponse.self, from: data) {
let contentType = serverData.payload.contentType
let message = serverData.message
let filename = payload.filename
let count = payload.data.count
print("contentType: ", contentType)
print("filename: ", filename)
print("data count: ", count)
showAlert("Server Data", message)
} else {
showAlert("Server-Response Error", "Unable to decode server response")
}
// Process data if needed
} else {
print("Error: Unexpected response")
}
} catch {
showAlert("Error", "\(error)")
}
}
Вот соответствующий сервер Vapor v4 (в файле маршрута):
struct ServerResponse: Content {
let message: String
let payload: PayloadModel?
// Provide an initializer that takes the required parameters
init(message: String, payload: PayloadModel? = nil) {
self.message = message
self.payload = payload
}
// Implement the required initializer for the Decodable protocol
init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
message = try container.decode(String.self, forKey: .message)
payload = try container.decodeIfPresent(PayloadModel.self, forKey: .payload)
}
// Define coding keys to match the properties
enum CodingKeys: String, CodingKey {
case message
case payload
}
}
func storeImage(req: Request) async throws -> ServerResponse {
// let payload = try req.content.decode(PayloadModel.self)
let response = ServerResponse(message: "Holy Shit, Batman! I'm from storeImage!!", payload: nil)
return response
}
Я сокращаю коды до минимума, чтобы найти причину отклонения размера.
Я отправил (опубликовано) строку, которая прошла успешно без ошибки размера.
- Я думал, что существует ограничение на размер Vapor, поэтому установил для него больший размер, чем размер изображения, что не изменило результат. Поэтому я прокомментировал:
Вопрос: Почему проблема с размером, что ее вызывает и какое решение:
[ ПРЕДУПРЕЖДЕНИЕ ] Abort.413: Слишком большая полезная нагрузка [идентификатор запроса: 3224652A-0F82-43E9-AA7A-92496FB7781D]
1 ответ
Это связано с тем, что Vapor собирает тело из потока в память перед тем, как попасть в обработчик маршрута. Чтобы остановить это и сделать ваше приложение неуязвимым для DDOS Vapor, задайте ограничение по умолчанию.
Вы можете либо увеличить лимит глобально, либо установить его только для этого одного маршрута. Это в документах здесь