| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 |
- import { readFile, rm } from 'node:fs/promises';
- import { join } from 'node:path';
- import tsconfigPaths from 'tsconfig-paths';
- import { describe, expect, it } from 'vitest';
- import { compile } from '../utils/compiler.js';
- import { debugLogger, noopLogger } from '../utils/logger.js';
- import { linguiBabelPlugin } from '../vite-plugin-lingui-babel.js';
- /**
- * Integration tests for the linguiBabelPlugin with actual npm packages.
- *
- * These tests verify that:
- * 1. The plugin discovery mechanism correctly identifies npm packages with dashboard extensions
- * 2. The linguiBabelPlugin transforms Lingui macros in those discovered packages
- *
- * This addresses the bug where third-party npm packages (like @vendure-ee/*) with
- * dashboard extensions using Lingui macros would fail to build because the macros
- * weren't being transformed.
- *
- * @see LINGUI_BABEL_PLUGIN_BUG.md
- */
- describe('linguiBabelPlugin with npm packages containing Lingui macros', () => {
- const fixtureDir = join(__dirname, 'fixtures-lingui-npm-plugin');
- const fakeNodeModules = join(fixtureDir, 'fake_node_modules');
- const tempDir = join(__dirname, './__temp/lingui-npm-plugin');
- it('should discover npm plugin and transform its Lingui macros', { timeout: 60_000 }, async () => {
- // Clean up temp directory
- await rm(tempDir, { recursive: true, force: true });
- // Register tsconfig paths so the test can resolve the fake npm package
- tsconfigPaths.register({
- baseUrl: fakeNodeModules,
- paths: {
- '@test-scope/lingui-plugin': [join(fakeNodeModules, '@test-scope/lingui-plugin')],
- },
- });
- // Step 1: Compile and discover plugins (like the real build process does)
- const result = await compile({
- outputPath: tempDir,
- vendureConfigPath: join(fixtureDir, 'vendure-config.ts'),
- logger: process.env.LOG ? debugLogger : noopLogger,
- pluginPackageScanner: {
- nodeModulesRoot: fakeNodeModules,
- },
- });
- // Verify the plugin was discovered
- expect(result.pluginInfo).toHaveLength(1);
- expect(result.pluginInfo[0].name).toBe('LinguiTestPlugin');
- expect(result.pluginInfo[0].dashboardEntryPath).toBe('./dashboard/index.tsx');
- expect(result.pluginInfo[0].sourcePluginPath).toBeUndefined(); // npm package, not local
- // Step 2: Extract the package path from the discovered plugin
- // (This is what the linguiBabelPlugin does in configResolved)
- const pluginPath = result.pluginInfo[0].pluginPath;
- expect(pluginPath).toContain('@test-scope/lingui-plugin');
- // Extract package name from path (simulating extractPackagePath)
- const packageName = '@test-scope/lingui-plugin';
- // Step 3: Create linguiBabelPlugin with the discovered package
- const plugin = linguiBabelPlugin({
- additionalPackagePaths: [packageName],
- });
- // Step 4: Read the actual dashboard file that contains Lingui macros
- const dashboardFilePath = join(fakeNodeModules, '@test-scope/lingui-plugin/dashboard/index.tsx');
- const dashboardCode = await readFile(dashboardFilePath, 'utf-8');
- // Verify the file contains Lingui macros before transformation
- expect(dashboardCode).toContain("from '@lingui/react/macro'");
- expect(dashboardCode).toContain("from '@lingui/core/macro'");
- expect(dashboardCode).toContain('<Trans>');
- expect(dashboardCode).toContain('t`');
- // Step 5: Transform the file using linguiBabelPlugin
- // @ts-expect-error - transform expects a different context but works for testing
- const transformed = await plugin.transform(dashboardCode, dashboardFilePath);
- // Step 6: Verify the macros were transformed
- expect(transformed).not.toBeNull();
- expect(transformed?.code).toBeDefined();
- // The transformed code should NOT contain macro imports
- expect(transformed?.code).not.toContain("from '@lingui/react/macro'");
- expect(transformed?.code).not.toContain("from '@lingui/core/macro'");
- // The transformed code should contain the runtime imports instead
- expect(transformed?.code).toContain('@lingui/react');
- expect(transformed?.code).toContain('@lingui/core');
- });
- it('should NOT transform Lingui macros in undiscovered npm packages', { timeout: 60_000 }, async () => {
- // Create plugin WITHOUT the package in additionalPackagePaths
- // (simulating a package that wasn't discovered as a Vendure plugin)
- const plugin = linguiBabelPlugin({
- additionalPackagePaths: [], // Empty - no packages discovered
- });
- // Read the dashboard file
- const dashboardFilePath = join(fakeNodeModules, '@test-scope/lingui-plugin/dashboard/index.tsx');
- const dashboardCode = await readFile(dashboardFilePath, 'utf-8');
- // Try to transform - should return null because package isn't in allowlist
- // @ts-expect-error - transform expects a different context but works for testing
- const transformed = await plugin.transform(dashboardCode, dashboardFilePath);
- // Should be null - file was skipped because it's in node_modules and not discovered
- expect(transformed).toBeNull();
- });
- it('should still transform @vendure/dashboard source files without discovery', async () => {
- // Create plugin with no discovered packages
- const plugin = linguiBabelPlugin();
- const code = `
- import { Trans } from '@lingui/react/macro';
- export function MyComponent() {
- return <Trans>Hello</Trans>;
- }
- `;
- // Simulate a file from @vendure/dashboard/src
- const id = join(fakeNodeModules, '@vendure/dashboard/src/components/Test.tsx');
- // @ts-expect-error - transform expects a different context but works for testing
- const transformed = await plugin.transform(code, id);
- // Should transform because @vendure/dashboard/src is always allowed
- expect(transformed).not.toBeNull();
- expect(transformed?.code).not.toContain("from '@lingui/react/macro'");
- });
- it('should still transform local files without discovery', async () => {
- // Create plugin with no discovered packages
- const plugin = linguiBabelPlugin();
- const code = `
- import { Trans } from '@lingui/react/macro';
- export function MyComponent() {
- return <Trans>Hello</Trans>;
- }
- `;
- // Simulate a local file (not in node_modules)
- const id = '/path/to/project/src/plugins/my-plugin/dashboard/Test.tsx';
- // @ts-expect-error - transform expects a different context but works for testing
- const transformed = await plugin.transform(code, id);
- // Should transform because local files are always allowed
- expect(transformed).not.toBeNull();
- expect(transformed?.code).not.toContain("from '@lingui/react/macro'");
- });
- });
|