import Dependencies import DependenciesMacros import Fluent import Foundation import ManualDCore import Validations extension DatabaseClient.Rooms: TestDependencyKey { public static let testValue = Self() public static func live(database: any Database) -> Self { .init( create: { projectID, request in let model = try request.toModel(projectID: projectID) try await model.validateAndSave(on: database) return try model.toDTO() }, createMany: { projectID, rooms in try await RoomModel.createMany(projectID: projectID, rooms: rooms, on: database) }, createFromCSV: { projectID, rows in database.logger.debug("\nCreate From CSV rows: \(rows)\n") // Filter out rows that delegate their airflow / load to another room, // these need to be created last. let rowsThatDelegate = rows.filter({ $0.delegatedToName != nil && $0.delegatedToName != "" }) // Filter out the rest of the rooms that don't delegate their airflow / loads. let initialRooms = rows.filter({ $0.delegatedToName == nil || $0.delegatedToName == "" }) .map(\.createModel) database.logger.debug("\nInitial rows: \(initialRooms)\n") let initialCreated = try await RoomModel.createMany( projectID: projectID, rooms: initialRooms, on: database ) database.logger.debug("\nInitially created rows: \(initialCreated)\n") let roomsThatDelegateModels = try rowsThatDelegate.reduce(into: [Room.Create]()) { array, row in database.logger.debug("\n\(row.name), delegating to: \(row.delegatedToName!)\n") guard let created = initialCreated.first(where: { $0.name == row.delegatedToName }) else { database.logger.debug( "\nUnable to find created room with name: \(row.delegatedToName!)\n" ) throw NotFoundError() } array.append( Room.Create.init( name: row.name, level: row.level, heatingLoad: row.heatingLoad, coolingTotal: row.coolingTotal, coolingSensible: row.coolingSensible, registerCount: 0, delegatedTo: created.id ) ) } return try await RoomModel.createMany( projectID: projectID, rooms: roomsThatDelegateModels, on: database ) + initialCreated }, delete: { id in guard let model = try await RoomModel.find(id, on: database) else { throw NotFoundError() } try await model.delete(on: database) }, deleteRectangularSize: { roomID, rectangularDuctID in guard let model = try await RoomModel.find(roomID, on: database) else { throw NotFoundError() } model.rectangularSizes?.removeAll { $0.id == rectangularDuctID } if model.rectangularSizes?.count == 0 { model.rectangularSizes = nil } if model.hasChanges { try await model.validateAndSave(on: database) } return try model.toDTO() }, get: { id in try await RoomModel.find(id, on: database).map { try $0.toDTO() } }, fetch: { projectID in try await RoomModel.query(on: database) .with(\.$project) .filter(\.$project.$id, .equal, projectID) .sort(\.$name, .ascending) .all() .map { try $0.toDTO() } }, update: { id, updates in guard let model = try await RoomModel.find(id, on: database) else { throw NotFoundError() } model.applyUpdates(updates) if model.hasChanges { try await model.validateAndSave(on: database) } return try model.toDTO() }, updateRectangularSize: { id, size in guard let model = try await RoomModel.find(id, on: database) else { throw NotFoundError() } var rectangularSizes = model.rectangularSizes ?? [] rectangularSizes.removeAll { $0.id == size.id } rectangularSizes.append(size) model.rectangularSizes = rectangularSizes try await model.save(on: database) return try model.toDTO() } ) } } extension Room.CSV.Row { fileprivate var createModel: Room.Create { assert(delegatedToName == nil || delegatedToName == "") return .init( name: name, level: level, heatingLoad: heatingLoad, coolingTotal: coolingTotal, coolingSensible: coolingSensible, registerCount: registerCount, delegatedTo: nil ) } } extension RoomModel { fileprivate static func createMany( projectID: Project.ID, rooms: [Room.Create], on database: any Database ) async throws -> [Room] { try await rooms.asyncMap { request in let model = try request.toModel(projectID: projectID) try await model.validateAndSave(on: database) return try model.toDTO() } } } extension Room.Create { func toModel(projectID: Project.ID) throws -> RoomModel { var registerCount = registerCount // Set register count appropriately when delegatedTo is set / changes. if delegatedTo != nil { registerCount = 0 } else if registerCount == 0 { registerCount = 1 } return .init( name: name, level: level?.rawValue, heatingLoad: heatingLoad, coolingLoad: coolingLoad, registerCount: registerCount, delegetedToID: delegatedTo, projectID: projectID ) } } extension Room { struct Migrate: AsyncMigration { let name = "CreateRoom" func prepare(on database: any Database) async throws { try await database.schema(RoomModel.schema) .id() .field("name", .string, .required) .field("level", .int8) .field("heatingLoad", .double, .required) .field("coolingLoad", .dictionary, .required) .field("registerCount", .int8, .required) .field("delegatedToID", .uuid, .references(RoomModel.schema, "id")) .field("rectangularSizes", .array) .field("createdAt", .datetime) .field("updatedAt", .datetime) .field( "projectID", .uuid, .required, .references(ProjectModel.schema, "id", onDelete: .cascade) ) .unique(on: "projectID", "name") .create() } func revert(on database: any Database) async throws { try await database.schema(RoomModel.schema).delete() } } } final class RoomModel: Model, @unchecked Sendable, Validatable { static let schema = "room" @ID(key: .id) var id: UUID? @Field(key: "name") var name: String @Field(key: "level") var level: Int? @Field(key: "heatingLoad") var heatingLoad: Double @Field(key: "coolingLoad") var coolingLoad: Room.CoolingLoad @Field(key: "registerCount") var registerCount: Int @OptionalParent(key: "delegatedToID") var room: RoomModel? @Field(key: "rectangularSizes") var rectangularSizes: [Room.RectangularSize]? @Timestamp(key: "createdAt", on: .create, format: .iso8601) var createdAt: Date? @Timestamp(key: "updatedAt", on: .update, format: .iso8601) var updatedAt: Date? @Parent(key: "projectID") var project: ProjectModel init() {} init( id: UUID? = nil, name: String, level: Int? = nil, heatingLoad: Double, coolingLoad: Room.CoolingLoad, registerCount: Int, delegetedToID: UUID? = nil, rectangularSizes: [Room.RectangularSize]? = nil, createdAt: Date? = nil, updatedAt: Date? = nil, projectID: Project.ID ) { self.id = id self.name = name self.level = level self.heatingLoad = heatingLoad self.coolingLoad = coolingLoad self.registerCount = registerCount $room.id = delegetedToID self.rectangularSizes = rectangularSizes self.createdAt = createdAt self.updatedAt = updatedAt $project.id = projectID } func toDTO() throws -> Room { try .init( id: requireID(), projectID: $project.id, name: name, level: level.map(Room.Level.init(rawValue:)), heatingLoad: heatingLoad, coolingLoad: coolingLoad, registerCount: registerCount, delegatedTo: $room.id, rectangularSizes: rectangularSizes, createdAt: createdAt!, updatedAt: updatedAt! ) } func applyUpdates(_ updates: Room.Update) { if let name = updates.name, name != self.name { self.name = name } if let level = updates.level?.rawValue, level != self.level { self.level = level } if let heatingLoad = updates.heatingLoad, heatingLoad != self.heatingLoad { self.heatingLoad = heatingLoad } if let coolingLoad = updates.coolingLoad, coolingLoad != self.coolingLoad { self.coolingLoad = coolingLoad } if let registerCount = updates.registerCount, registerCount != self.registerCount { self.registerCount = registerCount } if let rectangularSizes = updates.rectangularSizes, rectangularSizes != self.rectangularSizes { self.rectangularSizes = rectangularSizes } } var body: some Validation { Validator.accumulating { Validator.validate(\.name, with: .notEmpty()) .errorLabel("Name", inline: true) Validator.validate(\.heatingLoad, with: .greaterThanOrEquals(0)) .errorLabel("Heating Load", inline: true) Validator.validate(\.coolingLoad) .errorLabel("Cooling Load", inline: true) Validator.validate(\.registerCount, with: .greaterThanOrEquals($room.id == nil ? 1 : 0)) .errorLabel("Register Count", inline: true) Validator.validate(\.rectangularSizes) } } func validateAndSave(on database: Database) async throws { try self.validate() if let delegateTo = $room.id { guard let parent = try await RoomModel .query(on: database) .with(\.$room) .filter(\.$id == delegateTo) .first() else { throw ValidationError("Can not find room: \(delegateTo), to delegate airflow to.") } guard parent.$room.id == nil else { throw ValidationError( """ Attempting to delegate to: \(parent.name), that delegates to: \(parent.$room.name) Unable to delegate airflow to a room that already delegates it's airflow. """ ) } } try await save(on: database) } } extension Room.CoolingLoad: Validatable { public var body: some Validation { Validator.accumulating { // Ensure that at least one of the values is not nil. Validator.oneOf { Validator.validate(\.total, with: .notNil()) .errorLabel("Total or Sensible", inline: true) Validator.validate(\.sensible, with: .notNil()) .errorLabel("Total or Sensible", inline: true) } Validator.validate(\.total, with: Double.greaterThan(0).optional()) Validator.validate(\.sensible, with: Double.greaterThan(0).optional()) } } } extension Room.RectangularSize: Validatable { public var body: some Validation { Validator.accumulating { Validator.validate(\.register, with: Int.greaterThanOrEquals(1).optional()) .errorLabel("Register", inline: true) Validator.validate(\.height, with: Int.greaterThanOrEquals(1)) .errorLabel("Height", inline: true) } } }