initial. broken because it expects UUID support from mount which is not compiled in by default
This commit is contained in:
parent
f4d8e7bc14
commit
84bf08c3dc
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
build/
|
@ -1,4 +1,4 @@
|
||||
openwrt-auto-extroot
|
||||
====================
|
||||
* This is more of a template than something standalone
|
||||
|
||||
Use OpenWRT ImageBuilder to build a firmware that automatically formats and moves extroot to any inserted pendrive
|
||||
I've extracted this from another project, but I think it's useful
|
||||
enough for making it public.
|
||||
|
46
build.sh
Executable file
46
build.sh
Executable file
@ -0,0 +1,46 @@
|
||||
#!/bin/bash
|
||||
|
||||
absolutize ()
|
||||
{
|
||||
if [ ! -d "$1" ]; then
|
||||
echo
|
||||
echo "ERROR: '$1' doesn't exist or not a directory!"
|
||||
exit -1
|
||||
fi
|
||||
|
||||
pushd "$1" >/dev/null
|
||||
echo `pwd`
|
||||
popd >/dev/null
|
||||
}
|
||||
|
||||
TARGET_PLATFORM=$1
|
||||
BUILD=`dirname "$0"`"/build/"
|
||||
BUILD=`absolutize $BUILD`
|
||||
IMGTEMPDIR="${BUILD}/openwrt-build-image-extras"
|
||||
IMGBUILDERDIR="${BUILD}/OpenWrt-ImageBuilder-ar71xx_generic-for-linux-x86_64"
|
||||
IMGBUILDERURL="https://downloads.openwrt.org/barrier_breaker/14.07/ar71xx/generic/OpenWrt-ImageBuilder-ar71xx_generic-for-linux-x86_64.tar.bz2"
|
||||
|
||||
mkdir --parents ${BUILD}
|
||||
|
||||
rm -rf $IMGTEMPDIR
|
||||
cp -r image-extras $IMGTEMPDIR
|
||||
if [ -e image-extras.$TARGET_PLATFORM/ ]; then
|
||||
rsync -pr image-extras.$TARGET_PLATFORM/ $IMGTEMPDIR/
|
||||
fi
|
||||
|
||||
if [ ! -e ${IMGBUILDERDIR} ]; then
|
||||
pushd ${BUILD}
|
||||
wget --continue ${IMGBUILDERURL}
|
||||
tar jvxf OpenWrt-ImageBuilder*.tar.bz2
|
||||
popd
|
||||
fi
|
||||
|
||||
pushd ${IMGBUILDERDIR}
|
||||
|
||||
make image PROFILE=$TARGET_PLATFORM PACKAGES="wireless-tools firewall iptables fdisk blkid block-mount e2fsprogs kmod-fs-ext4 kmod-usb2 kmod-usb-uhci kmod-usb-ohci kmod-usb-storage kmod-usb-storage-extras luci kmod-mmc mount-utils ppp ppp-mod-pppoe ppp-mod-pppol2tp ppp-mod-pptp kmod-ppp kmod-pppoe" FILES=${IMGTEMPDIR}
|
||||
|
||||
pushd bin/ar71xx/
|
||||
ln -s ../../packages .
|
||||
popd
|
||||
|
||||
popd
|
11
image-extras.TLMR3020/etc/config/network
Normal file
11
image-extras.TLMR3020/etc/config/network
Normal file
@ -0,0 +1,11 @@
|
||||
|
||||
config interface 'loopback'
|
||||
option ifname 'lo'
|
||||
option proto 'static'
|
||||
option ipaddr '127.0.0.1'
|
||||
option netmask '255.0.0.0'
|
||||
|
||||
config interface 'lan'
|
||||
option ifname 'eth0'
|
||||
option type 'bridge'
|
||||
option proto 'dhcp'
|
27
image-extras/etc/config/fstab
Normal file
27
image-extras/etc/config/fstab
Normal file
@ -0,0 +1,27 @@
|
||||
config global 'automount'
|
||||
option from_fstab '0'
|
||||
option anon_mount '0'
|
||||
|
||||
config global 'autoswap'
|
||||
option from_fstab '0'
|
||||
option anon_swap '0'
|
||||
|
||||
config swap
|
||||
option uuid '05d615b3-bef8-460c-9a23-52db8d09e002'
|
||||
option enabled '1'
|
||||
|
||||
config mount
|
||||
option target '/overlay'
|
||||
option uuid '05d615b3-bef8-460c-9a23-52db8d09e000'
|
||||
option fstype 'ext4'
|
||||
option options 'rw,noatime'
|
||||
option enabled '1'
|
||||
option enabled_fsck '0'
|
||||
|
||||
config mount
|
||||
option target '/mnt/data'
|
||||
option uuid '05d615b3-bef8-460c-9a23-52db8d09e001'
|
||||
option fstype 'ext4'
|
||||
option options 'rw,noatime'
|
||||
option enabled '1'
|
||||
option enabled_fsck '0'
|
1
image-extras/etc/dropbear/authorized_keys
Normal file
1
image-extras/etc/dropbear/authorized_keys
Normal file
@ -0,0 +1 @@
|
||||
ssh-rsa AAAAB3NzaC1yc2EAAAABIwAAAQEAu9Nwb8tr91hvChHSjEbyS3P0c1+jKtKAdaFCRkyUjVwgCpuzDxiq0auuNulYIfD2oc+THJ6zymJUjWNrVipeUo8BmKkDSMgN0Qf5PlwcSiIj9vDbLqxmVnnvB6xGEROO215Y8XzMOgq8r3Z3WqRUZIeFDHC2sSwJKO3INgsLZd6IoDiM7Dza8pKzYPfY7jJ19JmK4S8lHG3YsoxTy2zkcwCI20sBekJU0iDGvOOJq5UbIumKsAm2uJkMKsKlxkDQr0Y+2J1l0iWBrUHonja6CieO5yNBWluA3DCqxa0pQW3dcOju3mGCQl0j8+3Iblu8lCGoQVSLQ3rUhekmz+cB2Q== alendvai laptop ssh key
|
10
image-extras/etc/logrotate.d/syslog.conf
Normal file
10
image-extras/etc/logrotate.d/syslog.conf
Normal file
@ -0,0 +1,10 @@
|
||||
/var/log/syslog
|
||||
{
|
||||
olddir archive
|
||||
rotate 30
|
||||
daily
|
||||
dateext
|
||||
notifempty
|
||||
missingok
|
||||
copytruncate
|
||||
}
|
6
image-extras/etc/rc.local
Normal file
6
image-extras/etc/rc.local
Normal file
@ -0,0 +1,6 @@
|
||||
# Put your custom commands here that should be executed once
|
||||
# the system init finished. By default this file does nothing.
|
||||
|
||||
/root/autoprovision-stage1.sh
|
||||
|
||||
exit 0
|
93
image-extras/root/autoprovision-functions.sh
Executable file
93
image-extras/root/autoprovision-functions.sh
Executable file
@ -0,0 +1,93 @@
|
||||
#!/bin/sh
|
||||
|
||||
# utility functions for the various stages of autoprovisioning
|
||||
|
||||
# these are also copy-pasted into other scripts and config files!
|
||||
rootUUID=05d615b3-bef8-460c-9a23-52db8d09e000
|
||||
dataUUID=05d615b3-bef8-460c-9a23-52db8d09e001
|
||||
swapUUID=05d615b3-bef8-460c-9a23-52db8d09e002
|
||||
|
||||
. /lib/ar71xx.sh
|
||||
|
||||
# let's try some defaults...
|
||||
autoprovisionUSBLed="tp-link:green:usb"
|
||||
autoprovisionStatusLed="tp-link:green:qss"
|
||||
|
||||
case $(ar71xx_board_name) in
|
||||
"tl-wr1043nd")
|
||||
autoprovisionUSBLed="tp-link:green:usb"
|
||||
autoprovisionStatusLed="tp-link:green:qss"
|
||||
;;
|
||||
"tl-mr3020")
|
||||
autoprovisionUSBLed="tp-link:green:wps"
|
||||
autoprovisionStatusLed="tp-link:green:wlan"
|
||||
;;
|
||||
"tl-wr2543n")
|
||||
autoprovisionUSBLed="tp-link:green:wps"
|
||||
autoprovisionStatusLed="tp-link:green:wlan5g"
|
||||
;;
|
||||
"tl-wdr4300")
|
||||
autoprovisionUSBLed="tp-link:green:usb1"
|
||||
autoprovisionStatusLed="tp-link:blue:qss"
|
||||
;;
|
||||
esac
|
||||
|
||||
log()
|
||||
{
|
||||
/usr/bin/logger -t autoprov -s $*
|
||||
}
|
||||
|
||||
setLedAttribute()
|
||||
{
|
||||
[ -f "/sys/class/leds/$1/$2" ] && echo "$3" > "/sys/class/leds/$1/$2"
|
||||
}
|
||||
|
||||
signalAutoprovisionWorking()
|
||||
{
|
||||
setLedAttribute ${autoprovisionStatusLed} trigger none
|
||||
setLedAttribute ${autoprovisionStatusLed} trigger timer
|
||||
setLedAttribute ${autoprovisionStatusLed} delay_on 2000
|
||||
setLedAttribute ${autoprovisionStatusLed} delay_off 2000
|
||||
}
|
||||
|
||||
signalAutoprovisionWaitingForUser()
|
||||
{
|
||||
setLedAttribute ${autoprovisionStatusLed} trigger none
|
||||
setLedAttribute ${autoprovisionStatusLed} trigger timer
|
||||
setLedAttribute ${autoprovisionStatusLed} delay_on 200
|
||||
setLedAttribute ${autoprovisionStatusLed} delay_off 300
|
||||
}
|
||||
|
||||
signalWaitingForPendrive()
|
||||
{
|
||||
setLedAttribute ${autoprovisionUSBLed} trigger none
|
||||
setLedAttribute ${autoprovisionUSBLed} trigger timer
|
||||
setLedAttribute ${autoprovisionUSBLed} delay_on 200
|
||||
setLedAttribute ${autoprovisionUSBLed} delay_off 300
|
||||
}
|
||||
|
||||
signalFormatting()
|
||||
{
|
||||
setLedAttribute ${autoprovisionUSBLed} trigger none
|
||||
setLedAttribute ${autoprovisionUSBLed} trigger timer
|
||||
setLedAttribute ${autoprovisionUSBLed} delay_on 1000
|
||||
setLedAttribute ${autoprovisionUSBLed} delay_off 1000
|
||||
}
|
||||
|
||||
stopSignallingAnything()
|
||||
{
|
||||
setLedAttribute ${autoprovisionStatusLed} trigger none
|
||||
setLedAttribute ${autoprovisionUSBLed} trigger usbdev
|
||||
}
|
||||
|
||||
setRootPassword()
|
||||
{
|
||||
local password=$1
|
||||
if [ "$password" == "" ]; then
|
||||
# set and forget a random password merely to disable telnet. login will go through ssh keys.
|
||||
password=$(</dev/urandom sed 's/[^A-Za-z0-9+_]//g' | head -c 22)
|
||||
fi
|
||||
#echo "Setting root password to '"$password"'"
|
||||
log "Setting root password"
|
||||
echo -e "$password\n$password\n" | passwd root
|
||||
}
|
142
image-extras/root/autoprovision-stage1.sh
Executable file
142
image-extras/root/autoprovision-stage1.sh
Executable file
@ -0,0 +1,142 @@
|
||||
#!/bin/sh
|
||||
|
||||
# autoprovision stage 1: this script will be executed upon boot without a valid extroot (i.e. when rc.local is found and run from the internal overlay)
|
||||
|
||||
. /root/autoprovision-functions.sh
|
||||
|
||||
getPendriveSize()
|
||||
{
|
||||
# this is needed for the mmc card in some (all?) Huawei 3G dongle.
|
||||
# details: https://dev.openwrt.org/ticket/10716#comment:4
|
||||
if [ -e /dev/sda ]; then
|
||||
# force re-read of the partition table
|
||||
head /dev/sda >/dev/null
|
||||
fi
|
||||
|
||||
if (grep -q sda /proc/partitions) then
|
||||
cat /sys/block/sda/size
|
||||
else
|
||||
echo 0
|
||||
fi
|
||||
}
|
||||
|
||||
hasBigEnoughPendrive()
|
||||
{
|
||||
local size=$(getPendriveSize)
|
||||
if [ $size -ge 2000000 ]; then
|
||||
log "Found a pendrive of size: $(($size / 2 / 1024)) MB"
|
||||
return 0
|
||||
else
|
||||
return 1
|
||||
fi
|
||||
}
|
||||
|
||||
setupPendrivePartitions()
|
||||
{
|
||||
# erase partition table
|
||||
dd if=/dev/zero of=/dev/sda bs=1M count=1
|
||||
|
||||
# first is 'swap'
|
||||
# second is 'root'
|
||||
# the rest is 'data'
|
||||
fdisk /dev/sda <<EOF
|
||||
o
|
||||
n
|
||||
p
|
||||
1
|
||||
|
||||
+64M
|
||||
n
|
||||
p
|
||||
2
|
||||
|
||||
+512M
|
||||
n
|
||||
p
|
||||
3
|
||||
|
||||
|
||||
t
|
||||
1
|
||||
82
|
||||
w
|
||||
q
|
||||
EOF
|
||||
log "Finished partitioning /dev/sda using fdisk"
|
||||
|
||||
until [ -e /dev/sda1 ]
|
||||
do
|
||||
echo "Waiting for a partitions to show up in /dev"
|
||||
sleep 1
|
||||
done
|
||||
|
||||
# just to be sure we wait a bit more (i've seen once that mkswap worked on /dev/sda1, but then mkfs errored that there's no /dev/sda2 (?!))
|
||||
sleep 3
|
||||
|
||||
mkswap -L swap -U $swapUUID /dev/sda1
|
||||
mkfs.ext4 -L root -U $rootUUID /dev/sda2
|
||||
mkfs.ext4 -L data -U $dataUUID /dev/sda3
|
||||
|
||||
log "Finished setting up filesystems"
|
||||
}
|
||||
|
||||
setupExtroot()
|
||||
{
|
||||
mkdir -p /mnt/extroot
|
||||
# we need to make the internal overlay read-only, otherwise the two md5's may be different
|
||||
# due to writing to the internal overlay from this point until the reboot.
|
||||
# files: /.extroot.md5sum (extroot) and /etc/extroot.md5sum (internal)
|
||||
mount -o remount,ro /
|
||||
|
||||
log "Remounted / as read-only"
|
||||
|
||||
mount UUID=$rootUUID /mnt/extroot
|
||||
tar -C /overlay -cvf - . | tar -C /mnt/extroot -xf -
|
||||
|
||||
# let's write a new rc.local on extroot which will shadow the one which is in the rom and runs stage1
|
||||
cat >/mnt/extroot/etc/rc.local <<EOF
|
||||
/root/autoprovision-stage2.sh
|
||||
exit 0
|
||||
EOF
|
||||
|
||||
# make sure that we shadow the /var -> /tmp symlink with the extroot, so that /var is permanent
|
||||
mkdir -p /mnt/extroot/var
|
||||
# KLUDGE: but /var/state is assumed to be transient, see https://dev.openwrt.org/ticket/12228
|
||||
cd /mnt/extroot/var
|
||||
ln -s /tmp state
|
||||
cd -
|
||||
|
||||
log "Finished setting up extroot"
|
||||
}
|
||||
|
||||
autoprovisionStage1()
|
||||
{
|
||||
signalAutoprovisionWorking
|
||||
|
||||
# this way it will set a random password and only ssh key based login will work
|
||||
setRootPassword ""
|
||||
|
||||
signalAutoprovisionWaitingForUser
|
||||
signalWaitingForPendrive
|
||||
|
||||
until hasBigEnoughPendrive
|
||||
do
|
||||
echo "Waiting for a pendrive to be inserted"
|
||||
sleep 3
|
||||
done
|
||||
|
||||
signalAutoprovisionWorking # to make it flash in sync with the USB led
|
||||
signalFormatting
|
||||
|
||||
sleep 1
|
||||
|
||||
setupPendrivePartitions
|
||||
sleep 1
|
||||
setupExtroot
|
||||
|
||||
sync
|
||||
stopSignallingAnything
|
||||
reboot
|
||||
}
|
||||
|
||||
autoprovisionStage1
|
72
image-extras/root/autoprovision-stage2.sh
Executable file
72
image-extras/root/autoprovision-stage2.sh
Executable file
@ -0,0 +1,72 @@
|
||||
#!/bin/sh
|
||||
|
||||
# autoprovision stage 2: this script will be executed upon boot if the extroot was successfully mounted (i.e. rc.local is run from the extroot overlay)
|
||||
|
||||
. /root/autoprovision-functions.sh
|
||||
|
||||
installPackages()
|
||||
{
|
||||
signalAutoprovisionWaitingForUser
|
||||
|
||||
until (opkg update)
|
||||
do
|
||||
log "opkg update failed. No internet connection? Retrying in 15 seconds..."
|
||||
sleep 15
|
||||
done
|
||||
|
||||
signalAutoprovisionWorking
|
||||
|
||||
log "Autoprovisioning stage2 is about to install packages"
|
||||
|
||||
# switch ssh from dropbear to openssh (needed to install sshtunnel)
|
||||
#opkg remove dropbear
|
||||
#opkg install openssh-server openssh-sftp-server sshtunnel
|
||||
|
||||
#/etc/init.d/sshd enable
|
||||
#mkdir /root/.ssh
|
||||
#chmod 0700 /root/.ssh
|
||||
#mv /etc/dropbear/authorized_keys /root/.ssh/
|
||||
#rm -rf /etc/dropbear
|
||||
|
||||
# install some more packages that don't need any extra steps
|
||||
opkg install lua luci ppp-mod-pppoe screen mc zip unzip logrotate
|
||||
|
||||
# this is needed for the vlans on tp-link 3020 with only a single hw ethernet port
|
||||
opkg install kmod-macvlan ip
|
||||
|
||||
# just in case if we were run in a firmware that didn't already had luci
|
||||
/etc/init.d/uhttpd enable
|
||||
}
|
||||
|
||||
autoprovisionStage2()
|
||||
{
|
||||
log "Autoprovisioning stage2 speaking"
|
||||
signalAutoprovisionWorking
|
||||
|
||||
# it's not the nicest way to test whether stage2 has been done already, but this is a shell script...
|
||||
if [ $(uci get system.@system[0].log_type) == "file" ]; then
|
||||
log "Seems like autoprovisioning stage2 has been done already. Running stage3."
|
||||
#/root/autoprovision-stage3.py
|
||||
else
|
||||
installPackages
|
||||
|
||||
crontab - <<EOF
|
||||
# */10 * * * * /root/autoprovision-stage3.py
|
||||
0 0 * * * /usr/sbin/logrotate /etc/logrotate.conf
|
||||
EOF
|
||||
|
||||
mkdir -p /var/log/archive
|
||||
|
||||
uci set system.@system[0].log_type=file
|
||||
uci set system.@system[0].log_file=/var/log/syslog
|
||||
uci set system.@system[0].log_size=0
|
||||
|
||||
uci commit
|
||||
sync
|
||||
reboot
|
||||
fi
|
||||
|
||||
stopSignallingAnything
|
||||
}
|
||||
|
||||
autoprovisionStage2
|
Loading…
x
Reference in New Issue
Block a user