fix(auth): SDK'yı kaldırıp ince REST katmanına geç (lib/appwrite-rest.ts)

Sorun:
- node-appwrite v20-25 hepsi Node 26'da bozuk (node-fetch-native-with-agent polyfill)
- appwrite browser SDK Server Action context'inde 'unexpected response' veriyor
  (büyük olasılıkla browser-only global'ları kontrol ederken)

Çözüm:
- Tüm Appwrite SDK'larını sil (appwrite + node-appwrite)
- lib/appwrite-rest.ts: native fetch üzerinde ~250 satırlık ince REST wrapper
  - account: createEmailPasswordSession, get, deleteSession
  - tablesDB: listRows, getRow, createRow, updateRow, deleteRow
  - storage: listFiles, createFile, deleteFile, fileViewUrl
  - Q helpers: equal, orderAsc/Desc, limit, offset
  - AppwriteError class
- Session secret cookie tabanlı auth korundu (isletmem-kovakcrm'deki desen)
- Tüm CRUD action ve query'ler REST katmanına bağlandı

End-to-end test edildi:
✓ Login (proper 401 hata mesajları, başarılı durumda redirect)
✓ Public read (services, blog, testimonials)
✓ Anonim create (contact form)
✓ Build (24 route)
✓ Sıfır SDK bağımlılığı — Node 26 sorunu yok
This commit is contained in:
Ege Can Komur
2026-05-20 02:29:19 +03:00
parent 4096b3d87b
commit 7eb0c1acc2
18 changed files with 503 additions and 333 deletions
+37 -17
View File
@@ -2,14 +2,27 @@
import { cookies } from "next/headers";
import { redirect } from "next/navigation";
import { Account, Client } from "appwrite";
import { account, AppwriteError } from "@/lib/appwrite-rest";
import { SESSION_COOKIE } from "@/lib/auth";
const endpoint = process.env.NEXT_PUBLIC_APPWRITE_ENDPOINT!;
const projectId = process.env.NEXT_PUBLIC_APPWRITE_PROJECT_ID!;
export type LoginState = { error?: string };
function friendly(err: unknown): string {
if (err instanceof AppwriteError) {
switch (err.type) {
case "user_invalid_credentials":
return "E-posta veya şifre hatalı.";
case "user_blocked":
return "Hesabınız engellenmiş.";
case "general_rate_limit_exceeded":
return "Çok fazla deneme. Birkaç dakika sonra tekrar deneyin.";
default:
return err.message;
}
}
return err instanceof Error ? err.message : "Giriş başarısız";
}
export async function loginAction(
_prev: LoginState | undefined,
formData: FormData,
@@ -19,28 +32,35 @@ export async function loginAction(
if (!email || !password) return { error: "E-posta ve şifre zorunlu" };
let session;
try {
const client = new Client().setEndpoint(endpoint).setProject(projectId);
const account = new Account(client);
const session = await account.createEmailPasswordSession({ email, password });
const store = await cookies();
store.set(SESSION_COOKIE, session.secret, {
httpOnly: true,
sameSite: "lax",
secure: process.env.NODE_ENV === "production",
path: "/",
expires: new Date(session.expire),
});
session = await account.createEmailPasswordSession(email, password);
} catch (err) {
const msg = err instanceof Error ? err.message : "Giriş başarısız";
return { error: msg };
return { error: friendly(err) };
}
const store = await cookies();
store.set(SESSION_COOKIE, session.secret, {
httpOnly: true,
sameSite: "lax",
secure: process.env.NODE_ENV === "production",
path: "/",
expires: new Date(session.expire),
});
redirect("/admin");
}
export async function logoutAction() {
const store = await cookies();
const secret = store.get(SESSION_COOKIE)?.value;
if (secret) {
try {
await account.deleteSession("current", secret);
} catch {
// ignore — cookie is cleared anyway
}
}
store.delete(SESSION_COOKIE);
redirect("/admin/login");
}