345 lines
14 KiB
Nim
345 lines
14 KiB
Nim
# The contents of this file are subject to the Common Public Attribution License
|
|
# Version 1.0 (the “License”); you may not use this file except in compliance
|
|
# with the License. You may obtain a copy of the License at
|
|
# https://myou.dev/licenses/LICENSE-CPAL. The License is based on the Mozilla
|
|
# Public License Version 1.1 but Sections 14 and 15 have been added to cover use
|
|
# of software over a computer network and provide for limited attribution for
|
|
# the Original Developer. In addition, Exhibit A has been modified to be
|
|
# consistent with Exhibit B.
|
|
#
|
|
# Software distributed under the License is distributed on an “AS IS” basis,
|
|
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for
|
|
# the specific language governing rights and limitations under the License.
|
|
#
|
|
# The Original Code is Myou Engine.
|
|
#
|
|
# the Original Developer is the Initial Developer.
|
|
#
|
|
# The Initial Developer of the Original Code is the Myou Engine developers.
|
|
# All portions of the code written by the Myou Engine developers are Copyright
|
|
# (c) 2024. All Rights Reserved.
|
|
#
|
|
# Alternatively, the contents of this file may be used under the terms of the
|
|
# GNU Affero General Public License version 3 (the [AGPL-3] License), in which
|
|
# case the provisions of [AGPL-3] License are applicable instead of those above.
|
|
#
|
|
# If you wish to allow use of your version of this file only under the terms of
|
|
# the [AGPL-3] License and not to allow others to use your version of this file
|
|
# under the CPAL, indicate your decision by deleting the provisions above and
|
|
# replace them with the notice and other provisions required by the [AGPL-3]
|
|
# License. If you do not delete the provisions above, a recipient may use your
|
|
# version of this file under either the CPAL or the [AGPL-3] License.
|
|
|
|
|
|
## A LoadableResource is similar to a JS promise or a Future, except that you can
|
|
## store it to load it any number of times, and instead of providing an object,
|
|
## you get a pointer, a length, and a function to tell it you're done with it.
|
|
## You cannot have more than one callback, but you can obtain the results of many
|
|
## with `loadAll` or get a callback when many are done with `allDone`
|
|
##
|
|
## TODO IMPORTANT: probably not thread safe, call all functions in the same thread
|
|
## if you use loadAll and allDone
|
|
##
|
|
## TODO: investigate the use of signals instead of callbacks
|
|
|
|
{.hint[ConvFromXtoItselfNotNeeded]:off.}
|
|
|
|
import std/tables
|
|
import arr_ref # for SliceMem
|
|
|
|
type LoadableResourceStatus* = enum
|
|
NotStarted
|
|
Started
|
|
Finished
|
|
Error
|
|
|
|
# type Result = tuple[ok: bool, err: string, data: SliceMem[byte]]
|
|
|
|
type LoadableResource* = ref object of RootObj
|
|
status: LoadableResourceStatus
|
|
start_func: proc(self: LoadableResource) {.closure.}
|
|
onload_func: proc(ok: bool, err: string, data: SliceMem[byte]) {.closure.}
|
|
cancel_func: proc() {.closure.}
|
|
str*: proc(): string
|
|
use_threads: bool
|
|
# result: ref Result # only to be used with loadAll
|
|
|
|
proc newLoadableResource*[T: LoadableResource](
|
|
start: proc(self: LoadableResource),
|
|
str: proc(): string = nil,
|
|
use_threads = false,
|
|
): T =
|
|
new(result)
|
|
result.start_func = start
|
|
result.str = str
|
|
if str == nil:
|
|
result.str = proc(): string = ""
|
|
result.use_threads = use_threads
|
|
|
|
when compileOption("threads"):
|
|
# main -> thread channels
|
|
var to_start: Channel[LoadableResource]
|
|
# main <- thread channels
|
|
var to_return: Channel[(LoadableResource, bool, string, SliceMem[byte])]
|
|
|
|
proc start*[T: LoadableResource](self: T) =
|
|
self.status = Started
|
|
if self.use_threads:
|
|
when compileOption("threads"): to_start.send self
|
|
else:
|
|
self.start_func(self)
|
|
|
|
proc `onload=`*[T: LoadableResource](self: T, onload_func: proc(ok: bool, err: string, data: SliceMem[byte])) =
|
|
self.onload_func = onload_func
|
|
|
|
proc onload*[T: LoadableResource](self: T, ok: bool, err: string, data = SliceMem[byte]()) =
|
|
if self.status == Started:
|
|
self.status = if ok: Finished else: Error
|
|
# if self.result != nil:
|
|
# self.result[] = (ok, err, data)
|
|
if self.onload_func != nil:
|
|
if self.use_threads:
|
|
when compileOption("threads"):
|
|
to_return.send((self.LoadableResource, ok, err, data))
|
|
else:
|
|
self.onload_func(ok, err, data)
|
|
|
|
proc cancel*[T: LoadableResource](self: T) =
|
|
if self.status != Started:
|
|
return
|
|
if self.cancel_func != nil:
|
|
self.cancel_func()
|
|
self.status = NotStarted
|
|
|
|
# proc allDone*(list: seq[LoadableResource], done: proc()) =
|
|
|
|
|
|
# proc loadAll*(list: seq[LoadableResource], onload: proc(all_ok: bool, result: seq[Result])) =
|
|
# for i,res in list:
|
|
# if not defined(release):
|
|
# assert res.status == NotStarted, "All resources in the list must be not started"
|
|
# assert res.onload_func == nil, "All resources in the list must not have onload"
|
|
# new(res.result)
|
|
# res.start()
|
|
# list.allDone, proc() =
|
|
# var results = newSeqOfCap[Result](list.len)
|
|
# for res in list:
|
|
# results.add res.result[]
|
|
# res.result = nil
|
|
|
|
when compileOption("threads"):
|
|
var worker: Thread[void]
|
|
proc workerThreadProc() {.thread.} =
|
|
while true:
|
|
let res = to_start.recv()
|
|
if res == nil:
|
|
break
|
|
cast[proc(self: LoadableResource) {.gcsafe.}](res.start_func)(res)
|
|
|
|
worker.createThread(workerThreadProc)
|
|
to_start.open()
|
|
to_return.open()
|
|
|
|
proc updateLoadableWorkerThreads*() =
|
|
while true:
|
|
let tried = to_return.tryRecv()
|
|
if not tried.dataAvailable:
|
|
break
|
|
let (res, ok, err, data) = tried.msg
|
|
res.onload_func(ok, err, data)
|
|
|
|
proc terminateLoadableWorkerThreads*() =
|
|
# TODO: test this
|
|
to_start.send(nil.LoadableResource)
|
|
worker.joinThread()
|
|
|
|
type Fetch* = ref object of LoadableResource
|
|
custom: pointer
|
|
|
|
when not defined(onlyLocalFiles):
|
|
when defined(emscripten):
|
|
type
|
|
FetchAttribute = enum
|
|
LoadToMemory = 1
|
|
StreamData = 2
|
|
PersistFile = 4
|
|
Append = 8
|
|
Replace = 16
|
|
NoDownload = 32
|
|
Synchronous = 64
|
|
Waitable = 128
|
|
emscripten_fetch_attr_t {.importc,nodecl,header:"emscripten/fetch.h".} = object
|
|
requestMethod: array[32, char]
|
|
userData: pointer
|
|
onsuccess: proc(fetch: ptr emscripten_fetch_t) {.cdecl.}
|
|
onerror: proc(fetch: ptr emscripten_fetch_t) {.cdecl.}
|
|
onprogress: proc(fetch: ptr emscripten_fetch_t) {.cdecl.}
|
|
onreadystatechange: proc(fetch: ptr emscripten_fetch_t) {.cdecl.}
|
|
attributes: FetchAttribute # uint32
|
|
timeoutMSecs: uint32
|
|
withCredentials: bool # actually int32
|
|
destinationPath: cstring
|
|
userName: cstring
|
|
password: cstring
|
|
requestHeaders: ptr UncheckedArray[cstring]
|
|
overriddenMimeType: cstring
|
|
requestData: cstring
|
|
requestDataSize: uint
|
|
emscripten_fetch_t {.importc,nodecl.} = object
|
|
id: uint32
|
|
userData: pointer
|
|
url: cstring
|
|
data: pointer
|
|
numBytes: uint64
|
|
dataOffset: uint64
|
|
totalBytes: uint64
|
|
readyState: uint16
|
|
status: uint16
|
|
statusText: array[64, char]
|
|
EmscriptenResult = enum
|
|
TimedOut = -8
|
|
NoData = -7
|
|
Failed = -6
|
|
InvalidParam = -5
|
|
UnknownTarget = -4
|
|
InvalidTarget = -3
|
|
FailedNotDeferred = -2
|
|
NotSupported = -1
|
|
Success = 0
|
|
Deferred = 1
|
|
const EMSCRIPTEN_RESULT_STRINGS = [
|
|
"TimedOut","NoData","Failed","InvalidParam",
|
|
"UnknownTarget","InvalidTarget","FailedNotDeferred",
|
|
"NotSupported","Success","Deferred"]
|
|
proc emscripten_fetch_attr_init(fetch_attr: var emscripten_fetch_attr_t) {.importc,nodecl.}
|
|
proc emscripten_fetch(fetch_attr: var emscripten_fetch_attr_t, url: cstring): ptr emscripten_fetch_t {.importc,nodecl.}
|
|
proc emscripten_fetch_wait(fetch: ptr emscripten_fetch_t, timeoutMSecs: float): EmscriptenResult {.importc,nodecl.}
|
|
proc emscripten_fetch_close(fetch: ptr emscripten_fetch_t): EmscriptenResult {.importc,nodecl.}
|
|
proc emscripten_fetch_get_response_headers_length(fetch: ptr emscripten_fetch_t): uint {.importc,nodecl.}
|
|
proc emscripten_fetch_get_response_headers(fetch: ptr emscripten_fetch_t, dst: cstring , dstSizeBytes: uint): uint {.importc,nodecl.}
|
|
proc emscripten_fetch_unpack_response_headers(headersString: cstring): ptr UncheckedArray[cstring] {.importc,nodecl.}
|
|
proc emscripten_fetch_free_unpacked_response_headers(unpackedHeaders: ptr UncheckedArray[cstring]) {.importc,nodecl.}
|
|
|
|
else:
|
|
import std/httpclient
|
|
|
|
import std/memfiles
|
|
import std/strutils
|
|
import std/strformat
|
|
|
|
func escapeUTF8*(s: string): string =
|
|
result = newStringOfCap(s.len + 5)
|
|
for i,c in s:
|
|
if c > ' ' and c <= '~':
|
|
result &= c
|
|
else:
|
|
result &= '%' & c.byte.toHex
|
|
|
|
# TODO: automatically disable threads when not in main thread
|
|
|
|
type ProtocolHandler* = proc(uri: string): proc(self: LoadableResource)
|
|
var custom_protocol_handlers: Table[string, ProtocolHandler]
|
|
var log_uri_handler: proc(uri: string)
|
|
|
|
proc registerCustomProtocol*(prefix: string, handler: ProtocolHandler) =
|
|
## Registers a handler for a custom protocol. The function will be run for
|
|
## each uri that is requested, and will return a start_func, which in turn
|
|
## will call self.onload()
|
|
custom_protocol_handlers[prefix] = handler
|
|
|
|
proc registerLogUriHandler*(handler: proc(uri: string)) =
|
|
log_uri_handler = handler
|
|
|
|
proc loadUri*(
|
|
uri: string,
|
|
onload_func: proc(ok: bool, err: string, data: SliceMem[byte]) = nil,
|
|
range = (-1,-1),
|
|
auto_start = true,
|
|
use_threads = true,
|
|
): Fetch {.discardable.} =
|
|
|
|
if log_uri_handler != nil:
|
|
log_uri_handler(uri)
|
|
|
|
for k,v in custom_protocol_handlers:
|
|
if uri.startswith k:
|
|
proc str(): string = uri
|
|
var self = newLoadableResource[Fetch](v(uri), str, false)
|
|
self.onload_func = onload_func
|
|
if auto_start:
|
|
start(self)
|
|
return self
|
|
|
|
var start_func: proc(self: LoadableResource)
|
|
var self: Fetch
|
|
var uri = uri
|
|
when compileOption("threads"):
|
|
var use_threads = use_threads
|
|
else:
|
|
var use_threads = false
|
|
when not defined(onlyLocalFiles):
|
|
when defined(emscripten):
|
|
const is_remote = true
|
|
else:
|
|
let is_remote = uri.startswith("http://") or uri.startswith("https://")
|
|
|
|
if is_remote:
|
|
uri = uri.escapeUTF8
|
|
when defined(emscripten):
|
|
use_threads = false # API is already threaded
|
|
start_func = proc(self: LoadableResource) =
|
|
var attr: emscripten_fetch_attr_t
|
|
emscripten_fetch_attr_init(attr)
|
|
attr.userData = cast[pointer](self)
|
|
attr.requestMethod.addr.copyMem("GET".cstring, 4)
|
|
attr.attributes = LoadToMemory
|
|
attr.onsuccess = proc(fetch: ptr emscripten_fetch_t) {.cdecl.} =
|
|
var self = cast[Fetch](fetch.userData)
|
|
self.custom = fetch.pointer
|
|
self.onload(true, "", newSliceMem(fetch.data, fetch.numBytes.int, proc() =
|
|
emscripten_fetch_close(fetch)
|
|
))
|
|
attr.onerror = proc(fetch: ptr emscripten_fetch_t) {.cdecl.} =
|
|
var self = cast[Fetch](fetch.userData)
|
|
let err_msg = $fetch.statusText.addr.cstring
|
|
let err = emscripten_fetch_close(fetch)
|
|
# self.onload(false, &"Error: {err_msg} ({EMSCRIPTEN_RESULT_STRINGS[err.int + 8]})", nil, 0)
|
|
self.onload(false, &"Error fetching {fetch.url}: {err_msg}")
|
|
discard emscripten_fetch(attr, uri.cstring)
|
|
else:
|
|
var client = newHttpClient()
|
|
var response: string
|
|
start_func = proc(self: LoadableResource) =
|
|
let self = cast[Fetch](self)
|
|
var ok = false
|
|
try:
|
|
response = client.getContent(uri)
|
|
ok = true
|
|
except:
|
|
self.onload(false, &"Error fetching {uri}: {getCurrentExceptionMsg()}")
|
|
if ok:
|
|
self.onload(true, "", newSliceMem(response[0].addr.pointer, response.len, proc() =
|
|
discard response
|
|
))
|
|
|
|
if not is_remote:
|
|
use_threads = false # TODO: detect networked file system?
|
|
# TODO: also test if there's better perf in local
|
|
start_func = proc(self: LoadableResource) =
|
|
try:
|
|
var memfile = memfiles.open(uri, mode=fmRead)
|
|
self.onload(true, "", newSliceMem(memfile.mem, memfile.size, proc() =
|
|
try:
|
|
memfile.close()
|
|
# this should never happen but destructors require it
|
|
except OSError: discard
|
|
))
|
|
except OSError:
|
|
self.onload(false, "Could not open file: " & uri)
|
|
proc str(): string = uri
|
|
self = newLoadableResource[Fetch](start_func, str, use_threads=use_threads)
|
|
self.onload_func = onload_func
|
|
if auto_start:
|
|
start(self)
|
|
return self
|
|
|