enable access to cloudflare runtime (#5103)

* enable access to cloudflare runtime

* added get runtime api
added context to the runtime in "advanced" mode

* added typings and adjusted some return vars

* added default types

* added usage description to changeset and readme

Co-authored-by: AirBorne04 <unknown>
Co-authored-by: AirBorne04 <>
This commit is contained in:
Daniel 2022-10-26 15:46:25 +02:00 committed by GitHub
parent 4efbfdd78d
commit d151d9f3f2
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 58 additions and 4 deletions

View file

@ -0,0 +1,6 @@
---
'@astrojs/cloudflare': patch
---
enable access to cloudflare runtime [KV, R2, Durable Objects]
- access native cloudflare runtime through `import { getRuntime } from "@astrojs/cloudflare/runtime"` now you can call `getRuntime(Astro.request)` and get access to the runtime environment

View file

@ -68,6 +68,18 @@ $ pnpm install wrangler --save-dev
It's then possible to update the preview script in your `package.json` to `"preview": "wrangler pages dev ./dist"`.This will allow you run your entire application locally with [Wrangler](https://github.com/cloudflare/wrangler2), which supports secrets, environment variables, KV namespaces, Durable Objects and [all other supported Cloudflare bindings](https://developers.cloudflare.com/pages/platform/functions/#adding-bindings). It's then possible to update the preview script in your `package.json` to `"preview": "wrangler pages dev ./dist"`.This will allow you run your entire application locally with [Wrangler](https://github.com/cloudflare/wrangler2), which supports secrets, environment variables, KV namespaces, Durable Objects and [all other supported Cloudflare bindings](https://developers.cloudflare.com/pages/platform/functions/#adding-bindings).
## Access to the cloudflare runtime
You have the posibility to access all the cloudflare bindings and environment variables from your astro pages and api routes through the adapter API
```
import { getRuntime } from "@astrojs/cloudflare/runtime";
getRuntime(Astro.request);
```
Depending your adapter mode (advanced = worker, directory = pages) the runtime object will look a little different due to the difference in the cloudflare API.
## Streams ## Streams
Some integrations such as [React](https://github.com/withastro/astro/tree/main/packages/integrations/react) rely on web streams. Currently Cloudflare Pages Functions require enabling a flag to support Streams. Some integrations such as [React](https://github.com/withastro/astro/tree/main/packages/integrations/react) rely on web streams. Currently Cloudflare Pages Functions require enabling a flag to support Streams.

View file

@ -1,6 +1,6 @@
{ {
"name": "@astrojs/cloudflare", "name": "@astrojs/cloudflare",
"description": "Deploy your site to cloudflare pages functions", "description": "Deploy your site to cloudflare workers or cloudflare pages",
"version": "3.1.0", "version": "3.1.0",
"type": "module", "type": "module",
"types": "./dist/index.d.ts", "types": "./dist/index.d.ts",
@ -19,6 +19,7 @@
"homepage": "https://docs.astro.build/en/guides/integrations-guide/cloudflare/", "homepage": "https://docs.astro.build/en/guides/integrations-guide/cloudflare/",
"exports": { "exports": {
".": "./dist/index.js", ".": "./dist/index.js",
"./runtime": "./dist/runtime.js",
"./server.advanced.js": "./dist/server.advanced.js", "./server.advanced.js": "./dist/server.advanced.js",
"./server.directory.js": "./dist/server.directory.js", "./server.directory.js": "./dist/server.directory.js",
"./package.json": "./package.json" "./package.json": "./package.json"

View file

@ -0,0 +1,28 @@
export type WorkerRuntime<T = unknown> = {
name: 'cloudflare';
env: T;
waitUntil(promise: Promise<any>): void;
passThroughOnException(): void;
};
export type PagesRuntime<T = unknown, U = unknown> = {
name: 'cloudflare';
env: T;
functionPath: string;
params: Record<string, string>;
data: U;
waitUntil(promise: Promise<any>): void;
next(request: Request): void;
};
export function getRuntime<T = unknown, U = unknown>(
request: Request
): WorkerRuntime<T> | PagesRuntime<T, U> {
if (!!request) {
return Reflect.get(request, Symbol.for('runtime'));
} else {
throw new Error(
'To retrieve the current cloudflare runtime you need to pass in the Astro request object'
);
}
}

View file

@ -5,12 +5,13 @@ import { App } from 'astro/app';
type Env = { type Env = {
ASSETS: { fetch: (req: Request) => Promise<Response> }; ASSETS: { fetch: (req: Request) => Promise<Response> };
name: string;
}; };
export function createExports(manifest: SSRManifest) { export function createExports(manifest: SSRManifest) {
const app = new App(manifest, false); const app = new App(manifest, false);
const fetch = async (request: Request, env: Env) => { const fetch = async (request: Request, env: Env, context: any) => {
const { origin, pathname } = new URL(request.url); const { origin, pathname } = new URL(request.url);
// static assets // static assets
@ -26,6 +27,7 @@ export function createExports(manifest: SSRManifest) {
Symbol.for('astro.clientAddress'), Symbol.for('astro.clientAddress'),
request.headers.get('cf-connecting-ip') request.headers.get('cf-connecting-ip')
); );
Reflect.set(request, Symbol.for('runtime'), { env, name: 'cloudflare', ...context });
let response = await app.render(request, routeData); let response = await app.render(request, routeData);
if (app.setCookieHeaders) { if (app.setCookieHeaders) {

View file

@ -9,12 +9,12 @@ export function createExports(manifest: SSRManifest) {
const onRequest = async ({ const onRequest = async ({
request, request,
next, next,
...runtimeEnv
}: { }: {
request: Request; request: Request;
next: (request: Request) => void; next: (request: Request) => void;
}) => { } & Record<string, unknown>) => {
const { origin, pathname } = new URL(request.url); const { origin, pathname } = new URL(request.url);
// static assets // static assets
if (manifest.assets.has(pathname)) { if (manifest.assets.has(pathname)) {
const assetRequest = new Request(`${origin}/static${pathname}`, request); const assetRequest = new Request(`${origin}/static${pathname}`, request);
@ -28,6 +28,11 @@ export function createExports(manifest: SSRManifest) {
Symbol.for('astro.clientAddress'), Symbol.for('astro.clientAddress'),
request.headers.get('cf-connecting-ip') request.headers.get('cf-connecting-ip')
); );
Reflect.set(request, Symbol.for('runtime'), {
...runtimeEnv,
name: 'cloudflare',
next,
});
let response = await app.render(request, routeData); let response = await app.render(request, routeData);
if (app.setCookieHeaders) { if (app.setCookieHeaders) {