From af3bdc37fb4d8b65043981257e942dfd8ba6651d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 18:52:26 +0000 Subject: [PATCH 01/12] feat(nicegui): make image table sortable --- anvil/nicegui/pages/registry.py | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/anvil/nicegui/pages/registry.py b/anvil/nicegui/pages/registry.py index 87e5213..0652c8b 100644 --- a/anvil/nicegui/pages/registry.py +++ b/anvil/nicegui/pages/registry.py @@ -24,8 +24,10 @@ def get_image_info() -> pandas.DataFrame: lambda layers: sum(layer["size"] for layer in layers) ) ) - data = data[["image_name", "name", "size"]].rename( - columns={"image_name": "image", "name": "tag", "size": "size"} + data = ( + data[["image_name", "name", "size"]] + .rename(columns={"image_name": "image", "name": "tag", "size": "size"}) + .sort_values(by=["image", "tag"], ascending=False) ) data["size"] = data["size"].apply(humanize.naturalsize) return data @@ -39,4 +41,7 @@ def content() -> None: with ui.card().classes("w-full"): ui.label("Image Overview").classes("text-h5") data = get_image_info() - ui.table.from_pandas(df=data).classes("w-full") + with ui.table.from_pandas(df=data).classes("w-full") as table: + table.columns[0]["sortable"] = True + table.columns[1]["sortable"] = True + table.columns[2]["sortable"] = True From b213c826cf224b5a21d45dcf3c0b5ef526f22ecd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 18:53:19 +0000 Subject: [PATCH 02/12] fix(registry): disable performance constraints (#52) trying to avoid 'StatusCode: 499, Client Closed Request' --- forge-pod.yml | 8 -------- 1 file changed, 8 deletions(-) diff --git a/forge-pod.yml b/forge-pod.yml index 54187c1..c7215f2 100644 --- a/forge-pod.yml +++ b/forge-pod.yml @@ -27,10 +27,6 @@ spec: containers: - name: traefik.${FORGE_DOMAIN_NAME} image: traefik # will be built on pod start - resources: - limits: - memory: 128Mi - cpu: 200m volumeMounts: - mountPath: /var/run/podman.sock name: podman-socket @@ -77,10 +73,6 @@ spec: containers: - name: docker.${FORGE_DOMAIN_NAME} image: registry # will be built on pod start - resources: - limits: - memory: 512Mi - cpu: 200m volumeMounts: - mountPath: /certs name: ublue-os_forge-certs-pvc From cec7512c570dd100d87b079c350440207abafde9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 19:00:20 +0000 Subject: [PATCH 03/12] feat(ansible): add possibility to add build-args to podman build job (#51) --- anvil/ansible/playbooks/project_build.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/anvil/ansible/playbooks/project_build.yml b/anvil/ansible/playbooks/project_build.yml index 5c32129..f824421 100644 --- a/anvil/ansible/playbooks/project_build.yml +++ b/anvil/ansible/playbooks/project_build.yml @@ -16,6 +16,7 @@ build: file: "{{ forge_container_file | default('Containerfile') }}" format: "{{ forge_container_format | default('oci') }}" + extra_args: "{{ forge_container_extra_args | default([]) | join(' ') }}" pull: false push: true push_args: From d1a0f7fff2be3854a9b6765dea77eedb529f42a3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 19:02:21 +0000 Subject: [PATCH 04/12] fix(ansible): use registry url for container image name --- anvil/ansible/playbooks/project_build.yml | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/anvil/ansible/playbooks/project_build.yml b/anvil/ansible/playbooks/project_build.yml index f824421..26b8862 100644 --- a/anvil/ansible/playbooks/project_build.yml +++ b/anvil/ansible/playbooks/project_build.yml @@ -2,6 +2,9 @@ - name: Build project hosts: host.ublue.local gather_facts: true + vars: + __image_regex_search: (?<=/)[^/]+(?=\.git) + __image_name: "{{ forge_registry_url }}/{{ forge_git_repository_url | regex_search(__image_regex_search) }}" pre_tasks: - name: DEBUG - forge variables ansible.builtin.include_role: @@ -10,7 +13,7 @@ tasks: - name: Build and push image to registry containers.podman.podman_image: - name: "{{ forge_git_repository_url | regex_search(__regex_search) }}" + name: "{{ __image_name }}" tag: latest path: "{{ forge_git_repository_destination }}" build: @@ -19,10 +22,6 @@ extra_args: "{{ forge_container_extra_args | default([]) | join(' ') }}" pull: false push: true - push_args: - dest: "{{ forge_registry_url }}" - vars: - __regex_search: (?<=/)[^/]+(?=\.git) async: 1800 poll: 0 register: __podman_image From 98c1803bfa99e24cf9ab02ef9388e60278c4895b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 19:03:28 +0000 Subject: [PATCH 05/12] chore(ansible): use better task names --- anvil/ansible/playbooks/project_build.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/anvil/ansible/playbooks/project_build.yml b/anvil/ansible/playbooks/project_build.yml index 26b8862..2ed01af 100644 --- a/anvil/ansible/playbooks/project_build.yml +++ b/anvil/ansible/playbooks/project_build.yml @@ -11,7 +11,7 @@ name: debug_forge_vars tasks: - - name: Build and push image to registry + - name: Build and upload container image containers.podman.podman_image: name: "{{ __image_name }}" tag: latest @@ -26,7 +26,7 @@ poll: 0 register: __podman_image - - name: Waiting for container build to finish + - name: Waiting for container build and upload to finish ansible.builtin.async_status: jid: "{{ __podman_image.ansible_job_id }}" register: __job_result @@ -34,7 +34,7 @@ retries: 1800 delay: 1 - - name: INFO | Status from build and push + - name: INFO | Result from container image build ansible.builtin.debug: msg: - "{{ __job_result.actions | to_nice_yaml(indent=2) }}" From 1214eda79716e318a055d930c9b569469e6c90a9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 19:04:10 +0000 Subject: [PATCH 06/12] chore(ansible): commit less time to async tasks --- anvil/ansible/playbooks/project_build.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/anvil/ansible/playbooks/project_build.yml b/anvil/ansible/playbooks/project_build.yml index 2ed01af..f5e0d81 100644 --- a/anvil/ansible/playbooks/project_build.yml +++ b/anvil/ansible/playbooks/project_build.yml @@ -22,7 +22,7 @@ extra_args: "{{ forge_container_extra_args | default([]) | join(' ') }}" pull: false push: true - async: 1800 + async: 900 poll: 0 register: __podman_image @@ -31,7 +31,7 @@ jid: "{{ __podman_image.ansible_job_id }}" register: __job_result until: __job_result.finished - retries: 1800 + retries: 900 delay: 1 - name: INFO | Result from container image build From 6297b8f951eaa597d2499116a1f4ab0c0ff0fa8c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Sun, 26 May 2024 19:04:58 +0000 Subject: [PATCH 07/12] feat(ansible): add additional tags to container image --- anvil/ansible/playbooks/project_build.yml | 28 +++++++++++++++++++++++ 1 file changed, 28 insertions(+) diff --git a/anvil/ansible/playbooks/project_build.yml b/anvil/ansible/playbooks/project_build.yml index f5e0d81..a3839c6 100644 --- a/anvil/ansible/playbooks/project_build.yml +++ b/anvil/ansible/playbooks/project_build.yml @@ -39,3 +39,31 @@ msg: - "{{ __job_result.actions | to_nice_yaml(indent=2) }}" - "{{ __job_result.image | to_nice_yaml(indent=2) }}" + + - name: Add additional tag to container image + containers.podman.podman_tag: + image: "{{ __image_name }}:latest" + target_names: + - "{{ __image_name }}:{{ forge_container_tag }}" + + - name: Push additional container image tag to registry + containers.podman.podman_image: + name: "{{ __image_name }}" + tag: "{{ forge_container_tag }}" + pull: false + push: true + async: 15 + poll: 0 + register: __podman_image + + - name: Waiting for additional container image tag push to finish + ansible.builtin.async_status: + jid: "{{ __podman_image.ansible_job_id }}" + register: __job_result + until: __job_result.finished + retries: 15 + delay: 1 + + - name: INFO | Result from additional container image tag push + ansible.builtin.debug: + msg: "{{ __job_result.actions | to_nice_yaml(indent=2) }}" From ae10ebc4acdf8e9bfab792cb2dbcaa22257c00d1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Mon, 27 May 2024 06:55:44 +0000 Subject: [PATCH 08/12] fix(nicegui): empty image table should use same column names as when images are available --- anvil/nicegui/pages/registry.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/anvil/nicegui/pages/registry.py b/anvil/nicegui/pages/registry.py index 0652c8b..fd38541 100644 --- a/anvil/nicegui/pages/registry.py +++ b/anvil/nicegui/pages/registry.py @@ -6,7 +6,7 @@ ## TODO: this should be async but I currently don't know how to implement this without button press def get_image_info() -> pandas.DataFrame: - data = pandas.DataFrame(columns=["image_name", "tag", "size"]) + data = pandas.DataFrame(columns=["image", "tag", "size"]) try: registry = DockerRegistry() all_image_info = registry.get_all_image_info() From 327c010ddddc62dfb7873adfe0b36d14ed567e2e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Mon, 27 May 2024 06:58:47 +0000 Subject: [PATCH 09/12] feat(nicegui): easy navigation to home by clicking on ublue icon --- anvil/nicegui/theme.py | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/anvil/nicegui/theme.py b/anvil/nicegui/theme.py index e7cd9a5..af93662 100644 --- a/anvil/nicegui/theme.py +++ b/anvil/nicegui/theme.py @@ -35,9 +35,10 @@ def frame( with ui.grid(columns=3).classes("w-full gap-0"): with ui.row(wrap=False).classes("col-span-1 justify-start"): menu() - ui.image(source=f"{project_root}/pages/assets/ublue-mini.svg").props( - "width=33px hight=auto" - ) + with ui.link(target="/"): + ui.image( + source=f"{project_root}/pages/assets/ublue-mini.svg" + ).props("width=33px hight=auto") ui.label(text="Forge").classes("text-h5") with ui.row(wrap=False).classes("col-span-1 justify-center"): ui.label(text=navigation_title).classes("text-h5") From e0df50076e6067ad65588a7bfe77818470b495c6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Mon, 27 May 2024 16:41:07 +0000 Subject: [PATCH 10/12] feat(ansible): add example configurations to the setup (#45) --- anvil/ansible/group_vars/all/container.yml | 3 +++ anvil/ansible/group_vars/all/git.yml | 4 ++-- anvil/ansible/playbooks/configure_host.yml | 24 +++++++++---------- .../playbooks/files/examples/bluefin-39.yml | 11 +++++++++ anvil/ansible/playbooks/project_build.yml | 17 ++++++------- anvil/nicegui/pages/ansible.py | 4 +++- 6 files changed, 38 insertions(+), 25 deletions(-) create mode 100644 anvil/ansible/playbooks/files/examples/bluefin-39.yml diff --git a/anvil/ansible/group_vars/all/container.yml b/anvil/ansible/group_vars/all/container.yml index 6be86b7..cba2027 100644 --- a/anvil/ansible/group_vars/all/container.yml +++ b/anvil/ansible/group_vars/all/container.yml @@ -1,3 +1,6 @@ --- +forge_container_name: "{{ forge_registry_url }}/{{ forge_git_repository_url | regex_search('(?<=/)[^/]+(?=\\.git)') }}" +forge_container_tag: "{{ ansible_date_time.date }}_{{ ansible_date_time.time | replace(':','')}}" forge_container_file: "Containerfile" forge_container_format: "oci" +forge_container_extra_args: [] diff --git a/anvil/ansible/group_vars/all/git.yml b/anvil/ansible/group_vars/all/git.yml index 1e72e26..fd8c8d1 100644 --- a/anvil/ansible/group_vars/all/git.yml +++ b/anvil/ansible/group_vars/all/git.yml @@ -1,5 +1,5 @@ --- # git variables -forge_git_repository_url: "https://github.com/ublue-os/bluefin.git" -forge_git_repository_destination: "{{ forge_data_volume_mountpoint }}/data/bluefin" +forge_git_repository_url: "" +forge_git_repository_destination: "" forge_git_repository_version: "main" diff --git a/anvil/ansible/playbooks/configure_host.yml b/anvil/ansible/playbooks/configure_host.yml index 32f4743..684ef60 100644 --- a/anvil/ansible/playbooks/configure_host.yml +++ b/anvil/ansible/playbooks/configure_host.yml @@ -36,19 +36,19 @@ changed_when: false become: true - - name: Create example extra-vars configuration file - ansible.builtin.copy: - dest: "{{ forge_data_volume_mountpoint }}/forge_example_vars.yml" - content: | - ## ublue-os forge extra-vars example configuration - ## For more details got to https://github.com/ublue-os/forge/blob/main/docs/variables.md - --- - {% for item in __vars_used %} - {{ item }}: {{ lookup('ansible.builtin.vars', item) }} - {% endfor %} + - name: Create example directory + ansible.builtin.file: + path: "{{ forge_data_volume_mountpoint }}/examples/" + state: directory + mode: "0755" + + - name: Create example configuration files + ansible.builtin.template: + src: "{{ item }}" + dest: "{{ forge_data_volume_mountpoint }}/examples/{{ item | regex_search(__regex_search) }}" backup: true owner: "{{ ansible_facts.env.USER }}" mode: "0644" + with_fileglob: "files/examples/*" vars: - __vars_used: "{{ lookup('ansible.builtin.varnames', __regex_search, wantlist=true) }}" - __regex_search: ^forge_(?!data).+ + __regex_search: "[^\/]+$" diff --git a/anvil/ansible/playbooks/files/examples/bluefin-39.yml b/anvil/ansible/playbooks/files/examples/bluefin-39.yml new file mode 100644 index 0000000..d42d9aa --- /dev/null +++ b/anvil/ansible/playbooks/files/examples/bluefin-39.yml @@ -0,0 +1,11 @@ +## ublue-os forge example configuration for bluefin +## For more details got to https://github.com/ublue-os/forge/blob/main/docs/variables.md +--- +forge_git_repository_url: https://github.com/ublue-os/bluefin.git +forge_git_repository_destination: "{{ forge_data_volume_mountpoint }}/data/bluefin" +forge_container_extra_args: + - --build-arg="BASE_IMAGE_NAME=silverblue" + - --build-arg="IMAGE_FLAVOR=main" + - --build-arg="AKMODS_FLAVOR=main" + - --build-arg="FEDORA_MAJOR_VERSION=39" + - --build-arg="TARGET_BASE=bluefin" diff --git a/anvil/ansible/playbooks/project_build.yml b/anvil/ansible/playbooks/project_build.yml index a3839c6..d1d679e 100644 --- a/anvil/ansible/playbooks/project_build.yml +++ b/anvil/ansible/playbooks/project_build.yml @@ -2,9 +2,6 @@ - name: Build project hosts: host.ublue.local gather_facts: true - vars: - __image_regex_search: (?<=/)[^/]+(?=\.git) - __image_name: "{{ forge_registry_url }}/{{ forge_git_repository_url | regex_search(__image_regex_search) }}" pre_tasks: - name: DEBUG - forge variables ansible.builtin.include_role: @@ -13,13 +10,13 @@ tasks: - name: Build and upload container image containers.podman.podman_image: - name: "{{ __image_name }}" + name: "{{ forge_container_name }}" tag: latest path: "{{ forge_git_repository_destination }}" build: - file: "{{ forge_container_file | default('Containerfile') }}" - format: "{{ forge_container_format | default('oci') }}" - extra_args: "{{ forge_container_extra_args | default([]) | join(' ') }}" + file: "{{ forge_container_file }}" + format: "{{ forge_container_format }}" + extra_args: "{{ forge_container_extra_args | join(' ') }}" pull: false push: true async: 900 @@ -42,13 +39,13 @@ - name: Add additional tag to container image containers.podman.podman_tag: - image: "{{ __image_name }}:latest" + image: "{{ forge_container_name }}:latest" target_names: - - "{{ __image_name }}:{{ forge_container_tag }}" + - "{{ forge_container_name }}:{{ forge_container_tag }}" - name: Push additional container image tag to registry containers.podman.podman_image: - name: "{{ __image_name }}" + name: "{{ forge_container_name }}" tag: "{{ forge_container_tag }}" pull: false push: true diff --git a/anvil/nicegui/pages/ansible.py b/anvil/nicegui/pages/ansible.py index 4b77f67..d83ef8e 100644 --- a/anvil/nicegui/pages/ansible.py +++ b/anvil/nicegui/pages/ansible.py @@ -14,7 +14,9 @@ async def load_configuration_file() -> None: global ANSIBLE_EXTRA_VARS result = await local_file_picker( - directory="/data", multiple=False, file_name_filter=".yml" + directory="/data", + multiple=False, + # file_name_filter=".yml", # TODO: limit to yml files but make sure folders are visible as well ) file_path = result[0] with open(file_path, "r") as file: From ee86b81f5b8844928aa4b6b9eb968e3782fb66dc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Mon, 27 May 2024 16:42:33 +0000 Subject: [PATCH 11/12] doc(main): updates --- docs/index.md | 19 +++++++++++-------- docs/variables.md | 22 +++++++++++----------- 2 files changed, 22 insertions(+), 19 deletions(-) diff --git a/docs/index.md b/docs/index.md index d06b925..21f04d5 100644 --- a/docs/index.md +++ b/docs/index.md @@ -41,21 +41,24 @@ ln -s $FORGE_POD_DATA_DIR $HOME/ublue-os_forge-data With this the data folder would be available in your home directory under `~/ublue-os_forge-data` -In that folder you will find an example configuration similar to this: +In that folder you will find an **examples** folder with example configurations similar to this: ```yaml -## ublue-os forge extra-vars example configuration +## ublue-os forge example configuration ## For more details got to https://github.com/ublue-os/forge/blob/main/docs/variables.md --- forge_git_repository_url: https://github.com/ublue-os/bluefin.git -forge_git_repository_destination: /var/home/stephan/.local/share/containers/storage/volumes/ublue-os_forge-data/_data/data/bluefin -forge_git_repository_version: main -forge_registry_url: registry.ublue.local +forge_git_repository_destination: "/var/home/stephan/.local/share/containers/storage/volumes/ublue-os_forge-data/_data/data/bluefin" +forge_container_extra_args: + - --build-arg="BASE_IMAGE_NAME=silverblue" + - --build-arg="IMAGE_FLAVOR=main" + - --build-arg="AKMODS_FLAVOR=main" + - --build-arg="FEDORA_MAJOR_VERSION=39" + - --build-arg="TARGET_BASE=bluefin" ``` -This file acts as a template. It has all available settings outlined for you. Simple copy -it and modify it to your liking. For each project you want to handle with the forge you can -create a dedicated file. +These files are a good starting point for your custom configuration. Simple copy those examples +you are interested in modify them to your liking. Details about the available variables are documented [here](./variables.md). diff --git a/docs/variables.md b/docs/variables.md index 743243b..3c57f24 100644 --- a/docs/variables.md +++ b/docs/variables.md @@ -10,16 +10,16 @@ The following configuration variables are available and can be set to your likin -| Name | Type | Default value | Description | -| ---------------------------------- | ---- | ------------------------------------------------- | -------------------------------------------------------------------------------- | -| `forge_container_file` | str | Containerfile | Path to the Containerfile for Podman to build | -| `forge_container_format` | str | oci | Format of the image Podman will build. Can be either `oci` or `docker` | -| `forge_git_repository_url` | str | | Git repository url | -| `forge_git_repository_destination` | str | `{{ forge_data_volume_mountpoint }}`/data/bluefin | Git destination where repository is cloned to. Can be any directory on your host | -| `forge_git_repository_version` | str | main | Git repository branch or tag or commit version | -| `forge_registry_url` | str | registry.ublue.local | Container registry url | +| Name | Type | Default value | Description | +| ---------------------------------- | ---- | ----------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------------------------------- | +| `forge_container_name` | str | Project name derived from `forge_git_repository_url` Example: `bluefin` | Container image name. | +| `forge_container_tag` | str | Evaluates to: `YYY-MM-DD_HHMMSS` Example: `2024-05-26_192206` | Container image tag. | +| `forge_container_file` | str | Containerfile | Path to the Containerfile for Podman to build | +| `forge_container_format` | str | oci | Format of the image Podman will build. Can be either `oci` or `docker` | +| `forge_container_extra_args` | list | [] | List of extra arguments which gets passed to the podman build process. Example: `[--build-arg="BASE_IMAGE_NAME=silverblue"]` | +| `forge_git_repository_url` | str | | Git repository url | +| `forge_git_repository_destination` | str | | Git destination where repository is cloned to. | +| `forge_git_repository_version` | str | main | Git repository branch or tag or commit version | +| `forge_registry_url` | str | registry.ublue.local | Container registry url | - -**_Note:_** The `{{ forge_data_volume_mountpoint }}` points to your ublue-os_forge-data -podman volume. From ef65492a90fae7d99e7dcfda0762cb663d202c2a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stephan=20L=C3=BCscher?= Date: Mon, 27 May 2024 16:43:25 +0000 Subject: [PATCH 12/12] chore(devcontainer): update spell-check dictionary --- .vscode/cspell_custom.txt | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/.vscode/cspell_custom.txt b/.vscode/cspell_custom.txt index 3a5a1c8..f1c92c1 100644 --- a/.vscode/cspell_custom.txt +++ b/.vscode/cspell_custom.txt @@ -1,19 +1,25 @@ aggrid +AKMODS CHACHA configmap Containerfile containerignore devcontainer devcontainers +dialtimeout dotenv ENDCOLOR ensurepath envsubst filepicker +forwardingtimeouts getent gitmessage +HHMMSS hostvars humanfriendly +idleconntimeout +idletimeout keygen LAZYGIT lightspeed @@ -30,10 +36,13 @@ notranslate pipx posix Proto +readtimeout redirections refreshable rvproxy serverstransport +serverstransports +silverblue traefik ublue varnames @@ -41,3 +50,4 @@ venvs VIRTUALENVS wantlist websecure +writetimeout