import Dependencies import DependenciesMacros import Fluent import Foundation import ManualDCore import Validations extension DatabaseClient.Projects: TestDependencyKey { public static let testValue = Self() public static func live(database: any Database) -> Self { .init( create: { userID, request in let model = request.toModel(userID: userID) try await model.validateAndSave(on: database) return try model.toDTO() }, delete: { id in guard let model = try await ProjectModel.find(id, on: database) else { throw NotFoundError() } try await model.delete(on: database) }, detail: { id in let model = try await ProjectModel.fetchDetail(for: id, on: database) // TODO: Different error ?? guard let equipmentInfo = model.equipment else { return nil } let trunks = try model.trunks.toDTO() return try .init( project: model.toDTO(), componentLosses: model.componentLosses.map { try $0.toDTO() }, equipmentInfo: equipmentInfo.toDTO(), equivalentLengths: model.equivalentLengths.map { try $0.toDTO() }, rooms: model.rooms.map { try $0.toDTO() }, trunks: trunks ) }, get: { id in try await ProjectModel.find(id, on: database).map { try $0.toDTO() } }, getCompletedSteps: { id in let model = try await ProjectModel.fetchDetail(for: id, on: database) var equivalentLengthsCompleted = false if model.equivalentLengths.filter({ $0.type == "supply" }).first != nil, model.equivalentLengths.filter({ $0.type == "return" }).first != nil { equivalentLengthsCompleted = true } return .init( equipmentInfo: model.equipment != nil, rooms: model.rooms.count > 0, equivalentLength: equivalentLengthsCompleted, frictionRate: model.componentLosses.count > 0 ) }, getSensibleHeatRatio: { id in guard let model = try await ProjectModel.query(on: database) .field(\.$id) .field(\.$sensibleHeatRatio) .filter(\.$id == id) .first() else { throw NotFoundError() } return model.sensibleHeatRatio }, fetch: { userID, request in try await ProjectModel.query(on: database) .sort(\.$createdAt, .descending) .with(\.$user) .filter(\.$user.$id == userID) .paginate(request) .map { try $0.toDTO() } }, update: { id, updates in guard let model = try await ProjectModel.find(id, on: database) else { throw NotFoundError() } model.applyUpdates(updates) if model.hasChanges { try await model.validateAndSave(on: database) } return try model.toDTO() } ) } } extension Project.Create { func toModel(userID: User.ID) -> ProjectModel { return .init( name: name, streetAddress: streetAddress, city: city, state: state, zipCode: zipCode, userID: userID ) } } extension Project { struct Migrate: AsyncMigration { let name = "CreateProject" func prepare(on database: any Database) async throws { try await database.schema(ProjectModel.schema) .id() .field("name", .string, .required) .field("streetAddress", .string, .required) .field("city", .string, .required) .field("state", .string, .required) .field("zipCode", .string, .required) .field("sensibleHeatRatio", .double) .field("createdAt", .string) .field("updatedAt", .string) .field("userID", .uuid, .required, .references(UserModel.schema, "id", onDelete: .cascade)) .unique(on: "userID", "name") .create() } func revert(on database: any Database) async throws { try await database.schema(ProjectModel.schema).delete() } } } // The Database model. final class ProjectModel: Model, @unchecked Sendable { static let schema = "project" @ID(key: .id) var id: UUID? @Field(key: "name") var name: String @Field(key: "streetAddress") var streetAddress: String @Field(key: "city") var city: String @Field(key: "state") var state: String @Field(key: "zipCode") var zipCode: String @Field(key: "sensibleHeatRatio") var sensibleHeatRatio: Double? @Timestamp(key: "createdAt", on: .create, format: .iso8601) var createdAt: Date? @Timestamp(key: "updatedAt", on: .update, format: .iso8601) var updatedAt: Date? @Children(for: \.$project) var componentLosses: [ComponentLossModel] @OptionalChild(for: \.$project) var equipment: EquipmentModel? @Children(for: \.$project) var equivalentLengths: [EffectiveLengthModel] @Children(for: \.$project) var rooms: [RoomModel] @Children(for: \.$project) var trunks: [TrunkModel] @Parent(key: "userID") var user: UserModel init() {} init( id: UUID? = nil, name: String, streetAddress: String, city: String, state: String, zipCode: String, sensibleHeatRatio: Double? = nil, userID: User.ID, createdAt: Date? = nil, updatedAt: Date? = nil ) { self.id = id self.name = name self.streetAddress = streetAddress self.city = city self.state = state self.zipCode = zipCode self.sensibleHeatRatio = sensibleHeatRatio $user.id = userID self.createdAt = createdAt self.updatedAt = updatedAt } func toDTO() throws -> Project { try .init( id: requireID(), name: name, streetAddress: streetAddress, city: city, state: state, zipCode: zipCode, sensibleHeatRatio: sensibleHeatRatio, createdAt: createdAt!, updatedAt: updatedAt! ) } func applyUpdates(_ updates: Project.Update) { if let name = updates.name, name != self.name { self.name = name } if let streetAddress = updates.streetAddress, streetAddress != self.streetAddress { self.streetAddress = streetAddress } if let city = updates.city, city != self.city { self.city = city } if let state = updates.state, state != self.state { self.state = state } if let zipCode = updates.zipCode, zipCode != self.zipCode { self.zipCode = zipCode } if let sensibleHeatRatio = updates.sensibleHeatRatio, sensibleHeatRatio != self.sensibleHeatRatio { self.sensibleHeatRatio = sensibleHeatRatio } } /// Returns a ``ProjectModel`` with all the relations eagerly loaded. static func fetchDetail( for projectID: Project.ID, on database: any Database ) async throws -> ProjectModel { guard let model = try await ProjectModel.query(on: database) .with(\.$componentLosses) .with(\.$equipment) .with(\.$equivalentLengths) .with(\.$rooms) .with( \.$trunks, { trunk in trunk.with( \.$rooms, { $0.with(\.$room) } ) } ) .filter(\.$id == projectID) .first() else { throw NotFoundError() } return model } } extension ProjectModel: Validatable { var body: some Validation { Validator.accumulating { Validator.validate(\.name, with: .notEmpty()) .errorLabel("Name", inline: true) Validator.validate(\.streetAddress, with: .notEmpty()) .errorLabel("Address", inline: true) Validator.validate(\.city, with: .notEmpty()) .errorLabel("City", inline: true) Validator.validate(\.state, with: .notEmpty()) .errorLabel("State", inline: true) Validator.validate(\.zipCode, with: .notEmpty()) .errorLabel("Zip", inline: true) Validator.validate(\.sensibleHeatRatio) { Validator { Double.greaterThan(0) Double.lessThanOrEquals(1.0) } .optional() } .errorLabel("Sensible Heat Ratio", inline: true) } } }