diff --git a/.yupdate.post b/.yupdate.post index 2d91b28549..c3560a7012 100755 --- a/.yupdate.post +++ b/.yupdate.post @@ -25,8 +25,8 @@ function restart_service() { if [ -n "$SERVICE_START" ]; then SERVICE_START_UNIX_TIME=$(date -d "$SERVICE_START" +"%s") - # find the date of the latest file in the gem - NEWEST_FILE_TIME=$(find /usr/lib*/ruby/gems/*/gems/$SERVICE_NAME-* -exec stat --format %Y "{}" \; | sort -nr | head -n 1) + # find the date of the latest file in the product configuration or in the gem + NEWEST_FILE_TIME=$(find /usr/share/agama/products.d /usr/lib*/ruby/gems/*/gems/$SERVICE_NAME-* -exec stat --format %Y "{}" \; | sort -nr | head -n 1) # when a file is newer than the start time then restart the service if [ -n "$NEWEST_FILE_TIME" ] && [ "$SERVICE_START_UNIX_TIME" -lt "$NEWEST_FILE_TIME" ]; then diff --git a/Rakefile b/Rakefile index 37ec973652..095acb3b1c 100644 --- a/Rakefile +++ b/Rakefile @@ -170,5 +170,21 @@ if ENV["YUPDATE_FORCE"] == "1" || File.exist?("/.packages.initrd") || live_iso? FileUtils.mkdir_p(File.join(destdir, "/usr/share")) FileUtils.cp_r("playwright/.", File.join(destdir, "/usr/share/agama-playwright")) end + + if ENV["YUPDATE_SKIP_PRODUCTS"] != "1" + files = Dir.glob("products.d/*.y{a}ml") + files.each do |f| + # the sources contain several products, update only the existing files + oldfile = File.join("/usr/share/agama/", f) + if File.exist?(oldfile) + target = File.join(destdir, "/usr/share/agama/", f) + FileUtils.mkdir_p(File.dirname(target)) + FileUtils.cp(f, target) + else + # if there is a new product file it needs to be copied manually + puts "Skipping product file: #{f}" + end + end + end end end diff --git a/doc/yaml_config.md b/doc/yaml_config.md index a98d48c929..c2a8e0de9a 100644 --- a/doc/yaml_config.md +++ b/doc/yaml_config.md @@ -38,6 +38,15 @@ Array of patterns that have to be selected. Array of patterns that should be selected but can be deselected or skipped if not available. +#### user\_patterns + +Array of patterns that are displayed in the pattern selector UI and user can +select them to install. + +If the list is empty then the pattern selector is not displayed. If the key is +not defined or the value is missing or is `null` then all available user visible +patterns are displayed. + ### security Options related to security diff --git a/products.d/ALP-Dolomite.yaml b/products.d/ALP-Dolomite.yaml index f87af0f9f5..d1cbea6f61 100644 --- a/products.d/ALP-Dolomite.yaml +++ b/products.d/ALP-Dolomite.yaml @@ -23,6 +23,8 @@ software: - alp_cockpit - alp_hardware optional_patterns: null # no optional pattern shared + # no user selectable patterns, do not display the pattern selector + user_patterns: [] mandatory_packages: - package: ppc64-diag # Needed for hardware-based installations archs: ppc64 diff --git a/products.d/agama-products-opensuse.changes b/products.d/agama-products-opensuse.changes index f84460af9e..eb6cd2d50f 100644 --- a/products.d/agama-products-opensuse.changes +++ b/products.d/agama-products-opensuse.changes @@ -1,3 +1,9 @@ +------------------------------------------------------------------- +Mon Dec 4 14:11:51 UTC 2023 - Ancor Gonzalez Sosa + +- Preliminary definitions of openSUSE MicroOS products +- Remove Leap 16.0 for now + ------------------------------------------------------------------- Mon Oct 30 14:38:51 UTC 2023 - Josef Reidinger diff --git a/products.d/agama-products-opensuse.spec b/products.d/agama-products-opensuse.spec index d2c23829f9..815663bc4a 100644 --- a/products.d/agama-products-opensuse.spec +++ b/products.d/agama-products-opensuse.spec @@ -52,8 +52,9 @@ install -m 0644 *.yaml %{buildroot}%{_datadir}/agama/products.d %files %dir %{_datadir}/agama %dir %{_datadir}/agama/products.d +%{_datadir}/agama/products.d/microos.yaml +%{_datadir}/agama/products.d/microos-desktop.yaml %{_datadir}/agama/products.d/tumbleweed.yaml -%{_datadir}/agama/products.d/leap16.yaml %files -n agama-products-ALP-Dolomite %dir %{_datadir}/agama diff --git a/products.d/leap16.yaml b/products.d/leap16.yaml deleted file mode 100644 index 30060510af..0000000000 --- a/products.d/leap16.yaml +++ /dev/null @@ -1,100 +0,0 @@ -id: Leap16 -name: openSUSE Leap 16.0 -archs: x86_64,aarch64 -# ------------------------------------------------------------------------------ -# WARNING: When changing the product description delete the translations located -# at the at translations/description key below to avoid using obsolete -# translations!! -# ------------------------------------------------------------------------------ -description: '[Experimental project] openSUSE Leap 16 is built on top of the - next generation Adaptable Linux Platform (ALP) from SUSE.' -# Do not manually change any translations! See README.md for more details. -translations: - description: - cs: "[Experimentální projekt] openSUSE Leap 16 je postaven na budoucí generaci - Adaptable Linux Platform (ALP) od SUSE." - sv: "[Experimentellt projekt] openSUSE Leap 16 är byggt ovanpå nästa generations - Anpassningsbara Linux Plattform (ALP) från SUSE." -software: - installation_repositories: - - url: https://download.opensuse.org/repositories/openSUSE:/Leap:/16.0/images/repo/Leap-16.0-x86_64-Media1/ - archs: x86_64 - - url: https://download.opensuse.org/repositories/openSUSE:/Leap:/16.0/images/repo/Leap-16.0-aarch64-Media1/ - archs: aarch64 - mandatory_patterns: - - alp_base - - alp_base_zypper - - alp_cockpit - - alp-container_runtime - - alp_defaults - optional_patterns: null # no optional pattern shared - mandatory_packages: null - optional_packages: null - base_product: Leap16 - -security: - lsm: selinux - available_lsms: - # apparmor: - # patterns: - # - apparmor - selinux: - patterns: - - alp_selinux - policy: enforcing - none: - patterns: null - -storage: - space_policy: delete - encryption: - method: luks2 - pbkd_function: pbkdf2 - tpm_luks_open: true - volumes: - - "/" - volume_templates: - - mount_path: "/" - filesystem: btrfs - btrfs: - snapshots: true - read_only: true - default_subvolume: "@" - subvolumes: - - path: root - - path: home - - path: opt - - path: srv - - path: boot/writable - - path: usr/local - - path: boot/grub2/arm64-efi - archs: aarch64 - - path: boot/grub2/i386-pc - archs: x86_64 - - path: boot/grub2/powerpc-ieee1275 - archs: ppc,!board_powernv - - path: boot/grub2/s390x-emu - archs: s390 - - path: boot/grub2/x86_64-efi - archs: x86_64 - - path: var - copy_on_write: false - size: - auto: false - min: 5 GiB - outline: - required: true - filesystems: - - btrfs - snapshots_configurable: false - - filesystem: xfs - size: - auto: false - outline: - required: false - filesystems: - - btrfs - - ext2 - - ext3 - - ext4 - - xfs diff --git a/products.d/microos-desktop.yaml b/products.d/microos-desktop.yaml new file mode 100644 index 0000000000..8a850ec72d --- /dev/null +++ b/products.d/microos-desktop.yaml @@ -0,0 +1,110 @@ +id: MicroOS-Desktop +name: openSUSE MicroOS Desktop +# ------------------------------------------------------------------------------ +# WARNING: When changing the product description delete the translations located +# at the at translations/description key below to avoid using obsolete +# translations!! +# ------------------------------------------------------------------------------ +description: 'A distribution for the desktop offering automatic updates and + rollback on top of the foundations of openSUSE MicroOS. Includes Podman + Container Runtime and allows to manage software using Gnome Software or + KDE Discover.' +# Do not manually change any translations! See README.md for more details. +translations: +software: + installation_repositories: + - url: https://download.opensuse.org/tumbleweed/repo/oss/ + archs: x86_64 + - url: https://download.opensuse.org/ports/aarch64/tumbleweed/repo/oss/ + archs: aarch64 + - url: https://download.opensuse.org/ports/zsystems/tumbleweed/repo/oss/ + archs: s390 + - url: https://download.opensuse.org/ports/ppc/tumbleweed/repo/oss/ + archs: ppc + - url: https://download.opensuse.org/tumbleweed/repo/non-oss/ + archs: x86_64 + # aarch64 does not have non-oss ports. Keep eye if it change + - url: https://download.opensuse.org/ports/zsystems/tumbleweed/repo/non-oss/ + archs: s390 + - url: https://download.opensuse.org/ports/ppc/tumbleweed/repo/non-oss/ + archs: ppc + - url: https://download.opensuse.org/update/tumbleweed/ + archs: x86_64 + - url: https://download.opensuse.org/ports/aarch64/update/tumbleweed/ + archs: aarch64 + - url: https://download.opensuse.org/ports/zsystems/update/tumbleweed/ + archs: s390 + - url: https://download.opensuse.org/ports/ppc/tumbleweed/repo/oss/ + archs: ppc + mandatory_patterns: + - microos_base + - microos_base_zypper + - microos_defaults + - microos_hardware + - container_runtime + - pattern: 32bit + archs: x86_64 + optional_patterns: + - microos_gnome_desktop + user_patterns: + - microos_gnome_desktop + - microos_kde_desktop + mandatory_packages: + - NetworkManager + optional_packages: null + base_product: MicroOS + +security: + lsm: selinux + available_lsms: + selinux: + patterns: + - microos_selinux + policy: enforcing + none: + patterns: null + +storage: + space_policy: delete + volumes: + - "/" + volume_templates: + - mount_path: "/" + filesystem: btrfs + btrfs: + snapshots: true + read_only: true + default_subvolume: "@" + subvolumes: + - path: home + - path: opt + - path: root + - path: srv + - path: usr/local + - path: boot/writable + # Unified var subvolume - https://lists.opensuse.org/opensuse-packaging/2017-11/msg00017.html + - path: var + copy_on_write: false + # Architecture specific subvolumes + - path: boot/grub2/arm64-efi + archs: aarch64 + - path: boot/grub2/arm-efi + archs: arm + - path: boot/grub2/i386-pc + archs: x86_64 + - path: boot/grub2/powerpc-ieee1275 + archs: ppc,!board_powernv + - path: boot/grub2/s390x-emu + archs: s390 + - path: boot/grub2/x86_64-efi + archs: x86_64 + - path: boot/grub2/riscv64-efi + archs: riscv64 + size: + auto: false + min: 5 GiB + outline: + required: true + snapshots_configurable: false + filesystems: + - btrfs diff --git a/products.d/microos.yaml b/products.d/microos.yaml new file mode 100644 index 0000000000..ec998f2eec --- /dev/null +++ b/products.d/microos.yaml @@ -0,0 +1,126 @@ +id: MicroOS +name: openSUSE MicroOS +# ------------------------------------------------------------------------------ +# WARNING: When changing the product description delete the translations located +# at the at translations/description key below to avoid using obsolete +# translations!! +# ------------------------------------------------------------------------------ +description: 'A quick, small distribution designed to host container workloads + with automated administration & patching. openSUSE MicroOS provides + transactional (atomic) updates upon a read-only btrfs root file system. + As rolling release distribution the software is always up-to-date.' +# Do not manually change any translations! See README.md for more details. +translations: +software: + installation_repositories: + - url: https://download.opensuse.org/tumbleweed/repo/oss/ + archs: x86_64 + - url: https://download.opensuse.org/ports/aarch64/tumbleweed/repo/oss/ + archs: aarch64 + - url: https://download.opensuse.org/ports/zsystems/tumbleweed/repo/oss/ + archs: s390 + - url: https://download.opensuse.org/ports/ppc/tumbleweed/repo/oss/ + archs: ppc + - url: https://download.opensuse.org/tumbleweed/repo/non-oss/ + archs: x86_64 + # aarch64 does not have non-oss ports. Keep eye if it change + - url: https://download.opensuse.org/ports/zsystems/tumbleweed/repo/non-oss/ + archs: s390 + - url: https://download.opensuse.org/ports/ppc/tumbleweed/repo/non-oss/ + archs: ppc + - url: https://download.opensuse.org/update/tumbleweed/ + archs: x86_64 + - url: https://download.opensuse.org/ports/aarch64/update/tumbleweed/ + archs: aarch64 + - url: https://download.opensuse.org/ports/zsystems/update/tumbleweed/ + archs: s390 + - url: https://download.opensuse.org/ports/ppc/tumbleweed/repo/oss/ + archs: ppc + mandatory_patterns: + - microos_base + - microos_base_zypper + - microos_defaults + - microos_hardware + - pattern: 32bit + archs: x86_64 + optional_patterns: null + user_patterns: + - container_runtime + - microos_ra_agent + - microos_ra_verifier + mandatory_packages: + - NetworkManager + optional_packages: null + base_product: MicroOS + +security: + lsm: selinux + available_lsms: + selinux: + patterns: + - microos_selinux + policy: enforcing + none: + patterns: null + +storage: + space_policy: delete + volumes: + - "/" + - "/var" + volume_templates: + - mount_path: "/" + filesystem: btrfs + btrfs: + snapshots: true + read_only: true + default_subvolume: "@" + subvolumes: + - path: home + - path: opt + - path: root + - path: srv + - path: usr/local + - path: boot/writable + # Unified var subvolume - https://lists.opensuse.org/opensuse-packaging/2017-11/msg00017.html + - path: var + copy_on_write: false + # Architecture specific subvolumes + - path: boot/grub2/arm64-efi + archs: aarch64 + - path: boot/grub2/arm-efi + archs: arm + - path: boot/grub2/i386-pc + archs: x86_64 + - path: boot/grub2/powerpc-ieee1275 + archs: ppc,!board_powernv + - path: boot/grub2/s390x-emu + archs: s390 + - path: boot/grub2/x86_64-efi + archs: x86_64 + - path: boot/grub2/riscv64-efi + archs: riscv64 + size: + auto: true + outline: + required: true + snapshots_configurable: false + filesystems: + - btrfs + auto_size: + base_min: 5 GiB + base_max: 25 GiB + max_fallback_for: + - "/var" + - mount_path: "/var" + filesystem: btrfs + mount_options: + - "x-initrd.mount" + - "nodatacow" + size: + auto: false + min: 5 GiB + outline: + required: false + filesystems: + - btrfs diff --git a/products.d/tumbleweed.yaml b/products.d/tumbleweed.yaml index 1ef74d4147..fc8e934e5a 100644 --- a/products.d/tumbleweed.yaml +++ b/products.d/tumbleweed.yaml @@ -47,6 +47,16 @@ software: mandatory_patterns: - enhanced_base # only pattern that is shared among all roles on TW optional_patterns: null # no optional pattern shared + user_patterns: + - basic_desktop + - xfce + - kde + - gnome + - yast2_basis + - yast2_desktop + - yast2_server + - multimedia + - office mandatory_packages: - NetworkManager optional_packages: null diff --git a/service/.rubocop.yml b/service/.rubocop.yml index ed2faf80db..137ea30537 100644 --- a/service/.rubocop.yml +++ b/service/.rubocop.yml @@ -24,3 +24,7 @@ Metrics/BlockLength: # assignment in method calls is used to document some params Lint/UselessAssignment: Enabled: false + +# be less strict +Metrics/AbcSize: + Max: 32 diff --git a/service/lib/agama/config.rb b/service/lib/agama/config.rb index 0e524d648c..3b4c2ba811 100644 --- a/service/lib/agama/config.rb +++ b/service/lib/agama/config.rb @@ -168,12 +168,14 @@ def merge(config) # # @param keys [Array] Config data keys of the collection. # @param property [Symbol|String|nil] Property to retrieve of the elements. + # @param default [Object] The default value returned when the value is not + # found or is not an array # # @return [Array] - def arch_elements_from(*keys, property: nil) + def arch_elements_from(*keys, property: nil, default: []) keys.map!(&:to_s) elements = products.dig(*keys) - return [] unless elements.is_a?(Array) + return default unless elements.is_a?(Array) elements.map do |element| if !element.is_a?(Hash) diff --git a/service/lib/agama/software/manager.rb b/service/lib/agama/software/manager.rb index be47087609..9c2aeb8cf9 100644 --- a/service/lib/agama/software/manager.rb +++ b/service/lib/agama/software/manager.rb @@ -214,6 +214,11 @@ def patterns(filtered) patterns = Y2Packager::Resolvable.find({ kind: :pattern }, preload) patterns = patterns.select(&:user_visible) if filtered + # only display the configured patterns + if product.user_patterns && filtered + patterns.select! { |p| product.user_patterns.include?(p.name) } + end + patterns end diff --git a/service/lib/agama/software/product.rb b/service/lib/agama/software/product.rb index 24bb8c0b2b..9e1179ce3a 100644 --- a/service/lib/agama/software/product.rb +++ b/service/lib/agama/software/product.rb @@ -73,6 +73,11 @@ class Product # @return [Array] attr_accessor :optional_patterns + # Optional user selectable patterns + # + # @return [Array] + attr_accessor :user_patterns + # Product translations. # # @example @@ -94,6 +99,8 @@ def initialize(id) @optional_packages = [] @mandatory_patterns = [] @optional_patterns = [] + # nil = display all visible patterns, [] = display no patterns + @user_patterns = nil @translations = {} end diff --git a/service/lib/agama/software/product_builder.rb b/service/lib/agama/software/product_builder.rb index b4ecfeddf8..c048be304e 100644 --- a/service/lib/agama/software/product_builder.rb +++ b/service/lib/agama/software/product_builder.rb @@ -47,6 +47,7 @@ def build product.optional_packages = data[:optional_packages] product.mandatory_patterns = data[:mandatory_patterns] product.optional_patterns = data[:optional_patterns] + product.user_patterns = data[:user_patterns] product.translations = attrs["translations"] || {} end end @@ -79,6 +80,9 @@ def product_data_from_config(id) ), optional_patterns: config.arch_elements_from( id, "software", "optional_patterns", property: :pattern + ), + user_patterns: config.arch_elements_from( + id, "software", "user_patterns", property: :pattern, default: nil ) } end diff --git a/service/test/agama/software/manager_test.rb b/service/test/agama/software/manager_test.rb index f899ab64b3..1603656c4f 100644 --- a/service/test/agama/software/manager_test.rb +++ b/service/test/agama/software/manager_test.rb @@ -229,12 +229,70 @@ describe "#products" do it "returns the list of known products" do products = subject.products - expect(products.size).to eq(3) + expect(products.size).to eq(4) expect(products).to all(be_a(Agama::Software::Product)) expect(products).to contain_exactly( an_object_having_attributes(id: "ALP-Dolomite"), an_object_having_attributes(id: "Tumbleweed"), - an_object_having_attributes(id: "Leap16") + an_object_having_attributes(id: "MicroOS"), + an_object_having_attributes(id: "MicroOS-Desktop") + ) + end + end + + describe "#patterns" do + it "returns only the specified patterns" do + expect(Y2Packager::Resolvable).to receive(:find).and_return( + [ + double( + arch: "x86_64", + category: "Base Technologies", + description: "YaST tools for installing your system.", + icon: "./yast", + kind: :pattern, + name: "yast2_install_wf", + order: "1240", + source: 0, + summary: "YaST Installation Packages", + user_visible: false, + version: "20220411-1.4" + ), + double( + arch: "x86_64", + category: "Base Technologies", + description: "YaST tools for basic system administration.", + icon: "./yast", + kind: :pattern, + name: "yast2_basis", + order: "1220", + source: 0, + summary: "YaST Base Utilities", + user_visible: true, + version: "20220411-1.4" + ), + double( + arch: "noarch", + category: "Graphical Environments", + description: + "Packages providing the Plasma desktop environment and " \ + "applications from KDE.", + icon: "./pattern-kde", + kind: :pattern, + name: "kde", + order: "1110", + source: 0, + summary: "KDE Applications and Plasma 5 Desktop", + user_visible: true, + version: "20230801-1.1" + ) + ] + ) + + allow(subject.product).to receive(:user_patterns).and_return(["kde"]) + patterns = subject.patterns(true) + + expect(patterns).to contain_exactly( + an_object_having_attributes(name: "kde") ) end end diff --git a/web/src/components/overview/SoftwareSection.jsx b/web/src/components/overview/SoftwareSection.jsx index fbcd81cee8..6046a9ab2f 100644 --- a/web/src/components/overview/SoftwareSection.jsx +++ b/web/src/components/overview/SoftwareSection.jsx @@ -34,6 +34,7 @@ const initialState = { errors: [], errorsRead: false, size: "", + patterns: {}, progress: { message: _("Reading software repositories"), current: 0, total: 0, finished: false } }; @@ -51,8 +52,8 @@ const reducer = (state, action) => { case "UPDATE_PROPOSAL": { if (state.busy) return state; - const { errors, size } = action.payload; - return { ...state, errors, size, errorsRead: true }; + const { errors, size, patterns } = action.payload; + return { ...state, errors, size, patterns, errorsRead: true }; } default: { @@ -82,8 +83,9 @@ export default function SoftwareSection({ showErrors }) { const updateProposal = async () => { const errors = await cancellablePromise(client.getIssues()); const size = await cancellablePromise(client.getUsedSpace()); + const patterns = await cancellablePromise(client.patterns(true)); - dispatch({ type: "UPDATE_PROPOSAL", payload: { errors, size } }); + dispatch({ type: "UPDATE_PROPOSAL", payload: { errors, size, patterns } }); }; updateProposal(); @@ -142,7 +144,8 @@ export default function SoftwareSection({ showErrors }) { icon="apps" loading={state.busy} errors={errors.map(toValidationError)} - path="/software" + // do not display the pattern selector when there are no patterns to display + path={Object.keys(state.patterns).length > 0 ? "/software" : null} > diff --git a/web/src/components/overview/SoftwareSection.test.jsx b/web/src/components/overview/SoftwareSection.test.jsx index d21df0e84c..d0fdf55711 100644 --- a/web/src/components/overview/SoftwareSection.test.jsx +++ b/web/src/components/overview/SoftwareSection.test.jsx @@ -29,9 +29,20 @@ import { SoftwareSection } from "~/components/overview"; jest.mock("~/client"); +const kdePattern = { + kde: [ + "Graphical Environments", + "Packages providing the Plasma desktop environment and applications from KDE.", + "./pattern-kde", + "KDE Applications and Plasma 5 Desktop", + "1110" + ] +}; + let getStatusFn = jest.fn().mockResolvedValue(IDLE); let getProgressFn = jest.fn().mockResolvedValue({}); let getIssuesFn = jest.fn().mockResolvedValue([]); +let patternsFn = jest.fn().mockResolvedValue(kdePattern); beforeEach(() => { createClient.mockImplementation(() => { @@ -42,6 +53,7 @@ beforeEach(() => { getIssues: getIssuesFn, onStatusChange: noop, onProgressChange: noop, + patterns: patternsFn, getUsedSpace: jest.fn().mockResolvedValue("500 MB") }, }; @@ -51,6 +63,7 @@ beforeEach(() => { describe("when the proposal is calculated", () => { beforeEach(() => { getStatusFn = jest.fn().mockResolvedValue(IDLE); + patternsFn = jest.fn().mockResolvedValue(kdePattern); }); it("renders the required space", async () => { @@ -59,6 +72,28 @@ describe("when the proposal is calculated", () => { await screen.findByText("500 MB"); }); + describe("patterns are available", () => { + it("the header is a link", async () => { + const { container } = installerRender(); + // wait until the component is fully rendered + await screen.findByText("Installation will take"); + expect(container.querySelector("h2 a[href='/software']")).not.toBeNull(); + }); + }); + + describe("no patterns are available", () => { + beforeEach(() => { + patternsFn = jest.fn().mockResolvedValue({}); + }); + + it("the header is a plain text", async () => { + const { container } = installerRender(); + // wait until the component is fully rendered + await screen.findByText("Installation will take"); + expect(container.querySelector("h2 a")).toBeNull(); + }); + }); + describe("and there are errors", () => { beforeEach(() => { getIssuesFn = jest.fn().mockResolvedValue([{ description: "Could not install..." }]);