linux-deployment-scripts/main.py
2024-11-10 18:03:47 +01:00

219 lines
7.2 KiB
Python
Executable file

#!/usr/bin/env python3
# linux-deployment-scripts
# Copyright (C) 2024 VM-Experiments
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
import subprocess
import sys
import os
import json
import shutil
downloads = "downloads/"
output = "output/"
chroot = "chroot/"
resources = "resources/"
installer_packages = "linux-image-amd64 grub-pc parted tar dosfstools util-linux e2fsprogs arch-install-scripts"
installer_amd64 = "http://deb.debian.org/debian/"
installer_amd64_name = "bookworm"
not_verbose = True
clean_tarballs = False
def command(command):
result = subprocess.run(command, capture_output=not_verbose, text=True, shell=True)
return result
# chroot associated functions
def prepare_chroot():
command(f"sudo mount --bind /dev '{chroot}/dev'")
command(f"sudo mount --bind /proc '{chroot}/proc'")
command(f"sudo mount --bind /sys '{chroot}/sys'")
command(f"sudo cp /etc/resolv.conf '{chroot}/etc/'")
def do_chroot_command(command_input):
command(f"sudo chroot '{chroot}' /bin/bash -c '{command_input}'")
def close_chroot():
command(f"sudo umount '{chroot}/dev'")
command(f"sudo umount '{chroot}/proc'")
command(f"sudo umount '{chroot}/sys'")
# handle rootfs image from internet
def rootfs_downloader(link, filename):
os.chdir(downloads)
command(f"wget -nc {link}")
command(f"sudo tar -xzf {filename} -C '../{chroot}'")
os.chdir("..")
def rootfs_creator(filename):
os.chdir(output)
command(f"sudo tar -czf {filename} -C '../{chroot}' .")
os.chdir("..")
# packaging rootfs
def rootfs_package(filename, installer):
infile = f"{filename}.tar.gz"
outfile = f"{filename}.iso"
close_chroot()
command(f"sudo rm -rf '{chroot}'")
command(f"sudo mkdir '{chroot}'")
command(
f"sudo debootstrap --arch amd64 {installer_amd64_name} {chroot} {installer_amd64}"
)
command(f"sudo cp '{infile}' '{chroot}/rootfs.tar.gz'")
if clean_tarballs:
command(f"sudo rm '{infile}'")
prepare_chroot()
do_chroot_command("apt update")
do_chroot_command(
f"DEBIAN_FRONTEND=noninteractive apt install -y {installer_packages}"
)
do_chroot_command("mkdir -p /etc/systemd/system/getty@tty1.service.d/")
do_chroot_command("touch /etc/systemd/system/getty@tty1.service.d/override.conf")
do_chroot_command(
'echo -e "[Service]\nExecStart=\nExecStart=/sbin/agetty --noclear --autologin root %I \\$TERM" | tee /etc/systemd/system/getty@tty1.service.d/override.conf'
)
do_chroot_command("systemctl daemon-reload")
do_chroot_command("systemctl enable getty@tty1.service")
do_chroot_command("echo 'localmachine' > /etc/hostname")
close_chroot()
command(f"sudo cp '{resources}/grub.cfg' '{chroot}/boot/grub/grub.cfg'")
command(f"sudo cp '{installer}' '{chroot}/root/install.sh'")
command(f"echo './install.sh' | sudo tee -a '{chroot}/root/.profile'")
command(f"echo './install.sh' | sudo tee -a '{chroot}/root/.bashrc'")
if shutil.which("grub-mkrescue"):
command(f"sudo grub-mkrescue -o '{outfile}' '{chroot}'")
elif shutil.which("grub2-mkrescue"):
command(f"sudo grub2-mkrescue -o '{outfile}' '{chroot}'")
command(f"sudo chown $(logname):$(logname) {outfile}")
# distro specific
def ubuntu_rootfs_prepare(
download_link, download_name, packages, identifier, script, package_manager
):
rootfs_downloader(download_link, download_name)
generic_rootfs_prepare(packages, identifier, script, package_manager)
def debian_rootfs_prepare(
download_link, download_name, packages, identifier, script, package_manager
):
command(f"sudo debootstrap --arch amd64 {download_name} {chroot} {download_link}")
generic_rootfs_prepare(packages, identifier, script, package_manager)
def generic_rootfs_prepare(packages, identifier, script, package_manager):
prepare_chroot()
do_chroot_command("apt update -y")
do_chroot_command("apt full-upgrade -y")
do_chroot_command(
f"DEBIAN_FRONTEND=noninteractive {package_manager} install -y {packages}"
)
do_chroot_command("echo 'localmachine' > /etc/hostname")
close_chroot()
rootfs_creator(f"{identifier}.tar.gz")
rootfs_package(f"{output}{identifier}", script)
# main
def main():
global not_verbose
global clean_tarballs
for index, arg in enumerate(sys.argv):
match arg:
case "-v":
not_verbose = False
case "--verbose":
not_verbose = False
case "-ct":
clean_tarballs = True
case "--clean-tarballs":
clean_tarballs = True
case "-h":
print("")
print("Help for Linux Deployment Scripts:")
print("")
print("-h / --help - Shows this menu.")
print(
"-v / --verbose - Displays the outputs of the different commands."
)
print(
"-ct / --clean-tarballs - Delete the tar.gz leftovers from the iso generation."
)
print("")
exit()
case "--help":
clean_tarballs = True
case _:
if index == 0:
continue
command(f"mkdir -p '{downloads}' '{output}'")
close_chroot()
command(f"sudo rm -rf '{chroot}'")
command(f"sudo mkdir '{chroot}'")
filename = arg
data = json.load(open(filename, "r"))
rootfs_type = data["rootfs_type"]
download_link = data["download_link"]
download_name = data["download_name"]
packages = data["packages"]
identifier = data["identifier"]
script = data["script"]
package_manager = data["package_manager"]
match rootfs_type:
case "download":
ubuntu_rootfs_prepare(
download_link,
download_name,
packages,
identifier,
script,
package_manager,
)
case "debootstrap":
debian_rootfs_prepare(
download_link,
download_name,
packages,
identifier,
script,
package_manager,
)
close_chroot()
command(f"sudo rm -rf '{chroot}'")
main()