-
Notifications
You must be signed in to change notification settings - Fork 5
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Dynamic imports #1
Comments
Federated wiki uses dynamic imports for rarely used or newly created item formatting plugins. We haven't yet figured out exactly how Deploy helps us, perhaps as insolation from older or non-compliant browsers. We usually stop thinking these things when we remember we can't load plugins. |
I'm trying to get https://github.com/exhibitionist-digital/ultra deployed (react 18, streaming SSR) but after forking it and removing the import maps I found that it uses dynamic imports, so it's currently not possible to deploy with. |
@Industrial Same...I don't really see the reason for dynamic import being disabled, but I'm also not a backend expert. I tried a lot of things but I eventually gave up and now use a DigitalOcean Droplet...but a Deno Deploy solution would be really awesome, not just because of the money I now have to pay for DO, but because Deno Deploy is so easy and fast to work with. |
I can see why dynamic imports of remote libs may be unwanted, but there is a lot of value in the ability to dynamically import local modules. Modern frontend frameworks could then have native code-splitting -- importing route and component files as needed, like @newtack explained above. Libraries like React and Solid are now supporting dynamic imports. It currently does work in Deno when server side rendering, so I imagine we will be seeing more of this in the near future: https://reactjs.org/docs/code-splitting.html#reactlazy Maybe we could declare all remote libs in an importmap -- then only allow local modules to be imported? @Industrial I did push and update to https://github.com/exhibitionist-digital/ultra which allows deployment to Deno Deploy, but yes, you need to ensure you don't actually use dynamic imports in your codebase. |
Here's an example similar to the situation as OP that doesn't work right now in Deno Deploy: https://github.com/BrunoBernardino/deno-boilerplate-simple-website/blob/5d32fb081d6080564e91a8ca8fdd6921ab9420cc/routes.ts#L31-L35 This means the alternative is to load all pages even if they're not going to be needed, so that it works in Deno Deploy: https://github.com/BrunoBernardino/deno-boilerplate-simple-website/blob/5d32fb081d6080564e91a8ca8fdd6921ab9420cc/routes.ts#L4-L12 Not having this functionality means it'll be impossible for any similar performance logic to be used with Deno Deploy, which seems like a shame. |
Not having dynamic imports prevents this from working: https://github.com/withastro/astro/tree/main/packages/integrations/deno |
I'm building an web library (imagine React but 100× better) but would need Dynamic imports for async importing of data URLs. Of course works in Deno, but I would also like it to deployable to Deno Deploy. Would be great if you'd add this! Thanks. |
* add version selector * dynamix import not supported in Deno deploy...so denoland/deploy_feedback#1
For those that land here from GQL (https://doc.deno.land/https://deno.land/x/gql@1.1.1/mod.ts or https://deno.land/x/gql@1.1.1), to avoid using the dynamic import in this module, I created a local version of the import {
GQLOptions,
runHttpQuery,
} from 'https://deno.land/x/gql@1.1.1/common.ts';
import { renderPlaygroundPage } from 'https://deno.land/x/gql@1.1.1/graphiql/render.ts';
import type { GQLRequest } from 'https://deno.land/x/gql@1.1.1/types.ts';
// !!! all code below is from https://deno.land/x/gql@1.1.1/http.ts
// except for removal of dynamic import
/**
* Create a new GraphQL HTTP middleware with schema, context etc
* @param {GQLOptions} options
*
* @example
* ```ts
* const graphql = await GraphQLHTTP({ schema })
*
* for await (const req of s) graphql(req)
* ```
*/
export function GraphQLHTTP<
Req extends GQLRequest = GQLRequest,
Ctx extends { request: Req } = { request: Req },
>({
playgroundOptions = {},
headers = {},
...options
}: GQLOptions<Ctx, Req>) {
return async (request: Req) => {
if (options.graphiql && request.method === 'GET') {
if (request.headers.get('Accept')?.includes('text/html')) {
// !!! only modified portion is here
// const { renderPlaygroundPage } = await import(
// "./graphiql/render.ts"
// );
// !!! end of modified portion
const playground = renderPlaygroundPage({
...playgroundOptions,
endpoint: '/graphql',
});
return new Response(playground, {
headers: new Headers({
'Content-Type': 'text/html',
...headers,
}),
});
} else {
return new Response(
'"Accept" header value must include text/html',
{
status: 400,
headers: new Headers(headers),
},
);
}
} else {
if (!['PUT', 'POST', 'PATCH'].includes(request.method)) {
return new Response('Method Not Allowed', {
status: 405,
headers: new Headers(headers),
});
} else {
try {
const result = await runHttpQuery<Req, Ctx>(
await request.json(),
options,
{ request },
);
return new Response(JSON.stringify(result, null, 2), {
status: 200,
headers: new Headers({
'Content-Type': 'application/json',
...headers,
}),
});
} catch (e) {
console.error(e);
return new Response('Malformed request body', {
status: 400,
headers: new Headers(headers),
});
}
}
}
};
} |
Any movement on this? It's keeping me from using Deno Deploy for a project Edit: I came up with a workaround; I'll share it here for others that may be stuck. My situation is one where I didn't strictly need dynamic imports (the modules to be imported all exist in source control), I just wanted to import them dynamically for ergonomics; when I add a new file in a certain directory, I want it to be picked up without going and manually configuring it to be loaded. What I did as a workaround is I wrote a script to walk that directory and generate a TypeScript file that statically imports all the relevant modules, and re-exports their contents in a way where they can be looked up by module path. This manifest is then checked into source control, so by the time it gets to Deno Deploy everything is static. Hopefully I'll be able to drop this mechanism one day. You can see it here: https://github.com/brundonsmith/site/blob/master/routes-manifest-generate.ts |
I'm hitting this because I need to set up a bunch of node-y / browser-y globals before my app's code is loaded due to a library not supporting deno - Azure/azure-sdk-for-js#13281 (comment) ( |
It's been over a year. This was the first issue created on this repository. If we were going to get a response, we would've gotten one by now. |
Has anyone tried out this ponyfill? It claims to let you import from a string on Deno Deploy. |
@MarkBennett Looking at the code, it should generally work. But as soon as you need to rely on anything related to module resolving or the module metadata itself (stuff like Also it bundles everything into one big function, which could lead to code being duplicated. |
Generate a route manifest to make it deployable on Deno Deploy. Inspired by Fresh [1] and Deno Deploy limitations [2] [1]: https://github.com/denoland/fresh [2]: denoland/deploy_feedback#1
Ultra 2.1.0 has an option to inline server dynamic imports while still retaining client side dynamic usage. Both with If this helps anyone else: exhibitionist-digital/ultra#195 |
I'm trying to run Observable notebooks in Deno so that I can write client and server-side code in notebooks. I prefer to import notebooks dynamically vs. maintaining a bunch of static imports somewhere in my Deno code. I have no issues when running my Deno script locally, but when I try to run them in Deno Deploy, I get:
Are there any workarounds? I don't think it's possible to |
@gnestor The "easiest" workaround would be to do what I've been using Deno Deploy since its inception and honestly it just doesn't work when things "get going". I need to move into fly.io for that (another example is that |
@BrunoBernardino Thanks for that suggestion. There will be cases where I want to run cells from a new notebook that's not included in the source, in which case I would need to re-run the script and update the source, which is doable but not ideal. I was able to work around this by using https://github.com/ayoreis/import and patching it: ayoreis/import#10 (comment) https://github.com/ayoreis/import works by essentially fetching the source code of the module, using esbuild to convert it into non-module Javascript, and creating an async function instance using it. If the native @BrunoBernardino I'll check out fly.io. I've looked at Cloudflare Workers as an alternative, but I ran into issues immediately trying to follow the instructions at https://deno.land/manual@v1.28.1/advanced/deploying_deno/cloudflare_workers. I have used AWS Lambda extensively for the past 5+ years and I have tried https://github.com/hayd/deno-lambda, but one of the biggest draws of Deno Deploy is the zero-config, and AWS is so much config. It looks like Google Cloud Run is another option and Digital Ocean and AWS LightSail for running Deno inside of Docker. Do people have any other recommendations that are closer to the Deno Deploy DX? I assume Vercel can run Deno and it has a similar DX. @ry and Deno team, any plans to allow dynamic imports in Deno Deploy? |
It looks like dynamic imports will also fail in Vercel but they have a workaround: https://github.com/vercel-community/deno#dynamic-imports
Is this workaround possible with Deno Deploy? |
I really wanted to avoid this, what happened to no build step, one of Deno original selling point |
Goddamnit, there should really be some kind of warning about this when using it in development, even if it's just some CLI warning, Deno already does a great job constantly warning us to not perform anti-patterns. This should be definitely added to avoid us developers wasting so much time developing a feature that will not work on Deno Deploy (if you guys want it to be the defacto choice for Deno websites, that is), but will work just fine everywhere else (like the user above mentioned using Digital Ocean). I did a whole thing where I created blog posts and would dynamically load them from I love the DX, but I'll be damned if the overall (lack of|outdated) documentation doesn't put a damper on my excitement about this runtime. |
Tell me about it. Compare https://github.com/BrunoBernardino/deno-boilerplate-simple-website/blob/main/routes.ts#L43-L44 (what's needed if Deno Deploy supported it): // NOTE: Use this instead once https://github.com/denoland/deploy_feedback/issues/1 is closed
// const { pageContent } = await import(`./pages/${id}.ts`); vs https://github.com/BrunoBernardino/deno-boilerplate-simple-website/blob/main/routes.ts#L10-L24 (what's necessary for it to work on Deno Deploy): // NOTE: This won't be necessary once https://github.com/denoland/deploy_feedback/issues/1 is closed
import * as indexPage from './pages/index.ts';
import * as ssrPage from './pages/ssr.ts';
import * as dynamicPage from './pages/dynamic.ts';
import * as formPage from './pages/form.ts';
import * as webComponentPage from './pages/web-component.ts';
import * as reactPage from './pages/react.tsx';
const pages = {
index: indexPage,
ssr: ssrPage,
dynamic: dynamicPage,
form: formPage,
webComponent: webComponentPage,
react: reactPage,
}; |
This is a show stopper for using MDX on Deno Deploy without introducing a build step, which is a shame. |
What I was surprised at is that Fresh states it has “No build step”, but actually it does a build internally. It's just hidden behind some UX design while development, but if one wants to track the project by Git (necessarily!), they has to be aware of this fact, as noticed in the generated
It's okay for me, but feels like this could introduce chances of deployment failure, like forgetting to regenerate |
As part of the migration to v3 we need to be able to update the documentation as well be able to have a method to embed playgrounds into our websites. This will be crucial for Effection, but also in derivative libraries and future products. Goals: 1. Fast deploys, easy previews. 2. Zero build 3. Just HTML™️ 4. Explicit vs Implicit Render This is just simple Deno server that serves HTML templates. The server definition is in `www/server.ts`. It uses a route recognizer to parse path and query parameters, and then calls an HTML template that is expressed in JavaScript (The data structure is defined by https://github.com/cowboyd/html) and then serialized to a DOM using [deno_dom](https://github.com/denoland/deno_dom) and Google's [incremental-dom` library](https://github.com/google/incremental-dom). Tailwind CSS is used as the atomic CSS framework. The documentation, like the v2 website is based on MDX. Ideally, we would like to not require a build step for this, and in development mode we can, by compiling the MDX into a JavaScript module. However, becaue Deno Deploy [does not support dynamic import](denoland/deploy_feedback#1), this is not an option at the moment. Future directions: - I think we can simplify things considerably by using [hast](https://github.com/syntax-tree/hastscript) as the structural representation of the HTML. This means several things: We can apply rehypejs plugins at any level, not just MDX, and also we do not need to use `deno_dom` nor `incremental-dom` for server side rendering since there is a direct transform from `hast` -> `string` - It would be nice to have autocompletion and type checking for tailwind classes. - Because we're based on Effection, we can use really nice helpers to interrupt a request mid-flight to redirect, render 404, etc...
As part of the migration to v3 we need to be able to update the documentation as well be able to have a method to embed playgrounds into our websites. This will be crucial for Effection, but also in derivative libraries and future products. Goals: 1. Fast deploys, easy previews. 2. Zero build 3. Just HTML™️ 4. Explicit vs Implicit Render This is just simple Deno server that serves HTML templates. The server definition is in `www/server.ts`. It uses a route recognizer to parse path and query parameters, and then calls an HTML template that is expressed in JavaScript (The data structure is defined by https://github.com/cowboyd/html) and then serialized to a DOM using [deno_dom](https://github.com/denoland/deno_dom) and Google's [incremental-dom` library](https://github.com/google/incremental-dom). Tailwind CSS is used as the atomic CSS framework. The documentation, like the v2 website is based on MDX. Ideally, we would like to not require a build step for this, and in development mode we can, by compiling the MDX into a JavaScript module. However, becaue Deno Deploy [does not support dynamic import](denoland/deploy_feedback#1), this is not an option at the moment. Future directions: - I think we can simplify things considerably by using [hast](https://github.com/syntax-tree/hastscript) as the structural representation of the HTML. This means several things: We can apply rehypejs plugins at any level, not just MDX, and also we do not need to use `deno_dom` nor `incremental-dom` for server side rendering since there is a direct transform from `hast` -> `string` - It would be nice to have autocompletion and type checking for tailwind classes. - Because we're based on Effection, we can use really nice helpers to interrupt a request mid-flight to redirect, render 404, etc...
This is now supported https://deno.com/deploy/changelog#statically-analyzable-dynamic-imports |
This is a great feature! Now the sky is the limit =) |
Actually only statically analyzable doesn't solve my use case :/ |
This doesn't help my use case either. If the issue is considered complete now, I'm glad that I've pivoted to another method of fetching my files, but it would have sucked if I had waited for this. |
It turn's out I was wrong here. You can use MDX without a build step on deno deploy. For any other MDX users who come across this, it supports an You can't do static imports inside your MDX, but you can just do static imports inside your JS code and then pass those values down into the mdx scope. |
As part of the migration to v3 we need to be able to update the documentation as well be able to have a method to embed playgrounds into our websites. This will be crucial for Effection, but also in derivative libraries and future products. Goals: 1. Fast deploys, easy previews. 2. Zero build 3. Just HTML™️ 4. Explicit vs Implicit Render This is just simple Deno server that serves HTML templates. The server definition is in `www/server.ts`. It uses a route recognizer to parse path and query parameters, and then calls an HTML template that is expressed in JavaScript (The data structure is defined by https://github.com/cowboyd/html) and then serialized to a DOM using [deno_dom](https://github.com/denoland/deno_dom) and Google's [incremental-dom` library](https://github.com/google/incremental-dom). Tailwind CSS is used as the atomic CSS framework. The documentation, like the v2 website is based on MDX. Ideally, we would like to not require a build step for this, and in development mode we can, by compiling the MDX into a JavaScript module. However, becaue Deno Deploy [does not support dynamic import](denoland/deploy_feedback#1), this is not an option at the moment. Future directions: - I think we can simplify things considerably by using [hast](https://github.com/syntax-tree/hastscript) as the structural representation of the HTML. This means several things: We can apply rehypejs plugins at any level, not just MDX, and also we do not need to use `deno_dom` nor `incremental-dom` for server side rendering since there is a direct transform from `hast` -> `string` - It would be nice to have autocompletion and type checking for tailwind classes. - Because we're based on Effection, we can use really nice helpers to interrupt a request mid-flight to redirect, render 404, etc...
I know this is not supported today, but would love to see this being supported in the future.
Use case: We have a dynamic server rendered web server where we use configuration and dynamic imports based on that configuration to fetch the relevant page handler.
The text was updated successfully, but these errors were encountered: