OSDN Git Service

Merge branch 'alteriso-3.1-plasma' into dev
[alterlinux/alterlinux.git] / build.sh
1 #!/usr/bin/env bash
2 #
3 # Yamada Hayao
4 # Twitter: @Hayao0819
5 # Email  : hayao@fascode.net
6 #
7 # (c) 2019-2021 Fascode Network.
8 #
9 # build.sh
10 #
11 # The main script that runs the build
12 #
13
14 set -eu
15
16 # Internal config
17 # Do not change these values.
18 script_path="$( cd -P "$( dirname "$(readlink -f "${0}")" )" && pwd )"
19 defaultconfig="${script_path}/default.conf"
20 tools_dir="${script_path}/tools"
21 module_dir="${script_path}/modules"
22 customized_username=false
23 customized_password=false
24 customized_kernel=false
25 customized_logpath=false
26 pkglist_args=()
27 makepkg_script_args=()
28 modules=()
29 DEFAULT_ARGUMENT=""
30 alteriso_version="3.1"
31
32 # Load config file
33 [[ ! -f "${defaultconfig}" ]] && "${tools_dir}/msg.sh" -a 'build.sh' error "${defaultconfig} was not found." && exit 1
34 for config in "${defaultconfig}" "${script_path}/custom.conf"; do
35     [[ -f "${config}" ]] && source "${config}"
36 done
37
38 umask 0022
39
40 # Message common function
41 # msg_common [type] [-n] [string]
42 msg_common(){
43     local _msg_opts=("-a" "build.sh") _type="${1}"
44     shift 1
45     [[ "${1}" = "-n" ]] && _msg_opts+=("-o" "-n") && shift 1
46     [[ "${msgdebug}" = true ]] && _msg_opts+=("-x")
47     [[ "${nocolor}"  = true ]] && _msg_opts+=("-n")
48     _msg_opts+=("${_type}" "${@}")
49     "${tools_dir}/msg.sh" "${_msg_opts[@]}"
50 }
51
52 # Show an INFO message
53 # ${1}: message string
54 msg_info() { msg_common info "${@}"; }
55
56 # Show an Warning message
57 # ${1}: message string
58 msg_warn() { msg_common warn "${@}"; }
59
60 # Show an debug message
61 # ${1}: message string
62 msg_debug() { 
63     [[ "${debug}" = true ]] && msg_common debug "${@}"
64     return 0
65 }
66
67 # Show an ERROR message then exit with status
68 # ${1}: message string
69 # ${2}: exit code number (with 0 does not exit)
70 msg_error() {
71     msg_common error "${1}"
72     [[ -n "${2:-}" ]] && exit "${2}"
73 }
74
75
76 # Usage: getclm <number>
77 # 標準入力から値を受けとり、引数で指定された列を抽出します。
78 getclm() { cut -d " " -f "${1}"; }
79
80 # Usage: echo_blank <number>
81 # 指定されたぶんの半角空白文字を出力します
82 echo_blank(){ yes " " 2> /dev/null  | head -n "${1}" | tr -d "\n"; }
83
84 _usage () {
85     echo "usage ${0} [options] [channel]"
86     echo
87     echo "A channel is a profile of AlterISO settings."
88     echo
89     echo " General options:"
90     echo "    -b | --boot-splash           Enable boot splash"
91     echo "    -e | --cleanup | --cleaning  Enable post-build cleaning"
92     echo "         --tarball               Build rootfs in tar.xz format"
93     echo "    -h | --help                  This help message and exit"
94     echo
95     echo "    -a | --arch <arch>           Set iso architecture"
96     echo "                                  Default: ${arch}"
97     echo "    -c | --comp-type <comp_type> Set SquashFS compression type (gzip, lzma, lzo, xz, zstd)"
98     echo "                                  Default: ${sfs_comp}"
99     echo "    -g | --gpgkey <key>          Set gpg key"
100     echo "                                  Default: ${gpg_key}"
101     echo "    -l | --lang <lang>           Specifies the default language for the live environment"
102     echo "                                  Default: ${locale_name}"
103     echo "    -k | --kernel <kernel>       Set special kernel type.See below for available kernels"
104     echo "                                  Default: ${defaultkernel}"
105     echo "    -o | --out <out_dir>         Set the output directory"
106     echo "                                  Default: ${out_dir}"
107     echo "    -p | --password <password>   Set a live user password"
108     echo "                                  Default: ${password}"
109     echo "    -t | --comp-opts <options>   Set compressor-specific options."
110     echo "                                  Default: empty"
111     echo "    -u | --user <username>       Set user name"
112     echo "                                  Default: ${username}"
113     echo "    -w | --work <work_dir>       Set the working directory"
114     echo "                                  Default: ${work_dir}"
115     echo
116
117     local blank="33" _arch _dirname _type
118
119     for _type in "locale" "kernel"; do
120         echo " ${_type} for each architecture:"
121         for _arch in $(find "${script_path}/system/" -maxdepth 1 -mindepth 1 -name "${_type}-*" -print0 | xargs -I{} -0 basename {} | sed "s|${_type}-||g"); do
122             echo -n "    ${_arch}$(echo_blank "$(( "${blank}" - 4 - "${#_arch}" ))")"
123             "${tools_dir}/${_type}.sh" -a "${_arch}" show
124         done
125         echo
126     done
127
128     echo " Channel:"
129     for _dirname in $(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" -d -b -n --line show | sed "s|.add$||g"); do
130         echo -ne "    ${_dirname}$(echo_blank "$(( "${blank}" - 4 - "${#_dirname}" ))")"
131         "${tools_dir}/channel.sh" --version "${alteriso_version}" --nocheck desc "${_dirname}"
132     done
133
134     echo
135     echo " Debug options: Please use at your own risk."
136     echo "    -d | --debug                 Enable debug messages"
137     echo "    -x | --bash-debug            Enable bash debug mode(set -xv)"
138     echo "         --channellist           Output the channel list and exit"
139     echo "         --gitversion            Add Git commit hash to image file version"
140     echo "         --logpath <file>        Set log file path (use with --log)"
141     echo "         --[no]log               (No) log ;re-run script with tee"
142     echo "         --msgdebug              Enables output debugging"
143     echo "         --noaur                 Ignore aur packages (Use only for debugging)"
144     echo "         --nocolor               No output colored output"
145     echo "         --[no]confirm           (No) check the settings before building"
146     echo "         --nochkver              No check the version of the channel"
147     echo "         --nodebug               No debug message"
148     echo "         --noefi                 No efi boot (Use only for debugging)"
149     echo "         --noloopmod             No check and load kernel module automatically"
150     echo "         --nodepend              No check package dependencies before building"
151     echo "         --noiso                 No build iso image (Use with --tarball)"
152     echo "         --pacman-debug          Enable pacman debug mode"
153     echo "         --normwork              No remove working dir"
154     echo "         --nopkgbuild            Ignore PKGBUILD (Use only for debugging)"
155     echo
156     echo " Many packages are installed from AUR, so specifying --noaur can cause problems."
157     echo
158     [[ -n "${1:-}" ]] && exit "${1}"
159 }
160
161 # Unmount helper Usage: _umount <target>
162 _umount() { if mountpoint -q "${1}"; then umount -lf "${1}"; fi; }
163
164 # Mount helper Usage: _mount <source> <target>
165 _mount() { ! mountpoint -q "${2}" && [[ -f "${1}" ]] && [[ -d "${2}" ]] && mount "${1}" "${2}"; return 0; }
166
167 # Unmount work dir
168 umount_work () {
169     local _args=("${build_dir}")
170     [[ "${debug}" = true ]] && _args=("-d" "${_args[@]}")
171     [[ "${nocolor}" = true ]] && _args+=("--nocolor")
172     "${tools_dir}/umount.sh" "${_args[@]}"
173 }
174
175 # Mount airootfs on "${airootfs_dir}"
176 mount_airootfs () {
177     mkdir -p "${airootfs_dir}"
178     _mount "${airootfs_dir}.img" "${airootfs_dir}"
179 }
180
181
182 # Helper function to run make_*() only one time.
183 run_once() {
184     set -eu
185     if [[ ! -e "${lockfile_dir}/build.${1}" ]]; then
186         msg_debug "Running ${1} ..."
187         mount_airootfs
188         eval "${@}"
189         mkdir -p "${lockfile_dir}"; touch "${lockfile_dir}/build.${1}"
190         umount_work
191     else
192         msg_debug "Skipped because ${1} has already been executed."
193     fi
194 }
195
196 # Show message when file is removed
197 # remove <file> <file> ...
198 remove() {
199     local _file
200     for _file in "${@}"; do msg_debug "Removing ${_file}"; rm -rf "${_file}"; done
201 }
202
203 # 強制終了時にアンマウント
204 umount_trap() {
205     local _status="${?}"
206     umount_work
207     msg_error "It was killed by the user.\nThe process may not have completed successfully."
208     exit "${_status}"
209 }
210
211 # 設定ファイルを読み込む
212 # load_config [file1] [file2] ...
213 load_config() {
214     local _file
215     for _file in "${@}"; do [[ -f "${_file}" ]] && source "${_file}" && msg_debug "The settings have been overwritten by the ${_file}"; done
216     return 0
217 }
218
219 # Display channel list
220 show_channel_list() {
221     local _args=("-v" "${alteriso_version}" show)
222     [[ "${nochkver}" = true ]] && _args+=("-n")
223     bash "${tools_dir}/channel.sh" "${_args[@]}"
224 }
225
226 # Execute command for each module. It will be executed with {} replaced with the module name.
227 # for_module <command>
228 for_module(){
229     local module
230     for module in "${modules[@]}"; do eval "${@//"{}"/${module}}"; done
231 }
232
233 # pacstrapを実行
234 _pacstrap(){
235     msg_info "Installing packages to ${airootfs_dir}/'..."
236     local _args=("-c" "-G" "-M" "--" "${airootfs_dir}" "${@}")
237     [[ "${pacman_debug}" = true ]] && _args+=(--debug)
238     pacstrap -C "${build_dir}/pacman.conf" "${_args[@]}"
239     msg_info "Packages installed successfully!"
240 }
241
242 # chroot環境でpacmanコマンドを実行
243 # /etc/alteriso-pacman.confを準備してコマンドを実行します
244 _run_with_pacmanconf(){
245     sed "s|^CacheDir     =|#CacheDir    =|g" "${build_dir}/pacman.conf" > "${airootfs_dir}/etc/alteriso-pacman.conf"
246     eval -- "${@}"
247     remove "${airootfs_dir}/etc/alteriso-pacman.conf"
248 }
249
250 # コマンドをchrootで実行する
251 _chroot_run() {
252     msg_debug "Run command in chroot\nCommand: ${*}"
253     eval -- arch-chroot "${airootfs_dir}" "${@}"
254 }
255
256 _cleanup_common () {
257     msg_info "Cleaning up what we can on airootfs..."
258
259     # Delete pacman database sync cache files (*.tar.gz)
260     [[ -d "${airootfs_dir}/var/lib/pacman" ]] && find "${airootfs_dir}/var/lib/pacman" -maxdepth 1 -type f -delete
261
262     # Delete pacman database sync cache
263     [[ -d "${airootfs_dir}/var/lib/pacman/sync" ]] && find "${airootfs_dir}/var/lib/pacman/sync" -delete
264
265     # Delete pacman package cache
266     [[ -d "${airootfs_dir}/var/cache/pacman/pkg" ]] && find "${airootfs_dir}/var/cache/pacman/pkg" -type f -delete
267
268     # Delete all log files, keeps empty dirs.
269     [[ -d "${airootfs_dir}/var/log" ]] && find "${airootfs_dir}/var/log" -type f -delete
270
271     # Delete all temporary files and dirs
272     [[ -d "${airootfs_dir}/var/tmp" ]] && find "${airootfs_dir}/var/tmp" -mindepth 1 -delete
273
274     # Delete package pacman related files.
275     find "${build_dir}" \( -name '*.pacnew' -o -name '*.pacsave' -o -name '*.pacorig' \) -delete
276
277     # Delete all cache file
278     [[ -d "${airootfs_dir}/var/cache" ]] && find "${airootfs_dir}/var/cache" -mindepth 1 -delete
279
280     # Create an empty /etc/machine-id
281     printf '' > "${airootfs_dir}/etc/machine-id"
282
283     msg_info "Done!"
284 }
285
286 _cleanup_airootfs(){
287     _cleanup_common
288     # Delete all files in /boot
289     [[ -d "${airootfs_dir}/boot" ]] && find "${airootfs_dir}/boot" -mindepth 1 -delete
290 }
291
292 _mkchecksum() {
293     msg_info "Creating md5 checksum ..."
294     echo "$(md5sum "${1}" | getclm 1) $(basename "${1}")" > "${1}.md5"
295     msg_info "Creating sha256 checksum ..."
296     echo "$(sha256sum "${1}" | getclm 1) $(basename "${1}")" > "${1}.sha256"
297 }
298
299 # Check the value of a variable that can only be set to true or false.
300 check_bool() {
301     local _value _variable
302     for _variable in "${@}"; do
303         msg_debug -n "Checking ${_variable}..."
304         eval ": \${${_variable}:=''}"
305         _value="$(eval echo "\$${_variable}")"
306         if [[ ! -v "${1}" ]] || [[ "${_value}"  = "" ]]; then
307             [[ "${debug}" = true ]] && echo ; msg_error "The variable name ${_variable} is empty." "1"
308         elif [[ ! "${_value}" = "true" ]] && [[ ! "${_value}" = "false" ]]; then
309             [[ "${debug}" = true ]] && echo ; msg_error "The variable name ${_variable} is not of bool type (${_variable} = ${_value})" "1"
310         elif [[ "${debug}" = true ]]; then
311             echo -e " ${_value}"
312         fi
313     done
314 }
315
316 _run_cleansh(){
317     bash $([[ "${bash_debug}" = true ]] && echo -n "-x" ) "${tools_dir}/clean.sh" -o -w "$(realpath "${build_dir}")" "$([[ "${debug}" = true ]] && printf "%s" "-d")" "$([[ "${noconfirm}" = true ]] && printf "%s" "-n")" "$([[ "${nocolor}" = true ]] && printf "%s" "--nocolor")"
318 }
319
320
321 # Check the build environment and create a directory.
322 prepare_env() {
323     # Check packages
324     if [[ "${nodepend}" = false ]]; then
325         local _check_failed=false _pkg _result=0
326         msg_info "Checking dependencies ..."
327         ! pacman -Qq pyalpm > /dev/null 2>&1 && msg_error "pyalpm is not installed." 1
328         for _pkg in "${dependence[@]}"; do
329             eval "${tools_dir}/package.py" "${_pkg}" "$( [[ "${debug}" = false ]] && echo "> /dev/null")" || _result="${?}"
330             if (( _result == 3 )) || (( _result == 4 )); then
331                 _check_failed=true
332             fi
333             _result=0
334         done
335         [[ "${_check_failed}" = true ]] && exit 1
336     fi
337
338     # Load loop kernel module
339     if [[ "${noloopmod}" = false ]]; then
340         [[ ! -d "/usr/lib/modules/$(uname -r)" ]] && msg_error "The currently running kernel module could not be found.\nProbably the system kernel has been updated.\nReboot your system to run the latest kernel." "1"
341         lsmod | getclm 1 | grep -x "loop" || modprobe loop
342     fi
343
344     # Check work dir
345     if [[ "${normwork}" = false ]]; then
346         msg_info "Deleting the contents of ${build_dir}..."
347         _run_cleansh
348     fi
349
350     # Set gpg key
351     if [[ -n "${gpg_key}" ]]; then
352         gpg --batch --output "${work_dir}/pubkey.gpg" --export "${gpg_key}"
353         exec {ARCHISO_GNUPG_FD}<>"${build_dir}/pubkey.gpg"
354         export ARCHISO_GNUPG_FD
355     fi
356
357     # 強制終了時に作業ディレクトリを削除する
358     local _trap_remove_work
359     _trap_remove_work() {
360         local status="${?}"
361         [[ "${normwork}" = false ]] && echo && _run_cleansh
362         exit "${status}"
363     }
364     trap '_trap_remove_work' 1 2 3 15
365
366     return 0
367 }
368
369
370 # Show settings.
371 show_settings() {
372     if [[ "${boot_splash}" = true ]]; then
373         msg_info "Boot splash is enabled."
374         msg_info "Theme is used ${theme_name}."
375     fi
376     msg_info "Language is ${locale_fullname}."
377     msg_info "Use the ${kernel} kernel."
378     msg_info "Live username is ${username}."
379     msg_info "Live user password is ${password}."
380     msg_info "The compression method of squashfs is ${sfs_comp}."
381     msg_info "Use the ${channel_name%.add} channel."
382     msg_info "Build with architecture ${arch}."
383     if [[ "${noconfirm}" = false ]]; then
384         echo -e "\nPress Enter to continue or Ctrl + C to cancel."
385         read -r
386     fi
387     trap 1 2 3 15
388     trap 'umount_trap' 1 2 3 15
389
390     return 0
391 }
392
393
394 # Preparation for build
395 prepare_build() {
396     # Show alteriso version
397     if [[ -d "${script_path}/.git" ]]; then
398         cd  "${script_path}"
399         msg_debug "The version of alteriso is $(git describe --long --tags | sed 's/\([^-]*-g\)/r\1/;s/-/./g')."
400         cd "${OLDPWD}"
401     fi
402
403     # Load configs
404     load_config "${channel_dir}/config.any" "${channel_dir}/config.${arch}"
405
406     # Debug mode
407     if [[ "${bash_debug}" = true ]]; then
408         set -x -v
409     fi
410
411     # Legacy mode
412     if [[ "$(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" ver "${channel_name}")" = "3.0" ]]; then
413         msg_warn "The module cannot be used because it works with Alter ISO3.0 compatibility."
414         if [[ -n "${include_extra+SET}" ]]; then
415             if [[ "${include_extra}" = true ]]; then
416                 modules=("base" "share" "share-extra" "calamares" "zsh-powerline")
417             else
418                 modules=("base" "share")
419             fi
420         fi
421     fi
422
423     local module_check
424     module_check(){
425         msg_debug "Checking ${1} module ..."
426         bash "${tools_dir}/module.sh" check "${1}" || msg_error "Module ${1} is not available." "1";
427     }
428     readarray -t modules < <(printf "%s\n" "${modules[@]}" | awk '!a[$0]++')
429     for_module "module_check {}"
430     for_module load_config "${module_dir}/{}/config.any" "${module_dir}/{}/config.${arch}"
431     msg_debug "Loaded modules: ${modules[*]}"
432     ! printf "%s\n" "${modules[@]}" | grep -x "share" >/dev/null 2>&1 && msg_warn "The share module is not loaded."
433
434     # Set kernel
435     [[ "${customized_kernel}" = false ]] && kernel="${defaultkernel}"
436
437     # Parse files
438     eval "$(bash "${tools_dir}/locale.sh" -s -a "${arch}" get "${locale_name}")"
439     eval "$(bash "${tools_dir}/kernel.sh" -s -c "${channel_name}" -a "${arch}" get "${kernel}")"
440
441     # Set username and password
442     [[ "${customized_username}" = false ]] && username="${defaultusername}"
443     [[ "${customized_password}" = false ]] && password="${defaultpassword}"
444
445     # gitversion
446     [[ "${gitversion}" = true ]] && iso_version="${iso_version}-${gitrev}"
447
448     # Generate iso file name.
449     local _channel_name="${channel_name%.add}-${locale_version}" 
450     iso_filename="${iso_name}-${_channel_name}-${iso_version}-${arch}.iso"
451     tar_filename="${iso_filename%.iso}.tar.xz"
452     [[ "${nochname}" = true ]] && iso_filename="${iso_name}-${iso_version}-${arch}.iso"
453     msg_debug "Iso filename is ${iso_filename}"
454
455     # check bool
456     check_bool boot_splash cleaning noconfirm nodepend customized_username customized_password noloopmod nochname tarball noiso noaur customized_syslinux norescue_entry debug bash_debug nocolor msgdebug noefi nosigcheck gitversion
457
458     # Check architecture for each channel
459     local _exit=0
460     bash "${tools_dir}/channel.sh" --version "${alteriso_version}" -a "${arch}" -n -b check "${channel_name}" || _exit="${?}"
461     if (( "${_exit}" != 0 )) && (( "${_exit}" != 1 )); then
462         msg_error "${channel_name} channel does not support current architecture (${arch})." "1"
463     fi
464
465     # Run with tee
466     if [[ ! "${logging}" = false ]]; then
467         [[ "${customized_logpath}" = false ]] && logging="${out_dir}/${iso_filename%.iso}.log"
468         mkdir -p "$(dirname "${logging}")" && touch "${logging}"
469         msg_warn "Re-run sudo ${0} ${ARGUMENT[*]} --nodepend --nolog --nocolor 2>&1 | tee ${logging}"
470         sudo "${0}" "${ARGUMENT[@]}" --nolog --nocolor --nodepend 2>&1 | tee "${logging}"
471         exit "${?}"
472     else
473         unset ARGUMENT
474     fi
475
476     # Set argument of pkglist.sh
477     pkglist_args=("-a" "${arch}" "-k" "${kernel}" "-c" "${channel_dir}" "-l" "${locale_name}" --line)
478     [[ "${boot_splash}"              = true ]] && pkglist_args+=("-b")
479     [[ "${debug}"                    = true ]] && pkglist_args+=("-d")
480     [[ "${memtest86}"                = true ]] && pkglist_args+=("-m")
481     [[ "${nocolor}"                  = true ]] && pkglist_args+=("--nocolor")
482     (( "${#additional_exclude_pkg[@]}" >= 1 )) && pkglist_args+=("-e" "${additional_exclude_pkg[*]}")
483     pkglist_args+=("${modules[@]}")
484
485     # Set argument of aur.sh and pkgbuild.sh
486     [[ "${bash_debug}"   = true ]] && makepkg_script_args+=("-x")
487     [[ "${pacman_debug}" = true ]] && makepkg_script_args+=("-d")
488
489     return 0
490 }
491
492
493 # Setup custom pacman.conf with current cache directories.
494 make_pacman_conf() {
495     # Pacman configuration file used only when building
496     # If there is pacman.conf for each channel, use that for building
497     local _pacman_conf _pacman_conf_list=("${script_path}/pacman-${arch}.conf" "${channel_dir}/pacman-${arch}.conf" "${script_path}/system/pacman-${arch}.conf")
498     for _pacman_conf in "${_pacman_conf_list[@]}"; do
499         if [[ -f "${_pacman_conf}" ]]; then
500             build_pacman_conf="${_pacman_conf}"
501             break
502         fi
503     done
504
505     msg_debug "Use ${build_pacman_conf}"
506     sed -r "s|^#?\\s*CacheDir.+|CacheDir     = ${cache_dir}|g" "${build_pacman_conf}" > "${build_dir}/pacman.conf"
507
508     [[ "${nosigcheck}" = true ]] && sed -ir "s|^s*SigLevel.+|SigLevel = Never|g" "${build_pacman_conf}"
509
510     [[ -n "$(find "${cache_dir}" -maxdepth 1 -name '*.pkg.tar.*' 2> /dev/null)" ]] && msg_info "Use cached package files in ${cache_dir}"
511
512     # Share any architecture packages
513     #while read -r _pkg; do
514     #    if [[ ! -f "${cache_dir}/$(basename "${_pkg}")" ]]; then
515     #        ln -s "${_pkg}" "${cache_dir}"
516     #    fi
517     #done < <(find "${cache_dir}/../" -type d -name "$(basename "${cache_dir}")" -prune -o -type f -name "*-any.pkg.tar.*" -printf "%p\n")
518
519     return 0
520 }
521
522 # Base installation (airootfs)
523 make_basefs() {
524     msg_info "Creating ext4 image of 32GiB..."
525     truncate -s 32G -- "${airootfs_dir}.img"
526     mkfs.ext4 -O '^has_journal,^resize_inode' -E 'lazy_itable_init=0' -m 0 -F -- "${airootfs_dir}.img" 32G
527     tune2fs -c "0" -i "0" "${airootfs_dir}.img"
528     msg_info "Done!"
529
530     msg_info "Mounting ${airootfs_dir}.img on ${airootfs_dir}"
531     mount_airootfs
532     msg_info "Done!"
533     return 0
534 }
535
536 # Additional packages (airootfs)
537 make_packages_repo() {
538     msg_debug "pkglist.sh ${pkglist_args[*]}"
539     readarray -t _pkglist < <("${tools_dir}/pkglist.sh" "${pkglist_args[@]}")
540
541     # Create a list of packages to be finally installed as packages.list directly under the working directory.
542     echo -e "# The list of packages that is installed in live cd.\n#\n" > "${build_dir}/packages.list"
543     printf "%s\n" "${_pkglist[@]}" >> "${build_dir}/packages.list"
544
545     # Install packages on airootfs
546     _pacstrap "${_pkglist[@]}"
547
548     return 0
549 }
550
551 make_packages_aur() {
552     readarray -t _pkglist_aur < <("${tools_dir}/pkglist.sh" --aur "${pkglist_args[@]}")
553
554     # Create a list of packages to be finally installed as packages.list directly under the working directory.
555     echo -e "\n# AUR packages.\n#\n" >> "${build_dir}/packages.list"
556     printf "%s\n" "${_pkglist_aur[@]}" >> "${build_dir}/packages.list"
557
558     # prepare for yay
559     cp -rf --preserve=mode "${script_path}/system/aur.sh" "${airootfs_dir}/root/aur.sh"
560
561     # Run aur script
562     _run_with_pacmanconf _chroot_run "bash" "/root/aur.sh" "${makepkg_script_args[@]}" "${_pkglist_aur[@]}"
563
564     # Remove script
565     remove "${airootfs_dir}/root/aur.sh"
566
567     return 0
568 }
569
570 make_pkgbuild() {
571     # Get PKGBUILD List
572     local _pkgbuild_dirs=("${channel_dir}/pkgbuild.any" "${channel_dir}/pkgbuild.${arch}")
573     for_module '_pkgbuild_dirs+=("${module_dir}/{}/pkgbuild.any" "${module_dir}/{}/pkgbuild.${arch}")'
574
575     # Copy PKGBUILD to work
576     mkdir -p "${airootfs_dir}/pkgbuilds/"
577     for _dir in $(find "${_pkgbuild_dirs[@]}" -type f -name "PKGBUILD" -print0 2>/dev/null | xargs -0 -I{} realpath {} | xargs -I{} dirname {}); do
578         msg_info "Find $(basename "${_dir}")"
579         cp -r "${_dir}" "${airootfs_dir}/pkgbuilds/"
580     done
581     
582     # copy buold script
583     cp -rf --preserve=mode "${script_path}/system/pkgbuild.sh" "${airootfs_dir}/root/pkgbuild.sh"
584
585     # Run build script
586     _run_with_pacmanconf _chroot_run "bash" "/root/pkgbuild.sh" "${makepkg_script_args[@]}" "/pkgbuilds"
587
588     # Remove script
589     remove "${airootfs_dir}/root/pkgbuild.sh"
590
591     return 0
592 }
593
594 # Customize installation (airootfs)
595 make_customize_airootfs() {
596     # Overwrite airootfs with customize_airootfs.
597     local _airootfs _airootfs_script_options _script _script_list _airootfs_list=() _main_script
598
599     for_module '_airootfs_list+=("${module_dir}/{}/airootfs.any" "${module_dir}/{}/airootfs.${arch}")'
600     _airootfs_list+=("${channel_dir}/airootfs.any" "${channel_dir}/airootfs.${arch}")
601
602     for _airootfs in "${_airootfs_list[@]}";do
603         if [[ -d "${_airootfs}" ]]; then
604             msg_debug "Copying airootfs ${_airootfs} ..."
605             cp -af "${_airootfs}"/* "${airootfs_dir}"
606         fi
607     done
608
609     # Replace /etc/mkinitcpio.conf if Plymouth is enabled.
610     if [[ "${boot_splash}" = true ]]; then
611         cp -f "${script_path}/mkinitcpio/mkinitcpio-plymouth.conf" "${airootfs_dir}/etc/mkinitcpio.conf"
612     else
613         cp -f "${script_path}/mkinitcpio/mkinitcpio.conf" "${airootfs_dir}/etc/mkinitcpio.conf"
614     fi
615     
616     # customize_airootfs options
617     # -b                        : Enable boot splash.
618     # -d                        : Enable debug mode.
619     # -g <locale_gen_name>      : Set locale-gen.
620     # -i <inst_dir>             : Set install dir
621     # -k <kernel config line>   : Set kernel name.
622     # -o <os name>              : Set os name.
623     # -p <password>             : Set password.
624     # -s <shell>                : Set user shell.
625     # -t                        : Set plymouth theme.
626     # -u <username>             : Set live user name.
627     # -x                        : Enable bash debug mode.
628     # -z <locale_time>          : Set the time zone.
629     # -l <locale_name>          : Set language.
630     #
631     # -j is obsolete in AlterISO3 and cannot be used.
632     # -r is obsolete due to the removal of rebuild.
633     # -k changed in AlterISO3 from passing kernel name to passing kernel configuration.
634
635     # Generate options of customize_airootfs.sh.
636     _airootfs_script_options="-p '${password}' -k '${kernel} ${kernel_filename} ${kernel_mkinitcpio_profile}' -u '${username}' -o '${os_name}' -i '${install_dir}' -s '${usershell}' -a '${arch}' -g '${locale_gen_name}' -l '${locale_name}' -z '${locale_time}' -t ${theme_name}"
637     [[ "${boot_splash}" = true ]] && _airootfs_script_options="${_airootfs_script_options} -b"
638     [[ "${debug}" = true       ]] && _airootfs_script_options="${_airootfs_script_options} -d"
639     [[ "${bash_debug}" = true  ]] && _airootfs_script_options="${_airootfs_script_options} -x"
640
641     _main_script="root/customize_airootfs.sh"
642
643     _script_list=(
644         "${airootfs_dir}/root/customize_airootfs_${channel_name}.sh"
645         "${airootfs_dir}/root/customize_airootfs_${channel_name%.add}.sh"
646     )
647
648     for_module '_script_list+=("${airootfs_dir}/root/customize_airootfs_{}.sh")'
649
650     # Create script
651     for _script in "${_script_list[@]}"; do
652         if [[ -f "${_script}" ]]; then
653             echo -e "\n$(cat "${_script}")" >> "${airootfs_dir}/${_main_script}"
654             remove "${_script}"
655         else
656             msg_debug "${_script} was not found."
657         fi
658     done
659
660     chmod 755 "${airootfs_dir}/${_main_script}"
661     cp "${airootfs_dir}/${_main_script}" "${build_dir}/$(basename ${_main_script})"
662     _chroot_run "${_main_script} ${_airootfs_script_options}"
663     remove "${airootfs_dir}/${_main_script}"
664
665     # /root permission https://github.com/archlinux/archiso/commit/d39e2ba41bf556674501062742190c29ee11cd59
666     chmod -f 750 "${airootfs_dir}/root"
667
668     return 0
669 }
670
671 # Copy mkinitcpio archiso hooks and build initramfs (airootfs)
672 make_setup_mkinitcpio() {
673     local _hook
674     mkdir -p "${airootfs_dir}/etc/initcpio/hooks" "${airootfs_dir}/etc/initcpio/install"
675
676     for _hook in "archiso" "archiso_shutdown" "archiso_pxe_common" "archiso_pxe_nbd" "archiso_pxe_http" "archiso_pxe_nfs" "archiso_loop_mnt"; do
677         cp "${script_path}/system/initcpio/hooks/${_hook}" "${airootfs_dir}/etc/initcpio/hooks"
678         cp "${script_path}/system/initcpio/install/${_hook}" "${airootfs_dir}/etc/initcpio/install"
679     done
680
681     sed -i "s|/usr/lib/initcpio/|/etc/initcpio/|g" "${airootfs_dir}/etc/initcpio/install/archiso_shutdown"
682     cp "${script_path}/system/initcpio/install/archiso_kms" "${airootfs_dir}/etc/initcpio/install"
683     cp "${script_path}/system/initcpio/archiso_shutdown" "${airootfs_dir}/etc/initcpio"
684     if [[ "${boot_splash}" = true ]]; then
685         cp "${script_path}/mkinitcpio/mkinitcpio-archiso-plymouth.conf" "${airootfs_dir}/etc/mkinitcpio-archiso.conf"
686     else
687         cp "${script_path}/mkinitcpio/mkinitcpio-archiso.conf" "${airootfs_dir}/etc/mkinitcpio-archiso.conf"
688     fi
689     if [[ "${gpg_key}" ]]; then
690       gpg --export "${gpg_key}" >"${build_dir}/gpgkey"
691       exec 17<>"${build_dir}/gpgkey"
692     fi
693
694     _chroot_run "mkinitcpio -c /etc/mkinitcpio-archiso.conf -k /boot/${kernel_filename} -g /boot/archiso.img"
695
696     [[ "${gpg_key}" ]] && exec 17<&-
697     
698     return 0
699 }
700
701 # Prepare kernel/initramfs ${install_dir}/boot/
702 make_boot() {
703     mkdir -p "${isofs_dir}/${install_dir}/boot/${arch}"
704     cp "${airootfs_dir}/boot/archiso.img" "${isofs_dir}/${install_dir}/boot/${arch}/archiso.img"
705     cp "${airootfs_dir}/boot/${kernel_filename}" "${isofs_dir}/${install_dir}/boot/${arch}/${kernel_filename}"
706
707     return 0
708 }
709
710 # Add other aditional/extra files to ${install_dir}/boot/
711 make_boot_extra() {
712     if [[ -e "${airootfs_dir}/boot/memtest86+/memtest.bin" ]]; then
713         install -m 0644 -- "${airootfs_dir}/boot/memtest86+/memtest.bin" "${isofs_dir}/${install_dir}/boot/memtest"
714         install -d -m 0755 -- "${isofs_dir}/${install_dir}/boot/licenses/memtest86+/"
715         install -m 0644 -- "${airootfs_dir}/usr/share/licenses/common/GPL2/license.txt" "${isofs_dir}/${install_dir}/boot/licenses/memtest86+/"
716     fi
717
718     local _ucode_image
719     msg_info "Preparing microcode for the ISO 9660 file system..."
720
721     for _ucode_image in {intel-uc.img,intel-ucode.img,amd-uc.img,amd-ucode.img,early_ucode.cpio,microcode.cpio}; do
722         if [[ -e "${airootfs_dir}/boot/${_ucode_image}" ]]; then
723             msg_info "Installimg ${_ucode_image} ..."
724             install -m 0644 -- "${airootfs_dir}/boot/${_ucode_image}" "${isofs_dir}/${install_dir}/boot/"
725             if [[ -e "${airootfs_dir}/usr/share/licenses/${_ucode_image%.*}/" ]]; then
726                 install -d -m 0755 -- "${isofs_dir}/${install_dir}/boot/licenses/${_ucode_image%.*}/"
727                 install -m 0644 -- "${airootfs_dir}/usr/share/licenses/${_ucode_image%.*}/"* "${isofs_dir}/${install_dir}/boot/licenses/${_ucode_image%.*}/"
728             fi
729         fi
730     done
731     msg_info "Done!"
732
733     return 0
734 }
735
736 # Prepare /${install_dir}/boot/syslinux
737 make_syslinux() {
738     mkdir -p "${isofs_dir}/syslinux"
739
740     # 一時ディレクトリに設定ファイルをコピー
741     mkdir -p "${build_dir}/syslinux/"
742     cp -a "${script_path}/syslinux/"* "${build_dir}/syslinux/"
743     if [[ -d "${channel_dir}/syslinux" ]] && [[ "${customized_syslinux}" = true ]]; then
744         cp -af "${channel_dir}/syslinux"* "${build_dir}/syslinux/"
745     fi
746
747     # copy all syslinux config to work dir
748     for _cfg in "${build_dir}/syslinux/"*.cfg; do
749         sed "s|%ARCHISO_LABEL%|${iso_label}|g;
750              s|%OS_NAME%|${os_name}|g;
751              s|%KERNEL_FILENAME%|${kernel_filename}|g;
752              s|%ARCH%|${arch}|g;
753              s|%INSTALL_DIR%|${install_dir}|g" "${_cfg}" > "${isofs_dir}/syslinux/${_cfg##*/}"
754     done
755
756     # Replace the SYSLINUX configuration file with or without boot splash.
757     local _use_config_name _no_use_config_name _pxe_or_sys
758     if [[ "${boot_splash}" = true ]]; then
759         _use_config_name=splash
760         _no_use_config_name=nosplash
761     else
762         _use_config_name=nosplash
763         _no_use_config_name=splash
764     fi
765     for _pxe_or_sys in "sys" "pxe"; do
766         remove "${isofs_dir}/syslinux/archiso_${_pxe_or_sys}_${_no_use_config_name}.cfg"
767         mv "${isofs_dir}/syslinux/archiso_${_pxe_or_sys}_${_use_config_name}.cfg" "${isofs_dir}/syslinux/archiso_${_pxe_or_sys}.cfg"
768     done
769
770     # Set syslinux wallpaper
771     if [[ -f "${channel_dir}/splash.png" ]]; then
772         cp "${channel_dir}/splash.png" "${isofs_dir}/syslinux"
773     else
774         cp "${script_path}/syslinux/splash.png" "${isofs_dir}/syslinux"
775     fi
776
777     # remove config
778     local _remove_config
779     function _remove_config() {
780         remove "${isofs_dir}/syslinux/${1}"
781         sed -i "s|$(grep "${1}" "${isofs_dir}/syslinux/archiso_sys_load.cfg")||g" "${isofs_dir}/syslinux/archiso_sys_load.cfg" 
782     }
783
784     [[ "${norescue_entry}" = true  ]] && _remove_config archiso_sys_rescue.cfg
785     [[ "${memtest86}"      = false ]] && _remove_config memtest86.cfg
786
787     # copy files
788     cp "${airootfs_dir}/usr/lib/syslinux/bios/"*.c32 "${isofs_dir}/syslinux"
789     cp "${airootfs_dir}/usr/lib/syslinux/bios/lpxelinux.0" "${isofs_dir}/syslinux"
790     cp "${airootfs_dir}/usr/lib/syslinux/bios/memdisk" "${isofs_dir}/syslinux"
791
792
793     if [[ -e "${isofs_dir}/syslinux/hdt.c32" ]]; then
794         install -d -m 0755 -- "${isofs_dir}/syslinux/hdt"
795         if [[ -e "${airootfs_dir}/usr/share/hwdata/pci.ids" ]]; then
796             gzip -c -9 "${airootfs_dir}/usr/share/hwdata/pci.ids" > "${isofs_dir}/syslinux/hdt/pciids.gz"
797         fi
798         find "${airootfs_dir}/usr/lib/modules" -name 'modules.alias' -print -exec gzip -c -9 '{}' ';' -quit > "${isofs_dir}/syslinux/hdt/modalias.gz"
799     fi
800
801     return 0
802 }
803
804 # Prepare /isolinux
805 make_isolinux() {
806     install -d -m 0755 -- "${isofs_dir}/syslinux"
807     sed "s|%INSTALL_DIR%|${install_dir}|g" "${script_path}/system/isolinux.cfg" > "${isofs_dir}/syslinux/isolinux.cfg"
808     install -m 0644 -- "${airootfs_dir}/usr/lib/syslinux/bios/isolinux.bin" "${isofs_dir}/syslinux/"
809     install -m 0644 -- "${airootfs_dir}/usr/lib/syslinux/bios/isohdpfx.bin" "${isofs_dir}/syslinux/"
810
811     return 0
812 }
813
814 # Prepare /EFI
815 make_efi() {
816     local _bootfile _use_config_name="nosplash" _efi_config_list=() _efi_config
817     [[ "${boot_splash}" = true ]] && _use_config_name="splash"
818     _bootfile="$(basename "$(ls "${airootfs_dir}/usr/lib/systemd/boot/efi/systemd-boot"*".efi" )")"
819
820     install -d -m 0755 -- "${isofs_dir}/EFI/boot"
821     install -m 0644 -- "${airootfs_dir}/usr/lib/systemd/boot/efi/${_bootfile}" "${isofs_dir}/EFI/boot/${_bootfile#systemd-}"
822
823     install -d -m 0755 -- "${isofs_dir}/loader/entries"
824     sed "s|%ARCH%|${arch}|g;" "${script_path}/efiboot/${_use_config_name}/loader.conf" > "${isofs_dir}/loader/loader.conf"
825
826     readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f -name "archiso-usb*.conf" -printf "%f\n" | grep -v "rescue")
827     [[ "${norescue_entry}" = false ]] && readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f  -name "archiso-usb*rescue*.conf" -printf "%f\n")
828
829     for _efi_config in "${_efi_config_list[@]}"; do
830         sed "s|%ARCHISO_LABEL%|${iso_label}|g;
831             s|%OS_NAME%|${os_name}|g;
832             s|%KERNEL_FILENAME%|${kernel_filename}|g;
833             s|%ARCH%|${arch}|g;
834             s|%INSTALL_DIR%|${install_dir}|g" \
835         "${script_path}/efiboot/${_use_config_name}/${_efi_config}" > "${isofs_dir}/loader/entries/$(basename "${_efi_config}" | sed "s|usb|${arch}|g")"
836     done
837
838     # edk2-shell based UEFI shell
839     local _efi_shell_arch
840     if [[ -d "${airootfs_dir}/usr/share/edk2-shell" ]]; then
841         for _efi_shell_arch in $(find "${airootfs_dir}/usr/share/edk2-shell" -mindepth 1 -maxdepth 1 -type d -print0 | xargs -0 -I{} basename {}); do
842             if [[ -f "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell_Full.efi" ]]; then
843                 cp "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell_Full.efi" "${isofs_dir}/EFI/shell_${_efi_shell_arch}.efi"
844             elif [[ -f "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell.efi" ]]; then
845                 cp "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell.efi" "${isofs_dir}/EFI/shell_${_efi_shell_arch}.efi"
846             else
847                 continue
848             fi
849             echo -e "title  UEFI Shell ${_efi_shell_arch}\nefi    /EFI/shell_${_efi_shell_arch}.efi" > "${isofs_dir}/loader/entries/uefi-shell-${_efi_shell_arch}.conf"
850         done
851     fi
852
853     return 0
854 }
855
856 # Prepare efiboot.img::/EFI for "El Torito" EFI boot mode
857 make_efiboot() {
858     truncate -s 128M "${build_dir}/efiboot.img"
859     mkfs.fat -n ARCHISO_EFI "${build_dir}/efiboot.img"
860
861     mkdir -p "${build_dir}/efiboot"
862     mount "${build_dir}/efiboot.img" "${build_dir}/efiboot"
863
864     mkdir -p "${build_dir}/efiboot/EFI/alteriso/${arch}" "${build_dir}/efiboot/EFI/boot" "${build_dir}/efiboot/loader/entries"
865     cp "${isofs_dir}/${install_dir}/boot/${arch}/${kernel_filename}" "${build_dir}/efiboot/EFI/alteriso/${arch}/${kernel_filename}.efi"
866     cp "${isofs_dir}/${install_dir}/boot/${arch}/archiso.img" "${build_dir}/efiboot/EFI/alteriso/${arch}/archiso.img"
867
868     local _ucode_image _efi_config _use_config_name="nosplash" _bootfile
869     for _ucode_image in "${airootfs_dir}/boot/"{intel-uc.img,intel-ucode.img,amd-uc.img,amd-ucode.img,early_ucode.cpio,microcode.cpio}; do
870         [[ -e "${_ucode_image}" ]] && cp "${_ucode_image}" "${build_dir}/efiboot/EFI/alteriso/"
871     done
872
873     cp "${airootfs_dir}/usr/share/efitools/efi/HashTool.efi" "${build_dir}/efiboot/EFI/boot/"
874
875     _bootfile="$(basename "$(ls "${airootfs_dir}/usr/lib/systemd/boot/efi/systemd-boot"*".efi" )")"
876     cp "${airootfs_dir}/usr/lib/systemd/boot/efi/${_bootfile}" "${build_dir}/efiboot/EFI/boot/${_bootfile#systemd-}"
877
878     [[ "${boot_splash}" = true ]] && _use_config_name="splash"
879     sed "s|%ARCH%|${arch}|g;" "${script_path}/efiboot/${_use_config_name}/loader.conf" > "${build_dir}/efiboot/loader/loader.conf"
880
881     find "${isofs_dir}/loader/entries/" -maxdepth 1 -mindepth 1 -name "uefi-shell*" -type f -printf "%p\0" | xargs -0 -I{} cp {} "${build_dir}/efiboot/loader/entries/"
882
883     readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f -name "archiso-cd*.conf" -printf "%f\n" | grep -v "rescue")
884     [[ "${norescue_entry}" = false ]] && readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f  -name "archiso-cd*rescue*.conf" -printf "%f\n")
885
886     for _efi_config in "${_efi_config_list[@]}"; do
887         sed "s|%ARCHISO_LABEL%|${iso_label}|g;
888             s|%OS_NAME%|${os_name}|g;
889             s|%KERNEL_FILENAME%|${kernel_filename}|g;
890             s|%ARCH%|${arch}|g;
891             s|%INSTALL_DIR%|${install_dir}|g" \
892         "${script_path}/efiboot/${_use_config_name}/${_efi_config}" > "${build_dir}/efiboot/loader/entries/$(basename "${_efi_config}" | sed "s|cd|${arch}|g")"
893     done
894
895     find "${isofs_dir}/EFI" -maxdepth 1 -mindepth 1 -name "shell*.efi" -printf "%p\0" | xargs -0 -I{} cp {} "${build_dir}/efiboot/EFI/"
896     umount -d "${build_dir}/efiboot"
897
898     return 0
899 }
900
901 # Compress tarball
902 make_tarball() {
903     # backup airootfs.img for tarball
904     msg_info "Copying airootfs.img ..."
905     cp "${airootfs_dir}.img" "${airootfs_dir}.img.org"
906
907     # Run script
908     mount_airootfs
909     [[ -f "${airootfs_dir}/root/optimize_for_tarball.sh" ]] && _chroot_run "bash /root/optimize_for_tarball.sh -u ${username}"
910
911     _cleanup_common
912     _chroot_run "mkinitcpio -P"
913
914     remove "${airootfs_dir}/root/optimize_for_tarball.sh"
915
916     mkdir -p "${out_dir}"
917     msg_info "Creating tarball..."
918     cd -- "${airootfs_dir}"
919     tar -v -J -p -c -f "${out_dir}/${tar_filename}" ./*
920     cd -- "${OLDPWD}"
921
922     _mkchecksum "${out_dir}/${tar_filename}"
923     msg_info "Done! | $(ls -sh "${out_dir}/${tar_filename}")"
924
925     remove "${airootfs_dir}.img"
926     mv "${airootfs_dir}.img.org" "${airootfs_dir}.img"
927
928     [[ "${noiso}" = true ]] && msg_info "The password for the live user and root is ${password}."
929     
930     return 0
931 }
932
933
934 # Build airootfs filesystem image
935 make_prepare() {
936     mount_airootfs
937
938     # Create packages list
939     msg_info "Creating a list of installed packages on live-enviroment..."
940     pacman-key --init
941     pacman -Q --sysroot "${airootfs_dir}" | tee "${isofs_dir}/${install_dir}/pkglist.${arch}.txt" "${build_dir}/packages-full.list" > /dev/null
942
943     # Cleanup
944     remove "${airootfs_dir}/root/optimize_for_tarball.sh"
945     _cleanup_airootfs
946
947     # Create squashfs
948     mkdir -p -- "${isofs_dir}/${install_dir}/${arch}"
949     msg_info "Creating SquashFS image, this may take some time..."
950     mksquashfs "${airootfs_dir}" "${build_dir}/iso/${install_dir}/${arch}/airootfs.sfs" -noappend -comp "${sfs_comp}" "${sfs_comp_opt[@]}"
951
952     # Create checksum
953     msg_info "Creating checksum file for self-test..."
954     echo "$(sha512sum "${isofs_dir}/${install_dir}/${arch}/airootfs.sfs" | getclm 1) airootfs.sfs" > "${isofs_dir}/${install_dir}/${arch}/airootfs.sha512"
955     msg_info "Done!"
956
957     # Sign with gpg
958     if [[ -v gpg_key ]] && (( "${#gpg_key}" != 0 )); then
959         msg_info "Creating signature file ($gpg_key) ..."
960         cd -- "${isofs_dir}/${install_dir}/${arch}"
961         gpg --detach-sign --default-key "${gpg_key}" "airootfs.sfs"
962         cd -- "${OLDPWD}"
963         msg_info "Done!"
964     fi
965
966     umount_work
967
968     [[ "${cleaning}" = true ]] && remove "${airootfs_dir}" "${airootfs_dir}.img"
969
970     return 0
971 }
972
973 make_alteriso_info(){
974     # iso version info
975     if [[ "${include_info}" = true ]]; then
976         local _info_file="${isofs_dir}/alteriso-info" _version="${iso_version}"
977         remove "${_info_file}"; touch "${_info_file}"
978         [[ -d "${script_path}/.git" ]] && [[ "${gitversion}" = false ]] && _version="${iso_version}-${gitrev}"
979         "${tools_dir}/alteriso-info.sh" -a "${arch}" -b "${boot_splash}" -c "${channel_name%.add}" -d "${iso_publisher}" -k "${kernel}" -o "${os_name}" -p "${password}" -u "${username}" -v "${_version}" > "${_info_file}"
980     fi
981
982     return 0
983 }
984
985 # Add files to the root of isofs
986 make_overisofs() {
987     local _over_isofs_list _isofs
988     _over_isofs_list=("${channel_dir}/over_isofs.any""${channel_dir}/over_isofs.${arch}")
989     for_module '_over_isofs_list+=("${module_dir}/{}/over_isofs.any""${module_dir}/{}/over_isofs.${arch}")'
990     for _isofs in "${_over_isofs_list[@]}"; do
991         [[ -d "${_isofs}" ]] && cp -af "${_isofs}"/* "${isofs_dir}"
992     done
993
994     return 0
995 }
996
997 # Build ISO
998 make_iso() {
999     local _iso_efi_boot_args=()
1000     # If exists, add an EFI "El Torito" boot image (FAT filesystem) to ISO-9660 image.
1001     if [[ -f "${build_dir}/efiboot.img" ]]; then
1002         _iso_efi_boot_args=(-append_partition 2 C12A7328-F81F-11D2-BA4B-00A0C93EC93B "${build_dir}/efiboot.img" -appended_part_as_gpt -eltorito-alt-boot -e --interval:appended_partition_2:all:: -no-emul-boot -isohybrid-gpt-basdat)
1003     fi
1004
1005     mkdir -p -- "${out_dir}"
1006     msg_info "Creating ISO image..."
1007     xorriso -as mkisofs \
1008         -iso-level 3 \
1009         -full-iso9660-filenames \
1010         -joliet \
1011         -joliet-long \
1012         -rational-rock \
1013         -volid "${iso_label}" \
1014         -appid "${iso_application}" \
1015         -publisher "${iso_publisher}" \
1016         -preparer "prepared by AlterISO" \
1017         -eltorito-boot syslinux/isolinux.bin \
1018         -eltorito-catalog syslinux/boot.cat \
1019         -no-emul-boot -boot-load-size 4 -boot-info-table \
1020         -isohybrid-mbr "${build_dir}/iso/syslinux/isohdpfx.bin" \
1021         "${_iso_efi_boot_args[@]}" \
1022         -output "${out_dir}/${iso_filename}" \
1023         "${build_dir}/iso/"
1024     _mkchecksum "${out_dir}/${iso_filename}"
1025     msg_info "Done! | $(ls -sh -- "${out_dir}/${iso_filename}")"
1026
1027     msg_info "The password for the live user and root is ${password}."
1028
1029     return 0
1030 }
1031
1032
1033 # Parse options
1034 OPTS=("a:" "b" "c:" "d" "e" "g:" "h" "j" "k:" "l:" "o:" "p:" "r" "t:" "u:" "w:" "x")
1035 OPTL=("arch:" "boot-splash" "comp-type:" "debug" "cleaning" "cleanup" "gpgkey:" "help" "lang:" "japanese" "kernel:" "out:" "password:" "comp-opts:" "user:" "work:" "bash-debug" "nocolor" "noconfirm" "nodepend" "gitversion" "msgdebug" "noloopmod" "tarball" "noiso" "noaur" "nochkver" "channellist" "config:" "noefi" "nodebug" "nosigcheck" "normwork" "log" "logpath:" "nolog" "nopkgbuild" "pacman-debug" "confirm")
1036 if ! OPT=$(getopt -o "$(printf "%s," "${OPTS[@]}")" -l "$(printf "%s," "${OPTL[@]}")" --  "${@}" "${DEFAULT_ARGUMENT[@]}"); then
1037     exit 1
1038 fi
1039
1040 eval set -- "${OPT}"
1041 msg_debug "Argument: ${OPT}"
1042 unset OPT OPTS OPTL DEFAULT_ARGUMENT
1043
1044 while true; do
1045     case "${1}" in
1046         -a | --arch)
1047             arch="${2}"
1048             shift 2
1049             ;;
1050         -b | --boot-splash)
1051             boot_splash=true
1052             shift 1
1053             ;;
1054         -c | --comp-type)
1055             case "${2}" in
1056                 "gzip" | "lzma" | "lzo" | "lz4" | "xz" | "zstd") sfs_comp="${2}" ;;
1057                 *) msg_error "Invaild compressors '${2}'" '1' ;;
1058             esac
1059             shift 2
1060             ;;
1061         -d | --debug)
1062             debug=true
1063             shift 1
1064             ;;
1065         -e | --cleaning | --cleanup)
1066             cleaning=true
1067             shift 1
1068             ;;
1069         -g | --gpgkey)
1070             gpg_key="${2}"
1071             shift 2
1072             ;;
1073         -h | --help)
1074             _usage
1075             exit 0
1076             ;;
1077         -j | --japanese)
1078             msg_error "This option is obsolete in AlterISO 3. To use Japanese, use \"-l ja\"." "1"
1079             ;;
1080         -k | --kernel)
1081             customized_kernel=true
1082             kernel="${2}"
1083             shift 2
1084             ;;
1085         -l | --lang)
1086             locale_name="${2}"
1087             shift 2
1088             ;;
1089         -o | --out)
1090             out_dir="${2}"
1091             shift 2
1092             ;;
1093         -p | --password)
1094             customized_password=true
1095             password="${2}"
1096             shift 2
1097             ;;
1098         -r | --tarball)
1099             tarball=true
1100             shift 1
1101             ;;
1102         -t | --comp-opts)
1103             if [[ "${2}" = "reset" ]]; then
1104                 sfs_comp_opt=()
1105             else
1106                 sfs_comp_opt=(${2})
1107             fi
1108             shift 2
1109             ;;
1110         -u | --user)
1111             customized_username=true
1112             username="$(echo -n "${2}" | sed 's/ //g' | tr '[:upper:]' '[:lower:]')"
1113             shift 2
1114             ;;
1115         -w | --work)
1116             work_dir="${2}"
1117             shift 2
1118             ;;
1119         -x | --bash-debug)
1120             debug=true
1121             bash_debug=true
1122             shift 1
1123             ;;
1124         --noconfirm)
1125             noconfirm=true
1126             shift 1
1127             ;;
1128         --confirm)
1129             noconfirm=false
1130             shift 1
1131             ;;
1132         --nodepend)
1133             nodepend=true
1134             shift 1
1135             ;;
1136         --nocolor)
1137             nocolor=true
1138             shift 1
1139             ;;
1140         --gitversion)
1141             if [[ -d "${script_path}/.git" ]]; then
1142                 gitversion=true
1143             else
1144                 msg_error "There is no git directory. You need to use git clone to use this feature." "1"
1145             fi
1146             shift 1
1147             ;;
1148         --msgdebug)
1149             msgdebug=true;
1150             shift 1
1151             ;;
1152         --noloopmod)
1153             noloopmod=true
1154             shift 1
1155             ;;
1156         --noiso)
1157             noiso=true
1158             shift 1
1159             ;;
1160         --noaur)
1161             noaur=true
1162             shift 1
1163             ;;
1164         --nochkver)
1165             nochkver=true
1166             shift 1
1167             ;;
1168         --nodebug)
1169             debug=false
1170             msgdebug=false
1171             bash_debug=false
1172             shift 1
1173             ;;
1174         --noefi)
1175             noefi=true
1176             shift 1
1177             ;;
1178         --channellist)
1179             show_channel_list
1180             exit 0
1181             ;;
1182         --config)
1183             source "${2}"
1184             shift 2
1185             ;;
1186         --pacman-debug)
1187             pacman_debug=true
1188             shift 1
1189             ;;
1190         --nosigcheck)
1191             nosigcheck=true
1192             shift 1
1193             ;;
1194         --normwork)
1195             normwork=true
1196             shift 1
1197             ;;
1198         --log)
1199             logging=true
1200             shift 1
1201             ;;
1202         --logpath)
1203             logging="${2}"
1204             customized_logpath=true
1205             shift 2
1206             ;;
1207         --nolog)
1208             logging=false
1209             shift 1
1210             ;;
1211         --nopkgbuild)
1212             nopkgbuild=true
1213             shift 1
1214             ;;
1215         --)
1216             shift
1217             break
1218             ;;
1219         *)
1220             msg_error "Invalid argument '${1}'"
1221             _usage 1
1222             ;;
1223     esac
1224 done
1225
1226 # Check root.
1227 if (( ! "${EUID}" == 0 )); then
1228     msg_warn "This script must be run as root." >&2
1229     msg_warn "Re-run 'sudo ${0} ${ARGUMENT[*]}'"
1230     sudo "${0}" "${ARGUMENT[@]}"
1231     exit "${?}"
1232 fi
1233
1234 # Show config message
1235 msg_debug "Use the default configuration file (${defaultconfig})."
1236 [[ -f "${script_path}/custom.conf" ]] && msg_debug "The default settings have been overridden by custom.conf"
1237
1238 # Debug mode
1239 [[ "${bash_debug}" = true ]] && set -x -v
1240
1241 # Check for a valid channel name
1242 if [[ -n "${1+SET}" ]]; then
1243     case "$(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" -n check "${1}"; printf "%d" "${?}")" in
1244         "2")
1245             msg_error "Invalid channel ${1}" "1"
1246             ;;
1247         "1")
1248             channel_dir="${1}"
1249             channel_name="$(basename "${1%/}")"
1250             ;;
1251         "0")
1252             channel_dir="${script_path}/channels/${1}"
1253             channel_name="${1}"
1254             ;;
1255     esac
1256 else
1257     channel_dir="${script_path}/channels/${channel_name}"
1258 fi
1259
1260 # Set vars
1261 build_dir="${work_dir}/build/${arch}"
1262 cache_dir="${work_dir}/cache/${arch}"
1263 airootfs_dir="${build_dir}/airootfs"
1264 isofs_dir="${build_dir}/iso"
1265 lockfile_dir="${build_dir}/lockfile"
1266 gitrev="$(cd "${script_path}"; git rev-parse --short HEAD)"
1267
1268 # Create dir
1269 for _dir in "${build_dir}" "${cache_dir}" "${airootfs_dir}" "${isofs_dir}" "${lockfile_dir}"; do
1270     mkdir -p "${_dir}"
1271 done
1272 ork_dir="$(realpath "${work_dir}")"
1273
1274 # Set for special channels
1275 if [[ -d "${channel_dir}.add" ]]; then
1276     channel_name="${1}"
1277     channel_dir="${channel_dir}.add"
1278 elif [[ "${channel_name}" = "clean" ]]; then
1279    _run_cleansh
1280     exit 0
1281 fi
1282
1283 # Check channel version
1284 msg_debug "channel path is ${channel_dir}"
1285 if [[ ! "$(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" ver "${channel_name}" | cut -d "." -f 1)" = "$(echo "${alteriso_version}" | cut -d "." -f 1)" ]] && [[ "${nochkver}" = false ]]; then
1286     msg_error "This channel does not support Alter ISO 3."
1287     if [[ -d "${script_path}/.git" ]]; then
1288         msg_error "Please run \"git checkout alteriso-2\"" "1"
1289     else
1290         msg_error "Please download old version here.\nhttps://github.com/FascodeNet/alterlinux/releases" "1"
1291     fi
1292 fi
1293
1294 set -eu
1295
1296 prepare_env
1297 prepare_build
1298 show_settings
1299 run_once make_pacman_conf
1300 run_once make_basefs # Mount airootfs
1301 run_once make_packages_repo
1302 [[ "${noaur}" = false ]] && run_once make_packages_aur
1303 [[ "${nopkgbuild}" = false ]] && run_once make_pkgbuild
1304 run_once make_customize_airootfs
1305 run_once make_setup_mkinitcpio
1306 [[ "${tarball}" = true ]] && run_once make_tarball
1307 if [[ "${noiso}" = false ]]; then
1308     run_once make_syslinux
1309     run_once make_isolinux
1310     run_once make_boot
1311     run_once make_boot_extra
1312     if [[ "${noefi}" = false ]]; then
1313         run_once make_efi
1314         run_once make_efiboot
1315     fi
1316     run_once make_alteriso_info
1317     run_once make_prepare
1318     run_once make_overisofs
1319     run_once make_iso
1320 fi
1321
1322 [[ "${cleaning}" = true ]] && _run_cleansh
1323
1324 exit 0