Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

perf: unresolvedUrlToModule promise cache #12725

Merged
merged 2 commits into from
Apr 6, 2023

Conversation

patak-dev
Copy link
Member

Description

wip

We thought this could have an effect, but in the examples we tried there are no concurrent callings of resolveUrl and ensureEntryFromUrl. @bluwy @sapphi-red maybe you could check in some of your apps.

I still think it is better to keep the promise given that these public functions could be called externally.


What is the purpose of this pull request?

  • Bug fix
  • New Feature
  • Documentation update
  • Other

@stackblitz
Copy link

stackblitz bot commented Apr 3, 2023

Review PR in StackBlitz Codeflow Run & review this pull request in StackBlitz Codeflow.

Copy link
Member

@bluwy bluwy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was able to see a promise cache hit when combined with vite-plugin-warmup sometimes, especially when combined with a slow resolving plugin:

  {
      name: 'slow',
      enforce: 'pre',
      async resolveId(id) {
        await new Promise((resolve) => setTimeout(resolve, 10))
      },
    },

But it's not always reproducible, and kinda an edgecase depending on the configuration. If the iife and promise doesn't incur much overhead, I guess we could continue with it still.

@patak-dev
Copy link
Member Author

We discussed this a bit more with @antfu, he is a bit concerned about merging this without more perf data. My take is that we should still merge this given that it makes the API more robust, and not only more performant. Still, I would like to wait for @sapphi-red's here. Maybe in some of his test cases would make the decision more clear.

@sapphi-red
Copy link
Member

The time didn't change on my machine. (I guess it's within the margin of error.)

before

┌─────────┬──────────────┬──────────────────────────────────────────────────────┬───────────────┬───────────────┐
│ (index) │     name     │                     startup time                     │ Root HMR time │ Leaf HMR time │
├─────────┼──────────────┼──────────────────────────────────────────────────────┼───────────────┼───────────────┤
│    0    │    'Vite'    │ '4763.3ms (including server start up time: 519.3ms)' │   '26.3ms'    │   '13.0ms'    │
│    1    │ 'Vite (swc)' │ '3316.0ms (including server start up time: 252.3ms)' │   '24.7ms'    │   '10.3ms'    │
└─────────┴──────────────┴──────────────────────────────────────────────────────┴───────────────┴───────────────┘

after

┌─────────┬──────────────┬──────────────────────────────────────────────────────┬───────────────┬───────────────┐
│ (index) │     name     │                     startup time                     │ Root HMR time │ Leaf HMR time │
├─────────┼──────────────┼──────────────────────────────────────────────────────┼───────────────┼───────────────┤
│    0    │    'Vite'    │ '4754.0ms (including server start up time: 512.3ms)' │   '26.0ms'    │   '12.7ms'    │
│    1    │ 'Vite (swc)' │ '3319.3ms (including server start up time: 250.7ms)' │   '23.7ms'    │   '10.3ms'    │
└─────────┴──────────────┴──────────────────────────────────────────────────────┴───────────────┴───────────────┘

@sapphi-red
Copy link
Member

I wonder if there would be a difference if we change this part to use Promise.all.

for (let index = 0; index < imports.length; index++) {
const {
s: start,
e: end,
ss: expStart,
se: expEnd,
d: dynamicIndex,
// #2083 User may use escape path,
// so use imports[index].n to get the unescaped string
n: specifier,
a: assertIndex,
} = imports[index]
const rawUrl = source.slice(start, end)
// check import.meta usage
if (rawUrl === 'import.meta') {
const prop = source.slice(end, end + 4)
if (prop === '.hot') {
hasHMR = true
const endHot = end + 4 + (source[end + 4] === '?' ? 1 : 0)
if (source.slice(endHot, endHot + 7) === '.accept') {
// further analyze accepted modules
if (source.slice(endHot, endHot + 14) === '.acceptExports') {
lexAcceptedHmrExports(
source,
source.indexOf('(', endHot + 14) + 1,
acceptedExports,
)
isPartiallySelfAccepting = true
} else if (
lexAcceptedHmrDeps(
source,
source.indexOf('(', endHot + 7) + 1,
acceptedUrls,
)
) {
isSelfAccepting = true
}
}
} else if (prop === '.env') {
hasEnv = true
}
continue
}
const isDynamicImport = dynamicIndex > -1
// strip import assertions as we can process them ourselves
if (!isDynamicImport && assertIndex > -1) {
str().remove(end + 1, expEnd)
}
// static import or valid string in dynamic import
// If resolvable, let's resolve it
if (specifier) {
// skip external / data uri
if (isExternalUrl(specifier) || isDataUrl(specifier)) {
continue
}
// skip ssr external
if (ssr) {
if (config.legacy?.buildSsrCjsExternalHeuristics) {
if (cjsShouldExternalizeForSSR(specifier, server._ssrExternals)) {
continue
}
} else if (shouldExternalizeForSSR(specifier, config)) {
continue
}
if (isBuiltin(specifier)) {
continue
}
}
// skip client
if (specifier === clientPublicPath) {
continue
}
// warn imports to non-asset /public files
if (
specifier[0] === '/' &&
!config.assetsInclude(cleanUrl(specifier)) &&
!specifier.endsWith('.json') &&
checkPublicFile(specifier, config)
) {
throw new Error(
`Cannot import non-asset file ${specifier} which is inside /public.` +
`JS/CSS files inside /public are copied as-is on build and ` +
`can only be referenced via <script src> or <link href> in html.`,
)
}
// normalize
const [url, resolvedId] = await normalizeUrl(specifier, start)
if (
!isDynamicImport &&
specifier &&
!specifier.includes('?') && // ignore custom queries
isCSSRequest(resolvedId) &&
!isModuleCSSRequest(resolvedId)
) {
const sourceExp = source.slice(expStart, start)
if (
sourceExp.includes('from') && // check default and named imports
!sourceExp.includes('__vite_glob_') // glob handles deprecation message itself
) {
const newImport =
sourceExp + specifier + `?inline` + source.slice(end, expEnd)
this.warn(
`\n` +
colors.cyan(importerModule.file) +
`\n` +
colors.reset(generateCodeFrame(source, start)) +
`\n` +
colors.yellow(
`Default and named imports from CSS files are deprecated. ` +
`Use the ?inline query instead. ` +
`For example: ${newImport}`,
),
)
}
}
// record as safe modules
server?.moduleGraph.safeModulesPath.add(fsPathFromUrl(url))
if (url !== specifier) {
let rewriteDone = false
if (
depsOptimizer?.isOptimizedDepFile(resolvedId) &&
!resolvedId.match(optimizedDepChunkRE)
) {
// for optimized cjs deps, support named imports by rewriting named imports to const assignments.
// internal optimized chunks don't need es interop and are excluded
// The browserHash in resolvedId could be stale in which case there will be a full
// page reload. We could return a 404 in that case but it is safe to return the request
const file = cleanUrl(resolvedId) // Remove ?v={hash}
const needsInterop = await optimizedDepNeedsInterop(
depsOptimizer.metadata,
file,
config,
ssr,
)
if (needsInterop === undefined) {
// Non-entry dynamic imports from dependencies will reach here as there isn't
// optimize info for them, but they don't need es interop. If the request isn't
// a dynamic import, then it is an internal Vite error
if (!file.match(optimizedDepDynamicRE)) {
config.logger.error(
colors.red(
`Vite Error, ${url} optimized info should be defined`,
),
)
}
} else if (needsInterop) {
debug?.(`${url} needs interop`)
interopNamedImports(str(), imports[index], url, index)
rewriteDone = true
}
}
// If source code imports builtin modules via named imports, the stub proxy export
// would fail as it's `export default` only. Apply interop for builtin modules to
// correctly throw the error message.
else if (
url.includes(browserExternalId) &&
source.slice(expStart, start).includes('{')
) {
interopNamedImports(str(), imports[index], url, index)
rewriteDone = true
}
if (!rewriteDone) {
const rewrittenUrl = JSON.stringify(url)
const s = isDynamicImport ? start : start - 1
const e = isDynamicImport ? end : end + 1
str().overwrite(s, e, rewrittenUrl, {
contentOnly: true,
})
}
}
// record for HMR import chain analysis
// make sure to unwrap and normalize away base
const hmrUrl = unwrapId(stripBase(url, base))
const isLocalImport = !isExternalUrl(hmrUrl) && !isDataUrl(hmrUrl)
if (isLocalImport) {
importedUrls.add(hmrUrl)
}
if (enablePartialAccept && importedBindings) {
extractImportedBindings(
resolvedId,
source,
imports[index],
importedBindings,
)
}
if (
!isDynamicImport &&
isLocalImport &&
config.server.preTransformRequests
) {
// pre-transform known direct imports
// These requests will also be registered in transformRequest to be awaited
// by the deps optimizer
const url = removeImportQuery(hmrUrl)
server.transformRequest(url, { ssr }).catch((e) => {
if (e?.code === ERR_OUTDATED_OPTIMIZED_DEP) {
// This are expected errors
return
}
// Unexpected error, log the issue but avoid an unhandled exception
config.logger.error(e.message)
})
}
} else if (!importer.startsWith(clientDir)) {
if (!isInNodeModules(importer)) {
// check @vite-ignore which suppresses dynamic import warning
const hasViteIgnore = hasViteIgnoreRE.test(
// complete expression inside parens
source.slice(dynamicIndex + 1, end),
)
if (!hasViteIgnore) {
this.warn(
`\n` +
colors.cyan(importerModule.file) +
`\n` +
colors.reset(generateCodeFrame(source, start)) +
colors.yellow(
`\nThe above dynamic import cannot be analyzed by Vite.\n` +
`See ${colors.blue(
`https://github.com/rollup/plugins/tree/master/packages/dynamic-import-vars#limitations`,
)} ` +
`for supported dynamic import formats. ` +
`If this is intended to be left as-is, you can use the ` +
`/* @vite-ignore */ comment inside the import() call to suppress this warning.\n`,
),
)
}
}
if (!ssr) {
const url = rawUrl.replace(cleanUpRawUrlRE, '').trim()
if (
!urlIsStringRE.test(url) ||
isExplicitImportRequired(url.slice(1, -1))
) {
needQueryInjectHelper = true
str().overwrite(
start,
end,
`__vite__injectQuery(${url}, 'import')`,
{ contentOnly: true },
)
}
}
}
}

@patak-dev
Copy link
Member Author

That's a good idea @sapphi-red. I was thinking about the same while reviewing #12619. But I didn't connect it to this PR. It may have a bigger effect after #12723 though. Worth exploring, I'll check it out if nobody beats me to it.

@patak-dev
Copy link
Member Author

I think we should merge this PR, even if the perf improvement is minimal it makes the API more robust.

@patak-dev patak-dev enabled auto-merge (squash) April 6, 2023 20:08
@patak-dev patak-dev merged commit 80c526e into main Apr 6, 2023
@patak-dev patak-dev deleted the perf/unresolved-url-to-promise-promise-cache branch April 6, 2023 20:17
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants