Skip to content

Commit 88dc01d

Browse files
committed
test(plugin-js-packages-e2e): create vulnerabilities e2e test
1 parent b230a3d commit 88dc01d

9 files changed

+393
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
import tseslint from 'typescript-eslint';
2+
import baseConfig from '../../eslint.config.js';
3+
4+
export default tseslint.config(...baseConfig, {
5+
files: ['**/*.ts'],
6+
languageOptions: {
7+
parserOptions: {
8+
projectService: true,
9+
tsconfigRootDir: import.meta.dirname,
10+
},
11+
},
12+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import jsPackagesPlugin from '@code-pushup/js-packages-plugin';
2+
import type { CoreConfig } from '@code-pushup/models';
3+
4+
export default {
5+
persist: { outputDir: './' },
6+
plugins: [await jsPackagesPlugin({ packageManager: 'npm' })],
7+
} satisfies CoreConfig;

e2e/plugin-js-packages-e2e/mocks/fixtures/npm-repo/package-lock.json

+191
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
{
2+
"dependencies": {
3+
"express": "3.0.0"
4+
}
5+
}
+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
{
2+
"name": "plugin-js-packages-e2e",
3+
"$schema": "../../node_modules/nx/schemas/project-schema.json",
4+
"projectType": "application",
5+
"sourceRoot": "e2e/plugin-js-packages-e2e/src",
6+
"implicitDependencies": ["cli", "plugin-js-packages"],
7+
"targets": {
8+
"lint": {
9+
"executor": "@nx/linter:eslint",
10+
"outputs": ["{options.outputFile}"],
11+
"options": {
12+
"lintFilePatterns": ["e2e/plugin-eslint-e2e/**/*.ts"]
13+
}
14+
},
15+
"e2e": {
16+
"executor": "@nx/vite:test",
17+
"options": {
18+
"configFile": "e2e/plugin-eslint-e2e/vite.config.e2e.ts"
19+
}
20+
}
21+
},
22+
"tags": ["scope:plugin", "type:e2e"]
23+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
import { cp } from 'node:fs/promises';
2+
import path from 'node:path';
3+
import { afterAll, beforeAll, expect, it } from 'vitest';
4+
import {
5+
type AuditReport,
6+
type Report,
7+
reportSchema,
8+
} from '@code-pushup/models';
9+
import { nxTargetProject } from '@code-pushup/test-nx-utils';
10+
import { teardownTestFolder } from '@code-pushup/test-setup';
11+
import { E2E_ENVIRONMENTS_DIR, TEST_OUTPUT_DIR } from '@code-pushup/test-utils';
12+
import { executeProcess, readJsonFile } from '@code-pushup/utils';
13+
14+
describe('plugin-js-packages', () => {
15+
const fixturesDir = path.join(
16+
'e2e',
17+
'plugin-js-packages-e2e',
18+
'mocks',
19+
'fixtures',
20+
);
21+
const fixturesNPMDir = path.join(fixturesDir, 'npm-repo');
22+
23+
const envRoot = path.join(
24+
E2E_ENVIRONMENTS_DIR,
25+
nxTargetProject(),
26+
TEST_OUTPUT_DIR,
27+
);
28+
const npmRepoDir = path.join(envRoot, 'npm-repo');
29+
30+
beforeAll(async () => {
31+
await cp(fixturesNPMDir, npmRepoDir, { recursive: true });
32+
});
33+
34+
afterAll(async () => {
35+
await teardownTestFolder(npmRepoDir);
36+
});
37+
38+
it('should run JS packages plugin for NPM and create report.json', async () => {
39+
const { code, stderr } = await executeProcess({
40+
command: 'node',
41+
args: [
42+
'../../node_modules/@code-pushup/cli/src/index.js',
43+
'collect',
44+
'--no-progress',
45+
],
46+
cwd: npmRepoDir,
47+
});
48+
49+
expect(code).toBe(0);
50+
expect(stderr).toBe('');
51+
52+
const report = await readJsonFile<Report>(
53+
path.join(npmRepoDir, 'report.json'),
54+
);
55+
56+
const plugin = report.plugins[0]!;
57+
const npmAuditProd = plugin.audits.find(
58+
({ slug }) => slug === 'npm-audit-prod',
59+
)!;
60+
const npmOutdatedProd = plugin.audits.find(
61+
({ slug }) => slug === 'npm-outdated-prod',
62+
)!;
63+
64+
expect(plugin.packageName).toBe('@code-pushup/js-packages-plugin');
65+
expect(plugin.audits).toHaveLength(4);
66+
67+
expect(npmAuditProd).toEqual<AuditReport>(
68+
expect.objectContaining({
69+
value: expect.any(Number),
70+
}),
71+
);
72+
expect(npmAuditProd.value).toBeGreaterThanOrEqual(7); // there are 7 vulnerabilities (6 high, 1 low) in prod dependency at the time
73+
expect(npmAuditProd.displayValue).toMatch(/\d vulnerabilities/);
74+
expect(npmAuditProd.details?.issues!.length).toBeGreaterThanOrEqual(7);
75+
76+
const expressConnectIssue = npmAuditProd.details!.issues![0]!;
77+
expect(expressConnectIssue?.severity).toBe('error');
78+
expect(expressConnectIssue?.message).toContain('express');
79+
expect(expressConnectIssue?.message).toContain('2.30.2');
80+
expect(expressConnectIssue?.message).toContain(
81+
'methodOverride Middleware Reflected Cross-Site Scripting in connect',
82+
);
83+
84+
expect(npmOutdatedProd.score).toBe(0);
85+
expect(npmOutdatedProd.value).toBe(1); // there is 1 outdated prod dependency at the time
86+
expect(npmOutdatedProd.displayValue).toBe(
87+
'1 major outdated package version',
88+
);
89+
expect(npmOutdatedProd.details?.issues).toHaveLength(1);
90+
91+
const expressOutdatedIssue = npmOutdatedProd.details!.issues![0]!;
92+
expect(expressOutdatedIssue.severity).toBe('error');
93+
expect(expressOutdatedIssue.message).toMatch(
94+
/^Package \[`express`]\(http:\/\/expressjs\.com\/?\) requires a \*\*major\*\* update from \*\*3.0.0\*\* to \*\*\d+\.\d+\.\d+\*\*\.$/,
95+
);
96+
97+
expect(() => reportSchema.parse(report)).not.toThrow();
98+
});
99+
});
+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
{
2+
"extends": "../../tsconfig.base.json",
3+
"compilerOptions": {
4+
"module": "ESNext",
5+
"forceConsistentCasingInFileNames": true,
6+
"strict": true,
7+
"noImplicitOverride": true,
8+
"noPropertyAccessFromIndexSignature": true,
9+
"noImplicitReturns": true,
10+
"noFallthroughCasesInSwitch": true,
11+
"types": ["vitest"]
12+
},
13+
"files": [],
14+
"include": [],
15+
"references": [
16+
{
17+
"path": "./tsconfig.test.json"
18+
}
19+
]
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"extends": "./tsconfig.json",
3+
"compilerOptions": {
4+
"outDir": "../../dist/out-tsc",
5+
"types": ["vitest/globals", "vitest/importMeta", "vite/client", "node"],
6+
"target": "ES2020"
7+
},
8+
"exclude": ["__test-env__/**"],
9+
"include": [
10+
"vite.config.e2e.ts",
11+
"tests/**/*.e2e.test.ts",
12+
"tests/**/*.d.ts",
13+
"mocks/**/*.ts"
14+
]
15+
}

0 commit comments

Comments
 (0)