-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathPostgreSQLDriver.swift
134 lines (124 loc) · 4.56 KB
/
PostgreSQLDriver.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
import Fluent
import PostgreSQL
import Random
public final class Driver: Fluent.Driver {
// The string value for the default identifier key.
//
// The `idKey` will be used when `Model.find(_:)` or other find by
// identifier methods are used.
//
// This value is overriden by entities that implement the `Entity.idKey`
// static property.
public let idKey: String
// The default type for values stored against the identifier key.
//
// The `idType` will be accessed by those Entity implementations
// which do not themselves implement `Entity.idType`.
public let idType: IdentifierType
// The naming convetion to use for foreign id keys, table names, etc.
// ex: snake_case vs. camelCase.
public let keyNamingConvention: KeyNamingConvention
// The master MySQL Database for read/write
public let master: PostgreSQL.Database
// The read replicas for read only
public let readReplicas: [PostgreSQL.Database]
// Stores query logger
public var queryLogger: QueryLogger?
// Attempts to establish a connection to a PostgreSQL database engine
// running on host.
//
// - parameter masterHostname: May be either a host name or an IP address.
// Defaults to "localhost".
// - parameter user: The PostgreSQL login ID.
// - parameter password: Password for user.
// - parameter database: Database name.
// The connection sets the default database to this value.
// - parameter port: If port is not 0, the value is used as
// the port number for the TCP/IP connection.
// - parameter socket: If socket is not NULL, the string specifies
// the socket or named pipe to use.
//
// - throws: `Error.connection(String)` if the call to connection fails.
//
public convenience init(
masterHostname: String,
readReplicaHostnames: [String],
user: String,
password: String,
database: String,
port: Int = 5432,
idKey: String = "id",
idType: IdentifierType = .int,
keyNamingConvention: KeyNamingConvention = .snake_case
) throws {
let master = try PostgreSQL.Database(
hostname: masterHostname,
port: port,
database: database,
user: user,
password: password
)
let readReplicas: [PostgreSQL.Database] = try readReplicaHostnames.map { hostname in
return try PostgreSQL.Database(
hostname: hostname,
port: port,
database: database,
user: user,
password: password
)
}
self.init(
master: master,
readReplicas: readReplicas,
idKey: idKey,
idType: idType,
keyNamingConvention: keyNamingConvention
)
}
// Creates the driver from an already initialized database.
public init(
master: PostgreSQL.Database,
readReplicas: [PostgreSQL.Database] = [],
idKey: String = "id",
idType: IdentifierType = .int,
keyNamingConvention: KeyNamingConvention = .snake_case
) {
self.master = master
self.readReplicas = readReplicas
self.idKey = idKey
self.idType = idType
self.keyNamingConvention = keyNamingConvention
}
// Creates a connection for executing queries. This method is used to
// automatically create a connection if any Executor methods are called on
// the Driver.
public func makeConnection(_ type: ConnectionType) throws -> Fluent.Connection {
let database: PostgreSQL.Database
switch type {
case .read:
database = readReplicas.random ?? master
case .readWrite:
database = master
}
let conn = try Connection(database.makeConnection())
conn.queryLogger = queryLogger
return conn
}
}
extension Driver: Transactable {
/// Executes a PostgreSQL transaction on a single connection.
///
/// The argument supplied to the closure is the connection
/// to use for this transaction.
///
/// It may be ignored if you are using Fluent and not performing
/// complex threading.
public func transaction<R>(_ closure: (Fluent.Connection) throws -> R) throws -> R {
let conn = try master.makeConnection()
return try conn.transaction {
let wrapped = PostgreSQLDriver.Connection(conn)
wrapped.queryLogger = self.queryLogger
return try closure(wrapped)
}
}
}