feat: Begins implementing route definitions.
This commit is contained in:
@@ -61,9 +61,10 @@ RUN export DEBIAN_FRONTEND=noninteractive DEBCONF_NONINTERACTIVE_SEEN=true \
|
|||||||
ca-certificates \
|
ca-certificates \
|
||||||
tzdata \
|
tzdata \
|
||||||
# If your app or its dependencies import FoundationNetworking, also install `libcurl4`.
|
# If your app or its dependencies import FoundationNetworking, also install `libcurl4`.
|
||||||
# libcurl4 \
|
libcurl4 \
|
||||||
# If your app or its dependencies import FoundationXML, also install `libxml2`.
|
# If your app or its dependencies import FoundationXML, also install `libxml2`.
|
||||||
# libxml2 \
|
# libxml2 \
|
||||||
|
sqlite3 \
|
||||||
&& rm -r /var/lib/apt/lists/*
|
&& rm -r /var/lib/apt/lists/*
|
||||||
|
|
||||||
# Create a vapor user and group with /app as its home directory
|
# Create a vapor user and group with /app as its home directory
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"originHash" : "aefc6edf3bfecf4e8b49731482d5e8f4fd78c1188ba5d90f5b78a36ab8106df6",
|
"originHash" : "62668360721c9ad13a7b961193242e083cbbf63a2bada2e8b1cc6bfeb4360fe6",
|
||||||
"pins" : [
|
"pins" : [
|
||||||
{
|
{
|
||||||
"identity" : "async-http-client",
|
"identity" : "async-http-client",
|
||||||
@@ -154,6 +154,15 @@
|
|||||||
"version" : "1.2.0"
|
"version" : "1.2.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"identity" : "swift-case-paths",
|
||||||
|
"kind" : "remoteSourceControl",
|
||||||
|
"location" : "https://github.com/pointfreeco/swift-case-paths",
|
||||||
|
"state" : {
|
||||||
|
"revision" : "e7039aaa4d9cf386fa8324a89f258c3f2c54d751",
|
||||||
|
"version" : "1.6.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"identity" : "swift-clocks",
|
"identity" : "swift-clocks",
|
||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
@@ -289,6 +298,15 @@
|
|||||||
"version" : "1.0.2"
|
"version" : "1.0.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"identity" : "swift-parsing",
|
||||||
|
"kind" : "remoteSourceControl",
|
||||||
|
"location" : "https://github.com/pointfreeco/swift-parsing",
|
||||||
|
"state" : {
|
||||||
|
"revision" : "3432cb81164dd3d69a75d0d63205be5fbae2c34b",
|
||||||
|
"version" : "0.14.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"identity" : "swift-service-context",
|
"identity" : "swift-service-context",
|
||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
@@ -316,6 +334,15 @@
|
|||||||
"version" : "1.4.0"
|
"version" : "1.4.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"identity" : "swift-url-routing",
|
||||||
|
"kind" : "remoteSourceControl",
|
||||||
|
"location" : "https://github.com/pointfreeco/swift-url-routing.git",
|
||||||
|
"state" : {
|
||||||
|
"revision" : "1cfd564259ecb1d324bb718a8f03e513dab738d2",
|
||||||
|
"version" : "0.6.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"identity" : "vapor",
|
"identity" : "vapor",
|
||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
|
|||||||
@@ -24,7 +24,8 @@ let package = Package(
|
|||||||
.package(url: "https://github.com/pointfreeco/swift-dependencies.git", from: "1.6.3"),
|
.package(url: "https://github.com/pointfreeco/swift-dependencies.git", from: "1.6.3"),
|
||||||
.package(url: "https://github.com/sliemeobn/elementary.git", from: "0.3.2"),
|
.package(url: "https://github.com/sliemeobn/elementary.git", from: "0.3.2"),
|
||||||
.package(url: "https://github.com/sliemeobn/elementary-htmx.git", from: "0.4.0"),
|
.package(url: "https://github.com/sliemeobn/elementary-htmx.git", from: "0.4.0"),
|
||||||
.package(url: "https://github.com/vapor-community/vapor-elementary.git", from: "0.1.0")
|
.package(url: "https://github.com/vapor-community/vapor-elementary.git", from: "0.1.0"),
|
||||||
|
.package(url: "https://github.com/pointfreeco/swift-url-routing.git", from: "0.6.2")
|
||||||
],
|
],
|
||||||
targets: [
|
targets: [
|
||||||
.executableTarget(
|
.executableTarget(
|
||||||
@@ -82,7 +83,8 @@ let package = Package(
|
|||||||
.target(
|
.target(
|
||||||
name: "SharedModels",
|
name: "SharedModels",
|
||||||
dependencies: [
|
dependencies: [
|
||||||
.product(name: "Dependencies", package: "swift-dependencies")
|
.product(name: "Dependencies", package: "swift-dependencies"),
|
||||||
|
.product(name: "URLRouting", package: "swift-url-routing")
|
||||||
],
|
],
|
||||||
swiftSettings: swiftSettings
|
swiftSettings: swiftSettings
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ struct PurchaseOrderTable: HTML {
|
|||||||
|
|
||||||
var content: some HTML {
|
var content: some HTML {
|
||||||
table(.id(.purchaseOrders())) {
|
table(.id(.purchaseOrders())) {
|
||||||
if page.items.count > 0 {
|
|
||||||
thead {
|
thead {
|
||||||
buttonRow
|
buttonRow
|
||||||
tableHeader
|
tableHeader
|
||||||
@@ -32,7 +31,6 @@ struct PurchaseOrderTable: HTML {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private var tableHeader: some HTML<HTMLTag.tr> {
|
private var tableHeader: some HTML<HTMLTag.tr> {
|
||||||
tr {
|
tr {
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import DatabaseClientLive
|
|||||||
import Dependencies
|
import Dependencies
|
||||||
import Fluent
|
import Fluent
|
||||||
import FluentSQLiteDriver
|
import FluentSQLiteDriver
|
||||||
import Leaf
|
|
||||||
import NIOSSL
|
import NIOSSL
|
||||||
import SharedModels
|
import SharedModels
|
||||||
import Vapor
|
import Vapor
|
||||||
@@ -45,8 +44,11 @@ public func configure(_ app: Application) async throws {
|
|||||||
try routes(app)
|
try routes(app)
|
||||||
}
|
}
|
||||||
|
|
||||||
if app.environment != .production {
|
// if app.environment != .production {
|
||||||
try await app.autoMigrate()
|
try await app.autoMigrate()
|
||||||
|
// }
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
app.asyncCommands.use(SeedCommand(), as: "seed")
|
app.asyncCommands.use(SeedCommand(), as: "seed")
|
||||||
}
|
#endif
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,6 +35,10 @@ func routes(_ app: Application) throws {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
app.get("health") { _ in
|
||||||
|
HTTPStatus.ok
|
||||||
|
}
|
||||||
|
|
||||||
app.post("login") { req in
|
app.post("login") { req in
|
||||||
@Dependency(\.database.users) var users
|
@Dependency(\.database.users) var users
|
||||||
let loginForm = try req.content.decode(User.Login.self)
|
let loginForm = try req.content.decode(User.Login.self)
|
||||||
|
|||||||
407
Sources/SharedModels/AppRoute.swift
Normal file
407
Sources/SharedModels/AppRoute.swift
Normal file
@@ -0,0 +1,407 @@
|
|||||||
|
import CasePathsCore
|
||||||
|
import Foundation
|
||||||
|
@preconcurrency import URLRouting
|
||||||
|
|
||||||
|
// TODO: Share view and api routes.
|
||||||
|
|
||||||
|
public enum ViewRoute: Sendable {
|
||||||
|
case employee(EmployeeRoute)
|
||||||
|
case purchaseOrder(PurchaseOrderRoute)
|
||||||
|
case select(SelectRoute)
|
||||||
|
case user(UserRoute)
|
||||||
|
case vendor(VendorRoute)
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.employee)) { EmployeeRoute.router }
|
||||||
|
Route(.case(Self.purchaseOrder)) { PurchaseOrderRoute.router }
|
||||||
|
Route(.case(Self.select)) { SelectRoute.router }
|
||||||
|
Route(.case(Self.user)) { UserRoute.router }
|
||||||
|
Route(.case(Self.vendor)) { VendorRoute.router }
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum EmployeeRoute: Sendable {
|
||||||
|
case create(Employee.Create)
|
||||||
|
case delete(id: Employee.ID)
|
||||||
|
case get(id: Employee.ID)
|
||||||
|
case form
|
||||||
|
case index
|
||||||
|
case update(id: Employee.ID, updates: Employee.Update)
|
||||||
|
|
||||||
|
static let rootPath = "employees"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(Employee.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.delete(id:))) {
|
||||||
|
Path { rootPath; UUID.parser() }
|
||||||
|
Method.delete
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; UUID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.form)) {
|
||||||
|
Path { rootPath; "create" }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { rootPath; UUID.parser() }
|
||||||
|
Method.put
|
||||||
|
Body(.json(Employee.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Add search.
|
||||||
|
public enum PurchaseOrderRoute: Sendable {
|
||||||
|
case create(PurchaseOrder.Create)
|
||||||
|
case form
|
||||||
|
case get(id: PurchaseOrder.ID)
|
||||||
|
case index
|
||||||
|
case next(page: Int, limit: Int)
|
||||||
|
|
||||||
|
static let rootPath = "purchase-orders"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(PurchaseOrder.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.form)) {
|
||||||
|
Path { rootPath; "create" }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; Digits() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.next(page:limit:))) {
|
||||||
|
Path { rootPath; "next" }
|
||||||
|
Method.get
|
||||||
|
Query {
|
||||||
|
Field("page", default: 1) { Digits() }
|
||||||
|
Field("limit", default: 25) { Digits() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum SelectRoute: Sendable {
|
||||||
|
case employee(context: Context)
|
||||||
|
case vendorBranches(context: Context)
|
||||||
|
|
||||||
|
public enum Context: String, Codable, Sendable, CaseIterable {
|
||||||
|
case purchaseOrderForm
|
||||||
|
case purchaseOrderSearch
|
||||||
|
}
|
||||||
|
|
||||||
|
static let rootPath = "select"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.employee(context:))) {
|
||||||
|
Path { rootPath; "employee" }
|
||||||
|
Method.get
|
||||||
|
Query {
|
||||||
|
Field("context") { Context.parser() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Route(.case(Self.vendorBranches(context:))) {
|
||||||
|
Path { rootPath; "vendor-branches" }
|
||||||
|
Method.get
|
||||||
|
Query {
|
||||||
|
Field("context") { Context.parser() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum UserRoute: Sendable {
|
||||||
|
case create(User.Create)
|
||||||
|
case delete(id: User.ID)
|
||||||
|
case form
|
||||||
|
case get(id: User.ID)
|
||||||
|
case index
|
||||||
|
case login(User.Login)
|
||||||
|
case update(id: User.ID, updates: User.Update)
|
||||||
|
|
||||||
|
static let rootPath = "users"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.delete(id:))) {
|
||||||
|
Path { rootPath; User.ID.parser() }
|
||||||
|
Method.delete
|
||||||
|
}
|
||||||
|
Route(.case(Self.form)) {
|
||||||
|
Path { rootPath; "create" }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; User.ID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.login)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Login.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { rootPath; User.ID.parser() }
|
||||||
|
// TODO: Use put or patch.
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum VendorRoute: Sendable {
|
||||||
|
case create(Vendor.Create)
|
||||||
|
case createBranch(VendorBranch.Create)
|
||||||
|
case form
|
||||||
|
case get(id: Vendor.ID)
|
||||||
|
case index
|
||||||
|
case update(id: Vendor.ID, updates: Vendor.Update)
|
||||||
|
|
||||||
|
static let rootPath = "vendors"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(Vendor.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.createBranch)) {
|
||||||
|
Path { rootPath; "branches" }
|
||||||
|
Method.post
|
||||||
|
Body(.json(VendorBranch.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.form)) {
|
||||||
|
Path { rootPath; "create" }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; Vendor.ID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { rootPath; Vendor.ID.parser() }
|
||||||
|
Method.put
|
||||||
|
Body(.json(Vendor.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum ApiRoute: Sendable {
|
||||||
|
|
||||||
|
case employee(EmployeeApiRoute)
|
||||||
|
case purchaseOrder(PurchaseOrderApiRoute)
|
||||||
|
case user(UserApiRoute)
|
||||||
|
case vendor(VendorApiRoute)
|
||||||
|
case vendorBranch(VendorBranchApiRoute)
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.employee)) { EmployeeApiRoute.router }
|
||||||
|
Route(.case(Self.purchaseOrder)) { PurchaseOrderApiRoute.router }
|
||||||
|
Route(.case(Self.user)) { UserApiRoute.router }
|
||||||
|
Route(.case(Self.vendor)) { VendorApiRoute.router }
|
||||||
|
Route(.case(Self.vendorBranch)) { VendorBranchApiRoute.router }
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum EmployeeApiRoute: Sendable {
|
||||||
|
case create(Employee.Create)
|
||||||
|
case delete(id: Employee.ID)
|
||||||
|
case get(id: Employee.ID)
|
||||||
|
case index
|
||||||
|
case update(id: Employee.ID, updates: Employee.Update)
|
||||||
|
|
||||||
|
static let rootPath = "employees"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(Employee.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.delete(id:))) {
|
||||||
|
Path { rootPath; UUID.parser() }
|
||||||
|
Method.delete
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; UUID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { rootPath; UUID.parser() }
|
||||||
|
Method.put
|
||||||
|
Body(.json(Employee.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum PurchaseOrderApiRoute: Sendable {
|
||||||
|
case create(PurchaseOrder.Create)
|
||||||
|
case delete(id: PurchaseOrder.ID)
|
||||||
|
case get(id: PurchaseOrder.ID)
|
||||||
|
case index
|
||||||
|
case page(page: Int, limit: Int)
|
||||||
|
|
||||||
|
static let rootPath = "purchase-orders"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(PurchaseOrder.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.delete(id:))) {
|
||||||
|
Path { rootPath; Digits() }
|
||||||
|
Method.delete
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; Digits() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.page(page:limit:))) {
|
||||||
|
Path { rootPath; "next" }
|
||||||
|
Method.get
|
||||||
|
Query {
|
||||||
|
Field("page", default: 1) { Digits() }
|
||||||
|
Field("limit", default: 25) { Digits() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum UserApiRoute: Sendable {
|
||||||
|
case create(User.Create)
|
||||||
|
case delete(id: User.ID)
|
||||||
|
case get(id: User.ID)
|
||||||
|
case index
|
||||||
|
case login(User.Login)
|
||||||
|
case update(id: User.ID, updates: User.Update)
|
||||||
|
|
||||||
|
static let rootPath = "users"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.delete(id:))) {
|
||||||
|
Path { rootPath; User.ID.parser() }
|
||||||
|
Method.delete
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; User.ID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.login)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Login.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { rootPath; User.ID.parser() }
|
||||||
|
// TODO: Use put or patch.
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum VendorApiRoute: Sendable {
|
||||||
|
case create(Vendor.Create)
|
||||||
|
case get(id: Vendor.ID)
|
||||||
|
case index
|
||||||
|
case update(id: Vendor.ID, updates: Vendor.Update)
|
||||||
|
|
||||||
|
static let rootPath = "vendors"
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.post
|
||||||
|
Body(.json(Vendor.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { rootPath; Vendor.ID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.index)) {
|
||||||
|
Path { rootPath }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { rootPath; Vendor.ID.parser() }
|
||||||
|
Method.put
|
||||||
|
Body(.json(Vendor.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum VendorBranchApiRoute: Sendable {
|
||||||
|
case create(VendorBranch.Create)
|
||||||
|
case delete(id: VendorBranch.ID)
|
||||||
|
case get(id: VendorBranch.ID)
|
||||||
|
case update(id: VendorBranch.ID, updates: VendorBranch.Update)
|
||||||
|
|
||||||
|
public static let router = OneOf {
|
||||||
|
Route(.case(Self.create)) {
|
||||||
|
Path { "vendors"; "branches" }
|
||||||
|
Method.post
|
||||||
|
Body(.json(VendorBranch.Create.self))
|
||||||
|
}
|
||||||
|
Route(.case(Self.delete(id:))) {
|
||||||
|
Path { "vendors"; "branches"; VendorBranch.ID.parser() }
|
||||||
|
Method.delete
|
||||||
|
}
|
||||||
|
Route(.case(Self.get(id:))) {
|
||||||
|
Path { "vendors"; "branches"; VendorBranch.ID.parser() }
|
||||||
|
Method.get
|
||||||
|
}
|
||||||
|
Route(.case(Self.update(id:updates:))) {
|
||||||
|
Path { "vendors"; "branches"; VendorBranch.ID.parser() }
|
||||||
|
Method.put
|
||||||
|
Body(.json(VendorBranch.Update.self))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -17,6 +17,12 @@ struct DatabaseClientTests {
|
|||||||
self.logger = logger
|
self.logger = logger
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
func testPath() {
|
||||||
|
let path = AppRoute.ViewRoute.router.path(for: .employee(.index))
|
||||||
|
#expect(path == "/employees")
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
func users() async throws {
|
func users() async throws {
|
||||||
try await withDatabase(migrations: User.Migrate()) {
|
try await withDatabase(migrations: User.Migrate()) {
|
||||||
|
|||||||
Reference in New Issue
Block a user