Skip to content
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 10 additions & 0 deletions .idea/.gitignore

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

58 changes: 58 additions & 0 deletions .idea/codeStyles/Project.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

5 changes: 5 additions & 0 deletions .idea/codeStyles/codeStyleConfig.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

9 changes: 9 additions & 0 deletions .idea/custom-daily-planner.iml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions .idea/inspectionProfiles/Project_Default.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions .idea/misc.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

8 changes: 8 additions & 0 deletions .idea/modules.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions .idea/prettier.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 6 additions & 0 deletions .idea/vcs.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

72 changes: 72 additions & 0 deletions src/app/api/check-email/route.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
import { supabaseAdmin } from "@/lib/supabase/admin";
import { isValidEmail } from "@/lib/validators";
import type { CheckEmailBody, CheckEmailFailRes, CheckEmailOkRes } from "@/types/auth";
import { NextResponse } from "next/server";

export async function POST(req: Request): Promise<Response> {
let body: CheckEmailBody;

try {
body = (await req.json()) as CheckEmailBody;
} catch {
const res: CheckEmailFailRes = {
ok: false,
available: false,
message: "์š”์ฒญ ๋ณธ๋ฌธ(JSON)์ด ์˜ฌ๋ฐ”๋ฅด์ง€ ์•Š์Šต๋‹ˆ๋‹ค.",
};
return NextResponse.json(res, { status: 400 });
}

const email = (body.email ?? "").trim().toLowerCase();

if (!email) {
const res: CheckEmailFailRes = {
ok: false,
available: false,
message: "์ด๋ฉ”์ผ์ด ๋น„์–ด ์žˆ์Šต๋‹ˆ๋‹ค.",
};
return NextResponse.json(res, { status: 400 });
}

if (!isValidEmail(email)) {
const res: CheckEmailFailRes = {
ok: false,
available: false,
message: "์ด๋ฉ”์ผ ํ˜•์‹์ด ์˜ฌ๋ฐ”๋ฅด์ง€ ์•Š์Šต๋‹ˆ๋‹ค.",
};
return NextResponse.json(res, { status: 400 });
}

const { data, error } = await supabaseAdmin
.from("profiles")
.select("id")
.eq("email", email)
.limit(1);

if (error) {
const res: CheckEmailFailRes = {
ok: false,
available: false,
message: "์„œ๋ฒ„ ํ™•์ธ ์ค‘ ์˜ค๋ฅ˜๊ฐ€ ๋ฐœ์ƒํ–ˆ์Šต๋‹ˆ๋‹ค.",
};
return NextResponse.json(res, { status: 500 });
}

const exists = (data?.length ?? 0) > 0;

if (exists) {
const res: CheckEmailOkRes = {
ok: true,
available: false,
message: "์ด๋ฏธ ๊ฐ€์ž…๋œ ์ด๋ฉ”์ผ์ž…๋‹ˆ๋‹ค.",
};
return NextResponse.json(res);
}

const res: CheckEmailOkRes = {
ok: true,
available: true,
message: "์‚ฌ์šฉ ๊ฐ€๋Šฅํ•œ ์ด๋ฉ”์ผ์ž…๋‹ˆ๋‹ค.",
};
return NextResponse.json(res);
}
31 changes: 31 additions & 0 deletions src/app/api/send-otp/route.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import { createClient } from "@supabase/supabase-js";
import { NextResponse } from "next/server";

type Body = { email?: string };

export async function POST(req: Request) {
const body = (await req.json()) as Body;
const email = (body.email ?? "").trim().toLowerCase();

if (!email) {
return NextResponse.json({ ok: false, message: "์ด๋ฉ”์ผ์ด ๋น„์–ด ์žˆ์Šต๋‹ˆ๋‹ค." }, { status: 400 });
}

const url = process.env.NEXT_PUBLIC_SUPABASE_URL!;
const anon = process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!;

const supabase = createClient(url, anon, { auth: { persistSession: false } });

const { error } = await supabase.auth.signInWithOtp({
email,
options: {
shouldCreateUser: true,
},
});

if (error) {
return NextResponse.json({ ok: false, message: error.message }, { status: 400 });
}

return NextResponse.json({ ok: true, message: "์ธ์ฆ ๋ฉ”์ผ์„ ๋ฐœ์†กํ–ˆ์Šต๋‹ˆ๋‹ค." });
}
99 changes: 99 additions & 0 deletions src/app/api/set-password/route.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
import { supabaseAdmin } from "@/lib/supabase/admin";
import { isValidEmail } from "@/lib/validators";
import { createClient } from "@supabase/supabase-js";
import { NextResponse } from "next/server";

export type SetPasswordBody = {
email?: string;
password?: string;
};

export type SetPasswordOkRes = {
ok: true;
message: string;
};

export type SetPasswordFailRes = {
ok: false;
message: string;
};

export type SetPasswordResponse = SetPasswordOkRes | SetPasswordFailRes;

const getBearerToken = (req: Request): string => {
const value = req.headers.get("authorization") ?? "";
if (!value.startsWith("Bearer ")) return "";
return value.slice("Bearer ".length).trim();
};

export async function POST(req: Request): Promise<Response> {
let body: SetPasswordBody;

try {
body = (await req.json()) as SetPasswordBody;
} catch {
const res: SetPasswordFailRes = {
ok: false,
message: "์š”์ฒญ ๋ณธ๋ฌธ(`JSON`)์ด ์˜ฌ๋ฐ”๋ฅด์ง€ ์•Š์Šต๋‹ˆ๋‹ค.",
};
return NextResponse.json(res, { status: 400 });
}

const accessToken = getBearerToken(req);
const email = (body.email ?? "").trim().toLowerCase();
const password = (body.password ?? "").trim();

if (!accessToken) {
const res: SetPasswordFailRes = { ok: false, message: "์ธ์ฆ ํ† ํฐ์ด ์—†์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res, { status: 401 });
}

if (!email) {
const res: SetPasswordFailRes = { ok: false, message: "์ด๋ฉ”์ผ์ด ๋น„์–ด ์žˆ์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res, { status: 400 });
}

if (!isValidEmail(email)) {
const res: SetPasswordFailRes = { ok: false, message: "์ด๋ฉ”์ผ ํ˜•์‹์ด ์˜ฌ๋ฐ”๋ฅด์ง€ ์•Š์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res, { status: 400 });
}

if (!password) {
const res: SetPasswordFailRes = { ok: false, message: "๋น„๋ฐ€๋ฒˆํ˜ธ๊ฐ€ ๋น„์–ด ์žˆ์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res, { status: 400 });
}

// โœ… 1) ํ† ํฐ์œผ๋กœ โ€œ๋ˆ„๊ตฌ์ธ์ง€โ€ ํ™•์ธ
const url = process.env.NEXT_PUBLIC_SUPABASE_URL!;
const anon = process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!;
const supabase = createClient(url, anon, { auth: { persistSession: false } });

const { data: userData, error: userError } = await supabase.auth.getUser(accessToken);

if (userError || !userData.user) {
const res: SetPasswordFailRes = { ok: false, message: "์ธ์ฆ ์ •๋ณด๊ฐ€ ์œ ํšจํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res, { status: 401 });
}

const authedEmail = (userData.user.email ?? "").toLowerCase();
if (authedEmail !== email) {
const res: SetPasswordFailRes = {
ok: false,
message: "์ธ์ฆ๋œ ์ด๋ฉ”์ผ๊ณผ ์š”์ฒญ ์ด๋ฉ”์ผ์ด ์ผ์น˜ํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค.",
};
return NextResponse.json(res, { status: 403 });
}

// โœ… 2) `Admin` ๊ถŒํ•œ์œผ๋กœ ๋น„๋ฐ€๋ฒˆํ˜ธ ์„ค์ •
const { error: updateError } = await supabaseAdmin.auth.admin.updateUserById(userData.user.id, {
password,
});

if (updateError) {
const res: SetPasswordFailRes = { ok: false, message: "๋น„๋ฐ€๋ฒˆํ˜ธ ์ €์žฅ ์ค‘ ์˜ค๋ฅ˜๊ฐ€ ๋ฐœ์ƒํ–ˆ์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res, { status: 500 });
}

const res: SetPasswordOkRes = { ok: true, message: "๋น„๋ฐ€๋ฒˆํ˜ธ๊ฐ€ ์„ค์ •๋˜์—ˆ์Šต๋‹ˆ๋‹ค." };
return NextResponse.json(res);
}
Loading
Loading