IOS (симулятор) -> Локальное изображение POST Vapor/png: Abort.413: слишком большая полезная нагрузка

Среда:
Vapor (v4.77+)
MacOS v13.4.1

Сценарий: попытка отправить изображение/png (466 КБ) POST на сервер.
Сервер является локальным для моего Mac: http://localhost:8080/upload.
Он работает одновременно с моим симулятором Xcode iOS.

info.plist Xcode:

Результат:

Начальная полезная нагрузка: 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
}

Я сокращаю коды до минимума, чтобы найти причину отклонения размера.
Я отправил (опубликовано) строку, которая прошла успешно без ошибки размера.

  1. Я думал, что существует ограничение на размер Vapor, поэтому установил для него больший размер, чем размер изображения, что не изменило результат. Поэтому я прокомментировал:

Вопрос: Почему проблема с размером, что ее вызывает и какое решение:

[ ПРЕДУПРЕЖДЕНИЕ ] Abort.413: Слишком большая полезная нагрузка [идентификатор запроса: 3224652A-0F82-43E9-AA7A-92496FB7781D]

1 ответ

Это связано с тем, что Vapor собирает тело из потока в память перед тем, как попасть в обработчик маршрута. Чтобы остановить это и сделать ваше приложение неуязвимым для DDOS Vapor, задайте ограничение по умолчанию.

Вы можете либо увеличить лимит глобально, либо установить его только для этого одного маршрута. Это в документах здесь

https://docs.vapor.codes/basics/routing/#body-streaming

Другие вопросы по тегам