-
Notifications
You must be signed in to change notification settings - Fork 32
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
Sessions #1055
base: main
Are you sure you want to change the base?
Conversation
// The required options depend on the driver | ||
options: { | ||
base: "session", | ||
host: process.env.REDIS_HOST, | ||
password: process.env.REDIS_PASSWORD, | ||
}, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If there are drivers that don't require options, don't options
become optional? How do we validate options against the driver? Is it possible to do this at the schema level?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, options are optional. We don't have schemas for the options, but we have types, and those do correctly handle the different driver options.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok. I read // The required options
and thought that they were required. So they are optional, but needs to be provided based on the driver.
Will we throw an error if those options are missing for specific drivers?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
// The required options depend on the driver
means some options will or will not be required, depending on the driver chosen.
The driver will throw the error if required options are missing.
proposals/0054-sessions.md
Outdated
All configuration is optional if the adapter has built-in support. This is the | ||
case for the Node adapter, which uses filesystem storage by default, and is | ||
expected to be the case for most adapters. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure this should be part of the proposal, specifically the part when we talk about specific adapters. Or, if we really need to, we should try to actually to be as generic as possible because we don't know what and how the specific adapters will behave.
// Required: the name of the Unstorage driver | ||
driver: "redis", |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have mixed feelings about using driver
in a "shared" configuration. What if the redis
driver isn't supported by a specific adapter or any other adapter? How can we tell the user that they can't use a certain adapter with certain driver?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do you mean by shared configuration?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
With "shared", I meant that If we define a driver
here, I would assume - as a user - that this driver will work regardless of any adapter used by. So it's a configuration "shared" by core and the adapter at play
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There's not such a tight relationship between an adapter and a driver. An adapter doesn't "support" a driver directly. It's more correct to say that some adapters do or do not support:
- Persistent memory
- FS access
- TCP sockets
So for your example, it's not that some adapters support redis
, it's that redis works over TCP sockets and some adapters may or may not support opening sockets. But we don't have such a feature map to know those sorts of requirements.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For any drivers that we aren't automatically enabling (so, basically all of them except fs) I think we need to delegate all the documentation to Unstorage, at least in the short term, and that includes which platforms are support. Anything else would be too hard to keep up to date. In future we may decide we want to own the drivers ourselves, but right now I think the idea is to rely on Unstorage wherever possible. If we need changes then we can contribute them upstream.
proposals/0054-sessions.md
Outdated
Because of the variety of environments to which Astro can be deployed, there is | ||
no single approach to storage that can be relied upon in all cases. For this | ||
reason, adapters should provide default session storage drivers where possible. | ||
Sessions are only available in on-demand rendered pages and API endpoints, so |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
and API endpoints
This is misleading, because endpoints that look like src/pages/schema.json.js
are run during the build, they create a physical file called src/pages/schea.json
and they aren't run in SSR anymore.
The endpoint needs to be on-demand, too.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The "on-demand rendered" is meant to apply to "pages and API endpoints". I'll reword it to be less ambiguous.
proposals/0054-sessions.md
Outdated
The adapter or integration is responsible for ensuring that they do not | ||
overwrite any user-defined driver configuration. Adapters may choose to accept | ||
their own configuration options which they can apply to the storage driver where | ||
needed. Adapters may provide a storage driver for use in development, or rely on | ||
the built-in node adapter which is provided by the dev server. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should really highlight what would happen if drivers go in conflict, and if so how users/developers can do that. Is Astro responsible for that? Is the adapter for that? Is the user responsible for that?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There can only be one driver, so I don't think there's an issue with conflicts there. Adapters are meant to prefer the user-defined drivers, though I don't think this is something we can enforce when adapters can do what they want to the config.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here's a concrete example that comes to mind
A project with a Cloudflare adapter that attempts to use a Netlify blob. What would happen in this occasion? How does Astro behave? It isn't clear to me from the RFC:
import cloudflare from "@astrojs/cloudflare"
import {defineConfg} from "astro/config"
defineConfig({
driver: 'netlify-blobs', // should use nstorage/drivers/netlify-blobs
adapter: cloudflare()
})
Is it possible to have such a combination? If so, how? It doesn't seem likely
In another case, the use of an adapter that doesn't allow FS storage:
import cloudflare from "@astrojs/cloudflare" // let's assume `node:fs` isn't compatible with cloudflare
import {defineConfg} from "astro/config"
defineConfig({
driver: 'fs',
adapter: cloudflare()
})
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think we need to do checks for people doing things that are clearly not allowed on that platform. Why would somebody choose Netlify blobs if they're not using Netlify? This isn't something we're doing automatically and hiding from them: this is equivalent to somebody choosing the Netlify adapter when they're deploying to Cloudflare, or using fs
in a page on a site deployed to Cloudflare.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why would somebody choose Netlify blobs if they're not using Netlify?
Mistakes, junior people who don't know very well what they're learning, possible differences between local behaviour and production behaviour, etc.
I would like to improve that DX if possible. Why can't an adapter accept a predefined list of supported storage?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, true it should work if you set the credentials manually
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't know if its true that you can't use the netlify-blobs driver with Cloudflare. If netlify-blobs operates over HTTP then more than likely it would work.
I don't know either, it was just an example to explain what I meant.
It would make more sense if they provided a list of drivers not supported
That's a step forward!
I'd really like to avoid coupling us so closely to the Unstorage driver implementation. One of the design goals for this is to avoid needing to maintain them. If we did that, we'd need to update all adapters whenever they add a new driver, and theoretically audit them all manually whenever that happens.
I understand that, however we need to strive towards a decent enough DX and UX for our users. Here are some considerations:
- We are responsible for the libraries we use as a service (see
sharp
,drizzle
, etc.), and we always try to create a certain layer of compatibility for our users. - Since we don't state which library we rely on (rightfully), Astro is free to change the underneath library (if it ever happens) without changing the user facing APIs. For images, for example, we remove
squoosh
, but the APIs are unchanged - AFAIK.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's a bigger question then. If we want to fully abstract away unstorage to the extent that we don't need to mention it then we need to document every driver ourselves. Now that may be a sensible thing to do, but it will mean we need to decide on a more limited list of ones that we will support.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We talked about this on a call, but I'll restate that we can't ensure that a user only uses compatible libraries with Astro. There are a number of places where a user might use a library that doesn't work:
- Remark plugins
- Vite plugins
- Framework components
There's a tradeoff between protecting the user from mistakes and giving them the power to do what they want. I think this is a case where there's a large amount of compatibility already, and users will naturally use drivers that make sense for their host anyways. Adding restrictions would make the API less powerful and unnecessarily prevent legitimate use-cases.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
unstorage
s raison d’être is to be the uncoupled interface and provide many implementations. It's used successfully by nitro and I can attest for its ergonomics. I don't think there's a compelling reason to reimplement it just to own the namespace it comes from
Regarding saving devs from themselves, obviously it comes from a good place but it's just a wild goose chase. The higher the guard rails, the more convoluted the hacks people would put in place to skip them. If a junior person who doesn't know very well what they're learning make a mistake then they would learn "the hard way". I personally believe that providing good descriptive errors is much more educational and effective than guardrails. Generally this is something astro is very good at, so it's not an odd expectation.
The suggested sessions API is already 99% existing. Add unstorage
and use Astro.locals
, mix in some cookie action and you're golden. The hard part of sessions was the async local context. With an RFC this obvious and implementation this lean it'll be a shame to bikeshed the issue when everyone involved can clearly see the value.
I hope it's ok that I butted in :)
proposals/0054-sessions.md
Outdated
- Type-safety. This may be added later, but there are questions about the best | ||
way to do this, because of issues with the way that the config is loaded. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Of course can be added later but adding it from the get-go would be nicer IMO.
To propose one option: Optional override in env.d.ts
.
Ideally everything could be inferred from the Astro config in some way but in the mean time I think a sufficient alternative would be to allow users to add types to their src/env.d.ts
to add explicit types for sessions. Similar to how it is done for Cloudflare types. For example, based on the examples above:
/// <reference path="../.astro/types.d.ts" />
/// <reference types="astro/client" />
declare namespace App {
interface Sessions {
cart: Array<string>;
}
}
Or if long-term things were to get more advanced some kind of schema
could be added as an option to the session config, similar to server actions, to both validate data & infer types from.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You must distinguish between typed and type-safe, sessions must be assumed unreliable and partial in the best of circumstances, enforcing it with a schema is just a trap for when the sessions start to skew or expire and suddenly no longer fit. I think either a FormData
-like API (as suggested) or a typing that gets Partial
-ized on the object is best, though it'll suck a bit ngl
Some suggestions from the API bash:
|
proposals/0054-sessions.md
Outdated
|
||
### `Astro.session.flash(key: string, value: any): void` | ||
|
||
Sets the value of the given key in the session. The value can be any serializable type. The value will be deleted after the next request. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How does Astro.sessions.flash
differ from Astro.locals
? Other than the fact that the latter can contain non-serializable values.
after the next request
The .rewrite
API creates a new request, does that mean that this value should disappear? For example, we navigate the page /foo
, which then executes Astro.rewrite('/failure')
. If pages/failure.astro
reads the flash, would there be any value?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
flash()
is persisted across requests. If you set Astro.locals
and then redirect, you won't have access to the result. The value passed to flash()
is persisted to storage, but is then deleted after the next request.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's what it's unclear to me:
Astro.locals
persists across requests, too.- It's unclear to me what "next request" means in this case because
Astro.rewrite
andnext()
create a newRequest
object
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How does Astro.locals
persist across requests? If I have a redirect, how would I read the value in the target page?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OK, I understand what you meant. I understood "across requests" differently. Astro.locals
doesn't persist.
I don't understand the usage of flash()
though. First, it's written that it persists across requests (page navigations), but then it's written that it disappears after the next request (page navigation). It seems like it's similar to Astro.locals
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Those can also sneak in, and the worse the network conditions are the more likely it is to happen. I can go into many specific real-world scenarios where this is very likely. Anything with SWR, pings, SSE, any sort of polling, prefetching, navigation, multiple-tabs, timeouts, there are so many reasons for requests to be sent in indeterminate times. This is even more pronounced in authenticated flows since most interactions would check the identity of the requester, these flows are also the most likely to benefit from the session API
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
flash
is an optional API. You can always use a regular session.set
if your use case may have extra requests.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
How is flash implemented in other frameworks that support it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I basically copied the Laravel API.
Sometimes you may wish to store items in the session for the next request. You may do so using the flash method. Data stored in the session using this method will be available immediately and during the subsequent HTTP request. After the subsequent HTTP request, the flashed data will be deleted. Flash data is primarily useful for short-lived status messages:
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
After playing with sessions a bit and based on my understanding of Laravel/Rails/Flask (which also have a flash) - I don't think the same API makes sense in a framework that leverages SSG as much as Astro does. My first impulse here is to make consuming a flash message intentional, i.e. an Action or an SSR-d layout calls consume()
and receives all pending flash messages, and then renders them as needed.
That doesn't feel like something that could be as "one size fits all" as you would want, and it might be worth it to just provide users with a recipe for flash message passing similar to the recipe for passing variables to client side scripts: https://docs.astro.build/en/guides/client-side-scripts/#pass-frontmatter-variables-to-scripts
It would probably look something like:
in env.d.ts
:
declare namespace App {
interface Locals {
flashMessages?: string[];
}
}
to append:
Astro.locals.flashMessages.push("message");
to consume (or just read them all and wipe the array, important thing is you need to consume them not just read them):
Astro.locals.flashMessages.shift();
to manage:
export const onRequest = defineMiddleware(async (context, next) => {
context.locals.flashMessages = await context.session?.get<string[]>("flashMessages") ?? [];
const result = await next();
context.session?.set("flashMessages", context.locals.flashMessages ?? []);
return result;
}
Off topic: I've been wondering how to turn the pass-data-to-on-page-script pattern into a utility function exposed by Astro, but came to the same conclusion that users would want to do too many different/unique things to create a one size fits all approach.
My concern is related to testing fs and running the wrk benchmark tool, as I discovered that my folder became filled with a large number of new sessions, potentially making the system vulnerable to running out of disk space. Is there a way to limit the folder/disk size allocated for storing sessions? |
are there any plans for adding built-in support for session signing and verification, similar to Remix's implementation. This would help protect against session tampering and provide developers with secure defaults. Key considerations:
|
@bkyerv no, I hadn't though of that. I'm not sure of the threat model though. It's never exposed to the client, so I'm not sure where it could be tampered with. |
@wildfiremedia this doesn't seem to be an option in the unstorage fs driver. It might be worth contributing, or at least making a feature request. It supports an in-memory LRU driver, but not anything fs-backed. |
I discovered that when I try to regenerate the session, having one opened(I see the session in cookies) the session id does not change. Looking at the code, I saw that the function that generate a new id, try to obtain it first from cookie. Is this the intended behaviour or should the id be regenerated even if the session is opened? |
This is a server side session implementation, so there's no session data on the client that can be tampered with. The only thing you could sign would be the session ID itself, which some sessions implementations do (such as express-session). It's not really necessary since the key is always validated against the server side database (as long as you aren't vulnerable to timing-based attacks). |
A feature I would like to see is the ability to use custom session read queries. For KV stores, it isn't super relevant, but if you are using an general purpose DB (e.g., postgres), it's great to have the session request include a join to also get the user data (if the session has a user ID) in a single query. |
Hey everyone. Trying to use session but with a custom driver, ala https://unstorage.unjs.io/guide/custom-driver however I am getting this error:
It seems like Astro is specifying that only a string is passed, that makes it impossible to pass in a customer driver such as:
If this only a TypeScript type check, could it be relaxed to |
@robertsosinski this isn't a typing issue, it's to do with the architecture: there's no way to pass the actual storage object to the runtime. The way to do this is to make the driver the import for your custom driver, so something like: session: {
driver: "./lib/myCustomStorage",
options: {
//..options here
}
} |
Will this allow for things like per-session tabs? We currently use the old-school method of sessions at work with our point of sale, the way PHP did it in the dark days where the session ID is a part of the URL, this means our stores are able to have multiple tabs open on each terminal. In the future I would love to move to Astro, but I fear we'll lose that ability, it isn't really a problem as far as I am concerned, but pushback can be real at times, some will adapt, others will complain for the sake of complaining. |
For this use case you'd need to build something using session storage on top of the per-browser-profile sessions. If you gave every new tab its own UUID (or similar) when it is created, you could use that to key into per-tab storage in the overall session. |
Session storage I can do, I guess where my confusion lies is how to utilise the browsers session storage, I assume I would do that using the islands feature? |
if you want to dynamically render the page based on a per-tab ID, then it still has to be something sent with the page request and unique to the tab. which basically means a url param still, unfortunately. it can be a part of the root path, like rather than the session ID in the url param, it should be a secondary ID, and probably can be a number or short string. that will then be compared to the set of secondary IDs that are in the server side session. so that way the session is still secure from XSS attacks (handled with http-only secure cookies), but you can have still have per-tab identities. this should also be how you manage it with PHP too. anything else that needs to use browser storage or in-memory values would have to make a secondary request from JS in the page. if functioning without JS is important, those solutions won't work. if initial page load performance is very important for whatever is rendered based on the per-tab ID, you also wouldn't want to wait for the multiple requests required by this approach. |
Thankfully speed isn't an issue, my existing system is already blazingly fast. We also don't currently use PHP, it's just an old-school system from the company that makes the database we use. Was trying to get away from stuff in the URL as that can lead to stale sessions, but if I can't, then I guess I can't. |
You could in theory write middleware to replace the cookies with URL params and add the params to all internal links, but it would be quite a complicated thing. I don't see us ever adding it to the core. |
Summary
An
Astro.session
primitive, with pluggable storage backends.Links