2023-06-07 08:08:43 +00:00
|
|
|
# Simple async pool driver for postgress.
|
|
|
|
# Inspired by: https://github.com/treeform/pg/
|
2024-06-28 10:34:57 +00:00
|
|
|
{.push raises: [].}
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
import
|
|
|
|
std/[sequtils, nre, strformat],
|
|
|
|
results,
|
|
|
|
chronos,
|
|
|
|
chronos/threadsync,
|
|
|
|
chronicles,
|
|
|
|
strutils
|
2024-08-29 20:56:14 +00:00
|
|
|
import ./dbconn, ../common, ../../../waku_core/time
|
2023-06-07 08:08:43 +00:00
|
|
|
|
|
|
|
type
|
|
|
|
# Database connection pool
|
|
|
|
PgAsyncPool* = ref object
|
|
|
|
connString: string
|
|
|
|
maxConnections: int
|
2024-10-01 21:36:03 +00:00
|
|
|
conns: seq[DbConnWrapper]
|
|
|
|
busySignal: ThreadSignalPtr ## signal to wait while the pool is busy
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc new*(T: type PgAsyncPool, dbUrl: string, maxConnections: int): DatabaseResult[T] =
|
2023-06-22 09:27:40 +00:00
|
|
|
var connString: string
|
|
|
|
|
|
|
|
try:
|
|
|
|
let regex = re("""^postgres:\/\/([^:]+):([^@]+)@([^:]+):(\d+)\/(.+)$""")
|
2024-03-15 23:08:47 +00:00
|
|
|
let matches = find(dbUrl, regex).get.captures
|
2023-06-22 09:27:40 +00:00
|
|
|
let user = matches[0]
|
2024-03-15 23:08:47 +00:00
|
|
|
let password = matches[1]
|
2023-06-22 09:27:40 +00:00
|
|
|
let host = matches[2]
|
|
|
|
let port = matches[3]
|
|
|
|
let dbName = matches[4]
|
|
|
|
connString =
|
|
|
|
fmt"user={user} host={host} port={port} dbname={dbName} password={password}"
|
2024-03-15 23:08:47 +00:00
|
|
|
except KeyError, InvalidUnicodeError, RegexInternalError, ValueError, StudyError,
|
|
|
|
SyntaxError:
|
2023-06-22 09:27:40 +00:00
|
|
|
return err("could not parse postgres string: " & getCurrentExceptionMsg())
|
2023-06-07 08:08:43 +00:00
|
|
|
|
|
|
|
let pool = PgAsyncPool(
|
|
|
|
connString: connString,
|
|
|
|
maxConnections: maxConnections,
|
2024-10-01 21:36:03 +00:00
|
|
|
conns: newSeq[DbConnWrapper](0),
|
2023-06-07 08:08:43 +00:00
|
|
|
)
|
|
|
|
|
2023-06-22 09:27:40 +00:00
|
|
|
return ok(pool)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
|
|
|
func isBusy(pool: PgAsyncPool): bool =
|
2024-10-01 21:36:03 +00:00
|
|
|
return pool.conns.mapIt(it.isPgDbConnBusy()).allIt(it)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc close*(pool: PgAsyncPool): Future[Result[void, string]] {.async.} =
|
2023-06-07 08:08:43 +00:00
|
|
|
## Gracefully wait and close all openned connections
|
|
|
|
# wait for the connections to be released and close them, without
|
|
|
|
# blocking the async runtime
|
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
debug "close PgAsyncPool"
|
|
|
|
await allFutures(pool.conns.mapIt(it.futBecomeFree))
|
|
|
|
debug "closing all connection PgAsyncPool"
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
for i in 0 ..< pool.conns.len:
|
2024-10-01 21:36:03 +00:00
|
|
|
if pool.conns[i].isPgDbConnOpen():
|
|
|
|
pool.conns[i].closeDbConn().isOkOr:
|
|
|
|
return err("error in close PgAsyncPool: " & $error)
|
|
|
|
pool.conns[i].setPgDbConnOpen(false)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
|
|
|
pool.conns.setLen(0)
|
|
|
|
|
|
|
|
return ok()
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc getFirstFreeConnIndex(pool: PgAsyncPool): DatabaseResult[int] =
|
|
|
|
for index in 0 ..< pool.conns.len:
|
2024-10-01 21:36:03 +00:00
|
|
|
if pool.conns[index].isPgDbConnBusy():
|
2023-10-31 13:46:46 +00:00
|
|
|
continue
|
|
|
|
|
|
|
|
## Pick up the first free connection and set it busy
|
|
|
|
return ok(index)
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc getConnIndex(pool: PgAsyncPool): Future[DatabaseResult[int]] {.async.} =
|
2023-06-07 08:08:43 +00:00
|
|
|
## Waits for a free connection or create if max connections limits have not been reached.
|
|
|
|
## Returns the index of the free connection
|
|
|
|
|
2023-10-31 13:46:46 +00:00
|
|
|
if not pool.isBusy():
|
|
|
|
return pool.getFirstFreeConnIndex()
|
|
|
|
|
|
|
|
## Pool is busy then
|
|
|
|
if pool.conns.len == pool.maxConnections:
|
|
|
|
## Can't create more connections. Wait for a free connection without blocking the async runtime.
|
2024-10-01 21:36:03 +00:00
|
|
|
let busyFuts = pool.conns.mapIt(it.futBecomeFree)
|
|
|
|
discard await one(busyFuts)
|
2023-10-31 13:46:46 +00:00
|
|
|
|
|
|
|
return pool.getFirstFreeConnIndex()
|
|
|
|
elif pool.conns.len < pool.maxConnections:
|
|
|
|
## stablish a new connection
|
2024-10-01 21:36:03 +00:00
|
|
|
let dbConn = DbConnWrapper.new(pool.connString).valueOr:
|
|
|
|
return err("error creating DbConnWrapper: " & $error)
|
2023-11-07 12:38:37 +00:00
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
pool.conns.add(dbConn)
|
2023-11-07 12:38:37 +00:00
|
|
|
return ok(pool.conns.len - 1)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2023-09-06 17:16:37 +00:00
|
|
|
proc resetConnPool*(pool: PgAsyncPool): Future[DatabaseResult[void]] {.async.} =
|
|
|
|
## Forces closing the connection pool.
|
|
|
|
## This proc is intended to be called when the connection with the database
|
|
|
|
## got interrupted from the database side or a connectivity problem happened.
|
|
|
|
|
|
|
|
(await pool.close()).isOkOr:
|
|
|
|
return err("error in resetConnPool: " & error)
|
|
|
|
|
|
|
|
return ok()
|
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
const SlowQueryThreshold = 1.seconds
|
2024-08-29 20:56:14 +00:00
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc pgQuery*(
|
|
|
|
pool: PgAsyncPool,
|
|
|
|
query: string,
|
|
|
|
args: seq[string] = newSeq[string](0),
|
|
|
|
rowCallback: DataProc = nil,
|
2024-08-29 20:56:14 +00:00
|
|
|
requestId: string = "",
|
2024-03-15 23:08:47 +00:00
|
|
|
): Future[DatabaseResult[void]] {.async.} =
|
2023-11-07 12:38:37 +00:00
|
|
|
let connIndex = (await pool.getConnIndex()).valueOr:
|
|
|
|
return err("connRes.isErr in query: " & $error)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-08-29 20:56:14 +00:00
|
|
|
let queryStartTime = getNowInNanosecondTime()
|
2024-10-01 21:36:03 +00:00
|
|
|
let dbConnWrapper = pool.conns[connIndex]
|
2024-03-15 23:08:47 +00:00
|
|
|
defer:
|
2024-08-29 20:56:14 +00:00
|
|
|
let queryDuration = getNowInNanosecondTime() - queryStartTime
|
2024-10-01 21:36:03 +00:00
|
|
|
if queryDuration > SlowQueryThreshold.nanos:
|
2024-08-29 20:56:14 +00:00
|
|
|
debug "pgQuery slow query",
|
|
|
|
query_duration_secs = (queryDuration / 1_000_000_000), query, requestId
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
(await dbConnWrapper.dbConnQuery(sql(query), args, rowCallback, requestId)).isOkOr:
|
2023-10-31 13:46:46 +00:00
|
|
|
return err("error in asyncpool query: " & $error)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
|
|
|
return ok()
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
proc runStmt*(
|
|
|
|
pool: PgAsyncPool,
|
|
|
|
stmtName: string,
|
|
|
|
stmtDefinition: string,
|
|
|
|
paramValues: seq[string],
|
|
|
|
paramLengths: seq[int32],
|
|
|
|
paramFormats: seq[int32],
|
|
|
|
rowCallback: DataProc = nil,
|
2024-08-29 20:56:14 +00:00
|
|
|
requestId: string = "",
|
2024-03-15 23:08:47 +00:00
|
|
|
): Future[DatabaseResult[void]] {.async.} =
|
2023-11-07 12:38:37 +00:00
|
|
|
## Runs a stored statement, for performance purposes.
|
|
|
|
## The stored statements are connection specific and is a technique of caching a very common
|
|
|
|
## queries within the same connection.
|
|
|
|
##
|
|
|
|
## rowCallback != nil when it is expected to retrieve info from the database.
|
|
|
|
## rowCallback == nil for queries that change the database state.
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2023-11-07 12:38:37 +00:00
|
|
|
let connIndex = (await pool.getConnIndex()).valueOr:
|
|
|
|
return err("Error in runStmt: " & $error)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
let dbConnWrapper = pool.conns[connIndex]
|
2024-08-29 20:56:14 +00:00
|
|
|
let queryStartTime = getNowInNanosecondTime()
|
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
defer:
|
2024-08-29 20:56:14 +00:00
|
|
|
let queryDuration = getNowInNanosecondTime() - queryStartTime
|
2024-10-01 21:36:03 +00:00
|
|
|
if queryDuration > SlowQueryThreshold.nanos:
|
2024-08-29 20:56:14 +00:00
|
|
|
debug "runStmt slow query",
|
|
|
|
query_duration = queryDuration / 1_000_000_000,
|
|
|
|
query = stmtDefinition,
|
|
|
|
requestId
|
2023-06-07 08:08:43 +00:00
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
if not pool.conns[connIndex].containsPreparedStmt(stmtName):
|
2023-11-07 12:38:37 +00:00
|
|
|
# The connection doesn't have that statement yet. Let's create it.
|
|
|
|
# Each session/connection has its own prepared statements.
|
|
|
|
let res = catch:
|
|
|
|
let len = paramValues.len
|
2024-10-01 21:36:03 +00:00
|
|
|
discard dbConnWrapper.getDbConn().prepare(stmtName, sql(stmtDefinition), len)
|
2023-11-07 12:38:37 +00:00
|
|
|
|
|
|
|
if res.isErr():
|
|
|
|
return err("failed prepare in runStmt: " & res.error.msg)
|
|
|
|
|
2024-10-01 21:36:03 +00:00
|
|
|
pool.conns[connIndex].inclPreparedStmt(stmtName)
|
2023-11-07 12:38:37 +00:00
|
|
|
|
2024-03-15 23:08:47 +00:00
|
|
|
(
|
2024-10-01 21:36:03 +00:00
|
|
|
await dbConnWrapper.dbConnQueryPrepared(
|
|
|
|
stmtName, paramValues, paramLengths, paramFormats, rowCallback, requestId
|
2024-03-15 23:08:47 +00:00
|
|
|
)
|
2023-11-07 12:38:37 +00:00
|
|
|
).isOkOr:
|
|
|
|
return err("error in runStmt: " & $error)
|
2023-06-07 08:08:43 +00:00
|
|
|
|
|
|
|
return ok()
|