Advertisement
thggdx

Untitled

Apr 5th, 2025 (edited)
276
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 28.75 KB | Source Code | 0 0
  1. --[[
  2. CC-MEK-SCADA Installer Utility
  3.  
  4. Copyright (c) 2023 - 2024 Mikayla Fischler
  5.  
  6. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
  7. associated documentation files (the "Software"), to deal in the Software without restriction,
  8. including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
  9. and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so.
  10.  
  11. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT
  12. LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
  13. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  14. WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
  15. SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  16. ]]--
  17.  
  18. local CCMSI_VERSION = "v1.21"
  19.  
  20. local install_dir = "/.install-cache"
  21. local manifest_path = "https://mikaylafischler.github.io/cc-mek-scada/manifests/"
  22. local repo_path = "https://gitproxy.click/https://raw.githubusercontent.com/MikaylaFischler/cc-mek-scada/"
  23.  
  24. ---@diagnostic disable-next-line: undefined-global
  25. local _is_pkt_env = pocket -- luacheck: ignore pocket
  26.  
  27. local function println(msg) print(tostring(msg)) end
  28.  
  29. -- stripped down & modified copy of log.dmesg
  30. local function print(msg)
  31.     msg = tostring(msg)
  32.  
  33.     local cur_x, cur_y = term.getCursorPos()
  34.     local out_w, out_h = term.getSize()
  35.  
  36.     -- jump to next line if needed
  37.     if cur_x == out_w then
  38.         cur_x = 1
  39.         if cur_y == out_h then
  40.             term.scroll(1)
  41.             term.setCursorPos(1, cur_y)
  42.         else
  43.             term.setCursorPos(1, cur_y + 1)
  44.         end
  45.     end
  46.  
  47.     -- wrap
  48.     local lines, remaining, s_start, s_end, ln = {}, true, 1, out_w + 1 - cur_x, 1
  49.     while remaining do
  50.         local line = string.sub(msg, s_start, s_end)
  51.  
  52.         if line == "" then
  53.             remaining = false
  54.         else
  55.             lines[ln] = line
  56.             s_start = s_end + 1
  57.             s_end = s_end + out_w
  58.             ln = ln + 1
  59.         end
  60.     end
  61.  
  62.     -- print
  63.     for i = 1, #lines do
  64.         cur_x, cur_y = term.getCursorPos()
  65.         if i > 1 and cur_x > 1 then
  66.             if cur_y == out_h then
  67.                 term.scroll(1)
  68.                 term.setCursorPos(1, cur_y)
  69.             else term.setCursorPos(1, cur_y + 1) end
  70.         end
  71.         term.write(lines[i])
  72.     end
  73. end
  74.  
  75. local opts = { ... }
  76. local mode, app, target
  77. local install_manifest = manifest_path.."main/install_manifest.json"
  78.  
  79. local function red() term.setTextColor(colors.red) end
  80. local function orange() term.setTextColor(colors.orange) end
  81. local function yellow() term.setTextColor(colors.yellow) end
  82. local function green() term.setTextColor(colors.green) end
  83. local function cyan() term.setTextColor(colors.cyan) end
  84. local function blue() term.setTextColor(colors.blue) end
  85. local function white() term.setTextColor(colors.white) end
  86. local function lgray() term.setTextColor(colors.lightGray) end
  87.  
  88. -- get command line option in list
  89. local function get_opt(opt, options)
  90.     for _, v in pairs(options) do if opt == v then return v end end
  91.     return nil
  92. end
  93.  
  94. -- wait for any key to be pressed
  95. ---@diagnostic disable-next-line: undefined-field
  96. local function any_key() os.pullEvent("key_up") end
  97.  
  98. -- ask the user yes or no
  99. local function ask_y_n(question, default)
  100.     print(question)
  101.     if default == true then print(" (Y/n)? ") else print(" (y/N)? ") end
  102.     local response = read();any_key()
  103.     if response == "" then return default
  104.     elseif response == "Y" or response == "y" then return true
  105.     elseif response == "N" or response == "n" then return false
  106.     else return nil end
  107. end
  108.  
  109. -- print out a white + blue text message
  110. local function pkg_message(message, package) white();print(message.." ");blue();println(package);white() end
  111.  
  112. -- indicate actions to be taken based on package differences for installs/updates
  113. local function show_pkg_change(name, v)
  114.     if v.v_local ~= nil then
  115.         if v.v_local ~= v.v_remote then
  116.             print("["..name.."] updating ");blue();print(v.v_local);white();print(" \xbb ");blue();println(v.v_remote);white()
  117.         elseif mode == "install" then
  118.             pkg_message("["..name.."] reinstalling", v.v_local)
  119.         end
  120.     else pkg_message("["..name.."] new install of", v.v_remote) end
  121.     return v.v_local ~= v.v_remote
  122. end
  123.  
  124. -- read the local manifest file
  125. local function read_local_manifest()
  126.     local local_ok = false
  127.     local local_manifest = {}
  128.     local imfile = fs.open("install_manifest.json", "r")
  129.     if imfile ~= nil then
  130.         local_ok, local_manifest = pcall(function () return textutils.unserializeJSON(imfile.readAll()) end)
  131.         imfile.close()
  132.     end
  133.     return local_ok, local_manifest
  134. end
  135.  
  136. -- get the manifest from GitHub
  137. local function get_remote_manifest()
  138.     local response, error = http.get(install_manifest)
  139.     if response == nil then
  140.         orange();println("Failed to get installation manifest from GitHub, cannot update or install.")
  141.         red();println("HTTP error: "..error);white()
  142.         return false, {}
  143.     end
  144.  
  145.     local ok, manifest = pcall(function () return textutils.unserializeJSON(response.readAll()) end)
  146.     if not ok then red();println("error parsing remote installation manifest");white() end
  147.  
  148.     return ok, manifest
  149. end
  150.  
  151. -- record the local installation manifest
  152. local function write_install_manifest(manifest, deps)
  153.     local versions = {}
  154.     for key, value in pairs(manifest.versions) do
  155.         local is_dep = false
  156.         for _, dep in pairs(deps) do
  157.             if (key == "bootloader" and dep == "system") or key == dep then
  158.                 is_dep = true;break
  159.             end
  160.         end
  161.         if key == app or key == "comms" or is_dep then versions[key] = value end
  162.     end
  163.  
  164.     manifest.versions = versions
  165.  
  166.     local imfile = fs.open("install_manifest.json", "w")
  167.     imfile.write(textutils.serializeJSON(manifest))
  168.     imfile.close()
  169. end
  170.  
  171. -- try at most 3 times to download a file from the repository and write into w_path base directory
  172. ---@return 0|1|2|3 success 0: ok, 1: download fail, 2: file open fail, 3: out of space
  173. local function http_get_file(file, w_path)
  174.     local dl, err
  175.     for i = 1, 3 do
  176.         dl, err = http.get(repo_path..file)
  177.         if dl then
  178.             if i > 1 then green();println("success!");lgray() end
  179.             local f = fs.open(w_path..file, "w")
  180.             if not f then return 2 end
  181.             local ok, msg = pcall(function() f.write(dl.readAll()) end)
  182.             f.close()
  183.             if not ok then
  184.                 if string.find(msg or "", "Out of space") ~= nil then
  185.                     red();println("[out of space]");lgray()
  186.                     return 3
  187.                 else return 2 end
  188.             end
  189.             break
  190.         else
  191.             red();println("HTTP Error: "..err)
  192.             if i < 3 then
  193.                 lgray();print("> retrying...")
  194.                 ---@diagnostic disable-next-line: undefined-field
  195.                 os.sleep(i/3.0)
  196.             else
  197.                 return 1
  198.             end
  199.         end
  200.     end
  201.     return 0
  202. end
  203.  
  204. -- recursively build a tree out of the file manifest
  205. local function gen_tree(manifest, log)
  206.     local function _tree_add(tree, split)
  207.         if #split > 1 then
  208.             local name = table.remove(split, 1)
  209.             if tree[name] == nil then tree[name] = {} end
  210.             table.insert(tree[name], _tree_add(tree[name], split))
  211.         else return split[1] end
  212.         return nil
  213.     end
  214.  
  215.     local list, tree = { log }, {}
  216.  
  217.     -- make a list of each and every file
  218.     for _, files in pairs(manifest.files) do for i = 1, #files do table.insert(list, files[i]) end end
  219.  
  220.     for i = 1, #list do
  221.         local split = {}
  222. ---@diagnostic disable-next-line: discard-returns
  223.         string.gsub(list[i], "([^/]+)", function(c) split[#split + 1] = c end)
  224.         if #split == 1 then table.insert(tree, list[i])
  225.         else table.insert(tree, _tree_add(tree, split)) end
  226.     end
  227.  
  228.     return tree
  229. end
  230.  
  231. local function _in_array(val, array)
  232.     for _, v in pairs(array) do if v == val then return true end end
  233.     return false
  234. end
  235.  
  236. local function _clean_dir(dir, tree)
  237.     if tree == nil then tree = {} end
  238.     local ls = fs.list(dir)
  239.     for _, val in pairs(ls) do
  240.         local path = dir.."/"..val
  241.         if fs.isDir(path) then
  242.             _clean_dir(path, tree[val])
  243.             if #fs.list(path) == 0 then fs.delete(path);println("deleted "..path) end
  244.         elseif (not _in_array(val, tree)) and (val ~= "config.lua" ) then ---@todo remove config.lua on full release
  245.             fs.delete(path)
  246.             println("deleted "..path)
  247.         end
  248.     end
  249. end
  250.  
  251. -- go through app/common directories to delete unused files
  252. local function clean(manifest)
  253.     local log = nil
  254.     if fs.exists(app..".settings") and settings.load(app..".settings") then
  255.         log = settings.get("LogPath")
  256.         if log:sub(1, 1) == "/" then log = log:sub(2) end
  257.     end
  258.  
  259.     local tree = gen_tree(manifest, log)
  260.  
  261.     table.insert(tree, "install_manifest.json")
  262.     table.insert(tree, "ccmsi.lua")
  263.  
  264.     local ls = fs.list("/")
  265.     for _, val in pairs(ls) do
  266.         if fs.isDriveRoot(val) then
  267.             yellow();println("skipped mount '"..val.."'")
  268.         elseif fs.isDir(val) then
  269.             if tree[val] ~= nil then lgray();_clean_dir("/"..val, tree[val])
  270.             else white(); if ask_y_n("delete the unused directory '"..val.."'") then lgray();_clean_dir("/"..val) end end
  271.             if #fs.list(val) == 0 then fs.delete(val);lgray();println("deleted empty directory '"..val.."'") end
  272.         elseif not _in_array(val, tree) and (string.find(val, ".settings") == nil) then
  273.             white();if ask_y_n("delete the unused file '"..val.."'") then fs.delete(val);lgray();println("deleted "..val) end
  274.         end
  275.     end
  276.  
  277.     white()
  278. end
  279.  
  280. -- get and validate command line options
  281.  
  282. if _is_pkt_env then println("- SCADA Installer "..CCMSI_VERSION.." -")
  283. else println("-- CC Mekanism SCADA Installer "..CCMSI_VERSION.." --") end
  284.  
  285. if #opts == 0 or opts[1] == "help" then
  286.     println("usage: ccmsi <mode> <app> <branch>")
  287.     if _is_pkt_env then
  288.     yellow();println("<mode>");lgray()
  289.     println(" check - check latest")
  290.     println(" install - fresh install")
  291.     println(" update - update app")
  292.     println(" uninstall - remove app")
  293.     yellow();println("<app>");lgray()
  294.     println(" reactor-plc")
  295.     println(" rtu")
  296.     println(" supervisor")
  297.     println(" coordinator")
  298.     println(" pocket")
  299.     println(" installer (update only)")
  300.     yellow();println("<branch>");lgray();
  301.     println(" main (default) | devel");white()
  302.     else
  303.     println("<mode>")
  304.     lgray()
  305.     println(" check       - check latest versions available")
  306.     yellow()
  307.     println("               ccmsi check <branch> for target")
  308.     lgray()
  309.     println(" install     - fresh install")
  310.     println(" update      - update files")
  311.     println(" uninstall   - delete files INCLUDING config/logs")
  312.     white();println("<app>");lgray()
  313.     println(" reactor-plc - reactor PLC firmware")
  314.     println(" rtu         - RTU firmware")
  315.     println(" supervisor  - supervisor server application")
  316.     println(" coordinator - coordinator application")
  317.     println(" pocket      - pocket application")
  318.     println(" installer   - ccmsi installer (update only)")
  319.     white();println("<branch>")
  320.     lgray();println(" main (default) | devel");white()
  321.     end
  322.     return
  323. else
  324.  
  325.     mode = get_opt(opts[1], { "check", "install", "update", "uninstall" })
  326.     if mode == nil then
  327.         red();println("Unrecognized mode.");white()
  328.         return
  329.     end
  330.  
  331.     local next_opt = 3
  332.     local apps = { "reactor-plc", "rtu", "supervisor", "coordinator", "pocket", "installer" }
  333.     app = get_opt(opts[2], apps)
  334.     if app == nil then
  335.         for _, a in pairs(apps) do
  336.             if fs.exists(a) and fs.isDir(a) then
  337.                 app = a
  338.                 next_opt = 2
  339.                 break
  340.             end
  341.         end
  342.     end
  343.  
  344.     if app == nil and mode ~= "check" then
  345.         red();println("Unrecognized application.");white()
  346.         return
  347.     elseif mode == "check" then
  348.         next_opt = 2
  349.     elseif app == "installer" and mode ~= "update" then
  350.         red();println("Installer app only supports 'update' option.");white()
  351.         return
  352.     end
  353.  
  354.     -- determine target
  355.     target = opts[next_opt]
  356.     if (target ~= "main") and (target ~= "devel") then
  357.         if (target and target ~= "") then yellow();println("Unknown target, defaulting to 'main'");white() end
  358.         target = "main"
  359.     end
  360.  
  361.     -- set paths
  362.     install_manifest = manifest_path..target.."/install_manifest.json"
  363.     repo_path = repo_path..target.."/"
  364. end
  365.  
  366. -- run selected mode
  367. if mode == "check" then
  368.     local ok, manifest = get_remote_manifest()
  369.     if not ok then return end
  370.  
  371.     local local_ok, local_manifest = read_local_manifest()
  372.     if not local_ok then
  373.         yellow();println("failed to load local installation information");white()
  374.         local_manifest = { versions = { installer = CCMSI_VERSION } }
  375.     else
  376.         local_manifest.versions.installer = CCMSI_VERSION
  377.     end
  378.  
  379.     -- list all versions
  380.     for key, value in pairs(manifest.versions) do
  381.         term.setTextColor(colors.purple)
  382.         local tag = string.format("%-14s", "["..key.."]")
  383.         if not _is_pkt_env then print(tag) end
  384.         if key == "installer" or (local_ok and (local_manifest.versions[key] ~= nil)) then
  385.             if _is_pkt_env then println(tag) end
  386.             blue();print(local_manifest.versions[key])
  387.             if value ~= local_manifest.versions[key] then
  388.                 white();print(" (")
  389.                 cyan();print(value);white();println(" available)")
  390.             else green();println(" (up to date)") end
  391.         elseif not _is_pkt_env then
  392.             lgray();print("not installed");white();print(" (latest ")
  393.             cyan();print(value);white();println(")")
  394.         end
  395.     end
  396.  
  397.     if manifest.versions.installer ~= local_manifest.versions.installer and not _is_pkt_env then
  398.         yellow();println("\nA different version of the installer is available, it is recommended to update (use 'ccmsi update installer').");white()
  399.     end
  400. elseif mode == "install" or mode == "update" then
  401.     local ok, r_manifest, l_manifest
  402.  
  403.     local update_installer = app == "installer"
  404.     ok, r_manifest = get_remote_manifest()
  405.     if not ok then return end
  406.  
  407.     local ver = {
  408.         app = { v_local = nil, v_remote = nil, changed = false },
  409.         boot = { v_local = nil, v_remote = nil, changed = false },
  410.         comms = { v_local = nil, v_remote = nil, changed = false },
  411.         common = { v_local = nil, v_remote = nil, changed = false },
  412.         graphics = { v_local = nil, v_remote = nil, changed = false },
  413.         lockbox = { v_local = nil, v_remote = nil, changed = false }
  414.     }
  415.  
  416.     -- try to find local versions
  417.     ok, l_manifest = read_local_manifest()
  418.     if mode == "update" and not update_installer then
  419.         if not ok then
  420.             red();println("Failed to load local installation information, cannot update.");white()
  421.             return
  422.         else
  423.             ver.boot.v_local = l_manifest.versions.bootloader
  424.             ver.app.v_local = l_manifest.versions[app]
  425.             ver.comms.v_local = l_manifest.versions.comms
  426.             ver.common.v_local = l_manifest.versions.common
  427.             ver.graphics.v_local = l_manifest.versions.graphics
  428.             ver.lockbox.v_local = l_manifest.versions.lockbox
  429.  
  430.             if l_manifest.versions[app] == nil then
  431.                 red();println("Another application is already installed, please uninstall it before installing a new application.");white()
  432.                 return
  433.             end
  434.         end
  435.     end
  436.  
  437.     if r_manifest.versions.installer ~= CCMSI_VERSION then
  438.         if not update_installer then yellow();println("A different version of the installer is available, it is recommended to update to it.");white() end
  439.         if update_installer or ask_y_n("Would you like to update now", true) then
  440.             lgray();println("GET ccmsi.lua")
  441.             local dl, err = http.get(repo_path.."ccmsi.lua")
  442.  
  443.             if dl == nil then
  444.                 red();println("HTTP Error: "..err)
  445.                 println("Installer download failed.");white()
  446.             else
  447.                 local handle = fs.open(debug.getinfo(1, "S").source:sub(2), "w") -- this file, regardless of name or location
  448.                 handle.write(dl.readAll())
  449.                 handle.close()
  450.                 green();println("Installer updated successfully.");white()
  451.             end
  452.  
  453.             return
  454.         end
  455.     elseif update_installer then
  456.         green();println("Installer already up-to-date.");white()
  457.         return
  458.     end
  459.  
  460.     ver.boot.v_remote = r_manifest.versions.bootloader
  461.     ver.app.v_remote = r_manifest.versions[app]
  462.     ver.comms.v_remote = r_manifest.versions.comms
  463.     ver.common.v_remote = r_manifest.versions.common
  464.     ver.graphics.v_remote = r_manifest.versions.graphics
  465.     ver.lockbox.v_remote = r_manifest.versions.lockbox
  466.  
  467.     green()
  468.     if mode == "install" then print("Installing ") else print("Updating ") end
  469.     println(app.." files...");white()
  470.  
  471.     ver.boot.changed = show_pkg_change("bootldr", ver.boot)
  472.     ver.common.changed = show_pkg_change("common", ver.common)
  473.     ver.comms.changed = show_pkg_change("comms", ver.comms)
  474.     if ver.comms.changed and ver.comms.v_local ~= nil then
  475.         print("[comms] ");yellow();println("other devices on the network will require an update");white()
  476.     end
  477.     ver.app.changed = show_pkg_change(app, ver.app)
  478.     ver.graphics.changed = show_pkg_change("graphics", ver.graphics)
  479.     ver.lockbox.changed = show_pkg_change("lockbox", ver.lockbox)
  480.  
  481.     -- start install/update
  482.  
  483.     local space_req = r_manifest.sizes.manifest
  484.     local space_avail = fs.getFreeSpace("/")
  485.  
  486.     local file_list = r_manifest.files
  487.     local size_list = r_manifest.sizes
  488.     local deps = r_manifest.depends[app]
  489.  
  490.     table.insert(deps, app)
  491.  
  492.     -- helper function to check if a dependency is unchanged
  493.     local function unchanged(dep)
  494.         if dep == "system" then return not ver.boot.changed
  495.         elseif dep == "graphics" then return not ver.graphics.changed
  496.         elseif dep == "lockbox" then return not ver.lockbox.changed
  497.         elseif dep == "common" then return not (ver.common.changed or ver.comms.changed)
  498.         elseif dep == app then return not ver.app.changed
  499.         else return true end
  500.     end
  501.  
  502.     local any_change = false
  503.  
  504.     for _, dep in pairs(deps) do
  505.         local size = size_list[dep]
  506.         space_req = space_req + size
  507.         any_change = any_change or not unchanged(dep)
  508.     end
  509.  
  510.     if mode == "update" and not any_change then
  511.         yellow();println("Nothing to do, everything is already up-to-date!");white()
  512.         return
  513.     end
  514.  
  515.     -- ask for confirmation
  516.     if not ask_y_n("Continue", false) then return end
  517.  
  518.     local single_file_mode = space_avail < space_req
  519.  
  520.     local success = true
  521.  
  522.     -- delete a file if the capitalization changes so that things work on Windows
  523.     ---@param path string
  524.     local function mitigate_case(path)
  525.         local dir, file = fs.getDir(path), fs.getName(path)
  526.         if not fs.isDir(dir) then return end
  527.         for _, p in ipairs(fs.list(dir)) do
  528.             if string.lower(p) == string.lower(file) then
  529.                 if p ~= file then fs.delete(path) end
  530.                 return
  531.             end
  532.         end
  533.     end
  534.  
  535.     ---@param dl_stat 1|2|3 download status
  536.     ---@param file string file name
  537.     ---@param attempt integer recursive attempt #
  538.     ---@param sf_install function installer function for recursion
  539.     local function handle_dl_fail(dl_stat, file, attempt, sf_install)
  540.         red()
  541.         if dl_stat == 1 then
  542.             println("failed to download "..file)
  543.         elseif dl_stat > 1 then
  544.             if dl_stat == 2 then println("filesystem error with "..file) else println("no space for "..file) end
  545.             if attempt == 1 then
  546.                 orange();println("re-attempting operation...");white()
  547.                 sf_install(2)
  548.             elseif attempt == 2 then
  549.                 yellow()
  550.                 if dl_stat == 2 then println("There was an error writing to a file.") else println("Insufficient space available.") end
  551.                 lgray()
  552.                 if dl_stat == 2 then
  553.                     println("This may be due to insufficent space available or file permission issues. The installer can now attempt to delete files not used by the SCADA system.")
  554.                 else
  555.                     println("The installer can now attempt to delete files not used by the SCADA system.")
  556.                 end
  557.                 white()
  558.                 if not ask_y_n("Continue", false) then
  559.                     success = false
  560.                     return
  561.                 end
  562.                 clean(r_manifest)
  563.                 sf_install(3)
  564.             elseif attempt == 3 then
  565.                 yellow()
  566.                 if dl_stat == 2 then println("There again was an error writing to a file.") else println("Insufficient space available.") end
  567.                 lgray()
  568.                 if dl_stat == 2 then
  569.                     println("This may be due to insufficent space available or file permission issues. Please delete any unused files you have on this computer then try again. Do not delete the "..app..".settings file unless you want to re-configure.")
  570.                 else
  571.                     println("Please delete any unused files you have on this computer then try again. Do not delete the "..app..".settings file unless you want to re-configure.")
  572.                 end
  573.                 white()
  574.                 success = false
  575.             end
  576.         end
  577.     end
  578.  
  579.     -- single file update routine: go through all files and replace one by one
  580.     ---@param attempt integer recursive attempt #
  581.     local function sf_install(attempt)
  582. ---@diagnostic disable-next-line: undefined-field
  583.         if attempt > 1 then os.sleep(2.0) end
  584.  
  585.         local abort_attempt = false
  586.         success = true
  587.  
  588.         for _, dep in pairs(deps) do
  589.             if mode == "update" and unchanged(dep) then
  590.                 pkg_message("skipping install of unchanged package", dep)
  591.             else
  592.                 pkg_message("installing package", dep)
  593.                 lgray()
  594.  
  595.                 -- beginning on the second try, delete the directory before starting
  596.                 if attempt >= 2 then
  597.                     if dep == "system" then
  598.                     elseif dep == "common" then
  599.                         if fs.exists("/scada-common") then
  600.                             fs.delete("/scada-common")
  601.                             println("deleted /scada-common")
  602.                         end
  603.                     else
  604.                         if fs.exists("/"..dep) then
  605.                             fs.delete("/"..dep)
  606.                             println("deleted /"..dep)
  607.                         end
  608.                     end
  609.                 end
  610.  
  611.                 local files = file_list[dep]
  612.                 for _, file in pairs(files) do
  613.                     println("GET "..file)
  614.                     mitigate_case(file)
  615.                     local dl_stat = http_get_file(file, "/")
  616.                     if dl_stat ~= 0 then
  617.                         abort_attempt = true
  618. ---@diagnostic disable-next-line: param-type-mismatch
  619.                         handle_dl_fail(dl_stat, file, attempt, sf_install)
  620.                         break
  621.                     end
  622.                 end
  623.             end
  624.             if abort_attempt or not success then break end
  625.         end
  626.     end
  627.  
  628.     -- handle update/install
  629.     if single_file_mode then sf_install(1)
  630.     else
  631.         if fs.exists(install_dir) then fs.delete(install_dir);fs.makeDir(install_dir) end
  632.  
  633.         -- download all dependencies
  634.         for _, dep in pairs(deps) do
  635.             if mode == "update" and unchanged(dep) then
  636.                 pkg_message("skipping download of unchanged package", dep)
  637.             else
  638.                 pkg_message("downloading package", dep)
  639.                 lgray()
  640.  
  641.                 local files = file_list[dep]
  642.                 for _, file in pairs(files) do
  643.                     println("GET "..file)
  644.                     local dl_stat = http_get_file(file, install_dir.."/")
  645.                     success = dl_stat == 0
  646.                     if dl_stat == 1 then
  647.                         red();println("failed to download "..file)
  648.                         break
  649.                     elseif dl_stat == 2 then
  650.                         red();println("filesystem error with "..file)
  651.                         break
  652.                     elseif dl_stat == 3 then
  653.                         -- this shouldn't occur in this mode
  654.                         red();println("no space for "..file)
  655.                         break
  656.                     end
  657.                 end
  658.             end
  659.             if not success then break end
  660.         end
  661.  
  662.         -- copy in downloaded files (installation)
  663.         if success then
  664.             for _, dep in pairs(deps) do
  665.                 if mode == "update" and unchanged(dep) then
  666.                     pkg_message("skipping install of unchanged package", dep)
  667.                 else
  668.                     pkg_message("installing package", dep)
  669.                     lgray()
  670.  
  671.                     local files = file_list[dep]
  672.                     for _, file in pairs(files) do
  673.                         local temp_file = install_dir.."/"..file
  674.                         if fs.exists(file) then fs.delete(file) end
  675.                         fs.move(temp_file, file)
  676.                     end
  677.                 end
  678.             end
  679.         end
  680.  
  681.         fs.delete(install_dir)
  682.     end
  683.  
  684.     if success then
  685.         write_install_manifest(r_manifest, deps)
  686.         green()
  687.         if mode == "install" then
  688.             println("Installation completed successfully.")
  689.         else println("Update completed successfully.") end
  690.         white();println("Ready to clean up unused files, press any key to continue...")
  691.         any_key();clean(r_manifest)
  692.         white();println("Done.")
  693.     else
  694.         red()
  695.         if single_file_mode then
  696.             if mode == "install" then
  697.                 println("Installation failed, files may have been skipped.")
  698.             else println("Update failed, files may have been skipped.") end
  699.         else
  700.             if mode == "install" then
  701.                 println("Installation failed.")
  702.             else orange();println("Update failed, existing files unmodified.") end
  703.         end
  704.     end
  705. elseif mode == "uninstall" then
  706.     local ok, manifest = read_local_manifest()
  707.     if not ok then
  708.         red();println("Error parsing local installation manifest.");white()
  709.         return
  710.     end
  711.  
  712.     if manifest.versions[app] == nil then
  713.         red();println("Error: '"..app.."' is not installed.")
  714.         return
  715.     end
  716.  
  717.     orange();println("Uninstalling all "..app.." files...")
  718.  
  719.     -- ask for confirmation
  720.     if not ask_y_n("Continue", false) then return end
  721.  
  722.     -- delete unused files first
  723.     clean(manifest)
  724.  
  725.     local file_list = manifest.files
  726.     local deps = manifest.depends[app]
  727.  
  728.     table.insert(deps, app)
  729.  
  730.     -- delete all installed files
  731.     lgray()
  732.     for _, dep in pairs(deps) do
  733.         local files = file_list[dep]
  734.         for _, file in pairs(files) do
  735.             if fs.exists(file) then fs.delete(file);println("deleted "..file) end
  736.         end
  737.  
  738.         local folder = files[1]
  739.         while true do
  740.             local dir = fs.getDir(folder)
  741.             if dir == "" or dir == ".." then break else folder = dir end
  742.         end
  743.  
  744.         if fs.isDir(folder) then
  745.             fs.delete(folder)
  746.             println("deleted directory "..folder)
  747.         end
  748.     end
  749.  
  750.     -- delete log file
  751.     local log_deleted = false
  752.     local settings_file = app..".settings"
  753.  
  754.     if fs.exists(settings_file) and settings.load(settings_file) then
  755.         local log = settings.get("LogPath")
  756.         if log ~= nil then
  757.             log_deleted = true
  758.             if fs.exists(log) then
  759.                 fs.delete(log)
  760.                 println("deleted log file "..log)
  761.             end
  762.         end
  763.     end
  764.  
  765.     if not log_deleted then
  766.         red();println("Failed to delete log file (it may not exist).");lgray()
  767.     end
  768.  
  769.     if fs.exists(settings_file) then
  770.         fs.delete(settings_file);println("deleted "..settings_file)
  771.     end
  772.  
  773.     fs.delete("install_manifest.json")
  774.     println("deleted install_manifest.json")
  775.  
  776.     green();println("Done!")
  777. end
  778.  
  779. white()
  780.  
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement