generated from treeform/nimtemplate
-
Notifications
You must be signed in to change notification settings - Fork 26
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
lower nim version dep, add nim version to github actions
- Loading branch information
Showing
5 changed files
with
228 additions
and
217 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,11 +1,11 @@ | ||
version = "1.5.2" | ||
version = "1.5.3" | ||
author = "Andre von Houck" | ||
description = "Puppy fetches HTML pages for Nim." | ||
license = "MIT" | ||
|
||
srcDir = "src" | ||
|
||
requires "nim >= 1.4.4" | ||
requires "nim >= 1.2.2" | ||
requires "urlly >= 0.2.0" | ||
requires "libcurl >= 1.0.0" | ||
requires "zippy >= 0.7.3" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,144 +1,145 @@ | ||
import puppy, std/locks, std/deques | ||
|
||
when not defined(gcArc): | ||
{.error: "Please use --gc:arc when using Puppy request pools.".} | ||
|
||
type | ||
RequestPoolObj = object | ||
lock: Lock | ||
cond: Cond | ||
threads: seq[Thread[ptr RequestPoolObj]] | ||
queue: Deque[ResponseHandle] | ||
closed: bool | ||
requestsCompleted: bool | ||
|
||
RequestPool* = ref RequestPoolObj | ||
|
||
ResponseHandleInternal = object | ||
lock: Lock | ||
request: Request | ||
response: Response | ||
error: ref PuppyError | ||
refCount: int | ||
|
||
ResponseHandle* = object | ||
internal: ptr ResponseHandleInternal | ||
|
||
proc free(internal: ptr ResponseHandleInternal) = | ||
deinitLock(internal.lock) | ||
internal.request = nil | ||
internal.response = nil | ||
internal.error = nil | ||
dealloc(internal) | ||
|
||
proc `=destroy`*(handle: var ResponseHandle) = | ||
if handle.internal != nil: | ||
acquire(handle.internal.lock) | ||
if handle.internal.refCount == 0: | ||
release(handle.internal.lock) | ||
free handle.internal | ||
else: | ||
dec handle.internal.refCount | ||
release(handle.internal.lock) | ||
when (NimMajor, NimMinor, NimPatch) >= (1, 6, 0): | ||
import puppy, std/locks, std/deques | ||
|
||
when not defined(gcArc): | ||
{.error: "Please use --gc:arc when using Puppy request pools.".} | ||
|
||
type | ||
RequestPoolObj = object | ||
lock: Lock | ||
cond: Cond | ||
threads: seq[Thread[ptr RequestPoolObj]] | ||
queue: Deque[ResponseHandle] | ||
closed: bool | ||
requestsCompleted: bool | ||
|
||
RequestPool* = ref RequestPoolObj | ||
|
||
ResponseHandleInternal = object | ||
lock: Lock | ||
request: Request | ||
response: Response | ||
error: ref PuppyError | ||
refCount: int | ||
|
||
ResponseHandle* = object | ||
internal: ptr ResponseHandleInternal | ||
|
||
proc free(internal: ptr ResponseHandleInternal) = | ||
deinitLock(internal.lock) | ||
internal.request = nil | ||
internal.response = nil | ||
internal.error = nil | ||
dealloc(internal) | ||
|
||
proc `=destroy`*(handle: var ResponseHandle) = | ||
if handle.internal != nil: | ||
acquire(handle.internal.lock) | ||
if handle.internal.refCount == 0: | ||
release(handle.internal.lock) | ||
free handle.internal | ||
else: | ||
dec handle.internal.refCount | ||
release(handle.internal.lock) | ||
|
||
proc `=`*(dst: var ResponseHandle, src: ResponseHandle) = | ||
if src.internal != nil: | ||
acquire(src.internal.lock) | ||
inc src.internal.refCount | ||
release(src.internal.lock) | ||
|
||
if dst.internal != nil: | ||
`=destroy`(dst) | ||
|
||
dst.internal = src.internal | ||
|
||
proc `=destroy`*(pool: var RequestPoolObj) = | ||
acquire(pool.lock) | ||
pool.closed = true | ||
release(pool.lock) | ||
|
||
proc `=`*(dst: var ResponseHandle, src: ResponseHandle) = | ||
if src.internal != nil: | ||
acquire(src.internal.lock) | ||
inc src.internal.refCount | ||
release(src.internal.lock) | ||
broadcast(pool.cond) | ||
|
||
if dst.internal != nil: | ||
`=destroy`(dst) | ||
joinThreads(pool.threads) | ||
|
||
dst.internal = src.internal | ||
deinitLock(pool.lock) | ||
deinitCond(pool.cond) | ||
`=destroy`(pool.threads) | ||
`=destroy`(pool.queue) | ||
|
||
proc `=destroy`*(pool: var RequestPoolObj) = | ||
acquire(pool.lock) | ||
pool.closed = true | ||
release(pool.lock) | ||
proc ready*(handle: ResponseHandle): bool = | ||
acquire(handle.internal.lock) | ||
result = handle.internal.response != nil | ||
release(handle.internal.lock) | ||
|
||
broadcast(pool.cond) | ||
proc response*(handle: ResponseHandle): Response = | ||
var error: ref PuppyError | ||
|
||
joinThreads(pool.threads) | ||
acquire(handle.internal.lock) | ||
result = handle.internal.response | ||
error = handle.internal.error | ||
release(handle.internal.lock) | ||
|
||
deinitLock(pool.lock) | ||
deinitCond(pool.cond) | ||
`=destroy`(pool.threads) | ||
`=destroy`(pool.queue) | ||
if error != nil: | ||
raise error | ||
|
||
proc ready*(handle: ResponseHandle): bool = | ||
acquire(handle.internal.lock) | ||
result = handle.internal.response != nil | ||
release(handle.internal.lock) | ||
proc workerProc(pool: ptr RequestPoolObj) {.raises: [].} = | ||
while true: | ||
acquire(pool.lock) | ||
|
||
proc response*(handle: ResponseHandle): Response = | ||
var error: ref PuppyError | ||
while pool.queue.len == 0 and not pool.closed: | ||
wait(pool.cond, pool.lock) | ||
|
||
acquire(handle.internal.lock) | ||
result = handle.internal.response | ||
error = handle.internal.error | ||
release(handle.internal.lock) | ||
if pool.closed: | ||
release(pool.lock) | ||
break | ||
|
||
if error != nil: | ||
raise error | ||
let handle = pool.queue.popFirst() | ||
|
||
proc workerProc(pool: ptr RequestPoolObj) {.raises: [].} = | ||
while true: | ||
acquire(pool.lock) | ||
release(pool.lock) | ||
|
||
while pool.queue.len == 0 and not pool.closed: | ||
wait(pool.cond, pool.lock) | ||
var | ||
response: Response | ||
error: ref PuppyError | ||
try: | ||
response = fetch(handle.internal.request) | ||
except PuppyError as e: | ||
error = e | ||
|
||
acquire(handle.internal.lock) | ||
handle.internal.response = response | ||
handle.internal.error = error | ||
release(handle.internal.lock) | ||
|
||
if pool.closed: | ||
acquire(pool.lock) | ||
pool.requestsCompleted = true | ||
release(pool.lock) | ||
break | ||
|
||
let handle = pool.queue.popFirst() | ||
proc newRequestPool*(maxInFlight: int): RequestPool = | ||
result = RequestPool() | ||
initLock(result.lock) | ||
initCond(result.cond) | ||
result.threads.setLen(maxInFlight) | ||
for thread in result.threads.mitems: | ||
createThread(thread, workerProc, cast[ptr RequestPoolObj](result)) | ||
|
||
proc fetch*(pool: RequestPool, request: Request): ResponseHandle = | ||
result.internal = cast[ptr ResponseHandleInternal]( | ||
alloc0(sizeof(ResponseHandleInternal)) | ||
) | ||
initLock(result.internal.lock) | ||
result.internal.request = request | ||
|
||
acquire(pool.lock) | ||
pool.queue.addLast(result) | ||
release(pool.lock) | ||
|
||
var | ||
response: Response | ||
error: ref PuppyError | ||
try: | ||
response = fetch(handle.internal.request) | ||
except PuppyError as e: | ||
error = e | ||
|
||
acquire(handle.internal.lock) | ||
handle.internal.response = response | ||
handle.internal.error = error | ||
release(handle.internal.lock) | ||
signal(pool.cond) | ||
|
||
proc requestsCompleted*(pool: RequestPool): bool = | ||
## Returns whether or not at least one request has completed since the last | ||
## time this proc was called. | ||
acquire(pool.lock) | ||
pool.requestsCompleted = true | ||
result = pool.requestsCompleted | ||
pool.requestsCompleted = false | ||
release(pool.lock) | ||
|
||
proc newRequestPool*(maxInFlight: int): RequestPool = | ||
result = RequestPool() | ||
initLock(result.lock) | ||
initCond(result.cond) | ||
result.threads.setLen(maxInFlight) | ||
for thread in result.threads.mitems: | ||
createThread(thread, workerProc, cast[ptr RequestPoolObj](result)) | ||
|
||
proc fetch*(pool: RequestPool, request: Request): ResponseHandle = | ||
result.internal = cast[ptr ResponseHandleInternal]( | ||
alloc0(sizeof(ResponseHandleInternal)) | ||
) | ||
initLock(result.internal.lock) | ||
result.internal.request = request | ||
|
||
acquire(pool.lock) | ||
pool.queue.addLast(result) | ||
release(pool.lock) | ||
|
||
signal(pool.cond) | ||
|
||
proc requestsCompleted*(pool: RequestPool): bool = | ||
## Returns whether or not at least one request has completed since the last | ||
## time this proc was called. | ||
acquire(pool.lock) | ||
result = pool.requestsCompleted | ||
pool.requestsCompleted = false | ||
release(pool.lock) |
Oops, something went wrong.