Routing¶
Beim Routing geht es um das Verteilen der eingehenden Serveranfragen, an die richtigen Anwendungsendpunkte. Endpunkte sind Einheiten zur Verarbeitung der Anfragen. Sie werden im Controller definiert und beim Starten der Anwendung registriert.
Grundlagen¶
Um das Ganze besser zu verstehen, werfen wir einen Blick auf den Aufbau einer solchen Serveranfrage.
GET /hello/vapor HTTP/1.1
host: vapor.codes
content-length: 0
Im Beispiel handelt es sich eine typische Anfrage an die URL /hello/vapor/
. Die selbe Anfrage wird erstellt, wenn wir im Browser folgenden Link aufrufen:
http://vapor.codes/hello/vapor
Anfragemethode¶
Ganz am Anfang der Serveranfrage steht die Anfragemethode. Wie im Beispiel, ist GET die meistgenutzte Methode, jedoch gibt es noch weitere Methoden, die zumeist in Verbindung mit CRUD zum Einsatz kommen.
Methode | Aktion | Beschreibung |
---|---|---|
GET | Read | Daten werden vom Server angefordert. |
POST | Create | Daten werden an den Server gesendet. |
PUT | Replace | Daten werden an den Server gesendet. |
PATCH | Update | Daten werden an den Server gesendet |
DELETE | Delete | Daten werden vom Server gelöscht. |
Anfragepfad¶
Auf die Methode folgt der Zielpfad der Anfrage. Die Zielpfad besteht aus einem Pfad und einer optionalen Zeichenabfolge ?
. Vapor benutzt beides um die Anfrage an den richtigen Endpunkt weiterzuleiten.
Endpunktmethoden¶
Vapor stellt alle Anfragemethoden als Methoden über die Application-Instanz zur Verfügung. Die Methoden akzeptieren einen oder mehrere Pfadangaben vom Typ String, die nachfolgend mit einem '/' getrennt zu einem Pfad zusammengestellt werden.
/// [controller.swift]
app.get("hello", "vapor") { req in
return "Hello, vapor!"
}
/// Die .on()-Variante ist ebenfalls möglich.
app.on(.GET, "hello", "vapor") { ... }
HTTP/1.1 200 OK
content-length: 13
content-type: text/plain; charset=utf-8
Hello, vapor!
Endpunktargumente¶
Durch das Voransetzen eines Doppelpunktes vor Parameterangabe zum Beispiel :name, erkennt Vapor, dass es sich hierbei um einem variablen Angabe handeln soll und somit jeder Parameter von Typ String akzeptiert wird. Über die Eigenschaft Parameters können wir nun auf den Angabe zugreifen.
/// [controllers.swift]
app.get("hello", ":name") { req -> String in
let name = req.parameters.get("name")!
return "Hello, \(name)!"
}
Wenn wir nun die Anfrage im Beispiel erneut ausführen, bekommen wir immer noch die selbe Antwort. Allerdings können wir nun hinter /hello/
einen beliebige Angabe machen, zum Beispiel /hello/swift
und bekommen folgende Antwort zurück:
GET /hello/swift HTTP/1.1
content-length: 0
HTTP/1.1 200 OK
content-length: 13
content-type: text/plain; charset=utf-8
Hello, swift!
Nachdem wir uns die Einführung angesehen haben, können wir uns den nachfolgenden Abschnitten widmen.
Endpunktdefinition¶
Methoden¶
Endpunkte können der Anwendung über die Instanz Application und den Methoden bekannt gemacht werden.
// [controllers.swift]
app.get("foo", "bar", "baz") { req in
...
}
Die Methode kann auch mit einem Rückgabewert versehen werden. Der Rückgabewert muss zwingend vom Typ ResponseEncodable sein.
app.get("foo") { req -> String in
return "bar"
}
// responds to OPTIONS /foo/bar/baz
app.on(.OPTIONS, "foo", "bar", "baz") { req in
...
}
Argumente¶
Die Endpunktmethoden akzeptieren eine Vielzahl von Argumenten. Es gibt vier Arten davon
Konstante¶
Bei der Konstante handelt es sich um eine statische Angabe. Somit werden von der Methode nur Anfragen mit einem übereinstimmen Pfad angenommen.
// responds to GET /foo/bar/baz
app.get("foo", "bar", "baz") { req in
...
}
Parameter¶
Beim Parameter handelt sich um eine variable Angabe. Somit werden jegliche Angaben entgegegen genommen. Dem Parameter muss ein Doppelpunkt vorangestellt werden. Die Deklaration nach dem Doppelpunkt steht für den Parameternamen. Mit dem Namen können wir später den Wert abfragen.
// responds to GET /foo/bar/baz
// responds to GET /foo/qux/baz
// ...
app.get("foo", ":bar", "baz") { req in
...
}
Wenn wir einen Parameter festlegen, wird der Wert der Angabe in der Eigenschaft Parameters auf der Instanz Request abgelegt und kann über den Namen abgefragt werden.
// responds to GET /hello/foo
// responds to GET /hello/bar
// ...
app.get("hello", ":name") { req -> String in
let name = req.parameters.get("name")!
return "Hello, \(name)!"
}
Tip
We can be sure that req.parameters.get
will never return nil
here since our route path includes :name
. However, if you are accessing route parameters in middleware or in code triggered by multiple routes, you will want to handle the possibility of nil
.
Tip
If you want to retrieve URL query params, e.g. /hello/?name=foo
you need to use Vapor's Content APIs to handle URL encoded data in the URL's query string. See Content
reference for more details.
req.parameters.get
also supports casting the parameter to LosslessStringConvertible
types automatically.
// responds to GET /number/42
// responds to GET /number/1337
// ...
app.get("number", ":x") { req -> String in
guard let int = req.parameters.get("x", as: Int.self) else {
throw Abort(.badRequest)
}
return "\(int) is a great number"
}
Sternchen¶
Für eine beliebige Angabe in einem Pfadabschnitt kann ein einfacher Asterisk angegeben werden. Es verhält sich ähnlich zu einer Parameterangabe, allerdings wird in diesem Fall der Wert verworfen.
// responds to GET /foo/bar/baz
// responds to GET /foo/qux/baz
// ...
app.get("foo", "*", "baz") { req in
...
}
Doppelsternchen¶
Für eine beliebige Angabe über mehrere Pfadabschnitte hinweg, können zwei Asterisk angegeben werden.
// responds to GET /foo/bar
// responds to GET /foo/bar/baz
// ...
app.get("foo", "**") { req in
...
}
Werte, die mit dem Pfadabschnitt übereinstimmen werden in der Eigenschaft parameters abgelegt und können mit der Methode getCatchall(:) abgerufen werden.
// responds to GET /hello/foo
// responds to GET /hello/foo/bar
// ...
app.get("hello", "**") { req -> String in
let name = req.parameters.getCatchall().joined(separator: " ")
return "Hello, \(name)!"
}
Verarbeitung¶
Wenn wir einen Endpunkt mit der Methode on(:) festlegen, können wir definieren, wie mit dem Inhalt umgegangen werden soll. Standardmäßig wird der Inhalt zwischengespeichert, bevor er an den Endpunkt übergeben wird. Das ist hilfreich, da Vapor den Anfrageinhalt nacheinander arbeiten kann, während zeitlgeich neue Anfrage eintreffen.
Vapor hat standardmäßig das Limit auf 16 KB festgelegt. Wir können allerdings den Wert mit der Eigenschaft Routes für alle Endpunkte überschreiben:
// Increases the streaming body collection limit to 500kb
app.routes.defaultMaxBodySize = "500kb"
Wenn das Limit erreicht wird, wird ein Fehler 413 (413 Payload Too Lage) ausgeworfen.
Der Wert kann aber auch für einen einzelnen Endpunkt abgeändert werden. Hierzu müssen wir der Methode beim Parameter body: einen Wert mitgeben. Wenn ein neuer Maximalwert mit angegeben wird, wird der Standardwert für den Endpunkt überschrieben.
// Collects streaming bodies (up to 1mb in size) before calling this route.
app.on(.POST, "listings", body: .collect(maxSize: "1mb")) { req in
// Handle request.
}
Bei leistungsintensivere Aufgaben, wie zum Beispiel das Hochladen von Dateien, kann das Zwischenspeichern des Inhalts den Arbeitsspeicher stark beanspruchen, daher ist es zu empfehlen, den Inhalt eher zu stream. In dem Fall bleibt req.body.data leer und die Daten müssen mit req.body.drain Stück für Stück entgegengenommen werden.
// Request body will not be collected into a buffer.
app.on(.POST, "upload", body: .stream) { req in
...
}
Groß- und Kleinschreibung¶
Grundsätzlich muss bei Endpunkten die Groß- und Kleinschreibung beachten werden. Bei Konstanten kann allerdings eine Ausnahme gemacht werden.
app.routes.caseInsensitive = true
Ansicht¶
Über die Eigenschaft all kann auf die Endpunkte zugegriffen werden.
print(app.routes.all) // [Route]
Vapor also ships with a routes
command that prints all available routes in an ASCII formatted table.
$ swift run App routes
+--------+----------------+
| GET | / |
+--------+----------------+
| GET | /hello |
+--------+----------------+
| GET | /todos |
+--------+----------------+
| POST | /todos |
+--------+----------------+
| DELETE | /todos/:todoID |
+--------+----------------+
Metadaten¶
Alle Endpunktmethoden liefern ein Objekt von Typ Route zurück. Damit können wir ihr Metadaten über die Sammlung userInfo mitgeben oder andere vordefinierte Methoden verwenden wie zum Beispiel, hinzufügen einer Beschreibung:
app.get("hello", ":name") { req in
...
}.description("says hello")
Endpunktgruppen¶
Endpunkte können zu Gruppen zusammengefasst werden. Der Name der Gruppe wird als Pfadabschnitt den enhaltenen Endpunkten vorangestellt.
let users = app.grouped("users")
// GET /users
users.get { req in
...
}
// POST /users
users.post { req in
...
}
// GET /users/:id
users.get(":id") { req in
let id = req.parameters.get("id")!
...
}
app.group("users") { users in
// GET /users
users.get { req in
...
}
// POST /users
users.post { req in
...
}
// GET /users/:id
users.get(":id") { req in
let id = req.parameters.get("id")!
...
}
}
Untergruppen¶
Gruppen können wiederum verschachteln werden.
app.group("users") { users in
// GET /users
users.get { ... }
// POST /users
users.post { ... }
users.group(":id") { user in
// GET /users/:id
user.get { ... }
// PATCH /users/:id
user.patch { ... }
// PUT /users/:id
user.put { ... }
}
}
Middleware¶
Gruppen können zudem mit Middlewares versehen werden.
app.get("fast-thing") { req in
...
}
app.group(RateLimitMiddleware(requestsPerMinute: 5)) { rateLimited in
rateLimited.get("slow-thing") { req in
...
}
}
app.post("login") { ... }
let auth = app.grouped(AuthMiddleware())
auth.get("dashboard") { ... }
auth.get("logout") { ... }
Weiterleitung¶
Für eine Weiterleitung kann es verschiedenste Gründe geben. Mit der Methode redirect(_:) über die Instanz Request können wir die Anfrage weiterleiten.
req.redirect(to: "/some/new/path")
/// redirect a page permanently
req.redirect(to: "/some/new/path", redirectType: .permanent)
Es gibt verschiedene Arten von Weiterleitungen:
Art | Statuscode | Beschreibung |
---|---|---|
permanent | 301 | Liefert einen Statuscode 301 zurück. |
normal | 303 | This is the default by Vapor and tells the client to follow the redirect with a GET request. |
temporary | 307 | This tells the client to preserve the HTTP method used in the request. |
To choose the proper redirection status code check out the full list |