6.9 KiB
Modelbindung
Mit der Modelbindung können wir den Inhalt oder die Zeichenfolge einer Serveranfrage an einen vordefiniertes Datenobjekt binden.
Grundlagen
Um das Binden besser zu verstehen, werfen wir einen kurzen Blick auf den Aufbau einer solchen Serveranfrage.
POST /greeting HTTP/1.1
content-type: application/json
content-length: 18
{"hello": "world"}
Die Angabe content-type in der Kopfzeile gibt Aufschluss über die Art des Inhaltes der Anfrage. Vapor nutzt die Angabe um den richtigen Kodierer zum Binden zu finden.
Im Beispiel können wir erkennen, dass es sich bei dem Inhalt um JSON-Daten handelt.
Binden des Inhalts
Zum Binden des Inhalts müssen wir zuerst eine Struktur vom Typ Codable anlegen. Indem wir das Objekt mit Vapor's Protokoll Content versehen, werden neben den eigentlichen Bindungsmethoden, der Typ mitvererbt.
struct Greeting: Content {
var hello: String
}
Über die Eigenschaft content können wir anschließend die Methode decode(_:) verwenden.
app.post("greeting") { req in
let greeting = try req.content.decode(Greeting.self)
print(greeting.hello) // "world"
return HTTPStatus.ok
}
Die Methode decode(_:) benutzt die entsprechende Angabe in der Serveranfrage um den passenden Kodierer aufzurufen.
Sollte kein passender Kodierer gefunden werden oder die Anfrage keine Angaben zum Inhalt besitzen, wird der Fehler 415 (415 Unsupported Media Type) zurückgeliefert.
Unterstützte Medien
Folgende Medien werden von Vapor standardmäßig unterstützt:
| Bezeichnung | Feldwert | Typ |
|---|---|---|
| JSON | application/json | .json |
| Multipart | multipart/form-data | .formData |
| URL-Encoded Form | application/x-www-form-urlencoded | .urlEncodedForm |
| Plaintext | text/plain | .plainText |
| HTML | text/html | .html |
Codable unterstützt leider nicht alle Medien.
Binden der Zeichenfolge
GET /hello?name=Vapor HTTP/1.1
content-length: 0
Ähnlich wie beim Binden des Inhalts müssen wir für das Binden der Zeichenfolge eine Struktur anlegen und es mit dem Protokoll Content versehen.
Zusätzlich müssen wir die Eigenschaft name als optional deklarieren, da Parameter in einer Zeichenfolge immer optional sind.
struct Hello: Content {
var name: String?
}
app.get("hello") { req -> String in
let hello = try req.query.decode(Hello.self)
return "Hello, \(hello.name ?? "Anonymous")"
}
Zudem können wir auch Einzelwerte aus der Zeichenabfolge abrufen:
app.get("hello") { req -> String in
let name: String? = req.query["name"]
...
}
Hooks
Vapor ruft automatisch jeweils die beiden Methoden beforeEncode und afterDecode eines Objektes von Typ Content auf.
Die Methoden sind standardmäßig funktionslos, können aber im Bedarfsfall überschrieben werden.
// Runs before this Content is encoded. `mutating` is only required for structs, not classes.
mutating func beforeEncode() throws {
guard
let name = self.name?.trimmingCharacters(in: .whitespacesAndNewlines),
!name.isEmpty
else {
throw Abort(.badRequest, reason: "Name must not be empty.")
}
self.name = name
}
// Runs after this Content is decoded. `mutating` is only required for structs, not classes.
mutating func afterDecode() throws {
self.name = self.name?.trimmingCharacters(in: .whitespacesAndNewlines)
if let name = self.name, name.isEmpty {
throw Abort(.badRequest, reason: "Name must not be empty.")
}
}
Standard überschreiben
Vapor's Standardkodierer kann global oder situationsabhängig überschrieben werden.
Global
Für eine globale Verwendung eines eigenen Kodierer müssen wir ihn der ContentConfiguration.global mitgeben.
// create a new JSON encoder that uses unix-timestamp dates
let encoder = JSONEncoder()
encoder.dateEncodingStrategy = .secondsSince1970
// override the global encoder used for the `.json` media type
ContentConfiguration.global.use(encoder: encoder, for: .json)
Situationsabhängig
Wir können aber auch den Bindungsmethoden abhängig von der Situation einen Kodierer mitgeben.
// create a new JSON decoder that uses unix-timestamp dates
let decoder = JSONDecoder()
decoder.dateDecodingStrategy = .secondsSince1970
// decodes Hello struct using custom decoder
let hello = try req.content.decode(Hello.self, using: decoder)
Benutzerdefinierte Kodierer
Kodierer für Inhalt
Vapor hat die folgenden zwei Protokolle zum Binden von Inhalt vordefiniert.
public protocol ContentEncoder {
func encode<E>(_ encodable: E, to body: inout ByteBuffer, headers: inout HTTPHeaders) throws
where E: Encodable
}
public protocol ContentDecoder {
func decode<D>(_ decodable: D.Type, from body: ByteBuffer, headers: HTTPHeaders) throws -> D
where D: Decodable
}
Indem wir einen unseren eigenen Kodierer mit diese beiden Protokolle versehen, kann er von ContentConfiguration entgegengenommen werden.
Kodierer für Zeichenfolge
Für das Binden einer Zeichenabfolge hat Vapor die folgenden zwei Protokolle vordefiniert.
public protocol URLQueryDecoder {
func decode<D>(_ decodable: D.Type, from url: URI) throws -> D
where D: Decodable
}
public protocol URLQueryEncoder {
func encode<E>(_ encodable: E, to url: inout URI) throws
where E: Encodable
}
ResponseEncodable
Another approach involves implementing ResponseEncodable on your types. Consider this trivial HTML wrapper type:
struct HTML {
let value: String
}
Then its ResponseEncodable implementation would look like this:
extension HTML: ResponseEncodable {
public func encodeResponse(for request: Request) -> EventLoopFuture<Response> {
var headers = HTTPHeaders()
headers.add(name: .contentType, value: "text/html")
return request.eventLoop.makeSucceededFuture(.init(
status: .ok, headers: headers, body: .init(string: value)
))
}
}
If you're using async/await you can use AsyncResponseEncodable:
extension HTML: AsyncResponseEncodable {
public func encodeResponse(for request: Request) async throws -> Response {
var headers = HTTPHeaders()
headers.add(name: .contentType, value: "text/html")
return .init(status: .ok, headers: headers, body: .init(string: value))
}
}
Note that this allows customizing the Content-Type header. See HTTPHeaders reference for more details.
You can then use HTML as a response type in your routes:
app.get { _ in
HTML(value: """
<html>
<body>
<h1>Hello, World!</h1>
</body>
</html>
""")
}