13 /* --------------------------------------------------------------------------
14 * This section should be sufficient to isolate the changes that any forked
15 * versions need to change with a custom support url.
18 const support_url = 'https://forum.openwrt.org/t/luci-attended-sysupgrade-support-thread/230552';
19 const support_link = E('a', { href: support_url }, _('this forum thread'));
21 function detailsBlock(title, content, pre) {
22 /* Formatter for discourse-based forum "details" block.
24 * If the above support_url is changed to github, say, then you'd
25 * probably need to get rid of the '[details...]' syntax.
27 return ! content ? '' : ''.concat(
28 '[details="', title, '"]\n',
36 /* -------------------------------------------------------------------------- */
38 const callPackagelist = rpc.declare({
40 method: 'packagelist',
43 const callSystemBoard = rpc.declare({
48 const callUpgradeStart = rpc.declare({
50 method: 'upgrade_start',
55 * Returns the branch of a given version. This helps to offer upgrades
56 * for point releases (aka within the branch).
59 * SNAPSHOT -> SNAPSHOT
60 * 21.02-SNAPSHOT -> 21.02
61 * 21.02.0-rc1 -> 21.02
64 * @param {string} version
65 * Input version from which to determine the branch
67 * The determined branch
69 function get_branch(version) {
70 return version.replace('-SNAPSHOT', '').split('.').slice(0, 2).join('.');
74 * The OpenWrt revision string contains both a hash as well as the number
75 * commits since the OpenWrt/LEDE reboot. It helps to determine if a
76 * snapshot is newer than another.
78 * @param {string} revision
79 * Revision string of a OpenWrt device
81 * The number of commits since OpenWrt/LEDE reboot
83 function get_revision_count(revision) {
84 return parseInt(revision.substring(1).split('-')[0]);
89 init: [ 0, _('Received build request')],
90 container_setup: [ 10, _('Setting up ImageBuilder')],
91 validate_revision: [ 20, _('Validating revision')],
92 validate_manifest: [ 30, _('Validating package selection')],
93 calculate_packages_hash: [ 40, _('Calculating package hash')],
94 building_image: [ 50, _('Generating firmware image')],
95 signing_images: [ 95, _('Signing images')],
96 done: [100, _('Completed generating firmware image')],
97 failed: [100, _('Failed to generate firmware image')],
99 /* Obsolete status values, retained for backward compatibility. */
100 download_imagebuilder: [ 20, _('Downloading ImageBuilder archive')],
101 unpack_imagebuilder: [ 40, _('Setting Up ImageBuilder')],
104 request_hash: new Map(),
107 applyPackageChanges: async function(package_info) {
108 let { url, target, version, packages } = package_info;
110 const overview_url = `${url}/api/v1/overview`;
111 const revision_url = `${url}/api/v1/revision/${version}/${target}`;
113 let changes, target_revision;
116 request.get(overview_url)
117 .then(response => response.json())
118 .then(json => json.branches)
119 .then(branches => branches[get_branch(version)])
120 .then(branch => { changes = branch.package_changes; })
122 throw Error(`Get overview failed:<br>${overview_url}<br>${error}`);
125 request.get(revision_url)
126 .then(response => response.json())
127 .then(json => json.revision)
128 .then(revision => { target_revision = get_revision_count(revision); })
130 throw Error(`Get revision failed:<br>${revision_url}<br>${error}`);
134 for (const change of changes) {
135 let idx = packages.indexOf(change.source);
136 if (idx >= 0 && change.revision <= target_revision) {
138 packages[idx] = change.target;
140 packages.splice(idx, 1);
146 selectImage: function (images, data, firmware) {
147 var filesystemFilter = function(e) {
148 return (e.filesystem == firmware.filesystem);
150 var typeFilter = function(e) {
151 let efi_targets = ['armsr', 'loongarch', 'x86'];
152 let efi_capable = efi_targets.some((tgt) => firmware.target.startsWith(tgt));
155 return (e.type == 'combined-efi');
157 return (e.type == 'combined');
160 return (e.type == 'sysupgrade' || e.type == 'combined');
163 return images.filter(filesystemFilter).filter(typeFilter)[0];
166 handle200: function (response, content, data, firmware) {
167 response = response.json();
168 let image = this.selectImage(response.images, data, firmware);
170 if (image.name != undefined) {
171 this.sha256_unsigned = image.sha256_unsigned;
172 let sysupgrade_url = `${data.url}/store/${response.bin_dir}/${image.name}`;
174 let keep = E('input', { type: 'checkbox' });
179 `${response.version_number} ${response.version_code}`,
184 if (data.advanced_mode == 1) {
201 E('a', { href: sysupgrade_url }, _('Download firmware image'))
203 if (data.rebuilder) {
204 fields.push(_('Rebuilds'), E('div', { id: 'rebuilder_status' }));
207 let table = E('div', { class: 'table' });
209 for (let i = 0; i < fields.length; i += 2) {
211 E('tr', { class: 'tr' }, [
212 E('td', { class: 'td left', width: '33%' }, [fields[i]]),
213 E('td', { class: 'td left' }, [fields[i + 1]]),
223 E('label', { class: 'btn' }, [
226 _('Keep settings and retain the current configuration'),
229 E('div', { class: 'right' }, [
230 E('div', { class: 'btn', click: ui.hideModal }, _('Cancel')),
235 class: 'btn cbi-button cbi-button-positive important',
236 click: ui.createHandlerFn(this, function () {
237 this.handleInstall(sysupgrade_url, keep.checked, image.sha256);
240 _('Install firmware image')
245 ui.showModal(_('Successfully created firmware image'), modal_body);
246 if (data.rebuilder) {
247 this.handleRebuilder(content, data, firmware);
252 handle202: function (response) {
253 if ('queue_position' in response) {
254 ui.showModal(_('Queued...'), [
257 { class: 'spinning' },
258 _('Request in build queue position %s').format(
259 response.queue_position
264 ui.showModal(_('Building Firmware...'), [
267 { class: 'spinning' },
268 _('Progress: %s%% %s').format(
269 this.steps[response.imagebuilder_status][0],
270 this.steps[response.imagebuilder_status][1]
277 handleError: function (response, data, firmware, request_hash) {
278 response = response.json();
279 const request_data = {
281 request_hash: request_hash,
282 sha256_unsigned: this.sha256_unsigned,
285 const request_str = JSON.stringify(request_data, null, 4);
286 if (typeof response.detail != "string") {
287 response.detail = JSON.stringify(response.detail, null, 4);
291 _('First, check'), ' ',
293 _('. If you don\'t find a solution there, then report all of the information below.')
299 class: 'btn cbi-button cbi-button-positive important',
300 style: 'margin-bottom: 1em; padding: 0.2em',
301 click: ui.createHandlerFn(this, function () {
302 var text = ''.concat(
303 // No translations in here as it's intended for the forum.
304 'Server response: %s\n\n'.format(response.detail),
305 detailsBlock('Request Data', request_str, true),
306 detailsBlock('STDOUT', response.stdout, true),
307 detailsBlock('STDERR', response.stderr, true),
310 navigator.clipboard.writeText(text);
312 ui.showModal(_('Data copied!'), [
314 _('Paste the contents of the clipboard to'), ' ',
318 E('div', { class: 'right' }, [
319 E('div', { class: 'btn', click: ui.hideModal }, _('Close')),
324 _('Copy error data to clipboard...')
327 E('p', _('Server response: %s').format(response.detail)),
328 E('p', _('Request Data:')),
329 E('pre', {}, request_str),
332 if (response.stdout) {
335 E('pre', response.stdout),
339 if (response.stderr) {
342 E('pre', response.stderr),
347 E('div', { class: 'right' }, [
348 E('div', { class: 'btn', click: ui.hideModal }, _('Close')),
352 ui.showModal(_('Error building the firmware image'), body);
355 handleRequest: function (server, main, content, data, firmware) {
356 let request_url = `${server}/api/v1/build`;
358 let local_content = content;
359 const request_hash = this.request_hash.get(server);
362 * If `request_hash` is available use a GET request instead of
363 * sending the entire object.
366 request_url += `/${request_hash}`;
372 .request(request_url, { method: method, content: local_content })
373 .then((response) => {
374 switch (response.status) {
376 response = response.json();
378 this.request_hash.set(server, response.request_hash);
381 this.handle202(response);
383 let view = document.getElementById(server);
384 view.innerText = `⏳ (${
385 this.steps[response.imagebuilder_status][0]
391 poll.remove(this.pollFn);
392 this.handle200(response, content, data, firmware);
394 poll.remove(this.rebuilder_polls[server]);
395 response = response.json();
396 let view = document.getElementById(server);
397 let image = this.selectImage(response.images, data, firmware);
398 if (image.sha256_unsigned == this.sha256_unsigned) {
399 view.innerText = '✅ %s'.format(server);
401 view.innerHTML = `⚠️ ${server} (<a href="${server}/store/${
403 }/${image.name}">${_('Download')}</a>)`;
407 default: // any error or unexpected responses
409 poll.remove(this.pollFn);
410 this.handleError(response, data, firmware, request_hash);
412 poll.remove(this.rebuilder_polls[server]);
413 document.getElementById(server).innerText = '🚫 %s'.format(
422 handleRebuilder: function (content, data, firmware) {
423 this.rebuilder_polls = {};
424 for (let rebuilder of data.rebuilder) {
425 this.rebuilder_polls[rebuilder] = L.bind(
434 poll.add(this.rebuilder_polls[rebuilder], 5);
435 document.getElementById(
437 ).innerHTML += `<p id="${rebuilder}">⏳ ${rebuilder}</p>`;
442 handleInstall: function (url, keep, sha256) {
443 ui.showModal(_('Downloading...'), [
446 { class: 'spinning' },
447 _('Downloading firmware from server to browser')
454 'Content-Type': 'application/x-www-form-urlencoded',
456 responseType: 'blob',
458 .then((response) => {
459 let form_data = new FormData();
460 form_data.append('sessionid', rpc.getSessionID());
461 form_data.append('filename', '/tmp/firmware.bin');
462 form_data.append('filemode', 600);
463 form_data.append('filedata', response.blob());
465 ui.showModal(_('Uploading...'), [
468 { class: 'spinning' },
469 _('Uploading firmware from browser to device')
474 .get(`${L.env.cgi_base}/cgi-upload`, {
478 .then((response) => response.json())
479 .then((response) => {
480 if (response.sha256sum != sha256) {
481 ui.showModal(_('Wrong checksum'), [
482 E('p', _('Error during download of firmware. Please try again')),
483 E('div', { class: 'btn', click: ui.hideModal }, _('Close')),
486 ui.showModal(_('Installing...'), [
487 E('div', { class: 'spinning' }, [
488 E('p', _('Installing the sysupgrade image...')),
490 _('Once the image is written, the system will reboot.')
492 _('This should take at least a minute, so please wait for the login screen.')
494 E('b', _('While you are waiting, do not unpower device!')),
498 L.resolveDefault(callUpgradeStart(keep), {}).then((response) => {
499 // Wait 10 seconds before we try to reconnect...
500 let hosts = keep ? [] : ['192.168.1.1', 'openwrt.lan'];
501 setTimeout(() => { ui.awaitReconnect(...hosts); }, 10000);
508 handleCheck: function (data, firmware) {
509 this.request_hash.clear();
510 let { url, revision, advanced_mode, branch } = data;
511 let { version, target, profile, packages } = firmware;
514 const endpoint = version.endsWith('SNAPSHOT') ? `revision/${version}/${target}` : 'overview';
515 const request_url = `${url}/api/v1/${endpoint}`;
517 ui.showModal(_('Searching...'), [
520 { class: 'spinning' },
521 _('Searching for an available sysupgrade of %s - %s').format(
528 L.resolveDefault(request.get(request_url)).then((response) => {
530 ui.showModal(_('Error connecting to upgrade server'), [
534 _('Could not reach API at "%s". Please try again later.').format(
538 E('pre', {}, response.responseText),
539 E('div', { class: 'right' }, [
540 E('div', { class: 'btn', click: ui.hideModal }, _('Close')),
545 if (version.endsWith('SNAPSHOT')) {
546 const remote_revision = response.json().revision;
548 get_revision_count(revision) < get_revision_count(remote_revision)
550 candidates.push([version, remote_revision]);
553 const latest = response.json().latest;
555 // ensure order: newest to oldest release
556 latest.sort().reverse();
558 for (let remote_version of latest) {
559 let remote_branch = get_branch(remote_version);
561 // already latest version installed
562 if (version == remote_version) {
566 candidates.unshift([remote_version, null]);
568 // don't offer branches older than the current
569 if (branch == remote_branch) {
575 // allow to re-install running firmware in advanced mode
576 if (advanced_mode == 1) {
577 candidates.unshift([version, revision]);
580 if (candidates.length) {
586 version: candidates[0][0],
587 packages: Object.keys(packages).sort(),
591 let map = new form.JSONMap(mapdata, '');
598 'Use defaults for the safest update'
600 o = s.option(form.ListValue, 'version', 'Select firmware version');
601 for (let candidate of candidates) {
602 if (candidate[0] == version && candidate[1] == revision) {
605 _('[installed] %s').format(
607 ? `${candidate[0]} - ${candidate[1]}`
614 candidate[1] ? `${candidate[0]} - ${candidate[1]}` : candidate[0]
619 if (advanced_mode == 1) {
620 o = s.option(form.Value, 'profile', _('Board Name / Profile'));
621 o = s.option(form.DynamicList, 'packages', _('Packages'));
624 L.resolveDefault(map.render()).then((form_rendered) => {
625 ui.showModal(_('New firmware upgrade available'), [
628 _('Currently running: %s - %s').format(
634 E('div', { class: 'right' }, [
635 E('div', { class: 'btn', click: ui.hideModal }, _('Cancel')),
640 class: 'btn cbi-button cbi-button-positive important',
641 click: ui.createHandlerFn(this, function () {
642 map.save().then(() => {
643 this.applyPackageChanges({
646 version: mapdata.request.version,
647 packages: mapdata.request.packages,
648 }).then((packages) => {
652 version: mapdata.request.version,
653 profile: mapdata.request.profile
655 this.pollFn = L.bind(function () {
656 this.handleRequest(url, true, content, data, firmware);
658 poll.add(this.pollFn, 5);
662 ui.addNotification(null, E('p', error.message));
668 _('Request firmware image')
674 ui.showModal(_('No upgrade available'), [
677 _('The device runs the latest firmware version %s - %s').format(
682 E('div', { class: 'right' }, [
683 E('div', { class: 'btn', click: ui.hideModal }, _('Close')),
690 load: async function () {
691 const promises = await Promise.all([
692 L.resolveDefault(callPackagelist(), {}),
693 L.resolveDefault(callSystemBoard(), {}),
694 L.resolveDefault(fs.stat('/sys/firmware/efi'), null),
695 uci.load('attendedsysupgrade'),
698 system_board: promises[1],
699 advanced_mode: uci.get_first('attendedsysupgrade', 'client', 'advanced_mode') || 0,
700 url: uci.get_first('attendedsysupgrade', 'server', 'url').replace(/\/+$/, ''),
701 branch: get_branch(promises[1].release.version),
702 revision: promises[1].release.revision,
704 rebuilder: uci.get_first('attendedsysupgrade', 'server', 'rebuilder'),
707 client: 'luci/' + promises[0].packages['luci-app-attendedsysupgrade'],
708 packages: promises[0].packages,
709 profile: promises[1].board_name,
710 target: promises[1].release.target,
711 version: promises[1].release.version,
713 filesystem: promises[1].rootfs_type,
715 // If the user has changed the rootfs partition size via owut,
716 // then make sure to keep new image the same size. A null value
717 // is interpreted by the ASU server as "default".
718 rootfs_size_mb: uci.get('attendedsysupgrade', 'owut', 'rootfs_size'),
720 return [data, firmware];
723 render: function (response) {
724 const data = response[0];
725 const firmware = response[1];
728 E('h2', _('Attended Sysupgrade')),
732 'The attended sysupgrade service allows to upgrade vanilla and custom firmware images easily.'
738 'This is done by building a new firmware on demand via an online service.'
743 _('Currently running: %s - %s').format(
751 class: 'btn cbi-button cbi-button-positive important',
752 click: ui.createHandlerFn(this, this.handleCheck, data, firmware),
754 _('Search for firmware upgrade')
758 handleSaveApply: null,