Search code examples
swiftmultipartform-datavaporleaf

Image Upload in Vapor 3 using PostgreSQL


I'm following this guys Martin Lasek Tutorials and now i'm at "image upload". It seems that no one has the answer to the question "How do you upload images i Vapor 3"

Db connection is ok, all the other values are saved.

This is mye create method:

    func create(_ req: Request) throws -> Future<Response> {

    return try req.content.decode(Question.self).flatMap { question in
        return question.save(on: req).map { _ in

            return req.redirect(to: "/form")
        }
    }
}

and Model:

final class Question: PostgreSQLModel {

var id: Int?
var questionText: String
var answers: [String]
var theme: String?
var imageName: String?
var imageData: File?

init(id: Int? = nil, questionText: String, answers: [String], theme: String, imageName: String?, imageData: File?) {

    self.id = id
    self.questionText = questionText
    self.answers = answers
    self.theme = theme
    self.imageName = imageName
    self.imageData = imageData
}

}

and Leaf template:

<form action="/save" method="POST" enctype="multipart/form-data" id="upload-form">
<input type="file" accept="image/png,image/jpg" name="image">
<input class="btn btn-success btn-block" type="submit" value="Legg til">
</form>

I know a method for managing files is needed and the raw image bytes to,

But how do i get there?


Solution

  • This uses automatic decoding of the multi-part form:

    router.get("upload") {
        request -> Future<View> in
        return try request.view().render("upload")
    }
    
    struct ExampleUpload: Content {
        let document: File
    }
    
    // this saves the file into a Question
    router.post(ExampleUpload.self, at:"upload") {
        request, upload -> Future<HTTPResponseStatus> in
        let question = try Question()
        question.imageData = upload.document.data
        question.imageName = upload.document.filename
        return question.save(on:request).transform(to: HTTPResponseStatus.ok)
    }
    

    The upload.leaf file is:

    <form method="POST" enctype="multipart/form-data">
    <input type="file" name="document" />
    <input type="submit" value="Send" />
    </form>
    

    Using the type File enables the local filename of the uploaded file to be accessed as well as the file data. If you add in the rest of the Question fields to the ExampleUpload structure, you can use the route to capture the whole form's fields.