feat: Moves api controller to it's own module.
This commit is contained in:
@@ -8,6 +8,8 @@ let package = Package(
|
||||
],
|
||||
products: [
|
||||
.executable(name: "App", targets: ["App"]),
|
||||
.library(name: "ApiController", targets: ["ApiController"]),
|
||||
.library(name: "ApiControllerLive", targets: ["ApiControllerLive"]),
|
||||
.library(name: "SharedModels", targets: ["SharedModels"]),
|
||||
.library(name: "DatabaseClient", targets: ["DatabaseClient"]),
|
||||
.library(name: "DatabaseClientLive", targets: ["DatabaseClientLive"]),
|
||||
@@ -37,6 +39,7 @@ let package = Package(
|
||||
.executableTarget(
|
||||
name: "App",
|
||||
dependencies: [
|
||||
.target(name: "ApiControllerLive"),
|
||||
.target(name: "DatabaseClientLive"),
|
||||
.target(name: "ViewControllerLive"),
|
||||
.product(name: "Fluent", package: "fluent"),
|
||||
@@ -54,27 +57,24 @@ let package = Package(
|
||||
],
|
||||
swiftSettings: swiftSettings
|
||||
),
|
||||
// .testTarget(
|
||||
// name: "AppTests",
|
||||
// dependencies: [
|
||||
// .target(name: "App"),
|
||||
// .target(name: "HtmlSnapshotTesting"),
|
||||
// .product(name: "XCTVapor", package: "vapor")
|
||||
// ],
|
||||
// resources: [
|
||||
// .copy("__Snapshots__")
|
||||
// ],
|
||||
// swiftSettings: swiftSettings
|
||||
// ),
|
||||
.testTarget(
|
||||
name: "ViewRouteTests",
|
||||
.target(
|
||||
name: "ApiController",
|
||||
dependencies: [
|
||||
.target(name: "App"),
|
||||
.product(name: "VaporTesting", package: "vapor")
|
||||
.target(name: "SharedModels"),
|
||||
.product(name: "Dependencies", package: "swift-dependencies"),
|
||||
.product(name: "DependenciesMacros", package: "swift-dependencies")
|
||||
],
|
||||
|
||||
swiftSettings: swiftSettings
|
||||
),
|
||||
.target(
|
||||
name: "ApiControllerLive",
|
||||
dependencies: [
|
||||
.target(name: "ApiController"),
|
||||
.target(name: "DatabaseClient")
|
||||
],
|
||||
swiftSettings: swiftSettings
|
||||
),
|
||||
|
||||
.testTarget(
|
||||
name: "ApiRouteTests",
|
||||
dependencies: [
|
||||
@@ -153,6 +153,15 @@ let package = Package(
|
||||
resources: [
|
||||
.copy("__Snapshots__")
|
||||
],
|
||||
swiftSettings: swiftSettings
|
||||
),
|
||||
.testTarget(
|
||||
name: "ViewRouteTests",
|
||||
dependencies: [
|
||||
.target(name: "App"),
|
||||
.product(name: "VaporTesting", package: "vapor")
|
||||
],
|
||||
|
||||
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 Fluent
|
||||
import Logging
|
||||
import SharedModels
|
||||
import Vapor
|
||||
|
||||
private let apiMiddleware: [any Middleware] = [
|
||||
UserPasswordAuthenticator(),
|
||||
UserTokenAuthenticator(),
|
||||
UserSessionAuthenticator(),
|
||||
User.guardMiddleware()
|
||||
]
|
||||
extension ApiController: DependencyKey {
|
||||
public static var liveValue: ApiController {
|
||||
.init(json: { try await $0.respond() })
|
||||
}
|
||||
}
|
||||
|
||||
extension ApiRoute {
|
||||
var middleware: [any Middleware]? { apiMiddleware }
|
||||
private extension ApiController.Request {
|
||||
|
||||
func respond(request: Request) async throws -> any AsyncResponseEncodable {
|
||||
switch self {
|
||||
func respond() async throws -> (any Encodable)? {
|
||||
@Dependency(\.database) var database
|
||||
|
||||
switch 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):
|
||||
return try await route.handleApiRequest(request: request)
|
||||
return try await route.handleApiRequest(logger: logger)
|
||||
case let .user(route):
|
||||
return try await route.handleApiRequest(request: request)
|
||||
return try await route.handleApiRequest(logger: logger)
|
||||
case let .vendor(route):
|
||||
return try await route.handleApiRequest(request: request)
|
||||
return try await route.handleApiRequest(logger: logger)
|
||||
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
|
||||
|
||||
switch self {
|
||||
case let .delete(id: id):
|
||||
try await database.employees.delete(id)
|
||||
return HTTPStatus.ok
|
||||
return nil
|
||||
case let .create(employee):
|
||||
return try await database.employees.create(employee)
|
||||
case .index:
|
||||
return try await database.employees.fetchAll()
|
||||
case let .get(id: id):
|
||||
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
|
||||
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
|
||||
switch self {
|
||||
case let .delete(id: id):
|
||||
try await purchaseOrders.delete(id)
|
||||
return HTTPStatus.ok
|
||||
return nil
|
||||
case .index:
|
||||
return try await purchaseOrders.fetchAll()
|
||||
case let .create(purchaseOrder):
|
||||
return try await purchaseOrders.create(purchaseOrder)
|
||||
case let .get(id: id):
|
||||
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
|
||||
case let .page(page: page, limit: limit):
|
||||
@@ -78,21 +91,22 @@ extension ApiRoute.PurchaseOrderRoute {
|
||||
}
|
||||
|
||||
// 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
|
||||
switch self {
|
||||
case let .delete(id: id):
|
||||
try await users.delete(id)
|
||||
return HTTPStatus.ok
|
||||
return nil
|
||||
case let .create(user):
|
||||
return try await users.create(user)
|
||||
case .index:
|
||||
return try await users.fetchAll()
|
||||
case let .get(id: id):
|
||||
logger.error("User not found for id: \(id)")
|
||||
guard let user = try await users.get(id) else {
|
||||
throw Abort(.badRequest, reason: "Employee not found")
|
||||
throw ApiError(reason: "Employee not found")
|
||||
}
|
||||
return user
|
||||
// case let .login(user):
|
||||
@@ -103,18 +117,19 @@ extension ApiRoute.UserRoute {
|
||||
}
|
||||
}
|
||||
|
||||
extension ApiRoute.VendorRoute {
|
||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
||||
private extension ApiRoute.VendorRoute {
|
||||
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||
@Dependency(\.database.vendors) var vendors
|
||||
switch self {
|
||||
case let .delete(id: id):
|
||||
try await vendors.delete(id)
|
||||
return HTTPStatus.ok
|
||||
return nil
|
||||
case let .create(vendor):
|
||||
return try await vendors.create(vendor)
|
||||
case let .get(id: id):
|
||||
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
|
||||
case let .update(id: id, updates: updates):
|
||||
@@ -128,13 +143,13 @@ extension ApiRoute.VendorRoute {
|
||||
}
|
||||
}
|
||||
|
||||
extension ApiRoute.VendorBranchRoute {
|
||||
func handleApiRequest(request: Request) async throws -> any AsyncResponseEncodable {
|
||||
private extension ApiRoute.VendorBranchRoute {
|
||||
func handleApiRequest(logger: Logger) async throws -> (any Encodable)? {
|
||||
@Dependency(\.database.vendorBranches) var vendorBranches
|
||||
switch self {
|
||||
case let .delete(id: id):
|
||||
try await vendorBranches.delete(id)
|
||||
return HTTPStatus.ok
|
||||
return nil
|
||||
case let .create(branch):
|
||||
return try await vendorBranches.create(branch)
|
||||
case let .index(for: optionalVendorID):
|
||||
@@ -144,7 +159,8 @@ extension ApiRoute.VendorBranchRoute {
|
||||
return try await vendorBranches.fetchAll(.for(vendorID: vendorID))
|
||||
case let .get(id: id):
|
||||
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
|
||||
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 Dependencies
|
||||
import Vapor
|
||||
@@ -8,14 +9,17 @@ import ViewControllerLive
|
||||
struct DependenciesMiddleware: AsyncMiddleware {
|
||||
|
||||
private let values: DependencyValues.Continuation
|
||||
private let apiController: ApiController
|
||||
private let database: DatabaseClient
|
||||
private let viewController: ViewController
|
||||
|
||||
init(
|
||||
database: DatabaseClient,
|
||||
apiController: ApiController = .liveValue,
|
||||
viewController: ViewController = .liveValue
|
||||
) {
|
||||
self.values = withEscapedDependencies { $0 }
|
||||
self.apiController = apiController
|
||||
self.database = database
|
||||
self.viewController = viewController
|
||||
}
|
||||
@@ -23,6 +27,7 @@ struct DependenciesMiddleware: AsyncMiddleware {
|
||||
func respond(to request: Request, chainingTo next: any AsyncResponder) async throws -> Response {
|
||||
try await values.yield {
|
||||
try await withDependencies {
|
||||
$0.apiController = apiController
|
||||
$0.database = database
|
||||
$0.dateFormatter = .liveValue
|
||||
$0.viewController = viewController
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
import ApiController
|
||||
import DatabaseClientLive
|
||||
import Dependencies
|
||||
import Elementary
|
||||
@@ -15,6 +16,24 @@ public func configure(
|
||||
_ app: Application,
|
||||
makeDatabaseClient: @escaping (any Database) -> DatabaseClient = { .live(database: $0) }
|
||||
) 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`
|
||||
let corsConfiguration = CORSMiddleware.Configuration(
|
||||
allowedOrigin: .all,
|
||||
@@ -27,11 +46,13 @@ public func configure(
|
||||
|
||||
app.middleware.use(FileMiddleware(publicDirectory: app.directory.publicDirectory))
|
||||
app.middleware.use(app.sessions.middleware)
|
||||
app.middleware.use(DependenciesMiddleware(database: databaseClient))
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
app.lifecycle.use(BrowserSyncHandler())
|
||||
#endif
|
||||
|
||||
private func setupDatabase(
|
||||
on app: Application,
|
||||
factory makeDatabaseClient: @escaping (any Database) -> DatabaseClient
|
||||
) async throws -> DatabaseClient {
|
||||
switch app.environment {
|
||||
case .production, .development:
|
||||
let dbFileName = Environment.get("SQLITE_FILENAME") ?? "db.sqlite"
|
||||
@@ -46,8 +67,10 @@ public func configure(
|
||||
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.
|
||||
app.get { req in
|
||||
req.redirect(to: ViewRoute.router.path(for: .purchaseOrder(.index)))
|
||||
@@ -64,18 +87,16 @@ public func configure(
|
||||
},
|
||||
use: siteHandler
|
||||
)
|
||||
}
|
||||
|
||||
if app.environment != .testing {
|
||||
try await app.autoMigrate()
|
||||
}
|
||||
|
||||
private func addCommands(to app: Application) {
|
||||
#if DEBUG
|
||||
app.asyncCommands.use(SeedCommand(), as: "seed")
|
||||
#endif
|
||||
app.asyncCommands.use(GenerateAdminUserCommand(), as: "generate-admin")
|
||||
}
|
||||
|
||||
extension SiteRoute {
|
||||
private extension SiteRoute {
|
||||
|
||||
func middleware() -> [any Middleware]? {
|
||||
switch self {
|
||||
@@ -90,14 +111,16 @@ extension SiteRoute {
|
||||
}
|
||||
|
||||
@Sendable
|
||||
func siteHandler(
|
||||
private func siteHandler(
|
||||
request: Request,
|
||||
route: SiteRoute
|
||||
) async throws -> any AsyncResponseEncodable {
|
||||
@Dependency(\.apiController) var apiController
|
||||
@Dependency(\.viewController) var viewController
|
||||
|
||||
switch route {
|
||||
case let .api(route):
|
||||
return try await route.respond(request: request)
|
||||
return try await apiController.respond(route, request: request)
|
||||
case .health:
|
||||
return HTTPStatus.ok
|
||||
case let .view(route):
|
||||
|
||||
@@ -25,7 +25,9 @@ public extension DatabaseClient.Users {
|
||||
} login: { login in
|
||||
try login.validate()
|
||||
|
||||
var query = UserModel.query(on: database)
|
||||
var query = UserModel
|
||||
.query(on: database)
|
||||
.with(\.$token)
|
||||
|
||||
if let username = login.username {
|
||||
query = query.filter(\UserModel.$username == username)
|
||||
@@ -37,15 +39,19 @@ public extension DatabaseClient.Users {
|
||||
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(
|
||||
id: token.requireID(),
|
||||
userID: user.requireID(),
|
||||
value: token.value
|
||||
)
|
||||
return token
|
||||
|
||||
} logout: { id in
|
||||
guard let token = try await UserTokenModel.find(id, on: database)
|
||||
|
||||
@@ -5,6 +5,7 @@ import Foundation
|
||||
public enum ApiRoute: Sendable, Equatable {
|
||||
|
||||
case employee(EmployeeRoute)
|
||||
case login(User.Login)
|
||||
case purchaseOrder(PurchaseOrderRoute)
|
||||
case user(UserRoute)
|
||||
case vendor(VendorRoute)
|
||||
@@ -17,6 +18,11 @@ public enum ApiRoute: Sendable, Equatable {
|
||||
rootPath
|
||||
EmployeeRoute.router
|
||||
}
|
||||
Route(.case(Self.login)) {
|
||||
Path { "api"; "v1"; "login" }
|
||||
Method.post
|
||||
Body(.json(User.Login.self))
|
||||
}
|
||||
Route(.case(Self.purchaseOrder)) {
|
||||
rootPath
|
||||
PurchaseOrderRoute.router
|
||||
|
||||
@@ -4,7 +4,6 @@ import Foundation
|
||||
|
||||
public enum ViewRoute: Sendable, Equatable {
|
||||
|
||||
// case index
|
||||
case employee(EmployeeRoute)
|
||||
case login(LoginRoute)
|
||||
case purchaseOrder(PurchaseOrderRoute)
|
||||
@@ -13,9 +12,6 @@ public enum ViewRoute: Sendable, Equatable {
|
||||
case vendorBranch(VendorBranchRoute)
|
||||
|
||||
public static let router = OneOf {
|
||||
// Route(.case(Self.index)) {
|
||||
// Method.get
|
||||
// }
|
||||
Route(.case(Self.employee)) { EmployeeRoute.router }
|
||||
Route(.case(Self.login)) { LoginRoute.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.vendorBranch)) { VendorBranchRoute.router }
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public extension ViewRoute {
|
||||
|
||||
@@ -47,6 +47,7 @@ public struct ViewController: Sendable {
|
||||
self.logger = logger
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension ViewController: TestDependencyKey {
|
||||
|
||||
Reference in New Issue
Block a user