mirror of
https://github.com/20kdc/OC-KittenOS.git
synced 2025-01-29 02:56:02 +11:00
156 lines
3.7 KiB
Lua
156 lines
3.7 KiB
Lua
-- This is released into the public domain.
|
|
-- No warranty is provided, implied or otherwise.
|
|
|
|
-- svc-app-claw-worker: Who stays stays. Who goes goes.
|
|
|
|
local callerPkg, _, destProx, packageId, downloadSrc, checked, primaryINet = ...
|
|
if callerPkg ~= "app-claw" then error("Internal process for app-claw's bidding.") end
|
|
-- This 'mutex' remains active as long as the process does.
|
|
neo.requireAccess("r.svc.app-claw-worker", "CLAW mutex")
|
|
|
|
local function wrapLines(ocb)
|
|
local buf = ""
|
|
return function (data)
|
|
if not data then
|
|
ocb(buf)
|
|
else
|
|
buf = buf .. data
|
|
buf = buf:gsub("[^\n]*\n", function (t)
|
|
ocb(t:sub(1, -2))
|
|
return ""
|
|
end)
|
|
end
|
|
end
|
|
end
|
|
|
|
local function download(url, cb, src)
|
|
if type(src) == "string" then
|
|
assert(primaryINet, "no internet")
|
|
local req, err = primaryINet.request(src .. url)
|
|
assert(req, err)
|
|
-- OpenComputers#535
|
|
req.finishConnect()
|
|
while true do
|
|
local n, n2 = req.read(neo.readBufSize)
|
|
cb(n)
|
|
if not n then
|
|
req.close()
|
|
if n2 then
|
|
error(n2)
|
|
else
|
|
cb(nil)
|
|
break
|
|
end
|
|
else
|
|
if n == "" then
|
|
neo.scheduleTimer(os.uptime() + 0.05)
|
|
while true do
|
|
local res = coroutine.yield()
|
|
if res == "k.timer" then break end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
else
|
|
local h, e = src.open(url, "rb")
|
|
assert(h, e)
|
|
repeat
|
|
local c = src.read(h, neo.readBufSize)
|
|
cb(c)
|
|
until not c
|
|
src.close(h)
|
|
end
|
|
end
|
|
|
|
local opInstall, opRemove
|
|
|
|
function opInstall(packageId, checked)
|
|
local gback = {} -- the ultimate strategy
|
|
local gdir = {}
|
|
local preinstall = {}
|
|
download("data/app-claw/" .. packageId .. ".c2x", wrapLines(function (l)
|
|
if l:sub(1, 1) == "?" and checked then
|
|
preinstall[l:sub(2)] = true
|
|
elseif l:sub(1, 1) == "+" then
|
|
table.insert(gback, l:sub(2))
|
|
elseif l:sub(1, 1) == "/" then
|
|
table.insert(gdir, l:sub(2))
|
|
end
|
|
end), downloadSrc)
|
|
for _, v in ipairs(gdir) do
|
|
destProx.makeDirectory(v)
|
|
assert(destProx.isDirectory(v), "unable to create dir " .. v)
|
|
end
|
|
gdir = nil
|
|
for k, _ in pairs(preinstall) do
|
|
if not destProx.exists("data/app-claw/" .. k .. ".c2x") then
|
|
opInstall(k, true)
|
|
end
|
|
end
|
|
preinstall = nil
|
|
for _, v in ipairs(gback) do
|
|
local f = destProx.open(v .. ".C2T", "wb")
|
|
assert(f, "unable to create download file")
|
|
local ok, e = pcall(download, v, function (b)
|
|
assert(destProx.write(f, b or ""), "unable to save data")
|
|
end, downloadSrc)
|
|
destProx.close(f)
|
|
if not ok then error(e) end
|
|
end
|
|
-- CRITICAL SECTION --
|
|
if destProx.exists("data/app-claw/" .. packageId .. ".c2x") then
|
|
opRemove(packageId, false)
|
|
end
|
|
for _, v in ipairs(gback) do
|
|
if destProx.exists(v) then
|
|
for _, v in ipairs(gback) do
|
|
destProx.remove(v .. ".C2T")
|
|
end
|
|
error("file conflict: " .. v)
|
|
end
|
|
end
|
|
for _, v in ipairs(gback) do
|
|
destProx.rename(v .. ".C2T", v)
|
|
end
|
|
end
|
|
|
|
function opRemove(packageId, checked)
|
|
if checked then
|
|
local dependents = {}
|
|
for _, pidf in ipairs(destProx.list("data/app-claw/")) do
|
|
if pidf:sub(-4) == ".c2x" then
|
|
local pid = pidf:sub(1, -5)
|
|
download("data/app-claw/" .. pidf, wrapLines(function (l)
|
|
if l == "?" .. packageId then
|
|
table.insert(dependents, pid)
|
|
end
|
|
end), destProx)
|
|
end
|
|
end
|
|
assert(not dependents[1], "Cannot remove " .. packageId .. ", required by:\n" .. table.concat(dependents, ", "))
|
|
end
|
|
local rmSchedule = {}
|
|
download("data/app-claw/" .. packageId .. ".c2x", wrapLines(function (l)
|
|
if l:sub(1, 1) == "+" then
|
|
table.insert(rmSchedule, l:sub(2))
|
|
end
|
|
end), destProx)
|
|
for _, v in ipairs(rmSchedule) do
|
|
destProx.remove(v)
|
|
end
|
|
end
|
|
|
|
local ok, e
|
|
if downloadSrc then
|
|
ok, e = pcall(opInstall, packageId, checked)
|
|
else
|
|
ok, e = pcall(opRemove, packageId, checked)
|
|
end
|
|
destProx = nil
|
|
downloadSrc = nil
|
|
primaryINet = nil
|
|
neo.executeAsync("app-claw", packageId)
|
|
if not ok then
|
|
error(e)
|
|
end
|