Use a connection provider rather than long-lived connections.
The previous implementation expected to work with a context object that contained a field `conn` that represented a valid database connection. However, this required the caller to manage the connection's lifecycle (close, renew, etc.). Now we expect to receive a context object that provides a `withConn` procedure or template that accepts a statement block and provides a `conn` variable to that code block. For example: createRecord(db: DbContext): Record = # withConn must be defined for DbContext db.withConn: # conn must be injected into the statement block context conn.exec(sql("INSERT INTO...")) In addition, this change provides a connection pooling mechanism (`DbConnPool`) as a default implementation for this hypothetical DbContext. There is also a new function `initPool` that will create an DbConnPool instance. Callers of this library can modify their DbContext objects to extend DbConnPool or simply be a type alias of DbConnPool.
This commit is contained in:
parent
8aad3cdb79
commit
3e19b3628d
@ -1,6 +1,6 @@
|
|||||||
# Package
|
# Package
|
||||||
|
|
||||||
version = "0.3.6"
|
version = "1.0.0"
|
||||||
author = "Jonathan Bernard"
|
author = "Jonathan Bernard"
|
||||||
description = "Lightweight Postgres ORM for Nim."
|
description = "Lightweight Postgres ORM for Nim."
|
||||||
license = "GPL-3.0"
|
license = "GPL-3.0"
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
import db_postgres, macros, options, sequtils, strutils, uuids
|
import std/db_postgres, std/macros, std/options, std/sequtils, std/strutils
|
||||||
import namespaced_logging
|
import namespaced_logging, uuids
|
||||||
|
|
||||||
from unicode import capitalize
|
from std/unicode import capitalize
|
||||||
|
|
||||||
|
import ./fiber_orm/pool
|
||||||
import ./fiber_orm/util
|
import ./fiber_orm/util
|
||||||
|
|
||||||
export
|
export
|
||||||
|
pool,
|
||||||
util.columnNamesForModel,
|
util.columnNamesForModel,
|
||||||
util.dbFormat,
|
util.dbFormat,
|
||||||
util.dbNameToIdent,
|
util.dbNameToIdent,
|
||||||
@ -18,7 +21,7 @@ type NotFoundError* = object of CatchableError
|
|||||||
var logNs {.threadvar.}: LoggingNamespace
|
var logNs {.threadvar.}: LoggingNamespace
|
||||||
|
|
||||||
template log(): untyped =
|
template log(): untyped =
|
||||||
if logNs.isNil: logNs = initLoggingNamespace(name = "fiber_orm", level = lvlDebug)
|
if logNs.isNil: logNs = initLoggingNamespace(name = "fiber_orm", level = lvlNotice)
|
||||||
logNs
|
logNs
|
||||||
|
|
||||||
proc newMutateClauses(): MutateClauses =
|
proc newMutateClauses(): MutateClauses =
|
||||||
@ -123,14 +126,27 @@ macro generateProcsForModels*(dbType: type, modelTypes: openarray[type]): untype
|
|||||||
let deleteName = ident("delete" & modelName)
|
let deleteName = ident("delete" & modelName)
|
||||||
let idType = typeOfColumn(t, "id")
|
let idType = typeOfColumn(t, "id")
|
||||||
result.add quote do:
|
result.add quote do:
|
||||||
proc `getName`*(db: `dbType`, id: `idType`): `t` = getRecord(db.conn, `t`, id)
|
proc `getName`*(db: `dbType`, id: `idType`): `t` =
|
||||||
proc `getAllName`*(db: `dbType`): seq[`t`] = getAllRecords(db.conn, `t`)
|
db.withConn: result = getRecord(conn, `t`, id)
|
||||||
|
|
||||||
|
proc `getAllName`*(db: `dbType`): seq[`t`] =
|
||||||
|
db.withConn: result = getAllRecords(conn, `t`)
|
||||||
|
|
||||||
proc `findWhereName`*(db: `dbType`, whereClause: string, values: varargs[string, dbFormat]): seq[`t`] =
|
proc `findWhereName`*(db: `dbType`, whereClause: string, values: varargs[string, dbFormat]): seq[`t`] =
|
||||||
return findRecordsWhere(db.conn, `t`, whereClause, values)
|
db.withConn:
|
||||||
proc `createName`*(db: `dbType`, rec: `t`): `t` = createRecord(db.conn, rec)
|
result = findRecordsWhere(conn, `t`, whereClause, values)
|
||||||
proc `updateName`*(db: `dbType`, rec: `t`): bool = updateRecord(db.conn, rec)
|
|
||||||
proc `deleteName`*(db: `dbType`, rec: `t`): bool = deleteRecord(db.conn, rec)
|
proc `createName`*(db: `dbType`, rec: `t`): `t` =
|
||||||
proc `deleteName`*(db: `dbType`, id: `idType`): bool = deleteRecord(db.conn, `t`, id)
|
db.withConn: result = createRecord(conn, rec)
|
||||||
|
|
||||||
|
proc `updateName`*(db: `dbType`, rec: `t`): bool =
|
||||||
|
db.withConn: result = updateRecord(conn, rec)
|
||||||
|
|
||||||
|
proc `deleteName`*(db: `dbType`, rec: `t`): bool =
|
||||||
|
db.withConn: result = deleteRecord(conn, rec)
|
||||||
|
|
||||||
|
proc `deleteName`*(db: `dbType`, id: `idType`): bool =
|
||||||
|
db.withConn: result = deleteRecord(conn, `t`, id)
|
||||||
|
|
||||||
macro generateLookup*(dbType: type, modelType: type, fields: seq[string]): untyped =
|
macro generateLookup*(dbType: type, modelType: type, fields: seq[string]): untyped =
|
||||||
let fieldNames = fields[1].mapIt($it)
|
let fieldNames = fields[1].mapIt($it)
|
||||||
@ -139,7 +155,7 @@ macro generateLookup*(dbType: type, modelType: type, fields: seq[string]): untyp
|
|||||||
# Create proc skeleton
|
# Create proc skeleton
|
||||||
result = quote do:
|
result = quote do:
|
||||||
proc `procName`*(db: `dbType`): seq[`modelType`] =
|
proc `procName`*(db: `dbType`): seq[`modelType`] =
|
||||||
return findRecordsBy(db.conn, `modelType`)
|
db.withConn: result = findRecordsBy(conn, `modelType`)
|
||||||
|
|
||||||
var callParams = quote do: @[]
|
var callParams = quote do: @[]
|
||||||
|
|
||||||
@ -149,10 +165,19 @@ macro generateLookup*(dbType: type, modelType: type, fields: seq[string]): untyp
|
|||||||
paramTuple.add(newColonExpr(ident("field"), newLit(identNameToDb(n))))
|
paramTuple.add(newColonExpr(ident("field"), newLit(identNameToDb(n))))
|
||||||
paramTuple.add(newColonExpr(ident("value"), ident(n)))
|
paramTuple.add(newColonExpr(ident("value"), ident(n)))
|
||||||
|
|
||||||
|
# Add the parameter to the outer call (the generated proc)
|
||||||
|
# result[3] is ProcDef -> [3]: FormalParams
|
||||||
result[3].add(newIdentDefs(ident(n), ident("string")))
|
result[3].add(newIdentDefs(ident(n), ident("string")))
|
||||||
|
|
||||||
|
# Build up the AST for the inner procedure call
|
||||||
callParams[1].add(paramTuple)
|
callParams[1].add(paramTuple)
|
||||||
|
|
||||||
result[6][0][0].add(callParams)
|
# Add the call params to the inner procedure call
|
||||||
|
# result[6][0][1][0][1] is
|
||||||
|
# ProcDef -> [6]: StmtList (body) -> [0]: Call ->
|
||||||
|
# [1]: StmtList (withConn body) -> [0]: Asgn (result =) ->
|
||||||
|
# [1]: Call (inner findRecords invocation)
|
||||||
|
result[6][0][1][0][1].add(callParams)
|
||||||
|
|
||||||
macro generateProcsForFieldLookups*(dbType: type, modelsAndFields: openarray[tuple[t: type, fields: seq[string]]]): untyped =
|
macro generateProcsForFieldLookups*(dbType: type, modelsAndFields: openarray[tuple[t: type, fields: seq[string]]]): untyped =
|
||||||
result = newStmtList()
|
result = newStmtList()
|
||||||
@ -166,7 +191,7 @@ macro generateProcsForFieldLookups*(dbType: type, modelsAndFields: openarray[tup
|
|||||||
# Create proc skeleton
|
# Create proc skeleton
|
||||||
let procDefAST = quote do:
|
let procDefAST = quote do:
|
||||||
proc `procName`*(db: `dbType`): seq[`modelType`] =
|
proc `procName`*(db: `dbType`): seq[`modelType`] =
|
||||||
return findRecordsBy(db.conn, `modelType`)
|
db.withConn: result = findRecordsBy(conn, `modelType`)
|
||||||
|
|
||||||
var callParams = quote do: @[]
|
var callParams = quote do: @[]
|
||||||
|
|
||||||
@ -179,6 +204,28 @@ macro generateProcsForFieldLookups*(dbType: type, modelsAndFields: openarray[tup
|
|||||||
procDefAST[3].add(newIdentDefs(ident(n), ident("string")))
|
procDefAST[3].add(newIdentDefs(ident(n), ident("string")))
|
||||||
callParams[1].add(paramTuple)
|
callParams[1].add(paramTuple)
|
||||||
|
|
||||||
procDefAST[6][0][0].add(callParams)
|
procDefAST[6][0][1][0][1].add(callParams)
|
||||||
|
|
||||||
result.add procDefAST
|
result.add procDefAST
|
||||||
|
|
||||||
|
proc initPool*(
|
||||||
|
connect: proc(): DbConn,
|
||||||
|
poolSize = 10,
|
||||||
|
hardCap = false,
|
||||||
|
healthCheckQuery = "SELECT 'true' AS alive"): DbConnPool =
|
||||||
|
|
||||||
|
initDbConnPool(DbConnPoolConfig(
|
||||||
|
connect: connect,
|
||||||
|
poolSize: poolSize,
|
||||||
|
hardCap: hardCap,
|
||||||
|
healthCheckQuery: healthCheckQuery))
|
||||||
|
|
||||||
|
template inTransaction*(db: DbConnPool, body: untyped) =
|
||||||
|
pool.withConn(db):
|
||||||
|
conn.exec(sql"BEGIN TRANSACTION")
|
||||||
|
try:
|
||||||
|
body
|
||||||
|
conn.exec(sql"COMMIT")
|
||||||
|
except:
|
||||||
|
conn.exec(sql"ROLLBACK")
|
||||||
|
raise getCurrentException()
|
||||||
|
97
src/fiber_orm/pool.nim
Normal file
97
src/fiber_orm/pool.nim
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
import std/db_postgres, std/sequtils, std/strutils, std/sugar
|
||||||
|
|
||||||
|
import namespaced_logging
|
||||||
|
|
||||||
|
|
||||||
|
type
|
||||||
|
DbConnPoolConfig* = object
|
||||||
|
connect*: () -> DbConn
|
||||||
|
poolSize*: int
|
||||||
|
hardCap*: bool
|
||||||
|
healthCheckQuery*: string
|
||||||
|
|
||||||
|
PooledDbConn = ref object
|
||||||
|
conn: DbConn
|
||||||
|
id: int
|
||||||
|
free: bool
|
||||||
|
|
||||||
|
DbConnPool* = ref object
|
||||||
|
conns: seq[PooledDbConn]
|
||||||
|
cfg: DbConnPoolConfig
|
||||||
|
lastId: int
|
||||||
|
|
||||||
|
var logNs {.threadvar.}: LoggingNamespace
|
||||||
|
|
||||||
|
template log(): untyped =
|
||||||
|
if logNs.isNil: logNs = initLoggingNamespace(name = "fiber_orm/pool", level = lvlNotice)
|
||||||
|
logNs
|
||||||
|
|
||||||
|
proc initDbConnPool*(cfg: DbConnPoolConfig): DbConnPool =
|
||||||
|
log().debug("Initializing new pool (size: " & $cfg.poolSize)
|
||||||
|
result = DbConnPool(
|
||||||
|
conns: @[],
|
||||||
|
cfg: cfg)
|
||||||
|
|
||||||
|
proc newConn(pool: DbConnPool): PooledDbConn =
|
||||||
|
log().debug("Creating a new connection to add to the pool.")
|
||||||
|
pool.lastId += 1
|
||||||
|
let conn = pool.cfg.connect()
|
||||||
|
result = PooledDbConn(
|
||||||
|
conn: conn,
|
||||||
|
id: pool.lastId,
|
||||||
|
free: true)
|
||||||
|
pool.conns.add(result)
|
||||||
|
|
||||||
|
proc maintain(pool: DbConnPool): void =
|
||||||
|
log().debug("Maintaining pool. $# connections." % [$pool.conns.len])
|
||||||
|
pool.conns.keepIf(proc (pc: PooledDbConn): bool =
|
||||||
|
if not pc.free: return true
|
||||||
|
|
||||||
|
try:
|
||||||
|
discard getRow(pc.conn, sql(pool.cfg.healthCheckQuery), [])
|
||||||
|
return true
|
||||||
|
except:
|
||||||
|
try: pc.conn.close() # try to close the connection
|
||||||
|
except: discard ""
|
||||||
|
return false
|
||||||
|
)
|
||||||
|
log().debug(
|
||||||
|
"Pruned dead connections. $# connections remaining." %
|
||||||
|
[$pool.conns.len])
|
||||||
|
|
||||||
|
let freeConns = pool.conns.filterIt(it.free)
|
||||||
|
if pool.conns.len > pool.cfg.poolSize and freeConns.len > 0:
|
||||||
|
let numToCull = min(freeConns.len, pool.conns.len - pool.cfg.poolSize)
|
||||||
|
let toCull = freeConns[0..numToCull]
|
||||||
|
pool.conns.keepIf((pc) => toCull.allIt(it.id != pc.id))
|
||||||
|
for culled in toCull:
|
||||||
|
try: culled.conn.close()
|
||||||
|
except: discard ""
|
||||||
|
log().debug(
|
||||||
|
"Trimming pool size. Culled $# free connections. $# connections remaining." %
|
||||||
|
[$toCull.len, $pool.conns.len])
|
||||||
|
|
||||||
|
proc take*(pool: DbConnPool): tuple[id: int, conn: DbConn] =
|
||||||
|
pool.maintain
|
||||||
|
let freeConns = pool.conns.filterIt(it.free)
|
||||||
|
|
||||||
|
log().debug(
|
||||||
|
"Providing a new connection ($# currently free)." % [$freeConns.len])
|
||||||
|
|
||||||
|
let reserved =
|
||||||
|
if freeConns.len > 0: freeConns[0]
|
||||||
|
else: pool.newConn()
|
||||||
|
|
||||||
|
reserved.free = false
|
||||||
|
log().debug("Reserve connection $#" % [$reserved.id])
|
||||||
|
return (id: reserved.id, conn: reserved.conn)
|
||||||
|
|
||||||
|
proc release*(pool: DbConnPool, connId: int): void =
|
||||||
|
log().debug("Reclaiming released connaction $#" % [$connId])
|
||||||
|
let foundConn = pool.conns.filterIt(it.id == connId)
|
||||||
|
if foundConn.len > 0: foundConn[0].free = true
|
||||||
|
|
||||||
|
template withConn*(pool: DbConnPool, stmt: untyped): untyped =
|
||||||
|
let (connId, conn {.inject.}) = take(pool)
|
||||||
|
stmt
|
||||||
|
release(pool, connId)
|
Loading…
x
Reference in New Issue
Block a user