attendedsysupgrade.htm 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  1. <%
  2. -- all lua code provided by https://github.com/jow-/
  3. -- thank you very much!
  4. function apply_acls(filename, session)
  5. local json = require "luci.jsonc"
  6. local util = require "luci.util"
  7. local fs = require "nixio.fs"
  8. local grants = { }
  9. local acl = json.parse(fs.readfile(filename))
  10. if type(acl) ~= "table" then
  11. return
  12. end
  13. local group, perms
  14. for group, perms in pairs(acl) do
  15. local perm, scopes
  16. for perm, scopes in pairs(perms) do
  17. if type(scopes) == "table" then
  18. local scope, objects
  19. for scope, objects in pairs(scopes) do
  20. if type(objects) == "table" then
  21. if not grants[scope] then
  22. grants[scope] = { }
  23. end
  24. if next(objects) == 1 then
  25. local _, object
  26. for _, object in ipairs(objects) do
  27. if not grants[scope][object] then
  28. grants[scope][object] = { }
  29. end
  30. table.insert(grants[scope][object], perm)
  31. end
  32. else
  33. local object, funcs
  34. for object, funcs in pairs(objects) do
  35. if type(funcs) == "table" then
  36. local _, func
  37. for _, func in ipairs(funcs) do
  38. if not grants[scope][object] then
  39. grants[scope][object] = { }
  40. end
  41. table.insert(grants[scope][object], func)
  42. end
  43. end
  44. end
  45. end
  46. end
  47. end
  48. end
  49. end
  50. end
  51. local _, scope, object, func
  52. for scope, _ in pairs(grants) do
  53. local objects = { }
  54. for object, _ in pairs(_) do
  55. for _, func in ipairs(_) do
  56. table.insert(objects, { object, func })
  57. end
  58. end
  59. util.ubus("session", "grant", {
  60. ubus_rpc_session = session,
  61. scope = scope, objects = objects
  62. })
  63. end
  64. end
  65. apply_acls("/usr/share/rpcd/acl.d/attendedsysupgrade.json", luci.dispatcher.context.authsession)
  66. apply_acls("/usr/share/rpcd/acl.d/packagelist.json", luci.dispatcher.context.authsession)
  67. %>
  68. <%+header%>
  69. <h2 name="content"><%:Attended Sysupgrade%></h2>
  70. <div class="container">
  71. <div style="display: none" id="update_info" class="alert-message info"></div>
  72. <div style="display: none" id="update_error" class="alert-message danger"></div>
  73. </div>
  74. <input class="cbi-button" value="search for updates" onclick="update_request()" type="button" id="update_button">
  75. <div style="display: none" id="packages" class="alert-message success"></div>
  76. <div class="cbi-value" id="update_packages_container" style="display: block">
  77. <label class="cbi-value-title" for="keep">search for package updates:</label>
  78. <div class="cbi-value-field">
  79. <input type="checkbox" name="update_packages" id="update_packages" />
  80. </div>
  81. </div>
  82. <div class="cbi-value" id="keep_container" style="display: none">
  83. <label class="cbi-value-title" for="keep">keep settings:</label>
  84. <div class="cbi-value-field">
  85. <input type="checkbox" name="keep" id="keep" checked="checked" />
  86. </div>
  87. </div>
  88. <script type="text/javascript">
  89. latest_version = "";
  90. data = {};
  91. ubus_counter = 1
  92. origin = document.location.href.replace(location.pathname, "")
  93. ubus_url = origin + "/ubus/"
  94. // requests to the update server
  95. function server_request(request_dict, path, callback) {
  96. url = data.update_server + "/" + path
  97. request_dict.distro = data.release.distribution;
  98. request_dict.target = data.release.target.split("\/")[0];
  99. request_dict.subtarget = data.release.target.split("\/")[1];
  100. var xmlhttp = new XMLHttpRequest();
  101. xmlhttp.open("POST", url, true);
  102. xmlhttp.setRequestHeader("Content-type", "application/json");
  103. xmlhttp.send(JSON.stringify(request_dict));
  104. xmlhttp.onerror = function(e) {
  105. update_error("update server down")
  106. }
  107. xmlhttp.addEventListener('load', function(event) {
  108. callback(xmlhttp)
  109. });
  110. }
  111. // requests ubus via rpcd
  112. function ubus_request(command, argument, params, callback) {
  113. request_data = {};
  114. request_data.jsonrpc = "2.0";
  115. request_data.id = ubus_counter;
  116. request_data.method = "call";
  117. request_data.params = [ data.ubus_rpc_session, command, argument, params ]
  118. ubus_counter++
  119. var xmlhttp = new XMLHttpRequest();
  120. xmlhttp.open("POST", ubus_url, true);
  121. xmlhttp.setRequestHeader("Content-type", "application/json");
  122. xmlhttp.onerror = function(e) {
  123. setTimeout(back_online, 5000)
  124. }
  125. xmlhttp.addEventListener('load', function(event) {
  126. if(command === "uci") {
  127. ubus_request_callback_uci(xmlhttp, callback)
  128. } else {
  129. ubus_request_callback(xmlhttp, callback)
  130. }
  131. });
  132. xmlhttp.send(JSON.stringify(request_data));
  133. }
  134. // handle ubus_requests, set variables or perform functions
  135. function ubus_request_callback(response_object, callback) {
  136. if(response_object.status === 200) {
  137. console.log(callback)
  138. if(typeof callback === "string") {
  139. response_json = JSON.parse(response_object.responseText).result[1]
  140. if (callback == "release") {
  141. latest_version = response_json.release.version
  142. }
  143. data[callback] = response_json[callback]
  144. } else {
  145. callback(response_object)
  146. }
  147. } else {
  148. console.log(respons_object.responseText)
  149. }
  150. }
  151. function ubus_request_callback_uci(response_object, callback) {
  152. if(response_object.status === 200) {
  153. console.log(callback)
  154. response_json = JSON.parse(response_object.responseText).result[1].value
  155. data[callback] = response_json
  156. document.getElementById("update_packages").checked = data.update_packages;
  157. } else {
  158. console.log(respons_object.responseText)
  159. }
  160. }
  161. // initial setup, get system information
  162. function setup() {
  163. data["ubus_rpc_session"] = "<%=luci.dispatcher.context.authsession%>"
  164. ubus_request("packagelist", "list", {}, "packagelist");
  165. ubus_request("system", "board", {}, "release");
  166. ubus_request("system", "board", {}, "board_name");
  167. ubus_request("system", "board", {}, "model");
  168. ubus_request("uci", "get", { "config": "attendedsysupgrade", "section": "updateserver", "option": "url" }, "update_server")
  169. ubus_request("uci", "get", { "config": "attendedsysupgrade", "section": "updateclient", "option": "update_packages" }, "update_packages")
  170. }
  171. // shows notification if update is available
  172. function update_info(info_output) {
  173. document.getElementById("update_info").style.display = "block";
  174. document.getElementById("update_info").innerHTML = info_output;
  175. }
  176. function update_error(error_output) {
  177. document.getElementById("update_error").style.display = "block";
  178. document.getElementById("update_error").innerHTML = error_output;
  179. document.getElementById("update_info").style.display = "none";
  180. }
  181. // asks server for news updates, actually only based on relesae not packages
  182. function update_request() {
  183. console.log("update_request")
  184. request_dict = {}
  185. request_dict.version = data.release.version;
  186. request_dict.packages = data.packagelist;
  187. if (document.getElementById("update_packages").checked == 1) {
  188. request_dict.update_packages = 1
  189. }
  190. server_request(request_dict, "update-request", update_request_callback)
  191. }
  192. function update_request_callback(response_object) {
  193. if (response_object.status === 500) {
  194. // python crashed
  195. update_error("internal server error, please try again later")
  196. console.log("update server issue")
  197. } else if (response_object.status === 502) {
  198. // python part offline
  199. update_error("internal server error, please try again later")
  200. console.log("update server issue")
  201. } else if (response_object.status === 503) {
  202. // handle overload
  203. update_error("server overloaded, retry in 5 minutes")
  204. console.log("server overloaded")
  205. setTimeout(update_request, 300000)
  206. } else if (response_object.status === 201) {
  207. update_info("imagebuilder not ready, please wait")
  208. console.log("setting up imagebuilder")
  209. setTimeout(update_request, 5000)
  210. } else if (response_object.status === 204) {
  211. // no updates
  212. update_info("no updates available")
  213. } else if (response_object.status === 400) {
  214. // bad request
  215. console.log(response_object.responseText)
  216. response_object_content = JSON.parse(response_object.responseText)
  217. update_error(response_object_content.error)
  218. } else if (response_object.status === 200) {
  219. // new release/updates
  220. response_object_content = JSON.parse(response_object.responseText)
  221. update_request_200(response_object_content)
  222. }
  223. }
  224. function back_online() {
  225. ubus_request("session", "login", {}, back_online_callback)
  226. }
  227. function back_online_callback(response_object) {
  228. if (response_object.status != 200) {
  229. setTimeout(back_online, 5000)
  230. } else {
  231. update_info("upgrade successfull!")
  232. document.getElementById("update_button").value = "reload page";
  233. document.getElementById("update_button").onclick = function() { location.reload(); }
  234. }
  235. }
  236. function update_request_200(response_content) {
  237. info_output = ""
  238. if(response_content.version != undefined) {
  239. info_output += "<h3>new update available</h3>"
  240. info_output += data.release.version + " to " + response_content.version
  241. latest_version = response_content.version;
  242. }
  243. if(response_content.updates != undefined) {
  244. info_output += "<h3>package updates available</h3>"
  245. for (update in response_content.updates) {
  246. info_output += "<b>" + update + "</b>: " + response_content.updates[update][1] + " to " + response_content.updates[update][0] + "</br>"
  247. }
  248. }
  249. data.packages = response_content.packages
  250. update_info(info_output)
  251. document.getElementById("update_button").value = "request image";
  252. document.getElementById("update_packages_container").style.display = "none";
  253. document.getElementById("update_button").onclick = image_request;
  254. }
  255. // request the image, need merge with update_request
  256. function image_request() {
  257. console.log("image_request")
  258. document.getElementById("update_packages_container").style.display = "none";
  259. request_dict = {}
  260. request_dict.version = latest_version;
  261. request_dict.board = data.board_name
  262. request_dict.packages = data.packages;
  263. request_dict.model = data.model
  264. server_request(request_dict, "image-request", image_request_handler)
  265. }
  266. function image_request_handler(response) {
  267. if (response.status === 400) {
  268. response_content = JSON.parse(response.responseText)
  269. update_error(response_content.error)
  270. } else if (response.status === 500) {
  271. image_request_500()
  272. } else if (response.status === 503) {
  273. update_error("please wait. server overloaded")
  274. // handle overload
  275. setTimeout(image_request, 30000)
  276. } else if (response.status === 201) {
  277. response_content = JSON.parse(response.responseText)
  278. if(response_content.queue != undefined) {
  279. // in queue
  280. update_info("please wait. you are in queue position " + response_content.queue)
  281. console.log("queued")
  282. } else {
  283. update_info("imagebuilder not ready, please wait")
  284. console.log("setting up imagebuilder")
  285. }
  286. setTimeout(image_request, 5000)
  287. } else if (response.status === 206) {
  288. // building
  289. console.log("building")
  290. update_info("building image")
  291. setTimeout(image_request, 5000)
  292. } else if (response.status === 200) {
  293. // ready to download
  294. response_content = JSON.parse(response.responseText)
  295. update_info("image created")
  296. document.getElementById("update_button").value = "sysupgrade"
  297. document.getElementById("update_button").onclick = function() {download_image(response_content.url); }
  298. document.getElementById("keep_container").style.display = "block";
  299. }
  300. }
  301. // uploads received blob data to the server using cgi-io
  302. function upload_image(blob) {
  303. var upload_request = new XMLHttpRequest();
  304. var form_data = new FormData();
  305. form_data.append("sessionid", data.ubus_rpc_session)
  306. form_data.append("filename", "/tmp/sysupgrade.bin")
  307. form_data.append("filemode", 755) // insecure?
  308. form_data.append("filedata", blob)
  309. upload_request.addEventListener('load', function(event) {
  310. // this checksum should be parsed
  311. document.getElementById("update_info").innerHTML = "flashing... please wait" // show fancy indicator http://www.ajaxload.info/
  312. ubus_request("attendedsysupgrade", "sysupgrade", { "keep_settings": document.getElementById("keep").checked }, 'done');
  313. });
  314. upload_request.addEventListener('error', function(event) {
  315. document.getElementById("update_info").innerHTML = "uploading failed, please retry"
  316. });
  317. upload_request.open('POST', origin + '/cgi-bin/cgi-upload');
  318. upload_request.send(form_data);
  319. }
  320. // download image from server once the url was received by image_request
  321. function download_image(url) {
  322. console.log("download_image")
  323. document.getElementById("update_button").value = "flashing..."
  324. document.getElementById("update_button").disabled = true;
  325. var download_request = new XMLHttpRequest();
  326. download_request.open("GET", url);
  327. download_request.responseType = "arraybuffer";
  328. download_request.onload = function () {
  329. if (this.status === 200) {
  330. var blob = new Blob([download_request.response], {type: "application/octet-stream"});
  331. upload_image(blob)
  332. }
  333. };
  334. document.getElementById("update_info").innerHTML = "downloading image"
  335. download_request.send();
  336. }
  337. document.onload = setup()
  338. </script>
  339. <%+footer%>