From 413b9747aabc21eecac2f7b42c2ff975c0019c75 Mon Sep 17 00:00:00 2001 From: Alexander Niebuhr Date: Thu, 28 Sep 2023 14:37:51 +0200 Subject: [PATCH] feat(cloudflare): add local mocking for DO bindings --- .changeset/hot-readers-live.md | 5 +++++ packages/integrations/cloudflare/README.md | 1 + packages/integrations/cloudflare/src/index.ts | 20 +++++++++++++++---- .../cloudflare/src/utils/parser.ts | 17 ++++++++++++++++ .../integrations/cloudflare/test/cf.test.js | 10 ++++++++++ .../test/fixtures/cf/src/pages/do.astro | 15 ++++++++++++++ .../cloudflare/test/fixtures/cf/wrangler.toml | 7 +++++++ 7 files changed, 71 insertions(+), 4 deletions(-) create mode 100644 .changeset/hot-readers-live.md create mode 100644 packages/integrations/cloudflare/test/fixtures/cf/src/pages/do.astro diff --git a/.changeset/hot-readers-live.md b/.changeset/hot-readers-live.md new file mode 100644 index 0000000000000..3aab80846f18e --- /dev/null +++ b/.changeset/hot-readers-live.md @@ -0,0 +1,5 @@ +--- +'@astrojs/cloudflare': minor +--- + +Introduce support for local Durable Objects bindings. Enhance development experience by allowing direct integration with `astro dev`. diff --git a/packages/integrations/cloudflare/README.md b/packages/integrations/cloudflare/README.md index 29cf4b0d2d3b9..0685ccd5e5ea6 100644 --- a/packages/integrations/cloudflare/README.md +++ b/packages/integrations/cloudflare/README.md @@ -218,6 +218,7 @@ Currently supported bindings: - [Cloudflare D1](https://developers.cloudflare.com/d1/) - [Cloudflare R2](https://developers.cloudflare.com/r2/) - [Cloudflare Workers KV](https://developers.cloudflare.com/kv/) +- [Cloudflare Durable Objects](https://developers.cloudflare.com/durable-objects/) You can access the runtime from Astro components through `Astro.locals` inside any .astro` file. diff --git a/packages/integrations/cloudflare/src/index.ts b/packages/integrations/cloudflare/src/index.ts index 2c5ea16367fdf..ddbf29ddd8477 100644 --- a/packages/integrations/cloudflare/src/index.ts +++ b/packages/integrations/cloudflare/src/index.ts @@ -12,7 +12,13 @@ import glob from 'tiny-glob'; import { getAdapter } from './getAdapter.js'; import { deduplicatePatterns } from './utils/deduplicatePatterns.js'; import { getCFObject } from './utils/getCFObject.js'; -import { getD1Bindings, getEnvVars, getKVBindings, getR2Bindings } from './utils/parser.js'; +import { + getD1Bindings, + getDOBindings, + getEnvVars, + getKVBindings, + getR2Bindings, +} from './utils/parser.js'; import { prependForwardSlash } from './utils/prependForwardSlash.js'; import { rewriteWasmImportPath } from './utils/rewriteWasmImportPath.js'; import { wasmModuleLoader } from './utils/wasm-module-loader.js'; @@ -20,8 +26,6 @@ import { wasmModuleLoader } from './utils/wasm-module-loader.js'; export type { AdvancedRuntime } from './entrypoints/server.advanced.js'; export type { DirectoryRuntime } from './entrypoints/server.directory.js'; -// export { Response as MiniflareResponse } from 'miniflare'; - type Options = { mode?: 'directory' | 'advanced'; functionPerRoute?: boolean; @@ -115,7 +119,7 @@ export default function createIntegration(args?: Options): AstroIntegration { const D1Bindings = await getD1Bindings(); const R2Bindings = await getR2Bindings(); const KVBindings = await getKVBindings(); - + const DOBindings = await getDOBindings(); let bindingsEnv = new Object({}); // fix for the error "kj/filesystem-disk-unix.c++:1709: warning: PWD environment variable doesn't match current directory." @@ -135,6 +139,8 @@ export default function createIntegration(args?: Options): AstroIntegration { r2Persist: true, kvNamespaces: KVBindings, kvPersist: true, + durableObjects: DOBindings, + durableObjectsPersist: true, }); await _mf.ready; @@ -150,6 +156,12 @@ export default function createIntegration(args?: Options): AstroIntegration { const namespace = await _mf.getKVNamespace(KVBinding); Reflect.set(bindingsEnv, KVBinding, namespace); } + for (const key in DOBindings) { + if (Object.prototype.hasOwnProperty.call(DOBindings, key)) { + const DO = await _mf.getDurableObjectNamespace(key); + Reflect.set(bindingsEnv, key, DO); + } + } const mfCache = await _mf.getCaches(); process.env.PWD = originalPWD; diff --git a/packages/integrations/cloudflare/src/utils/parser.ts b/packages/integrations/cloudflare/src/utils/parser.ts index af64e68c68aa9..de239a1ee7071 100644 --- a/packages/integrations/cloudflare/src/utils/parser.ts +++ b/packages/integrations/cloudflare/src/utils/parser.ts @@ -172,3 +172,20 @@ export async function getKVBindings() { ); return bindings; } + +export function getDOBindings(): Record< + string, + { scriptName?: string | undefined; unsafeUniqueKey?: string | undefined; className: string } +> { + const { rawConfig } = parseConfig(); + if (!rawConfig) return {}; + if (!rawConfig?.durable_objects) return {}; + const output = new Object({}) as Record< + string, + { scriptName?: string | undefined; unsafeUniqueKey?: string | undefined; className: string } + >; + for (const binding of rawConfig?.durable_objects.bindings) { + Reflect.set(output, binding.name, { className: binding.class_name }); + } + return output; +} diff --git a/packages/integrations/cloudflare/test/cf.test.js b/packages/integrations/cloudflare/test/cf.test.js index 0276edd6a5a6b..0078f30245b3b 100644 --- a/packages/integrations/cloudflare/test/cf.test.js +++ b/packages/integrations/cloudflare/test/cf.test.js @@ -114,4 +114,14 @@ describe('Astro Cloudflare Runtime', () => { expect($('#hasPRODKV').text()).to.equal('true'); expect($('#hasACCESS').text()).to.equal('true'); }); + + it('adds DO mocking', async () => { + expect(await fixture.pathExists('../.mf/do')).to.be.true; + + let res = await fixture.fetch('/do'); + expect(res.status).to.equal(200); + let html = await res.text(); + let $ = cheerio.load(html); + expect($('#hasDO').text()).to.equal('true'); + }); }); diff --git a/packages/integrations/cloudflare/test/fixtures/cf/src/pages/do.astro b/packages/integrations/cloudflare/test/fixtures/cf/src/pages/do.astro new file mode 100644 index 0000000000000..e338c8e8f9cc4 --- /dev/null +++ b/packages/integrations/cloudflare/test/fixtures/cf/src/pages/do.astro @@ -0,0 +1,15 @@ +--- +const runtime = Astro.locals.runtime; +--- + + + + + + + DO + + +
{!!runtime.env.DO}
+ + diff --git a/packages/integrations/cloudflare/test/fixtures/cf/wrangler.toml b/packages/integrations/cloudflare/test/fixtures/cf/wrangler.toml index a0fc8ddb5cd93..17fd88ea5cb95 100644 --- a/packages/integrations/cloudflare/test/fixtures/cf/wrangler.toml +++ b/packages/integrations/cloudflare/test/fixtures/cf/wrangler.toml @@ -28,3 +28,10 @@ bucket_name = '' binding = 'R2_PROD' # <~ valid JavaScript variable name bucket_name = '' +[[durable_objects.bindings]] +name = "DO" +class_name = "DurableObjectExample" + +[[durable_objects.bindings]] +name = "DO_PROD" +class_name = "DurableObjectProductionExample"