Files
learn-languages/src/modules/auth/auth-action.ts
goddonebianu c01c94abd0 refactor: 替换服务端 console.log/error 为 winston logger
- folder-action.ts: 18处 console.log -> log.error
- auth-action.ts: 4处 console.error -> log.error
- dictionary-action/service.ts: 3处 -> log.error
- translator-action/service.ts: 3处 -> log.error
- bigmodel/translator/orchestrator.ts: console -> log.debug/info/error
- bigmodel/tts.ts: console -> log.error/warn
- bigmodel/dictionary/*.ts: console -> log.error/debug/info

客户端代码(browser、page.tsx)保留 console.error
2026-03-08 14:58:43 +08:00

183 lines
5.0 KiB
TypeScript

"use server";
import { auth } from "@/auth";
import { headers } from "next/headers";
import { redirect } from "next/navigation";
import { ValidateError } from "@/lib/errors";
import { createLogger } from "@/lib/logger";
import {
ActionInputGetUserProfileByUsername,
ActionInputSignIn,
ActionInputSignUp,
ActionOutputAuth,
ActionOutputUserProfile,
validateActionInputGetUserProfileByUsername,
validateActionInputSignIn,
validateActionInputSignUp
} from "./auth-action-dto";
import {
serviceGetUserProfileByUsername,
serviceSignIn,
serviceSignUp
} from "./auth-service";
// Re-export types for use in components
export type { ActionOutputAuth, ActionOutputUserProfile } from "./auth-action-dto";
const log = createLogger("auth-action");
/**
* Sign up action
* Creates a new user account
*/
export async function actionSignUp(prevState: ActionOutputAuth | undefined, formData: FormData): Promise<ActionOutputAuth> {
try {
// Extract form data
const rawData = {
email: formData.get("email") as string,
username: formData.get("username") as string,
password: formData.get("password") as string,
redirectTo: formData.get("redirectTo") as string | undefined,
};
// Validate input
const dto: ActionInputSignUp = validateActionInputSignUp(rawData);
// Call service layer
const result = await serviceSignUp({
email: dto.email,
username: dto.username,
password: dto.password,
name: dto.username,
});
if (!result.success) {
return {
success: false,
message: "Registration failed. Email or username may already be taken.",
};
}
// Redirect on success
redirect(dto.redirectTo || "/");
} catch (e) {
if (e instanceof Error && e.message.includes('NEXT_REDIRECT')) {
throw e;
}
if (e instanceof ValidateError) {
return {
success: false,
message: e.message,
};
}
log.error("Sign up failed", { error: e });
return {
success: false,
message: "Registration failed. Please try again later.",
};
}
}
/**
* Sign in action
* Authenticates a user
*/
export async function actionSignIn(_prevState: ActionOutputAuth | undefined, formData: FormData): Promise<ActionOutputAuth> {
try {
// Extract form data
const rawData = {
identifier: formData.get("identifier") as string,
password: formData.get("password") as string,
redirectTo: formData.get("redirectTo") as string | undefined,
};
// Validate input
const dto: ActionInputSignIn = validateActionInputSignIn(rawData);
// Call service layer
const result = await serviceSignIn({
identifier: dto.identifier,
password: dto.password,
});
if (!result.success) {
return {
success: false,
message: "Invalid email/username or password.",
errors: {
identifier: ["Invalid email/username or password"],
},
};
}
// Redirect on success
redirect(dto.redirectTo || "/");
} catch (e) {
if (e instanceof Error && e.message.includes('NEXT_REDIRECT')) {
throw e;
}
if (e instanceof ValidateError) {
return {
success: false,
message: e.message,
};
}
log.error("Sign in failed", { error: e });
return {
success: false,
message: "Sign in failed. Please check your credentials.",
};
}
}
/**
* Sign out action
* Signs out the current user
*/
export async function signOutAction() {
try {
await auth.api.signOut({
headers: await headers()
});
redirect("/login");
} catch (e) {
if (e instanceof Error && e.message.includes('NEXT_REDIRECT')) {
throw e;
}
log.error("Sign out failed", { error: e });
redirect("/login");
}
}
/**
* Get user profile by username
* Returns user profile data for display
*/
export async function actionGetUserProfileByUsername(dto: ActionInputGetUserProfileByUsername): Promise<ActionOutputUserProfile> {
try {
const userProfile = await serviceGetUserProfileByUsername(dto);
if (!userProfile) {
return {
success: false,
message: "User not found",
};
}
return {
success: true,
message: "User profile retrieved successfully",
data: userProfile,
};
} catch (e) {
log.error("Get user profile failed", { error: e });
return {
success: false,
message: "Failed to retrieve user profile",
};
}
}