feat: Moves api controller to it's own module.
This commit is contained in:
@@ -8,6 +8,8 @@ let package = Package(
|
|||||||
],
|
],
|
||||||
products: [
|
products: [
|
||||||
.executable(name: "App", targets: ["App"]),
|
.executable(name: "App", targets: ["App"]),
|
||||||
|
.library(name: "ApiController", targets: ["ApiController"]),
|
||||||
|
.library(name: "ApiControllerLive", targets: ["ApiControllerLive"]),
|
||||||
.library(name: "SharedModels", targets: ["SharedModels"]),
|
.library(name: "SharedModels", targets: ["SharedModels"]),
|
||||||
.library(name: "DatabaseClient", targets: ["DatabaseClient"]),
|
.library(name: "DatabaseClient", targets: ["DatabaseClient"]),
|
||||||
.library(name: "DatabaseClientLive", targets: ["DatabaseClientLive"]),
|
.library(name: "DatabaseClientLive", targets: ["DatabaseClientLive"]),
|
||||||
@@ -37,6 +39,7 @@ let package = Package(
|
|||||||
.executableTarget(
|
.executableTarget(
|
||||||
name: "App",
|
name: "App",
|
||||||
dependencies: [
|
dependencies: [
|
||||||
|
.target(name: "ApiControllerLive"),
|
||||||
.target(name: "DatabaseClientLive"),
|
.target(name: "DatabaseClientLive"),
|
||||||
.target(name: "ViewControllerLive"),
|
.target(name: "ViewControllerLive"),
|
||||||
.product(name: "Fluent", package: "fluent"),
|
.product(name: "Fluent", package: "fluent"),
|
||||||
@@ -54,27 +57,24 @@ let package = Package(
|
|||||||
],
|
],
|
||||||
swiftSettings: swiftSettings
|
swiftSettings: swiftSettings
|
||||||
),
|
),
|
||||||
// .testTarget(
|
.target(
|
||||||
// name: "AppTests",
|
name: "ApiController",
|
||||||
// dependencies: [
|
|
||||||
// .target(name: "App"),
|
|
||||||
// .target(name: "HtmlSnapshotTesting"),
|
|
||||||
// .product(name: "XCTVapor", package: "vapor")
|
|
||||||
// ],
|
|
||||||
// resources: [
|
|
||||||
// .copy("__Snapshots__")
|
|
||||||
// ],
|
|
||||||
// swiftSettings: swiftSettings
|
|
||||||
// ),
|
|
||||||
.testTarget(
|
|
||||||
name: "ViewRouteTests",
|
|
||||||
dependencies: [
|
dependencies: [
|
||||||
.target(name: "App"),
|
.target(name: "SharedModels"),
|
||||||
.product(name: "VaporTesting", package: "vapor")
|
.product(name: "Dependencies", package: "swift-dependencies"),
|
||||||
|
.product(name: "DependenciesMacros", package: "swift-dependencies")
|
||||||
],
|
],
|
||||||
|
|
||||||
swiftSettings: swiftSettings
|
swiftSettings: swiftSettings
|
||||||
),
|
),
|
||||||
|
.target(
|
||||||
|
name: "ApiControllerLive",
|
||||||
|
dependencies: [
|
||||||
|
.target(name: "ApiController"),
|
||||||
|
.target(name: "DatabaseClient")
|
||||||
|
],
|
||||||
|
swiftSettings: swiftSettings
|
||||||
|
),
|
||||||
|
|
||||||
.testTarget(
|
.testTarget(
|
||||||
name: "ApiRouteTests",
|
name: "ApiRouteTests",
|
||||||
dependencies: [
|
dependencies: [
|
||||||
@@ -153,6 +153,15 @@ let package = Package(
|
|||||||
resources: [
|
resources: [
|
||||||
.copy("__Snapshots__")
|
.copy("__Snapshots__")
|
||||||
],
|
],
|
||||||
|
swiftSettings: swiftSettings
|
||||||
|
),
|
||||||
|
.testTarget(
|
||||||
|
name: "ViewRouteTests",
|
||||||
|
dependencies: [
|
||||||
|
.target(name: "App"),
|
||||||
|
.product(name: "VaporTesting", package: "vapor")
|
||||||
|
],
|
||||||
|
|
||||||
swiftSettings: swiftSettings
|
swiftSettings: swiftSettings
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
|||||||
34
Sources/ApiController/ApiController.swift
Normal file
34
Sources/ApiController/ApiController.swift
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import Dependencies
|
||||||
|
import DependenciesMacros
|
||||||
|
import Logging
|
||||||
|
import SharedModels
|
||||||
|
|
||||||
|
public extension DependencyValues {
|
||||||
|
var apiController: ApiController {
|
||||||
|
get { self[ApiController.self] }
|
||||||
|
set { self[ApiController.self] = newValue }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@DependencyClient
|
||||||
|
public struct ApiController: Sendable {
|
||||||
|
public var json: @Sendable (Request) async throws -> (any Encodable)?
|
||||||
|
|
||||||
|
public func json(_ route: ApiRoute, logger: Logger) async throws -> (any Encodable)? {
|
||||||
|
try await json(.init(route, logger: logger))
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct Request: Sendable {
|
||||||
|
public let route: ApiRoute
|
||||||
|
public let logger: Logger
|
||||||
|
|
||||||
|
public init(_ route: ApiRoute, logger: Logger) {
|
||||||
|
self.route = route
|
||||||
|
self.logger = logger
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extension ApiController: TestDependencyKey {
|
||||||
|
public static let testValue: ApiController = Self()
|
||||||
|
}
|
||||||
@@ -1,51 +1,63 @@
|
|||||||
import DatabaseClientLive
|
@_exported import ApiController
|
||||||
|
import DatabaseClient
|
||||||
import Dependencies
|
import Dependencies
|
||||||
import Fluent
|
import Logging
|
||||||
import SharedModels
|
import SharedModels
|
||||||
import Vapor
|
|
||||||
|
|
||||||
private let apiMiddleware: [any Middleware] = [
|
extension ApiController: DependencyKey {
|
||||||
UserPasswordAuthenticator(),
|
public static var liveValue: ApiController {
|
||||||
UserTokenAuthenticator(),
|
.init(json: { try await $0.respond() })
|
||||||
UserSessionAuthenticator(),
|
}
|
||||||
User.guardMiddleware()
|
}
|
||||||
]
|
|
||||||
|
|
||||||
extension ApiRoute {
|
private extension ApiController.Request {
|
||||||
var middleware: [any Middleware]? { apiMiddleware }
|
|
||||||
|
|
||||||
func respond(request: Request) async throws -> any AsyncResponseEncodable {
|
func respond() async throws -> (any Encodable)? {
|
||||||
switch self {
|
@Dependency(\.database) var database
|
||||||
|
|
||||||
|
switch route {
|
||||||
case let .employee(route):
|
case let .employee(route):
|
||||||
return try await route.handleApiRequest(request: request)
|
return try await route.handleApiRequest(logger: logger)
|
||||||
|
case let .login(login):
|
||||||
|
return try await TokenResponse(token: database.users.login(login))
|
||||||
case let .purchaseOrder(route):
|
case let .purchaseOrder(route):
|
||||||
return try await route.handleApiRequest(request: request)
|
return try await route.handleApiRequest(logger: logger)
|
||||||
case let .user(route):
|
case let .user(route):
|
||||||
return try await route.handleApiRequest(request: request)
|
return try await route.handleApiRequest(logger: logger)
|
||||||
case let .vendor(route):
|
case let .vendor(route):
|
||||||
return try await route.handleApiRequest(request: request)
|
return try await route.handleApiRequest(logger: logger)
|
||||||
case let .vendorBranch(route):
|
case let .vendorBranch(route):
|
||||||
return try await route.handleApiRequest(request: request)
|
return try await route.handleApiRequest(logger: logger)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension ApiRoute.EmployeeRoute {
|
private struct TokenResponse: Encodable {
|
||||||
|
let token: String
|
||||||
|
|
||||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
init(token: User.Token) {
|
||||||
|
self.token = token.value
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private extension ApiRoute.EmployeeRoute {
|
||||||
|
|
||||||
|
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||||
@Dependency(\.database) var database
|
@Dependency(\.database) var database
|
||||||
|
|
||||||
switch self {
|
switch self {
|
||||||
case let .delete(id: id):
|
case let .delete(id: id):
|
||||||
try await database.employees.delete(id)
|
try await database.employees.delete(id)
|
||||||
return HTTPStatus.ok
|
return nil
|
||||||
case let .create(employee):
|
case let .create(employee):
|
||||||
return try await database.employees.create(employee)
|
return try await database.employees.create(employee)
|
||||||
case .index:
|
case .index:
|
||||||
return try await database.employees.fetchAll()
|
return try await database.employees.fetchAll()
|
||||||
case let .get(id: id):
|
case let .get(id: id):
|
||||||
guard let employee = try await database.employees.get(id) else {
|
guard let employee = try await database.employees.get(id) else {
|
||||||
throw Abort(.badRequest, reason: "Employee not found")
|
logger.error("Employee not found for id: \(id)")
|
||||||
|
throw ApiError(reason: "Employee not found")
|
||||||
}
|
}
|
||||||
return employee
|
return employee
|
||||||
case let .update(id: id, updates: updates):
|
case let .update(id: id, updates: updates):
|
||||||
@@ -54,21 +66,22 @@ extension ApiRoute.EmployeeRoute {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension ApiRoute.PurchaseOrderRoute {
|
private extension ApiRoute.PurchaseOrderRoute {
|
||||||
|
|
||||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||||
@Dependency(\.database.purchaseOrders) var purchaseOrders
|
@Dependency(\.database.purchaseOrders) var purchaseOrders
|
||||||
switch self {
|
switch self {
|
||||||
case let .delete(id: id):
|
case let .delete(id: id):
|
||||||
try await purchaseOrders.delete(id)
|
try await purchaseOrders.delete(id)
|
||||||
return HTTPStatus.ok
|
return nil
|
||||||
case .index:
|
case .index:
|
||||||
return try await purchaseOrders.fetchAll()
|
return try await purchaseOrders.fetchAll()
|
||||||
case let .create(purchaseOrder):
|
case let .create(purchaseOrder):
|
||||||
return try await purchaseOrders.create(purchaseOrder)
|
return try await purchaseOrders.create(purchaseOrder)
|
||||||
case let .get(id: id):
|
case let .get(id: id):
|
||||||
guard let output = try await purchaseOrders.get(id) else {
|
guard let output = try await purchaseOrders.get(id) else {
|
||||||
throw Abort(.badRequest, reason: "Purchase order not found.")
|
logger.error("Purchase Order not found for id: \(id)")
|
||||||
|
throw ApiError(reason: "Purchase order not found.")
|
||||||
}
|
}
|
||||||
return output
|
return output
|
||||||
case let .page(page: page, limit: limit):
|
case let .page(page: page, limit: limit):
|
||||||
@@ -78,21 +91,22 @@ extension ApiRoute.PurchaseOrderRoute {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Add Login.
|
// TODO: Add Login.
|
||||||
extension ApiRoute.UserRoute {
|
private extension ApiRoute.UserRoute {
|
||||||
|
|
||||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||||
@Dependency(\.database.users) var users
|
@Dependency(\.database.users) var users
|
||||||
switch self {
|
switch self {
|
||||||
case let .delete(id: id):
|
case let .delete(id: id):
|
||||||
try await users.delete(id)
|
try await users.delete(id)
|
||||||
return HTTPStatus.ok
|
return nil
|
||||||
case let .create(user):
|
case let .create(user):
|
||||||
return try await users.create(user)
|
return try await users.create(user)
|
||||||
case .index:
|
case .index:
|
||||||
return try await users.fetchAll()
|
return try await users.fetchAll()
|
||||||
case let .get(id: id):
|
case let .get(id: id):
|
||||||
|
logger.error("User not found for id: \(id)")
|
||||||
guard let user = try await users.get(id) else {
|
guard let user = try await users.get(id) else {
|
||||||
throw Abort(.badRequest, reason: "Employee not found")
|
throw ApiError(reason: "Employee not found")
|
||||||
}
|
}
|
||||||
return user
|
return user
|
||||||
// case let .login(user):
|
// case let .login(user):
|
||||||
@@ -103,18 +117,19 @@ extension ApiRoute.UserRoute {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension ApiRoute.VendorRoute {
|
private extension ApiRoute.VendorRoute {
|
||||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||||
@Dependency(\.database.vendors) var vendors
|
@Dependency(\.database.vendors) var vendors
|
||||||
switch self {
|
switch self {
|
||||||
case let .delete(id: id):
|
case let .delete(id: id):
|
||||||
try await vendors.delete(id)
|
try await vendors.delete(id)
|
||||||
return HTTPStatus.ok
|
return nil
|
||||||
case let .create(vendor):
|
case let .create(vendor):
|
||||||
return try await vendors.create(vendor)
|
return try await vendors.create(vendor)
|
||||||
case let .get(id: id):
|
case let .get(id: id):
|
||||||
guard let vendor = try await vendors.get(id) else {
|
guard let vendor = try await vendors.get(id) else {
|
||||||
throw Abort(.badRequest, reason: "Employee not found")
|
logger.error("Vendor not found for id: \(id)")
|
||||||
|
throw ApiError(reason: "Vendor not found")
|
||||||
}
|
}
|
||||||
return vendor
|
return vendor
|
||||||
case let .update(id: id, updates: updates):
|
case let .update(id: id, updates: updates):
|
||||||
@@ -128,13 +143,13 @@ extension ApiRoute.VendorRoute {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extension ApiRoute.VendorBranchRoute {
|
private extension ApiRoute.VendorBranchRoute {
|
||||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||||
@Dependency(\.database.vendorBranches) var vendorBranches
|
@Dependency(\.database.vendorBranches) var vendorBranches
|
||||||
switch self {
|
switch self {
|
||||||
case let .delete(id: id):
|
case let .delete(id: id):
|
||||||
try await vendorBranches.delete(id)
|
try await vendorBranches.delete(id)
|
||||||
return HTTPStatus.ok
|
return nil
|
||||||
case let .create(branch):
|
case let .create(branch):
|
||||||
return try await vendorBranches.create(branch)
|
return try await vendorBranches.create(branch)
|
||||||
case let .index(for: optionalVendorID):
|
case let .index(for: optionalVendorID):
|
||||||
@@ -144,7 +159,8 @@ extension ApiRoute.VendorBranchRoute {
|
|||||||
return try await vendorBranches.fetchAll(.for(vendorID: vendorID))
|
return try await vendorBranches.fetchAll(.for(vendorID: vendorID))
|
||||||
case let .get(id: id):
|
case let .get(id: id):
|
||||||
guard let branch = try await vendorBranches.get(id) else {
|
guard let branch = try await vendorBranches.get(id) else {
|
||||||
throw Abort(.badRequest, reason: "Employee not found")
|
logger.error("Vendor branch not found for id: \(id)")
|
||||||
|
throw ApiError(reason: "Vendor branch not found")
|
||||||
}
|
}
|
||||||
return branch
|
return branch
|
||||||
case let .update(id: id, updates: updates):
|
case let .update(id: id, updates: updates):
|
||||||
@@ -152,3 +168,7 @@ extension ApiRoute.VendorBranchRoute {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct ApiError: Error {
|
||||||
|
let reason: String
|
||||||
|
}
|
||||||
35
Sources/App/Extensions/ApiController+respond.swift
Normal file
35
Sources/App/Extensions/ApiController+respond.swift
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
import ApiController
|
||||||
|
import SharedModels
|
||||||
|
import Vapor
|
||||||
|
|
||||||
|
extension ApiController {
|
||||||
|
|
||||||
|
func respond(_ route: ApiRoute, request: Vapor.Request) async throws -> any AsyncResponseEncodable {
|
||||||
|
guard let encodable = try await json(route, logger: request.logger) else {
|
||||||
|
return HTTPStatus.ok
|
||||||
|
}
|
||||||
|
return AnyJSONResponse(value: encodable)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct AnyJSONResponse: AsyncResponseEncodable {
|
||||||
|
public var headers: HTTPHeaders = ["Content-Type": "application/json"]
|
||||||
|
let value: any Encodable
|
||||||
|
|
||||||
|
init(additionalHeaders: HTTPHeaders = [:], value: any Encodable) {
|
||||||
|
if additionalHeaders.contains(name: .contentType) {
|
||||||
|
self.headers = additionalHeaders
|
||||||
|
} else {
|
||||||
|
headers.add(contentsOf: additionalHeaders)
|
||||||
|
}
|
||||||
|
self.value = value
|
||||||
|
}
|
||||||
|
|
||||||
|
func encodeResponse(for request: Request) async throws -> Response {
|
||||||
|
try Response(
|
||||||
|
status: .ok,
|
||||||
|
headers: headers,
|
||||||
|
body: .init(data: JSONEncoder().encode(value))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
20
Sources/App/Middleware/ApiRoute+middleware.swift
Normal file
20
Sources/App/Middleware/ApiRoute+middleware.swift
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import DatabaseClientLive
|
||||||
|
import SharedModels
|
||||||
|
import Vapor
|
||||||
|
|
||||||
|
private let apiMiddleware: [any Middleware] = [
|
||||||
|
UserPasswordAuthenticator(),
|
||||||
|
UserTokenAuthenticator(),
|
||||||
|
UserSessionAuthenticator(),
|
||||||
|
User.guardMiddleware()
|
||||||
|
]
|
||||||
|
|
||||||
|
extension ApiRoute {
|
||||||
|
var middleware: [any Middleware]? {
|
||||||
|
switch self {
|
||||||
|
case .login: return nil
|
||||||
|
default:
|
||||||
|
return apiMiddleware
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import ApiControllerLive
|
||||||
import DatabaseClientLive
|
import DatabaseClientLive
|
||||||
import Dependencies
|
import Dependencies
|
||||||
import Vapor
|
import Vapor
|
||||||
@@ -8,14 +9,17 @@ import ViewControllerLive
|
|||||||
struct DependenciesMiddleware: AsyncMiddleware {
|
struct DependenciesMiddleware: AsyncMiddleware {
|
||||||
|
|
||||||
private let values: DependencyValues.Continuation
|
private let values: DependencyValues.Continuation
|
||||||
|
private let apiController: ApiController
|
||||||
private let database: DatabaseClient
|
private let database: DatabaseClient
|
||||||
private let viewController: ViewController
|
private let viewController: ViewController
|
||||||
|
|
||||||
init(
|
init(
|
||||||
database: DatabaseClient,
|
database: DatabaseClient,
|
||||||
|
apiController: ApiController = .liveValue,
|
||||||
viewController: ViewController = .liveValue
|
viewController: ViewController = .liveValue
|
||||||
) {
|
) {
|
||||||
self.values = withEscapedDependencies { $0 }
|
self.values = withEscapedDependencies { $0 }
|
||||||
|
self.apiController = apiController
|
||||||
self.database = database
|
self.database = database
|
||||||
self.viewController = viewController
|
self.viewController = viewController
|
||||||
}
|
}
|
||||||
@@ -23,6 +27,7 @@ struct DependenciesMiddleware: AsyncMiddleware {
|
|||||||
func respond(to request: Request, chainingTo next: any AsyncResponder) async throws -> Response {
|
func respond(to request: Request, chainingTo next: any AsyncResponder) async throws -> Response {
|
||||||
try await values.yield {
|
try await values.yield {
|
||||||
try await withDependencies {
|
try await withDependencies {
|
||||||
|
$0.apiController = apiController
|
||||||
$0.database = database
|
$0.database = database
|
||||||
$0.dateFormatter = .liveValue
|
$0.dateFormatter = .liveValue
|
||||||
$0.viewController = viewController
|
$0.viewController = viewController
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import ApiController
|
||||||
import DatabaseClientLive
|
import DatabaseClientLive
|
||||||
import Dependencies
|
import Dependencies
|
||||||
import Elementary
|
import Elementary
|
||||||
@@ -15,6 +16,24 @@ public func configure(
|
|||||||
_ app: Application,
|
_ app: Application,
|
||||||
makeDatabaseClient: @escaping (any Database) -> DatabaseClient = { .live(database: $0) }
|
makeDatabaseClient: @escaping (any Database) -> DatabaseClient = { .live(database: $0) }
|
||||||
) async throws {
|
) async throws {
|
||||||
|
// Setup the database client.
|
||||||
|
let databaseClient = try await setupDatabase(on: app, factory: makeDatabaseClient)
|
||||||
|
// Add the global middlewares.
|
||||||
|
addMiddleware(to: app, database: databaseClient)
|
||||||
|
#if DEBUG
|
||||||
|
// Live reload of the application for development when launched with the `./swift-dev` command
|
||||||
|
app.lifecycle.use(BrowserSyncHandler())
|
||||||
|
#endif
|
||||||
|
// Add our route handlers.
|
||||||
|
addRoutes(to: app)
|
||||||
|
if app.environment != .testing {
|
||||||
|
try await app.autoMigrate()
|
||||||
|
}
|
||||||
|
// Add our custom cli-commands to the application.
|
||||||
|
addCommands(to: app)
|
||||||
|
}
|
||||||
|
|
||||||
|
private func addMiddleware(to app: Application, database databaseClient: DatabaseClient) {
|
||||||
// cors middleware should come before default error middleware using `at: .beginning`
|
// cors middleware should come before default error middleware using `at: .beginning`
|
||||||
let corsConfiguration = CORSMiddleware.Configuration(
|
let corsConfiguration = CORSMiddleware.Configuration(
|
||||||
allowedOrigin: .all,
|
allowedOrigin: .all,
|
||||||
@@ -27,11 +46,13 @@ public func configure(
|
|||||||
|
|
||||||
app.middleware.use(FileMiddleware(publicDirectory: app.directory.publicDirectory))
|
app.middleware.use(FileMiddleware(publicDirectory: app.directory.publicDirectory))
|
||||||
app.middleware.use(app.sessions.middleware)
|
app.middleware.use(app.sessions.middleware)
|
||||||
|
app.middleware.use(DependenciesMiddleware(database: databaseClient))
|
||||||
|
}
|
||||||
|
|
||||||
#if DEBUG
|
private func setupDatabase(
|
||||||
app.lifecycle.use(BrowserSyncHandler())
|
on app: Application,
|
||||||
#endif
|
factory makeDatabaseClient: @escaping (any Database) -> DatabaseClient
|
||||||
|
) async throws -> DatabaseClient {
|
||||||
switch app.environment {
|
switch app.environment {
|
||||||
case .production, .development:
|
case .production, .development:
|
||||||
let dbFileName = Environment.get("SQLITE_FILENAME") ?? "db.sqlite"
|
let dbFileName = Environment.get("SQLITE_FILENAME") ?? "db.sqlite"
|
||||||
@@ -46,8 +67,10 @@ public func configure(
|
|||||||
try await app.migrations.add(databaseClient.migrations())
|
try await app.migrations.add(databaseClient.migrations())
|
||||||
}
|
}
|
||||||
|
|
||||||
app.middleware.use(DependenciesMiddleware(database: databaseClient))
|
return databaseClient
|
||||||
|
}
|
||||||
|
|
||||||
|
private func addRoutes(to app: Application) {
|
||||||
// Redirect the index path to purchase order route.
|
// Redirect the index path to purchase order route.
|
||||||
app.get { req in
|
app.get { req in
|
||||||
req.redirect(to: ViewRoute.router.path(for: .purchaseOrder(.index)))
|
req.redirect(to: ViewRoute.router.path(for: .purchaseOrder(.index)))
|
||||||
@@ -64,18 +87,16 @@ public func configure(
|
|||||||
},
|
},
|
||||||
use: siteHandler
|
use: siteHandler
|
||||||
)
|
)
|
||||||
|
}
|
||||||
|
|
||||||
if app.environment != .testing {
|
private func addCommands(to app: Application) {
|
||||||
try await app.autoMigrate()
|
|
||||||
}
|
|
||||||
|
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
app.asyncCommands.use(SeedCommand(), as: "seed")
|
app.asyncCommands.use(SeedCommand(), as: "seed")
|
||||||
#endif
|
#endif
|
||||||
app.asyncCommands.use(GenerateAdminUserCommand(), as: "generate-admin")
|
app.asyncCommands.use(GenerateAdminUserCommand(), as: "generate-admin")
|
||||||
}
|
}
|
||||||
|
|
||||||
extension SiteRoute {
|
private extension SiteRoute {
|
||||||
|
|
||||||
func middleware() -> [any Middleware]? {
|
func middleware() -> [any Middleware]? {
|
||||||
switch self {
|
switch self {
|
||||||
@@ -90,14 +111,16 @@ extension SiteRoute {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Sendable
|
@Sendable
|
||||||
func siteHandler(
|
private func siteHandler(
|
||||||
request: Request,
|
request: Request,
|
||||||
route: SiteRoute
|
route: SiteRoute
|
||||||
) async throws -> any AsyncResponseEncodable {
|
) async throws -> any AsyncResponseEncodable {
|
||||||
|
@Dependency(\.apiController) var apiController
|
||||||
@Dependency(\.viewController) var viewController
|
@Dependency(\.viewController) var viewController
|
||||||
|
|
||||||
switch route {
|
switch route {
|
||||||
case let .api(route):
|
case let .api(route):
|
||||||
return try await route.respond(request: request)
|
return try await apiController.respond(route, request: request)
|
||||||
case .health:
|
case .health:
|
||||||
return HTTPStatus.ok
|
return HTTPStatus.ok
|
||||||
case let .view(route):
|
case let .view(route):
|
||||||
|
|||||||
@@ -25,7 +25,9 @@ public extension DatabaseClient.Users {
|
|||||||
} login: { login in
|
} login: { login in
|
||||||
try login.validate()
|
try login.validate()
|
||||||
|
|
||||||
var query = UserModel.query(on: database)
|
var query = UserModel
|
||||||
|
.query(on: database)
|
||||||
|
.with(\.$token)
|
||||||
|
|
||||||
if let username = login.username {
|
if let username = login.username {
|
||||||
query = query.filter(\UserModel.$username == username)
|
query = query.filter(\UserModel.$username == username)
|
||||||
@@ -37,15 +39,19 @@ public extension DatabaseClient.Users {
|
|||||||
throw NotFoundError()
|
throw NotFoundError()
|
||||||
}
|
}
|
||||||
|
|
||||||
let token = try user.generateToken()
|
let token: User.Token
|
||||||
|
|
||||||
try await token.save(on: database)
|
// Check if the user already has a token.
|
||||||
|
if let userToken = user.token {
|
||||||
|
token = try userToken.toDTO()
|
||||||
|
} else {
|
||||||
|
// Generate a new token for the user if they didn't have one.
|
||||||
|
let tokenModel = try user.generateToken()
|
||||||
|
try await tokenModel.save(on: database)
|
||||||
|
token = try tokenModel.toDTO()
|
||||||
|
}
|
||||||
|
|
||||||
return try User.Token(
|
return token
|
||||||
id: token.requireID(),
|
|
||||||
userID: user.requireID(),
|
|
||||||
value: token.value
|
|
||||||
)
|
|
||||||
|
|
||||||
} logout: { id in
|
} logout: { id in
|
||||||
guard let token = try await UserTokenModel.find(id, on: database)
|
guard let token = try await UserTokenModel.find(id, on: database)
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import Foundation
|
|||||||
public enum ApiRoute: Sendable, Equatable {
|
public enum ApiRoute: Sendable, Equatable {
|
||||||
|
|
||||||
case employee(EmployeeRoute)
|
case employee(EmployeeRoute)
|
||||||
|
case login(User.Login)
|
||||||
case purchaseOrder(PurchaseOrderRoute)
|
case purchaseOrder(PurchaseOrderRoute)
|
||||||
case user(UserRoute)
|
case user(UserRoute)
|
||||||
case vendor(VendorRoute)
|
case vendor(VendorRoute)
|
||||||
@@ -17,6 +18,11 @@ public enum ApiRoute: Sendable, Equatable {
|
|||||||
rootPath
|
rootPath
|
||||||
EmployeeRoute.router
|
EmployeeRoute.router
|
||||||
}
|
}
|
||||||
|
Route(.case(Self.login)) {
|
||||||
|
Path { "api"; "v1"; "login" }
|
||||||
|
Method.post
|
||||||
|
Body(.json(User.Login.self))
|
||||||
|
}
|
||||||
Route(.case(Self.purchaseOrder)) {
|
Route(.case(Self.purchaseOrder)) {
|
||||||
rootPath
|
rootPath
|
||||||
PurchaseOrderRoute.router
|
PurchaseOrderRoute.router
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import Foundation
|
|||||||
|
|
||||||
public enum ViewRoute: Sendable, Equatable {
|
public enum ViewRoute: Sendable, Equatable {
|
||||||
|
|
||||||
// case index
|
|
||||||
case employee(EmployeeRoute)
|
case employee(EmployeeRoute)
|
||||||
case login(LoginRoute)
|
case login(LoginRoute)
|
||||||
case purchaseOrder(PurchaseOrderRoute)
|
case purchaseOrder(PurchaseOrderRoute)
|
||||||
@@ -13,9 +12,6 @@ public enum ViewRoute: Sendable, Equatable {
|
|||||||
case vendorBranch(VendorBranchRoute)
|
case vendorBranch(VendorBranchRoute)
|
||||||
|
|
||||||
public static let router = OneOf {
|
public static let router = OneOf {
|
||||||
// Route(.case(Self.index)) {
|
|
||||||
// Method.get
|
|
||||||
// }
|
|
||||||
Route(.case(Self.employee)) { EmployeeRoute.router }
|
Route(.case(Self.employee)) { EmployeeRoute.router }
|
||||||
Route(.case(Self.login)) { LoginRoute.router }
|
Route(.case(Self.login)) { LoginRoute.router }
|
||||||
Route(.case(Self.purchaseOrder)) { PurchaseOrderRoute.router }
|
Route(.case(Self.purchaseOrder)) { PurchaseOrderRoute.router }
|
||||||
@@ -23,7 +19,6 @@ public enum ViewRoute: Sendable, Equatable {
|
|||||||
Route(.case(Self.vendor)) { VendorRoute.router }
|
Route(.case(Self.vendor)) { VendorRoute.router }
|
||||||
Route(.case(Self.vendorBranch)) { VendorBranchRoute.router }
|
Route(.case(Self.vendorBranch)) { VendorBranchRoute.router }
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public extension ViewRoute {
|
public extension ViewRoute {
|
||||||
|
|||||||
@@ -47,6 +47,7 @@ public struct ViewController: Sendable {
|
|||||||
self.logger = logger
|
self.logger = logger
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
extension ViewController: TestDependencyKey {
|
extension ViewController: TestDependencyKey {
|
||||||
|
|||||||
Reference in New Issue
Block a user