From a9dc0625cf1e3bb4e26766df754a1072dda672a1 Mon Sep 17 00:00:00 2001
From: Michael Kriese <michael.kriese@visualon.de>
Date: Wed, 6 Sep 2023 13:26:22 +0200
Subject: [PATCH] fix: better branch code coverage (#24270)

---
 .github/workflows/build.yml                   |  2 +-
 jest.config.ts                                |  4 +-
 .../internal/auto-generate-replacements.ts    |  3 +-
 lib/config/presets/internal/monorepo.ts       |  8 ++--
 .../datasource/bitbucket-tags/index.spec.ts   | 18 ++++++++
 lib/modules/datasource/conan/index.spec.ts    | 27 ++++++++++++
 lib/modules/datasource/conda/index.spec.ts    |  5 ++-
 .../datasource/dart-version/index.spec.ts     |  9 +++-
 .../datasource/endoflife-date/index.spec.ts   |  2 +-
 .../datasource/flutter-version/index.spec.ts  |  2 +-
 .../datasource/flutter-version/index.ts       |  3 +-
 lib/modules/datasource/hermit/index.ts        |  5 ++-
 lib/modules/datasource/index.ts               |  5 ++-
 .../datasource/jenkins-plugins/index.spec.ts  |  2 -
 lib/modules/datasource/pod/index.spec.ts      | 20 +++++++++
 lib/modules/datasource/pod/index.ts           |  9 ++--
 .../datasource/sbt-package/util.spec.ts       |  7 ++++
 lib/modules/datasource/sbt-package/util.ts    |  3 +-
 .../datasource/terraform-module/index.ts      |  5 ++-
 .../manager/cake/__fixtures__/build.cake      |  2 +-
 .../cake/__snapshots__/index.spec.ts.snap     | 42 -------------------
 lib/modules/manager/cake/index.spec.ts        |  4 +-
 .../manager/custom/regex/strategies.ts        | 15 ++++---
 lib/modules/manager/docker-compose/extract.ts |  4 +-
 .../multi_and_nested_image_values.yaml        |  2 +-
 .../manager/jsonnet-bundler/artifacts.ts      |  7 ++--
 .../manager/jsonnet-bundler/extract.ts        |  3 +-
 lib/modules/manager/mint/extract.spec.ts      |  4 ++
 .../manager/woodpecker/extract.spec.ts        |  1 +
 lib/modules/platform/codecommit/index.spec.ts | 13 +++++-
 lib/modules/platform/codecommit/index.ts      |  5 ++-
 lib/modules/platform/utils/pr-body.ts         |  8 ++--
 lib/modules/versioning/debian/index.spec.ts   |  4 --
 lib/modules/versioning/debian/index.ts        |  8 ++--
 lib/modules/versioning/pep440/range.ts        | 25 ++++++-----
 lib/modules/versioning/redhat/index.ts        |  2 +-
 lib/modules/versioning/rez/index.ts           | 21 ++++++----
 lib/modules/versioning/swift/index.spec.ts    |  4 ++
 lib/util/array.spec.ts                        | 11 ++++-
 lib/util/array.ts                             |  9 ++++
 lib/util/string.spec.ts                       |  9 +++-
 lib/util/string.ts                            |  4 ++
 .../model/commit-message-factory.ts           |  5 ++-
 .../model/semantic-commit-message.ts          |  4 +-
 44 files changed, 231 insertions(+), 124 deletions(-)
 create mode 100644 lib/modules/datasource/sbt-package/util.spec.ts
 delete mode 100644 lib/modules/manager/cake/__snapshots__/index.spec.ts.snap

diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml
index a913e289a6..cd1b11e43d 100644
--- a/.github/workflows/build.yml
+++ b/.github/workflows/build.yml
@@ -404,7 +404,7 @@ jobs:
       - name: Check coverage threshold
         run: |
           pnpm nyc check-coverage -t ./coverage/nyc \
-            --branches 98 \
+            --branches 98.99 \
             --functions 100 \
             --lines 100 \
             --statements 100
diff --git a/jest.config.ts b/jest.config.ts
index 76b3eaa996..d354593760 100644
--- a/jest.config.ts
+++ b/jest.config.ts
@@ -213,7 +213,9 @@ const config: JestConfig = {
   cacheDirectory: '.cache/jest',
   clearMocks: true,
   collectCoverage: true,
-  coverageReporters: ci ? ['lcovonly', 'json'] : ['html', 'text-summary'],
+  coverageReporters: ci
+    ? ['lcovonly', 'json']
+    : ['html', 'text-summary', 'json'],
   transform: {
     '\\.ts$': [
       'ts-jest',
diff --git a/lib/config/presets/internal/auto-generate-replacements.ts b/lib/config/presets/internal/auto-generate-replacements.ts
index 83b07c015b..ece2db24bf 100644
--- a/lib/config/presets/internal/auto-generate-replacements.ts
+++ b/lib/config/presets/internal/auto-generate-replacements.ts
@@ -1,3 +1,4 @@
+import { coerceArray } from '../../../util/array';
 import type { PackageRule } from '../../types';
 import type { Preset } from '../types';
 
@@ -45,7 +46,7 @@ export function addPresets(
   presets: Record<string, Preset>,
   ...templates: PresetTemplate[]
 ): void {
-  const ext = presets.all?.extends ?? [];
+  const ext = coerceArray(presets.all?.extends);
   for (const template of templates) {
     const { title, description, packageRules } = template;
     presets[title] = {
diff --git a/lib/config/presets/internal/monorepo.ts b/lib/config/presets/internal/monorepo.ts
index f5d8ca1654..d57743e8e7 100644
--- a/lib/config/presets/internal/monorepo.ts
+++ b/lib/config/presets/internal/monorepo.ts
@@ -1,4 +1,4 @@
-import is from '@sindresorhus/is';
+import { toArray } from '../../../util/array';
 import type { Preset } from '../types';
 
 /* eslint sort-keys: ["error", "asc", {caseSensitive: false, natural: true}] */
@@ -482,20 +482,20 @@ export const presets: Record<string, Preset> = {};
 for (const [name, value] of Object.entries(repoGroups)) {
   presets[name] = {
     description: `${name} monorepo`,
-    matchSourceUrls: is.array(value) ? value : [value],
+    matchSourceUrls: toArray(value),
   };
 }
 
 for (const [name, value] of Object.entries(orgGroups)) {
   presets[name] = {
     description: `${name} monorepo`,
-    matchSourceUrlPrefixes: is.array(value) ? value : [value],
+    matchSourceUrlPrefixes: toArray(value),
   };
 }
 
 for (const [name, value] of Object.entries(patternGroups)) {
   presets[name] = {
     description: `${name} monorepo`,
-    matchPackagePatterns: is.array(value) ? value : [value],
+    matchPackagePatterns: toArray(value),
   };
 }
diff --git a/lib/modules/datasource/bitbucket-tags/index.spec.ts b/lib/modules/datasource/bitbucket-tags/index.spec.ts
index d2340ea579..02f7022a9a 100644
--- a/lib/modules/datasource/bitbucket-tags/index.spec.ts
+++ b/lib/modules/datasource/bitbucket-tags/index.spec.ts
@@ -124,5 +124,23 @@ describe('modules/datasource/bitbucket-tags/index', () => {
       expect(res).toBeString();
       expect(res).toBe('123');
     });
+
+    it('returns null for missing hash', async () => {
+      const body = {
+        name: 'v1.0.0',
+      };
+      httpMock
+        .scope('https://api.bitbucket.org')
+        .get('/2.0/repositories/some/dep2/refs/tags/v1.0.0')
+        .reply(200, body);
+      const res = await getDigest(
+        {
+          datasource,
+          packageName: 'some/dep2',
+        },
+        'v1.0.0'
+      );
+      expect(res).toBeNull();
+    });
   });
 });
diff --git a/lib/modules/datasource/conan/index.spec.ts b/lib/modules/datasource/conan/index.spec.ts
index e9025f4386..f87096f168 100644
--- a/lib/modules/datasource/conan/index.spec.ts
+++ b/lib/modules/datasource/conan/index.spec.ts
@@ -51,6 +51,17 @@ describe('modules/datasource/conan/index', () => {
         '3a9b47caee2e2c1d3fb7d97788339aa8'
       );
     });
+
+    it('returns null for missing revision', async () => {
+      const version = '1.8.1';
+      httpMock
+        .scope(nonDefaultRegistryUrl)
+        .get(`/v2/conans/poco/${version}/_/_/revisions`)
+        .reply(200, []);
+      digestConfig.packageName = `poco/${version}@_/_`;
+      digestConfig.currentDigest = '4fc13d60fd91ba44fefe808ad719a5af';
+      expect(await getDigest(digestConfig, version)).toBeNull();
+    });
   });
 
   describe('getReleases', () => {
@@ -180,6 +191,22 @@ describe('modules/datasource/conan/index', () => {
       });
     });
 
+    it('works with empty releases', async () => {
+      httpMock
+        .scope('https://api.github.com')
+        .get(
+          '/repos/conan-io/conan-center-index/contents/recipes/poco/config.yml'
+        )
+        .reply(200, '');
+      expect(
+        await getPkgReleases({
+          ...config,
+          registryUrls: [defaultRegistryUrl],
+          packageName: 'poco/1.2@_/_',
+        })
+      ).toBeNull();
+    });
+
     it('rejects userAndChannel for Conan Center', async () => {
       expect(
         await getPkgReleases({
diff --git a/lib/modules/datasource/conda/index.spec.ts b/lib/modules/datasource/conda/index.spec.ts
index 0085c36c69..e77c16d2ae 100644
--- a/lib/modules/datasource/conda/index.spec.ts
+++ b/lib/modules/datasource/conda/index.spec.ts
@@ -31,7 +31,10 @@ describe('modules/datasource/conda/index', () => {
     });
 
     it('returns null for empty result', async () => {
-      httpMock.scope(defaultRegistryUrl).get(depUrl).reply(200, {});
+      httpMock
+        .scope(defaultRegistryUrl)
+        .get(depUrl)
+        .reply(200, { versions: [] });
       expect(
         await getPkgReleases({
           datasource,
diff --git a/lib/modules/datasource/dart-version/index.spec.ts b/lib/modules/datasource/dart-version/index.spec.ts
index ae41cd725d..4d612e7e56 100644
--- a/lib/modules/datasource/dart-version/index.spec.ts
+++ b/lib/modules/datasource/dart-version/index.spec.ts
@@ -34,7 +34,14 @@ describe('modules/datasource/dart-version/index', () => {
     });
 
     it('returns null for empty 200 OK', async () => {
-      httpMock.scope(baseUrl).get(urlPath).reply(200, []);
+      const scope = httpMock.scope(baseUrl);
+      for (const channel of channels) {
+        scope
+          .get(
+            `/storage/v1/b/dart-archive/o?delimiter=%2F&prefix=channels%2F${channel}%2Frelease%2F&alt=json`
+          )
+          .reply(200, { prefixes: [] });
+      }
       expect(
         await getPkgReleases({
           datasource,
diff --git a/lib/modules/datasource/endoflife-date/index.spec.ts b/lib/modules/datasource/endoflife-date/index.spec.ts
index b564776ba1..2c29e0e3c3 100644
--- a/lib/modules/datasource/endoflife-date/index.spec.ts
+++ b/lib/modules/datasource/endoflife-date/index.spec.ts
@@ -100,7 +100,7 @@ describe('modules/datasource/endoflife-date/index', () => {
     });
 
     it('returns null for empty result', async () => {
-      httpMock.scope(registryUrl).get(eksMockPath).reply(200, {});
+      httpMock.scope(registryUrl).get(eksMockPath).reply(200, []);
       expect(
         await getPkgReleases({
           datasource,
diff --git a/lib/modules/datasource/flutter-version/index.spec.ts b/lib/modules/datasource/flutter-version/index.spec.ts
index 8e333b6658..5af2bb5e92 100644
--- a/lib/modules/datasource/flutter-version/index.spec.ts
+++ b/lib/modules/datasource/flutter-version/index.spec.ts
@@ -32,7 +32,7 @@ describe('modules/datasource/flutter-version/index', () => {
     });
 
     it('returns null for empty 200 OK', async () => {
-      httpMock.scope(baseUrl).get(urlPath).reply(200, []);
+      httpMock.scope(baseUrl).get(urlPath).reply(200, { releases: [] });
       expect(
         await getPkgReleases({
           datasource,
diff --git a/lib/modules/datasource/flutter-version/index.ts b/lib/modules/datasource/flutter-version/index.ts
index 803da054fa..293134377f 100644
--- a/lib/modules/datasource/flutter-version/index.ts
+++ b/lib/modules/datasource/flutter-version/index.ts
@@ -54,10 +54,9 @@ export class FlutterVersionDatasource extends Datasource {
           releaseTimestamp: release_date,
           isStable: channel === 'stable',
         }));
+      return result.releases.length ? result : null;
     } catch (err) {
       this.handleGenericErrors(err);
     }
-
-    return result.releases.length ? result : null;
   }
 }
diff --git a/lib/modules/datasource/hermit/index.ts b/lib/modules/datasource/hermit/index.ts
index cb95421f5f..5fd91d5b4a 100644
--- a/lib/modules/datasource/hermit/index.ts
+++ b/lib/modules/datasource/hermit/index.ts
@@ -5,6 +5,7 @@ import { getApiBaseUrl } from '../../../util/github/url';
 import { GithubHttp } from '../../../util/http/github';
 import { regEx } from '../../../util/regex';
 import { streamToString } from '../../../util/streams';
+import { coerceString } from '../../../util/string';
 import { parseUrl } from '../../../util/url';
 import { id } from '../../versioning/hermit';
 import { Datasource } from '../datasource';
@@ -106,8 +107,8 @@ export class HermitDatasource extends Datasource {
   })
   async getHermitSearchManifest(u: URL): Promise<HermitSearchResult[] | null> {
     const registryUrl = u.toString();
-    const host = u.host ?? '';
-    const groups = this.pathRegex.exec(u.pathname ?? '')?.groups;
+    const host = coerceString(u.host);
+    const groups = this.pathRegex.exec(coerceString(u.pathname))?.groups;
     if (!groups) {
       logger.warn(
         { registryUrl },
diff --git a/lib/modules/datasource/index.ts b/lib/modules/datasource/index.ts
index 3bc7181d22..cd6b3b155f 100644
--- a/lib/modules/datasource/index.ts
+++ b/lib/modules/datasource/index.ts
@@ -3,6 +3,7 @@ import { dequal } from 'dequal';
 import { HOST_DISABLED } from '../../constants/error-messages';
 import { logger } from '../../logger';
 import { ExternalHostError } from '../../types/errors/external-host-error';
+import { coerceArray } from '../../util/array';
 import * as memCache from '../../util/cache/memory';
 import * as packageCache from '../../util/cache/package';
 import { clone } from '../../util/clone';
@@ -149,10 +150,10 @@ async function mergeRegistries(
         continue;
       }
       if (combinedRes) {
-        for (const existingRelease of combinedRes.releases || []) {
+        for (const existingRelease of coerceArray(combinedRes.releases)) {
           existingRelease.registryUrl ??= combinedRes.registryUrl;
         }
-        for (const additionalRelease of res.releases || []) {
+        for (const additionalRelease of coerceArray(res.releases)) {
           additionalRelease.registryUrl = res.registryUrl;
         }
         combinedRes = { ...res, ...combinedRes };
diff --git a/lib/modules/datasource/jenkins-plugins/index.spec.ts b/lib/modules/datasource/jenkins-plugins/index.spec.ts
index b45707cb40..3d386e61e8 100644
--- a/lib/modules/datasource/jenkins-plugins/index.spec.ts
+++ b/lib/modules/datasource/jenkins-plugins/index.spec.ts
@@ -22,7 +22,6 @@ const jenkinsPluginsVersions: JenkinsPluginsVersionsResponse = {
       '1.0.0': {
         version: '1.0.0',
         url: 'https://download.example.com',
-        buildDate: 'Jan 01, 2020',
       },
       '2.0.0': {
         version: '2.0.0',
@@ -83,7 +82,6 @@ describe('modules/datasource/jenkins-plugins/index', () => {
         releases: [
           {
             downloadUrl: 'https://download.example.com',
-            releaseTimestamp: '2020-01-01T00:00:00.000Z',
             version: '1.0.0',
           },
           {
diff --git a/lib/modules/datasource/pod/index.spec.ts b/lib/modules/datasource/pod/index.spec.ts
index 3f90ac87f0..9ac7b8c9af 100644
--- a/lib/modules/datasource/pod/index.spec.ts
+++ b/lib/modules/datasource/pod/index.spec.ts
@@ -1,6 +1,7 @@
 import { getPkgReleases } from '..';
 import * as httpMock from '../../../../test/http-mock';
 import { EXTERNAL_HOST_ERROR } from '../../../constants/error-messages';
+import * as hostRules from '../../../util/host-rules';
 import * as rubyVersioning from '../../versioning/ruby';
 import { PodDatasource } from '.';
 
@@ -20,6 +21,7 @@ describe('modules/datasource/pod/index', () => {
   describe('getReleases', () => {
     beforeEach(() => {
       jest.resetAllMocks();
+      hostRules.clear();
     });
 
     it('returns null for invalid inputs', async () => {
@@ -37,6 +39,16 @@ describe('modules/datasource/pod/index', () => {
       ).toBeNull();
     });
 
+    it('returns null disabled host', async () => {
+      hostRules.add({ matchHost: cocoapodsHost, enabled: false });
+      expect(
+        await getPkgReleases({
+          datasource: PodDatasource.id,
+          packageName: 'foobar',
+        })
+      ).toBeNull();
+    });
+
     it('returns null for empty result', async () => {
       // FIXME: why get request?
       httpMock
@@ -119,6 +131,14 @@ describe('modules/datasource/pod/index', () => {
       await expect(getPkgReleases(config)).rejects.toThrow(EXTERNAL_HOST_ERROR);
     });
 
+    it('throws for 500', async () => {
+      httpMock
+        .scope(cocoapodsHost)
+        .get('/all_pods_versions_a_c_b.txt')
+        .reply(500);
+      await expect(getPkgReleases(config)).rejects.toThrow(EXTERNAL_HOST_ERROR);
+    });
+
     it('returns null for unknown error', async () => {
       httpMock
         .scope(cocoapodsHost)
diff --git a/lib/modules/datasource/pod/index.ts b/lib/modules/datasource/pod/index.ts
index 6cf771fa19..92cb4aadd8 100644
--- a/lib/modules/datasource/pod/index.ts
+++ b/lib/modules/datasource/pod/index.ts
@@ -68,7 +68,6 @@ function handleError(packageName: string, err: HttpError): void {
   } else if (statusCode === 404) {
     logger.debug(errorData, 'Package lookup error');
   } else if (err.message === HOST_DISABLED) {
-    // istanbul ignore next
     logger.trace(errorData, 'Host disabled');
   } else {
     logger.warn(errorData, 'CocoaPods lookup failure: Unknown error');
@@ -77,8 +76,8 @@ function handleError(packageName: string, err: HttpError): void {
 
 function isDefaultRepo(url: string): boolean {
   const match = githubRegex.exec(url);
-  if (match) {
-    const { account, repo } = match.groups ?? {};
+  if (match?.groups) {
+    const { account, repo } = match.groups;
     return (
       account.toLowerCase() === 'cocoapods' && repo.toLowerCase() === 'specs'
     ); // https://github.com/CocoaPods/Specs.git
@@ -228,9 +227,9 @@ export class PodDatasource extends Datasource {
 
     let result: ReleaseResult | null = null;
     const match = githubRegex.exec(baseUrl);
-    if (match) {
+    if (match?.groups) {
       baseUrl = massageGithubUrl(baseUrl);
-      const { hostURL, account, repo } = match?.groups ?? {};
+      const { hostURL, account, repo } = match.groups;
       const opts = { hostURL, account, repo };
       result = await this.getReleasesFromGithub(podName, opts);
     } else {
diff --git a/lib/modules/datasource/sbt-package/util.spec.ts b/lib/modules/datasource/sbt-package/util.spec.ts
new file mode 100644
index 0000000000..4ca0304265
--- /dev/null
+++ b/lib/modules/datasource/sbt-package/util.spec.ts
@@ -0,0 +1,7 @@
+import { getLatestVersion } from './util';
+
+describe('modules/datasource/sbt-package/util', () => {
+  it('gets latest version', () => {
+    expect(getLatestVersion(['1.0.0', '3.0.0', '2.0.0'])).toBe('3.0.0');
+  });
+});
diff --git a/lib/modules/datasource/sbt-package/util.ts b/lib/modules/datasource/sbt-package/util.ts
index 7af9db18cc..c67973a4ef 100644
--- a/lib/modules/datasource/sbt-package/util.ts
+++ b/lib/modules/datasource/sbt-package/util.ts
@@ -1,3 +1,4 @@
+import { coerceArray } from '../../../util/array';
 import { regEx } from '../../../util/regex';
 import { compare } from '../../versioning/maven/compare';
 
@@ -7,7 +8,7 @@ export function parseIndexDir(
   content: string,
   filterFn = (x: string): boolean => !regEx(/^\.+/).test(x)
 ): string[] {
-  const unfiltered = content.match(linkRegExp) ?? [];
+  const unfiltered = coerceArray(content.match(linkRegExp));
   return unfiltered.filter(filterFn);
 }
 
diff --git a/lib/modules/datasource/terraform-module/index.ts b/lib/modules/datasource/terraform-module/index.ts
index 17bbf9063a..e85a5c7c99 100644
--- a/lib/modules/datasource/terraform-module/index.ts
+++ b/lib/modules/datasource/terraform-module/index.ts
@@ -1,6 +1,7 @@
 import { logger } from '../../../logger';
 import { cache } from '../../../util/cache/package/decorator';
 import { regEx } from '../../../util/regex';
+import { coerceString } from '../../../util/string';
 import * as hashicorpVersioning from '../../versioning/hashicorp';
 import type { GetReleasesConfig, ReleaseResult } from '../types';
 import { TerraformDatasource } from './base';
@@ -163,7 +164,7 @@ export class TerraformModuleDatasource extends TerraformDatasource {
 
   private static getRegistryRepository(
     packageName: string,
-    registryUrl = ''
+    registryUrl: string | undefined
   ): RegistryRepository {
     let registry: string;
     const split = packageName.split('/');
@@ -171,7 +172,7 @@ export class TerraformModuleDatasource extends TerraformDatasource {
       [registry] = split;
       split.shift();
     } else {
-      registry = registryUrl;
+      registry = coerceString(registryUrl);
     }
     if (!regEx(/^https?:\/\//).test(registry)) {
       registry = `https://${registry}`;
diff --git a/lib/modules/manager/cake/__fixtures__/build.cake b/lib/modules/manager/cake/__fixtures__/build.cake
index c678319560..faffeb069f 100644
--- a/lib/modules/manager/cake/__fixtures__/build.cake
+++ b/lib/modules/manager/cake/__fixtures__/build.cake
@@ -1,5 +1,5 @@
 foo
-#addin nuget:?package=Foo.Foo&version=1.1.1
+#addin nuget:?package=Foo.Foo
 #addin "nuget:?package=Bim.Bim&version=6.6.6"
 #tool nuget:https://example.com?package=Bar.Bar&version=2.2.2
 #module nuget:file:///tmp/?package=Baz.Baz&version=3.3.3
diff --git a/lib/modules/manager/cake/__snapshots__/index.spec.ts.snap b/lib/modules/manager/cake/__snapshots__/index.spec.ts.snap
deleted file mode 100644
index 1ab20c798f..0000000000
--- a/lib/modules/manager/cake/__snapshots__/index.spec.ts.snap
+++ /dev/null
@@ -1,42 +0,0 @@
-// Jest Snapshot v1, https://goo.gl/fbAQLP
-
-exports[`modules/manager/cake/index extracts 1`] = `
-{
-  "deps": [
-    {
-      "currentValue": "1.1.1",
-      "datasource": "nuget",
-      "depName": "Foo.Foo",
-    },
-    {
-      "currentValue": "6.6.6",
-      "datasource": "nuget",
-      "depName": "Bim.Bim",
-    },
-    {
-      "currentValue": "2.2.2",
-      "datasource": "nuget",
-      "depName": "Bar.Bar",
-      "registryUrls": [
-        "https://example.com",
-      ],
-    },
-    {
-      "currentValue": "3.3.3",
-      "datasource": "nuget",
-      "depName": "Baz.Baz",
-      "skipReason": "unsupported-url",
-    },
-    {
-      "currentValue": "1.0.3",
-      "datasource": "nuget",
-      "depName": "Cake.7zip",
-    },
-    {
-      "currentValue": "1.0.0",
-      "datasource": "nuget",
-      "depName": "Cake.asciidoctorj",
-    },
-  ],
-}
-`;
diff --git a/lib/modules/manager/cake/index.spec.ts b/lib/modules/manager/cake/index.spec.ts
index e129be4c12..0fb2e913fc 100644
--- a/lib/modules/manager/cake/index.spec.ts
+++ b/lib/modules/manager/cake/index.spec.ts
@@ -3,9 +3,9 @@ import { extractPackageFile } from '.';
 
 describe('modules/manager/cake/index', () => {
   it('extracts', () => {
-    expect(extractPackageFile(Fixtures.get('build.cake'))).toMatchSnapshot({
+    expect(extractPackageFile(Fixtures.get('build.cake'))).toMatchObject({
       deps: [
-        { depName: 'Foo.Foo', currentValue: '1.1.1' },
+        { depName: 'Foo.Foo', currentValue: undefined },
         { depName: 'Bim.Bim', currentValue: '6.6.6' },
         { depName: 'Bar.Bar', registryUrls: ['https://example.com'] },
         { depName: 'Baz.Baz', skipReason: 'unsupported-url' },
diff --git a/lib/modules/manager/custom/regex/strategies.ts b/lib/modules/manager/custom/regex/strategies.ts
index 182c43e693..a77c072be4 100644
--- a/lib/modules/manager/custom/regex/strategies.ts
+++ b/lib/modules/manager/custom/regex/strategies.ts
@@ -12,7 +12,7 @@ import {
 
 export function handleAny(
   content: string,
-  packageFile: string,
+  _packageFile: string,
   config: RegexManagerConfig
 ): PackageDependency[] {
   return config.matchStrings
@@ -20,7 +20,12 @@ export function handleAny(
     .flatMap((regex) => regexMatchAll(regex, content)) // match all regex to content, get all matches, reduce to single array
     .map((matchResult) =>
       createDependency(
-        { groups: matchResult.groups ?? {}, replaceString: matchResult[0] },
+        {
+          groups:
+            matchResult.groups ??
+            /* istanbul ignore next: can this happen? */ {},
+          replaceString: matchResult[0],
+        },
         config
       )
     )
@@ -30,7 +35,7 @@ export function handleAny(
 
 export function handleCombination(
   content: string,
-  packageFile: string,
+  _packageFile: string,
   config: RegexManagerConfig
 ): PackageDependency[] {
   const matches = config.matchStrings
@@ -43,7 +48,7 @@ export function handleCombination(
 
   const extraction = matches
     .map((match) => ({
-      groups: match.groups ?? {},
+      groups: match.groups ?? /* istanbul ignore next: can this happen? */ {},
       replaceString:
         match?.groups?.currentValue ?? match?.groups?.currentDigest
           ? match[0]
@@ -93,7 +98,7 @@ function processRecursive(parameters: RecursionParameter): PackageDependency[] {
       },
       config
     );
-    return result ? [result] : [];
+    return result ? [result] : /* istanbul ignore next: can this happen? */ [];
   }
   return regexMatchAll(regexes[index], content).flatMap((match) => {
     return processRecursive({
diff --git a/lib/modules/manager/docker-compose/extract.ts b/lib/modules/manager/docker-compose/extract.ts
index be6014ab0d..a61843992d 100644
--- a/lib/modules/manager/docker-compose/extract.ts
+++ b/lib/modules/manager/docker-compose/extract.ts
@@ -71,7 +71,9 @@ export function extractPackageFile(
 
     // Image name/tags for services are only eligible for update if they don't
     // use variables and if the image is not built locally
-    const deps = Object.values(services || {})
+    const deps = Object.values(
+      services || /* istanbul ignore next: can never happen */ {}
+    )
       .filter((service) => is.string(service?.image) && !service?.build)
       .map((service) => {
         const dep = getDep(service.image, true, extractConfig.registryAliases);
diff --git a/lib/modules/manager/helm-values/__fixtures__/multi_and_nested_image_values.yaml b/lib/modules/manager/helm-values/__fixtures__/multi_and_nested_image_values.yaml
index afb3969032..5664de2a21 100644
--- a/lib/modules/manager/helm-values/__fixtures__/multi_and_nested_image_values.yaml
+++ b/lib/modules/manager/helm-values/__fixtures__/multi_and_nested_image_values.yaml
@@ -28,4 +28,4 @@ empty_key:
 coreImage:
   registry: docker.io
   repository: bitnami/harbor-core
-  tag: 2.1.3-debian-10-r38
+  version: 2.1.3-debian-10-r38
diff --git a/lib/modules/manager/jsonnet-bundler/artifacts.ts b/lib/modules/manager/jsonnet-bundler/artifacts.ts
index c3216055fd..19027353bd 100644
--- a/lib/modules/manager/jsonnet-bundler/artifacts.ts
+++ b/lib/modules/manager/jsonnet-bundler/artifacts.ts
@@ -1,6 +1,7 @@
 import { quote } from 'shlex';
 import { TEMPORARY_ERROR } from '../../../constants/error-messages';
 import { logger } from '../../../logger';
+import { coerceArray } from '../../../util/array';
 import { exec } from '../../../util/exec';
 import type { ExecOptions, ToolConstraint } from '../../../util/exec/types';
 import { readLocalFile } from '../../../util/fs';
@@ -66,7 +67,7 @@ export async function updateArtifacts(
 
     const res: UpdateArtifactsResult[] = [];
 
-    for (const f of status.modified ?? []) {
+    for (const f of coerceArray(status.modified)) {
       res.push({
         file: {
           type: 'addition',
@@ -75,7 +76,7 @@ export async function updateArtifacts(
         },
       });
     }
-    for (const f of status.not_added ?? []) {
+    for (const f of coerceArray(status.not_added)) {
       res.push({
         file: {
           type: 'addition',
@@ -84,7 +85,7 @@ export async function updateArtifacts(
         },
       });
     }
-    for (const f of status.deleted ?? []) {
+    for (const f of coerceArray(status.deleted)) {
       res.push({
         file: {
           type: 'deletion',
diff --git a/lib/modules/manager/jsonnet-bundler/extract.ts b/lib/modules/manager/jsonnet-bundler/extract.ts
index 28d1f42715..f2565b3c5e 100644
--- a/lib/modules/manager/jsonnet-bundler/extract.ts
+++ b/lib/modules/manager/jsonnet-bundler/extract.ts
@@ -1,6 +1,7 @@
 import { join } from 'upath';
 import { logger } from '../../../logger';
 import { coerceArray } from '../../../util/array';
+import { coerceString } from '../../../util/string';
 import { parseUrl } from '../../../util/url';
 import type { PackageDependency, PackageFileContent } from '../types';
 import type { Dependency, JsonnetFile } from './types';
@@ -52,7 +53,7 @@ function extractDependency(dependency: Dependency): PackageDependency | null {
   const depName = join(
     gitRemote.host,
     gitRemote.pathname.replace(/\.git$/, ''),
-    dependency.source.git.subdir ?? ''
+    coerceString(dependency.source.git.subdir)
   );
 
   return {
diff --git a/lib/modules/manager/mint/extract.spec.ts b/lib/modules/manager/mint/extract.spec.ts
index 561159c345..d0506a0b1e 100644
--- a/lib/modules/manager/mint/extract.spec.ts
+++ b/lib/modules/manager/mint/extract.spec.ts
@@ -3,6 +3,10 @@ import { extractPackageFile } from '.';
 
 describe('modules/manager/mint/extract', () => {
   describe('extractPackageFile()', () => {
+    it('returns null for empty', () => {
+      expect(extractPackageFile('')).toBeNull();
+    });
+
     it('Mintfile With Version Description', () => {
       const res = extractPackageFile(codeBlock`
         SwiftGen/SwiftGen@6.6.1
diff --git a/lib/modules/manager/woodpecker/extract.spec.ts b/lib/modules/manager/woodpecker/extract.spec.ts
index 90774059fe..d08ca2bbcd 100644
--- a/lib/modules/manager/woodpecker/extract.spec.ts
+++ b/lib/modules/manager/woodpecker/extract.spec.ts
@@ -11,6 +11,7 @@ describe('modules/manager/woodpecker/extract', () => {
 
     it('returns null for non-object YAML', () => {
       expect(extractPackageFile('nothing here', '', {})).toBeNull();
+      expect(extractPackageFile('clone: null', '', {})).toBeNull();
     });
 
     it('returns null for malformed YAML', () => {
diff --git a/lib/modules/platform/codecommit/index.spec.ts b/lib/modules/platform/codecommit/index.spec.ts
index 6dfc43483a..e82ff378cf 100644
--- a/lib/modules/platform/codecommit/index.spec.ts
+++ b/lib/modules/platform/codecommit/index.spec.ts
@@ -42,6 +42,9 @@ describe('modules/platform/codecommit/index', () => {
   });
 
   beforeEach(() => {
+    delete process.env.AWS_REGION;
+    delete process.env.AWS_ACCESS_KEY_ID;
+    delete process.env.AWS_SECRET_ACCESS_KEY;
     codeCommitClient.reset();
     config.prList = undefined;
     config.repository = undefined;
@@ -70,7 +73,6 @@ describe('modules/platform/codecommit/index', () => {
     });
 
     it('should init with env vars', async () => {
-      const temp = process.env.AWS_REGION;
       process.env.AWS_REGION = 'REGION';
       await expect(
         codeCommit.initPlatform({
@@ -80,7 +82,6 @@ describe('modules/platform/codecommit/index', () => {
       ).resolves.toEqual({
         endpoint: 'https://git-codecommit.REGION.amazonaws.com/',
       });
-      process.env.AWS_REGION = temp;
     });
 
     it('should ', async () => {
@@ -588,6 +589,14 @@ describe('modules/platform/codecommit/index', () => {
       const res = await codeCommit.getJsonFile('file.json');
       expect(res).toEqual({ foo: 'bar' });
     });
+
+    it('returns null', async () => {
+      codeCommitClient
+        .on(GetFileCommand)
+        .resolvesOnce({ fileContent: undefined });
+      const res = await codeCommit.getJsonFile('file.json');
+      expect(res).toBeNull();
+    });
   });
 
   describe('getRawFile()', () => {
diff --git a/lib/modules/platform/codecommit/index.ts b/lib/modules/platform/codecommit/index.ts
index dbc6b4109a..c9877db7f1 100644
--- a/lib/modules/platform/codecommit/index.ts
+++ b/lib/modules/platform/codecommit/index.ts
@@ -13,6 +13,7 @@ import {
 } from '../../../constants/error-messages';
 import { logger } from '../../../logger';
 import type { BranchStatus, PrState } from '../../../types';
+import { coerceArray } from '../../../util/array';
 import * as git from '../../../util/git';
 import { regEx } from '../../../util/regex';
 import { sanitize } from '../../../util/sanitize';
@@ -163,7 +164,7 @@ export async function getPrList(): Promise<CodeCommitPr[]> {
     return fetchedPrs;
   }
 
-  const prIds = listPrsResponse.pullRequestIds ?? [];
+  const prIds = coerceArray(listPrsResponse.pullRequestIds);
 
   for (const prId of prIds) {
     const prRes = await client.getPr(prId);
@@ -291,7 +292,7 @@ export async function getRepos(): Promise<string[]> {
 
   const res: string[] = [];
 
-  const repoNames = reposRes?.repositories ?? [];
+  const repoNames = coerceArray(reposRes?.repositories);
 
   for (const repo of repoNames) {
     if (repo.repositoryName) {
diff --git a/lib/modules/platform/utils/pr-body.ts b/lib/modules/platform/utils/pr-body.ts
index a64c78bfc7..4dcf7cc4c0 100644
--- a/lib/modules/platform/utils/pr-body.ts
+++ b/lib/modules/platform/utils/pr-body.ts
@@ -11,14 +11,14 @@ export function smartTruncate(input: string, len: number): string {
   }
 
   const reMatch = re.exec(input);
-  if (!reMatch) {
+  if (!reMatch?.groups) {
     return input.substring(0, len);
   }
 
   const divider = `\n\n</details>\n\n---\n\n### Configuration`;
-  const preNotes = reMatch.groups?.preNotes ?? '';
-  const releaseNotes = reMatch.groups?.releaseNotes ?? '';
-  const postNotes = reMatch.groups?.postNotes ?? '';
+  const preNotes = reMatch.groups.preNotes;
+  const releaseNotes = reMatch.groups.releaseNotes;
+  const postNotes = reMatch.groups.postNotes;
 
   const availableLength =
     len - (preNotes.length + postNotes.length + divider.length);
diff --git a/lib/modules/versioning/debian/index.spec.ts b/lib/modules/versioning/debian/index.spec.ts
index 8db760b072..ce19fa9a12 100644
--- a/lib/modules/versioning/debian/index.spec.ts
+++ b/lib/modules/versioning/debian/index.spec.ts
@@ -11,10 +11,6 @@ describe('modules/versioning/debian/index', () => {
     Settings.now = () => dt.valueOf();
   });
 
-  afterEach(() => {
-    jest.resetAllMocks();
-  });
-
   it.each`
     version           | expected
     ${undefined}      | ${false}
diff --git a/lib/modules/versioning/debian/index.ts b/lib/modules/versioning/debian/index.ts
index d1c9b740ea..310df655bd 100644
--- a/lib/modules/versioning/debian/index.ts
+++ b/lib/modules/versioning/debian/index.ts
@@ -30,7 +30,7 @@ export class DebianVersioningApi extends GenericVersioningApi {
     const schedule = this._distroInfo.getSchedule(
       this._rollingReleases.getVersionByLts(version)
     );
-    return (isValid && schedule && RELEASE_PROP in schedule) ?? false;
+    return isValid && schedule !== null && RELEASE_PROP in schedule;
   }
 
   override isStable(version: string): boolean {
@@ -43,7 +43,6 @@ export class DebianVersioningApi extends GenericVersioningApi {
   override getNewValue({
     currentValue,
     rangeStrategy,
-    currentVersion,
     newVersion,
   }: NewValueConfig): string {
     if (rangeStrategy === 'pin') {
@@ -83,7 +82,10 @@ export class DebianVersioningApi extends GenericVersioningApi {
     // newVersion is [oldold|old|]stable
     // current value is numeric
     if (this._rollingReleases.has(newVersion)) {
-      return this._rollingReleases.schedule(newVersion)?.version ?? newVersion;
+      return (
+        this._rollingReleases.schedule(newVersion)?.version ??
+        /* istanbul ignore next: should never happen */ newVersion
+      );
     }
 
     return this._distroInfo.getVersionByCodename(newVersion);
diff --git a/lib/modules/versioning/pep440/range.ts b/lib/modules/versioning/pep440/range.ts
index a3d60c1d27..54eca71de4 100644
--- a/lib/modules/versioning/pep440/range.ts
+++ b/lib/modules/versioning/pep440/range.ts
@@ -2,6 +2,7 @@ import { gte, lt, lte, satisfies } from '@renovatebot/pep440';
 import { parse as parseRange } from '@renovatebot/pep440/lib/specifier.js';
 import { parse as parseVersion } from '@renovatebot/pep440/lib/version.js';
 import { logger } from '../../../logger';
+import { coerceArray } from '../../../util/array';
 import { regEx } from '../../../util/regex';
 import type { NewValueConfig } from '../types';
 
@@ -28,8 +29,9 @@ type UserPolicy =
  * @returns A {@link UserPolicy}
  */
 function getRangePrecision(ranges: Range[]): UserPolicy {
-  const bound: number[] =
-    parseVersion((ranges[1] || ranges[0]).version)?.release ?? [];
+  const bound = coerceArray(
+    parseVersion((ranges[1] || ranges[0]).version)?.release
+  );
   let rangePrecision = -1;
   // range is defined by a single bound.
   // ie. <1.2.2.3,
@@ -39,7 +41,7 @@ function getRangePrecision(ranges: Range[]): UserPolicy {
   }
   // Range is defined by both upper and lower bounds.
   if (ranges.length === 2) {
-    const lowerBound: number[] = parseVersion(ranges[0].version)?.release ?? [];
+    const lowerBound = coerceArray(parseVersion(ranges[0].version)?.release);
     rangePrecision = bound.findIndex((el, index) => el > lowerBound[index]);
   }
   // Tune down Major precision if followed by a zero
@@ -74,11 +76,12 @@ function getFutureVersion(
   newVersion: string,
   baseVersion?: string
 ): number[] {
-  const toRelease: number[] = parseVersion(newVersion)?.release ?? [];
-  const baseRelease: number[] =
-    parseVersion(baseVersion ?? newVersion)?.release ?? [];
+  const toRelease = coerceArray(parseVersion(newVersion)?.release);
+  const baseRelease = coerceArray(
+    parseVersion(baseVersion ?? newVersion)?.release
+  );
   return baseRelease.map((_, index) => {
-    const toPart: number = toRelease[index] ?? 0;
+    const toPart = toRelease[index] ?? 0;
     if (index < policy) {
       return toPart;
     }
@@ -303,8 +306,8 @@ function updateRangeValue(
     return range.operator + futureVersion + '.*';
   }
   if (range.operator === '~=') {
-    const baseVersion = parseVersion(range.version)?.release ?? [];
-    const futureVersion = parseVersion(newVersion)?.release ?? [];
+    const baseVersion = coerceArray(parseVersion(range.version)?.release);
+    const futureVersion = coerceArray(parseVersion(newVersion)?.release);
     const baseLen = baseVersion.length;
     const newVerLen = futureVersion.length;
     // trim redundant trailing version specifiers
@@ -410,7 +413,7 @@ function handleWidenStrategy(
   return newRanges.map((range) => {
     // newVersion is over the upper bound
     if (range.operator === '<' && gte(newVersion, range.version)) {
-      const upperBound = parseVersion(range.version)?.release ?? [];
+      const upperBound = coerceArray(parseVersion(range.version)?.release);
       const len = upperBound.length;
       // Match the precision of the smallest specifier if other than 0
       if (upperBound[len - 1] !== 0) {
@@ -474,7 +477,7 @@ function handleReplaceStrategy(
         return '>=' + newVersion;
       }
       // update the lower bound to reflect the accepted new version
-      const lowerBound = parseVersion(range.version)?.release ?? [];
+      const lowerBound = coerceArray(parseVersion(range.version)?.release);
       const rangePrecision = lowerBound.length - 1;
       let newBase = getFutureVersion(rangePrecision, newVersion);
       if (trimZeros) {
diff --git a/lib/modules/versioning/redhat/index.ts b/lib/modules/versioning/redhat/index.ts
index 21e468c907..e950358238 100644
--- a/lib/modules/versioning/redhat/index.ts
+++ b/lib/modules/versioning/redhat/index.ts
@@ -20,7 +20,7 @@ class RedhatVersioningApi extends GenericVersioningApi {
 
     const { major, minor, patch, releaseMajor, releaseMinor } = matches;
     const release = [
-      typeof major === 'undefined' ? 0 : Number.parseInt(major, 10),
+      Number.parseInt(major, 10),
       typeof minor === 'undefined' ? 0 : Number.parseInt(minor, 10),
       typeof patch === 'undefined' ? 0 : Number.parseInt(patch, 10),
       typeof releaseMajor === 'undefined'
diff --git a/lib/modules/versioning/rez/index.ts b/lib/modules/versioning/rez/index.ts
index d094875f05..06ab51161d 100644
--- a/lib/modules/versioning/rez/index.ts
+++ b/lib/modules/versioning/rez/index.ts
@@ -1,5 +1,6 @@
 import type { RangeStrategy } from '../../../types/versioning';
 import { regEx } from '../../../util/regex';
+import { coerceString } from '../../../util/string';
 import { api as npm } from '../npm';
 import { api as pep440 } from '../pep440';
 import type { NewValueConfig, VersioningApi } from '../types';
@@ -160,10 +161,12 @@ function getNewValue({
     const lowerAscVersionCurrent = matchAscRange.groups.range_lower_asc_version;
     const upperAscVersionCurrent = matchAscRange.groups.range_upper_asc_version;
     const [lowerBoundAscPep440, upperBoundAscPep440] = pep440Value.split(', ');
-    const lowerAscVersionNew =
-      regEx(versionGroup).exec(lowerBoundAscPep440)?.[0] ?? '';
-    const upperAscVersionNew =
-      regEx(versionGroup).exec(upperBoundAscPep440)?.[0] ?? '';
+    const lowerAscVersionNew = coerceString(
+      regEx(versionGroup).exec(lowerBoundAscPep440)?.[0]
+    );
+    const upperAscVersionNew = coerceString(
+      regEx(versionGroup).exec(upperBoundAscPep440)?.[0]
+    );
     const lowerBoundAscNew = lowerBoundAscCurrent.replace(
       lowerAscVersionCurrent,
       lowerAscVersionNew
@@ -189,10 +192,12 @@ function getNewValue({
     const [lowerBoundDescPep440, upperBoundDescPep440] =
       pep440Value.split(', ');
 
-    const upperDescVersionNew =
-      regEx(versionGroup).exec(upperBoundDescPep440)?.[0] ?? '';
-    const lowerDescVersionNew =
-      regEx(versionGroup).exec(lowerBoundDescPep440)?.[0] ?? '';
+    const upperDescVersionNew = coerceString(
+      regEx(versionGroup).exec(upperBoundDescPep440)?.[0]
+    );
+    const lowerDescVersionNew = coerceString(
+      regEx(versionGroup).exec(lowerBoundDescPep440)?.[0]
+    );
     const upperBoundDescNew = upperBoundDescCurrent.replace(
       upperDescVersionCurrent,
       upperDescVersionNew
diff --git a/lib/modules/versioning/swift/index.spec.ts b/lib/modules/versioning/swift/index.spec.ts
index ee47bb36f3..b79c6addd3 100644
--- a/lib/modules/versioning/swift/index.spec.ts
+++ b/lib/modules/versioning/swift/index.spec.ts
@@ -61,6 +61,7 @@ describe('modules/versioning/swift/index', () => {
     versions                          | range           | expected
     ${['1.2.3', '1.2.4', '1.2.5']}    | ${'..<"1.2.4"'} | ${'1.2.3'}
     ${['v1.2.3', 'v1.2.4', 'v1.2.5']} | ${'..<"1.2.4"'} | ${'1.2.3'}
+    ${['v1.2.3', 'v1.2.4', 'v1.2.5']} | ${''}           | ${null}
   `(
     'minSatisfyingVersion($versions, "$range") === "$expected"',
     ({ versions, range, expected }) => {
@@ -73,6 +74,7 @@ describe('modules/versioning/swift/index', () => {
     ${['1.2.3', '1.2.4', '1.2.5']}    | ${'..<"1.2.4"'} | ${'1.2.3'}
     ${['v1.2.3', 'v1.2.4', 'v1.2.5']} | ${'..<"1.2.4"'} | ${'1.2.3'}
     ${['1.2.3', '1.2.4', '1.2.5']}    | ${'..."1.2.4"'} | ${'1.2.4'}
+    ${['1.2.3', '1.2.4', '1.2.5']}    | ${''}           | ${null}
   `(
     'getSatisfyingVersion($versions, "$range") === "$expected"',
     ({ versions, range, expected }) => {
@@ -86,6 +88,7 @@ describe('modules/versioning/swift/index', () => {
     ${'v1.2.3'} | ${'..."1.2.4"'} | ${false}
     ${'1.2.3'}  | ${'"1.2.4"...'} | ${true}
     ${'v1.2.3'} | ${'"1.2.4"...'} | ${true}
+    ${'v1.2.3'} | ${''}           | ${false}
   `(
     'isLessThanRange("$version", "$range") === "$expected"',
     ({ version, range, expected }) => {
@@ -99,6 +102,7 @@ describe('modules/versioning/swift/index', () => {
     ${'v1.2.4'} | ${'..."1.2.4"'} | ${true}
     ${'1.2.4'}  | ${'..."1.2.3"'} | ${false}
     ${'v1.2.4'} | ${'..."1.2.3"'} | ${false}
+    ${'v1.2.4'} | ${''}           | ${false}
   `(
     'matches("$version", "$range") === "$expected"',
     ({ version, range, expected }) => {
diff --git a/lib/util/array.spec.ts b/lib/util/array.spec.ts
index 57356be7de..965aa62570 100644
--- a/lib/util/array.spec.ts
+++ b/lib/util/array.spec.ts
@@ -1,4 +1,4 @@
-import { isNotNullOrUndefined } from './array';
+import { isNotNullOrUndefined, toArray } from './array';
 
 describe('util/array', () => {
   it.each`
@@ -9,4 +9,13 @@ describe('util/array', () => {
   `('.isNotNullOrUndefined', ({ a, exp }) => {
     expect(isNotNullOrUndefined(a)).toEqual(exp);
   });
+
+  it.each`
+    a            | exp
+    ${null}      | ${[null]}
+    ${undefined} | ${[undefined]}
+    ${[]}        | ${[]}
+  `('.toArray', ({ a, exp }) => {
+    expect(toArray(a)).toEqual(exp);
+  });
 });
diff --git a/lib/util/array.ts b/lib/util/array.ts
index 0175fabf26..f4ad9412fe 100644
--- a/lib/util/array.ts
+++ b/lib/util/array.ts
@@ -19,3 +19,12 @@ export function isNotNullOrUndefined<T>(
 ): value is T {
   return !is.nullOrUndefined(value);
 }
+
+/**
+ * Converts a single value or an array of values to an array of values.
+ * @param value a single value or an array of values
+ * @returns array of values
+ */
+export function toArray<T>(value: T | T[]): T[] {
+  return is.array(value) ? value : [value];
+}
diff --git a/lib/util/string.spec.ts b/lib/util/string.spec.ts
index 202b15e5d1..3394a02364 100644
--- a/lib/util/string.spec.ts
+++ b/lib/util/string.spec.ts
@@ -1,4 +1,4 @@
-import { looseEquals, replaceAt } from './string';
+import { coerceString, looseEquals, replaceAt } from './string';
 
 describe('util/string', () => {
   describe('replaceAt', () => {
@@ -32,4 +32,11 @@ describe('util/string', () => {
       expect(looseEquals(null, '')).toBeFalse();
     });
   });
+
+  it('coerceString', () => {
+    expect(coerceString('foo')).toBe('foo');
+    expect(coerceString('')).toBe('');
+    expect(coerceString(undefined)).toBe('');
+    expect(coerceString(null)).toBe('');
+  });
 });
diff --git a/lib/util/string.ts b/lib/util/string.ts
index d62a1be1a2..ce6bad089d 100644
--- a/lib/util/string.ts
+++ b/lib/util/string.ts
@@ -82,3 +82,7 @@ export function copystr(x: string): string {
   buf.write(x, 'utf8');
   return buf.toString('utf8');
 }
+
+export function coerceString(val: string | null | undefined): string {
+  return val ?? '';
+}
diff --git a/lib/workers/repository/model/commit-message-factory.ts b/lib/workers/repository/model/commit-message-factory.ts
index 3feb7c594e..e0c65ffa72 100644
--- a/lib/workers/repository/model/commit-message-factory.ts
+++ b/lib/workers/repository/model/commit-message-factory.ts
@@ -1,4 +1,5 @@
 import type { RenovateSharedConfig } from '../../../config/types';
+import { coerceString } from '../../../util/string';
 import type { CommitMessage } from './commit-message';
 import { CustomCommitMessage } from './custom-commit-message';
 import { SemanticCommitMessage } from './semantic-commit-message';
@@ -29,8 +30,8 @@ export class CommitMessageFactory {
   private createSemanticCommitMessage(): SemanticCommitMessage {
     const message = new SemanticCommitMessage();
 
-    message.type = this._config.semanticCommitType ?? '';
-    message.scope = this._config.semanticCommitScope ?? '';
+    message.type = coerceString(this._config.semanticCommitType);
+    message.scope = coerceString(this._config.semanticCommitScope);
 
     return message;
   }
diff --git a/lib/workers/repository/model/semantic-commit-message.ts b/lib/workers/repository/model/semantic-commit-message.ts
index a3e5d15888..1b20315b33 100644
--- a/lib/workers/repository/model/semantic-commit-message.ts
+++ b/lib/workers/repository/model/semantic-commit-message.ts
@@ -27,11 +27,11 @@ export class SemanticCommitMessage extends CommitMessage {
   static fromString(value: string): SemanticCommitMessage | undefined {
     const match = value.match(SemanticCommitMessage.REGEXP);
 
-    if (!match) {
+    if (!match?.groups) {
       return undefined;
     }
 
-    const { groups = {} } = match;
+    const { groups } = match;
     const message = new SemanticCommitMessage();
     message.type = groups.type;
     message.scope = groups.scope;
-- 
GitLab