0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2025-01-06 22:10:10 -05:00

fix(#2846): handle destructured private env usage during SSR (#2861)

* fix(#2846): handle destructured private env usage during SSR

* test: add destructured env vars test

* fix: support destructured env vars

* fix: only inline referenced vars
This commit is contained in:
Nate Moore 2022-03-24 15:10:08 -05:00 committed by GitHub
parent eb4ac82acf
commit 671727ba83
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 36 additions and 5 deletions

View file

@ -34,11 +34,14 @@ function getPrivateEnv(viteConfig: vite.ResolvedConfig, astroConfig: AstroConfig
return Object.fromEntries(privateKeys.map((key) => [key, JSON.stringify(fullEnv[key])])); return Object.fromEntries(privateKeys.map((key) => [key, JSON.stringify(fullEnv[key])]));
} }
function referencesPrivateKey(source: string, privateEnv: Record<string, any>) { function getReferencedPrivateKeys(source: string, privateEnv: Record<string, any>): Set<string> {
const references = new Set<string>();
for (const key of Object.keys(privateEnv)) { for (const key of Object.keys(privateEnv)) {
if (source.includes(key)) return true; if (source.includes(key)) {
references.add(key);
} }
return false; }
return references;
} }
export default function envVitePlugin({ config: astroConfig }: EnvPluginOptions): vite.PluginOption { export default function envVitePlugin({ config: astroConfig }: EnvPluginOptions): vite.PluginOption {
@ -68,6 +71,13 @@ export default function envVitePlugin({ config: astroConfig }: EnvPluginOptions)
if (privateEnv) { if (privateEnv) {
const entries = Object.entries(privateEnv).map(([key, value]) => [`import.meta.env.${key}`, value]); const entries = Object.entries(privateEnv).map(([key, value]) => [`import.meta.env.${key}`, value]);
replacements = Object.fromEntries(entries); replacements = Object.fromEntries(entries);
// These additional replacements are needed to match Vite
replacements = Object.assign(replacements, {
// This catches destructed `import.meta.env` calls,
// BUT we only want to inject private keys referenced in the file.
// We overwrite this value on a per-file basis.
'import.meta.env': `({})`,
})
pattern = new RegExp( pattern = new RegExp(
// Do not allow preceding '.', but do allow preceding '...' for spread operations // Do not allow preceding '.', but do allow preceding '...' for spread operations
'(?<!(?<!\\.\\.)\\.)\\b(' + '(?<!(?<!\\.\\.)\\.)\\b(' +
@ -84,7 +94,8 @@ export default function envVitePlugin({ config: astroConfig }: EnvPluginOptions)
} }
if (!privateEnv || !pattern) return source; if (!privateEnv || !pattern) return source;
if (!referencesPrivateKey(source, privateEnv)) return source; const references = getReferencedPrivateKeys(source, privateEnv);
if (references.size === 0) return source;
// Find matches for *private* env and do our own replacement. // Find matches for *private* env and do our own replacement.
const s = new MagicString(source); const s = new MagicString(source);
@ -93,7 +104,15 @@ export default function envVitePlugin({ config: astroConfig }: EnvPluginOptions)
while ((match = pattern.exec(source))) { while ((match = pattern.exec(source))) {
const start = match.index; const start = match.index;
const end = start + match[0].length; const end = start + match[0].length;
const replacement = '' + replacements[match[1]]; let replacement = '' + replacements[match[1]];
// If we match exactly `import.meta.env`, define _only_ referenced private variables
if (match[0] === 'import.meta.env') {
replacement = `(Object.assign(import.meta.env,{`
for (const key of references.values()) {
replacement += `${key}:${privateEnv[key]},`
}
replacement += '}))'
}
s.overwrite(start, end, replacement); s.overwrite(start, end, replacement);
} }

View file

@ -23,6 +23,13 @@ describe('Environment Variables', () => {
expect(indexHtml).to.include('BLUE_BAYOU'); expect(indexHtml).to.include('BLUE_BAYOU');
}); });
it('does render destructured public env and private env', async () => {
let indexHtml = await fixture.readFile('/destructured/index.html');
expect(indexHtml).to.include('CLUB_33');
expect(indexHtml).to.include('BLUE_BAYOU');
});
it('includes public env in client-side JS', async () => { it('includes public env in client-side JS', async () => {
let dirs = await fixture.readdir('/'); let dirs = await fixture.readdir('/');
let found = false; let found = false;

View file

@ -0,0 +1,5 @@
---
const { PUBLIC_PLACE, SECRET_PLACE } = import.meta.env;
---
<environment-variable>{PUBLIC_PLACE}</environment-variable>
<environment-variable>{SECRET_PLACE}</environment-variable>