feat: Adds view route parsing tests.

This commit is contained in:
2025-01-20 10:51:07 -05:00
parent 2de85ed758
commit affd9b5d81
14 changed files with 552 additions and 59 deletions

View File

@@ -24,9 +24,9 @@ struct EmployeeForm: HTML {
employee == nil employee == nil
? .hx.swap(.beforeEnd.transition(true).swap("0.5s")) ? .hx.swap(.beforeEnd.transition(true).swap("0.5s"))
: .hx.swap(.outerHTML.transition(true).swap("0.5s")), : .hx.swap(.outerHTML.transition(true).swap("0.5s")),
.custom( .hx.on(
name: "hx-on::after-request", .afterRequest,
value: "if (event.detail.successful) toggleContent('float'); window.location.href='/employees';" .ifSuccessful(.toggleContent(.float), .setWindowLocation(to: .employee(.index)))
) )
) { ) {
div(.class("row")) { div(.class("row")) {
@@ -53,7 +53,7 @@ struct EmployeeForm: HTML {
.attributes( .attributes(
.hx.confirm("Are you sure you want to delete this employee?"), .hx.confirm("Are you sure you want to delete this employee?"),
.hx.delete(route: .employee(.delete(id: employee.id))), .hx.delete(route: .employee(.delete(id: employee.id))),
.hx.target("#employee_\(employee.id)"), .hx.target(.id(.employee(.row(id: employee.id)))),
.hx.swap(.outerHTML.transition(true).swap("1s")) .hx.swap(.outerHTML.transition(true).swap("1s"))
) )
} }

View File

@@ -28,10 +28,6 @@ struct UserForm: HTML, Sendable {
.afterRequest, .afterRequest,
.ifSuccessful(.resetForm, .toggleContent(.float)) .ifSuccessful(.resetForm, .toggleContent(.float))
) )
// .custom(
// name: "hx-on::after-request",
// value: "if(event.detail.successful) this.reset(); toggleContent('float');"
// )
) { ) {
if case let .login(next) = context, let next { if case let .login(next) = context, let next {
input(.type(.hidden), .name("next"), .value(next)) input(.type(.hidden), .name("next"), .value(next))

View File

@@ -31,7 +31,7 @@ public struct Employee: Codable, Equatable, Identifiable, Sendable {
} }
public extension Employee { public extension Employee {
struct Create: Codable, Sendable { struct Create: Codable, Sendable, Equatable {
public let firstName: String public let firstName: String
public let lastName: String public let lastName: String
public let active: Bool? public let active: Bool?
@@ -47,7 +47,7 @@ public extension Employee {
} }
} }
struct Update: Codable, Sendable { struct Update: Codable, Sendable, Equatable {
public let firstName: String? public let firstName: String?
public let lastName: String? public let lastName: String?
public let active: Bool? public let active: Bool?

View File

@@ -41,7 +41,7 @@ public struct PurchaseOrder: Codable, Equatable, Identifiable, Sendable {
public extension PurchaseOrder { public extension PurchaseOrder {
struct Create: Codable, Sendable { struct Create: Codable, Sendable, Equatable {
public let id: Int? public let id: Int?
public let workOrder: Int? public let workOrder: Int?
@@ -73,7 +73,7 @@ public extension PurchaseOrder {
} }
} }
struct CreateIntermediate: Codable, Sendable { struct CreateIntermediate: Codable, Sendable, Equatable {
public let id: Int? public let id: Int?
public let workOrder: Int? public let workOrder: Int?
@@ -115,7 +115,7 @@ public extension PurchaseOrder {
} }
} }
enum SearchContext: Sendable { enum SearchContext: Sendable, Equatable {
case customer(String) case customer(String)
case vendor(VendorBranch.ID) case vendor(VendorBranch.ID)
case employee(Employee.ID) case employee(Employee.ID)

View File

@@ -2,7 +2,8 @@ import CasePathsCore
import Foundation import Foundation
@preconcurrency import URLRouting @preconcurrency import URLRouting
public enum ViewRoute: Sendable { // swiftlint:disable file_length
public enum ViewRoute: Sendable, Equatable {
case employee(EmployeeRoute) case employee(EmployeeRoute)
case login(LoginRoute) case login(LoginRoute)
@@ -24,7 +25,7 @@ public enum ViewRoute: Sendable {
public extension ViewRoute { public extension ViewRoute {
enum EmployeeRoute: Sendable { enum EmployeeRoute: Sendable, Equatable {
case create(Employee.Create) case create(Employee.Create)
case delete(id: Employee.ID) case delete(id: Employee.ID)
case form case form
@@ -43,15 +44,11 @@ public extension ViewRoute {
FormData { FormData {
Field("firstName", .string) Field("firstName", .string)
Field("lastName", .string) Field("lastName", .string)
Field("active") { Optionally { Bool.parser() } } Optionally { Field("active") { Bool.parser() } }
} }
.map(.memberwise(Employee.Create.init)) .map(.memberwise(Employee.Create.init))
} }
} }
Route(.case(Self.index)) {
Path { rootPath }
Method.get
}
Route(.case(Self.delete(id:))) { Route(.case(Self.delete(id:))) {
Path { rootPath; Employee.ID.parser() } Path { rootPath; Employee.ID.parser() }
Method.delete Method.delete
@@ -60,14 +57,22 @@ public extension ViewRoute {
Path { rootPath; Employee.ID.parser() } Path { rootPath; Employee.ID.parser() }
Method.get Method.get
} }
Route(.case(Self.form)) {
Path { rootPath; "create" }
Method.get
}
Route(.case(Self.index)) {
Path { rootPath }
Method.get
}
Route(.case(Self.update(id:updates:))) { Route(.case(Self.update(id:updates:))) {
Path { rootPath; Employee.ID.parser() } Path { rootPath; Employee.ID.parser() }
Method.put Method.put
Body { Body {
FormData { FormData {
Field("firstName") { Optionally { CharacterSet.alphanumerics.map(.string) } } Optionally { Field("firstName") { CharacterSet.alphanumerics.map(.string) } }
Field("lastName") { Optionally { CharacterSet.alphanumerics.map(.string) } } Optionally { Field("lastName") { CharacterSet.alphanumerics.map(.string) } }
Field("active") { Optionally { Bool.parser() } } Optionally { Field("active") { Bool.parser() } }
} }
.map(.memberwise(Employee.Update.init)) .map(.memberwise(Employee.Update.init))
} }
@@ -85,7 +90,7 @@ public extension ViewRoute {
public extension ViewRoute { public extension ViewRoute {
enum LoginRoute: Sendable { enum LoginRoute: Sendable, Equatable {
case index(next: String? = nil) case index(next: String? = nil)
case post(Request) case post(Request)
@@ -96,8 +101,10 @@ public extension ViewRoute {
Method.get Method.get
Path { rootPath } Path { rootPath }
Query { Query {
Field("next", default: nil) { Optionally {
Optionally { CharacterSet.urlPathAllowed.map(.string) } Field("next", default: nil) {
CharacterSet.urlPathAllowed.map(.string)
}
} }
} }
} }
@@ -108,8 +115,10 @@ public extension ViewRoute {
FormData { FormData {
Field("username", .string) Field("username", .string)
Field("password", .string) Field("password", .string)
Field("next", default: nil) { Optionally {
Optionally { CharacterSet.urlPathAllowed.map(.string) } Field("next", default: nil) {
CharacterSet.urlPathAllowed.map(.string)
}
} }
} }
.map(.memberwise(Request.init)) .map(.memberwise(Request.init))
@@ -121,12 +130,18 @@ public extension ViewRoute {
public let username: String public let username: String
public let password: String public let password: String
public let next: String? public let next: String?
public init(username: String, password: String, next: String? = nil) {
self.username = username
self.password = password
self.next = next
}
} }
} }
} }
public extension ViewRoute { public extension ViewRoute {
enum PurchaseOrderRoute: Sendable { enum PurchaseOrderRoute: Sendable, Equatable {
case create(PurchaseOrder.Create) case create(PurchaseOrder.Create)
case delete(id: PurchaseOrder.ID) case delete(id: PurchaseOrder.ID)
case form case form
@@ -143,11 +158,11 @@ public extension ViewRoute {
Method.post Method.post
Body { Body {
FormData { FormData {
Field("id") { Optionally { PurchaseOrder.ID.parser() } } Optionally { Field("id") { PurchaseOrder.ID.parser() } }
Field("workOrder") { Optionally { Int.parser() } } Optionally { Field("workOrder") { Int.parser() } }
Field("materials", .string) Field("materials", .string)
Field("customer", .string) Field("customer", .string)
Field("truckStock") { Optionally { Bool.parser() } } Optionally { Field("truckStock") { Bool.parser() } }
Field("createdByID") { User.ID.parser() } Field("createdByID") { User.ID.parser() }
Field("createdForID") { Employee.ID.parser() } Field("createdForID") { Employee.ID.parser() }
Field("vendorBranchID") { VendorBranch.ID.parser() } Field("vendorBranchID") { VendorBranch.ID.parser() }
@@ -185,9 +200,9 @@ public extension ViewRoute {
} }
} }
public enum Search: Sendable { public enum Search: Sendable, Equatable {
case index(context: Context? = nil, table: Bool? = nil) case index(context: Context? = nil, table: Bool? = nil)
case search(Request) case request(Request)
static let rootPath = Path { "purchase-orders"; "search" } static let rootPath = Path { "purchase-orders"; "search" }
@@ -196,24 +211,33 @@ public extension ViewRoute {
rootPath rootPath
Method.get Method.get
Query { Query {
Field("context", default: .employee) { Optionally { Search.Context.parser() } } Optionally { Field("context", default: .employee) { Search.Context.parser() } }
Field("table", default: nil) { Optionally { Bool.parser() } } Optionally { Field("table", default: nil) { Bool.parser() } }
} }
} }
Route(.case(Search.search)) { Route(.case(Search.request)) {
rootPath rootPath
Method.post Method.post
Body(.json(Search.Request.self)) Body {
FormData {
Field("context") { Context.parser() }
Optionally { Field("createdForID") { Employee.ID.parser() } }
Optionally { Field("customerSearch", .string) }
Optionally { Field("vendorBranchID") { VendorBranch.ID.parser() } }
}
.map(.memberwise(Request.init))
}
} }
} }
public enum Context: String, Codable, CaseIterable, Sendable { public enum Context: String, Codable, CaseIterable, Equatable, Sendable {
case employee case employee
case customer case customer
case vendor case vendor
} }
public struct Request: Codable, Sendable { // TODO: Create a validation or potentially turn this into an enum with the valid states.
public struct Request: Codable, Equatable, Sendable {
public let context: Context public let context: Context
public let createdForID: Employee.ID? public let createdForID: Employee.ID?
public let customerSearch: String? public let customerSearch: String?
@@ -231,14 +255,13 @@ public extension ViewRoute {
self.vendorBranchID = vendorBranchID self.vendorBranchID = vendorBranchID
} }
} }
} }
} }
} }
public extension ViewRoute { public extension ViewRoute {
enum SelectContext: String, Codable, Sendable, CaseIterable { enum SelectContext: String, Codable, Equatable, Sendable, CaseIterable {
case purchaseOrderForm case purchaseOrderForm
case purchaseOrderSearch case purchaseOrderSearch
} }
@@ -246,7 +269,7 @@ public extension ViewRoute {
} }
public extension ViewRoute { public extension ViewRoute {
enum UserRoute: Sendable { enum UserRoute: Sendable, Equatable {
case create(User.Create) case create(User.Create)
case delete(id: User.ID) case delete(id: User.ID)
case form case form
@@ -260,7 +283,15 @@ public extension ViewRoute {
Route(.case(Self.create)) { Route(.case(Self.create)) {
Path { rootPath } Path { rootPath }
Method.post Method.post
Body(.json(User.Create.self)) Body {
FormData {
Field("username", .string)
Field("email", .string)
Field("password", .string)
Field("confirmPassword", .string)
}
.map(.memberwise(User.Create.init))
}
} }
Route(.case(Self.delete(id:))) { Route(.case(Self.delete(id:))) {
Path { rootPath; User.ID.parser() } Path { rootPath; User.ID.parser() }
@@ -283,15 +314,11 @@ public extension ViewRoute {
Method.patch Method.patch
Body { Body {
FormData { FormData {
Field("username") { Optionally { Field("username") {
Optionally { CharacterSet.alphanumerics.map(.string) } CharacterSet.alphanumerics.map(.string)
} }
Field("email") {
Optionally {
// TODO: Not sure if this is correct.
Rest().map(.string)
}
} }
Optionally { Field("email", .string) }
} }
.map(.memberwise(User.Update.init)) .map(.memberwise(User.Update.init))
} }
@@ -302,7 +329,7 @@ public extension ViewRoute {
public extension ViewRoute { public extension ViewRoute {
enum VendorRoute: Sendable { enum VendorRoute: Sendable, Equatable {
case create(Vendor.Create) case create(Vendor.Create)
case delete(id: Vendor.ID) case delete(id: Vendor.ID)
case form case form
@@ -356,7 +383,7 @@ public extension ViewRoute {
public extension ViewRoute { public extension ViewRoute {
enum VendorBranchRoute: Sendable { enum VendorBranchRoute: Sendable, Equatable {
case create(VendorBranch.Create) case create(VendorBranch.Create)
case delete(id: VendorBranch.ID) case delete(id: VendorBranch.ID)
case select(context: ViewRoute.SelectContext) case select(context: ViewRoute.SelectContext)
@@ -387,3 +414,5 @@ public extension ViewRoute {
} }
} }
} }
// swiftlint:enable file_length

View File

@@ -26,7 +26,7 @@ public struct User: Codable, Equatable, Identifiable, Sendable {
public extension User { public extension User {
struct Create: Codable, Sendable { struct Create: Codable, Sendable, Equatable {
public let username: String public let username: String
public let email: String public let email: String
public let password: String public let password: String
@@ -45,7 +45,7 @@ public extension User {
} }
} }
struct Login: Codable, Sendable { struct Login: Codable, Sendable, Equatable {
public let username: String? public let username: String?
public let email: String? public let email: String?
public let password: String public let password: String
@@ -80,6 +80,11 @@ public extension User {
struct Update: Codable, Equatable, Sendable { struct Update: Codable, Equatable, Sendable {
public let username: String? public let username: String?
public let email: String? public let email: String?
public init(username: String?, email: String?) {
self.username = username
self.email = email
}
} }
} }

View File

@@ -25,7 +25,7 @@ public struct Vendor: Codable, Equatable, Identifiable, Sendable {
public extension Vendor { public extension Vendor {
struct Create: Codable, Sendable { struct Create: Codable, Sendable, Equatable {
public let name: String public let name: String
public init(name: String) { public init(name: String) {
@@ -33,7 +33,7 @@ public extension Vendor {
} }
} }
struct Update: Codable, Sendable { struct Update: Codable, Sendable, Equatable {
public let name: String public let name: String
public init(name: String) { public init(name: String) {

View File

@@ -24,7 +24,7 @@ public struct VendorBranch: Codable, Equatable, Identifiable, Sendable {
} }
public extension VendorBranch { public extension VendorBranch {
struct Create: Codable, Sendable { struct Create: Codable, Sendable, Equatable {
public let name: String public let name: String
public let vendorID: Vendor.ID public let vendorID: Vendor.ID
@@ -56,7 +56,7 @@ public extension VendorBranch {
} }
} }
struct Update: Codable, Sendable { struct Update: Codable, Sendable, Equatable {
public let name: String? public let name: String?
public init(name: String?) { public init(name: String?) {

View File

@@ -0,0 +1,101 @@
import Dependencies
import Foundation
import SharedModels
import Testing
import URLRouting
@Suite("EmployeeViewRouteTests")
struct EmployeeViewRouteTests {
let router = ViewRoute.router
@Test
func employeeCreate() throws {
var request = URLRequestData(
method: "POST",
path: "/employees",
body: .init("firstName=Blob&lastName=Esquire&active=true".utf8)
)
let route = try router.parse(&request)
#expect(
route == .employee(.create(.init(firstName: "Blob", lastName: "Esquire", active: true)))
)
}
@Test
func employeeDelete() throws {
let id = UUID(0)
var request = URLRequestData(
method: "DELETE",
path: "/employees/\(id)"
)
let route = try router.parse(&request)
#expect(
route == .employee(.delete(id: id))
)
}
@Test
func employeeForm() throws {
var request = URLRequestData(
method: "GET",
path: "/employees/create"
)
let route = try router.parse(&request)
#expect(
route == .employee(.form)
)
}
@Test
func employeeGet() throws {
let id = UUID(0)
var request = URLRequestData(
method: "GET",
path: "/employees/\(id)"
)
let route = try router.parse(&request)
#expect(
route == .employee(.get(id: id))
)
}
@Test
func employeeIndex() throws {
var request = URLRequestData(
method: "GET",
path: "/employees"
)
let route = try router.parse(&request)
#expect(
route == .employee(.index)
)
}
@Test
func employeeUpdate() throws {
let id = UUID(0)
var request = URLRequestData(
method: "PUT",
path: "/employees/\(id)",
body: .init("firstName=Blob&lastName=Esquire&active=true".utf8)
)
let route = try router.parse(&request)
#expect(
route == .employee(.update(
id: id,
updates: .init(firstName: "Blob", lastName: "Esquire", active: true)
))
)
}
@Test
func employeeSelect() throws {
var request = URLRequestData(
method: "GET",
path: "/employees/select",
query: ["context": ["purchaseOrderForm"]]
)
let route = try router.parse(&request)
#expect(route == .employee(.select(context: .purchaseOrderForm)))
}
}

View File

@@ -0,0 +1,32 @@
import Dependencies
import Foundation
import SharedModels
import Testing
import URLRouting
@Suite("LoginViewRouteTests")
struct LoginViewRouteTests {
let router = ViewRoute.router
@Test
func get() throws {
var request = URLRequestData(
method: "GET",
path: "/login",
query: ["next": ["/users"]]
)
let route = try router.parse(&request)
#expect(route == .login(.index(next: "/users")))
}
@Test
func post() throws {
var request = URLRequestData(
method: "POST",
path: "/login",
body: .init("username=foo&password=super-secret&next=/users".utf8)
)
let route = try router.parse(&request)
#expect(route == .login(.post(.init(username: "foo", password: "super-secret", next: "/users"))))
}
}

View File

@@ -0,0 +1,129 @@
import Dependencies
import Foundation
import SharedModels
import Testing
import URLRouting
@Suite("PurchaseOrderViewRouteTests")
struct PurchaseOrderViewRouteTests {
let router = ViewRoute.router
@Test
func create() throws {
let id = UUID(0)
var request = URLRequestData(
method: "POST",
path: "/purchase-orders",
body: .init("id=1&workOrder=12345&materials=some&customer=Testy&truckStock=false&createdByID=\(id)&createdForID=\(id)&vendorBranchID=\(id)".utf8)
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.create(.init(
id: 1,
workOrder: 12345,
materials: "some",
customer: "Testy",
truckStock: false,
createdByID: id,
createdForID: id,
vendorBranchID: id
))))
}
@Test
func delete() throws {
let id = 1
var request = URLRequestData(
method: "DELETE",
path: "/purchase-orders/\(id)"
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.delete(id: id)))
}
@Test
func form() throws {
var request = URLRequestData(
method: "GET",
path: "/purchase-orders/create"
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.form))
}
@Test
func get() throws {
let id = 1
var request = URLRequestData(
method: "GET",
path: "/purchase-orders/\(id)"
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.get(id: id)))
}
@Test
func index() throws {
var request = URLRequestData(
method: "GET",
path: "/purchase-orders"
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.index))
}
@Test
func page() throws {
var request = URLRequestData(
method: "GET",
path: "/purchase-orders/next"
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.page(page: 1, limit: 25)))
var request2 = URLRequestData(
method: "GET",
path: "/purchase-orders/next",
query: ["page": ["2"], "limit": ["50"]]
)
let route2 = try router.parse(&request2)
#expect(route2 == .purchaseOrder(.page(page: 2, limit: 50)))
}
@Test
func searchIndex() throws {
var request = URLRequestData(
method: "GET",
path: "/purchase-orders/search",
query: ["context": ["customer"]]
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.search(.index(context: .customer, table: nil))))
var request2 = URLRequestData(
method: "GET",
path: "/purchase-orders/search"
)
let route2 = try router.parse(&request2)
#expect(route2 == .purchaseOrder(.search(.index(context: .employee, table: nil))))
var request3 = URLRequestData(
method: "GET",
path: "/purchase-orders/search",
query: ["context": ["vendor"], "table": ["true"]]
)
let route3 = try router.parse(&request3)
#expect(route3 == .purchaseOrder(.search(.index(context: .vendor, table: true))))
}
@Test
func searchPost() throws {
let id = UUID(0)
var request = URLRequestData(
method: "POST",
path: "/purchase-orders/search",
body: .init("context=employee&createdForID=\(id)".utf8)
)
let route = try router.parse(&request)
#expect(route == .purchaseOrder(.search(.request(.init(context: .employee, createdForID: id)))))
}
}

View File

@@ -0,0 +1,81 @@
import Dependencies
import Foundation
import SharedModels
import Testing
import URLRouting
@Suite("UserViewRouteTests")
struct UserViewRouteTests {
let router = ViewRoute.router
@Test
func create() throws {
var request = URLRequestData(
method: "POST",
path: "/users",
body: .init("username=foo&email=foo@bar.com&password=super-secret&confirmPassword=super-secret".utf8)
)
let route = try router.parse(&request)
#expect(
route == .user(.create(.init(
username: "foo",
email: "foo@bar.com",
password: "super-secret",
confirmPassword: "super-secret"
))))
}
@Test
func delete() throws {
let id = UUID(0)
var request = URLRequestData(
method: "DELETE",
path: "/users/\(id)"
)
let route = try router.parse(&request)
#expect(route == .user(.delete(id: id)))
}
@Test
func form() throws {
var request = URLRequestData(
method: "GET",
path: "/users/create"
)
let route = try router.parse(&request)
#expect(route == .user(.form))
}
@Test
func get() throws {
let id = UUID(0)
var request = URLRequestData(
method: "GET",
path: "/users/\(id)"
)
let route = try router.parse(&request)
#expect(route == .user(.get(id: id)))
}
@Test
func index() throws {
var request = URLRequestData(
method: "GET",
path: "/users"
)
let route = try router.parse(&request)
#expect(route == .user(.index))
}
@Test
func update() throws {
let id = UUID(0)
var request = URLRequestData(
method: "PATCH",
path: "/users/\(id)",
body: .init("username=bar&email=bar@foo.com".utf8)
)
let route = try router.parse(&request)
#expect(route == .user(.update(id: id, updates: .init(username: "bar", email: "bar@foo.com"))))
}
}

View File

@@ -0,0 +1,45 @@
import Dependencies
import Foundation
import SharedModels
import Testing
import URLRouting
@Suite("VendorBranchViewRouteTests")
struct VendorBranchViewRouteTests {
let router = ViewRoute.router
@Test
func create() throws {
let id = UUID(0)
var request = URLRequestData(
method: "POST",
path: "/vendors/branches",
body: .init("name=Test&vendorID=\(id)".utf8)
)
let route = try router.parse(&request)
#expect(route == .vendorBranch(.create(.init(name: "Test", vendorID: id))))
}
@Test
func delete() throws {
let id = UUID(0)
var request = URLRequestData(
method: "DELETE",
path: "/vendors/branches/\(id)"
)
let route = try router.parse(&request)
#expect(route == .vendorBranch(.delete(id: id)))
}
@Test
func select() throws {
var request = URLRequestData(
method: "GET",
path: "/vendors/branches/select",
query: ["context": ["purchaseOrderForm"]]
)
let route = try router.parse(&request)
#expect(route == .vendorBranch(.select(context: .purchaseOrderForm)))
}
}

View File

@@ -0,0 +1,75 @@
import Dependencies
import Foundation
import SharedModels
import Testing
import URLRouting
@Suite("VendorViewRouteTests")
struct VendorViewRouteTests {
let router = ViewRoute.router
@Test
func create() throws {
var request = URLRequestData(
method: "POST",
path: "/vendors",
body: .init("name=Test".utf8)
)
let route = try router.parse(&request)
#expect(route == .vendor(.create(.init(name: "Test"))))
}
@Test
func delete() throws {
let id = UUID(0)
var request = URLRequestData(
method: "DELETE",
path: "/vendors/\(id)"
)
let route = try router.parse(&request)
#expect(route == .vendor(.delete(id: id)))
}
@Test
func get() throws {
let id = UUID(0)
var request = URLRequestData(
method: "GET",
path: "/vendors/\(id)"
)
let route = try router.parse(&request)
#expect(route == .vendor(.get(id: id)))
}
@Test
func form() throws {
var request = URLRequestData(
method: "GET",
path: "/vendors/create"
)
let route = try router.parse(&request)
#expect(route == .vendor(.form))
}
@Test
func index() throws {
var request = URLRequestData(
method: "GET",
path: "/vendors"
)
let route = try router.parse(&request)
#expect(route == .vendor(.index))
}
@Test
func update() throws {
let id = UUID(0)
var request = URLRequestData(
method: "PUT",
path: "/vendors/\(id)",
body: .init("name=Test".utf8)
)
let route = try router.parse(&request)
#expect(route == .vendor(.update(id: id, updates: .init(name: "Test"))))
}
}