source: lab/vendor/lxc/0.8.0~rc1-4ubuntu37/templates/lxc-archlinux @ 175

Last change on this file since 175 was 175, checked in by mitty, 12 years ago
  • /usr/lib/lxc/templates of lxc 0.8.0~rc1-4ubuntu37 on Ubuntu 12.10 (beta)
  • Property svn:executable set to *
File size: 13.5 KB
Line 
1#!/bin/bash
2
3#
4# template script for generating Arch linux container for LXC
5#
6
7#
8# lxc: linux Container library
9
10# Authors:
11# Alexander Vladimirov <idkfa@vlan1.ru>
12
13# This library is free software; you can redistribute it and/or
14# modify it under the terms of the GNU Lesser General Public
15# License as published by the Free Software Foundation; either
16# version 2.1 of the License, or (at your option) any later version.
17
18# This library is distributed in the hope that it will be useful,
19# but WITHOUT ANY WARRANTY; without even the implied warranty of
20# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
21# Lesser General Public License for more details.
22
23# You should have received a copy of the GNU Lesser General Public
24# License along with this library; if not, write to the Free Software
25# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26
27# defaults
28arch=$(arch)
29cache=/var/cache/lxc/arch/${arch}
30lxc_network_type="veth"
31lxc_network_link="br0"
32default_path=/var/lib/lxc
33default_rc_locale="en-US.UTF-8"
34default_rc_timezone="UTC"
35host_mirror="http://mirrors.kernel.org/archlinux/\$repo/os/$arch"
36
37# sort of minimal package set
38base_packages=(
39    "filesystem"
40    "initscripts"
41    "coreutils"
42    "module-init-tools"
43    "procps"
44    "psmisc"
45    "pacman"
46    "bash"
47    "syslog-ng"
48    "cronie"
49    "iproute2"
50    "iputils"
51    "inetutils"
52    "dhcpcd"
53    "dnsutils"
54    "nano"
55    "grep"
56    "less"
57    "gawk"
58    "sed"
59    "tar"
60    "wget"
61    "gzip"
62    "which"
63)
64declare -a additional_packages
65
66[ -f /etc/arch-release ] && is_arch=true
67
68# find and extract parameter value from given config file
69# ${1} - file to read parameter from
70# ${2} - parameter name
71# ${result} - result value on success
72function read_parameter_value {
73    [ -f ${1} ] && [ "${2}" ] || return 1
74    local pattern="^[[:space:]]*${2}[[:space:]]*=[[:space:]]*"
75    local str=$(grep "${pattern}" "${1}")
76    local str=${str/#$(grep -o "${pattern}" "${1}")/}
77    result=${str//\"/}
78    return 0
79}
80
81# split comma-separated string into an array
82# ${1} - string to split
83# ${2} - separator (default is ",")
84# ${result} - result value on success
85function split_string {
86    local ifs=${IFS}
87    IFS="${2:-,}"
88    read -a result < <(echo "${1}")
89    IFS=${ifs}
90    return 0
91}
92
93# Arch-specific preconfiguration for container
94function configure_arch {
95    # read locale and timezone defaults from system rc.conf if running on Arch
96    if [ "${is_arch}" ]; then
97        read_parameter_value "/etc/rc.conf" "LOCALE"
98        rc_locale=${result:-${default_rc_locale}}
99        read_parameter_value "/etc/rc.conf" "TIMEZONE"
100        rc_timezone=${result:-${default_rc_timezone}}
101    else
102        rc_locale=${default_rc_locale}
103        rc_timezone=${default_rc_timezone}
104    fi
105
106    echo "Setting up rc.conf"
107    cat > "${rootfs_path}/etc/rc.conf" << EOF
108# /etc/rc.conf - Main Configuration for Arch Linux
109LOCALE="${rc_locale}"
110DAEMON_LOCALE="no"
111HARDWARECLOCK="local"
112TIMEZONE="${rc_timezone}"
113KEYMAP=us
114CONSOLEFONT=
115CONSOLEMAP=
116USECOLOR="yes"
117MODULES=()
118HOSTNAME="${name}"
119interface=eth0
120address=
121netmask=
122broadcast=
123gateway=
124DAEMONS=(syslog-ng crond network)
125EOF
126
127    if [ -e "${rootfs_path}/etc/locale.gen" ]; then
128        sed -i 's@^#\(en_US\.UTF-8\)@\1@' "${rootfs_path}/etc/locale.gen"
129        if [ ! "${rc_locale}" = "en_US.UTF-8" ]; then
130            echo "${rc_locale} ${rc_locale##*.}" >> "${rootfs_path}/etc/locale.gen"
131        fi
132        chroot "${rootfs_path}" locale-gen
133    fi
134    cp "${rootfs_path}/usr/share/zoneinfo/${rc_timezone}" \
135       "${rootfs_path}/etc/localtime"
136
137    echo "Setting up rc.sysinit"
138    cat > "${rootfs_path}/etc/rc.sysinit.lxc" << EOF
139#!/bin/bash
140. /etc/rc.conf
141. /etc/rc.d/functions
142
143echo "starting Arch Linux"
144rm -f \$(find /var/run -name '*pid')
145rm -f /run/daemons/*
146rm -f /var/lock/subsys/*
147rm -f /etc/mtab
148touch /etc/mtab
149run_hook sysinit_end
150EOF
151
152    echo "Setting up rc.shutdown"
153    cat > "${rootfs_path}/etc/rc.shutdown.lxc" << EOF
154#!/bin/bash
155. /etc/rc.conf
156. /etc/rc.d/functions
157stty onlcr
158run_hook shutdown_start
159[[ -x /etc/rc.local.shutdown ]] && /etc/rc.local.shutdown
160stop_all_daemons
161run_hook shutdown_prekillall
162kill_all
163run_hook shutdown_postkillall
164[[ \${TIMEZONE} ]] && cp --remove-destination "/usr/share/zoneinfo/\${TIMEZONE}" /etc/localtime
165halt -w
166umount -a -r -t nodevtmpfs,notmpfs,nosysfs,noproc,nodevpts -O no_netdev
167run_hook shutdown_postumount
168run_hook shutdown_poweroff
169if [[ \${RUNLEVEL} = 0 ]]; then
170    poweroff -d -f -i
171else
172    reboot -d -f -i
173fi
174# vim: set ts=2 sw=2 noet:
175EOF
176    chmod 755 "${rootfs_path}/etc/rc.shutdown.lxc" "${rootfs_path}/etc/rc.sysinit.lxc"
177
178    echo "Setting up inittab"
179    cat > "${rootfs_path}/etc/inittab" << EOF
180id:3:initdefault:
181rc::sysinit:/etc/rc.sysinit.lxc
182rs:S1:wait:/etc/rc.single
183rm:2345:wait:/etc/rc.multi
184rh:06:wait:/etc/rc.shutdown.lxc
185su:S:wait:/sbin/sulogin -p
186c1:2345:respawn:/sbin/agetty -8 38400 tty1 linux
187EOF
188
189    echo "Setting up hosts"
190    cat > "${rootfs_path}/etc/hosts" << EOF
191127.0.0.1   localhost.localdomain   localhost ${name}
192::1     localhost.localdomain   localhost
193EOF
194
195    echo "Setting up nameserver"
196    grep nameserver /etc/resolv.conf > "${rootfs_path}/etc/resolv.conf"
197
198    echo "Setting up device nodes"
199    mkdir -m 755 "${rootfs_path}/dev/pts"
200    mkdir -m 1777 "${rootfs_path}/dev/shm"
201    mknod -m 666 "${rootfs_path}/dev/null" c 1 3
202    mknod -m 666 "${rootfs_path}/dev/full" c 1 7
203    mknod -m 666 "${rootfs_path}/dev/random" c 1 8
204    mknod -m 666 "${rootfs_path}/dev/urandom" c 1 9
205    mknod -m 666 "${rootfs_path}/dev/tty0" c 4 0
206    mknod -m 666 "${rootfs_path}/dev/tty1" c 4 1
207    mknod -m 666 "${rootfs_path}/dev/tty2" c 4 2
208    mknod -m 666 "${rootfs_path}/dev/tty3" c 4 3
209    mknod -m 666 "${rootfs_path}/dev/tty4" c 4 4
210    mknod -m 600 "${rootfs_path}/dev/initctl" p
211    mknod -m 666 "${rootfs_path}/dev/tty" c 5 0
212    mknod -m 666 "${rootfs_path}/dev/console" c 5 1
213    mknod -m 666 "${rootfs_path}/dev/ptmx" c 5 2
214
215    return 0
216}
217
218# write container configuration files
219function copy_configuration {
220    mkdir -p "${config_path}"
221    grep -q "^lxc.rootfs" "${config_path}/config" 2>/dev/null || echo "lxc.rootfs=${rootfs_path}" >> "${config_path}/config"
222    cat > "${config_path}/config" << EOF
223lxc.utsname=${name}
224lxc.tty=4
225lxc.pts=1024
226lxc.mount=${config_path}/fstab
227#networking
228lxc.network.type=${lxc_network_type}
229lxc.network.flags=up
230lxc.network.link=${lxc_network_link}
231lxc.network.name=eth0
232lxc.network.mtu=1500
233#cgroups
234lxc.cgroup.devices.deny = a
235# /dev/null and zero
236lxc.cgroup.devices.allow = c 1:3 rwm
237lxc.cgroup.devices.allow = c 1:5 rwm
238# consoles
239lxc.cgroup.devices.allow = c 5:1 rwm
240lxc.cgroup.devices.allow = c 5:0 rwm
241lxc.cgroup.devices.allow = c 4:0 rwm
242lxc.cgroup.devices.allow = c 4:1 rwm
243# /dev/{,u}random
244lxc.cgroup.devices.allow = c 1:9 rwm
245lxc.cgroup.devices.allow = c 1:8 rwm
246# /dev/pts
247lxc.cgroup.devices.allow = c 136:* rwm
248lxc.cgroup.devices.allow = c 5:2 rwm
249# rtc
250lxc.cgroup.devices.allow = c 254:0 rwm
251EOF
252
253    cat > "${config_path}/fstab" << EOF
254none ${rootfs_path}/dev/pts devpts defaults 0 0
255none ${rootfs_path}/proc proc nodev,noexec,nosuid 0 0
256none ${rootfs_path}/sys sysfs defaults 0 0
257none ${rootfs_path}/dev/shm tmpfs defaults 0 0
258EOF
259
260    if [ ${?} -ne 0 ]; then
261        echo "Failed to configure container"
262        return 1
263    fi
264
265    return 0
266}
267
268# lock chroot and mount subdirectories before installing container
269function mount_chroot {
270    echo "mounting chroot"
271    umask 0022
272    [ -e "${rootfs_path}/sys" ] || mkdir "${rootfs_path}/sys"
273    mount -t sysfs sysfs "${rootfs_path}/sys"
274    [ -e "${rootfs_path}/proc" ] || mkdir "${rootfs_path}/proc"
275    mount -t proc proc "${rootfs_path}/proc"
276    [ -e "${rootfs_path}/dev" ] || mkdir "${rootfs_path}/dev"
277    mount -t tmpfs dev "${rootfs_path}/dev" -o mode=0755,size=10M,nosuid
278    mknod -m 666 "${rootfs_path}/dev/null" c 1 3
279    mknod -m 666 "${rootfs_path}/dev/zero" c 1 5
280    mknod -m 600 "${rootfs_path}/dev/console" c 5 1
281    mknod -m 644 "${rootfs_path}/dev/random" c 1 8
282    mknod -m 644 "${rootfs_path}/dev/urandom" c 1 9
283    mknod -m 666 "${rootfs_path}/dev/tty" c 5 0
284    mknod -m 666 "${rootfs_path}/dev/tty0" c 4 0
285    mknod -m 666 "${rootfs_path}/dev/full" c 1 7
286    ln -s /proc/kcore "${rootfs_path}/dev/core"
287    ln -s /proc/self/fd "${rootfs_path}/dev/fd"
288    ln -s /proc/self/fd/0 "${rootfs_path}/dev/stdin"
289    ln -s /proc/self/fd/1 "${rootfs_path}/dev/stdout"
290    ln -s /proc/self/fd/2 "${rootfs_path}/dev/stderr"
291    [ -e "${rootfs_path}/dev/shm" ] || mkdir "${rootfs_path}/dev/shm"
292    mount -t tmpfs shm "${rootfs_path}/dev/shm" -o nodev,nosuid,size=128M
293    [ -e "${rootfs_path}/dev/pts" ] || mkdir "${rootfs_path}/dev/pts"
294    mount -t devpts devpts "${rootfs_path}/dev/pts" -o newinstance,ptmxmode=666
295    ln -s pts/ptmx "${rootfs_path}/dev/ptmx"
296    [ -e "${cache_dir}" ] || mkdir -p "${cache_dir}"
297    [ -e "${rootfs_path}/${cache_dir}" ] || mkdir -p "${rootfs_path}/${cache_dir}"
298    mount -o bind "${cache_dir}" "${rootfs_path}/${cache_dir}"
299    if [ -n "${host_mirror_path}" ]; then
300        [ -e "${rootfs_path}/${host_mirror_path}" ] || mkdir -p "${rootfs_path}/${host_mirror_path}"
301        mount -o bind "${host_mirror_path}" "${rootfs_path}/${host_mirror_path}"
302        mount -o remount,ro,bind "${host_mirror_path}" "${rootfs_path}/${host_mirror_path}"
303    fi
304    trap 'umount_chroot' EXIT INT QUIT TERM HUP
305}
306
307function umount_chroot {
308    if [ -z "${umount_done}" ]; then
309        echo "unmounting chroot"
310        umount "${rootfs_path}/proc"
311        umount "${rootfs_path}/sys"
312        umount "${rootfs_path}/dev/pts"
313        umount "${rootfs_path}/dev/shm"
314        umount "${rootfs_path}/dev"
315        umount "${rootfs_path}/${cache_dir}"
316        [ -n "${host_mirror_path}" ] && umount "${rootfs_path}/${host_mirror_path}"
317        umount_done=1
318    fi
319}
320
321# install packages within container chroot
322function install_arch {
323    pacman_config=$(mktemp)
324
325    cat <<EOF > "${pacman_config}"
326[options]
327HoldPkg      = pacman glibc
328SyncFirst    = pacman
329Architecture = auto
330#IgnorePkg    = udev
331[core]
332Include = /etc/pacman.d/mirrorlist
333Server = ${host_mirror}
334[extra]
335Include = /etc/pacman.d/mirrorlist
336Server = ${host_mirror}
337[community]
338Include = /etc/pacman.d/mirrorlist
339Server = ${host_mirror}
340EOF
341
342    mkdir -p "${rootfs_path}/var/lib/pacman/sync"
343    mkdir -p "${rootfs_path}/etc"
344
345    if echo "${host_mirror}" | grep -q 'file://'; then
346        host_mirror_path=$(echo "${host_mirror}" | sed -E 's#file://(/.*)/\$repo/os/\$arch#\1#g')
347    fi
348    cache_dir=$( (grep -m 1 '^CacheDir' "${pacman_config}" || echo 'CacheDir = /var/cache/pacman/pkg') | sed 's/CacheDir\s*=\s*//')
349    mount_chroot
350    params="--root ${rootfs_path} --config=${pacman_config} --noconfirm"
351    if ! pacman -Sydd ${params} --dbonly udev; then
352        echo "Failed to preinstall udev package record"
353        return 1
354    fi
355    if ! pacman -S ${params} ${base_packages[@]}; then
356        echo "Failed to install container packages"
357        return 1
358    fi
359    [ -d "${rootfs_path}/lib/modules" ] && ldconfig -r "${rootfs_path}"
360    mv "${pacman_config}" "${rootfs_path}/etc/pacman.conf"
361    umount_chroot
362    return 0
363}
364
365usage()
366{
367    cat <<EOF
368usage:
369    ${1} -n|--name=<container_name>
370        [-P|--packages=<pkg1,pkg2,...>] [-p|--path=<path>] [-h|--help]
371Mandatory args:
372  -n,--name         container name, used to as an identifier for that container from now on
373Optional args:
374  -p,--path         path to where the container rootfs will be created, defaults to /var/lib/lxc. The container config will go under /var/lib/lxc in that case
375  -P,--packages     preinstall additional packages, comma-separated list
376  -h,--help         print this help
377EOF
378    return 0
379}
380
381options=$(getopt -o hp:P:n:cm: -l help,path:,packages:,name:,clean,mirror: -- "${@}")
382if [ ${?} -ne 0 ]; then
383    usage $(basename ${0})
384    exit 1
385fi
386eval set -- "${options}"
387
388while true
389do
390    case "${1}" in
391    -h|--help)      usage ${0} && exit 0;;
392    -p|--path)      path=${2}; shift 2;;
393    -n|--name)      name=${2}; shift 2;;
394    -P|--packages)  additional_packages=${2}; shift 2;;
395    -m|--mirror)    host_mirror=${2}; shift 2;;
396    --)             shift 1; break ;;
397    *)              break ;;
398    esac
399done
400
401if [ -z "${name}" ]; then
402    echo "missing required 'name' parameter"
403    exit 1
404fi
405
406type pacman >/dev/null 2>&1
407if [ ${?} -ne 0 ]; then
408    echo "'pacman' command is missing, refer to wiki.archlinux.org for information about installing pacman"
409    exit 1
410fi
411
412if [ -z "${path}" ]; then
413    path="${default_path}/${name}"
414fi
415
416if [ "${EUID}" != "0" ]; then
417    echo "This script should be run as 'root'"
418    exit 1
419fi
420
421rootfs_path="${path}/rootfs"
422# check for 'lxc.rootfs' passed in through default config by lxc-create
423if grep -q '^lxc.rootfs' $path/config 2>/dev/null ; then
424    rootfs_path=`grep 'lxc.rootfs =' $path/config | awk -F= '{ print $2 }'`
425fi
426config_path="${default_path}/${name}"
427
428revert()
429{
430    echo "Interrupted, so cleaning up"
431    lxc-destroy -n "${name}"
432    # maybe was interrupted before copy config
433    rm -rf "${path}/${name}"
434    rm -rf "${default_path}/${name}"
435    exit 1
436}
437
438trap revert SIGHUP SIGINT SIGTERM
439
440copy_configuration
441if [ ${?} -ne 0 ]; then
442    echo "failed write configuration file"
443    rm -rf "${config_path}"
444    exit 1
445fi
446
447if [ ${#additional_packages[@]} -gt 0 ]; then
448    split_string ${additional_packages}
449    base_packages+=(${result[@]})
450fi
451
452install_arch
453if [ ${?} -ne 0 ]; then
454    echo "failed to install Arch linux"
455    rm -rf "${config_path}" "${path}"
456    exit 1
457fi
458
459configure_arch
460if [ ${?} -ne 0 ]; then
461    echo "failed to configure Arch linux for a container"
462    rm -rf "${config_path}" "${path}"
463    exit 1
464fi
465
466echo "container rootfs and config created"
Note: See TracBrowser for help on using the repository browser.