-
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #8 from Team3256/supa-auth
- Loading branch information
Showing
58 changed files
with
2,261 additions
and
2,713 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
# Update these with your Supabase details from your project settings > API | ||
# https://app.supabase.com/project/_/settings/api | ||
NEXT_PUBLIC_SUPABASE_URL=your-project-url | ||
NEXT_PUBLIC_SUPABASE_ANON_KEY=your-anon-key |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,34 +1,93 @@ | ||
This is a [Next.js](https://nextjs.org/) project bootstrapped with [`create-next-app`](https://github.com/vercel/next.js/tree/canary/packages/create-next-app). | ||
<a href="https://demo-nextjs-with-supabase.vercel.app/"> | ||
<img alt="Next.js and Supabase Starter Kit - the fastest way to build apps with Next.js and Supabase" src="https://demo-nextjs-with-supabase.vercel.app/opengraph-image.png"> | ||
<h1 align="center">Next.js and Supabase Starter Kit</h1> | ||
</a> | ||
|
||
## Getting Started | ||
<p align="center"> | ||
The fastest way to build apps with Next.js and Supabase | ||
</p> | ||
|
||
First, run the development server: | ||
<p align="center"> | ||
<a href="#features"><strong>Features</strong></a> · | ||
<a href="#demo"><strong>Demo</strong></a> · | ||
<a href="#deploy-to-vercel"><strong>Deploy to Vercel</strong></a> · | ||
<a href="#clone-and-run-locally"><strong>Clone and run locally</strong></a> · | ||
<a href="#feedback-and-issues"><strong>Feedback and issues</strong></a> | ||
<a href="#more-supabase-examples"><strong>More Examples</strong></a> | ||
</p> | ||
<br/> | ||
|
||
```bash | ||
npm run dev | ||
# or | ||
yarn dev | ||
``` | ||
## Features | ||
|
||
Open [http://localhost:3000](http://localhost:3000) with your browser to see the result. | ||
- Works across the entire [Next.js](https://nextjs.org) stack | ||
- App Router | ||
- Pages Router | ||
- Middleware | ||
- Client | ||
- Server | ||
- It just works! | ||
- supabase-ssr. A package to configure Supabase Auth to use cookies | ||
- Styling with [Tailwind CSS](https://tailwindcss.com) | ||
- Optional deployment with [Supabase Vercel Integration and Vercel deploy](#deploy-your-own) | ||
- Environment variables automatically assigned to Vercel project | ||
|
||
You can start editing the page by modifying `pages/index.js`. The page auto-updates as you edit the file. | ||
## Demo | ||
|
||
[API routes](https://nextjs.org/docs/api-routes/introduction) can be accessed on [http://localhost:3000/api/hello](http://localhost:3000/api/hello). This endpoint can be edited in `pages/api/hello.js`. | ||
You can view a fully working demo at [demo-nextjs-with-supabase.vercel.app](https://demo-nextjs-with-supabase.vercel.app/). | ||
|
||
The `pages/api` directory is mapped to `/api/*`. Files in this directory are treated as [API routes](https://nextjs.org/docs/api-routes/introduction) instead of React pages. | ||
## Deploy to Vercel | ||
|
||
## Learn More | ||
Vercel deployment will guide you through creating a Supabase account and project. | ||
|
||
To learn more about Next.js, take a look at the following resources: | ||
After installation of the Supabase integration, all relevant environment variables will be assigned to the project so the deployment is fully functioning. | ||
|
||
- [Next.js Documentation](https://nextjs.org/docs) - learn about Next.js features and API. | ||
- [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial. | ||
[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/clone?repository-url=https%3A%2F%2Fgithub.com%2Fvercel%2Fnext.js%2Ftree%2Fcanary%2Fexamples%2Fwith-supabase&project-name=nextjs-with-supabase&repository-name=nextjs-with-supabase&demo-title=nextjs-with-supabase&demo-description=This%20starter%20configures%20Supabase%20Auth%20to%20use%20cookies%2C%20making%20the%20user's%20session%20available%20throughout%20the%20entire%20Next.js%20app%20-%20Client%20Components%2C%20Server%20Components%2C%20Route%20Handlers%2C%20Server%20Actions%20and%20Middleware.&demo-url=https%3A%2F%2Fdemo-nextjs-with-supabase.vercel.app%2F&external-id=https%3A%2F%2Fgithub.com%2Fvercel%2Fnext.js%2Ftree%2Fcanary%2Fexamples%2Fwith-supabase&demo-image=https%3A%2F%2Fdemo-nextjs-with-supabase.vercel.app%2Fopengraph-image.png&integration-ids=oac_VqOgBHqhEoFTPzGkPd7L0iH6) | ||
|
||
You can check out [the Next.js GitHub repository](https://github.com/vercel/next.js/) - your feedback and contributions are welcome! | ||
The above will also clone the Starter kit to your GitHub, you can clone that locally and develop locally. | ||
|
||
## Deploy on Vercel | ||
If you wish to just develop locally and not deploy to Vercel, [follow the steps below](#clone-and-run-locally). | ||
|
||
The easiest way to deploy your Next.js app is to use the [Vercel Platform](https://vercel.com/new?utm_medium=default-template&filter=next.js&utm_source=create-next-app&utm_campaign=create-next-app-readme) from the creators of Next.js. | ||
## Clone and run locally | ||
|
||
Check out our [Next.js deployment documentation](https://nextjs.org/docs/deployment) for more details. | ||
1. You'll first need a Supabase project which can be made [via the Supabase dashboard](https://database.new) | ||
|
||
2. Create a Next.js app using the Supabase Starter template npx command | ||
|
||
```bash | ||
npx create-next-app -e with-supabase | ||
``` | ||
|
||
3. Use `cd` to change into the app's directory | ||
|
||
```bash | ||
cd name-of-new-app | ||
``` | ||
|
||
4. Rename `.env.local.example` to `.env.local` and update the following: | ||
|
||
``` | ||
NEXT_PUBLIC_SUPABASE_URL=[INSERT SUPABASE PROJECT URL] | ||
NEXT_PUBLIC_SUPABASE_ANON_KEY=[INSERT SUPABASE PROJECT API ANON KEY] | ||
``` | ||
|
||
Both `NEXT_PUBLIC_SUPABASE_URL` and `NEXT_PUBLIC_SUPABASE_ANON_KEY` can be found in [your Supabase project's API settings](https://app.supabase.com/project/_/settings/api) | ||
|
||
5. You can now run the Next.js local development server: | ||
|
||
```bash | ||
npm run dev | ||
``` | ||
|
||
The starter kit should now be running on [localhost:3000](http://localhost:3000/). | ||
|
||
> Check out [the docs for Local Development](https://supabase.com/docs/guides/getting-started/local-development) to also run Supabase locally. | ||
## Feedback and issues | ||
|
||
Please file feedback and issues over on the [Supabase GitHub org](https://github.com/supabase/supabase/issues/new/choose). | ||
|
||
## More Supabase examples | ||
|
||
- [Next.js Subscription Payments Starter](https://github.com/vercel/nextjs-subscription-payments) | ||
- [Cookie-based Auth and the Next.js 13 App Router (free course)](https://youtube.com/playlist?list=PL5S4mPUpp4OtMhpnp93EFSo42iQ40XjbF) | ||
- [Supabase Auth and the Next.js App Router](https://github.com/supabase/supabase/tree/master/examples/auth/nextjs) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
import { createClient } from '@/utils/supabase/server' | ||
import { NextResponse } from 'next/server' | ||
import { cookies } from 'next/headers' | ||
|
||
export async function GET(request: Request) { | ||
// The `/auth/callback` route is required for the server-side auth flow implemented | ||
// by the Auth Helpers package. It exchanges an auth code for the user's session. | ||
// https://supabase.com/docs/guides/auth/auth-helpers/nextjs#managing-sign-in-with-code-exchange | ||
const requestUrl = new URL(request.url) | ||
const code = requestUrl.searchParams.get('code') | ||
|
||
if (code) { | ||
const cookieStore = cookies() | ||
const supabase = createClient(cookieStore) | ||
await supabase.auth.exchangeCodeForSession(code) | ||
} | ||
|
||
// URL to redirect to after sign in process completes | ||
return NextResponse.redirect(requestUrl.origin) | ||
} |
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
@tailwind base; | ||
@tailwind components; | ||
@tailwind utilities; | ||
|
||
@layer base { | ||
:root { | ||
--background: 200 20% 98%; | ||
--btn-background: 200 10% 91%; | ||
--btn-background-hover: 200 10% 89%; | ||
--foreground: 200 50% 3%; | ||
} | ||
|
||
@media (prefers-color-scheme: dark) { | ||
:root { | ||
--background: 200 50% 3%; | ||
--btn-background: 200 10% 9%; | ||
--btn-background-hover: 200 10% 12%; | ||
--foreground: 200 20% 96%; | ||
} | ||
} | ||
} | ||
|
||
@layer base { | ||
* { | ||
@apply border-foreground/20; | ||
} | ||
} | ||
|
||
.animate-in { | ||
animation: animateIn 0.3s ease 0.15s both; | ||
} | ||
|
||
@keyframes animateIn { | ||
from { | ||
opacity: 0; | ||
transform: translateY(10px); | ||
} | ||
to { | ||
opacity: 1; | ||
transform: translateY(0); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
import { GeistSans } from 'geist/font/sans' | ||
import './globals.css' | ||
|
||
const defaultUrl = process.env.VERCEL_URL | ||
? `https://${process.env.VERCEL_URL}` | ||
: 'http://localhost:3000' | ||
|
||
export const metadata = { | ||
metadataBase: new URL(defaultUrl), | ||
title: 'Next.js and Supabase Starter Kit', | ||
description: 'The fastest way to build apps with Next.js and Supabase', | ||
} | ||
|
||
export default function RootLayout({ | ||
children, | ||
}: { | ||
children: React.ReactNode | ||
}) { | ||
return ( | ||
<html lang="en" className={GeistSans.className}> | ||
<body className="bg-background text-foreground"> | ||
<main className="min-h-screen flex flex-col items-center"> | ||
{children} | ||
</main> | ||
</body> | ||
</html> | ||
) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,118 @@ | ||
import Link from 'next/link' | ||
import { headers, cookies } from 'next/headers' | ||
import { createClient } from '@/utils/supabase/server' | ||
import { redirect } from 'next/navigation' | ||
|
||
export default function Login({ | ||
searchParams, | ||
}: { | ||
searchParams: { message: string } | ||
}) { | ||
const signIn = async (formData: FormData) => { | ||
'use server' | ||
|
||
const email = formData.get('email') as string | ||
const password = formData.get('password') as string | ||
const cookieStore = cookies() | ||
const supabase = createClient(cookieStore) | ||
|
||
const { error } = await supabase.auth.signInWithPassword({ | ||
email, | ||
password, | ||
}) | ||
|
||
if (error) { | ||
return redirect('/login?message=Could not authenticate user') | ||
} | ||
|
||
return redirect('/') | ||
} | ||
|
||
const signUp = async (formData: FormData) => { | ||
'use server' | ||
|
||
const origin = headers().get('origin') | ||
const email = formData.get('email') as string | ||
const password = formData.get('password') as string | ||
const cookieStore = cookies() | ||
const supabase = createClient(cookieStore) | ||
|
||
const { error } = await supabase.auth.signUp({ | ||
email, | ||
password, | ||
options: { | ||
emailRedirectTo: `${origin}/auth/callback`, | ||
}, | ||
}) | ||
|
||
if (error) { | ||
return redirect('/login?message=Could not authenticate user') | ||
} | ||
|
||
return redirect('/login?message=Check email to continue sign in process') | ||
} | ||
|
||
return ( | ||
<div className="flex-1 flex flex-col w-full px-8 sm:max-w-md justify-center gap-2"> | ||
<Link | ||
href="/" | ||
className="absolute left-8 top-8 py-2 px-4 rounded-md no-underline text-foreground bg-btn-background hover:bg-btn-background-hover flex items-center group text-sm" | ||
> | ||
<svg | ||
xmlns="http://www.w3.org/2000/svg" | ||
width="24" | ||
height="24" | ||
viewBox="0 0 24 24" | ||
fill="none" | ||
stroke="currentColor" | ||
strokeWidth="2" | ||
strokeLinecap="round" | ||
strokeLinejoin="round" | ||
className="mr-2 h-4 w-4 transition-transform group-hover:-translate-x-1" | ||
> | ||
<polyline points="15 18 9 12 15 6" /> | ||
</svg>{' '} | ||
Back | ||
</Link> | ||
|
||
<form | ||
className="animate-in flex-1 flex flex-col w-full justify-center gap-2 text-foreground" | ||
action={signIn} | ||
> | ||
<label className="text-md" htmlFor="email"> | ||
</label> | ||
<input | ||
className="rounded-md px-4 py-2 bg-inherit border mb-6" | ||
name="email" | ||
placeholder="[email protected]" | ||
required | ||
/> | ||
<label className="text-md" htmlFor="password"> | ||
Password | ||
</label> | ||
<input | ||
className="rounded-md px-4 py-2 bg-inherit border mb-6" | ||
type="password" | ||
name="password" | ||
placeholder="••••••••" | ||
required | ||
/> | ||
<button className="bg-green-700 rounded-md px-4 py-2 text-foreground mb-2"> | ||
Sign In | ||
</button> | ||
<button | ||
formAction={signUp} | ||
className="border border-foreground/20 rounded-md px-4 py-2 text-foreground mb-2" | ||
> | ||
Sign Up | ||
</button> | ||
{searchParams?.message && ( | ||
<p className="mt-4 p-4 bg-foreground/10 text-foreground text-center"> | ||
{searchParams.message} | ||
</p> | ||
)} | ||
</form> | ||
</div> | ||
) | ||
} |
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Oops, something went wrong.