2021-12-20 16:27:36 +00:00
|
|
|
import { readFileSync } from 'fs'
|
2021-11-21 16:14:46 +00:00
|
|
|
import type { AutoImport } from '@nuxt/schema'
|
2022-01-13 17:54:33 +00:00
|
|
|
import { expect, describe, it } from 'vitest'
|
2021-12-20 16:27:36 +00:00
|
|
|
import { join } from 'pathe'
|
|
|
|
import { createCommonJS, findExports } from 'mlly'
|
2021-11-05 14:39:14 +00:00
|
|
|
import * as VueFunctions from 'vue'
|
2021-10-20 09:47:18 +00:00
|
|
|
import { AutoImportContext, updateAutoImportContext } from '../src/auto-imports/context'
|
2021-10-11 08:07:27 +00:00
|
|
|
import { TransformPlugin } from '../src/auto-imports/transform'
|
2021-11-05 14:39:14 +00:00
|
|
|
import { Nuxt3AutoImports } from '../src/auto-imports/imports'
|
2021-08-10 00:27:23 +00:00
|
|
|
|
2021-10-18 13:39:53 +00:00
|
|
|
describe('auto-imports:transform', () => {
|
|
|
|
const autoImports: AutoImport[] = [
|
|
|
|
{ name: 'ref', as: 'ref', from: 'vue' },
|
2021-12-21 14:28:45 +00:00
|
|
|
{ name: 'computed', as: 'computed', from: 'bar' },
|
|
|
|
{ name: 'foo', as: 'foo', from: 'excluded' }
|
2021-10-18 13:39:53 +00:00
|
|
|
]
|
|
|
|
|
2021-12-21 14:28:45 +00:00
|
|
|
const ctx = {
|
|
|
|
autoImports,
|
|
|
|
map: new Map(),
|
|
|
|
transform: {
|
|
|
|
exclude: [/excluded/]
|
|
|
|
}
|
|
|
|
} as AutoImportContext
|
2021-10-20 09:47:18 +00:00
|
|
|
updateAutoImportContext(ctx)
|
|
|
|
|
|
|
|
const transformPlugin = TransformPlugin.raw(ctx, { framework: 'rollup' })
|
2021-10-18 13:39:53 +00:00
|
|
|
const transform = (code: string) => transformPlugin.transform.call({ error: null, warn: null }, code, '')
|
2021-08-10 00:27:23 +00:00
|
|
|
|
2021-08-30 11:55:57 +00:00
|
|
|
it('should correct inject', async () => {
|
2021-10-12 12:24:43 +00:00
|
|
|
expect(await transform('const a = ref(0)')).to.equal('import { ref } from \'vue\';const a = ref(0)')
|
2021-10-18 13:39:53 +00:00
|
|
|
expect(await transform('import { computed as ref } from "foo"; const a = ref(0)')).to.include('import { computed } from \'bar\';')
|
2021-08-10 00:27:23 +00:00
|
|
|
})
|
|
|
|
|
2021-10-12 12:24:43 +00:00
|
|
|
it('should ignore existing imported', async () => {
|
|
|
|
expect(await transform('import { ref } from "foo"; const a = ref(0)')).to.equal(null)
|
|
|
|
expect(await transform('import ref from "foo"; const a = ref(0)')).to.equal(null)
|
|
|
|
expect(await transform('import { z as ref } from "foo"; const a = ref(0)')).to.equal(null)
|
|
|
|
expect(await transform('let ref = () => {}; const a = ref(0)')).to.equal(null)
|
|
|
|
expect(await transform('let { ref } = Vue; const a = ref(0)')).to.equal(null)
|
|
|
|
expect(await transform('let [\ncomputed,\nref\n] = Vue; const a = ref(0); const b = ref(0)')).to.equal(null)
|
2021-08-30 11:55:57 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
it('should ignore comments', async () => {
|
|
|
|
const result = await transform('// import { computed } from "foo"\n;const a = computed(0)')
|
|
|
|
expect(result).to.equal('import { computed } from \'bar\';// import { computed } from "foo"\n;const a = computed(0)')
|
2021-08-10 00:27:23 +00:00
|
|
|
})
|
2021-12-21 14:28:45 +00:00
|
|
|
|
|
|
|
it('should exclude files from transform', async () => {
|
|
|
|
expect(await transform('const a = foo()')).to.not.include('import { foo } from "excluded"')
|
|
|
|
})
|
2021-08-10 00:27:23 +00:00
|
|
|
})
|
2021-11-05 14:39:14 +00:00
|
|
|
|
2021-12-20 16:27:36 +00:00
|
|
|
const excludedNuxtHelpers = ['useHydration']
|
|
|
|
|
2022-01-21 09:51:19 +00:00
|
|
|
const typeExports = [
|
|
|
|
'AsyncDataOptions',
|
|
|
|
'AsyncData',
|
|
|
|
'FetchResult',
|
|
|
|
'UseFetchOptions',
|
|
|
|
'CookieOptions',
|
|
|
|
'CookieRef'
|
|
|
|
]
|
|
|
|
|
2021-12-20 16:27:36 +00:00
|
|
|
describe('auto-imports:nuxt3', () => {
|
|
|
|
try {
|
|
|
|
const { __dirname } = createCommonJS(import.meta.url)
|
|
|
|
const entrypointContents = readFileSync(join(__dirname, '../src/app/composables/index.ts'), 'utf8')
|
|
|
|
|
|
|
|
const names = findExports(entrypointContents).flatMap(i => i.names || i.name)
|
|
|
|
for (const name of names) {
|
2022-01-21 09:51:19 +00:00
|
|
|
if (excludedNuxtHelpers.includes(name) || typeExports.includes(name)) {
|
2021-12-20 16:27:36 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
it(`should register ${name} globally`, () => {
|
|
|
|
expect(Nuxt3AutoImports.find(a => a.from === '#app').names).to.include(name)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
console.log(e)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2021-11-05 14:39:14 +00:00
|
|
|
const excludedVueHelpers = [
|
2022-01-13 17:54:33 +00:00
|
|
|
'__esModule',
|
|
|
|
'devtools',
|
2021-11-05 14:39:14 +00:00
|
|
|
'EffectScope',
|
|
|
|
'ReactiveEffect',
|
|
|
|
'stop',
|
|
|
|
'camelize',
|
|
|
|
'capitalize',
|
|
|
|
'normalizeClass',
|
|
|
|
'normalizeProps',
|
|
|
|
'normalizeStyle',
|
|
|
|
'toDisplayString',
|
|
|
|
'toHandlerKey',
|
|
|
|
'BaseTransition',
|
|
|
|
'Comment',
|
|
|
|
'Fragment',
|
|
|
|
'KeepAlive',
|
|
|
|
'Static',
|
|
|
|
'Suspense',
|
|
|
|
'Teleport',
|
|
|
|
'Text',
|
|
|
|
'callWithAsyncErrorHandling',
|
|
|
|
'callWithErrorHandling',
|
|
|
|
'cloneVNode',
|
|
|
|
'compatUtils',
|
|
|
|
'createBlock',
|
|
|
|
'createCommentVNode',
|
|
|
|
'createElementBlock',
|
|
|
|
'createElementVNode',
|
|
|
|
'createHydrationRenderer',
|
|
|
|
'createPropsRestProxy',
|
|
|
|
'createRenderer',
|
|
|
|
'createSlots',
|
|
|
|
'createStaticVNode',
|
|
|
|
'createTextVNode',
|
|
|
|
'createVNode',
|
|
|
|
'getTransitionRawChildren',
|
|
|
|
'guardReactiveProps',
|
|
|
|
'handleError',
|
|
|
|
'initCustomFormatter',
|
|
|
|
'isMemoSame',
|
|
|
|
'isRuntimeOnly',
|
|
|
|
'isVNode',
|
|
|
|
'mergeDefaults',
|
|
|
|
'mergeProps',
|
|
|
|
'openBlock',
|
|
|
|
'popScopeId',
|
|
|
|
'pushScopeId',
|
|
|
|
'queuePostFlushCb',
|
|
|
|
'registerRuntimeCompiler',
|
|
|
|
'renderList',
|
|
|
|
'renderSlot',
|
|
|
|
'resolveComponent',
|
|
|
|
'resolveDirective',
|
|
|
|
'resolveDynamicComponent',
|
|
|
|
'resolveFilter',
|
|
|
|
'resolveTransitionHooks',
|
|
|
|
'setBlockTracking',
|
|
|
|
'setDevtoolsHook',
|
|
|
|
'setTransitionHooks',
|
|
|
|
'ssrContextKey',
|
|
|
|
'ssrUtils',
|
|
|
|
'toHandlers',
|
|
|
|
'transformVNodeArgs',
|
|
|
|
'useSSRContext',
|
|
|
|
'version',
|
|
|
|
'warn',
|
|
|
|
'watchPostEffect',
|
|
|
|
'watchSyncEffect',
|
|
|
|
'withAsyncContext',
|
|
|
|
'Transition',
|
|
|
|
'TransitionGroup',
|
|
|
|
'VueElement',
|
|
|
|
'createApp',
|
|
|
|
'createSSRApp',
|
|
|
|
'defineCustomElement',
|
|
|
|
'defineSSRCustomElement',
|
|
|
|
'hydrate',
|
|
|
|
'initDirectivesForSSR',
|
|
|
|
'render',
|
|
|
|
'useCssVars',
|
|
|
|
'vModelCheckbox',
|
|
|
|
'vModelDynamic',
|
|
|
|
'vModelRadio',
|
|
|
|
'vModelSelect',
|
|
|
|
'vModelText',
|
|
|
|
'vShow',
|
|
|
|
'compile'
|
|
|
|
]
|
|
|
|
|
|
|
|
describe('auto-imports:vue', () => {
|
|
|
|
for (const name of Object.keys(VueFunctions)) {
|
|
|
|
if (excludedVueHelpers.includes(name)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
it(`should register ${name} globally`, () => {
|
2022-01-13 17:54:33 +00:00
|
|
|
expect(Nuxt3AutoImports.find(a => a.from === 'vue').names).toContain(name)
|
2021-11-05 14:39:14 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|