all:
  children:
    tempest:
      hosts:
        controller: null
    zuul_unreachable:
      hosts: {}
  hosts:
    controller:
      ansible_connection: ssh
      ansible_host: 162.253.55.36
      ansible_port: 22
      ansible_python_interpreter: auto
      ansible_user: zuul
      configure_swap_size: 8192
      devstack_local_conf:
        post-config:
          $NEUTRON_CONF:
            DEFAULT:
              global_physnet_mtu: '{{ external_bridge_mtu }}'
          /etc/magnum/magnum.conf:
            cluster_template:
              kubernetes_allowed_network_drivers: calico,cilium
              kubernetes_default_network_driver: calico
            nova_client:
              api_version: 2.15
          /etc/manila/manila.conf:
            generic:
              connect_share_server_to_tenant_network: true
              driver_handles_share_servers: true
      devstack_localrc:
        ADMIN_PASSWORD: secretadmin
        DATABASE_PASSWORD: secretdatabase
        DEBUG_LIBVIRT_COREDUMPS: true
        DISABLE_AMP_IMAGE_BUILD: true
        ENABLE_SYSCTL_MEM_TUNING: true
        ENABLE_SYSCTL_NET_TUNING: true
        ENABLE_ZSWAP: true
        ERROR_ON_CLONE: true
        FIXED_RANGE: 10.1.0.0/20
        FLOATING_RANGE: 172.24.5.0/24
        GIT_BASE: https://github.com
        HOST_IP: '{{ hostvars[''controller''][''nodepool''][''private_ipv4''] }}'
        IPV4_ADDRS_SAFE_TO_USE: 10.1.0.0/20
        LIBVIRT_TYPE: '{{ devstack_libvirt_type | default("qemu") }}'
        LOGFILE: /opt/stack/logs/devstacklog.txt
        LOG_COLOR: false
        MAGNUM_GUEST_IMAGE_URL: '{{ image_url }}'
        MANILA_DEFAULT_SHARE_TYPE_EXTRA_SPECS: snapshot_support=True create_share_from_snapshot_support=True
        MANILA_ENABLED_BACKENDS: generic
        MANILA_USE_SERVICE_INSTANCE_PASSWORD: true
        NETWORK_GATEWAY: 10.1.0.1
        NOVA_LIBVIRT_TB_CACHE_SIZE: 128
        NOVA_VNC_ENABLED: true
        OCTAVIA_NODE: api
        OVN_DBS_LOG_LEVEL: dbg
        PUBLIC_BRIDGE_MTU: '{{ external_bridge_mtu }}'
        PUBLIC_NETWORK_GATEWAY: 172.24.5.1
        RABBIT_PASSWORD: secretrabbit
        SERVICE_HOST: '{{ hostvars[''controller''][''nodepool''][''private_ipv4'']
          }}'
        SERVICE_PASSWORD: secretservice
        SWIFT_HASH: 1234123412341234
        SWIFT_REPLICAS: 1
        SWIFT_START_ALL_SERVICES: false
        VERBOSE: true
        VERBOSE_NO_TIMESTAMP: true
      devstack_plugins:
        barbican: https://github.com/openstack/barbican
        magnum: https://review.opendev.org/openstack/magnum
        magnum-cluster-api: https://github.com/vexxhost/magnum-cluster-api
        manila: https://github.com/openstack/manila
        octavia: https://github.com/openstack/octavia
        ovn-octavia-provider: https://github.com/openstack/ovn-octavia-provider
      devstack_services:
        base: false
        c-api: true
        c-bak: true
        c-sch: true
        c-vol: true
        dstat: false
        etcd3: true
        file_tracker: true
        g-api: true
        horizon: false
        key: true
        memory_tracker: true
        mysql: true
        n-api: true
        n-api-meta: true
        n-cond: true
        n-cpu: true
        n-novnc: true
        n-sch: true
        o-api: true
        o-da: true
        o-hk: true
        octavia: true
        openstack-cli-server: true
        ovn-controller: true
        ovn-northd: true
        ovs-vswitchd: true
        ovsdb-server: true
        placement-api: true
        q-ovn-agent: true
        q-svc: true
        rabbit: true
        s-account: false
        s-container: false
        s-object: false
        s-proxy: false
        tempest: false
        tls-proxy: true
      extensions_to_txt:
        auto: true
        conf: true
        localrc: true
        log: true
        stackenv: true
      image_url: https://github.com/vexxhost/capo-image-elements/releases/latest/download/ubuntu-22.04-{{
        kube_tag }}.qcow2
      kube_tag: v1.33.10
      network_driver: cilium
      nodepool:
        az: nova
        cloud: public
        external_id: 9093d97e-c1d5-4d35-a6fb-d6edf40363ff
        host_id: 7b8a4cf3090e32af994f4889a62f1003a5f74592d8295d185d35c3b9
        interface_ip: 162.253.55.36
        label: ubuntu-noble-16
        node_properties: {}
        private_ipv4: 162.253.55.36
        private_ipv6: null
        provider: yul1
        public_ipv4: 162.253.55.36
        public_ipv6: 2604:e100:1:0:f816:3eff:febb:cc93
        region: ca-ymq-1
        slot: null
      zuul_copy_output:
        /etc/ceph: logs
        /etc/glusterfs/glusterd.vol: logs
        /etc/libvirt: logs
        /etc/lvm: logs
        /etc/resolv.conf: logs
        /etc/sudoers: logs
        /etc/sudoers.d: logs
        /var/log/ceph: logs
        /var/log/glusterfs: logs
        /var/log/libvirt: logs
        /var/log/mysql: logs
        /var/log/openvswitch: logs
        /var/log/postgresql: logs
        /var/log/rabbitmq: logs
        /var/log/unbound.log: logs
        '{{ devstack_conf_dir }}/.localrc.auto': logs
        '{{ devstack_conf_dir }}/.stackenv': logs
        '{{ devstack_conf_dir }}/local.conf': logs
        '{{ devstack_conf_dir }}/localrc': logs
        '{{ devstack_full_log}}': logs
        '{{ devstack_log_dir }}/atop': logs
        '{{ devstack_log_dir }}/devstacklog.txt': logs
        '{{ devstack_log_dir }}/devstacklog.txt.summary': logs
        '{{ devstack_log_dir }}/dstat-csv.log': logs
        '{{ devstack_log_dir }}/qemu.coredump': logs
        '{{ devstack_log_dir }}/tcpdump.pcap': logs
        '{{ devstack_log_dir }}/worlddump-latest.txt': logs
        '{{ stage_dir }}/apache': logs
        '{{ stage_dir }}/apache_config': logs
        '{{ stage_dir }}/audit.log': logs
        '{{ stage_dir }}/core': logs
        '{{ stage_dir }}/deprecations.log': logs
        '{{ stage_dir }}/df.txt': logs
        '{{ stage_dir }}/dpkg-l.txt': logs
        '{{ stage_dir }}/etc': logs
        '{{ stage_dir }}/iptables.txt': logs
        '{{ stage_dir }}/listen53.txt': logs
        '{{ stage_dir }}/mount.txt': logs
        '{{ stage_dir }}/performance.json': logs
        '{{ stage_dir }}/pip2-freeze.txt': logs
        '{{ stage_dir }}/pip3-freeze.txt': logs
        '{{ stage_dir }}/rpm-qa.txt': logs
        '{{ stage_dir }}/services.txt': logs
        '{{ stage_dir }}/verify_tempest_conf.log': logs
      zuul_node:
        az: nova
        cloud: public
        external_id: 9093d97e-c1d5-4d35-a6fb-d6edf40363ff
        host_id: 7b8a4cf3090e32af994f4889a62f1003a5f74592d8295d185d35c3b9
        interface_ip: 162.253.55.36
        label: ubuntu-noble-16
        node_properties: {}
        private_ipv4: 162.253.55.36
        private_ipv6: null
        provider: yul1
        public_ipv4: 162.253.55.36
        public_ipv6: 2604:e100:1:0:f816:3eff:febb:cc93
        region: ca-ymq-1
        slot: null
        uuid: null
  vars:
    configure_swap_size: 8192
    devstack_local_conf:
      post-config:
        $NEUTRON_CONF:
          DEFAULT:
            global_physnet_mtu: '{{ external_bridge_mtu }}'
        /etc/magnum/magnum.conf:
          cluster_template:
            kubernetes_allowed_network_drivers: calico,cilium
            kubernetes_default_network_driver: calico
          nova_client:
            api_version: 2.15
        /etc/manila/manila.conf:
          generic:
            connect_share_server_to_tenant_network: true
            driver_handles_share_servers: true
    devstack_localrc:
      ADMIN_PASSWORD: secretadmin
      DATABASE_PASSWORD: secretdatabase
      DEBUG_LIBVIRT_COREDUMPS: true
      DISABLE_AMP_IMAGE_BUILD: true
      ENABLE_SYSCTL_MEM_TUNING: true
      ENABLE_SYSCTL_NET_TUNING: true
      ENABLE_ZSWAP: true
      ERROR_ON_CLONE: true
      FIXED_RANGE: 10.1.0.0/20
      FLOATING_RANGE: 172.24.5.0/24
      GIT_BASE: https://github.com
      HOST_IP: '{{ hostvars[''controller''][''nodepool''][''private_ipv4''] }}'
      IPV4_ADDRS_SAFE_TO_USE: 10.1.0.0/20
      LIBVIRT_TYPE: '{{ devstack_libvirt_type | default("qemu") }}'
      LOGFILE: /opt/stack/logs/devstacklog.txt
      LOG_COLOR: false
      MAGNUM_GUEST_IMAGE_URL: '{{ image_url }}'
      MANILA_DEFAULT_SHARE_TYPE_EXTRA_SPECS: snapshot_support=True create_share_from_snapshot_support=True
      MANILA_ENABLED_BACKENDS: generic
      MANILA_USE_SERVICE_INSTANCE_PASSWORD: true
      NETWORK_GATEWAY: 10.1.0.1
      NOVA_LIBVIRT_TB_CACHE_SIZE: 128
      NOVA_VNC_ENABLED: true
      OCTAVIA_NODE: api
      OVN_DBS_LOG_LEVEL: dbg
      PUBLIC_BRIDGE_MTU: '{{ external_bridge_mtu }}'
      PUBLIC_NETWORK_GATEWAY: 172.24.5.1
      RABBIT_PASSWORD: secretrabbit
      SERVICE_HOST: '{{ hostvars[''controller''][''nodepool''][''private_ipv4''] }}'
      SERVICE_PASSWORD: secretservice
      SWIFT_HASH: 1234123412341234
      SWIFT_REPLICAS: 1
      SWIFT_START_ALL_SERVICES: false
      VERBOSE: true
      VERBOSE_NO_TIMESTAMP: true
    devstack_plugins:
      barbican: https://github.com/openstack/barbican
      magnum: https://review.opendev.org/openstack/magnum
      magnum-cluster-api: https://github.com/vexxhost/magnum-cluster-api
      manila: https://github.com/openstack/manila
      octavia: https://github.com/openstack/octavia
      ovn-octavia-provider: https://github.com/openstack/ovn-octavia-provider
    devstack_services:
      base: false
      c-api: true
      c-bak: true
      c-sch: true
      c-vol: true
      dstat: false
      etcd3: true
      file_tracker: true
      g-api: true
      horizon: false
      key: true
      memory_tracker: true
      mysql: true
      n-api: true
      n-api-meta: true
      n-cond: true
      n-cpu: true
      n-novnc: true
      n-sch: true
      o-api: true
      o-da: true
      o-hk: true
      octavia: true
      openstack-cli-server: true
      ovn-controller: true
      ovn-northd: true
      ovs-vswitchd: true
      ovsdb-server: true
      placement-api: true
      q-ovn-agent: true
      q-svc: true
      rabbit: true
      s-account: false
      s-container: false
      s-object: false
      s-proxy: false
      tempest: false
      tls-proxy: true
    extensions_to_txt:
      auto: true
      conf: true
      localrc: true
      log: true
      stackenv: true
    image_url: https://github.com/vexxhost/capo-image-elements/releases/latest/download/ubuntu-22.04-{{
      kube_tag }}.qcow2
    kube_tag: v1.33.10
    network_driver: cilium
    zuul:
      _inheritance_path:
      - '<Job base explicit: None implied: {MatchAny:{ImpliedBranchMatcher:main}}
        source: vexxhost/zuul-config/zuul.d/jobs.yaml@main#1>'
      - '<Job openstack-multinode-fips explicit: None implied: {MatchAny:{ImpliedBranchMatcher:main}}
        source: vexxhost/zuul-config/zuul.d/jobs.yaml@main#17>'
      - '<Job devstack-base explicit: None implied: {MatchAny:{ImpliedBranchMatcher:master}}
        source: openstack/devstack/.zuul.yaml@master#426>'
      - '<Job devstack-minimal explicit: None implied: {MatchAny:{ImpliedBranchMatcher:master}}
        source: openstack/devstack/.zuul.yaml@master#560>'
      - '<Job devstack explicit: None implied: {MatchAny:{ImpliedBranchMatcher:master}}
        source: openstack/devstack/.zuul.yaml@master#603>'
      - '<Job magnum-cluster-api-devstack explicit: None implied: {MatchAny:{ImpliedBranchMatcher:main}}
        source: vexxhost/magnum-cluster-api/zuul.d/jobs.yaml@main#1>'
      - '<Job magnum-cluster-api-hydrophone explicit: None implied: {MatchAny:{ImpliedBranchMatcher:main}}
        source: vexxhost/magnum-cluster-api/zuul.d/jobs.yaml@main#60>'
      - '<Job magnum-cluster-api-hydrophone-v1.33.10 explicit: None implied: {MatchAny:{ImpliedBranchMatcher:main}}
        source: vexxhost/magnum-cluster-api/zuul.d/jobs.yaml@main#72>'
      - '<Job magnum-cluster-api-hydrophone-v1.33.10-cilium explicit: None implied:
        {MatchAny:{ImpliedBranchMatcher:main}} source: vexxhost/magnum-cluster-api/zuul.d/jobs.yaml@main#84>'
      - '<Job magnum-cluster-api-hydrophone-v1.33.10-cilium explicit: None implied:
        None source: vexxhost/magnum-cluster-api/zuul.d/project.yaml@main#1>'
      ansible_version: '9'
      attempts: 1
      branch: main
      build: a8ea62124168471e83397f5d02cc5b29
      build_refs:
      - branch: main
        change: '956'
        change_message: "fix: patch nested object fields individually to avoid clobbering
          existing values\n\nDuring cluster upgrades, several JSON patches were replacing
          entire nested objects (`op: add` at the parent path), which removed pre-existing
          fields like `provider` that were set by CAPO or by a previous ClusterClass
          version.\n\n## Root Cause\n\nThe old code used single JSON patches that
          replaced entire nested objects with serialized variable values. When optional
          fields were omitted (via `skip_serializing_if`), the patch value only contained
          the explicitly set fields \u2014 which overwrote any existing fields on
          the template, losing values set by CAPO or other controllers.\n\n## Fix\n\nSplit
          whole-object patches into **field-level `ClusterClassPatch` entries** across
          multiple features:\n\n### `apiServerLoadBalancer`\n\n| Patch | Field | Op
          | Guard |\n|-------|-------|----|-------|\n| `apiServerLoadBalancerEnabled`
          | `enabled` | `replace` | always |\n| `apiServerLoadBalancerProvider` |
          `provider` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerFlavor`
          | `flavor` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerAvailabilityZone`
          | `availabilityZone` | `add` | `enabledIf` \u2014 skipped when not set |\n\n###
          `externalNetwork`, `serverGroup`, `network`\n\nConverted from whole-object
          replacement to field-level patching of `/id` only, preserving any other
          fields (e.g., `filter`) that CAPO may set.\n\n### `boot_volume` (deferred)\n\n`boot_volume.rs`
          sets all current `rootVolume` fields (`type`, `sizeGiB`, `availabilityZone`),
          so there is no clobbering risk today. Adding `rootVolume: {}` to the base
          template would break the disabled test which expects `root_volume: None`.\n\n##
          Infrastructure Improvements\n\n- **`JsonPatchOp` enum**: Added `Add` and
          `Replace` variants to replace free-form `&str` for the `op` parameter, preventing
          typos at compile time.\n- **`variable_field_patch()` helper**: Extracted
          a reusable helper in `mod.rs` that builds a single-field `ClusterClassPatches`
          entry, reducing ~80 lines of boilerplate per feature to a single function
          call.\n- **Base template parent objects**: Added empty parent objects (`apiServerLoadBalancer`,
          `externalNetwork`, `network`, `serverGroup`) to base templates so sub-field
          patch paths have valid parents (required by RFC 6902).\n\n## Test Infrastructure
          Improvements\n\nUpdated `src/features/test.rs` to better match CAPI's behavior:\n\n-
          **`to_rendered_value`**: Resolves dotted variable paths by walking the JSON
          tree, preserving types like `bool` instead of rendering through Go templates
          to strings. Panics on unresolvable variables, matching CAPI's `calculateValue`
          error behavior.\n- **`to_rendered_patch`** / **`to_patch`**: Return values
          directly and panic on unresolvable variables, ensuring enabled patches with
          misspelled or missing variables are caught as test failures.\n- **`is_enabled`**:
          Special-cases `ExecError::NoFiledFor` (gtmpl's missing map key error) as
          \"disabled\" to match Go's `text/template` default behavior. All other render
          errors are propagated.\n\n## Regression Tests\n\n- `test_patches_preserve_existing_fields`:
          Verifies that patching only `enabled` does NOT remove pre-existing `provider`
          and `flavor` fields from the template.\n- `test_patches_update_existing_provider`:
          Verifies that when a new provider value is specified, it correctly replaces
          the existing one."
        change_url: https://github.com/vexxhost/magnum-cluster-api/pull/956
        commit_id: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        patchset: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        project:
          canonical_hostname: github.com
          canonical_name: github.com/vexxhost/magnum-cluster-api
          name: vexxhost/magnum-cluster-api
          short_name: magnum-cluster-api
          src_dir: src/github.com/vexxhost/magnum-cluster-api
        src_dir: src/github.com/vexxhost/magnum-cluster-api
        topic: null
      buildset: 40897718fe054ea2b520d2b2909f7285
      buildset_refs:
      - branch: main
        change: '956'
        change_message: "fix: patch nested object fields individually to avoid clobbering
          existing values\n\nDuring cluster upgrades, several JSON patches were replacing
          entire nested objects (`op: add` at the parent path), which removed pre-existing
          fields like `provider` that were set by CAPO or by a previous ClusterClass
          version.\n\n## Root Cause\n\nThe old code used single JSON patches that
          replaced entire nested objects with serialized variable values. When optional
          fields were omitted (via `skip_serializing_if`), the patch value only contained
          the explicitly set fields \u2014 which overwrote any existing fields on
          the template, losing values set by CAPO or other controllers.\n\n## Fix\n\nSplit
          whole-object patches into **field-level `ClusterClassPatch` entries** across
          multiple features:\n\n### `apiServerLoadBalancer`\n\n| Patch | Field | Op
          | Guard |\n|-------|-------|----|-------|\n| `apiServerLoadBalancerEnabled`
          | `enabled` | `replace` | always |\n| `apiServerLoadBalancerProvider` |
          `provider` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerFlavor`
          | `flavor` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerAvailabilityZone`
          | `availabilityZone` | `add` | `enabledIf` \u2014 skipped when not set |\n\n###
          `externalNetwork`, `serverGroup`, `network`\n\nConverted from whole-object
          replacement to field-level patching of `/id` only, preserving any other
          fields (e.g., `filter`) that CAPO may set.\n\n### `boot_volume` (deferred)\n\n`boot_volume.rs`
          sets all current `rootVolume` fields (`type`, `sizeGiB`, `availabilityZone`),
          so there is no clobbering risk today. Adding `rootVolume: {}` to the base
          template would break the disabled test which expects `root_volume: None`.\n\n##
          Infrastructure Improvements\n\n- **`JsonPatchOp` enum**: Added `Add` and
          `Replace` variants to replace free-form `&str` for the `op` parameter, preventing
          typos at compile time.\n- **`variable_field_patch()` helper**: Extracted
          a reusable helper in `mod.rs` that builds a single-field `ClusterClassPatches`
          entry, reducing ~80 lines of boilerplate per feature to a single function
          call.\n- **Base template parent objects**: Added empty parent objects (`apiServerLoadBalancer`,
          `externalNetwork`, `network`, `serverGroup`) to base templates so sub-field
          patch paths have valid parents (required by RFC 6902).\n\n## Test Infrastructure
          Improvements\n\nUpdated `src/features/test.rs` to better match CAPI's behavior:\n\n-
          **`to_rendered_value`**: Resolves dotted variable paths by walking the JSON
          tree, preserving types like `bool` instead of rendering through Go templates
          to strings. Panics on unresolvable variables, matching CAPI's `calculateValue`
          error behavior.\n- **`to_rendered_patch`** / **`to_patch`**: Return values
          directly and panic on unresolvable variables, ensuring enabled patches with
          misspelled or missing variables are caught as test failures.\n- **`is_enabled`**:
          Special-cases `ExecError::NoFiledFor` (gtmpl's missing map key error) as
          \"disabled\" to match Go's `text/template` default behavior. All other render
          errors are propagated.\n\n## Regression Tests\n\n- `test_patches_preserve_existing_fields`:
          Verifies that patching only `enabled` does NOT remove pre-existing `provider`
          and `flavor` fields from the template.\n- `test_patches_update_existing_provider`:
          Verifies that when a new provider value is specified, it correctly replaces
          the existing one."
        change_url: https://github.com/vexxhost/magnum-cluster-api/pull/956
        commit_id: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        patchset: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        project:
          canonical_hostname: github.com
          canonical_name: github.com/vexxhost/magnum-cluster-api
          name: vexxhost/magnum-cluster-api
          short_name: magnum-cluster-api
          src_dir: src/github.com/vexxhost/magnum-cluster-api
        src_dir: src/github.com/vexxhost/magnum-cluster-api
        topic: null
      change: '956'
      change_message: "fix: patch nested object fields individually to avoid clobbering
        existing values\n\nDuring cluster upgrades, several JSON patches were replacing
        entire nested objects (`op: add` at the parent path), which removed pre-existing
        fields like `provider` that were set by CAPO or by a previous ClusterClass
        version.\n\n## Root Cause\n\nThe old code used single JSON patches that replaced
        entire nested objects with serialized variable values. When optional fields
        were omitted (via `skip_serializing_if`), the patch value only contained the
        explicitly set fields \u2014 which overwrote any existing fields on the template,
        losing values set by CAPO or other controllers.\n\n## Fix\n\nSplit whole-object
        patches into **field-level `ClusterClassPatch` entries** across multiple features:\n\n###
        `apiServerLoadBalancer`\n\n| Patch | Field | Op | Guard |\n|-------|-------|----|-------|\n|
        `apiServerLoadBalancerEnabled` | `enabled` | `replace` | always |\n| `apiServerLoadBalancerProvider`
        | `provider` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerFlavor`
        | `flavor` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerAvailabilityZone`
        | `availabilityZone` | `add` | `enabledIf` \u2014 skipped when not set |\n\n###
        `externalNetwork`, `serverGroup`, `network`\n\nConverted from whole-object
        replacement to field-level patching of `/id` only, preserving any other fields
        (e.g., `filter`) that CAPO may set.\n\n### `boot_volume` (deferred)\n\n`boot_volume.rs`
        sets all current `rootVolume` fields (`type`, `sizeGiB`, `availabilityZone`),
        so there is no clobbering risk today. Adding `rootVolume: {}` to the base
        template would break the disabled test which expects `root_volume: None`.\n\n##
        Infrastructure Improvements\n\n- **`JsonPatchOp` enum**: Added `Add` and `Replace`
        variants to replace free-form `&str` for the `op` parameter, preventing typos
        at compile time.\n- **`variable_field_patch()` helper**: Extracted a reusable
        helper in `mod.rs` that builds a single-field `ClusterClassPatches` entry,
        reducing ~80 lines of boilerplate per feature to a single function call.\n-
        **Base template parent objects**: Added empty parent objects (`apiServerLoadBalancer`,
        `externalNetwork`, `network`, `serverGroup`) to base templates so sub-field
        patch paths have valid parents (required by RFC 6902).\n\n## Test Infrastructure
        Improvements\n\nUpdated `src/features/test.rs` to better match CAPI's behavior:\n\n-
        **`to_rendered_value`**: Resolves dotted variable paths by walking the JSON
        tree, preserving types like `bool` instead of rendering through Go templates
        to strings. Panics on unresolvable variables, matching CAPI's `calculateValue`
        error behavior.\n- **`to_rendered_patch`** / **`to_patch`**: Return values
        directly and panic on unresolvable variables, ensuring enabled patches with
        misspelled or missing variables are caught as test failures.\n- **`is_enabled`**:
        Special-cases `ExecError::NoFiledFor` (gtmpl's missing map key error) as \"disabled\"
        to match Go's `text/template` default behavior. All other render errors are
        propagated.\n\n## Regression Tests\n\n- `test_patches_preserve_existing_fields`:
        Verifies that patching only `enabled` does NOT remove pre-existing `provider`
        and `flavor` fields from the template.\n- `test_patches_update_existing_provider`:
        Verifies that when a new provider value is specified, it correctly replaces
        the existing one."
      change_url: https://github.com/vexxhost/magnum-cluster-api/pull/956
      child_jobs: []
      commit_id: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
      event_id: 6068b100-31ed-11f1-83ce-805d1d9502e4
      executor:
        hostname: 0a8996d2b663
        inventory_file: /var/lib/zuul/builds/a8ea62124168471e83397f5d02cc5b29/ansible/inventory.yaml
        log_root: /var/lib/zuul/builds/a8ea62124168471e83397f5d02cc5b29/work/logs
        result_data_file: /var/lib/zuul/builds/a8ea62124168471e83397f5d02cc5b29/work/results.json
        src_root: /var/lib/zuul/builds/a8ea62124168471e83397f5d02cc5b29/work/src
        work_root: /var/lib/zuul/builds/a8ea62124168471e83397f5d02cc5b29/work
      include_vars: []
      items:
      - branch: main
        change: '956'
        change_message: "fix: patch nested object fields individually to avoid clobbering
          existing values\n\nDuring cluster upgrades, several JSON patches were replacing
          entire nested objects (`op: add` at the parent path), which removed pre-existing
          fields like `provider` that were set by CAPO or by a previous ClusterClass
          version.\n\n## Root Cause\n\nThe old code used single JSON patches that
          replaced entire nested objects with serialized variable values. When optional
          fields were omitted (via `skip_serializing_if`), the patch value only contained
          the explicitly set fields \u2014 which overwrote any existing fields on
          the template, losing values set by CAPO or other controllers.\n\n## Fix\n\nSplit
          whole-object patches into **field-level `ClusterClassPatch` entries** across
          multiple features:\n\n### `apiServerLoadBalancer`\n\n| Patch | Field | Op
          | Guard |\n|-------|-------|----|-------|\n| `apiServerLoadBalancerEnabled`
          | `enabled` | `replace` | always |\n| `apiServerLoadBalancerProvider` |
          `provider` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerFlavor`
          | `flavor` | `add` | `enabledIf` \u2014 skipped when not set |\n| `apiServerLoadBalancerAvailabilityZone`
          | `availabilityZone` | `add` | `enabledIf` \u2014 skipped when not set |\n\n###
          `externalNetwork`, `serverGroup`, `network`\n\nConverted from whole-object
          replacement to field-level patching of `/id` only, preserving any other
          fields (e.g., `filter`) that CAPO may set.\n\n### `boot_volume` (deferred)\n\n`boot_volume.rs`
          sets all current `rootVolume` fields (`type`, `sizeGiB`, `availabilityZone`),
          so there is no clobbering risk today. Adding `rootVolume: {}` to the base
          template would break the disabled test which expects `root_volume: None`.\n\n##
          Infrastructure Improvements\n\n- **`JsonPatchOp` enum**: Added `Add` and
          `Replace` variants to replace free-form `&str` for the `op` parameter, preventing
          typos at compile time.\n- **`variable_field_patch()` helper**: Extracted
          a reusable helper in `mod.rs` that builds a single-field `ClusterClassPatches`
          entry, reducing ~80 lines of boilerplate per feature to a single function
          call.\n- **Base template parent objects**: Added empty parent objects (`apiServerLoadBalancer`,
          `externalNetwork`, `network`, `serverGroup`) to base templates so sub-field
          patch paths have valid parents (required by RFC 6902).\n\n## Test Infrastructure
          Improvements\n\nUpdated `src/features/test.rs` to better match CAPI's behavior:\n\n-
          **`to_rendered_value`**: Resolves dotted variable paths by walking the JSON
          tree, preserving types like `bool` instead of rendering through Go templates
          to strings. Panics on unresolvable variables, matching CAPI's `calculateValue`
          error behavior.\n- **`to_rendered_patch`** / **`to_patch`**: Return values
          directly and panic on unresolvable variables, ensuring enabled patches with
          misspelled or missing variables are caught as test failures.\n- **`is_enabled`**:
          Special-cases `ExecError::NoFiledFor` (gtmpl's missing map key error) as
          \"disabled\" to match Go's `text/template` default behavior. All other render
          errors are propagated.\n\n## Regression Tests\n\n- `test_patches_preserve_existing_fields`:
          Verifies that patching only `enabled` does NOT remove pre-existing `provider`
          and `flavor` fields from the template.\n- `test_patches_update_existing_provider`:
          Verifies that when a new provider value is specified, it correctly replaces
          the existing one."
        change_url: https://github.com/vexxhost/magnum-cluster-api/pull/956
        commit_id: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        patchset: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        project:
          canonical_hostname: github.com
          canonical_name: github.com/vexxhost/magnum-cluster-api
          name: vexxhost/magnum-cluster-api
          short_name: magnum-cluster-api
          src_dir: src/github.com/vexxhost/magnum-cluster-api
        topic: null
      job: magnum-cluster-api-hydrophone-v1.33.10-cilium
      jobtags: []
      max_attempts: 3
      message: Zml4OiBwYXRjaCBuZXN0ZWQgb2JqZWN0IGZpZWxkcyBpbmRpdmlkdWFsbHkgdG8gYXZvaWQgY2xvYmJlcmluZyBleGlzdGluZyB2YWx1ZXMKCkR1cmluZyBjbHVzdGVyIHVwZ3JhZGVzLCBzZXZlcmFsIEpTT04gcGF0Y2hlcyB3ZXJlIHJlcGxhY2luZyBlbnRpcmUgbmVzdGVkIG9iamVjdHMgKGBvcDogYWRkYCBhdCB0aGUgcGFyZW50IHBhdGgpLCB3aGljaCByZW1vdmVkIHByZS1leGlzdGluZyBmaWVsZHMgbGlrZSBgcHJvdmlkZXJgIHRoYXQgd2VyZSBzZXQgYnkgQ0FQTyBvciBieSBhIHByZXZpb3VzIENsdXN0ZXJDbGFzcyB2ZXJzaW9uLgoKIyMgUm9vdCBDYXVzZQoKVGhlIG9sZCBjb2RlIHVzZWQgc2luZ2xlIEpTT04gcGF0Y2hlcyB0aGF0IHJlcGxhY2VkIGVudGlyZSBuZXN0ZWQgb2JqZWN0cyB3aXRoIHNlcmlhbGl6ZWQgdmFyaWFibGUgdmFsdWVzLiBXaGVuIG9wdGlvbmFsIGZpZWxkcyB3ZXJlIG9taXR0ZWQgKHZpYSBgc2tpcF9zZXJpYWxpemluZ19pZmApLCB0aGUgcGF0Y2ggdmFsdWUgb25seSBjb250YWluZWQgdGhlIGV4cGxpY2l0bHkgc2V0IGZpZWxkcyDigJQgd2hpY2ggb3Zlcndyb3RlIGFueSBleGlzdGluZyBmaWVsZHMgb24gdGhlIHRlbXBsYXRlLCBsb3NpbmcgdmFsdWVzIHNldCBieSBDQVBPIG9yIG90aGVyIGNvbnRyb2xsZXJzLgoKIyMgRml4CgpTcGxpdCB3aG9sZS1vYmplY3QgcGF0Y2hlcyBpbnRvICoqZmllbGQtbGV2ZWwgYENsdXN0ZXJDbGFzc1BhdGNoYCBlbnRyaWVzKiogYWNyb3NzIG11bHRpcGxlIGZlYXR1cmVzOgoKIyMjIGBhcGlTZXJ2ZXJMb2FkQmFsYW5jZXJgCgp8IFBhdGNoIHwgRmllbGQgfCBPcCB8IEd1YXJkIHwKfC0tLS0tLS18LS0tLS0tLXwtLS0tfC0tLS0tLS18CnwgYGFwaVNlcnZlckxvYWRCYWxhbmNlckVuYWJsZWRgIHwgYGVuYWJsZWRgIHwgYHJlcGxhY2VgIHwgYWx3YXlzIHwKfCBgYXBpU2VydmVyTG9hZEJhbGFuY2VyUHJvdmlkZXJgIHwgYHByb3ZpZGVyYCB8IGBhZGRgIHwgYGVuYWJsZWRJZmAg4oCUIHNraXBwZWQgd2hlbiBub3Qgc2V0IHwKfCBgYXBpU2VydmVyTG9hZEJhbGFuY2VyRmxhdm9yYCB8IGBmbGF2b3JgIHwgYGFkZGAgfCBgZW5hYmxlZElmYCDigJQgc2tpcHBlZCB3aGVuIG5vdCBzZXQgfAp8IGBhcGlTZXJ2ZXJMb2FkQmFsYW5jZXJBdmFpbGFiaWxpdHlab25lYCB8IGBhdmFpbGFiaWxpdHlab25lYCB8IGBhZGRgIHwgYGVuYWJsZWRJZmAg4oCUIHNraXBwZWQgd2hlbiBub3Qgc2V0IHwKCiMjIyBgZXh0ZXJuYWxOZXR3b3JrYCwgYHNlcnZlckdyb3VwYCwgYG5ldHdvcmtgCgpDb252ZXJ0ZWQgZnJvbSB3aG9sZS1vYmplY3QgcmVwbGFjZW1lbnQgdG8gZmllbGQtbGV2ZWwgcGF0Y2hpbmcgb2YgYC9pZGAgb25seSwgcHJlc2VydmluZyBhbnkgb3RoZXIgZmllbGRzIChlLmcuLCBgZmlsdGVyYCkgdGhhdCBDQVBPIG1heSBzZXQuCgojIyMgYGJvb3Rfdm9sdW1lYCAoZGVmZXJyZWQpCgpgYm9vdF92b2x1bWUucnNgIHNldHMgYWxsIGN1cnJlbnQgYHJvb3RWb2x1bWVgIGZpZWxkcyAoYHR5cGVgLCBgc2l6ZUdpQmAsIGBhdmFpbGFiaWxpdHlab25lYCksIHNvIHRoZXJlIGlzIG5vIGNsb2JiZXJpbmcgcmlzayB0b2RheS4gQWRkaW5nIGByb290Vm9sdW1lOiB7fWAgdG8gdGhlIGJhc2UgdGVtcGxhdGUgd291bGQgYnJlYWsgdGhlIGRpc2FibGVkIHRlc3Qgd2hpY2ggZXhwZWN0cyBgcm9vdF92b2x1bWU6IE5vbmVgLgoKIyMgSW5mcmFzdHJ1Y3R1cmUgSW1wcm92ZW1lbnRzCgotICoqYEpzb25QYXRjaE9wYCBlbnVtKio6IEFkZGVkIGBBZGRgIGFuZCBgUmVwbGFjZWAgdmFyaWFudHMgdG8gcmVwbGFjZSBmcmVlLWZvcm0gYCZzdHJgIGZvciB0aGUgYG9wYCBwYXJhbWV0ZXIsIHByZXZlbnRpbmcgdHlwb3MgYXQgY29tcGlsZSB0aW1lLgotICoqYHZhcmlhYmxlX2ZpZWxkX3BhdGNoKClgIGhlbHBlcioqOiBFeHRyYWN0ZWQgYSByZXVzYWJsZSBoZWxwZXIgaW4gYG1vZC5yc2AgdGhhdCBidWlsZHMgYSBzaW5nbGUtZmllbGQgYENsdXN0ZXJDbGFzc1BhdGNoZXNgIGVudHJ5LCByZWR1Y2luZyB+ODAgbGluZXMgb2YgYm9pbGVycGxhdGUgcGVyIGZlYXR1cmUgdG8gYSBzaW5nbGUgZnVuY3Rpb24gY2FsbC4KLSAqKkJhc2UgdGVtcGxhdGUgcGFyZW50IG9iamVjdHMqKjogQWRkZWQgZW1wdHkgcGFyZW50IG9iamVjdHMgKGBhcGlTZXJ2ZXJMb2FkQmFsYW5jZXJgLCBgZXh0ZXJuYWxOZXR3b3JrYCwgYG5ldHdvcmtgLCBgc2VydmVyR3JvdXBgKSB0byBiYXNlIHRlbXBsYXRlcyBzbyBzdWItZmllbGQgcGF0Y2ggcGF0aHMgaGF2ZSB2YWxpZCBwYXJlbnRzIChyZXF1aXJlZCBieSBSRkMgNjkwMikuCgojIyBUZXN0IEluZnJhc3RydWN0dXJlIEltcHJvdmVtZW50cwoKVXBkYXRlZCBgc3JjL2ZlYXR1cmVzL3Rlc3QucnNgIHRvIGJldHRlciBtYXRjaCBDQVBJJ3MgYmVoYXZpb3I6CgotICoqYHRvX3JlbmRlcmVkX3ZhbHVlYCoqOiBSZXNvbHZlcyBkb3R0ZWQgdmFyaWFibGUgcGF0aHMgYnkgd2Fsa2luZyB0aGUgSlNPTiB0cmVlLCBwcmVzZXJ2aW5nIHR5cGVzIGxpa2UgYGJvb2xgIGluc3RlYWQgb2YgcmVuZGVyaW5nIHRocm91Z2ggR28gdGVtcGxhdGVzIHRvIHN0cmluZ3MuIFBhbmljcyBvbiB1bnJlc29sdmFibGUgdmFyaWFibGVzLCBtYXRjaGluZyBDQVBJJ3MgYGNhbGN1bGF0ZVZhbHVlYCBlcnJvciBiZWhhdmlvci4KLSAqKmB0b19yZW5kZXJlZF9wYXRjaGAqKiAvICoqYHRvX3BhdGNoYCoqOiBSZXR1cm4gdmFsdWVzIGRpcmVjdGx5IGFuZCBwYW5pYyBvbiB1bnJlc29sdmFibGUgdmFyaWFibGVzLCBlbnN1cmluZyBlbmFibGVkIHBhdGNoZXMgd2l0aCBtaXNzcGVsbGVkIG9yIG1pc3NpbmcgdmFyaWFibGVzIGFyZSBjYXVnaHQgYXMgdGVzdCBmYWlsdXJlcy4KLSAqKmBpc19lbmFibGVkYCoqOiBTcGVjaWFsLWNhc2VzIGBFeGVjRXJyb3I6Ok5vRmlsZWRGb3JgIChndG1wbCdzIG1pc3NpbmcgbWFwIGtleSBlcnJvcikgYXMgImRpc2FibGVkIiB0byBtYXRjaCBHbydzIGB0ZXh0L3RlbXBsYXRlYCBkZWZhdWx0IGJlaGF2aW9yLiBBbGwgb3RoZXIgcmVuZGVyIGVycm9ycyBhcmUgcHJvcGFnYXRlZC4KCiMjIFJlZ3Jlc3Npb24gVGVzdHMKCi0gYHRlc3RfcGF0Y2hlc19wcmVzZXJ2ZV9leGlzdGluZ19maWVsZHNgOiBWZXJpZmllcyB0aGF0IHBhdGNoaW5nIG9ubHkgYGVuYWJsZWRgIGRvZXMgTk9UIHJlbW92ZSBwcmUtZXhpc3RpbmcgYHByb3ZpZGVyYCBhbmQgYGZsYXZvcmAgZmllbGRzIGZyb20gdGhlIHRlbXBsYXRlLgotIGB0ZXN0X3BhdGNoZXNfdXBkYXRlX2V4aXN0aW5nX3Byb3ZpZGVyYDogVmVyaWZpZXMgdGhhdCB3aGVuIGEgbmV3IHByb3ZpZGVyIHZhbHVlIGlzIHNwZWNpZmllZCwgaXQgY29ycmVjdGx5IHJlcGxhY2VzIHRoZSBleGlzdGluZyBvbmUu
      override_checkout: master
      patchset: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
      pipeline: check
      playbook_context:
        playbook_projects:
          trusted/project_0/github.com/vexxhost/zuul-config:
            canonical_name: github.com/vexxhost/zuul-config
            checkout: main
            commit: 298983cd1253e6833abdb49d87d912527e0e6597
          trusted/project_1/opendev.org/zuul/zuul-jobs:
            canonical_name: opendev.org/zuul/zuul-jobs
            checkout: master
            commit: c75fe6ef19c05b98349573c971950c51bbf24758
          trusted/project_2/github.com/vexxhost/zuul-jobs:
            canonical_name: github.com/vexxhost/zuul-jobs
            checkout: main
            commit: a6e68243e02ef030ce5e75f8b67630880c475f33
          untrusted/project_0/opendev.org/openstack/devstack:
            canonical_name: opendev.org/openstack/devstack
            checkout: master
            commit: 0f28a404c2e4576d1b45eb123ec4e9e1b3274eea
          untrusted/project_1/opendev.org/openstack/openstack-zuul-jobs:
            canonical_name: opendev.org/openstack/openstack-zuul-jobs
            checkout: master
            commit: 3d5175f90e389f3240a8400a792abaaeb51bee3b
          untrusted/project_2/github.com/vexxhost/zuul-config:
            canonical_name: github.com/vexxhost/zuul-config
            checkout: main
            commit: 298983cd1253e6833abdb49d87d912527e0e6597
          untrusted/project_3/opendev.org/zuul/zuul-jobs:
            canonical_name: opendev.org/zuul/zuul-jobs
            checkout: master
            commit: c75fe6ef19c05b98349573c971950c51bbf24758
          untrusted/project_4/github.com/vexxhost/zuul-jobs:
            canonical_name: github.com/vexxhost/zuul-jobs
            checkout: main
            commit: a6e68243e02ef030ce5e75f8b67630880c475f33
          untrusted/project_5/github.com/vexxhost/magnum-cluster-api:
            canonical_name: github.com/vexxhost/magnum-cluster-api
            checkout: main
            commit: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
        playbooks:
        - path: untrusted/project_5/github.com/vexxhost/magnum-cluster-api/zuul.d/playbooks/hydrophone/run.yml
          roles:
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/playbook_0/role_1/devstack
            link_target: untrusted/project_0/opendev.org/openstack/devstack
            role_path: ansible/playbook_0/role_1/devstack/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/playbook_0/role_2/openstack-zuul-jobs
            link_target: untrusted/project_1/opendev.org/openstack/openstack-zuul-jobs
            role_path: ansible/playbook_0/role_2/openstack-zuul-jobs/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/playbook_0/role_4/zuul-jobs
            link_target: untrusted/project_3/opendev.org/zuul/zuul-jobs
            role_path: ansible/playbook_0/role_4/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/playbook_0/role_5/zuul-jobs
            link_target: untrusted/project_4/github.com/vexxhost/zuul-jobs
            role_path: ansible/playbook_0/role_5/zuul-jobs/roles
        post_playbooks:
        - path: untrusted/project_5/github.com/vexxhost/magnum-cluster-api/zuul.d/playbooks/hydrophone/post.yml
          roles:
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_0/role_1/devstack
            link_target: untrusted/project_0/opendev.org/openstack/devstack
            role_path: ansible/post_playbook_0/role_1/devstack/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_0/role_2/openstack-zuul-jobs
            link_target: untrusted/project_1/opendev.org/openstack/openstack-zuul-jobs
            role_path: ansible/post_playbook_0/role_2/openstack-zuul-jobs/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_0/role_4/zuul-jobs
            link_target: untrusted/project_3/opendev.org/zuul/zuul-jobs
            role_path: ansible/post_playbook_0/role_4/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/post_playbook_0/role_5/zuul-jobs
            link_target: untrusted/project_4/github.com/vexxhost/zuul-jobs
            role_path: ansible/post_playbook_0/role_5/zuul-jobs/roles
        - path: untrusted/project_0/opendev.org/openstack/devstack/playbooks/post.yaml
          roles:
          - checkout: master
            checkout_description: playbook branch
            link_name: ansible/post_playbook_1/role_0/devstack
            link_target: untrusted/project_0/opendev.org/openstack/devstack
            role_path: ansible/post_playbook_1/role_0/devstack/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_1/role_1/openstack-zuul-jobs
            link_target: untrusted/project_1/opendev.org/openstack/openstack-zuul-jobs
            role_path: ansible/post_playbook_1/role_1/openstack-zuul-jobs/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_1/role_3/zuul-jobs
            link_target: untrusted/project_3/opendev.org/zuul/zuul-jobs
            role_path: ansible/post_playbook_1/role_3/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/post_playbook_1/role_4/zuul-jobs
            link_target: untrusted/project_4/github.com/vexxhost/zuul-jobs
            role_path: ansible/post_playbook_1/role_4/zuul-jobs/roles
        - path: trusted/project_0/github.com/vexxhost/zuul-config/playbooks/base/post.yaml
          roles:
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_2/role_1/zuul-jobs
            link_target: trusted/project_1/opendev.org/zuul/zuul-jobs
            role_path: ansible/post_playbook_2/role_1/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/post_playbook_2/role_2/zuul-jobs
            link_target: trusted/project_2/github.com/vexxhost/zuul-jobs
            role_path: ansible/post_playbook_2/role_2/zuul-jobs/roles
        - path: trusted/project_0/github.com/vexxhost/zuul-config/playbooks/base/post-logs.yaml
          roles:
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/post_playbook_3/role_1/zuul-jobs
            link_target: trusted/project_1/opendev.org/zuul/zuul-jobs
            role_path: ansible/post_playbook_3/role_1/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/post_playbook_3/role_2/zuul-jobs
            link_target: trusted/project_2/github.com/vexxhost/zuul-jobs
            role_path: ansible/post_playbook_3/role_2/zuul-jobs/roles
        pre_playbooks:
        - path: trusted/project_0/github.com/vexxhost/zuul-config/playbooks/base/pre.yaml
          roles:
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/pre_playbook_0/role_1/zuul-jobs
            link_target: trusted/project_1/opendev.org/zuul/zuul-jobs
            role_path: ansible/pre_playbook_0/role_1/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/pre_playbook_0/role_2/zuul-jobs
            link_target: trusted/project_2/github.com/vexxhost/zuul-jobs
            role_path: ansible/pre_playbook_0/role_2/zuul-jobs/roles
        - path: untrusted/project_0/opendev.org/openstack/devstack/playbooks/pre.yaml
          roles:
          - checkout: master
            checkout_description: playbook branch
            link_name: ansible/pre_playbook_1/role_0/devstack
            link_target: untrusted/project_0/opendev.org/openstack/devstack
            role_path: ansible/pre_playbook_1/role_0/devstack/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/pre_playbook_1/role_1/openstack-zuul-jobs
            link_target: untrusted/project_1/opendev.org/openstack/openstack-zuul-jobs
            role_path: ansible/pre_playbook_1/role_1/openstack-zuul-jobs/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/pre_playbook_1/role_3/zuul-jobs
            link_target: untrusted/project_3/opendev.org/zuul/zuul-jobs
            role_path: ansible/pre_playbook_1/role_3/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/pre_playbook_1/role_4/zuul-jobs
            link_target: untrusted/project_4/github.com/vexxhost/zuul-jobs
            role_path: ansible/pre_playbook_1/role_4/zuul-jobs/roles
        - path: untrusted/project_5/github.com/vexxhost/magnum-cluster-api/zuul.d/playbooks/hydrophone/pre.yml
          roles:
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/pre_playbook_2/role_1/devstack
            link_target: untrusted/project_0/opendev.org/openstack/devstack
            role_path: ansible/pre_playbook_2/role_1/devstack/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/pre_playbook_2/role_2/openstack-zuul-jobs
            link_target: untrusted/project_1/opendev.org/openstack/openstack-zuul-jobs
            role_path: ansible/pre_playbook_2/role_2/openstack-zuul-jobs/roles
          - checkout: master
            checkout_description: job override ref
            link_name: ansible/pre_playbook_2/role_4/zuul-jobs
            link_target: untrusted/project_3/opendev.org/zuul/zuul-jobs
            role_path: ansible/pre_playbook_2/role_4/zuul-jobs/roles
          - checkout: main
            checkout_description: zuul branch
            link_name: ansible/pre_playbook_2/role_5/zuul-jobs
            link_target: untrusted/project_4/github.com/vexxhost/zuul-jobs
            role_path: ansible/pre_playbook_2/role_5/zuul-jobs/roles
      post_review: false
      post_timeout: null
      pre_timeout: null
      project:
        canonical_hostname: github.com
        canonical_name: github.com/vexxhost/magnum-cluster-api
        name: vexxhost/magnum-cluster-api
        short_name: magnum-cluster-api
        src_dir: src/github.com/vexxhost/magnum-cluster-api
      projects:
        github.com/novnc/novnc:
          canonical_hostname: github.com
          canonical_name: github.com/novnc/novnc
          checkout: master
          checkout_description: job override ref
          commit: 8e1ebdffba02e651c399dacef841f8941f6ad6e4
          name: novnc/novnc
          required: true
          short_name: novnc
          src_dir: src/github.com/novnc/novnc
        github.com/vexxhost/magnum-cluster-api:
          canonical_hostname: github.com
          canonical_name: github.com/vexxhost/magnum-cluster-api
          checkout: main
          checkout_description: zuul branch
          commit: 02e02b5ad23a4a54d32c81d446f37b5d0b212ace
          name: vexxhost/magnum-cluster-api
          required: false
          short_name: magnum-cluster-api
          src_dir: src/github.com/vexxhost/magnum-cluster-api
        opendev.org/openstack/barbican:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/barbican
          checkout: master
          checkout_description: job override ref
          commit: f8a331a40eb21e6c8f37e07794d57aa98b120af9
          name: openstack/barbican
          required: true
          short_name: barbican
          src_dir: src/opendev.org/openstack/barbican
        opendev.org/openstack/cinder:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/cinder
          checkout: master
          checkout_description: job override ref
          commit: 81c48732b13b4512ca63ac0fc91ccd7b0eec2863
          name: openstack/cinder
          required: true
          short_name: cinder
          src_dir: src/opendev.org/openstack/cinder
        opendev.org/openstack/devstack:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/devstack
          checkout: master
          checkout_description: job override ref
          commit: 0f28a404c2e4576d1b45eb123ec4e9e1b3274eea
          name: openstack/devstack
          required: true
          short_name: devstack
          src_dir: src/opendev.org/openstack/devstack
        opendev.org/openstack/glance:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/glance
          checkout: master
          checkout_description: job override ref
          commit: 3f8cce1d7415d8358a9679c585ce3db32c3fde8e
          name: openstack/glance
          required: true
          short_name: glance
          src_dir: src/opendev.org/openstack/glance
        opendev.org/openstack/keystone:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/keystone
          checkout: master
          checkout_description: job override ref
          commit: 91688d9dde8fcd09e6484423466a4513dbff5df7
          name: openstack/keystone
          required: true
          short_name: keystone
          src_dir: src/opendev.org/openstack/keystone
        opendev.org/openstack/magnum:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/magnum
          checkout: master
          checkout_description: job override ref
          commit: ec08b8b22b807b741c27a74e2497d58781d96fc2
          name: openstack/magnum
          required: true
          short_name: magnum
          src_dir: src/opendev.org/openstack/magnum
        opendev.org/openstack/manila:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/manila
          checkout: master
          checkout_description: job override ref
          commit: 54ac755d93d6c4128ddb8814b5d7f3906234fff4
          name: openstack/manila
          required: true
          short_name: manila
          src_dir: src/opendev.org/openstack/manila
        opendev.org/openstack/neutron:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/neutron
          checkout: master
          checkout_description: job override ref
          commit: 54befe6c096275cfe0049a2f1ab1abbee3140c8c
          name: openstack/neutron
          required: true
          short_name: neutron
          src_dir: src/opendev.org/openstack/neutron
        opendev.org/openstack/nova:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/nova
          checkout: master
          checkout_description: job override ref
          commit: b84864d9781ece8387daa967be5208b859e20d97
          name: openstack/nova
          required: true
          short_name: nova
          src_dir: src/opendev.org/openstack/nova
        opendev.org/openstack/octavia:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/octavia
          checkout: master
          checkout_description: job override ref
          commit: c86b945c21b7d5d34cf13a4e9d52810d657682da
          name: openstack/octavia
          required: true
          short_name: octavia
          src_dir: src/opendev.org/openstack/octavia
        opendev.org/openstack/os-test-images:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/os-test-images
          checkout: master
          checkout_description: job override ref
          commit: 5d0367e03788764f41da8effffa14e3eac513201
          name: openstack/os-test-images
          required: true
          short_name: os-test-images
          src_dir: src/opendev.org/openstack/os-test-images
        opendev.org/openstack/ovn-octavia-provider:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/ovn-octavia-provider
          checkout: master
          checkout_description: job override ref
          commit: a04e63332c1a1fb2f1aceb20892f9aabefb75cf7
          name: openstack/ovn-octavia-provider
          required: true
          short_name: ovn-octavia-provider
          src_dir: src/opendev.org/openstack/ovn-octavia-provider
        opendev.org/openstack/placement:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/placement
          checkout: master
          checkout_description: job override ref
          commit: e3db398fba279721121892323e6260c6932797c1
          name: openstack/placement
          required: true
          short_name: placement
          src_dir: src/opendev.org/openstack/placement
        opendev.org/openstack/python-magnumclient:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/python-magnumclient
          checkout: master
          checkout_description: job override ref
          commit: bfc9dbc2aa9a113c12e591a87f774a6d986a981f
          name: openstack/python-magnumclient
          required: true
          short_name: python-magnumclient
          src_dir: src/opendev.org/openstack/python-magnumclient
        opendev.org/openstack/requirements:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/requirements
          checkout: master
          checkout_description: job override ref
          commit: b763c1f6e124482045dd2140cb2c1a698b0e9524
          name: openstack/requirements
          required: true
          short_name: requirements
          src_dir: src/opendev.org/openstack/requirements
        opendev.org/openstack/swift:
          canonical_hostname: opendev.org
          canonical_name: opendev.org/openstack/swift
          checkout: master
          checkout_description: job override ref
          commit: e9c92154b4bc75454810ba0c779f000f4d620532
          name: openstack/swift
          required: true
          short_name: swift
          src_dir: src/opendev.org/openstack/swift
      ref: refs/pull/956/head
      resources: {}
      tenant: oss
      timeout: 7200
      topic: null
      voting: true
    zuul_copy_output:
      /etc/ceph: logs
      /etc/glusterfs/glusterd.vol: logs
      /etc/libvirt: logs
      /etc/lvm: logs
      /etc/resolv.conf: logs
      /etc/sudoers: logs
      /etc/sudoers.d: logs
      /var/log/ceph: logs
      /var/log/glusterfs: logs
      /var/log/libvirt: logs
      /var/log/mysql: logs
      /var/log/openvswitch: logs
      /var/log/postgresql: logs
      /var/log/rabbitmq: logs
      /var/log/unbound.log: logs
      '{{ devstack_conf_dir }}/.localrc.auto': logs
      '{{ devstack_conf_dir }}/.stackenv': logs
      '{{ devstack_conf_dir }}/local.conf': logs
      '{{ devstack_conf_dir }}/localrc': logs
      '{{ devstack_full_log}}': logs
      '{{ devstack_log_dir }}/atop': logs
      '{{ devstack_log_dir }}/devstacklog.txt': logs
      '{{ devstack_log_dir }}/devstacklog.txt.summary': logs
      '{{ devstack_log_dir }}/dstat-csv.log': logs
      '{{ devstack_log_dir }}/qemu.coredump': logs
      '{{ devstack_log_dir }}/tcpdump.pcap': logs
      '{{ devstack_log_dir }}/worlddump-latest.txt': logs
      '{{ stage_dir }}/apache': logs
      '{{ stage_dir }}/apache_config': logs
      '{{ stage_dir }}/audit.log': logs
      '{{ stage_dir }}/core': logs
      '{{ stage_dir }}/deprecations.log': logs
      '{{ stage_dir }}/df.txt': logs
      '{{ stage_dir }}/dpkg-l.txt': logs
      '{{ stage_dir }}/etc': logs
      '{{ stage_dir }}/iptables.txt': logs
      '{{ stage_dir }}/listen53.txt': logs
      '{{ stage_dir }}/mount.txt': logs
      '{{ stage_dir }}/performance.json': logs
      '{{ stage_dir }}/pip2-freeze.txt': logs
      '{{ stage_dir }}/pip3-freeze.txt': logs
      '{{ stage_dir }}/rpm-qa.txt': logs
      '{{ stage_dir }}/services.txt': logs
      '{{ stage_dir }}/verify_tempest_conf.log': logs
