Send patches - preferably formatted by git format-patch - to patches at archlinux32 dot org.
summaryrefslogtreecommitdiff
path: root/archinstall/scripts
diff options
context:
space:
mode:
Diffstat (limited to 'archinstall/scripts')
-rw-r--r--archinstall/scripts/__init__.py0
-rw-r--r--archinstall/scripts/guided.py247
-rw-r--r--archinstall/scripts/list.py10
-rw-r--r--archinstall/scripts/minimal.py106
-rw-r--r--archinstall/scripts/only_hd.py78
-rw-r--r--archinstall/scripts/swiss.py308
-rw-r--r--archinstall/scripts/unattended.py19
7 files changed, 768 insertions, 0 deletions
diff --git a/archinstall/scripts/__init__.py b/archinstall/scripts/__init__.py
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/archinstall/scripts/__init__.py
diff --git a/archinstall/scripts/guided.py b/archinstall/scripts/guided.py
new file mode 100644
index 00000000..385ff500
--- /dev/null
+++ b/archinstall/scripts/guided.py
@@ -0,0 +1,247 @@
+from pathlib import Path
+from typing import Any, TYPE_CHECKING, Optional
+
+import archinstall
+from archinstall import info, debug
+from archinstall import SysInfo
+from archinstall.lib import locale
+from archinstall.lib import disk
+from archinstall.lib.global_menu import GlobalMenu
+from archinstall.lib.configuration import ConfigurationOutput
+from archinstall.lib.installer import Installer
+from archinstall.lib.menu import Menu
+from archinstall.lib.models import AudioConfiguration
+from archinstall.lib.models.bootloader import Bootloader
+from archinstall.lib.models.network_configuration import NetworkConfiguration
+from archinstall.lib.profile.profiles_handler import profile_handler
+
+if TYPE_CHECKING:
+ _: Any
+
+
+if archinstall.arguments.get('help'):
+ print("See `man archinstall` for help.")
+ exit(0)
+
+
+def ask_user_questions():
+ """
+ First, we'll ask the user for a bunch of user input.
+ Not until we're satisfied with what we want to install
+ will we continue with the actual installation steps.
+ """
+
+ # ref: https://github.com/archlinux/archinstall/pull/831
+ # we'll set NTP to true by default since this is also
+ # the default value specified in the menu options; in
+ # case it will be changed by the user we'll also update
+ # the system immediately
+ global_menu = GlobalMenu(data_store=archinstall.arguments)
+
+ global_menu.enable('archinstall-language')
+
+ # Set which region to download packages from during the installation
+ global_menu.enable('mirror_config')
+
+ global_menu.enable('locale_config')
+
+ global_menu.enable('disk_config', mandatory=True)
+
+ # Specify disk encryption options
+ global_menu.enable('disk_encryption')
+
+ # Ask which boot-loader to use (will only ask if we're in UEFI mode, otherwise will default to GRUB)
+ global_menu.enable('bootloader')
+
+ global_menu.enable('uki')
+
+ global_menu.enable('swap')
+
+ # Get the hostname for the machine
+ global_menu.enable('hostname')
+
+ # Ask for a root password (optional, but triggers requirement for super-user if skipped)
+ global_menu.enable('!root-password', mandatory=True)
+
+ global_menu.enable('!users', mandatory=True)
+
+ # Ask for archinstall-specific profiles_bck (such as desktop environments etc)
+ global_menu.enable('profile_config')
+
+ # Ask about audio server selection if one is not already set
+ global_menu.enable('audio_config')
+
+ # Ask for preferred kernel:
+ global_menu.enable('kernels', mandatory=True)
+
+ global_menu.enable('packages')
+
+ if archinstall.arguments.get('advanced', False):
+ # Enable parallel downloads
+ global_menu.enable('parallel downloads')
+
+ # Ask or Call the helper function that asks the user to optionally configure a network.
+ global_menu.enable('network_config')
+
+ global_menu.enable('timezone')
+
+ global_menu.enable('ntp')
+
+ global_menu.enable('additional-repositories')
+
+ global_menu.enable('__separator__')
+
+ global_menu.enable('save_config')
+ global_menu.enable('install')
+ global_menu.enable('abort')
+
+ global_menu.run()
+
+
+def perform_installation(mountpoint: Path):
+ """
+ Performs the installation steps on a block device.
+ Only requirement is that the block devices are
+ formatted and setup prior to entering this function.
+ """
+ info('Starting installation...')
+ disk_config: disk.DiskLayoutConfiguration = archinstall.arguments['disk_config']
+
+ # Retrieve list of additional repositories and set boolean values appropriately
+ enable_testing = 'testing' in archinstall.arguments.get('additional-repositories', [])
+ enable_multilib = 'multilib' in archinstall.arguments.get('additional-repositories', [])
+ run_mkinitcpio = not archinstall.arguments.get('uki')
+ locale_config: locale.LocaleConfiguration = archinstall.arguments['locale_config']
+ disk_encryption: disk.DiskEncryption = archinstall.arguments.get('disk_encryption', None)
+
+ with Installer(
+ mountpoint,
+ disk_config,
+ disk_encryption=disk_encryption,
+ kernels=archinstall.arguments.get('kernels', ['linux'])
+ ) as installation:
+ # Mount all the drives to the desired mountpoint
+ if disk_config.config_type != disk.DiskLayoutType.Pre_mount:
+ installation.mount_ordered_layout()
+
+ installation.sanity_check()
+
+ if disk_config.config_type != disk.DiskLayoutType.Pre_mount:
+ if disk_encryption and disk_encryption.encryption_type != disk.EncryptionType.NoEncryption:
+ # generate encryption key files for the mounted luks devices
+ installation.generate_key_files()
+
+ if mirror_config := archinstall.arguments.get('mirror_config', None):
+ installation.set_mirrors(mirror_config, on_target=False)
+
+ installation.minimal_installation(
+ testing=enable_testing,
+ multilib=enable_multilib,
+ mkinitcpio=run_mkinitcpio,
+ hostname=archinstall.arguments.get('hostname', 'archlinux'),
+ locale_config=locale_config
+ )
+
+ if mirror_config := archinstall.arguments.get('mirror_config', None):
+ installation.set_mirrors(mirror_config, on_target=True)
+
+ if archinstall.arguments.get('swap'):
+ installation.setup_swap('zram')
+
+ if archinstall.arguments.get("bootloader") == Bootloader.Grub and SysInfo.has_uefi():
+ installation.add_additional_packages("grub")
+
+ installation.add_bootloader(
+ archinstall.arguments["bootloader"],
+ archinstall.arguments.get('uki', False)
+ )
+
+ # If user selected to copy the current ISO network configuration
+ # Perform a copy of the config
+ network_config: Optional[NetworkConfiguration] = archinstall.arguments.get('network_config', None)
+
+ if network_config:
+ network_config.install_network_config(
+ installation,
+ archinstall.arguments.get('profile_config', None)
+ )
+
+ if users := archinstall.arguments.get('!users', None):
+ installation.create_users(users)
+
+ audio_config: Optional[AudioConfiguration] = archinstall.arguments.get('audio_config', None)
+ if audio_config:
+ audio_config.install_audio_config(installation)
+ else:
+ info("No audio server will be installed")
+
+ if archinstall.arguments.get('packages', None) and archinstall.arguments.get('packages', None)[0] != '':
+ installation.add_additional_packages(archinstall.arguments.get('packages', None))
+
+ if profile_config := archinstall.arguments.get('profile_config', None):
+ profile_handler.install_profile_config(installation, profile_config)
+
+ if timezone := archinstall.arguments.get('timezone', None):
+ installation.set_timezone(timezone)
+
+ if archinstall.arguments.get('ntp', False):
+ installation.activate_time_synchronization()
+
+ if archinstall.accessibility_tools_in_use():
+ installation.enable_espeakup()
+
+ if (root_pw := archinstall.arguments.get('!root-password', None)) and len(root_pw):
+ installation.user_set_pw('root', root_pw)
+
+ if profile_config := archinstall.arguments.get('profile_config', None):
+ profile_config.profile.post_install(installation)
+
+ # If the user provided a list of services to be enabled, pass the list to the enable_service function.
+ # Note that while it's called enable_service, it can actually take a list of services and iterate it.
+ if archinstall.arguments.get('services', None):
+ installation.enable_service(archinstall.arguments.get('services', []))
+
+ # If the user provided custom commands to be run post-installation, execute them now.
+ if archinstall.arguments.get('custom-commands', None):
+ archinstall.run_custom_user_commands(archinstall.arguments['custom-commands'], installation)
+
+ installation.genfstab()
+
+ info("For post-installation tips, see https://wiki.archlinux.org/index.php/Installation_guide#Post-installation")
+
+ if not archinstall.arguments.get('silent'):
+ prompt = str(_('Would you like to chroot into the newly created installation and perform post-installation configuration?'))
+ choice = Menu(prompt, Menu.yes_no(), default_option=Menu.yes()).run()
+ if choice.value == Menu.yes():
+ try:
+ installation.drop_to_shell()
+ except:
+ pass
+
+ debug(f"Disk states after installing: {disk.disk_layouts()}")
+
+
+if not archinstall.arguments.get('silent'):
+ ask_user_questions()
+
+config_output = ConfigurationOutput(archinstall.arguments)
+
+if not archinstall.arguments.get('silent'):
+ config_output.show()
+
+config_output.save()
+
+if archinstall.arguments.get('dry_run'):
+ exit(0)
+
+if not archinstall.arguments.get('silent'):
+ input(str(_('Press Enter to continue.')))
+
+fs_handler = disk.FilesystemHandler(
+ archinstall.arguments['disk_config'],
+ archinstall.arguments.get('disk_encryption', None)
+)
+
+fs_handler.perform_filesystem_operations()
+
+perform_installation(archinstall.storage.get('MOUNT_POINT', Path('/mnt')))
diff --git a/archinstall/scripts/list.py b/archinstall/scripts/list.py
new file mode 100644
index 00000000..0e0363a1
--- /dev/null
+++ b/archinstall/scripts/list.py
@@ -0,0 +1,10 @@
+import glob
+import pathlib
+
+print("The following are viable --script options:")
+
+for script in [pathlib.Path(x) for x in glob.glob(f"{pathlib.Path(__file__).parent}/*.py")]:
+ if script.stem in ['__init__', 'list']:
+ continue
+
+ print(f" {script.stem}") \ No newline at end of file
diff --git a/archinstall/scripts/minimal.py b/archinstall/scripts/minimal.py
new file mode 100644
index 00000000..f6650ff8
--- /dev/null
+++ b/archinstall/scripts/minimal.py
@@ -0,0 +1,106 @@
+from pathlib import Path
+from typing import TYPE_CHECKING, Any, List
+
+import archinstall
+from archinstall import info
+from archinstall import Installer, ConfigurationOutput
+from archinstall.default_profiles.minimal import MinimalProfile
+from archinstall.lib.interactions import suggest_single_disk_layout, select_devices
+from archinstall.lib.models import Bootloader, User
+from archinstall.lib.profile import ProfileConfiguration, profile_handler
+from archinstall.lib import disk
+
+if TYPE_CHECKING:
+ _: Any
+
+
+info("Minimal only supports:")
+info(" * Being installed to a single disk")
+
+if archinstall.arguments.get('help', None):
+ info(" - Optional disk encryption via --!encryption-password=<password>")
+ info(" - Optional filesystem type via --filesystem=<fs type>")
+ info(" - Optional systemd network via --network")
+
+
+def perform_installation(mountpoint: Path):
+ disk_config: disk.DiskLayoutConfiguration = archinstall.arguments['disk_config']
+ disk_encryption: disk.DiskEncryption = archinstall.arguments.get('disk_encryption', None)
+
+ with Installer(
+ mountpoint,
+ disk_config,
+ disk_encryption=disk_encryption,
+ kernels=archinstall.arguments.get('kernels', ['linux'])
+ ) as installation:
+ # Strap in the base system, add a boot loader and configure
+ # some other minor details as specified by this profile and user.
+ if installation.minimal_installation():
+ installation.set_hostname('minimal-arch')
+ installation.add_bootloader(Bootloader.Systemd)
+
+ # Optionally enable networking:
+ if archinstall.arguments.get('network', None):
+ installation.copy_iso_network_config(enable_services=True)
+
+ installation.add_additional_packages(['nano', 'wget', 'git'])
+
+ profile_config = ProfileConfiguration(MinimalProfile())
+ profile_handler.install_profile_config(installation, profile_config)
+
+ user = User('devel', 'devel', False)
+ installation.create_users(user)
+
+ # Once this is done, we output some useful information to the user
+ # And the installation is complete.
+ info("There are two new accounts in your installation after reboot:")
+ info(" * root (password: airoot)")
+ info(" * devel (password: devel)")
+
+
+def prompt_disk_layout():
+ fs_type = None
+ if filesystem := archinstall.arguments.get('filesystem', None):
+ fs_type = disk.FilesystemType(filesystem)
+
+ devices = select_devices()
+ modifications = suggest_single_disk_layout(devices[0], filesystem_type=fs_type)
+
+ archinstall.arguments['disk_config'] = disk.DiskLayoutConfiguration(
+ config_type=disk.DiskLayoutType.Default,
+ device_modifications=[modifications]
+ )
+
+
+def parse_disk_encryption():
+ if enc_password := archinstall.arguments.get('!encryption-password', None):
+ modification: List[disk.DeviceModification] = archinstall.arguments['disk_config']
+ partitions: List[disk.PartitionModification] = []
+
+ # encrypt all partitions except the /boot
+ for mod in modification:
+ partitions += list(filter(lambda x: x.mountpoint != Path('/boot'), mod.partitions))
+
+ archinstall.arguments['disk_encryption'] = disk.DiskEncryption(
+ encryption_type=disk.EncryptionType.Luks,
+ encryption_password=enc_password,
+ partitions=partitions
+ )
+
+
+prompt_disk_layout()
+parse_disk_encryption()
+
+config_output = ConfigurationOutput(archinstall.arguments)
+config_output.show()
+
+input(str(_('Press Enter to continue.')))
+
+fs_handler = disk.FilesystemHandler(
+ archinstall.arguments['disk_config'],
+ archinstall.arguments.get('disk_encryption', None)
+)
+
+fs_handler.perform_filesystem_operations()
+
+perform_installation(archinstall.storage.get('MOUNT_POINT', Path('/mnt')))
diff --git a/archinstall/scripts/only_hd.py b/archinstall/scripts/only_hd.py
new file mode 100644
index 00000000..2293cbee
--- /dev/null
+++ b/archinstall/scripts/only_hd.py
@@ -0,0 +1,78 @@
+from pathlib import Path
+
+import archinstall
+from archinstall import debug
+from archinstall.lib.installer import Installer
+from archinstall.lib.configuration import ConfigurationOutput
+from archinstall.lib import disk
+
+
+def ask_user_questions():
+ global_menu = archinstall.GlobalMenu(data_store=archinstall.arguments)
+
+ global_menu.enable('archinstall-language')
+
+ global_menu.enable('disk_config', mandatory=True)
+ global_menu.enable('disk_encryption')
+ global_menu.enable('swap')
+
+ global_menu.enable('save_config')
+ global_menu.enable('install')
+ global_menu.enable('abort')
+
+ global_menu.run()
+
+
+def perform_installation(mountpoint: Path):
+ """
+ Performs the installation steps on a block device.
+ Only requirement is that the block devices are
+ formatted and setup prior to entering this function.
+ """
+ disk_config: disk.DiskLayoutConfiguration = archinstall.arguments['disk_config']
+ disk_encryption: disk.DiskEncryption = archinstall.arguments.get('disk_encryption', None)
+
+ with Installer(
+ mountpoint,
+ disk_config,
+ disk_encryption=disk_encryption,
+ kernels=archinstall.arguments.get('kernels', ['linux'])
+ ) as installation:
+ # Mount all the drives to the desired mountpoint
+ # This *can* be done outside of the installation, but the installer can deal with it.
+ if archinstall.arguments.get('disk_config'):
+ installation.mount_ordered_layout()
+
+ # to generate a fstab directory holder. Avoids an error on exit and at the same time checks the procedure
+ target = Path(f"{mountpoint}/etc/fstab")
+ if not target.parent.exists():
+ target.parent.mkdir(parents=True)
+
+ # For support reasons, we'll log the disk layout post installation (crash or no crash)
+ debug(f"Disk states after installing: {disk.disk_layouts()}")
+
+
+if not archinstall.arguments.get('silent'):
+ ask_user_questions()
+
+config_output = ConfigurationOutput(archinstall.arguments)
+if not archinstall.arguments.get('silent'):
+ config_output.show()
+
+config_output.save()
+
+if archinstall.arguments.get('dry_run'):
+ exit(0)
+
+if not archinstall.arguments.get('silent'):
+ input('Press Enter to continue.')
+
+fs_handler = disk.FilesystemHandler(
+ archinstall.arguments['disk_config'],
+ archinstall.arguments.get('disk_encryption', None)
+)
+
+fs_handler.perform_filesystem_operations()
+
+
+perform_installation(archinstall.storage.get('MOUNT_POINT', Path('/mnt')))
diff --git a/archinstall/scripts/swiss.py b/archinstall/scripts/swiss.py
new file mode 100644
index 00000000..8813fd91
--- /dev/null
+++ b/archinstall/scripts/swiss.py
@@ -0,0 +1,308 @@
+from enum import Enum
+from pathlib import Path
+from typing import TYPE_CHECKING, Any, Dict, Optional
+
+import archinstall
+from archinstall import SysInfo, info, debug
+from archinstall.lib import models
+from archinstall.lib import disk
+from archinstall.lib import locale
+from archinstall.lib.models import AudioConfiguration
+from archinstall.lib.profile.profiles_handler import profile_handler
+from archinstall.lib import menu
+from archinstall.lib.global_menu import GlobalMenu
+from archinstall.lib.installer import Installer
+from archinstall.lib.configuration import ConfigurationOutput
+
+if TYPE_CHECKING:
+ _: Any
+
+
+class ExecutionMode(Enum):
+ Full = 'full'
+ Lineal = 'lineal'
+ Only_HD = 'only-hd'
+ Only_OS = 'only-os'
+ Minimal = 'minimal'
+
+
+def select_mode() -> ExecutionMode:
+ options = [str(e.value) for e in ExecutionMode]
+ choice = menu.Menu(
+ str(_('Select an execution mode')),
+ options,
+ default_option=ExecutionMode.Full.value,
+ skip=False
+ ).run()
+
+ return ExecutionMode(choice.single_value)
+
+
+class SetupMenu(GlobalMenu):
+ def __init__(self, storage_area: Dict[str, Any]):
+ super().__init__(data_store=storage_area)
+
+ def setup_selection_menu_options(self):
+ super().setup_selection_menu_options()
+
+ self._menu_options['mode'] = menu.Selector(
+ 'Execution mode',
+ lambda x : select_mode(),
+ display_func=lambda x: x.value if x else '',
+ default=ExecutionMode.Full)
+
+ self._menu_options['continue'] = menu.Selector(
+ 'Continue',
+ exec_func=lambda n,v: True)
+
+ self.enable('archinstall-language')
+ self.enable('ntp')
+ self.enable('mode')
+ self.enable('continue')
+ self.enable('abort')
+
+ def exit_callback(self):
+ if self._data_store.get('mode', None):
+ archinstall.arguments['mode'] = self._data_store['mode']
+ info(f"Archinstall will execute under {archinstall.arguments['mode']} mode")
+
+
+class SwissMainMenu(GlobalMenu):
+ def __init__(
+ self,
+ data_store: Dict[str, Any],
+ exec_mode: ExecutionMode = ExecutionMode.Full
+ ):
+ self._execution_mode = exec_mode
+ super().__init__(data_store)
+
+ def setup_selection_menu_options(self):
+ super().setup_selection_menu_options()
+
+ options_list = []
+ mandatory_list = []
+
+ match self._execution_mode:
+ case ExecutionMode.Full | ExecutionMode.Lineal:
+ options_list = [
+ 'mirror_config', 'disk_config',
+ 'disk_encryption', 'swap', 'bootloader', 'hostname', '!root-password',
+ '!users', 'profile_config', 'audio_config', 'kernels', 'packages', 'additional-repositories', 'network_config',
+ 'timezone', 'ntp'
+ ]
+
+ if archinstall.arguments.get('advanced', False):
+ options_list.extend(['locale_config'])
+
+ mandatory_list = ['disk_config', 'bootloader', 'hostname']
+ case ExecutionMode.Only_HD:
+ options_list = ['disk_config', 'disk_encryption','swap']
+ mandatory_list = ['disk_config']
+ case ExecutionMode.Only_OS:
+ options_list = [
+ 'mirror_config','bootloader', 'hostname',
+ '!root-password', '!users', 'profile_config', 'audio_config', 'kernels',
+ 'packages', 'additional-repositories', 'network_config', 'timezone', 'ntp'
+ ]
+
+ mandatory_list = ['hostname']
+
+ if archinstall.arguments.get('advanced', False):
+ options_list += ['locale_config']
+ case ExecutionMode.Minimal:
+ pass
+ case _:
+ info(f' Execution mode {self._execution_mode} not supported')
+ exit(1)
+
+ if self._execution_mode != ExecutionMode.Lineal:
+ options_list += ['save_config', 'install']
+
+ if not archinstall.arguments.get('advanced', False):
+ options_list.append('archinstall-language')
+
+ options_list += ['abort']
+
+ for entry in mandatory_list:
+ self.enable(entry, mandatory=True)
+
+ for entry in options_list:
+ self.enable(entry)
+
+
+def ask_user_questions(exec_mode: ExecutionMode = ExecutionMode.Full):
+ """
+ First, we'll ask the user for a bunch of user input.
+ Not until we're satisfied with what we want to install
+ will we continue with the actual installation steps.
+ """
+ if archinstall.arguments.get('advanced', None):
+ setup_area: Dict[str, Any] = {}
+ setup = SetupMenu(setup_area)
+
+ if exec_mode == ExecutionMode.Lineal:
+ for entry in setup.list_enabled_options():
+ if entry in ('continue', 'abort'):
+ continue
+ if not setup.option(entry).enabled:
+ continue
+ setup.exec_option(entry)
+ else:
+ setup.run()
+
+ archinstall.arguments['archinstall-language'] = setup_area.get('archinstall-language')
+
+ with SwissMainMenu(data_store=archinstall.arguments, exec_mode=exec_mode) as menu:
+ if mode == ExecutionMode.Lineal:
+ for entry in menu.list_enabled_options():
+ if entry in ('install', 'abort'):
+ continue
+ menu.exec_option(entry)
+ archinstall.arguments[entry] = menu.option(entry).get_selection()
+ else:
+ menu.run()
+
+
+def perform_installation(mountpoint: Path, exec_mode: ExecutionMode):
+ disk_config: disk.DiskLayoutConfiguration = archinstall.arguments['disk_config']
+ disk_encryption: disk.DiskEncryption = archinstall.arguments.get('disk_encryption', None)
+
+ enable_testing = 'testing' in archinstall.arguments.get('additional-repositories', [])
+ enable_multilib = 'multilib' in archinstall.arguments.get('additional-repositories', [])
+ locale_config: locale.LocaleConfiguration = archinstall.arguments['locale_config']
+
+ with Installer(
+ mountpoint,
+ disk_config,
+ disk_encryption=disk_encryption,
+ kernels=archinstall.arguments.get('kernels', ['linux'])
+ ) as installation:
+ if exec_mode in [ExecutionMode.Full, ExecutionMode.Only_HD]:
+ installation.mount_ordered_layout()
+
+ installation.sanity_check()
+
+ if disk_config.config_type != disk.DiskLayoutType.Pre_mount:
+ if disk_encryption and disk_encryption.encryption_type != disk.EncryptionType.NoEncryption:
+ # generate encryption key files for the mounted luks devices
+ installation.generate_key_files()
+
+ if mirror_config := archinstall.arguments.get('mirror_config', None):
+ installation.set_mirrors(mirror_config)
+
+ installation.minimal_installation(
+ testing=enable_testing,
+ multilib=enable_multilib,
+ hostname=archinstall.arguments.get('hostname', 'archlinux'),
+ locale_config=locale_config
+ )
+
+ if mirror_config := archinstall.arguments.get('mirror_config', None):
+ installation.set_mirrors(mirror_config, on_target=True)
+
+ if archinstall.arguments.get('swap'):
+ installation.setup_swap('zram')
+
+ if archinstall.arguments.get("bootloader") == models.Bootloader.Grub and SysInfo.has_uefi():
+ installation.add_additional_packages("grub")
+
+ installation.add_bootloader(archinstall.arguments["bootloader"])
+
+ # If user selected to copy the current ISO network configuration
+ # Perform a copy of the config
+ network_config = archinstall.arguments.get('network_config', None)
+
+ if network_config:
+ network_config.install_network_config(
+ installation,
+ archinstall.arguments.get('profile_config', None)
+ )
+
+ if users := archinstall.arguments.get('!users', None):
+ installation.create_users(users)
+
+ audio_config: Optional[AudioConfiguration] = archinstall.arguments.get('audio_config', None)
+ if audio_config:
+ audio_config.install_audio_config(installation)
+ else:
+ info("No audio server will be installed")
+
+ if archinstall.arguments.get('packages', None) and archinstall.arguments.get('packages', None)[0] != '':
+ installation.add_additional_packages(archinstall.arguments.get('packages', []))
+
+ if profile_config := archinstall.arguments.get('profile_config', None):
+ profile_handler.install_profile_config(installation, profile_config)
+
+ if timezone := archinstall.arguments.get('timezone', None):
+ installation.set_timezone(timezone)
+
+ if archinstall.arguments.get('ntp', False):
+ installation.activate_time_synchronization()
+
+ if archinstall.accessibility_tools_in_use():
+ installation.enable_espeakup()
+
+ if (root_pw := archinstall.arguments.get('!root-password', None)) and len(root_pw):
+ installation.user_set_pw('root', root_pw)
+
+ if profile_config := archinstall.arguments.get('profile_config', None):
+ profile_config.profile.post_install(installation)
+
+ # If the user provided a list of services to be enabled, pass the list to the enable_service function.
+ # Note that while it's called enable_service, it can actually take a list of services and iterate it.
+ if archinstall.arguments.get('services', None):
+ installation.enable_service(archinstall.arguments.get('services', []))
+
+ # If the user provided custom commands to be run post-installation, execute them now.
+ if archinstall.arguments.get('custom-commands', None):
+ archinstall.run_custom_user_commands(archinstall.arguments['custom-commands'], installation)
+
+ installation.genfstab()
+
+ info("For post-installation tips, see https://wiki.archlinux.org/index.php/Installation_guide#Post-installation")
+
+ if not archinstall.arguments.get('silent'):
+ prompt = str(
+ _('Would you like to chroot into the newly created installation and perform post-installation configuration?'))
+ choice = menu.Menu(prompt, menu.Menu.yes_no(), default_option=menu.Menu.yes()).run()
+ if choice.value == menu.Menu.yes():
+ try:
+ installation.drop_to_shell()
+ except:
+ pass
+
+ debug(f"Disk states after installing: {disk.disk_layouts()}")
+
+
+param_mode = archinstall.arguments.get('mode', ExecutionMode.Full.value).lower()
+
+try:
+ mode = ExecutionMode(param_mode)
+except KeyError:
+ info(f'Mode "{param_mode}" is not supported')
+ exit(1)
+
+if not archinstall.arguments.get('silent'):
+ ask_user_questions(mode)
+
+config_output = ConfigurationOutput(archinstall.arguments)
+if not archinstall.arguments.get('silent'):
+ config_output.show()
+
+config_output.save()
+
+if archinstall.arguments.get('dry_run'):
+ exit(0)
+
+if not archinstall.arguments.get('silent'):
+ input('Press Enter to continue.')
+
+if mode in (ExecutionMode.Full, ExecutionMode.Only_HD):
+ fs_handler = disk.FilesystemHandler(
+ archinstall.arguments['disk_config'],
+ archinstall.arguments.get('disk_encryption', None)
+ )
+
+ fs_handler.perform_filesystem_operations()
+
+perform_installation(archinstall.storage.get('MOUNT_POINT', Path('/mnt')), mode)
diff --git a/archinstall/scripts/unattended.py b/archinstall/scripts/unattended.py
new file mode 100644
index 00000000..5ae4ae3d
--- /dev/null
+++ b/archinstall/scripts/unattended.py
@@ -0,0 +1,19 @@
+import time
+
+import archinstall
+from archinstall import info
+from archinstall import profile
+
+for p in profile.profile_handler.get_mac_addr_profiles():
+ # Tailored means it's a match for this machine
+ # based on it's MAC address (or some other criteria
+ # that fits the requirements for this machine specifically).
+ info(f'Found a tailored profile for this machine called: "{p.name}"')
+
+ print('Starting install in:')
+ for i in range(10, 0, -1):
+ print(f'{i}...')
+ time.sleep(1)
+
+ install_session = archinstall.storage['installation_session']
+ p.install(install_session)