Embeddable magic links
Learn how to build custom magic link flows to increase user engagement and reduce drop off in transactional emails, SMSs and anywhere else you can imagine.
Overview
A "magic link" is a link, that when pressed, will automatically authenticate your user, so that they can quickly perform some action on your site with less friction than if they had to sign in manually.
Common use cases include:
- Welcome emails, when users are added off a waitlist.
- Promotional emails for users.
- Recovering abandoned carts.
- Surveys or questionnaires
Creating magic links
Embeddable magic links leverage sign-in tokens which can be created from your backend.
Create a sign in token1curl --request POST \2--url 'https://api.clerk.dev/v1/sign_in_tokens' \3-H 'Authorization: Bearer {{bapi}}' \4-H 'Content-Type: application/json' \5-d '{6"user_id": "user_28dVmJleKwFWrefTJN7skmrbtJi",7}'
This will return a token, which can then be embedded as a query param in any link, something like:
https://your-site.com/welcome?token=THE_TOKEN
Once you have the link, this can be embedded anywhere, such as an email.
Accepting magic links
Now that you have a way to process a token, you'll need to setup a page on your frontend to get the token from the query string, call the sign in function, then perform whatever action you originally had in mind.
Signing a user in with a token is very similar to all of our other custom flows.
React pseudo-code
Sign the user in1// Grab the sign in token from the query string2const queryParams = new URLSearchParams(window.location.search);3const signInToken = queryParams.get('token');45const { signIn, setSession } = useSignIn();67// Create a sign in with the ticket strategy, and the sign-in-token8const res = await signIn.create({9strategy: "ticket",10ticket: "signInToken",11});1213// Set the session as active, and then do whatever you need to!14setSession(res.createdSessionId, () => console.log("SignedIn!"));
Complete NextJS example
NextJS complete page1import { InferGetServerSidePropsType, GetServerSideProps } from "next";2import { useUser, useSignIn } from "@clerk/nextjs";3import { useEffect, useState } from "react";45// Grab the query param server side, and pass through props6export const getServerSideProps: GetServerSideProps = async (context) => {7return {8props: { signInToken: context.query.token ? context.query.token : null },9};10};1112const AcceptToken = ({13signInToken,14}: InferGetServerSidePropsType<typeof getServerSideProps>) => {15const { signIn, setSession } = useSignIn();16const { user } = useUser();17const [signInProcessed, setSignInProcessed] = useState<boolean>(false);1819useEffect(() => {20if (!signIn || !setSession || !signInToken) {21return;22}2324const aFunc = async () => {25try {26// Create a signIn with the token, note that you need to use the "ticket" strategy.27const res = await signIn.create({28strategy: "ticket",29ticket: signInToken as string,30});3132setSession(res.createdSessionId, () => {33setSignInProcessed(true);34});35} catch (err) {36setSignInProcessed(true);37}38};3940aFunc();41}, [signIn, setSession]);4243if (!signInToken) {44return <div>no token provided</div>;45}4647if (!signInProcessed) {48return <div>loading</div>;49}5051if (!user) {52return <div>error invalid token</div>;53}5455return <div>Signed in as {user.id}</div>;56};5758export default AcceptToken;59