mkarchiso 17.7 KB
Newer Older
1
#!/bin/bash
2

3
4
set -e -u

5
6
export LANG=C

7
8
9
app_name=${0##*/}
arch=$(uname -m)
pkg_list=""
10
run_cmd=""
11
12
13
14
15
16
quiet="y"
pacman_conf="/etc/pacman.conf"
export iso_label="ARCH_$(date +%Y%m)"
iso_publisher="Arch Linux <http://www.archlinux.org>"
iso_application="Arch Linux Live/Rescue CD"
install_dir="arch"
17
18
work_dir="work"
out_dir="out"
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49

# Show an INFO message
# $1: message string
_msg_info() {
    local _msg="${1}"
    echo "[mkarchiso] INFO: ${_msg}"
}

# Show an ERROR message then exit with status
# $1: message string
# $2: exit code number (with 0 does not exit)
_msg_error() {
    local _msg="${1}"
    local _error=${2}
    echo
    echo "[mkarchiso] ERROR: ${_msg}"
    echo
    if [[ ${_error} -gt 0 ]]; then
        exit ${_error}
    fi
}

# Show space usage similar to df, but better formatted.
# $1: mount-point or mounted device.
_show_space_usage () {
    local _where="${1}"
    local _fs _total _used _avail _pct_u=0 _mnt
    read _fs _total _used _avail _pct_u _mnt < <(df -m "${_where}" | tail -1) &> /dev/null
    _msg_info "Total: ${_total} MiB (100%) | Used: ${_used} MiB (${_pct_u}) | Avail: ${_avail} MiB ($((100 - ${_pct_u%\%}))%)"
}

50
_chroot_init() {
51
52
53
    if [[ -f "${work_dir}/mkarchiso.init" ]]; then
        _msg_info "Initial enviroment already installed, skipping."
    else
54
55
        mkdir -p ${work_dir}/root-image
        _pacman "base syslinux"
56
        : > "${work_dir}/mkarchiso.init"
57
58
59
60
    fi
}

_chroot_run() {
61
    eval arch-chroot ${work_dir}/root-image "${run_cmd}"
62
63
}

64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
# Mount a filesystem (trap signals in case of error for unmounting it
# $1: source image
# $2: mount-point
_mount_fs() {
    local _src="${1}"
    local _dst="${2}"
    trap "_umount_fs ${_src}" EXIT HUP INT TERM
    mkdir -p "${_dst}"
    _msg_info "Mounting '${_src}' on '${_dst}'"
    mount "${_src}" "${_dst}"
    _show_space_usage "${_dst}"
}

# Unmount a filesystem (and untrap signals)
# $1: mount-point or device/image
_umount_fs() {
    local _dst="${1}"
    _show_space_usage "${_dst}"
    _msg_info "Unmounting '${_dst}'"
    umount "${_dst}"
    rmdir "${_dst}"
    trap - EXIT HUP INT TERM
}

# Compare if a file/directory (source) is newer than other file (target)
# $1: source file/directory
# $2: target file
# return: 0 if target does not exists or if target is older than source.
#         1 if target is newer than source
_is_directory_changed() {
    local _src="${1}"
    local _dst="${2}"

    if [ -e "${_dst}" ]; then
        if [[ $(find ${_src} -newer ${_dst} | wc -l) -gt 0 ]]; then
            _msg_info "Target '${_dst}' is older than '${_src}', updating."
            rm -f "${_dst}"
            return 0
        else
            _msg_info "Target '${_dst}' is up to date with '${_src}', skipping."
            return 1
        fi
    else
        _msg_info "Target '${_dst}' does not exist, making it from '${_src}'"
        return 0
    fi
}

# Show help usage, with an exit status.
# $1: exit status number.
_usage ()
115
{
116
    echo "usage ${app_name} [options] command <command options>"
117
    echo " general options:"
118
    echo "    -p PACKAGE(S)    Package(s) to install, can be used multiple times"
119
    echo "    -r <command>     Run <command> inside root-image"
120
121
    echo "    -C <file>        Config file for pacman."
    echo "                     Default: '${pacman_conf}'"
122
    echo "    -L <label>       Set a label for the disk"
123
    echo "                     Default: '${iso_label}'"
124
    echo "    -P <publisher>   Set a publisher for the disk"
125
    echo "                     Default: '${iso_publisher}'"
126
    echo "    -A <application> Set an application name for the disk"
127
    echo "                     Default: '${iso_application}'"
128
    echo "    -D <install_dir> Set an install_dir. All files will by located here."
129
    echo "                     Default: '${install_dir}'"
130
    echo "                     NOTE: Max 8 characters, use only [a-z0-9]"
131
    echo "    -w <work_dir>    Set the working directory"
132
    echo "                     Default: '${work_dir}'"
133
    echo "    -o <out_dir>     Set the output directory"
134
    echo "                     Default: '${out_dir}'"
135
136
    echo "    -v               Enable verbose output"
    echo "    -h               This message"
137
    echo " commands:"
138
139
140
141
142
143
    echo "   init"
    echo "      Make base layout and install base group"
    echo "   install"
    echo "      Install all specified packages (-p)"
    echo "   run"
    echo "      run command specified by -r"
144
    echo "   prepare"
145
    echo "      build all images"
146
    echo "   checksum"
147
    echo "      make a checksum.md5 for self-test"
148
149
    echo "   pkglist"
    echo "      make a pkglist.txt of packages installed on root-image"
150
    echo "   iso <image name>"
151
    echo "      build an iso image from the working dir"
152
    exit ${1}
153
154
}

155
# Shows configuration according to command mode.
156
# $1: init | install | run | prepare | checksum | iso
157
158
159
160
161
162
163
164
165
_show_config () {
    local _mode="$1"
    echo
    _msg_info "Configuration settings"
    _msg_info "                  Command:   ${command_name}"
    _msg_info "             Architecture:   ${arch}"
    _msg_info "        Working directory:   ${work_dir}"
    _msg_info "   Installation directory:   ${install_dir}"
    case "${_mode}" in
166
167
168
169
        init)
            _msg_info "       Pacman config file:   ${pacman_conf}"
            ;;
        install)
170
171
172
            _msg_info "       Pacman config file:   ${pacman_conf}"
            _msg_info "                 Packages:   ${pkg_list}"
            ;;
173
174
175
        run)
            _msg_info "              Run command:   ${run_cmd}"
            ;;
176
177
        prepare)
            ;;
178
179
        checksum)
            ;;
180
181
        pkglist)
            ;;
182
183
184
185
186
187
        iso)
            _msg_info "               Image name:   ${img_name}"
            _msg_info "               Disk label:   ${iso_label}"
            _msg_info "           Disk publisher:   ${iso_publisher}"
            _msg_info "         Disk application:   ${iso_application}"
            ;;
188
    esac
189
190
    echo
}
191

192
# Install desired packages to root-image
193
194
_pacman ()
{
195
    _msg_info "Installing packages to '${work_dir}/root-image/'..."
196
197

    if [[ "${quiet}" = "y" ]]; then
198
        pacstrap -C "${pacman_conf}" -c -d -G -M "${work_dir}/root-image" $* &> /dev/null
199
    else
200
        pacstrap -C "${pacman_conf}" -c -d -G -M "${work_dir}/root-image" $*
201
    fi
202

203
    _msg_info "Packages installed successfully!"
204
205
}

206
207
# Cleanup root-image
_cleanup () {
208
    _msg_info "Cleaning up what we can on root-image..."
209

210
    # Delete initcpio image(s)
211
    if [[ -d "${work_dir}/root-image/boot" ]]; then
212
213
214
215
216
        find "${work_dir}/root-image/boot" -type f -name '*.img' -delete
    fi
    # Delete kernel(s)
    if [[ -d "${work_dir}/root-image/boot" ]]; then
        find "${work_dir}/root-image/boot" -type f -name 'vmlinuz*' -delete
217
218
219
    fi
    # Delete pacman database sync cache files (*.tar.gz)
    if [[ -d "${work_dir}/root-image/var/lib/pacman" ]]; then
220
        find "${work_dir}/root-image/var/lib/pacman" -maxdepth 1 -type f -delete
221
222
223
    fi
    # Delete pacman database sync cache
    if [[ -d "${work_dir}/root-image/var/lib/pacman/sync" ]]; then
224
        find "${work_dir}/root-image/var/lib/pacman/sync" -delete
225
226
227
    fi
    # Delete pacman package cache
    if [[ -d "${work_dir}/root-image/var/cache/pacman/pkg" ]]; then
228
        find "${work_dir}/root-image/var/cache/pacman/pkg" -type f -delete
229
230
231
    fi
    # Delete all log files, keeps empty dirs.
    if [[ -d "${work_dir}/root-image/var/log" ]]; then
232
        find "${work_dir}/root-image/var/log" -type f -delete
233
234
235
    fi
    # Delete all temporary files and dirs
    if [[ -d "${work_dir}/root-image/var/tmp" ]]; then
236
        find "${work_dir}/root-image/var/tmp" -mindepth 1 -delete
237
    fi
238
    # Delete package pacman related files.
239
    find "${work_dir}" \( -name "*.pacnew" -o -name "*.pacsave" -o -name "*.pacorig" \) -delete
240
    _msg_info "Done!"
241
}
242

243
244
245
246
247
248
# Makes a SquashFS filesystem image of file/directory passes as argument with desired compression.
# $1: Source file/directory
# $2: SquashFS compression type (gzip | lzo | xz)
_mksfs () {
    local _src="${1}"
    local _sfs_comp="${2}"
249

250
251
    if [[ ! -e "${work_dir}/${_src}" ]]; then
        _msg_error "The path '${work_dir}/${_src}' does not exist" 1
252
253
    fi

254
255
256
257
258
    local _sfs_img="${work_dir}/${_src}.sfs"

    _msg_info "Creating SquashFS image for '${work_dir}/${_src}', This may take some time..."
    local _seconds=${SECONDS}
    if [[ "${quiet}" = "y" ]]; then
259
        mksquashfs "${work_dir}/${_src}" "${_sfs_img}" -noappend -comp "${_sfs_comp}" -no-progress &> /dev/null
260
    else
261
        mksquashfs "${work_dir}/${_src}" "${_sfs_img}" -noappend -comp "${_sfs_comp}" -no-progress
262
    fi
263
264
    _seconds=$((SECONDS - _seconds))
    printf "[mkarchiso] INFO: Image creation done in %02d:%02d minutes\n" $((_seconds / 60)) $((_seconds % 60))
265
}
266

267
268
# Makes a filesystem from a source directory.
# $1: Source directory
269
# $2: Target filesystem type (ext4 | ext3 | ext2 | xfs | btrfs)
270
271
272
273
274
# $3: Size of target filesystem. Can be an absolute value in MiB, or relative value of desired free space (1% - 99%)
_mkfs () {
    local _src="${1}"
    local _fs_type="${2}"
    local _fs_size="${3}"
275

276
277
    local _fs_src="${work_dir}/${_src}"
    local _fs_img="${work_dir}/${_src}.fs"
278

279
280
    if [[ ! -e "${_fs_src}" ]]; then
        _msg_error "The path '${_fs_src}' does not exist" 1
281
282
    fi

283
284
    local _spc_used
    _spc_used=$(du -sxm "${_fs_src}" | awk '{print $1}')
285

286
287
288
289
290
291
292
293
294
295
    # Caculate FS size with desired % of free space, adds 10% overhead to used space.
    if [[ ${_fs_size} != ${_fs_size%\%} ]]; then
        if [[ ${_fs_size%\%} -le 0 || ${_fs_size%\%} -ge 100 ]]; then
            _msg_error "Invalid percentage of free space specified '${_fs_size}' on '${_src}', should be 0% < x < 100%" 1
        fi
        _fs_size=$((_spc_used * 110 / (100 - ${_fs_size%\%})))
    else
        local _spc_used_over=$((_spc_used * 11 / 10))
        if [[ ${_fs_size} -lt ${_spc_used_over} ]]; then
            _msg_error "Filesystem size specified '${_fs_size}' MiB for '${_src}' is too small, must be at least '${_spc_used_over}' MiB" 1
296
297
298
        fi
    fi

299
    _msg_info "Creating ${_fs_type} image of ${_fs_size} MiB..."
300
    rm -f "${_fs_img}"
301
    truncate -s ${_fs_size}M "${_fs_img}"
302
303
304
    local _qflag=""
    if [[ ${quiet} == "y" ]]; then
        _qflag="-q"
305
    fi
306
307
    case "${_fs_type}" in
        ext4)
308
            mkfs.ext4 ${_qflag} -O ^has_journal -E lazy_itable_init=0 -m 0 -F "${_fs_img}"
309
310
311
312
313
314
315
316
317
318
319
320
321
            tune2fs -c 0 -i 0 "${_fs_img}" &> /dev/null
            ;;
        ext3)
            mkfs.ext3 ${_qflag} -m 0 -F "${_fs_img}"
            tune2fs -c 0 -i 0 "${_fs_img}" &> /dev/null
            ;;
        ext2)
            mkfs.ext2 ${_qflag} -m 0 -F "${_fs_img}"
            tune2fs -c 0 -i 0 "${_fs_img}" &> /dev/null
            ;;
        xfs)
            mkfs.xfs ${_qflag} "${_fs_img}"
            ;;
322
323
324
        btrfs)
            mkfs.btrfs -M "${_fs_img}"
            ;;
325
326
327
328
        *)
            _msg_error "Invalid filesystem: ${_fs_type}" 1
            ;;
    esac
329
    _msg_info "Done!"
330
    _mount_fs "${_fs_img}" "${work_dir}/mnt/${_src}"
331
    _msg_info "Copying '${_fs_src}/' to '${work_dir}/mnt/${_src}/'..."
332
    cp -aT "${_fs_src}/" "${work_dir}/mnt/${_src}/"
333
    _msg_info "Done!"
334
    _umount_fs "${work_dir}/mnt/${_src}"
335
336
}

337
338
339
command_checksum () {
    _show_config checksum

340
341
342
343
    local _chk_arch

    for _chk_arch in i686 x86_64; do
        if _is_directory_changed "${work_dir}/iso/${install_dir}" "${work_dir}/iso/${install_dir}/checksum.${_chk_arch}.md5"; then
344
            _msg_info "Creating checksum file for self-test (${_chk_arch})..."
345
346
347
348
349
350
351
352
353
354
355
356
            cd "${work_dir}/iso/${install_dir}"
            if [[ -d "${_chk_arch}" ]]; then
                md5sum aitab > checksum.${_chk_arch}.md5
                find ${_chk_arch} -type f -print0 | xargs -0 md5sum >> checksum.${_chk_arch}.md5
                if [[ -d "any" ]]; then
                    find any -type f -print0 | xargs -0 md5sum >> checksum.${_chk_arch}.md5
                fi
            fi
            cd ${OLDPWD}
            _msg_info "Done!"
        fi
    done
357
358
}

359
360
361
362
command_pkglist () {
    _show_config pkglist

    if _is_directory_changed "${work_dir}/root-image/var/lib/pacman/local" "${work_dir}/iso/${install_dir}/pkglist.${arch}.txt"; then
363
        _msg_info "Creating a list of installed packages on live-enviroment..."
364
365
366
367
368
369
370
371
        pacman -Sl -r "${work_dir}/root-image" --config "${pacman_conf}" | \
            awk '/\[installed\]$/ {print $1 "/" $2 "-" $3}' > \
            "${work_dir}/iso/${install_dir}/pkglist.${arch}.txt"
        _msg_info "Done!"
    fi

}

372
# Create an ISO9660 filesystem from "iso" directory.
373
command_iso () {
374
375
    local _iso_efi_boot_args=""

376
377
378
    if [[ ! -f "${work_dir}/iso/isolinux/isolinux.bin" ]]; then
         _msg_error "The file '${work_dir}/iso/isolinux/isolinux.bin' does not exist." 1
    fi
379
380
381
    if [[ ! -f "${work_dir}/iso/isolinux/isohdpfx.bin" ]]; then
         _msg_error "The file '${work_dir}/iso/isolinux/isohdpfx.bin' does not exist." 1
    fi
382

383
384
    # If exists, add an EFI "El Torito" boot image (FAT filesystem) to ISO-9660 image.
    if [[ -f "${work_dir}/iso/EFI/archiso/efiboot.img" ]]; then
385
386
387
388
        _iso_efi_boot_args="-eltorito-alt-boot
                            -e EFI/archiso/efiboot.img
                            -no-emul-boot
                            -isohybrid-gpt-basdat"
389
390
    fi

391
392
    _show_config iso

393
394
    if _is_directory_changed "${work_dir}/iso" "${out_dir}/${img_name}"; then
        mkdir -p ${out_dir}
395
396
397
398
399
        _msg_info "Creating ISO image..."
        local _qflag=""
        if [[ ${quiet} == "y" ]]; then
            _qflag="-quiet"
        fi
400
        xorriso -as mkisofs ${_qflag} \
401
            -iso-level 3 \
402
403
404
405
406
407
408
            -full-iso9660-filenames \
            -volid "${iso_label}" \
            -appid "${iso_application}" \
            -publisher "${iso_publisher}" \
            -preparer "prepared by mkarchiso" \
            -eltorito-boot isolinux/isolinux.bin \
            -eltorito-catalog isolinux/boot.cat \
409
            -no-emul-boot -boot-load-size 4 -boot-info-table \
410
            -isohybrid-mbr ${work_dir}/iso/isolinux/isohdpfx.bin \
411
            ${_iso_efi_boot_args} \
412
413
            -output "${out_dir}/${img_name}" \
            "${work_dir}/iso/"
414
        _msg_info "Done! | $(ls -sh ${out_dir}/${img_name})"
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
    fi
}

# Parse aitab and create each filesystem specified on that, and push it in "iso" directory.
command_prepare () {
    if [[ ! -f "${work_dir}/iso/${install_dir}/aitab" ]]; then
         _msg_error "The file '${work_dir}/iso/${install_dir}/aitab' does not exist." 1
    fi
    _show_config prepare

    _cleanup
    local _aitab_img _aitab_mnt _aitab_arch _aitab_sfs_comp _aitab_fs_type _aitab_fs_size
    while read _aitab_img _aitab_mnt _aitab_arch _aitab_sfs_comp _aitab_fs_type _aitab_fs_size ; do
        if [[ ${_aitab_img} =~ ^# ]]; then
            continue
        fi
431
432
433
        if [[ "${_aitab_arch}" != "any" && "${_aitab_arch}" != "${arch}" ]]; then
            continue
        fi
434
435
436
437
        local _src="${work_dir}/${_aitab_img}"
        local _dst="${work_dir}/iso/${install_dir}/${_aitab_arch}"
        mkdir -p "${_dst}"
        if [[ ${_aitab_fs_type} != "none" ]]; then
438
439
440
441
442
            if _is_directory_changed "${_src}" "${_dst}/${_aitab_img}.fs.sfs"; then
                _mkfs ${_aitab_img} ${_aitab_fs_type} ${_aitab_fs_size}
                _mksfs ${_aitab_img}.fs ${_aitab_sfs_comp}
                mv "${_src}.fs.sfs" "${_dst}"
                rm "${_src}.fs"
443
444
445
446
447
448
449
450
451
452
453
454
            fi
        else
            if _is_directory_changed "${_src}" "${_dst}/${_aitab_img}.sfs"; then
                _mksfs ${_aitab_img} ${_aitab_sfs_comp}
                mv "${work_dir}/${_aitab_img}.sfs" "${_dst}"
            fi
        fi
    done < "${work_dir}/iso/${install_dir}/aitab"
}

# Install packages on root-image.
# A basic check to avoid double execution/reinstallation is done via hashing package names.
455
command_install () {
456
457
458
459
460
461
462
463
464
465
466
467
    if [[ ! -f "${pacman_conf}" ]]; then
        _msg_error "Pacman config file '${pacman_conf}' does not exist" 1
    fi

    #trim spaces
    pkg_list="$(echo ${pkg_list})"

    if [[ -z ${pkg_list} ]]; then
        _msg_error "Packages must be specified" 0
        _usage 1
    fi

468
    _show_config install
469
470
471

    local _pkg_list_hash
    _pkg_list_hash=$(echo ${pkg_list} | sort -u | md5sum | cut -c1-32)
472
    if [[ -f "${work_dir}/install.${_pkg_list_hash}" ]]; then
473
474
475
        _msg_info "These packages are already installed, skipping."
    else
        _pacman "${pkg_list}"
476
        : > "${work_dir}/install.${_pkg_list_hash}"
477
    fi
478
479
}

480
481
482
483
484
485
486
487
488
command_init() {
    _show_config init
    _chroot_init
}

command_run() {
    _show_config run
    _chroot_run
}
489
490
491
492
493

if [[ ${EUID} -ne 0 ]]; then
    _msg_error "This script must be run as root." 1
fi

494
while getopts 'p:r:C:L:P:A:D:w:o:vh' arg; do
495
496
    case "${arg}" in
        p) pkg_list="${pkg_list} ${OPTARG}" ;;
497
        r) run_cmd="${OPTARG}" ;;
498
499
500
501
502
        C) pacman_conf="${OPTARG}" ;;
        L) iso_label="${OPTARG}" ;;
        P) iso_publisher="${OPTARG}" ;;
        A) iso_application="${OPTARG}" ;;
        D) install_dir="${OPTARG}" ;;
503
504
        w) work_dir="${OPTARG}" ;;
        o) out_dir="${OPTARG}" ;;
505
506
507
508
509
510
511
512
513
514
515
516
517
518
        v) quiet="n" ;;
        h|?) _usage 0 ;;
        *)
            _msg_error "Invalid argument '${arg}'" 0
            _usage 1
            ;;
    esac
done

shift $((OPTIND - 1))

if [[ $# -lt 1 ]]; then
    _msg_error "No command specified" 0
    _usage 1
519
fi
520
521
522
command_name="${1}"

case "${command_name}" in
523
524
525
526
527
528
529
530
    init)
        command_init
        ;;
    install)
        command_install
        ;;
    run)
        command_run
531
532
533
534
        ;;
    prepare)
        command_prepare
        ;;
535
536
537
    checksum)
        command_checksum
        ;;
538
539
540
    pkglist)
        command_pkglist
        ;;
541
    iso)
542
        if [[ $# -lt 2 ]]; then
543
544
545
            _msg_error "No image specified" 0
            _usage 1
        fi
546
        img_name="${2}"
547
548
549
550
551
552
553
        command_iso
        ;;
    *)
        _msg_error "Invalid command name '${command_name}'" 0
        _usage 1
        ;;
esac
554
555

# vim:ts=4:sw=4:et: