As I've written about before, I'm a big fan of Renovate.

One of my favourite Renovate features is the Custom Manager Support using Regex, which allows you to capture dependencies in files that Renovate doesn't understand out-of-the-box.

However, this brings to mind an xkdc:

[Man with sunglasses talking (or, alternatively, rapping) to Cueball.] Sunglasses: If you're havin' perl problems I feel bad for you, son- Sunglasses: I got 99 problems, Sunglasses: So I used regular expressions. Sunglasses: Now I have 100 problems.

Having to maintain the regexes can get rather tricky, especially as some of them can be rather complex, or work across multiple lines.

Over the years I've written a fair few of these, and in the recent month I've been writing quite a few more, and have really been feeling the pain of trying to ensure they work.

I recently added some additional logging to Renovate for this, alongside running Renovate one-offs, but it wasn't quite the experience I wanted - I really wanted a test harness. So that's what I built!

Fixing a regex

I noticed recently that my article about Buildkite agent image management with Renovate had a bug, so let's use this as an example of how to test-drive a fix.

Setting up the test harness

The completed code can be found in the repository on GitLab.com. Below you'll find a step-by-step guide for setting it up.

Let's start by creating a new Typescript project:

npm i --save-dev typescript npm i --save-dev @tsconfig/node18 # in the case you want to pin to the version of Renovate you're using npm i --save-dev renovate@37.363.5

Then we'll create the tsconfig.json:

{ "extends": "@tsconfig/node18/tsconfig.json" }

Next, we want to set up our actual test framework, which in this case we'll use Jest:

npm i --save-dev jest @types/jest npm i --save-dev ts-jest npx ts-jest config:init

(You don't have to use Jest, but I wanted to take advantage of its snapshotting functionality for better visibility and control over assertions, and it happens to be what Renovate itself uses)

Now we've got our dependencies, we want to set up the renovate.json that we want to test:

{ "regexManagers": [ { "fileMatch": [ "buildkite\\.ya?ml", "\\.buildkite/.+\\.ya?ml$", "\\.buildkite/.+/.+\\.ya?ml$" ], "matchStrings": [ "image:\\s*\"?(?<depName>[^\\s]+):(?<currentValue>[^\\s\"]+)\"?" ], "datasourceTemplate": "docker" } ] }

With that setup complete, we can now start writing tests:

import { extractPackageFile } from 'renovate/dist/modules/manager/custom/regex' import { ExtractConfig } from 'renovate/dist/modules/manager/types' describe('renovate.json', () => { const baseConfig = require('./renovate.json') const packageFile = 'UNUSED' describe('buildkite images', () => { // unfortunately we have to index into this right now, until https://github.com/renovatebot/renovate/issues/21760 is complete const config: ExtractConfig = baseConfig.customManagers[0] const fileContents = { 'image with tag and quotes': ` image: "golang:1.19" `, 'image with tag but no quotes': ` image: golang:1.19 `, } it('matches an image with tag and quotes', () => { const content = fileContents['image with tag and quotes'] const res = extractPackageFile(content, packageFile, config) expect(res).toMatchSnapshot({ deps: [ { depName: 'golang', currentValue: '1.19' } ] }) }) it('matches an image with tag but no quotes', () => { const content = fileContents['image with tag but no quotes'] const res = extractPackageFile(content, packageFile, config) expect(res).toMatchSnapshot({ deps: [ { depName: 'golang', currentValue: '1.19' } ] }) }) }) });

This isn't exhaustive testing, but gives us a good starting point.

Test-driving new functionality

So now we've got the harness set up, let's try and change the regex, rather than just work based on what we already have in place.

Let's say that we also want to update Docker images referenced in Buildkite pipelines that are using digest pinning, such as:

image: "golang:1.22@sha256:0b55ab82ac2a54a6f8f85ec8b943b9e470c39e32c109b766bbc1b801f3fa8d3b"

Well, we can start by writing a new test case:

it('matches an image with tag and digest and quotes', () => { const content = fileContents['image with tag and digest and quotes'] const res = extractPackageFile(content, packageFile, config) expect(res).toMatchSnapshot({ deps: [ { depName: 'golang', currentValue: '1.22', currentDigest: 'sha256:0b55ab82ac2a54a6f8f85ec8b943b9e470c39e32c109b766bbc1b801f3fa8d3b' } ] }) })

Now when we run this we can see a failure:

● renovate.json › buildkite images › matches an image with tag and digest and quotes expect(received).toMatchSnapshot(properties) Snapshot name: `renovate.json buildkite images matches an image with tag and digest and quotes 1` - Expected properties - 3 + Received value + 2 Object { "deps": Array [ Object { - "currentDigest": "sha256:0b55ab82ac2a54a6f8f85ec8b943b9e470c39e32c109b766bbc1b801f3fa8d3b", - "currentValue": "1.22", - "depName": "golang", + "currentValue": "0b55ab82ac2a54a6f8f85ec8b943b9e470c39e32c109b766bbc1b801f3fa8d3b", + "depName": "golang:1.22@sha256", }, ], } 57 | const res = extractPackageFile(content, packageFile, config) 58 | > 59 | expect(res).toMatchSnapshot({ | ^ 60 | deps: [ 61 | { 62 | depName: 'golang', at Object.<anonymous> (renovate.spec.ts:59:16) › 1 snapshot failed. Snapshot Summary › 1 snapshot failed from 1 test suite. Inspect your code changes or run `npm run npx -- -u` to update them.

We can see here that our regex is currently capturing the digest as part of the name of the dependency and the digest as the actual tag, which isn't right 😅

From here, we can then either perform some trial-and-error, or do some more calculated work to modify our regex in our renovate.json, in this case to:

- "image:\\s*\"?(?<depName>[^\\s]+):(?<currentValue>[^\\s\"]+)\"?" + "image:\\s*\"?(?<depName>[^\\s:@\"]+)(?::(?<currentValue>[-a-zA-Z0-9.]+))?(?:@(?<currentDigest>sha256:[a-zA-Z0-9]+))?\"?"

Now when we run this, our tests pass 🙌🏼

Even if it helps no one else, this is going to save me a tonne of time.

Recent content in articles on Jamie Tanna | Software Engineer

https://www.jvt.me/kind/articles/