Remove second snowpack instance (#368)
* Remove second snowpack instance * Document import.meta.env.SSR * Remove unnecessary Promise.all
This commit is contained in:
parent
ce9336115e
commit
1bab906539
7 changed files with 45 additions and 67 deletions
5
.changeset/rare-nails-sneeze.md
Normal file
5
.changeset/rare-nails-sneeze.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': minor
|
||||||
|
---
|
||||||
|
|
||||||
|
Removes a second instance of snowpack which degraded peformance
|
18
docs/api.md
18
docs/api.md
|
@ -146,6 +146,24 @@ Astro will generate an RSS 2.0 feed at `/feed/[collection].xml` (for example, `/
|
||||||
<link rel="alternate" type="application/rss+xml" title="My RSS Feed" href="/feed/podcast.xml" />
|
<link rel="alternate" type="application/rss+xml" title="My RSS Feed" href="/feed/podcast.xml" />
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### `import.meta`
|
||||||
|
|
||||||
|
All ESM modules include a `import.meta` property. Astro adds `import.meta.env` through [Snowpack](https://www.snowpack.dev/).
|
||||||
|
|
||||||
|
__import.meta.env.SSR__ can be used to know when rendering on the server. Some times you might want different logic, for example a component that should only be rendered in the client:
|
||||||
|
|
||||||
|
```jsx
|
||||||
|
import { h } from 'preact';
|
||||||
|
|
||||||
|
export default function() {
|
||||||
|
return (
|
||||||
|
import.meta.env.SSR ?
|
||||||
|
<div class="spinner"></div> :
|
||||||
|
<FancyComponent />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
[canonical]: https://en.wikipedia.org/wiki/Canonical_link_element
|
[canonical]: https://en.wikipedia.org/wiki/Canonical_link_element
|
||||||
[config]: ../README.md#%EF%B8%8F-configuration
|
[config]: ../README.md#%EF%B8%8F-configuration
|
||||||
[docs-collections]: ./collections.md
|
[docs-collections]: ./collections.md
|
||||||
|
|
|
@ -117,7 +117,7 @@ function PluginSearchPageLive() {
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function PluginSearchPage(props) {
|
export default function PluginSearchPage(props) {
|
||||||
return import.meta.env.astro ? (
|
return import.meta.env.SSR ? (
|
||||||
<div>Loading...</div>
|
<div>Loading...</div>
|
||||||
) : (
|
) : (
|
||||||
<PluginSearchPageLive {...props} />
|
<PluginSearchPageLive {...props} />
|
||||||
|
|
|
@ -58,7 +58,7 @@ export async function build(astroConfig: AstroConfig, logging: LogOptions = defa
|
||||||
const mode: RuntimeMode = 'production';
|
const mode: RuntimeMode = 'production';
|
||||||
const runtime = await createRuntime(astroConfig, { mode, logging: runtimeLogging });
|
const runtime = await createRuntime(astroConfig, { mode, logging: runtimeLogging });
|
||||||
const { runtimeConfig } = runtime;
|
const { runtimeConfig } = runtime;
|
||||||
const { backendSnowpack: snowpack } = runtimeConfig;
|
const { snowpack } = runtimeConfig;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// 0. erase build directory
|
// 0. erase build directory
|
||||||
|
|
|
@ -20,12 +20,9 @@ interface RuntimeConfig {
|
||||||
astroConfig: AstroConfig;
|
astroConfig: AstroConfig;
|
||||||
logging: LogOptions;
|
logging: LogOptions;
|
||||||
mode: RuntimeMode;
|
mode: RuntimeMode;
|
||||||
backendSnowpack: SnowpackDevServer;
|
snowpack: SnowpackDevServer;
|
||||||
backendSnowpackRuntime: SnowpackServerRuntime;
|
snowpackRuntime: SnowpackServerRuntime;
|
||||||
backendSnowpackConfig: SnowpackConfig;
|
snowpackConfig: SnowpackConfig;
|
||||||
frontendSnowpack: SnowpackDevServer;
|
|
||||||
frontendSnowpackRuntime: SnowpackServerRuntime;
|
|
||||||
frontendSnowpackConfig: SnowpackConfig;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// info needed for collection generation
|
// info needed for collection generation
|
||||||
|
@ -50,7 +47,7 @@ configureSnowpackLogger(snowpackLogger);
|
||||||
|
|
||||||
/** Pass a URL to Astro to resolve and build */
|
/** Pass a URL to Astro to resolve and build */
|
||||||
async function load(config: RuntimeConfig, rawPathname: string | undefined): Promise<LoadResult> {
|
async function load(config: RuntimeConfig, rawPathname: string | undefined): Promise<LoadResult> {
|
||||||
const { logging, backendSnowpackRuntime, frontendSnowpack } = config;
|
const { logging, snowpackRuntime, snowpack } = config;
|
||||||
const { pages: pagesRoot, buildOptions, devOptions } = config.astroConfig;
|
const { pages: pagesRoot, buildOptions, devOptions } = config.astroConfig;
|
||||||
|
|
||||||
let origin = buildOptions.site ? new URL(buildOptions.site).origin : `http://localhost:${devOptions.port}`;
|
let origin = buildOptions.site ? new URL(buildOptions.site).origin : `http://localhost:${devOptions.port}`;
|
||||||
|
@ -62,7 +59,7 @@ async function load(config: RuntimeConfig, rawPathname: string | undefined): Pro
|
||||||
const searchResult = searchForPage(fullurl, pagesRoot);
|
const searchResult = searchForPage(fullurl, pagesRoot);
|
||||||
if (searchResult.statusCode === 404) {
|
if (searchResult.statusCode === 404) {
|
||||||
try {
|
try {
|
||||||
const result = await frontendSnowpack.loadUrl(reqPath);
|
const result = await snowpack.loadUrl(reqPath);
|
||||||
if (!result) throw new Error(`Unable to load ${reqPath}`);
|
if (!result) throw new Error(`Unable to load ${reqPath}`);
|
||||||
// success
|
// success
|
||||||
return {
|
return {
|
||||||
|
@ -88,7 +85,7 @@ async function load(config: RuntimeConfig, rawPathname: string | undefined): Pro
|
||||||
let rss: { data: any[] & CollectionRSS } = {} as any;
|
let rss: { data: any[] & CollectionRSS } = {} as any;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const mod = await backendSnowpackRuntime.importModule(snowpackURL);
|
const mod = await snowpackRuntime.importModule(snowpackURL);
|
||||||
debug(logging, 'resolve', `${reqPath} -> ${snowpackURL}`);
|
debug(logging, 'resolve', `${reqPath} -> ${snowpackURL}`);
|
||||||
|
|
||||||
// handle collection
|
// handle collection
|
||||||
|
@ -297,10 +294,8 @@ interface RuntimeOptions {
|
||||||
}
|
}
|
||||||
|
|
||||||
interface CreateSnowpackOptions {
|
interface CreateSnowpackOptions {
|
||||||
env: Record<string, any>;
|
|
||||||
mode: RuntimeMode;
|
mode: RuntimeMode;
|
||||||
resolvePackageUrl?: (pkgName: string) => Promise<string>;
|
resolvePackageUrl?: (pkgName: string) => Promise<string>;
|
||||||
target: 'frontend' | 'backend';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const DEFAULT_HMR_PORT = 12321;
|
const DEFAULT_HMR_PORT = 12321;
|
||||||
|
@ -309,11 +304,11 @@ const DEFAULT_RENDERERS = ['@astrojs/renderer-vue', '@astrojs/renderer-svelte',
|
||||||
/** Create a new Snowpack instance to power Astro */
|
/** Create a new Snowpack instance to power Astro */
|
||||||
async function createSnowpack(astroConfig: AstroConfig, options: CreateSnowpackOptions) {
|
async function createSnowpack(astroConfig: AstroConfig, options: CreateSnowpackOptions) {
|
||||||
const { projectRoot, pages: pagesRoot, renderers = DEFAULT_RENDERERS } = astroConfig;
|
const { projectRoot, pages: pagesRoot, renderers = DEFAULT_RENDERERS } = astroConfig;
|
||||||
const { env, mode, resolvePackageUrl, target } = options;
|
const { mode, resolvePackageUrl } = options;
|
||||||
|
|
||||||
const frontendPath = new URL('./frontend/', import.meta.url);
|
const frontendPath = new URL('./frontend/', import.meta.url);
|
||||||
const resolveDependency = (dep: string) => resolve.sync(dep, { basedir: fileURLToPath(projectRoot) });
|
const resolveDependency = (dep: string) => resolve.sync(dep, { basedir: fileURLToPath(projectRoot) });
|
||||||
const isHmrEnabled = mode === 'development' && target === 'backend';
|
const isHmrEnabled = mode === 'development';
|
||||||
|
|
||||||
let snowpack: SnowpackDevServer;
|
let snowpack: SnowpackDevServer;
|
||||||
let astroPluginOptions: {
|
let astroPluginOptions: {
|
||||||
|
@ -419,9 +414,6 @@ async function createSnowpack(astroConfig: AstroConfig, options: CreateSnowpackO
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const envConfig = snowpackConfig.env || (snowpackConfig.env = {});
|
|
||||||
Object.assign(envConfig, env);
|
|
||||||
|
|
||||||
snowpack = await startSnowpackServer(
|
snowpack = await startSnowpackServer(
|
||||||
{
|
{
|
||||||
config: snowpackConfig,
|
config: snowpackConfig,
|
||||||
|
@ -438,53 +430,34 @@ async function createSnowpack(astroConfig: AstroConfig, options: CreateSnowpackO
|
||||||
|
|
||||||
/** Core Astro runtime */
|
/** Core Astro runtime */
|
||||||
export async function createRuntime(astroConfig: AstroConfig, { mode, logging }: RuntimeOptions): Promise<AstroRuntime> {
|
export async function createRuntime(astroConfig: AstroConfig, { mode, logging }: RuntimeOptions): Promise<AstroRuntime> {
|
||||||
|
let snowpack: SnowpackDevServer;
|
||||||
const timer: Record<string, number> = {};
|
const timer: Record<string, number> = {};
|
||||||
const resolvePackageUrl = async (pkgName: string) => frontendSnowpack.getUrlForPackage(pkgName);
|
const resolvePackageUrl = async (pkgName: string) => snowpack.getUrlForPackage(pkgName);
|
||||||
|
|
||||||
timer.backend = performance.now();
|
timer.backend = performance.now();
|
||||||
const {
|
const {
|
||||||
snowpack: backendSnowpack,
|
snowpack: snowpackInstance,
|
||||||
snowpackRuntime: backendSnowpackRuntime,
|
snowpackRuntime,
|
||||||
snowpackConfig: backendSnowpackConfig,
|
snowpackConfig,
|
||||||
} = await createSnowpack(astroConfig, {
|
} = await createSnowpack(astroConfig, {
|
||||||
target: 'backend',
|
|
||||||
env: {
|
|
||||||
astro: true,
|
|
||||||
},
|
|
||||||
mode,
|
mode,
|
||||||
resolvePackageUrl,
|
resolvePackageUrl,
|
||||||
});
|
});
|
||||||
debug(logging, 'core', `backend snowpack created [${stopTimer(timer.backend)}]`);
|
snowpack = snowpackInstance;
|
||||||
|
debug(logging, 'core', `snowpack created [${stopTimer(timer.backend)}]`);
|
||||||
timer.frontend = performance.now();
|
|
||||||
const {
|
|
||||||
snowpack: frontendSnowpack,
|
|
||||||
snowpackRuntime: frontendSnowpackRuntime,
|
|
||||||
snowpackConfig: frontendSnowpackConfig,
|
|
||||||
} = await createSnowpack(astroConfig, {
|
|
||||||
target: 'frontend',
|
|
||||||
env: {
|
|
||||||
astro: false,
|
|
||||||
},
|
|
||||||
mode,
|
|
||||||
});
|
|
||||||
debug(logging, 'core', `frontend snowpack created [${stopTimer(timer.frontend)}]`);
|
|
||||||
|
|
||||||
const runtimeConfig: RuntimeConfig = {
|
const runtimeConfig: RuntimeConfig = {
|
||||||
astroConfig,
|
astroConfig,
|
||||||
logging,
|
logging,
|
||||||
mode,
|
mode,
|
||||||
backendSnowpack,
|
snowpack,
|
||||||
backendSnowpackRuntime,
|
snowpackRuntime,
|
||||||
backendSnowpackConfig,
|
snowpackConfig,
|
||||||
frontendSnowpack,
|
|
||||||
frontendSnowpackRuntime,
|
|
||||||
frontendSnowpackConfig,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
runtimeConfig,
|
runtimeConfig,
|
||||||
load: load.bind(null, runtimeConfig),
|
load: load.bind(null, runtimeConfig),
|
||||||
shutdown: () => Promise.all([backendSnowpack.shutdown(), frontendSnowpack.shutdown()]).then(() => void 0),
|
shutdown: () => snowpack.shutdown(),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
import { logger as snowpackLogger } from 'snowpack';
|
import { logger as snowpackLogger } from 'snowpack';
|
||||||
import { defaultLogLevel } from './logger.js';
|
import { defaultLogLevel } from './logger.js';
|
||||||
|
|
||||||
const onceMessages = ['Ready!', 'watching for file changes'].map((str) => new RegExp(`\\[snowpack\\](.*?)${str}`));
|
|
||||||
|
|
||||||
const neverWarn = new RegExp(
|
const neverWarn = new RegExp(
|
||||||
'(' +
|
'(' +
|
||||||
/(run "snowpack init" to create a project config file.)|/.source +
|
/(run "snowpack init" to create a project config file.)|/.source +
|
||||||
|
@ -26,20 +24,4 @@ export function configureSnowpackLogger(logger: typeof snowpackLogger) {
|
||||||
}
|
}
|
||||||
console.error(message);
|
console.error(message);
|
||||||
});
|
});
|
||||||
|
|
||||||
logger.on('info', (message) => {
|
|
||||||
// Cache messages that should only be shown once.
|
|
||||||
// This is due to having 2 snowpack instances. Once that is removed we can
|
|
||||||
// get rid of this workaround.
|
|
||||||
if (messageCache.has(message)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const shouldBeCached = onceMessages.some((exp) => exp.test(message));
|
|
||||||
if (shouldBeCached) {
|
|
||||||
messageCache.add(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(message);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { h } from 'preact';
|
import { h } from 'preact';
|
||||||
|
|
||||||
export default function (props) {
|
export default function (props) {
|
||||||
return <div id="fallback">{import.meta.env.astro ? 'static' : 'dynamic'}</div>;
|
return <div id="fallback">{import.meta.env.SSR ? 'static' : 'dynamic'}</div>;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue