Tiago Freitas
Tiago Freitas
CCConvex Community
Created by Ahmed on 2/9/2025 in #support-community
Convex, react flow and collaboration
did you make this work? I started working on it too and also not sure about the best approach. react flow expects arrays of nodes and edges as props and we can implement components for the individual nodes/edges but not sure if we can’t use the reactive hooks as it needs to have access to all the nodes/edges in its internal zustand store
4 replies
CCConvex Community
Created by Tiago Freitas on 2/27/2025 in #support-community
nextauth auth with convex tokens
if I don't add the debouce to forceRefreshToken I get into a loop
9 replies
CCConvex Community
Created by Tiago Freitas on 2/27/2025 in #support-community
nextauth auth with convex tokens
in my providers.tsx:
const convex = new ConvexReactClient(process.env.NEXT_PUBLIC_CONVEX_URL!);

function useAuth() {
const { data: session, status, update } = useSession();
const typedSession = session as SessionWithConvex | null;
const isLoading = status === "loading";
const isAuthenticated = !!typedSession?.user;

const lastRefreshTime = useRef(0);
const debounceMs = 1000; // 1-second debounce

const fetchAccessToken = useCallback(
async ({ forceRefreshToken }: { forceRefreshToken: boolean }) => {
const now = Date.now();
if (forceRefreshToken && now - lastRefreshTime.current < debounceMs) {
console.log("Debouncing token refresh, returning current token");
return typedSession?.convexToken || null;
}

if (forceRefreshToken) {
lastRefreshTime.current = now;
const newSession = await update();
return (newSession as SessionWithConvex)?.convexToken || null;
}
return typedSession?.convexToken || null;
},
[typedSession, update]
);

return useMemo(
() => ({
isLoading,
isAuthenticated,
fetchAccessToken,
}),
[isLoading, isAuthenticated, fetchAccessToken]
);
}

export function Providers({ children }: { children: ReactNode }) {
return (
<SessionProvider>
<ConvexProviderWithAuth client={convex} useAuth={useAuth}>
{children}
</ConvexProviderWithAuth>
</SessionProvider>
);
}
const convex = new ConvexReactClient(process.env.NEXT_PUBLIC_CONVEX_URL!);

function useAuth() {
const { data: session, status, update } = useSession();
const typedSession = session as SessionWithConvex | null;
const isLoading = status === "loading";
const isAuthenticated = !!typedSession?.user;

const lastRefreshTime = useRef(0);
const debounceMs = 1000; // 1-second debounce

const fetchAccessToken = useCallback(
async ({ forceRefreshToken }: { forceRefreshToken: boolean }) => {
const now = Date.now();
if (forceRefreshToken && now - lastRefreshTime.current < debounceMs) {
console.log("Debouncing token refresh, returning current token");
return typedSession?.convexToken || null;
}

if (forceRefreshToken) {
lastRefreshTime.current = now;
const newSession = await update();
return (newSession as SessionWithConvex)?.convexToken || null;
}
return typedSession?.convexToken || null;
},
[typedSession, update]
);

return useMemo(
() => ({
isLoading,
isAuthenticated,
fetchAccessToken,
}),
[isLoading, isAuthenticated, fetchAccessToken]
);
}

export function Providers({ children }: { children: ReactNode }) {
return (
<SessionProvider>
<ConvexProviderWithAuth client={convex} useAuth={useAuth}>
{children}
</ConvexProviderWithAuth>
</SessionProvider>
);
}
9 replies
CCConvex Community
Created by Tiago Freitas on 2/27/2025 in #support-community
nextauth auth with convex tokens
@ballingt I still have an issue with convex calliing forceRefreshToken and that forces a nextauth session update which causes a loop. I have to debounce a second to stop the loop
9 replies
CCConvex Community
Created by Tiago Freitas on 2/27/2025 in #support-community
nextauth auth with convex tokens
that helped got it to work, now just trying to solve a nextauth loop issue generating tokens on all pages. just to be sure, your guides are meant to use convex auth as a provider with users saved in convex. if I want to keep using mextauth db for users and pages auth, i should not use the convex adapter and just save the tokens in the session to send to convex
9 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
so when/if you implement Yjs sync, you don't need to use OT anymore for Tiptap? or a combination of OT and Yjs depending on the situation?
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
this sounds awesome and is just what I needed. Yjs has been a nightmare to sync with react and keep performant (as I'm using it for other data not just blocknote), and replacing it with convex for my whole app in one go sounds perfect. then I won't need a Yjs websocket server anymore right, convex will handle that replacing livekit?
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
and I guess a custom presence implementation using convex for tiptap is more complex than implementing the Yjs delta sync with convex? makes sense to support more of upstream code, just worry about complexity, OT only was sounding good if it worked
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
so that solution would do away with a yjs websocket server and be peer to peer only, with convex a persistence but using deltas right. and OT is not performant enough to do browser tab sync? but google docs is fast enough for that. because if OT is good enough for most use cases, Yjs just adds a lot of complexity. or is it that tiptap collaboration features already use Yjs directly and so a prosemirror OT does not sync everything?
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
I need to ditch blocknote because the schema is too limited for nested blocks with rules on the block types allowed. but will take some development effort, but I guess tiptap would still be the best way to do it, or is there a better alternative
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
Also can you comment on automerge vs yjs in case a crdt is needed
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
If it's performant enough without yjs for most use cases then you just need to add offline to convex itself and no need for Yjs
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
As for 2., they are already powered by yjs, so what would be the difference here? Sync yjs with convex? What is the advantage compared to just storing the Yjs doc in convex or another persistence method?
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
So it already syncs tiptap and blocknote right? A prosemirror plug-in would only add the ability of using prosemirror without tiptap. Isn't that what scroll by @RJ already does?
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
This is perfect timing
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
another reason is that tiptap and blocknote already do a lot of the work to create an extensible collaborative block document editor but use Yjs extensively, and reimplementing in pure prosemirror OT could be difficult, but I'm open to alternatives that work better with convex
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
aparently OT implementation is a lot more difficult, and harder to work offline, so most OSS systems use CRDTs
38 replies
CCConvex Community
Created by Tiago Freitas on 12/4/2024 in #support-community
CRDT for text editors
just what I read somewhere, was not aware of that. but if that is the case, why do you have the automerge article on your website, instead of using convex as the storage for prosemirror OT or something similar? is there a reference implementation / project?
38 replies
CCConvex Community
Created by Tiago Freitas on 11/29/2024 in #support-community
Convex migrations in self-hosted
I guess its because they use a local sqlite db in the browser, which is good for offline mode, that convex does not support yet, but probably has issues with multiple tabs
14 replies
CCConvex Community
Created by Tiago Freitas on 11/29/2024 in #support-community
Convex migrations in self-hosted
good catch, that is defitely a huge issue. collaboration features are important and I am currently using yjs that works but the sync with the react state manager got too complex and slow. multiple tabs may not happen much in production but are usually a way to demo that realtime data sync works correctly, so it doesn't bode well for powersync
14 replies