attendedsysupgrade.htm 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520
  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. %>
  67. <%+header%>
  68. <h2 name="content"><%:Attended Sysupgrade%></h2>
  69. <div class="cbi-map-descr">
  70. Easily search and install new releases and package upgrades. Sysupgrade firmware are created on demand based on locally installed packages.
  71. </div>
  72. <div style="display: none" id="info_box" class="alert-message info"></div>
  73. <div style="display: none" id="error_box" class="alert-message danger"></div>
  74. <div style="display: none" id="packages" class="alert-message success"></div>
  75. <p>
  76. <textarea style="display: none; width: 100%;" id="edit_packages" rows="15"></textarea>
  77. </p>
  78. <fieldset class="cbi-section">
  79. <form method="post" action="">
  80. <div class="cbi-selection-node">
  81. <div class="cbi-value" id="keep_container" style="display: none">
  82. <div class="cbi-section-descr">
  83. Check "Keep settings" to retain the current configuration (requires a compatible firmware).
  84. </div>
  85. <label class="cbi-value-title" for="keep">Keep settings:</label>
  86. <div class="cbi-value-field">
  87. <input name="keep" id="keep" checked="checked" type="checkbox">
  88. </div>
  89. </div>
  90. <div class="cbi-value" id="edit_button" style="display: none">
  91. <div class="cbi-value-field">
  92. <input class="cbi-button" value="Edit installed packages" onclick="edit_packages()" type="button">
  93. </div>
  94. </div>
  95. <div class="cbi-value cbi-value" id="server_div" style="display:none">
  96. <label class="cbi-value-title" for="server">Server:</label>
  97. <div class="cbi-value-field">
  98. <input onclick="edit_server()" class="cbi-button cbi-button-edit" value="" type="button" id="server" name="server">
  99. </div>
  100. </div>
  101. <div class="cbi-value cbi-value-last">
  102. <div class="cbi-value-field">
  103. <input class="cbi-button cbi-button-apply" value="Search for upgrades" style="display: none" onclick="upgrade_check()" type="button" id="upgrade_button">
  104. </div>
  105. </div>
  106. </div>
  107. </form>
  108. </fieldset>
  109. <script type="text/javascript">
  110. data = {};
  111. origin = document.location.href.replace(location.pathname, "")
  112. ubus_url = origin + "/ubus/"
  113. function set_server() {
  114. document.getElementById("error_box").style.display = "none";
  115. data.url = document.getElementById("server").value;
  116. ubus_call("uci", "set", { "config": "attendedsysupgrade", "section": "server", values: { "url": data.url } })
  117. ubus_call("uci", "commit", { "config": "attendedsysupgrade" })
  118. var server_button = document.getElementById("server")
  119. server_button.type = 'button';
  120. server_button.className = 'cbi-button cbi-button-edit';
  121. server_button.parentElement.removeChild(document.getElementById("button_set"));
  122. server_button.onclick = edit_server;
  123. }
  124. function edit_server() {
  125. document.getElementById("server").type = 'text';
  126. document.getElementById("server").onkeydown = function(event) {
  127. if(event.key === 'Enter') {
  128. set_server();
  129. return false;
  130. }
  131. }
  132. document.getElementById("server").className = '';
  133. document.getElementById("server").onclick = null;
  134. button_set = document.createElement("input");
  135. button_set.type = "button";
  136. button_set.value = "Save";
  137. button_set.name = "button_set";
  138. button_set.id = "button_set";
  139. button_set.className = 'cbi-button cbi-button-edit';
  140. button_set.style = 'background-image: url("/luci-static/resources/cbi/save.gif");'
  141. button_set.onclick = set_server
  142. document.getElementById("server").parentElement.appendChild(button_set);
  143. }
  144. function edit_packages() {
  145. data.edit_packages = true
  146. document.getElementById("edit_button").style.display = "none";
  147. document.getElementById("edit_packages").value = data.packages.join("\n");
  148. document.getElementById("edit_packages").style.display = "block";
  149. }
  150. // requests to the upgrade server
  151. function server_request(request_dict, path, callback) {
  152. request_dict.distro = data.release.distribution;
  153. request_dict.target = data.release.target.split("\/")[0];
  154. request_dict.subtarget = data.release.target.split("\/")[1];
  155. var request = new XMLHttpRequest();
  156. request.open("POST", data.url + "/" + path, true);
  157. request.setRequestHeader("Content-type", "application/json");
  158. request.send(JSON.stringify(request_dict));
  159. request.onerror = function(e) {
  160. error_box("upgrade server down")
  161. document.getElementById("server_div").style.display = "block";
  162. }
  163. request.addEventListener('load', function(event) {
  164. callback(request)
  165. });
  166. }
  167. // initial setup, get system information
  168. function setup() {
  169. data["ubus_rpc_session"] = "<%=luci.dispatcher.context.authsession%>"
  170. ubus_call("rpc-sys", "packagelist", {}, "packages");
  171. ubus_call("system", "board", {}, "release");
  172. ubus_call("system", "board", {}, "board_name");
  173. ubus_call("system", "board", {}, "model");
  174. ubus_call("system", "info", {}, "memory");
  175. uci_get({ "config": "attendedsysupgrade", "section": "server", "option": "url" })
  176. uci_get({ "config": "attendedsysupgrade", "section": "client", "option": "upgrade_packages" })
  177. uci_get({ "config": "attendedsysupgrade", "section": "client", "option": "advanced_mode" })
  178. uci_get({ "config": "attendedsysupgrade", "section": "client", "option": "auto_search" })
  179. setup_ready();
  180. }
  181. function setup_ready() {
  182. // checks if a async ubus calls have finished
  183. if(ubus_counter != ubus_closed) {
  184. setTimeout(setup_ready, 300)
  185. } else {
  186. if(data.auto_search == 1) {
  187. upgrade_check();
  188. } else {
  189. document.getElementById("upgrade_button").style.display = "block";
  190. document.getElementById("server_div").style.display = "block";
  191. document.getElementById("server").value = data.url;
  192. }
  193. }
  194. }
  195. function uci_get(option) {
  196. // simple wrapper to get a uci value store in data.<option>
  197. ubus_call("uci", "get", option, option["option"])
  198. }
  199. ubus_counter = 0;
  200. ubus_closed = 0;
  201. function ubus_call(command, argument, params, variable) {
  202. var request_data = {};
  203. request_data.jsonrpc = "2.0";
  204. request_data.id = ubus_counter;
  205. request_data.method = "call";
  206. request_data.params = [ data.ubus_rpc_session, command, argument, params ]
  207. request_json = JSON.stringify(request_data)
  208. ubus_counter++;
  209. var request = new XMLHttpRequest();
  210. request.open("POST", ubus_url, true);
  211. request.setRequestHeader("Content-type", "application/json");
  212. request.onload = function(event) {
  213. if(request.status === 200) {
  214. var response = JSON.parse(request.responseText)
  215. if(!("error" in response) && "result" in response) {
  216. if(response.result.length === 2) {
  217. if(command === "uci") {
  218. data[variable] = response.result[1].value
  219. } else {
  220. data[variable] = response.result[1][variable]
  221. }
  222. }
  223. } else {
  224. error_box("<b>Ubus call faild:</b></br>Request: " + request_json + "</br>Response: " + JSON.stringify(response))
  225. }
  226. ubus_closed++;
  227. }
  228. }
  229. request.send(request_json);
  230. }
  231. function info_box(info_output, loading) {
  232. // Shows notification if upgrade is available
  233. // If loading is true then an "processing" animation is added
  234. document.getElementById("info_box").style.display = "block";
  235. var loading_image = '';
  236. if(loading) {
  237. loading_image = '<img src="/luci-static/resources/icons/loading.gif" alt="Loading" style="vertical-align:middle">';
  238. }
  239. document.getElementById("info_box").innerHTML = loading_image + info_output;
  240. }
  241. function error_box(error_output) {
  242. // Shows erros in red box
  243. document.getElementById("error_box").style.display = "block";
  244. document.getElementById("error_box").innerHTML = error_output;
  245. document.getElementById("info_box").style.display = "none";
  246. }
  247. function upgrade_check() {
  248. // Asks server for new firmware
  249. // If data.upgrade_packages is set to true search for new package versions as well
  250. document.getElementById("error_box").style.display = "none";
  251. document.getElementById("server_div").style.display = "none";
  252. info_box("Searching for upgrades", true);
  253. var request_dict = {}
  254. request_dict.version = data.release.version;
  255. request_dict.packages = data.packages;
  256. request_dict.upgrade_packages = data.upgrade_packages
  257. server_request(request_dict, "api/upgrade-check", upgrade_check_callback)
  258. }
  259. function upgrade_check_callback(request_text) {
  260. var request_json = JSON.parse(request_text)
  261. // create simple output to tell user whats going to be upgrade (release/packages)
  262. var info_output = ""
  263. if(request_json.version != undefined) {
  264. info_output += "<h3>New firmware release available</h3>"
  265. info_output += data.release.version + " to " + request_json.version
  266. data.latest_version = request_json.version;
  267. }
  268. if(request_json.upgrades != undefined) {
  269. info_output += "<h3>Package upgrades available</h3>"
  270. for (upgrade in request_json.upgrades) {
  271. info_output += "<b>" + upgrade + "</b>: " + request_json.upgrades[upgrade][1] + " to " + request_json.upgrades[upgrade][0] + "</br>"
  272. }
  273. }
  274. data.packages = request_json.packages
  275. info_box(info_output)
  276. if(data.advanced_mode == 1) {
  277. document.getElementById("edit_button").style.display = "block";
  278. }
  279. var upgrade_button = document.getElementById("upgrade_button")
  280. upgrade_button.value = "Request firmware";
  281. upgrade_button.style.display = "block";
  282. upgrade_button.disabled = false;
  283. upgrade_button.onclick = upgrade_request;
  284. }
  285. function upgrade_request() {
  286. // Request the image
  287. // Needed values
  288. // version/release
  289. // board_name or model (server tries to find the corrent profile)
  290. // packages
  291. // The rest is added by server_request()
  292. document.getElementById("upgrade_button").disabled = true;
  293. document.getElementById("edit_packages").style.display = "none";
  294. document.getElementById("edit_button").style.display = "none";
  295. document.getElementById("keep_container").style.display = "none";
  296. var request_dict = {}
  297. request_dict.version = data.latest_version;
  298. request_dict.board = data.board_name
  299. request_dict.model = data.model
  300. if(data.edit_packages == true) {
  301. request_dict.packages = document.getElementById("edit_packages").value.split("\n")
  302. } else {
  303. request_dict.packages = data.packages;
  304. }
  305. server_request(request_dict, "api/upgrade-request", upgrade_request_callback)
  306. }
  307. function upgrade_request_callback(request) {
  308. // ready to download
  309. var request_json = JSON.parse(request);
  310. data.sysupgrade_url = request_json.sysupgrade;
  311. data.checksum = request_json.checksum;
  312. data.filesize = request_json.filesize;
  313. info_output = "Firmware created"
  314. if(data.advanced_mode == 1) {
  315. info_output += '</br><a target="_blank" href="' + data.sysupgrade_url + '.log">Build log</a>'
  316. }
  317. info_box(info_output);
  318. document.getElementById("keep_container").style.display = "block";
  319. var upgrade_button = document.getElementById("upgrade_button")
  320. upgrade_button.disabled = false;
  321. upgrade_button.style.display = "block";
  322. upgrade_button.value = "Flash firmware";
  323. upgrade_button.onclick = download_image;
  324. }
  325. function flash_image() {
  326. // Flash image via rpc-sys upgrade_start
  327. info_box("Flashing firmware. Don't unpower device", true)
  328. ubus_call("rpc-sys", "upgrade_start", { "keep": document.getElementById("keep").checked }, 'message');
  329. ping_max = 3600; // in seconds
  330. setTimeout(ping_ubus, 10000)
  331. }
  332. function ping_ubus() {
  333. // Tries to connect to ubus. If the connection fails the device is likely still rebooting.
  334. // If more time than ping_max passes update may failed
  335. if(ping_max > 0) {
  336. ping_max--;
  337. var request = new XMLHttpRequest();
  338. request.open("GET", ubus_url, true);
  339. request.addEventListener('error', function(event) {
  340. info_box("Rebooting device", true);
  341. setTimeout(ping_ubus, 1000)
  342. });
  343. request.addEventListener('load', function(event) {
  344. info_box("Success! Please reload web interface");
  345. document.getElementById("upgrade_button").value = "reload page";
  346. document.getElementById("upgrade_button").style.display = "block";
  347. document.getElementById("upgrade_button").disabled = false;
  348. document.getElementById("upgrade_button").onclick = function() { location.reload(); }
  349. });
  350. request.send();
  351. } else {
  352. error_box("Web interface could not reconnect to your device. Please reload web interface or check device manually")
  353. }
  354. }
  355. function upload_image(blob) {
  356. // Uploads received blob data to the server using cgi-io
  357. var request = new XMLHttpRequest();
  358. var form_data = new FormData();
  359. form_data.append("sessionid", data.ubus_rpc_session)
  360. form_data.append("filename", "/tmp/firmware.bin")
  361. form_data.append("filemode", 755) // insecure?
  362. form_data.append("filedata", blob)
  363. request.addEventListener('load', function(event) {
  364. request_json = JSON.parse(request.responseText)
  365. if(data.checksum != request_json.checksum) {
  366. error_box("Checksum missmatch! Please retry")
  367. } else {
  368. flash_image();
  369. }
  370. });
  371. request.addEventListener('error', function(event) {
  372. info_box("Upload of firmware failed, please retry by reloading web interface")
  373. });
  374. request.open('POST', origin + '/cgi-bin/cgi-upload');
  375. request.send(form_data);
  376. }
  377. function download_image() {
  378. // Download image from server once the url was received by upgrade_request
  379. if(data.filesize > data.memory.free) {
  380. error_box("Not enough free memory to download firmware. Please stop unneeded services on router and retry")
  381. } else {
  382. document.getElementById("keep_container").style.display = "none";
  383. document.getElementById("upgrade_button").style.display = "none";
  384. var download_request = new XMLHttpRequest();
  385. download_request.open("GET", data.sysupgrade_url);
  386. download_request.responseType = "arraybuffer";
  387. download_request.onload = function () {
  388. if (this.status === 200) {
  389. var blob = new Blob([download_request.response], {type: "application/octet-stream"});
  390. upload_image(blob)
  391. }
  392. };
  393. info_box("Downloading firmware", true);
  394. download_request.send();
  395. }
  396. }
  397. function server_request(request_dict, path, callback) {
  398. request_dict.distro = data.release.distribution;
  399. request_dict.target = data.release.target.split("\/")[0];
  400. request_dict.subtarget = data.release.target.split("\/")[1];
  401. var request = new XMLHttpRequest();
  402. request.open("POST", data.url + "/" + path, true);
  403. request.setRequestHeader("Content-type", "application/json");
  404. request.send(JSON.stringify(request_dict));
  405. request.onerror = function(e) {
  406. error_box("Upgrade server down or could not connect")
  407. document.getElementById("server_div").style.display = "block";
  408. }
  409. request.addEventListener('load', function(event) {
  410. request_text = request.responseText;
  411. if (request.status === 200) {
  412. callback(request_text)
  413. } else if (request.status === 202) {
  414. var imagebuilder = request.getResponseHeader("X-Imagebuilder-Status");
  415. if(imagebuilder === "queue") {
  416. // in queue
  417. var queue = request.getResponseHeader("X-Build-Queue-Position");
  418. info_box("In build queue position " + queue, true)
  419. console.log("queued");
  420. } else if(imagebuilder === "initialize") {
  421. info_box("Setting up ImageBuilder", true)
  422. console.log("Setting up imagebuilder");
  423. } else if(imagebuilder === "building") {
  424. info_box("Building image");
  425. console.log("building");
  426. } else {
  427. info_box("Processing request");
  428. console.log(imagebuilder)
  429. }
  430. setTimeout(function() { server_request(request_dict, path, callback) }, 5000)
  431. } else if (request.status === 204) {
  432. // no upgrades available
  433. info_box("No upgrades available")
  434. } else if (request.status === 400) {
  435. // bad request
  436. request_json = JSON.parse(request_text)
  437. error_box(request_json.error)
  438. } else if (request.status === 412) {
  439. // this is a bit generic
  440. error_box("Unsupported device, release, target, subtraget or board")
  441. } else if (request.status === 413) {
  442. error_box("No firmware created due to image size. Try again with less packages selected.")
  443. } else if (request.status === 422) {
  444. error_box("Unknown package in request")
  445. } else if (request.status === 500) {
  446. request_json = JSON.parse(request_text)
  447. error_box_content = "<b>Internal server error</b></br>"
  448. error_box_content += request_json.error
  449. if(request_json.log != undefined) {
  450. data.log_url = request_json.log
  451. }
  452. error_box(error_box_content)
  453. } else if (request.status === 501) {
  454. error_box("No sysupgrade file produced, may not supported by modell.")
  455. } else if (request.status === 502) {
  456. // python part offline
  457. error_box("Server down for maintenance")
  458. setTimeout(function() { server_request(request_dict, path, callback) }, 30000)
  459. } else if (request.status === 503) {
  460. error_box("Server overloaded")
  461. setTimeout(function() { server_request(request_dict, path, callback) }, 30000)
  462. }
  463. });
  464. }
  465. document.onload = setup()
  466. </script>
  467. <%+footer%>