components_api_ApiTask.bs
' bsc-disable-file print-locations — legacy print() sites; migration to m.log.* tracked by tech-debt.md#legacy-print-statements
import "pkg:/source/api/baseRequest.bs"
import "pkg:/source/constants/timeouts.bs"
import "pkg:/source/roku_modules/rr/Requests.brs"
import "pkg:/source/utils/misc.bs"
sub init()
m.top.functionName = "runApiLoop"
end sub
' Continuous-server event loop (Architecture C).
' Waits for request field writes, executes HTTP via roku-requests, writes response.
' Never exits — lives for the lifetime of the app alongside the global pool.
sub runApiLoop()
port = CreateObject("roMessagePort")
m.top.observeField("request", port)
' Signal that our observer is registered. The ApiQueueTask coordinator waits
' for all pool slots to be ready before dispatching — without this, the
' coordinator could write to our request field before we're listening,
' causing a permanently dead slot.
m.top.isReady = true
while true
msg = wait(0, port)
if type(msg) = "roSGNodeEvent" and msg.getField() = "request"
' Safe to read m.top.request directly: the coordinator only dispatches
' one request per slot at a time (m.inFlight prevents reuse until we
' write the response), so the field always reflects our pending request.
req = m.top.request
if isValid(req)
m.top.response = executeRequest(req)
end if
end if
end while
end sub
' Executes a single HTTP request via roku-requests.
' @param req - request AA: { requestId, method, url, headers?, body?, timeout? }
' @return response AA: { requestId, ok, statusCode, json, text }
function executeRequest(req as object) as object
method = req.method
if not isValid(method) or method = "" then method = "GET"
' Auth header first; merge in any caller-provided extras
headers = { Authorization: buildAuthHeader() }
if isValid(req.headers) and type(req.headers) = "roAssociativeArray"
headers.append(req.headers)
end if
' timeout in request AA is seconds; rr_Requests expects milliseconds
timeout = timeouts.HTTP_MS
if isValid(req.timeout) then timeout = req.timeout * 1000
args = {
headers: headers,
timeout: timeout,
useCache: false
}
if isValid(req.body) and req.body <> ""
args.data = req.body
' POST/PUT/PATCH with a body requires Content-Type so the server knows the format.
' Only set if not already provided by caller-supplied headers.
if not isValid(headers["Content-Type"])
headers["Content-Type"] = "application/json"
end if
end if
if not isValid(req.url) or req.url = ""
print "[ApiTask] request rejected: url is invalid or empty"
return {
requestId: req.requestId,
ok: false,
statusCode: 0,
json: invalid,
text: ""
}
end if
r = rr_Requests().request(method, req.url, args)
return {
requestId: req.requestId,
ok: r.ok,
statusCode: r.statusCode,
json: r.json,
text: r.text
}
end function