From e8a1fc9a40947b949080107138eade09f06f5868 Mon Sep 17 00:00:00 2001 From: Kang Ming Date: Wed, 6 Mar 2024 15:10:13 +0800 Subject: [PATCH] feat: add method for anonymous sign-in (#858) ## What kind of change does this PR introduce? * https://github.com/supabase/gotrue/issues/68 * Adds the `signInAnonymously` method to support https://github.com/supabase/gotrue/pull/1460 * Passing in user metadata / the captcha token in `signInAnonymously` looks like: `signInAnonymously({ options: { data, captchaToken }})` which follows the same format as the other sign up / sign in methods --- example/react/package.json | 3 +++ example/react/src/App.js | 19 ++++++++++++++++--- src/GoTrueClient.ts | 37 ++++++++++++++++++++++++++++++++++++- src/lib/types.ts | 27 ++++++++++++++++++++------- 4 files changed, 75 insertions(+), 11 deletions(-) diff --git a/example/react/package.json b/example/react/package.json index 9d7398d5a..3ecd47ac3 100644 --- a/example/react/package.json +++ b/example/react/package.json @@ -12,6 +12,9 @@ "react-scripts": "5.0.1", "tailwindcss": "^1.8.10" }, + "devDependencies": { + "@babel/plugin-proposal-private-property-in-object": "^7.21.11" + }, "scripts": { "build:tailwind": "tailwindcss build src/App.css -o src/tailwind.output.css", "prestart": "npm run build:tailwind", diff --git a/example/react/src/App.js b/example/react/src/App.js index 7353a3bf2..ebc4c656f 100644 --- a/example/react/src/App.js +++ b/example/react/src/App.js @@ -41,13 +41,11 @@ function App() { console.log(event, session) if (event === 'SIGNED_OUT' || event === 'SIGNED_IN') { setSession(session) - window.location.reload() } }) return () => { - if (subscription) { - console.log('Subscription: ', subscription) + if (subscription.subscription) { subscription.unsubscribe() } } @@ -94,6 +92,10 @@ function App() { let { error } = await auth.signOut() if (error) console.log('Error: ', error) } + async function handleSignInAnonymously(data) { + let { error } = await auth.signInAnonymously({ options: { data } }) + if (error) alert(error.message) + } async function forgotPassword() { var email = prompt('Please enter your email:') if (email === null || email === '') { @@ -393,6 +395,17 @@ function App() { +
+ + + +
diff --git a/src/GoTrueClient.ts b/src/GoTrueClient.ts index 4702ace87..88b00bf42 100644 --- a/src/GoTrueClient.ts +++ b/src/GoTrueClient.ts @@ -87,7 +87,7 @@ import type { AuthFlowType, LockFunc, UserIdentity, - WeakPassword, + SignInAnonymouslyCredentials, } from './lib/types' polyfillGlobalThis() // Make "globalThis" available @@ -361,6 +361,41 @@ export default class GoTrueClient { } } + async signInAnonymously(credentials?: SignInAnonymouslyCredentials): Promise { + try { + await this._removeSession() + + const res = await _request(this.fetch, 'POST', `${this.url}/signup`, { + headers: this.headers, + body: { + data: credentials?.options?.data ?? {}, + gotrue_meta_security: { captcha_token: credentials?.options?.captchaToken }, + }, + xform: _sessionResponse, + }) + const { data, error } = res + + if (error || !data) { + return { data: { user: null, session: null }, error: error } + } + const session: Session | null = data.session + const user: User | null = data.user + + if (data.session) { + await this._saveSession(data.session) + await this._notifyAllSubscribers('SIGNED_IN', session) + } + + return { data: { user, session }, error: null } + } catch (error) { + if (isAuthError(error)) { + return { data: { user: null, session: null }, error } + } + + throw error + } + } + /** * Creates a new user. * diff --git a/src/lib/types.ts b/src/lib/types.ts index 38e7440c4..180b3e1f3 100644 --- a/src/lib/types.ts +++ b/src/lib/types.ts @@ -366,7 +366,7 @@ export interface UserAttributes { nonce?: string /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * The `data` should be a JSON object that includes user-specific info, such as their first and last name. * @@ -376,7 +376,7 @@ export interface UserAttributes { export interface AdminUserAttributes extends Omit { /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * * The `user_metadata` should be a JSON object that includes user-specific info, such as their first and last name. @@ -452,6 +452,19 @@ export interface UpdatableFactorAttributes { friendlyName: string } +export type SignInAnonymouslyCredentials = { + options?: { + /** + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. + * + * The `data` should be a JSON object that includes user-specific info, such as their first and last name. + */ + data?: object + /** Verification token received when the user completes the captcha on the site. */ + captchaToken?: string + } +} + export type SignUpWithPasswordCredentials = | { /** The user's email address. */ @@ -462,7 +475,7 @@ export type SignUpWithPasswordCredentials = /** The redirect url embedded in the email link */ emailRedirectTo?: string /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * The `data` should be a JSON object that includes user-specific info, such as their first and last name. */ @@ -478,7 +491,7 @@ export type SignUpWithPasswordCredentials = password: string options?: { /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * The `data` should be a JSON object that includes user-specific info, such as their first and last name. */ @@ -521,7 +534,7 @@ export type SignInWithPasswordlessCredentials = /** If set to false, this method will not create a new user. Defaults to true. */ shouldCreateUser?: boolean /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * The `data` should be a JSON object that includes user-specific info, such as their first and last name. */ @@ -537,7 +550,7 @@ export type SignInWithPasswordlessCredentials = /** If set to false, this method will not create a new user. Defaults to true. */ shouldCreateUser?: boolean /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * The `data` should be a JSON object that includes user-specific info, such as their first and last name. */ @@ -708,7 +721,7 @@ export type GenerateEmailChangeLinkParams = { export interface GenerateLinkOptions { /** - * A custom data object to store the user's metadata. This maps to the `auth.users.user_metadata` column. + * A custom data object to store the user's metadata. This maps to the `auth.users.raw_user_meta_data` column. * * The `data` should be a JSON object that includes user-specific info, such as their first and last name. */