mihaaai
mihaaai
Explore posts from servers
BABetter Auth
Created by mihaaai on 3/30/2025 in #bug-reports
Can't generate migrations using D1 on Cloudflare Workers
There is no way to run migrations while using D1 with Kysely on Cloudflare for 2 reasons: - The DB is exposed only at runtime, aka we cannot use the better-auth CLI to detect the configuration since it's only looking for variables exported (when using D1 we construct the configuration with each request and we only have a function) - The usage of getMigrations is still undocumented and probably abandoned since It's returning generic errors from D1 such as Error: D1_ERROR: not authorized: SQLITE_AUTH
const { getMigrations } = await import("better-auth/db");
const { auth } = await import("@/lib/auth");

const GET_HANDLER = async () => {
try {
console.log(auth().options.database.db);
const { toBeCreated, toBeAdded, compileMigrations } = await getMigrations(
auth().options,
);

if (!toBeCreated.length && !toBeAdded.length) {
return Response.json(
{ status: "success", message: "No schema changes detected" },
{ status: 200 },
);
}

const schema = await compileMigrations();

return Response.json(
{ status: "success", message: schema },
{ status: 200 },
);
} catch (err) {
console.error(err);
return new Response(null, { status: 500 });
}
};

export { GET_HANDLER as GET };
const { getMigrations } = await import("better-auth/db");
const { auth } = await import("@/lib/auth");

const GET_HANDLER = async () => {
try {
console.log(auth().options.database.db);
const { toBeCreated, toBeAdded, compileMigrations } = await getMigrations(
auth().options,
);

if (!toBeCreated.length && !toBeAdded.length) {
return Response.json(
{ status: "success", message: "No schema changes detected" },
{ status: 200 },
);
}

const schema = await compileMigrations();

return Response.json(
{ status: "success", message: schema },
{ status: 200 },
);
} catch (err) {
console.error(err);
return new Response(null, { status: 500 });
}
};

export { GET_HANDLER as GET };
This is from within a Next.js app running with Opennext on Cloudflare Workers, everything else better-auth related works like a charm, but if we can't make migration generation is useless, applying the migrations can be done with wrangler against D1, but if we cannot generate the schema according to the current state of the DB we cannot add plugins efficiently by possible knowing what tables to alter and what to add or remove. Any help would be much appreciated!
5 replies
CDCloudflare Developers
Created by mihaaai on 3/27/2025 in #workers-help
Chain multiple Workers/DOs with RPC stub forwarding across Service Bindings
I'm trying to achieve the most of the performance I can get by the ecosystem in a situation where I need to bounce a request from an introduction worker to a deeply nested Service Binding DO in my architecture: Until now I've been using HTTP across all services passing requests over the fetch() handler: Introduction Worker (HTTP) -> Worker A (HTTP) -> Worker B (HTTP) -> Worker C (HTTP) -> Worker D (HTTP) -> DO (HTTP) What I'd like achieve is switching to RPC for testing performance and speed: Introduction Worker (HTTP) -> Worker A (RPC) -> Worker B (RPC) -> Worker C (RPC) -> Worker D (RPC) -> DO (RPC) I'd also need to adopt a zero-copy strategy at step 1 (or all step if possible) and just stream it all over to Worker D where it's gonna be needed for processing for the first time. According to docs an approach of such is possible with RPC since it supports both sending the whole Request as an argument as well as a ReadableStream but I'm unsure to the level of depth I can achieve, examples are few and only with 2 levels, also for ReadableStream there are no examples. For the ReadableStream case is not clear if we're allowed to pass to a stub function an argument of { reader } and then at each nested level of depth, create a new couple of TransformStream to { write } to the received reader and pass the next { reader } deeper. Same doubts about the Forwarding RPC stubs strategy, how deep can I chain them? If anyone ever got to a situation of such can you please share your experience? (I'm aware of the 1Mb limit for RPC, let's suppose the request is less than that)
1 replies
CDCloudflare Developers
Created by mihaaai on 1/27/2025 in #workers-help
Is it possible to use Cap'n Proto, to deserialize payloads instead of JSON, in Workers requests?
I'd like to know if anyone has been using Cap'n Proto to deserialize binary incoming requests payloads to a Worker before. I'm expecting a high volume of data and I'd need to optimize workers performance to avoid deserializing (and also serializing on the emitter side) from Json as text the payload of each request. I've seen that @kenton is the author of both Cap'n Proto and Workers and that these 2 are used underneath the CF architecture, but I was wondering if it's also possible to use a standalone lib for Workers that can achieve this. Researches didn't bring much information, and also JS libraries for Cap'n Proto are very outdated.
4 replies
BABetter Auth
Created by mihaaai on 12/17/2024 in #help
With Organization plugin, `getFullOrganization` and `setActive` is always returning `null`
I'm building a Organiztion dash, from where I need to inspect organizations once created. Although I'm the owner of the admin once I create it, I cannot get the organization details from the client because both getFullOrganization and setActive return a 200 with null. I tried both with slug and id (I'd prefer to use slug) but they behave the same. From what I understood from the docs, you need to have an activeOrganization binded to your session to be able to inspect it, but what is the best way to inspect each organization without relying on active ones? I'm the "super admin" of the service, (also using the admin plugin) and I'm also the owner of each organization created by me, so I should be able to inspect any organization I need and make operations over it without restrictions
4 replies
BABetter Auth
Created by mihaaai on 12/15/2024 in #help
How to use `organizationLimit` with function? Why it returns boolean?
While using the Organization and Admin Plugins I need to make sure that admin Users can be in as many orgs as owner while each other org role (admin and member) can only be without a single org. The organizationLimit seems to be the setting I'm looking for, but I don't understand the return signature, the docs: https://www.better-auth.com/docs/plugins/organization#options say either a number or a function that return a boolean? I can use the function to check against the user but I'd still need the function to return a number, either let's say 100 for admins/owners and only 1 for any other role. But I can only return boolean from the function? what for?
8 replies
BABetter Auth
Created by mihaaai on 12/9/2024 in #help
Auth error for Cloudflare D1 Migrations over remote environments
I have a Hono app with better-auth over Cloudflare Workers that is binded to a D1 database. On the local environment everything works fine including migrations, but on remote the same endpoint I use for migrations fails with Error: D1_ERROR: not authorized: SQLITE_AUTH The migration function:
const auth = initAuth(c);

const { toBeCreated, toBeAdded, runMigrations } = await getMigrations(
auth.options
);

if (!toBeCreated.length && !toBeAdded.length) {
return c.json({ status: 403, message: "No migrations to run" }, 403);
}

await runMigrations(); // Fails here

return c.json({
status: 200,
message: "Database migration ran successfully!",
});
const auth = initAuth(c);

const { toBeCreated, toBeAdded, runMigrations } = await getMigrations(
auth.options
);

if (!toBeCreated.length && !toBeAdded.length) {
return c.json({ status: 403, message: "No migrations to run" }, 403);
}

await runMigrations(); // Fails here

return c.json({
status: 200,
message: "Database migration ran successfully!",
});
This only happens on deployed D1 databases, on the local mocked through wrangler migrations work well. It's not because of my API since I don't have any filter on the migratipn route yet, any idea why is happening and how to bypass it?
1 replies
CDCloudflare Developers
Created by mihaaai on 12/5/2024 in #workers-help
How to access Bindings inside custom functions of an RPC Worker?
I need to be able to access the env to be able to reference some vars or other CF services from the RPC worker, this worker will be called from another worker through Service Bindings
import { WorkerEntrypoint } from "cloudflare:workers";

export default class MyRPCWorker extends WorkerEntrypoint {
async fetch(req: Request, env: Env) { console.log("env as the second argument is valid in the fetch handler"); }

async customFunction(param1: string) { console.log("I need to access the env of the current RPC worker here when calling this method from another worker using Service Bindings...") }
}
import { WorkerEntrypoint } from "cloudflare:workers";

export default class MyRPCWorker extends WorkerEntrypoint {
async fetch(req: Request, env: Env) { console.log("env as the second argument is valid in the fetch handler"); }

async customFunction(param1: string) { console.log("I need to access the env of the current RPC worker here when calling this method from another worker using Service Bindings...") }
}
I didn't find anything within docs nor other discussions
2 replies
BABetter Auth
Created by mihaaai on 10/16/2024 in #bug-reports
Can't use Cloudflare D1 database with Hono backend on Cloudflare Workers
Due to the fact that the env context is only exposed to incoming requests, It can't seem to be possible to create a config for better-auth on Hono using the D1 Database from Cloudflare Workers. This example of D1 with better-auth is for the OpenNext new implementation for Workers, I'm wondering if there's a way to set it up also on Hono. I've been able to make it work from SolidStart with a similar configuration but I'm stuck on Hono, and I'd prefer to have the server backend of better-auth on a separate Hono Worker without relying on CF Functions from the full-stack app.
import { betterAuth } from "better-auth";
import { anonymous } from "better-auth/plugins";
import { Kysely } from "kysely";
import { D1Dialect } from "kysely-d1";
// Can't get a reference of D1 from here since it's outside the incoming request scope

export const auth = betterAuth({
database: {
db: new Kysely({
dialect: new D1Dialect({
database: process.env.DB as unknown as D1Database, // this results in database: undefined
}),
}),
type: "sqlite",
},
emailAndPassword: {
enabled: true,
},
plugins: [anonymous()],
});
import { betterAuth } from "better-auth";
import { anonymous } from "better-auth/plugins";
import { Kysely } from "kysely";
import { D1Dialect } from "kysely-d1";
// Can't get a reference of D1 from here since it's outside the incoming request scope

export const auth = betterAuth({
database: {
db: new Kysely({
dialect: new D1Dialect({
database: process.env.DB as unknown as D1Database, // this results in database: undefined
}),
}),
type: "sqlite",
},
emailAndPassword: {
enabled: true,
},
plugins: [anonymous()],
});
15 replies
HHono
Created by mihaaai on 10/16/2024 in #help
Is there a way to get bindings outside the `c` Context of an inbound Request in Cloudflare Workers?
I'm trying to setup the better-auth with D1 but following the guide it pops out that I need to have a valid reference of my binded D1 database outside of the inbount request context (the c.env). There are other examples of using D1 with better-auth but I can't figure out how to make it work within a Hono app deployed on Cloudflare Workers.
import { betterAuth } from "better-auth";
import { anonymous } from "better-auth/plugins";
import { Kysely } from "kysely";
import { D1Dialect } from "kysely-d1";

export const auth = betterAuth({
database: {
db: new Kysely({
dialect: new D1Dialect({
database: process.env.DB as unknown as D1Database, // this results in database: undefined
}),
}),
type: "sqlite",
},
emailAndPassword: {
enabled: true,
},
plugins: [anonymous()],
});
import { betterAuth } from "better-auth";
import { anonymous } from "better-auth/plugins";
import { Kysely } from "kysely";
import { D1Dialect } from "kysely-d1";

export const auth = betterAuth({
database: {
db: new Kysely({
dialect: new D1Dialect({
database: process.env.DB as unknown as D1Database, // this results in database: undefined
}),
}),
type: "sqlite",
},
emailAndPassword: {
enabled: true,
},
plugins: [anonymous()],
});
The configuration file is within /src/lib/auth.ts but isn't called directly nowhere within the Hono app I have a worker-configuration.d.ts in my src directory declaring the DB as a D1Database type, and the wrangler binding as well.
11 replies
SSolidJS
Created by mihaaai on 9/2/2024 in #support
How to trigger server function for SSR when transitioning routes?
Is there any way to achieve a new SSR call to the backend for each route I navigate() with useNavigate() within a [slug] pattern? I have a createAsync() in [slug].tsx that is supposed to get some data from the backend based on the route I get to, but it triggers only with a full refresh of the tab in the browser or fisrt page load, if I try to navigate() from inside the app the createAsync() function is not called anymore, it's a "use server" declared function, with a cache() wrapper, just like the examples on docs. Ideally I wouldn't use <A> components and stick to useNavigate() but I'm open to anything would make this work. Maybe this is the correct behavior of my setup but I couldn't find anything related to my use-case in the docs
7 replies
CDCloudflare Developers
Created by mihaaai on 7/16/2024 in #workers-help
Doubt about the Service Binding subrequests billing
Although the ecosystem is growing and I'm really glad about all the additions, I'm having a hard time decrypting the docs across all the services. I'm confused about the billing of subrequests made in a Service Binding context. I'm wondering if they still count as against the subrequest of the original request, If I have an API on Worker A and make a request through RPC to Worker B will this count as 2 worker requests or just one? From here I understand that it will increase the counter to 2 https://developers.cloudflare.com/workers/runtime-apis/bindings/service-bindings/#limits But from here https://developers.cloudflare.com/workers/platform/pricing/#workers I see that subrequests are not billed inbound. I remember a couple of months ago asking a similar questions and got a hint about a Twitter post where the proposal to make Service Binding requests count as only one request being in discussed inside CF, but never got any news ever since. I lost the source of the Tweet. TLDR: Will making a request from a worker to another worker through Service Bindings increase the worker requests counter to 2 or more or it will be counted only as one request to the first worker called?
1 replies
SSolidJS
Created by mihaaai on 7/12/2024 in #support
How to use `createResource` value to initialize another signal on the client?
I'm trying to fetch from my API a json that contains a link to Spotify preview (an audio) and then feed it to the audio primitive, the problem is that at render time it's undefined until it resolves the request and return the json, so the audio is undefined and I cannot play it.
const SpotifyPlayer = () => {
const [song] = createResource(fetchSong);
const [isPlaying, setIsPlaying] = createSignal<boolean>(false);
const [audio, controls] = createAudio(song().item.preview_url, isPlaying); // song() is possibly undefined, if I song()? TS stops complaining but the audio wn't play, because at the init of the component the song().item.preview_url is undefined

return <>...</>
}
const SpotifyPlayer = () => {
const [song] = createResource(fetchSong);
const [isPlaying, setIsPlaying] = createSignal<boolean>(false);
const [audio, controls] = createAudio(song().item.preview_url, isPlaying); // song() is possibly undefined, if I song()? TS stops complaining but the audio wn't play, because at the init of the component the song().item.preview_url is undefined

return <>...</>
}
How to handle such situations with Solid? Maybe with onMount? Typescript keeps complaining about the song() being possibly undefined and it's right, but how it's the correct way of using the values returned by createResources in the body of the function before it returns? For more context I'm using this within Astro in a client:only"solid-js" which means that the code is not wrapped in <Suspense> and it's executed only on the client (my API updates the last song played in Spotify each 10 mins so I need the component to also return the last song played at each refresh). If I set another client: directive it works without any problem but I'll only get the last song played during the last build of the app, insted of it being dynamic. No SSR is involved, everything is SSG
3 replies