feat: Working on hummingbird app
This commit is contained in:
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"originHash" : "5fd82be053af28c2638151b00890a2e86e39fa8f499795235ca85c4080aa9364",
|
"originHash" : "68600988594927fcb25434476f598e1a813cb64030b7d9d8bf5b480e13a11989",
|
||||||
"pins" : [
|
"pins" : [
|
||||||
{
|
{
|
||||||
"identity" : "async-http-client",
|
"identity" : "async-http-client",
|
||||||
@@ -154,6 +154,15 @@
|
|||||||
"version" : "1.2.0"
|
"version" : "1.2.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"identity" : "swift-argument-parser",
|
||||||
|
"kind" : "remoteSourceControl",
|
||||||
|
"location" : "https://github.com/apple/swift-argument-parser.git",
|
||||||
|
"state" : {
|
||||||
|
"revision" : "41982a3656a71c768319979febd796c6fd111d5c",
|
||||||
|
"version" : "1.5.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"identity" : "swift-asn1",
|
"identity" : "swift-asn1",
|
||||||
"kind" : "remoteSourceControl",
|
"kind" : "remoteSourceControl",
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ let package = Package(
|
|||||||
.macOS(.v14)
|
.macOS(.v14)
|
||||||
],
|
],
|
||||||
products: [
|
products: [
|
||||||
|
.executable(name: "App", targets: ["App"]),
|
||||||
|
.executable(name: "HApp", targets: ["HApp"]),
|
||||||
.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"])
|
||||||
@@ -23,7 +25,9 @@ let package = Package(
|
|||||||
// 🔵 Non-blocking, event-driven networking for Swift. Used for custom executors
|
// 🔵 Non-blocking, event-driven networking for Swift. Used for custom executors
|
||||||
.package(url: "https://github.com/apple/swift-nio.git", from: "2.65.0"),
|
.package(url: "https://github.com/apple/swift-nio.git", from: "2.65.0"),
|
||||||
.package(url: "https://github.com/pointfreeco/swift-dependencies.git", from: "1.6.3"),
|
.package(url: "https://github.com/pointfreeco/swift-dependencies.git", from: "1.6.3"),
|
||||||
.package(url: "https://github.com/hummingbird-project/hummingbird-auth.git", from: "2.0.2")
|
.package(url: "https://github.com/hummingbird-project/hummingbird.git", from: "2.0.0"),
|
||||||
|
.package(url: "https://github.com/hummingbird-project/hummingbird-auth.git", from: "2.0.2"),
|
||||||
|
.package(url: "https://github.com/apple/swift-argument-parser.git", from: "1.3.0")
|
||||||
],
|
],
|
||||||
targets: [
|
targets: [
|
||||||
.executableTarget(
|
.executableTarget(
|
||||||
@@ -49,6 +53,14 @@ let package = Package(
|
|||||||
],
|
],
|
||||||
swiftSettings: swiftSettings
|
swiftSettings: swiftSettings
|
||||||
),
|
),
|
||||||
|
.executableTarget(
|
||||||
|
name: "HApp",
|
||||||
|
dependencies: [
|
||||||
|
.product(name: "Hummingbird", package: "hummingbird"),
|
||||||
|
.product(name: "ArgumentParser", package: "swift-argument-parser")
|
||||||
|
],
|
||||||
|
swiftSettings: swiftSettings
|
||||||
|
),
|
||||||
.target(
|
.target(
|
||||||
name: "DatabaseClient",
|
name: "DatabaseClient",
|
||||||
dependencies: [
|
dependencies: [
|
||||||
|
|||||||
@@ -10,16 +10,16 @@ public extension DatabaseClient.Users {
|
|||||||
.init { create in
|
.init { create in
|
||||||
let model = try create.toModel()
|
let model = try create.toModel()
|
||||||
try await model.save(on: database)
|
try await model.save(on: database)
|
||||||
return model.toDTO()
|
return try model.toDTO()
|
||||||
} delete: { id in
|
} delete: { id in
|
||||||
guard let model = try await UserModel.find(id, on: database) else {
|
guard let model = try await UserModel.find(id, on: database) else {
|
||||||
throw NotFoundError()
|
throw NotFoundError()
|
||||||
}
|
}
|
||||||
try await model.delete(on: database)
|
try await model.delete(on: database)
|
||||||
} fetchAll: {
|
} fetchAll: {
|
||||||
try await UserModel.query(on: database).all().map { $0.toDTO() }
|
try await UserModel.query(on: database).all().map { try $0.toDTO() }
|
||||||
} get: { id in
|
} get: { id in
|
||||||
try await UserModel.find(id, on: database).map { $0.toDTO() }
|
try await UserModel.find(id, on: database).map { try $0.toDTO() }
|
||||||
} login: { login in
|
} login: { login in
|
||||||
try login.validate()
|
try login.validate()
|
||||||
|
|
||||||
@@ -168,13 +168,13 @@ final class UserModel: Model, @unchecked Sendable {
|
|||||||
self.passwordHash = passwordHash
|
self.passwordHash = passwordHash
|
||||||
}
|
}
|
||||||
|
|
||||||
func toDTO() -> User {
|
func toDTO() throws -> User {
|
||||||
.init(
|
try .init(
|
||||||
id: id,
|
id: requireID(),
|
||||||
createdAt: createdAt,
|
|
||||||
email: email,
|
email: email,
|
||||||
updatedAt: updatedAt,
|
username: username,
|
||||||
username: username
|
createdAt: createdAt,
|
||||||
|
updatedAt: updatedAt
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ public extension DatabaseClient.VendorBranches {
|
|||||||
.init { create in
|
.init { create in
|
||||||
let model = try create.toModel()
|
let model = try create.toModel()
|
||||||
try await model.save(on: db)
|
try await model.save(on: db)
|
||||||
return model.toDTO()
|
return try model.toDTO()
|
||||||
} delete: { id in
|
} delete: { id in
|
||||||
guard let model = try await VendorBranchModel.find(id, on: db) else {
|
guard let model = try await VendorBranchModel.find(id, on: db) else {
|
||||||
throw NotFoundError()
|
throw NotFoundError()
|
||||||
@@ -29,22 +29,22 @@ public extension DatabaseClient.VendorBranches {
|
|||||||
.with(\.$branches)
|
.with(\.$branches)
|
||||||
.first()?
|
.first()?
|
||||||
.branches
|
.branches
|
||||||
.map { $0.toDTO() }
|
.map { try $0.toDTO() }
|
||||||
|
|
||||||
guard let branches else { throw NotFoundError() }
|
guard let branches else { throw NotFoundError() }
|
||||||
return branches
|
return branches
|
||||||
}
|
}
|
||||||
|
|
||||||
return try await query.all().map { $0.toDTO() }
|
return try await query.all().map { try $0.toDTO() }
|
||||||
} get: { id in
|
} get: { id in
|
||||||
try await VendorBranchModel.find(id, on: db).map { $0.toDTO() }
|
try await VendorBranchModel.find(id, on: db).map { try $0.toDTO() }
|
||||||
} update: { id, updates in
|
} update: { id, updates in
|
||||||
guard let model = try await VendorBranchModel.find(id, on: db) else {
|
guard let model = try await VendorBranchModel.find(id, on: db) else {
|
||||||
throw NotFoundError()
|
throw NotFoundError()
|
||||||
}
|
}
|
||||||
try model.applyUpdates(updates)
|
try model.applyUpdates(updates)
|
||||||
try await model.save(on: db)
|
try await model.save(on: db)
|
||||||
return model.toDTO()
|
return try model.toDTO()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,16 +53,19 @@ public extension DatabaseClient.VendorBranches {
|
|||||||
extension VendorBranch {
|
extension VendorBranch {
|
||||||
|
|
||||||
struct Migrate: AsyncMigration {
|
struct Migrate: AsyncMigration {
|
||||||
|
|
||||||
let name = "CreateVendorBranch"
|
let name = "CreateVendorBranch"
|
||||||
|
|
||||||
|
fileprivate typealias FieldKey = VendorBranch.Key.Field
|
||||||
|
|
||||||
func prepare(on database: any Database) async throws {
|
func prepare(on database: any Database) async throws {
|
||||||
try await database.schema(VendorBranchModel.schema)
|
try await database.schema(VendorBranchModel.schema)
|
||||||
.id()
|
.id()
|
||||||
.field("name", .string, .required)
|
.field(.string(FieldKey.name), .string, .required)
|
||||||
.field("vendor_id", .uuid, .required)
|
.field(.string(FieldKey.vendorID), .uuid, .required)
|
||||||
.field("created_at", .datetime)
|
.field(.string(FieldKey.createdAt), .datetime)
|
||||||
.field("updated_at", .datetime)
|
.field(.string(FieldKey.updatedAt), .datetime)
|
||||||
.foreignKey("vendor_id", references: VendorModel.schema, "id", onDelete: .cascade)
|
.foreignKey(.string(FieldKey.vendorID), references: VendorModel.schema, "id", onDelete: .cascade)
|
||||||
.create()
|
.create()
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,23 +99,36 @@ extension VendorBranch.Update {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private extension VendorBranch {
|
||||||
|
enum Key {
|
||||||
|
static let schema = "vendor_branch"
|
||||||
|
enum Field {
|
||||||
|
static let createdAt = "created_at"
|
||||||
|
static let name = "name"
|
||||||
|
static let updatedAt = "updated_at"
|
||||||
|
static let vendorID = "vendor_id"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
final class VendorBranchModel: Model, @unchecked Sendable {
|
final class VendorBranchModel: Model, @unchecked Sendable {
|
||||||
|
|
||||||
static let schema = "vendor_branch"
|
fileprivate typealias FieldKey = VendorBranch.Key.Field
|
||||||
|
static let schema = VendorBranch.Key.schema
|
||||||
|
|
||||||
@ID(key: .id)
|
@ID(key: .id)
|
||||||
var id: UUID?
|
var id: UUID?
|
||||||
|
|
||||||
@Field(key: "name")
|
@Field(key: .string(FieldKey.name))
|
||||||
var name: String
|
var name: String
|
||||||
|
|
||||||
@Timestamp(key: "created_at", on: .create)
|
@Timestamp(key: .string(FieldKey.createdAt), on: .create)
|
||||||
var createdAt: Date?
|
var createdAt: Date?
|
||||||
|
|
||||||
@Timestamp(key: "updated_at", on: .update)
|
@Timestamp(key: .string(FieldKey.updatedAt), on: .update)
|
||||||
var updatedAt: Date?
|
var updatedAt: Date?
|
||||||
|
|
||||||
@Parent(key: "vendor_id")
|
@Parent(key: .string(FieldKey.vendorID))
|
||||||
var vendor: VendorModel
|
var vendor: VendorModel
|
||||||
|
|
||||||
init() {}
|
init() {}
|
||||||
@@ -123,9 +139,9 @@ final class VendorBranchModel: Model, @unchecked Sendable {
|
|||||||
$vendor.id = vendorId
|
$vendor.id = vendorId
|
||||||
}
|
}
|
||||||
|
|
||||||
func toDTO() -> VendorBranch {
|
func toDTO() throws -> VendorBranch {
|
||||||
.init(
|
try .init(
|
||||||
id: id,
|
id: requireID(),
|
||||||
name: name,
|
name: name,
|
||||||
vendorID: $vendor.id,
|
vendorID: $vendor.id,
|
||||||
createdAt: createdAt,
|
createdAt: createdAt,
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ public extension DatabaseClient.Vendors {
|
|||||||
.init { create in
|
.init { create in
|
||||||
let model = try create.toModel()
|
let model = try create.toModel()
|
||||||
try await model.save(on: db)
|
try await model.save(on: db)
|
||||||
return model.toDTO()
|
return try model.toDTO()
|
||||||
} delete: { id in
|
} delete: { id in
|
||||||
guard let model = try await VendorModel.find(id, on: db) else {
|
guard let model = try await VendorModel.find(id, on: db) else {
|
||||||
throw NotFoundError()
|
throw NotFoundError()
|
||||||
@@ -27,7 +27,7 @@ public extension DatabaseClient.Vendors {
|
|||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
return try await query.all().map { $0.toDTO(includeBranches: withBranches) }
|
return try await query.all().map { try $0.toDTO(includeBranches: withBranches) }
|
||||||
|
|
||||||
} get: { id, request in
|
} get: { id, request in
|
||||||
var query = VendorModel.query(on: db).filter(\.$id == id)
|
var query = VendorModel.query(on: db).filter(\.$id == id)
|
||||||
@@ -35,14 +35,14 @@ public extension DatabaseClient.Vendors {
|
|||||||
if withBranches {
|
if withBranches {
|
||||||
query = query.with(\.$branches)
|
query = query.with(\.$branches)
|
||||||
}
|
}
|
||||||
return try await query.first().map { $0.toDTO(includeBranches: withBranches) }
|
return try await query.first().map { try $0.toDTO(includeBranches: withBranches) }
|
||||||
} update: { id, updates in
|
} update: { id, updates in
|
||||||
guard let model = try await VendorModel.find(id, on: db) else {
|
guard let model = try await VendorModel.find(id, on: db) else {
|
||||||
throw NotFoundError()
|
throw NotFoundError()
|
||||||
}
|
}
|
||||||
try model.applyUpdates(updates)
|
try model.applyUpdates(updates)
|
||||||
try await model.save(on: db)
|
try await model.save(on: db)
|
||||||
return model.toDTO()
|
return try model.toDTO()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -119,12 +119,12 @@ final class VendorModel: Model, @unchecked Sendable {
|
|||||||
self.name = name
|
self.name = name
|
||||||
}
|
}
|
||||||
|
|
||||||
func toDTO(includeBranches: Bool? = nil) -> Vendor {
|
func toDTO(includeBranches: Bool? = nil) throws -> Vendor {
|
||||||
.init(
|
try .init(
|
||||||
id: id,
|
id: requireID(),
|
||||||
name: name,
|
name: name,
|
||||||
branches: ($branches.value != nil && $branches.value!.count > 0)
|
branches: ($branches.value != nil && $branches.value!.count > 0)
|
||||||
? $branches.value!.map { $0.toDTO() }
|
? $branches.value!.map { try $0.toDTO() }
|
||||||
: [],
|
: [],
|
||||||
createdAt: createdAt,
|
createdAt: createdAt,
|
||||||
updatedAt: updatedAt
|
updatedAt: updatedAt
|
||||||
|
|||||||
80
Sources/HApp/App+build.swift
Normal file
80
Sources/HApp/App+build.swift
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
import Hummingbird
|
||||||
|
import Logging
|
||||||
|
|
||||||
|
public struct AppConfiguration {
|
||||||
|
|
||||||
|
public let hostname: String
|
||||||
|
public let port: Int
|
||||||
|
public let logLevel: Logger.Level?
|
||||||
|
|
||||||
|
public init(hostname: String, port: Int, logLevel: Logger.Level? = nil) {
|
||||||
|
self.hostname = hostname
|
||||||
|
self.port = port
|
||||||
|
self.logLevel = logLevel
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Build application
|
||||||
|
/// - Parameter arguments: application arguments
|
||||||
|
public func buildApplication(_ arguments: AppConfiguration) async throws -> some ApplicationProtocol {
|
||||||
|
let environment = Environment()
|
||||||
|
|
||||||
|
let logger = {
|
||||||
|
var logger = Logger(label: "Todos")
|
||||||
|
logger.logLevel = arguments.logLevel ??
|
||||||
|
environment.get("LOG_LEVEL").map { Logger.Level(rawValue: $0) ?? .info } ??
|
||||||
|
.info
|
||||||
|
|
||||||
|
return logger
|
||||||
|
}()
|
||||||
|
|
||||||
|
let router = Router()
|
||||||
|
|
||||||
|
// Add middleware
|
||||||
|
|
||||||
|
router.addMiddleware {
|
||||||
|
// logging middleware
|
||||||
|
LogRequestsMiddleware(.info)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add health endpoint
|
||||||
|
|
||||||
|
router.get("/health") { _, _ -> HTTPResponse.Status in
|
||||||
|
return .ok
|
||||||
|
}
|
||||||
|
|
||||||
|
// let router = buildRouter()
|
||||||
|
//
|
||||||
|
let app = Application(
|
||||||
|
router: router,
|
||||||
|
|
||||||
|
configuration: .init(
|
||||||
|
address: .hostname(arguments.hostname, port: arguments.port),
|
||||||
|
serverName: "Todos"
|
||||||
|
),
|
||||||
|
logger: logger
|
||||||
|
)
|
||||||
|
|
||||||
|
return app
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Build router
|
||||||
|
|
||||||
|
// func buildRouter() -> Router<AppRequestContext> {
|
||||||
|
// let router = Router()
|
||||||
|
//
|
||||||
|
// // Add middleware
|
||||||
|
//
|
||||||
|
// router.addMiddleware {
|
||||||
|
// // logging middleware
|
||||||
|
// LogRequestsMiddleware(.info)
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// // Add health endpoint
|
||||||
|
//
|
||||||
|
// router.get("/health") { _, _ -> HTTPResponse.Status in
|
||||||
|
// return .ok
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// return router
|
||||||
|
// }
|
||||||
29
Sources/HApp/App.swift
Normal file
29
Sources/HApp/App.swift
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import ArgumentParser
|
||||||
|
import Hummingbird
|
||||||
|
import Logging
|
||||||
|
|
||||||
|
@main
|
||||||
|
struct App: AsyncParsableCommand {
|
||||||
|
|
||||||
|
@Option(name: .shortAndLong)
|
||||||
|
var hostname: String = "127.0.0.1"
|
||||||
|
|
||||||
|
@Option(name: .shortAndLong)
|
||||||
|
var port: Int = 8080
|
||||||
|
|
||||||
|
@Option(name: .shortAndLong)
|
||||||
|
var logLevel: Logger.Level?
|
||||||
|
|
||||||
|
func run() async throws {
|
||||||
|
let app = try await buildApplication(.init(hostname: hostname, port: port, logLevel: logLevel))
|
||||||
|
try await app.runService()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Extend `Logger.Level` so it can be used as an argument
|
||||||
|
|
||||||
|
#if hasFeature(RetroactiveAttribute)
|
||||||
|
extension Logger.Level: @retroactive ExpressibleByArgument {}
|
||||||
|
#else
|
||||||
|
extension Logger.Level: ExpressibleByArgument {}
|
||||||
|
#endif
|
||||||
@@ -4,35 +4,32 @@ import Foundation
|
|||||||
public struct User: Codable, Equatable, Identifiable, Sendable {
|
public struct User: Codable, Equatable, Identifiable, Sendable {
|
||||||
|
|
||||||
public var id: UUID
|
public var id: UUID
|
||||||
public var createdAt: Date
|
|
||||||
public var email: String
|
public var email: String
|
||||||
public var updatedAt: Date
|
|
||||||
public var username: String
|
public var username: String
|
||||||
|
public var createdAt: Date?
|
||||||
|
public var updatedAt: Date?
|
||||||
|
|
||||||
public init(
|
public init(
|
||||||
id: UUID? = nil,
|
id: UUID,
|
||||||
createdAt: Date? = nil,
|
|
||||||
email: String,
|
email: String,
|
||||||
updatedAt: Date? = nil,
|
username: String,
|
||||||
username: String
|
createdAt: Date? = nil,
|
||||||
|
updatedAt: Date? = nil
|
||||||
) {
|
) {
|
||||||
@Dependency(\.date) var date
|
self.id = id
|
||||||
@Dependency(\.uuid) var uuid
|
self.createdAt = createdAt
|
||||||
|
|
||||||
self.id = id ?? uuid()
|
|
||||||
self.createdAt = createdAt ?? date.now
|
|
||||||
self.email = email
|
self.email = email
|
||||||
self.updatedAt = updatedAt ?? date.now
|
self.updatedAt = updatedAt
|
||||||
self.username = username
|
self.username = username
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public extension User {
|
// public extension User {
|
||||||
static var mocks: [Self] {
|
// static var mocks: [Self] {
|
||||||
[
|
// [
|
||||||
.init(email: "blob@test.com", username: "blob"),
|
// .init(email: "blob@test.com", username: "blob"),
|
||||||
.init(email: "blob-jr@test.com", username: "blob-jr"),
|
// .init(email: "blob-jr@test.com", username: "blob-jr"),
|
||||||
.init(email: "blob-sr@test.com", username: "blob-sr")
|
// .init(email: "blob-sr@test.com", username: "blob-sr")
|
||||||
]
|
// ]
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|||||||
@@ -5,34 +5,31 @@ public struct Vendor: Codable, Equatable, Identifiable, Sendable {
|
|||||||
public var id: UUID
|
public var id: UUID
|
||||||
public var name: String
|
public var name: String
|
||||||
public var branches: [VendorBranch]?
|
public var branches: [VendorBranch]?
|
||||||
public var createdAt: Date
|
public var createdAt: Date?
|
||||||
public var updatedAt: Date
|
public var updatedAt: Date?
|
||||||
|
|
||||||
public init(
|
public init(
|
||||||
id: UUID? = nil,
|
id: UUID,
|
||||||
name: String,
|
name: String,
|
||||||
branches: [VendorBranch]? = nil,
|
branches: [VendorBranch]? = nil,
|
||||||
createdAt: Date? = nil,
|
createdAt: Date? = nil,
|
||||||
updatedAt: Date? = nil
|
updatedAt: Date? = nil
|
||||||
) {
|
) {
|
||||||
@Dependency(\.date) var date
|
self.id = id
|
||||||
@Dependency(\.uuid) var uuid
|
|
||||||
|
|
||||||
self.id = id ?? uuid()
|
|
||||||
self.name = name
|
self.name = name
|
||||||
self.branches = branches
|
self.branches = branches
|
||||||
self.createdAt = createdAt ?? date.now
|
self.createdAt = createdAt
|
||||||
self.updatedAt = updatedAt ?? date.now
|
self.updatedAt = updatedAt
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public extension Vendor {
|
// public extension Vendor {
|
||||||
|
//
|
||||||
static var mocks: [Self] {
|
// static var mocks: [Self] {
|
||||||
[
|
// [
|
||||||
.init(name: "Corken"),
|
// .init(name: "Corken"),
|
||||||
.init(name: "Johnstone"),
|
// .init(name: "Johnstone"),
|
||||||
.init(name: "Winstel Controls")
|
// .init(name: "Winstel Controls")
|
||||||
]
|
// ]
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|||||||
@@ -5,23 +5,20 @@ public struct VendorBranch: Codable, Equatable, Identifiable, Sendable {
|
|||||||
public var id: UUID
|
public var id: UUID
|
||||||
public var name: String
|
public var name: String
|
||||||
public var vendorID: Vendor.ID
|
public var vendorID: Vendor.ID
|
||||||
public var createdAt: Date
|
public var createdAt: Date?
|
||||||
public var updatedAt: Date
|
public var updatedAt: Date?
|
||||||
|
|
||||||
public init(
|
public init(
|
||||||
id: UUID? = nil,
|
id: UUID,
|
||||||
name: String,
|
name: String,
|
||||||
vendorID: Vendor.ID,
|
vendorID: Vendor.ID,
|
||||||
createdAt: Date? = nil,
|
createdAt: Date? = nil,
|
||||||
updatedAt: Date? = nil
|
updatedAt: Date? = nil
|
||||||
) {
|
) {
|
||||||
@Dependency(\.date) var date
|
self.id = id
|
||||||
@Dependency(\.uuid) var uuid
|
|
||||||
|
|
||||||
self.id = id ?? uuid()
|
|
||||||
self.name = name
|
self.name = name
|
||||||
self.vendorID = vendorID
|
self.vendorID = vendorID
|
||||||
self.createdAt = createdAt ?? date.now
|
self.createdAt = createdAt
|
||||||
self.updatedAt = updatedAt ?? date.now
|
self.updatedAt = updatedAt
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user