mirror of
https://github.com/j-a-n/raspberrymatic-addon-rmupdate.git
synced 2023-10-10 13:37:40 +02:00
Replace /lib just before reboot, patch cmdline.txt and /etc/fstab
This commit is contained in:
parent
7c9d7e9492
commit
4af69aba51
@ -128,6 +128,66 @@ proc ::rmupdate::is_system_upgradeable {} {
|
|||||||
return 1
|
return 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
proc ::rmupdate::get_part_id {device} {
|
||||||
|
#set data [exec blkid $device]
|
||||||
|
#foreach d [split $data "\n"] {
|
||||||
|
# # recent busybox version needed
|
||||||
|
# regexp {PARTUUID="([^"]+)"} $d match partuuid
|
||||||
|
# if { [info exists partuuid] } {
|
||||||
|
# return $partuuid
|
||||||
|
# }
|
||||||
|
#}
|
||||||
|
foreach f [glob /dev/disk/by-partuuid/*] {
|
||||||
|
set d ""
|
||||||
|
catch {
|
||||||
|
set d [file readlink $f]
|
||||||
|
}
|
||||||
|
if { [file tail $d] == [file tail $device] } {
|
||||||
|
return [file tail $f]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
proc ::rmupdate::update_cmdline {cmdline root} {
|
||||||
|
set fd [open $cmdline r]
|
||||||
|
set data [read $fd]
|
||||||
|
close $fd
|
||||||
|
|
||||||
|
regsub -all "root=\[a-zA-Z0-9=/-\]+\s" $data "root=${root} " data
|
||||||
|
|
||||||
|
set fd [open $cmdline w]
|
||||||
|
puts $fd $data
|
||||||
|
close $fd
|
||||||
|
}
|
||||||
|
|
||||||
|
proc ::rmupdate::update_fstab {fstab {boot ""} {root ""} {user ""}} {
|
||||||
|
set ndata ""
|
||||||
|
set fd [open $fstab r]
|
||||||
|
set data [read $fd]
|
||||||
|
foreach d [split $data "\n"] {
|
||||||
|
set filesystem ""
|
||||||
|
regexp {^([^#]\S+)\s+(\S+)\s+(\S+)\s+(\S+)\s+(\S+)\s+(\S+).*} $d match filesystem mountpoint type options dump pass
|
||||||
|
if { [info exists filesystem] } {
|
||||||
|
if {$filesystem != ""} {
|
||||||
|
if {$mountpoint == "/" && $root != ""} {
|
||||||
|
regsub -all $filesystem $d $root d
|
||||||
|
} elseif {$mountpoint == "/boot" && $boot != ""} {
|
||||||
|
regsub -all $filesystem $d $boot d
|
||||||
|
} elseif {$mountpoint == "/usr/local" && $user != ""} {
|
||||||
|
regsub -all $filesystem $d $user d
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
append ndata "${d}\n"
|
||||||
|
}
|
||||||
|
close $fd
|
||||||
|
|
||||||
|
set fd [open $fstab w]
|
||||||
|
puts $fd $ndata
|
||||||
|
close $fd
|
||||||
|
}
|
||||||
|
|
||||||
proc ::rmupdate::mount_image_partition {image partition mountpoint} {
|
proc ::rmupdate::mount_image_partition {image partition mountpoint} {
|
||||||
variable loop_dev
|
variable loop_dev
|
||||||
variable sys_dev
|
variable sys_dev
|
||||||
@ -211,6 +271,8 @@ proc ::rmupdate::check_sizes {image} {
|
|||||||
proc ::rmupdate::update_filesystems {image {dryrun 0}} {
|
proc ::rmupdate::update_filesystems {image {dryrun 0}} {
|
||||||
variable mnt_new
|
variable mnt_new
|
||||||
variable mnt_cur
|
variable mnt_cur
|
||||||
|
variable sys_dev
|
||||||
|
|
||||||
set extra_args ""
|
set extra_args ""
|
||||||
if {$dryrun != 0} {
|
if {$dryrun != 0} {
|
||||||
set extra_args "--dry-run"
|
set extra_args "--dry-run"
|
||||||
@ -228,9 +290,30 @@ proc ::rmupdate::update_filesystems {image {dryrun 0}} {
|
|||||||
mount_system_partition $partition $mnt_cur
|
mount_system_partition $partition $mnt_cur
|
||||||
|
|
||||||
write_log "Rsyncing filesystem of partition ${partition}."
|
write_log "Rsyncing filesystem of partition ${partition}."
|
||||||
set data [exec rsync ${extra_args} --progress --archive --delete "${mnt_new}/" "${mnt_cur}"]
|
if {$partition == 2} {
|
||||||
|
exec rsync ${extra_args} --progress --archive --delete --exclude=/lib "${mnt_new}/" "${mnt_cur}"
|
||||||
|
exec rsync ${extra_args} --progress --archive --delete "${mnt_new}/lib/" "${mnt_cur}/lib.rmupdate"
|
||||||
|
} else {
|
||||||
|
exec rsync ${extra_args} --progress --archive --delete "${mnt_new}/" "${mnt_cur}"
|
||||||
|
}
|
||||||
write_log "Rsync finished."
|
write_log "Rsync finished."
|
||||||
|
|
||||||
|
if {$partition == 1} {
|
||||||
|
write_log "Update cmdline."
|
||||||
|
if {$dryrun == 0} {
|
||||||
|
update_cmdline "${mnt_cur}/cmdline.txt" "${sys_dev}p2"
|
||||||
|
#set partid [get_part_id "${sys_dev}p2"]
|
||||||
|
#if { $partid != "" } {
|
||||||
|
# set_root_in_cmdline "${mnt_cur}/cmdline.txt" "PARTUUID=${partid}"
|
||||||
|
#}
|
||||||
|
}
|
||||||
|
} elseif {$partition == 2} {
|
||||||
|
write_log "Update fstab."
|
||||||
|
if {$dryrun == 0} {
|
||||||
|
update_fstab "${mnt_cur}/etc/fstab" "${sys_dev}p1" "/dev/root" "${sys_dev}p3"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
umount $mnt_new
|
umount $mnt_new
|
||||||
umount $mnt_cur
|
umount $mnt_cur
|
||||||
}
|
}
|
||||||
@ -261,7 +344,7 @@ proc ::rmupdate::get_available_firmware_downloads {} {
|
|||||||
variable release_url
|
variable release_url
|
||||||
set rpi_version [get_rpi_version]
|
set rpi_version [get_rpi_version]
|
||||||
set download_urls [list]
|
set download_urls [list]
|
||||||
set data [exec wget "${release_url}" --no-check-certificate -q -O-]
|
set data [exec /usr/bin/wget "${release_url}" --no-check-certificate -q -O-]
|
||||||
foreach d [split $data ">"] {
|
foreach d [split $data ">"] {
|
||||||
set href ""
|
set href ""
|
||||||
regexp {<\s*a\s+href\s*=\s*"([^"]+/releases/download/[^"]+)\.zip"} $d match href
|
regexp {<\s*a\s+href\s*=\s*"([^"]+/releases/download/[^"]+)\.zip"} $d match href
|
||||||
@ -309,15 +392,15 @@ proc ::rmupdate::download_firmware {version} {
|
|||||||
set archive_file "${img_dir}/${archive_file}"
|
set archive_file "${img_dir}/${archive_file}"
|
||||||
file mkdir $img_dir
|
file mkdir $img_dir
|
||||||
if {$log_file != ""} {
|
if {$log_file != ""} {
|
||||||
exec wget "${download_url}" --show-progress --progress=dot:giga --no-check-certificate --quiet --output-document=$archive_file 2>>${log_file}
|
exec /usr/bin/wget "${download_url}" --show-progress --progress=dot:giga --no-check-certificate --quiet --output-document=$archive_file 2>>${log_file}
|
||||||
write_log ""
|
write_log ""
|
||||||
} else {
|
} else {
|
||||||
exec wget "${download_url}" --no-check-certificate --quiet --output-document=$archive_file
|
exec /usr/bin/wget "${download_url}" --no-check-certificate --quiet --output-document=$archive_file
|
||||||
}
|
}
|
||||||
write_log "Download completed."
|
write_log "Download completed."
|
||||||
|
|
||||||
write_log "Extracting firmware ${archive_file}."
|
write_log "Extracting firmware ${archive_file}."
|
||||||
set data [exec unzip -ql "${archive_file}"]
|
set data [exec /usr/bin/unzip -ql "${archive_file}"]
|
||||||
set img_file ""
|
set img_file ""
|
||||||
foreach d [split $data "\n"] {
|
foreach d [split $data "\n"] {
|
||||||
regexp {\s+(\S+\.img)\s*$} $d match img_file
|
regexp {\s+(\S+\.img)\s*$} $d match img_file
|
||||||
@ -328,7 +411,7 @@ proc ::rmupdate::download_firmware {version} {
|
|||||||
if { $img_file == "" } {
|
if { $img_file == "" } {
|
||||||
error "Failed to extract image from archive."
|
error "Failed to extract image from archive."
|
||||||
}
|
}
|
||||||
exec unzip "${archive_file}" "${img_file}" -o -d "${img_dir}"
|
exec /usr/bin/unzip "${archive_file}" "${img_file}" -o -d "${img_dir}"
|
||||||
set img_file "${img_dir}/${img_file}"
|
set img_file "${img_dir}/${img_file}"
|
||||||
puts "${img_file} ${image_file}"
|
puts "${img_file} ${image_file}"
|
||||||
if {$img_file != $image_file} {
|
if {$img_file != $image_file} {
|
||||||
@ -418,7 +501,7 @@ proc ::rmupdate::delete_firmware_image {version} {
|
|||||||
eval {file delete [glob "${img_dir}/*${version}.img"]}
|
eval {file delete [glob "${img_dir}/*${version}.img"]}
|
||||||
}
|
}
|
||||||
|
|
||||||
proc ::rmupdate::install_firmware_version {version {reboot 1}} {
|
proc ::rmupdate::install_firmware_version {version {reboot 1} {dryrun 0}} {
|
||||||
if {[rmupdate::install_process_running]} {
|
if {[rmupdate::install_process_running]} {
|
||||||
error "Another install process is running."
|
error "Another install process is running."
|
||||||
}
|
}
|
||||||
@ -447,13 +530,41 @@ proc ::rmupdate::install_firmware_version {version {reboot 1}} {
|
|||||||
}
|
}
|
||||||
|
|
||||||
check_sizes $firmware_image
|
check_sizes $firmware_image
|
||||||
update_filesystems $firmware_image
|
update_filesystems $firmware_image $dryrun
|
||||||
|
|
||||||
file delete $install_lock
|
file delete $install_lock
|
||||||
|
|
||||||
if {$reboot} {
|
if {$reboot} {
|
||||||
|
if { [file exist /lib.rmupdate] } {
|
||||||
|
write_log "Replacing /lib and rebooting system."
|
||||||
|
} else {
|
||||||
write_log "Rebooting system."
|
write_log "Rebooting system."
|
||||||
exec /bin/sh -c "/bin/sleep 5; /sbin/reboot -f" &
|
}
|
||||||
|
}
|
||||||
|
# Write success marker for web interface
|
||||||
|
write_log "INSTALL_FIRMWARE_SUCCESS"
|
||||||
|
after 3000
|
||||||
|
|
||||||
|
if {$reboot} {
|
||||||
|
if { [file exist /lib.rmupdate] } {
|
||||||
|
exec mount -o remount,rw /
|
||||||
|
exec rsync --archive --delete /lib.rmupdate/ /lib
|
||||||
|
|
||||||
|
set fd [open /proc/sys/kernel/sysrq "a"]
|
||||||
|
puts $fd "1"
|
||||||
|
close $fd
|
||||||
|
set fd [open /proc/sysrq-trigger "a"]
|
||||||
|
puts $fd "s"
|
||||||
|
close $fd
|
||||||
|
set fd [open /proc/sysrq-trigger "a"]
|
||||||
|
puts $fd "u"
|
||||||
|
close $fd
|
||||||
|
set fd [open /proc/sysrq-trigger "a"]
|
||||||
|
puts $fd "b"
|
||||||
|
close $fd
|
||||||
|
} else {
|
||||||
|
exec /bin/sh -c "sleep 5; reboot -f " &
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -83,8 +83,23 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function installation_finished() {
|
||||||
|
if (installation_running) {
|
||||||
|
var version = installation_running;
|
||||||
|
$('[data-install-version="' + version + '"]').removeClass('loading');
|
||||||
|
installation_running = "";
|
||||||
|
msg = 'Firmware ' + version + ' successfully installed.'
|
||||||
|
display_message('success', msg, 6000000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function update_install_log() {
|
function update_install_log() {
|
||||||
rest("GET", "/read_install_log", null, function(data) {
|
rest("GET", "/read_install_log", null, function(data) {
|
||||||
|
var regex_success = /INSTALL_FIRMWARE_SUCCESS\n/g;
|
||||||
|
if (regex_success.test(data)) {
|
||||||
|
installation_finished();
|
||||||
|
}
|
||||||
|
data = data.replace(regex_success, "");
|
||||||
$('#log-content').html(data.replace(/\n/g, '<br />'));
|
$('#log-content').html(data.replace(/\n/g, '<br />'));
|
||||||
$('#modal-log').modal('refresh');
|
$('#modal-log').modal('refresh');
|
||||||
$('#log-content').scrollTop($('#log-content').prop("scrollHeight"));
|
$('#log-content').scrollTop($('#log-content').prop("scrollHeight"));
|
||||||
@ -117,9 +132,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
clear_message();
|
clear_message();
|
||||||
rest("POST", "/start_install_firmware", JSON.stringify({"version":version, "reboot":reboot, "dryrun":dryrun}),
|
rest("POST", "/start_install_firmware", JSON.stringify({"version":version, "reboot":reboot, "dryrun":dryrun}),
|
||||||
function(data) {
|
function(data) {
|
||||||
$('[data-install-version="' + installation_running + '"]').removeClass('loading');
|
installation_finished();
|
||||||
installation_running = "";
|
|
||||||
display_message('success', 'Firmware ' + version + ' successfully installed.', 6000000);
|
|
||||||
if (!reboot) {
|
if (!reboot) {
|
||||||
get_firmware_info();
|
get_firmware_info();
|
||||||
}
|
}
|
||||||
@ -128,8 +141,8 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
$('[data-install-version="' + installation_running + '"]').removeClass('loading');
|
$('[data-install-version="' + installation_running + '"]').removeClass('loading');
|
||||||
installation_running = "";
|
installation_running = "";
|
||||||
//$('#modal-log').modal('hide');
|
//$('#modal-log').modal('hide');
|
||||||
if (!reboot) {
|
if (installation_running) {
|
||||||
get_firmware_info();
|
//get_firmware_info();
|
||||||
default_error_callback(xhr, ajaxOptions, thrownError);
|
default_error_callback(xhr, ajaxOptions, thrownError);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -75,7 +75,8 @@ proc process {} {
|
|||||||
} elseif {[lindex $path 1] == "delete_firmware_image"} {
|
} elseif {[lindex $path 1] == "delete_firmware_image"} {
|
||||||
regexp {\"version\"\s*:\s*\"([\d\.]+)\"} $data match version
|
regexp {\"version\"\s*:\s*\"([\d\.]+)\"} $data match version
|
||||||
if { [info exists version] && $version != "" } {
|
if { [info exists version] && $version != "" } {
|
||||||
return "\"[rmupdate::delete_firmware_image $version]\""
|
set res [rmupdate::delete_firmware_image $version]
|
||||||
|
return "\"${res}\""
|
||||||
} else {
|
} else {
|
||||||
error "Invalid version: ${data}"
|
error "Invalid version: ${data}"
|
||||||
}
|
}
|
||||||
|
11
rmupdate
11
rmupdate
@ -6,7 +6,16 @@ ADDON_DIR=/usr/local/addons/${ADDON_NAME}
|
|||||||
RCD_DIR=${CONFIG_DIR}/rc.d
|
RCD_DIR=${CONFIG_DIR}/rc.d
|
||||||
|
|
||||||
case "$1" in
|
case "$1" in
|
||||||
""|start|stop|restart)
|
""|start)
|
||||||
|
if [ -e /lib.rmupdate ]; then
|
||||||
|
mount -o remount,rw /
|
||||||
|
rm -r /lib.rmupdate
|
||||||
|
mount -o remount,ro /
|
||||||
|
fi
|
||||||
|
echo "Done."
|
||||||
|
;;
|
||||||
|
|
||||||
|
stop|restart)
|
||||||
echo "Nothing to do."
|
echo "Nothing to do."
|
||||||
;;
|
;;
|
||||||
|
|
||||||
|
BIN
rmupdate.tar.gz
BIN
rmupdate.tar.gz
Binary file not shown.
Loading…
Reference in New Issue
Block a user