From 97aa1fb7cc7595ca696fbe72fdf4982dd9815dfd Mon Sep 17 00:00:00 2001
From: Charles Lyding <19598772+clydin@users.noreply.github.com>
Date: Fri, 18 Dec 2020 20:46:28 -0500
Subject: [PATCH] test(@angular-devkit/build-angular): add browser builder
assets option tests
This change adds expanded unit tests for the browser builder's `assets` option using the builder test harness.
---
.../src/browser/tests/options/assets_spec.ts | 394 ++++++++++++++++++
1 file changed, 394 insertions(+)
create mode 100644 packages/angular_devkit/build_angular/src/browser/tests/options/assets_spec.ts
diff --git a/packages/angular_devkit/build_angular/src/browser/tests/options/assets_spec.ts b/packages/angular_devkit/build_angular/src/browser/tests/options/assets_spec.ts
new file mode 100644
index 0000000000..7893869156
--- /dev/null
+++ b/packages/angular_devkit/build_angular/src/browser/tests/options/assets_spec.ts
@@ -0,0 +1,394 @@
+/**
+ * @license
+ * Copyright Google Inc. All Rights Reserved.
+ *
+ * Use of this source code is governed by an MIT-style license that can be
+ * found in the LICENSE file at https://angular.io/license
+ */
+// tslint:disable:no-big-function
+import { buildWebpackBrowser } from '../../index';
+import { BASE_OPTIONS, BROWSER_BUILDER_INFO, describeBuilder } from '../setup';
+
+describeBuilder(buildWebpackBrowser, BROWSER_BUILDER_INFO, (harness) => {
+ describe('Option: "assets"', () => {
+ beforeEach(async () => {
+ // Application code is not needed for asset tests
+ await harness.writeFile('src/main.ts', '');
+ });
+
+ it('supports an empty array value', async () => {
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+ });
+
+ it('supports mixing shorthand and longhand syntax', async () => {
+ await harness.writeFile('src/files/test.svg', '');
+ await harness.writeFile('src/files/another.file', 'asset file');
+ await harness.writeFile('src/extra.file', 'extra file');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: ['src/extra.file', { glob: '*', input: 'src/files', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/extra.file').content.toBe('extra file');
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').content.toBe('asset file');
+ });
+
+ describe('shorthand syntax', () => {
+ it('copies a single asset', async () => {
+ await harness.writeFile('src/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: ['src/test.svg'],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ });
+
+ it('copies multiple assets', async () => {
+ await harness.writeFile('src/test.svg', '');
+ await harness.writeFile('src/another.file', 'asset file');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: ['src/test.svg', 'src/another.file'],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').content.toBe('asset file');
+ });
+
+ it('copies an asset with directory and maintains directory in output', async () => {
+ await harness.writeFile('src/subdirectory/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: ['src/subdirectory/test.svg'],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/subdirectory/test.svg').content.toBe('');
+ });
+
+ it('does not fail if asset does not exist', async () => {
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: ['src/test.svg'],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').toNotExist();
+ });
+
+ it('throws exception if asset path is not within project source root', async () => {
+ await harness.writeFile('test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: ['test.svg'],
+ });
+
+ const { result, error } = await harness.executeOnce({ outputLogsOnException: false });
+
+ expect(result).toBeUndefined();
+ expect(error).toEqual(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching('path must start with the project source root'),
+ }),
+ );
+
+ harness.expectFile('dist/test.svg').toNotExist();
+ });
+ });
+
+ describe('longhand syntax', () => {
+ it('copies a single asset', async () => {
+ await harness.writeFile('src/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ });
+
+ it('copies multiple assets as separate entries', async () => {
+ await harness.writeFile('src/test.svg', '');
+ await harness.writeFile('src/another.file', 'asset file');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [
+ { glob: 'test.svg', input: 'src', output: '.' },
+ { glob: 'another.file', input: 'src', output: '.' },
+ ],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').content.toBe('asset file');
+ });
+
+ it('copies multiple assets with a single entry glob pattern', async () => {
+ await harness.writeFile('src/test.svg', '');
+ await harness.writeFile('src/another.file', 'asset file');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: '{test.svg,another.file}', input: 'src', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').content.toBe('asset file');
+ });
+
+ it('copies multiple assets with a wildcard glob pattern', async () => {
+ await harness.writeFile('src/files/test.svg', '');
+ await harness.writeFile('src/files/another.file', 'asset file');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: '*', input: 'src/files', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').content.toBe('asset file');
+ });
+
+ it('copies multiple assets with a recursive wildcard glob pattern', async () => {
+ await harness.writeFiles({
+ 'src/files/test.svg': '',
+ 'src/files/another.file': 'asset file',
+ 'src/files/nested/extra.file': 'extra file',
+ });
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: '**/*', input: 'src/files', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').content.toBe('asset file');
+ harness.expectFile('dist/nested/extra.file').content.toBe('extra file');
+ });
+
+ it('automatically ignores "." prefixed files when using wildcard glob pattern', async () => {
+ await harness.writeFile('src/files/.gitkeep', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: '*', input: 'src/files', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/.gitkeep').toNotExist();
+ });
+
+ it('supports ignoring a specific file when using a glob pattern', async () => {
+ await harness.writeFiles({
+ 'src/files/test.svg': '',
+ 'src/files/another.file': 'asset file',
+ 'src/files/nested/extra.file': 'extra file',
+ });
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [
+ { glob: '**/*', input: 'src/files', output: '.', ignore: ['another.file'] },
+ ],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').toNotExist();
+ harness.expectFile('dist/nested/extra.file').content.toBe('extra file');
+ });
+
+ it('supports ignoring with a glob pattern when using a glob pattern', async () => {
+ await harness.writeFiles({
+ 'src/files/test.svg': '',
+ 'src/files/another.file': 'asset file',
+ 'src/files/nested/extra.file': 'extra file',
+ });
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [
+ { glob: '**/*', input: 'src/files', output: '.', ignore: ['**/*.file'] },
+ ],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ harness.expectFile('dist/another.file').toNotExist();
+ harness.expectFile('dist/nested/extra.file').toNotExist();
+ });
+
+ it('copies an asset with directory and maintains directory in output', async () => {
+ await harness.writeFile('src/subdirectory/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'subdirectory/test.svg', input: 'src', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/subdirectory/test.svg').content.toBe('');
+ });
+
+ it('does not fail if asset does not exist', async () => {
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').toNotExist();
+ });
+
+ it('uses project output path when output option is empty string', async () => {
+ await harness.writeFile('src/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: '' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ });
+
+ it('uses project output path when output option is "."', async () => {
+ await harness.writeFile('src/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: '.' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ });
+
+ it('uses project output path when output option is "/"', async () => {
+ await harness.writeFile('src/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: '/' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/test.svg').content.toBe('');
+ });
+
+ it('creates a project output sub-path when output option path does not exist', async () => {
+ await harness.writeFile('src/test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: 'subdirectory' }],
+ });
+
+ const { result } = await harness.executeOnce();
+
+ expect(result?.success).toBe(true);
+
+ harness.expectFile('dist/subdirectory/test.svg').content.toBe('');
+ });
+
+ it('throws exception if output option is not within project output path', async () => {
+ await harness.writeFile('test.svg', '');
+
+ harness.useTarget('build', {
+ ...BASE_OPTIONS,
+ assets: [{ glob: 'test.svg', input: 'src', output: '..' }],
+ });
+
+ const { result, error } = await harness.executeOnce({ outputLogsOnException: false });
+
+ expect(result).toBeUndefined();
+ expect(error).toEqual(
+ jasmine.objectContaining({
+ message: jasmine.stringMatching(
+ 'An asset cannot be written to a location outside of the output path',
+ ),
+ }),
+ );
+
+ harness.expectFile('dist/test.svg').toNotExist();
+ });
+ });
+ });
+});