feat: Moves api controller to it's own module.

This commit is contained in:
2025-01-25 15:54:02 -05:00
parent 67e689b51e
commit 0fad024350
14 changed files with 234 additions and 80 deletions

View File

@@ -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
) )
], ],

View 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()
}

View File

@@ -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
}

View 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))
)
}
}

View 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
}
}
}

View File

@@ -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

View File

@@ -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):

View File

@@ -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)

View File

@@ -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

View File

@@ -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 {

View File

@@ -47,6 +47,7 @@ public struct ViewController: Sendable {
self.logger = logger self.logger = logger
} }
} }
} }
extension ViewController: TestDependencyKey { extension ViewController: TestDependencyKey {