/** * Copyright 2023 actions-toolkit authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import {describe, expect, it, test} from '@jest/globals'; import * as fs from 'fs'; import * as path from 'path'; import {Util} from '../src/util'; describe('getInputList', () => { it('single line correctly', async () => { await setInput('foo', 'bar'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar']); }); it('empty correctly', async () => { setInput('foo', ''); const res = Util.getInputList('foo'); expect(res).toEqual([]); }); it('multiline correctly', async () => { setInput('foo', 'bar\nbaz'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar', 'baz']); }); it('empty lines correctly', async () => { setInput('foo', 'bar\n\nbaz'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar', 'baz']); }); it('comma correctly', async () => { setInput('foo', 'bar,baz'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar', 'baz']); }); it('empty result correctly', async () => { setInput('foo', 'bar,baz,'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar', 'baz']); }); it('different new lines correctly', async () => { setInput('foo', 'bar\r\nbaz'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar', 'baz']); }); it('different new lines and comma correctly', async () => { setInput('foo', 'bar\r\nbaz,bat'); const res = Util.getInputList('foo'); expect(res).toEqual(['bar', 'baz', 'bat']); }); it('multiline and ignoring comma correctly', async () => { setInput('cache-from', 'user/app:cache\ntype=local,src=path/to/dir'); const res = Util.getInputList('cache-from', {ignoreComma: true}); expect(res).toEqual(['user/app:cache', 'type=local,src=path/to/dir']); }); it('multiline and ignoring comment correctly', async () => { setInput('labels', 'foo=bar\nbar=qux#baz'); const res = Util.getInputList('labels'); expect(res).toEqual(['foo=bar', 'bar=qux#baz']); }); it('multiline with comment', async () => { setInput('labels', 'foo=bar\nbar=qux#baz'); const res = Util.getInputList('labels', {comment: '#'}); expect(res).toEqual(['foo=bar', 'bar=qux']); }); it('different new lines and ignoring comma correctly', async () => { setInput('cache-from', 'user/app:cache\r\ntype=local,src=path/to/dir'); const res = Util.getInputList('cache-from', {ignoreComma: true}); expect(res).toEqual(['user/app:cache', 'type=local,src=path/to/dir']); }); it('do not escape surrounding quotes', async () => { setInput('driver-opts', `"env.no_proxy=localhost,127.0.0.1,.mydomain"`); const res = Util.getInputList('driver-opts', {ignoreComma: true, quote: false}); expect(res).toEqual(['"env.no_proxy=localhost,127.0.0.1,.mydomain"']); }); it('escape surrounding quotes', async () => { setInput('platforms', 'linux/amd64\n"linux/arm64,linux/arm/v7"'); const res = Util.getInputList('platforms'); expect(res).toEqual(['linux/amd64', 'linux/arm64', 'linux/arm/v7']); }); it('multiline values', async () => { setInput( 'secrets', `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 "MYSECRET=aaaaaaaa bbbbbbb ccccccccc" FOO=bar` ); const res = Util.getInputList('secrets', {ignoreComma: true}); expect(res).toEqual([ 'GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', `MYSECRET=aaaaaaaa bbbbbbb ccccccccc`, 'FOO=bar' ]); }); it('multiline values with empty lines', async () => { setInput( 'secrets', `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 "MYSECRET=aaaaaaaa bbbbbbb ccccccccc" FOO=bar "EMPTYLINE=aaaa bbbb ccc"` ); const res = Util.getInputList('secrets', {ignoreComma: true}); expect(res).toEqual([ 'GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', `MYSECRET=aaaaaaaa bbbbbbb ccccccccc`, 'FOO=bar', `EMPTYLINE=aaaa bbbb ccc` ]); }); it('multiline values without quotes', async () => { setInput( 'secrets', `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 MYSECRET=aaaaaaaa bbbbbbb ccccccccc FOO=bar` ); const res = Util.getInputList('secrets', {ignoreComma: true}); expect(res).toEqual(['GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', 'MYSECRET=aaaaaaaa', 'bbbbbbb', 'ccccccccc', 'FOO=bar']); }); it('large multiline values', async () => { const pgp = fs.readFileSync(path.join(__dirname, '.fixtures', 'pgp.txt'), {encoding: 'utf-8'}); setInput( 'secrets', `"GPG_KEY=${pgp}" FOO=bar` ); const res = Util.getInputList('secrets', {ignoreComma: true}); expect(res).toEqual([`GPG_KEY=${pgp}`, 'FOO=bar']); }); it('multiline values escape quotes', async () => { setInput( 'secrets', `GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789 "MYSECRET=aaaaaaaa bbbb""bbb ccccccccc" FOO=bar` ); const res = Util.getInputList('secrets', {ignoreComma: true}); expect(res).toEqual([ 'GIT_AUTH_TOKEN=abcdefgh,ijklmno=0123456789', `MYSECRET=aaaaaaaa bbbb"bbb ccccccccc`, 'FOO=bar' ]); }); it('keep quotes', async () => { const output = `type=image,"name=ghcr.io/nginxinc/nginx-unprivileged,docker.io/nginxinc/nginx-unprivileged",push-by-digest=true,name-canonical=true,push=true`; setInput('outputs', output); expect(Util.getInputList('outputs', {ignoreComma: true, quote: false})).toEqual([output]); }); }); describe('asyncForEach', () => { it('executes async tasks sequentially', async () => { const testValues = [1, 2, 3, 4, 5]; const results: number[] = []; await Util.asyncForEach(testValues, async value => { results.push(value); }); expect(results).toEqual(testValues); }); }); describe('isValidURL', () => { test.each([ ['https://github.com/docker/buildx.git', true], ['https://github.com/docker/buildx.git#refs/pull/648/head', true], ['git@github.com:moby/buildkit.git', false], ['git://github.com/user/repo.git', false], ['github.com/moby/buildkit.git#main', false], ['v0.4.1', false] ])('given %p', async (url, expected) => { expect(Util.isValidURL(url)).toEqual(expected); }); }); describe('isValidRef', () => { test.each([ ['https://github.com/docker/buildx.git', true], ['https://github.com/docker/buildx.git#refs/pull/648/head', true], ['git@github.com:moby/buildkit.git', true], ['git://github.com/user/repo.git', true], ['github.com/moby/buildkit.git#main', true], ['v0.4.1', false] ])('given %p', async (url, expected) => { expect(Util.isValidRef(url)).toEqual(expected); }); }); describe('trimPrefix', () => { test.each([ ['', 'abc', ''], ['abc', 'a', 'bc'], ['abc', 'ab', 'c'], ['abc', '', 'abc'], ['abc', '', 'abc'], ['abc', 'd', 'abc'], ['abc', 'abc', ''], ['abc', 'abcd', 'abc'], ['abcdabc', 'abc', 'dabc'], ['abcabc', 'abc', 'abc'], ['abcdabc', 'd', 'abcdabc'] ])('given %p', async (str, prefix, expected) => { expect(Util.trimPrefix(str, prefix)).toEqual(expected); }); }); describe('trimSuffix', () => { test.each([ ['', 'abc', ''], ['abc', 'c', 'ab'], ['abc', '', 'abc'], ['abc', 'bc', 'a'], ['abc', 'abc', ''], ['abc', 'abcd', 'abc'], ['abc', 'aabc', 'abc'], ['abcdabc', 'abc', 'abcd'], ['abcabc', 'abc', 'abc'], ['abcdabc', 'd', 'abcdabc'] ])('given %p', async (str, suffix, expected) => { expect(Util.trimSuffix(str, suffix)).toEqual(expected); }); }); describe('hash', () => { it('returns 2c26b46b68ffc68ff99b453c1d30413413422d706483bfa0f98a5e886266e7ae', async () => { expect(Util.hash('foo')).toEqual('2c26b46b68ffc68ff99b453c1d30413413422d706483bfa0f98a5e886266e7ae'); }); }); // https://github.com/golang/go/blob/f6b93a4c358b28b350dd8fe1780c1f78e520c09c/src/strconv/atob_test.go#L36-L58 describe('parseBool', () => { [ {input: '', expected: false, throwsError: true}, {input: 'asdf', expected: false, throwsError: true}, {input: '0', expected: false, throwsError: false}, {input: 'f', expected: false, throwsError: false}, {input: 'F', expected: false, throwsError: false}, {input: 'FALSE', expected: false, throwsError: false}, {input: 'false', expected: false, throwsError: false}, {input: 'False', expected: false, throwsError: false}, {input: '1', expected: true, throwsError: false}, {input: 't', expected: true, throwsError: false}, {input: 'T', expected: true, throwsError: false}, {input: 'TRUE', expected: true, throwsError: false}, {input: 'true', expected: true, throwsError: false}, {input: 'True', expected: true, throwsError: false} ].forEach(({input, expected, throwsError}) => { test(`parseBool("${input}")`, () => { if (throwsError) { // eslint-disable-next-line jest/no-conditional-expect expect(() => Util.parseBool(input)).toThrow(); } else { // eslint-disable-next-line jest/no-conditional-expect expect(Util.parseBool(input)).toBe(expected); } }); }); }); describe('formatFileSize', () => { test('should return "0 Bytes" when given 0 bytes', () => { expect(Util.formatFileSize(0)).toBe('0 Bytes'); }); test('should format bytes to KB correctly', () => { expect(Util.formatFileSize(1024)).toBe('1 KB'); expect(Util.formatFileSize(2048)).toBe('2 KB'); expect(Util.formatFileSize(1500)).toBe('1.46 KB'); }); test('should format bytes to MB correctly', () => { expect(Util.formatFileSize(1024 * 1024)).toBe('1 MB'); expect(Util.formatFileSize(2.5 * 1024 * 1024)).toBe('2.5 MB'); expect(Util.formatFileSize(3.8 * 1024 * 1024)).toBe('3.8 MB'); }); test('should format bytes to GB correctly', () => { expect(Util.formatFileSize(1024 * 1024 * 1024)).toBe('1 GB'); expect(Util.formatFileSize(2.5 * 1024 * 1024 * 1024)).toBe('2.5 GB'); expect(Util.formatFileSize(3.8 * 1024 * 1024 * 1024)).toBe('3.8 GB'); }); test('should format bytes to TB correctly', () => { expect(Util.formatFileSize(1024 * 1024 * 1024 * 1024)).toBe('1 TB'); expect(Util.formatFileSize(2.5 * 1024 * 1024 * 1024 * 1024)).toBe('2.5 TB'); expect(Util.formatFileSize(3.8 * 1024 * 1024 * 1024 * 1024)).toBe('3.8 TB'); }); }); describe('generateRandomString', () => { it('should generate a random string of default length 10', async () => { const res = Util.generateRandomString(); expect(typeof res).toBe('string'); expect(res.length).toBe(10); expect(/^[0-9a-f]+$/i.test(res)).toBe(true); }); it('should generate a random string of specified length', async () => { const length = 15; const res = Util.generateRandomString(length); expect(typeof res).toBe('string'); expect(res.length).toBe(15); expect(/^[0-9a-f]+$/i.test(res)).toBe(true); }); }); describe('stringToUnicodeEntities', () => { it('should convert a string to Unicode entities', () => { const input = 'Hello, World!'; const expected = 'Hello, World!'; const result = Util.stringToUnicodeEntities(input); expect(result).toEqual(expected); }); it('should handle an empty string', () => { const input = ''; const expected = ''; const result = Util.stringToUnicodeEntities(input); expect(result).toEqual(expected); }); it('should handle special characters', () => { const input = '@#^&*()'; const expected = '@#^&*()'; const result = Util.stringToUnicodeEntities(input); expect(result).toEqual(expected); }); it('should handle non-English characters', () => { const input = 'こんにちは'; const expected = 'こんにちは'; const result = Util.stringToUnicodeEntities(input); expect(result).toEqual(expected); }); }); describe('countLines', () => { it('counts total number of lines correctly', () => { const text = `This is a sample text with multiple lines`; const result = Util.countLines(text); expect(result).toEqual(10); // Including empty lines }); it('handles edge case with empty string', () => { const text = ''; const result = Util.countLines(text); expect(result).toEqual(1); // Empty string should have 1 line }); it('handles edge case with single line', () => { const text = 'Single line text'; const result = Util.countLines(text); expect(result).toEqual(1); // Single line should have 1 line }); it('handles multiple types of line breaks', () => { const text = `Line 1\r\nLine 2\rLine 3\nLine 4`; const result = Util.countLines(text); expect(result).toEqual(4); // Different line break types should be counted correctly }); }); describe('isPathRelativeTo', () => { it('should return true for a child path directly inside the parent path', () => { const parentPath = '/home/user/projects'; const childPath = '/home/user/projects/subproject'; expect(Util.isPathRelativeTo(parentPath, childPath)).toBe(true); }); it('should return true for a deeply nested child path inside the parent path', () => { const parentPath = '/home/user'; const childPath = '/home/user/projects/subproject/module'; expect(Util.isPathRelativeTo(parentPath, childPath)).toBe(true); }); it('should return false for a child path outside the parent path', () => { const parentPath = '/home/user/projects'; const childPath = '/home/user/otherprojects/subproject'; expect(Util.isPathRelativeTo(parentPath, childPath)).toBe(false); }); it('should return true for a child path specified with relative segments', () => { const parentPath = '/home/user/projects'; const childPath = '/home/user/projects/../projects/subproject'; expect(Util.isPathRelativeTo(parentPath, childPath)).toBe(true); }); it('should return false when the child path is actually a parent path', () => { const parentPath = '/home/user/projects/subproject'; const childPath = '/home/user/projects'; expect(Util.isPathRelativeTo(parentPath, childPath)).toBe(false); }); }); // See: https://github.com/actions/toolkit/blob/a1b068ec31a042ff1e10a522d8fdf0b8869d53ca/packages/core/src/core.ts#L89 function getInputName(name: string): string { return `INPUT_${name.replace(/ /g, '_').toUpperCase()}`; } function setInput(name: string, value: string): void { process.env[getInputName(name)] = value; }