UNPKG

one

Version:

One is a new React Framework that makes Vite serve both native and web.

312 lines (271 loc) 9.32 kB
import { describe, expect, it } from 'vitest' import { transformTreeShakeClient } from './clientTreeShakePlugin' describe('clientTreeShakePlugin', () => { describe('transformTreeShakeClient', () => { it('should remove loader export and its imports', async () => { const code = ` import { serverOnlyModule } from 'server-only-pkg' import { Text } from 'react-native' import { useLoader } from 'one' export function loader() { return serverOnlyModule() } export default function Page() { const data = useLoader(loader) return <Text>{data}</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() // The server-only import should be removed expect(result!.code).not.toContain('server-only-pkg') // But react-native and one should remain expect(result!.code).toContain('react-native') expect(result!.code).toContain('one') }) it('should remove loader export with dynamic import', async () => { const code = ` import { Text } from 'react-native' import { useLoader } from 'one' export async function loader() { const { serverFn } = await import('server-only-pkg') return serverFn() } export default function Page() { const data = useLoader(loader) return <Text>{data}</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() // Dynamic import should also be removed along with the loader expect(result!.code).not.toContain('server-only-pkg') }) it('should remove loader with const arrow function export', async () => { const code = ` import { serverOnlyModule } from 'server-only-pkg' import { Text } from 'react-native' import { useLoader } from 'one' export const loader = async () => { return serverOnlyModule() } export default function Page() { const data = useLoader(loader) return <Text>{data}</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() // The server-only import should be removed expect(result!.code).not.toContain('server-only-pkg') }) it('should handle multiple imports where only some are used in loader', async () => { const code = ` import { serverFn } from 'server-only-pkg' import { sharedUtil } from 'shared-pkg' import { Text } from 'react-native' import { useLoader } from 'one' export function loader() { return serverFn() } export default function Page() { const data = useLoader(loader) return <Text>{sharedUtil(data)}</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() // server-only-pkg should be removed (only used in loader) expect(result!.code).not.toContain('server-only-pkg') // shared-pkg should remain (used in component) expect(result!.code).toContain('shared-pkg') }) it('should remove generateStaticParams and its imports', async () => { const code = ` import { getRoutes } from 'server-only-pkg' import { Text } from 'react-native' export function generateStaticParams() { return getRoutes() } export default function Page() { return <Text>Hello</Text> } ` const result = await transformTreeShakeClient(code, '/app/[slug].tsx') expect(result).toBeDefined() expect(result!.code).not.toContain('server-only-pkg') }) it('should not transform files without loader or generateStaticParams', async () => { const code = ` import { Text } from 'react-native' export default function Page() { return <Text>Hello</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeUndefined() }) it('should embed routeId in loader stub when root is provided', async () => { const code = ` import { serverOnlyModule } from 'server-only-pkg' import { useLoader } from 'one' export async function loader() { return serverOnlyModule() } export default function Layout() { const data = useLoader(loader) return data } ` const result = await transformTreeShakeClient( code, '/project/app/_layout.tsx', '/project' ) expect(result).toBeDefined() // routeId should be relative to app/ directory to match route contextKey format expect(result!.code).toContain('export function loader() {return "./_layout.tsx"}') expect(result!.code).not.toContain('__vxrn__loader__') }) it('should embed routeId with render mode suffix in loader stub', async () => { const code = ` import { serverOnlyModule } from 'server-only-pkg' import { useLoader } from 'one' export async function loader() { return serverOnlyModule() } export default function Layout() { const data = useLoader(loader) return data } ` const result = await transformTreeShakeClient( code, '/project/app/user+ssr.tsx', '/project' ) expect(result).toBeDefined() expect(result!.code).toContain('export function loader() {return "./user+ssr.tsx"}') }) it('should fall back to __vxrn__loader__ stub when no root provided', async () => { const code = ` import { serverOnlyModule } from 'server-only-pkg' import { useLoader } from 'one' export async function loader() { return serverOnlyModule() } export default function Page() { const data = useLoader(loader) return data } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() expect(result!.code).toContain('__vxrn__loader__') }) it('should not transform route file when loader is imported (no inline declaration)', async () => { // importing a loader from another file is not a supported pattern // the plugin only handles inline loader declarations const code = ` import { loader } from './loaders/my-loader' import { useLoader } from 'one' export { loader } export default function Layout() { const data = useLoader(loader) return data } ` const result = await transformTreeShakeClient( code, '/project/app/_layout.tsx', '/project' ) expect(result).toBeUndefined() }) it('should tree-shake server imports from source file where loader is defined', async () => { // when loader is defined in a separate file, that file gets its own transform pass // server-only imports should be removed, loader replaced with routeId stub const sourceCode = ` import { db } from 'server-only-db' export async function loader() { return db.query('SELECT * FROM users') } export function helperUsedByClient() { return 'hello' } ` const result = await transformTreeShakeClient( sourceCode, '/project/app/loaders/my-loader.ts', '/project' ) expect(result).toBeDefined() // server-only import removed expect(result!.code).not.toContain('server-only-db') // client-safe export preserved expect(result!.code).toContain('helperUsedByClient') // loader replaced with stub (routeId relative to app/ dir) expect(result!.code).toContain( 'export function loader() {return "./loaders/my-loader.ts"}' ) }) it('should not transform re-export from source syntax', async () => { // importing/re-exporting a loader from another file is not a supported pattern const code = ` export { loader } from './loaders/shared-loader' import { useLoader } from 'one' export default function Page() { return 'hello' } ` const result = await transformTreeShakeClient( code, '/project/app/page.tsx', '/project' ) expect(result).toBeUndefined() }) it('should preserve type-only imports during tree shaking', async () => { const code = ` import type { SomeType } from 'types-pkg' import { serverFn } from 'server-only-pkg' import { Text } from 'react-native' import { useLoader } from 'one' export function loader(): SomeType { return serverFn() } export default function Page() { const data = useLoader(loader) return <Text>{data}</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() // Type import should be preserved expect(result!.code).toContain('types-pkg') // Server-only import should be removed expect(result!.code).not.toContain('server-only-pkg') }) it('should preserve type imports even when used only in loader', async () => { const code = ` import type { APIGuildMember } from '@discordjs/core' import { getClient } from 'discord-client' import { Text } from 'react-native' import { useLoader } from 'one' export function loader(): APIGuildMember { return getClient() } export default function Page() { const data = useLoader(loader) return <Text>{JSON.stringify(data)}</Text> } ` const result = await transformTreeShakeClient(code, '/app/index.tsx') expect(result).toBeDefined() // Type import should be preserved (TypeScript erases these at compile time) expect(result!.code).toContain('@discordjs/core') // Runtime import should be removed expect(result!.code).not.toContain('discord-client') }) }) })