-
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(nitro-utils): Export Rollup Plugin `wrapServerEntryWithDynamicIm…
…port`
- Loading branch information
Showing
19 changed files
with
625 additions
and
408 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
module.exports = { | ||
extends: ['../../.eslintrc.js'], | ||
env: { | ||
node: true, | ||
}, | ||
overrides: [ | ||
{ | ||
files: ['src/**'], | ||
rules: { | ||
'@sentry-internal/sdk/no-optional-chaining': 'off', | ||
}, | ||
}, | ||
{ | ||
files: ['src/metrics/**'], | ||
rules: { | ||
'@typescript-eslint/explicit-function-return-type': 'off', | ||
'@typescript-eslint/no-non-null-assertion': 'off', | ||
}, | ||
}, | ||
], | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
MIT License | ||
|
||
Copyright (c) 2020-2024 Functional Software, Inc. dba Sentry | ||
|
||
Permission is hereby granted, free of charge, to any person obtaining a copy of | ||
this software and associated documentation files (the "Software"), to deal in | ||
the Software without restriction, including without limitation the rights to | ||
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies | ||
of the Software, and to permit persons to whom the Software is furnished to do | ||
so, subject to the following conditions: | ||
|
||
The above copyright notice and this permission notice shall be included in all | ||
copies or substantial portions of the Software. | ||
|
||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | ||
SOFTWARE. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,23 @@ | ||
<p align="center"> | ||
<a href="https://sentry.io/?utm_source=github&utm_medium=logo" target="_blank"> | ||
<img src="https://sentry-brand.storage.googleapis.com/sentry-wordmark-dark-280x84.png" alt="Sentry" width="280" height="84"> | ||
</a> | ||
</p> | ||
|
||
# Sentry Utilities for Nitro-based SDKs | ||
|
||
[![npm version](https://img.shields.io/npm/v/@sentry-internal/nitro-utils.svg)](https://www.npmjs.com/package/@sentry-internal/nitro-utils) | ||
[![npm dm](https://img.shields.io/npm/dm/@sentry-internal/nitro-utils.svg)](https://www.npmjs.com/package/@sentry-internal/nitro-utils) | ||
[![npm dt](https://img.shields.io/npm/dt/@sentry-internal/nitro-utils.svg)](https://www.npmjs.com/package/@sentry-internal/nitro-utils) | ||
|
||
## Links | ||
|
||
- [Official SDK Docs](https://docs.sentry.io/quickstart/) | ||
- [TypeDoc](http://getsentry.github.io/sentry-node/) | ||
|
||
## General | ||
|
||
Common utilities used by Sentry SDKs that use Nitro on the server-side. | ||
|
||
Note: This package is only meant to be used internally, and as such is not part of our public API contract and does not | ||
follow semver. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
{ | ||
"name": "@sentry-internal/nitro-utils", | ||
"version": "8.36.0", | ||
"description": "Utilities for all Sentry SDKs with Nitro on the server-side", | ||
"repository": "git://github.com/getsentry/sentry-javascript.git", | ||
"homepage": "https://github.com/getsentry/sentry-javascript/tree/master/packages/nitro-utils", | ||
"author": "Sentry", | ||
"license": "MIT", | ||
"private": true, | ||
"engines": { | ||
"node": ">=14.18" | ||
}, | ||
"files": [ | ||
"/build" | ||
], | ||
"main": "build/cjs/index.js", | ||
"module": "build/esm/index.js", | ||
"types": "build/types/index.d.ts", | ||
"exports": { | ||
"./package.json": "./package.json", | ||
".": { | ||
"import": { | ||
"types": "./build/types/index.d.ts", | ||
"default": "./build/esm/index.js" | ||
}, | ||
"require": { | ||
"types": "./build/types/index.d.ts", | ||
"default": "./build/cjs/index.js" | ||
} | ||
} | ||
}, | ||
"typesVersions": { | ||
"<4.9": { | ||
"build/types/index.d.ts": [ | ||
"build/types-ts3.8/index.d.ts" | ||
] | ||
} | ||
}, | ||
"publishConfig": { | ||
"access": "public" | ||
}, | ||
"dependencies": { | ||
"@sentry/core": "8.36.0", | ||
"@sentry/types": "8.36.0", | ||
"@sentry/utils": "8.36.0" | ||
}, | ||
"scripts": { | ||
"build": "run-p build:transpile build:types", | ||
"build:dev": "yarn build", | ||
"build:transpile": "rollup -c rollup.npm.config.mjs", | ||
"build:types": "run-s build:types:core build:types:downlevel", | ||
"build:types:core": "tsc -p tsconfig.types.json", | ||
"build:types:downlevel": "yarn downlevel-dts build/types build/types-ts3.8 --to ts3.8", | ||
"build:watch": "run-p build:transpile:watch build:types:watch", | ||
"build:dev:watch": "run-p build:transpile:watch build:types:watch", | ||
"build:transpile:watch": "rollup -c rollup.npm.config.mjs --watch", | ||
"build:types:watch": "tsc -p tsconfig.types.json --watch", | ||
"build:tarball": "npm pack", | ||
"clean": "rimraf build coverage sentry-internal-nitro-utils-*.tgz", | ||
"fix": "eslint . --format stylish --fix", | ||
"lint": "eslint . --format stylish", | ||
"test": "yarn test:unit", | ||
"test:unit": "vitest run", | ||
"test:watch": "vitest --watch", | ||
"yalc:publish": "yalc publish --push --sig" | ||
}, | ||
"volta": { | ||
"extends": "../../package.json" | ||
}, | ||
"sideEffects": false | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
import { makeBaseNPMConfig, makeNPMConfigVariants } from '@sentry-internal/rollup-utils'; | ||
|
||
export default makeNPMConfigVariants( | ||
makeBaseNPMConfig({ | ||
packageSpecificConfig: { | ||
output: { | ||
// set exports to 'named' or 'auto' so that rollup doesn't warn | ||
exports: 'named', | ||
// set preserveModules to true because we don't want to bundle everything into one file. | ||
preserveModules: | ||
process.env.SENTRY_BUILD_PRESERVE_MODULES === undefined | ||
? true | ||
: Boolean(process.env.SENTRY_BUILD_PRESERVE_MODULES), | ||
}, | ||
}, | ||
}), | ||
); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export { wrapServerEntryWithDynamicImport } from './rollupPlugins/wrapServerEntryWithDynamicImport'; |
219 changes: 219 additions & 0 deletions
219
packages/nitro-utils/src/rollupPlugins/wrapServerEntryWithDynamicImport.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,219 @@ | ||
import { consoleSandbox, flatten } from '@sentry/utils'; | ||
import type { InputPluginOption } from 'rollup'; | ||
|
||
export const SENTRY_WRAPPED_ENTRY = '?sentry-query-wrapped-entry'; | ||
export const SENTRY_WRAPPED_FUNCTIONS = '?sentry-query-wrapped-functions='; | ||
export const SENTRY_REEXPORTED_FUNCTIONS = '?sentry-query-reexported-functions='; | ||
export const QUERY_END_INDICATOR = 'SENTRY-QUERY-END'; | ||
|
||
/** | ||
* A Rollup plugin which wraps the server entry with a dynamic `import()`. This makes it possible to initialize Sentry first | ||
* by using a regular `import` and load the server after that. | ||
* This also works with serverless `handler` functions, as it re-exports the `handler`. | ||
* | ||
* @param config Configuration options for the Rollup Plugin | ||
* @param config.serverConfigFileName Name of the Sentry server config (without file extension). E.g. 'sentry.server.config' | ||
* @param config.resolvedServerConfigPath Resolved path of the Sentry server config (based on `src` directory) | ||
* @param config.entryPointWrappedFunctions Exported bindings of the server entry file, which are wrapped as async function. E.g. ['default', 'handler', 'server'] | ||
* @param config.additionalImports Adds additional imports to the entry file. Can be e.g. 'import-in-the-middle/hook.mjs' | ||
* @param config.debug Whether debug logs are enabled in the build time environment | ||
*/ | ||
export function wrapServerEntryWithDynamicImport(config: { | ||
serverConfigFileName: string; | ||
resolvedServerConfigPath: string; | ||
entrypointWrappedFunctions: string[]; | ||
additionalImports?: string[]; | ||
debug?: boolean; | ||
}): InputPluginOption { | ||
const { serverConfigFileName, resolvedServerConfigPath, entrypointWrappedFunctions, additionalImports, debug } = | ||
config; | ||
|
||
return { | ||
name: 'sentry-wrap-server-entry-with-dynamic-import', | ||
async resolveId(source, importer, options) { | ||
if (source.includes(`/${serverConfigFileName}`)) { | ||
return { id: source, moduleSideEffects: true }; | ||
} | ||
|
||
if (additionalImports && additionalImports.includes(source)) { | ||
// When importing additional imports like "import-in-the-middle/hook.mjs" in the returned code of the `load()` function below: | ||
// By setting `moduleSideEffects` to `true`, the import is added to the bundle, although nothing is imported from it | ||
// By importing "import-in-the-middle/hook.mjs", we can make sure this file is included, as not all node builders are including files imported with `module.register()`. | ||
// Prevents the error "Failed to register ESM hook Error: Cannot find module 'import-in-the-middle/hook.mjs'" | ||
return { id: source, moduleSideEffects: true, external: true }; | ||
} | ||
|
||
if (options.isEntry && source.includes('.mjs') && !source.includes(`.mjs${SENTRY_WRAPPED_ENTRY}`)) { | ||
const resolution = await this.resolve(source, importer, options); | ||
|
||
// If it cannot be resolved or is external, just return it so that Rollup can display an error | ||
if (!resolution || (resolution && resolution.external)) return resolution; | ||
|
||
const moduleInfo = await this.load(resolution); | ||
|
||
moduleInfo.moduleSideEffects = true; | ||
|
||
// The enclosing `if` already checks for the suffix in `source`, but a check in `resolution.id` is needed as well to prevent multiple attachment of the suffix | ||
return resolution.id.includes(`.mjs${SENTRY_WRAPPED_ENTRY}`) | ||
? resolution.id | ||
: resolution.id | ||
// Concatenates the query params to mark the file (also attaches names of re-exports - this is needed for serverless functions to re-export the handler) | ||
.concat(SENTRY_WRAPPED_ENTRY) | ||
.concat( | ||
constructWrappedFunctionExportQuery(moduleInfo.exportedBindings, entrypointWrappedFunctions, debug), | ||
) | ||
.concat(QUERY_END_INDICATOR); | ||
} | ||
return null; | ||
}, | ||
load(id: string) { | ||
if (id.includes(`.mjs${SENTRY_WRAPPED_ENTRY}`)) { | ||
const entryId = removeSentryQueryFromPath(id); | ||
|
||
// Mostly useful for serverless `handler` functions | ||
const reExportedFunctions = | ||
id.includes(SENTRY_WRAPPED_FUNCTIONS) || id.includes(SENTRY_REEXPORTED_FUNCTIONS) | ||
? constructFunctionReExport(id, entryId) | ||
: ''; | ||
|
||
return ( | ||
// Regular `import` of the Sentry config | ||
`import ${JSON.stringify(resolvedServerConfigPath)};\n` + | ||
// Dynamic `import()` for the previous, actual entry point. | ||
// `import()` can be used for any code that should be run after the hooks are registered (https://nodejs.org/api/module.html#enabling) | ||
`import(${JSON.stringify(entryId)});\n` + | ||
// By importing additional imports like "import-in-the-middle/hook.mjs", we can make sure this file wil be included, as not all node builders are including files imported with `module.register()`. | ||
`${additionalImports ? additionalImports.map(importPath => `import "${importPath}";\n`) : ''}` + | ||
`${reExportedFunctions}\n` | ||
); | ||
} | ||
|
||
return null; | ||
}, | ||
}; | ||
} | ||
|
||
/** | ||
* Strips the Sentry query part from a path. | ||
* Example: example/path?sentry-query-wrapped-entry?sentry-query-functions-reexport=foo,SENTRY-QUERY-END -> /example/path | ||
* | ||
* **Only exported for testing** | ||
*/ | ||
export function removeSentryQueryFromPath(url: string): string { | ||
// eslint-disable-next-line @sentry-internal/sdk/no-regexp-constructor | ||
const regex = new RegExp(`\\${SENTRY_WRAPPED_ENTRY}.*?\\${QUERY_END_INDICATOR}`); | ||
return url.replace(regex, ''); | ||
} | ||
|
||
/** | ||
* Extracts and sanitizes function re-export and function wrap query parameters from a query string. | ||
* If it is a default export, it is not considered for re-exporting. | ||
* | ||
* **Only exported for testing** | ||
*/ | ||
export function extractFunctionReexportQueryParameters(query: string): { wrap: string[]; reexport: string[] } { | ||
// Regex matches the comma-separated params between the functions query | ||
// eslint-disable-next-line @sentry-internal/sdk/no-regexp-constructor | ||
const wrapRegex = new RegExp( | ||
`\\${SENTRY_WRAPPED_FUNCTIONS}(.*?)(\\${QUERY_END_INDICATOR}|\\${SENTRY_REEXPORTED_FUNCTIONS})`, | ||
); | ||
// eslint-disable-next-line @sentry-internal/sdk/no-regexp-constructor | ||
const reexportRegex = new RegExp(`\\${SENTRY_REEXPORTED_FUNCTIONS}(.*?)(\\${QUERY_END_INDICATOR})`); | ||
|
||
const wrapMatch = query.match(wrapRegex); | ||
const reexportMatch = query.match(reexportRegex); | ||
|
||
const wrap = | ||
wrapMatch && wrapMatch[1] | ||
? wrapMatch[1] | ||
.split(',') | ||
.filter(param => param !== '') | ||
// Sanitize, as code could be injected with another rollup plugin | ||
.map((str: string) => str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')) | ||
: []; | ||
|
||
const reexport = | ||
reexportMatch && reexportMatch[1] | ||
? reexportMatch[1] | ||
.split(',') | ||
.filter(param => param !== '' && param !== 'default') | ||
// Sanitize, as code could be injected with another rollup plugin | ||
.map((str: string) => str.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')) | ||
: []; | ||
|
||
return { wrap, reexport }; | ||
} | ||
|
||
/** | ||
* Constructs a comma-separated string with all functions that need to be re-exported later from the server entry. | ||
* It uses Rollup's `exportedBindings` to determine the functions to re-export. Functions which should be wrapped | ||
* (e.g. serverless handlers) are wrapped by Sentry. | ||
* | ||
* **Only exported for testing** | ||
*/ | ||
export function constructWrappedFunctionExportQuery( | ||
exportedBindings: Record<string, string[]> | null, | ||
entrypointWrappedFunctions: string[], | ||
debug?: boolean, | ||
): string { | ||
// `exportedBindings` can look like this: `{ '.': [ 'handler' ] }` or `{ '.': [], './firebase-gen-1.mjs': [ 'server' ] }` | ||
// The key `.` refers to exports within the current file, while other keys show from where exports were imported first. | ||
const functionsToExport = flatten(Object.values(exportedBindings || {})).reduce( | ||
(functions, currFunctionName) => { | ||
if (entrypointWrappedFunctions.includes(currFunctionName)) { | ||
functions.wrap.push(currFunctionName); | ||
} else { | ||
functions.reexport.push(currFunctionName); | ||
} | ||
return functions; | ||
}, | ||
{ wrap: [], reexport: [] } as { wrap: string[]; reexport: string[] }, | ||
); | ||
|
||
if (debug && functionsToExport.wrap.length === 0) { | ||
consoleSandbox(() => | ||
// eslint-disable-next-line no-console | ||
console.warn( | ||
"[Sentry] No functions found to wrap. In case the server needs to export async functions other than `handler` or `server`, consider adding the name(s) to Sentry's build options `sentry.entrypointWrappedFunctions` in `nuxt.config.ts`.", | ||
), | ||
); | ||
} | ||
|
||
const wrapQuery = functionsToExport.wrap.length | ||
? `${SENTRY_WRAPPED_FUNCTIONS}${functionsToExport.wrap.join(',')}` | ||
: ''; | ||
const reexportQuery = functionsToExport.reexport.length | ||
? `${SENTRY_REEXPORTED_FUNCTIONS}${functionsToExport.reexport.join(',')}` | ||
: ''; | ||
|
||
return [wrapQuery, reexportQuery].join(''); | ||
} | ||
|
||
/** | ||
* Constructs a code snippet with function reexports (can be used in Rollup plugins as a return value for `load()`) | ||
* | ||
* **Only exported for testing** | ||
*/ | ||
export function constructFunctionReExport(pathWithQuery: string, entryId: string): string { | ||
const { wrap: wrapFunctions, reexport: reexportFunctions } = extractFunctionReexportQueryParameters(pathWithQuery); | ||
|
||
return wrapFunctions | ||
.reduce( | ||
(functionsCode, currFunctionName) => | ||
functionsCode.concat( | ||
`async function ${currFunctionName}_sentryWrapped(...args) {\n` + | ||
` const res = await import(${JSON.stringify(entryId)});\n` + | ||
` return res.${currFunctionName}.call(this, ...args);\n` + | ||
'}\n' + | ||
`export { ${currFunctionName}_sentryWrapped as ${currFunctionName} };\n`, | ||
), | ||
'', | ||
) | ||
.concat( | ||
reexportFunctions.reduce( | ||
(functionsCode, currFunctionName) => | ||
functionsCode.concat(`export { ${currFunctionName} } from ${JSON.stringify(entryId)};`), | ||
'', | ||
), | ||
); | ||
} |
Oops, something went wrong.