认证系统

Better Auth 配置、OAuth 提供商设置、手机号登录、Magic Link、2FA 和微信登录集成指南

概览

项目使用 Better Auth 作为认证框架。Better Auth 是一个开源的 TypeScript 认证库,帮你处理用户注册、登录、会话管理等核心功能,无需从零实现。

  • 配置位置: apps/01mvp-web/src/lib/auth/auth-config.ts
  • 环境变量: apps/01mvp-web/.env.local

登录方式一览

项目支持以下登录方式,默认已开启邮箱 + 密码登录,其余方式需配置对应环境变量后才会生效。

登录方式所需环境变量默认状态
邮箱 + 密码BETTER_AUTH_SECRET, DATABASE_URL已开启
GitHub OAuthGITHUB_CLIENT_ID, GITHUB_CLIENT_SECRET未配置则隐藏
Google OAuthGOOGLE_CLIENT_ID, GOOGLE_CLIENT_SECRET未配置则隐藏
微信扫码 (PC)WECHAT_WEBSITE_APP_ID, WECHAT_WEBSITE_APP_SECRET未配置则隐藏
微信授权 (手机)WECHAT_SERVICE_ACCOUNT_APP_ID, WECHAT_SERVICE_ACCOUNT_APP_SECRET未配置则隐藏
微信小程序WECHAT_MINIPROGRAM_APP_ID, WECHAT_MINIPROGRAM_APP_SECRET未配置则隐藏
手机号 + 短信验证码SMS_PROVIDER 及对应短信服务商的密钥已开启(需配置短信服务)
Magic Link(邮件链接)邮件服务配置(EMAIL_FROM 等)已开启
用户名登录无需额外配置已开启
双因素认证 (2FA)config.auth.enableTwoFactor(代码配置)已开启

环境变量配置

基础配置

这些变量是认证系统运行的必要条件。

# 必填:Better Auth 签名密钥(至少 32 字符,用于加密 session token)
BETTER_AUTH_SECRET=your-secret-key-min-32-chars

# 可选:应用 URL(默认从请求中推断,建议显式设置)
BETTER_AUTH_URL=http://localhost:7001

# 数据库连接(Drizzle 使用 PostgreSQL)
DATABASE_URL=postgresql://user:password@localhost:5432/dbname

# 可选:信任的来源域名(逗号分隔,用于 CORS 和 CSRF 防护)
TRUSTED_ORIGINS=https://app.example.com,https://www.example.com

# 可选:跨子域名 Cookie(实现单点登录 SSO)
COOKIE_DOMAIN=.example.com

BETTER_AUTH_SECRET 在生产环境必须使用随机生成的强密钥,绝对不要使用示例中的值。可以用 openssl rand -base64 32 生成。

OAuth 提供商配置

环境变量说明用于
GITHUB_CLIENT_IDGitHub OAuth App 的 Client IDGitHub 登录
GITHUB_CLIENT_SECRETGitHub OAuth App 的 Client SecretGitHub 登录
GOOGLE_CLIENT_IDGoogle Cloud Console 的 Client IDGoogle 登录
GOOGLE_CLIENT_SECRETGoogle Cloud Console 的 Client SecretGoogle 登录

微信登录配置

环境变量说明
WECHAT_WEBSITE_APP_ID微信开放平台网站应用 AppID(PC 扫码登录)
WECHAT_WEBSITE_APP_SECRET微信开放平台网站应用 AppSecret
WECHAT_SERVICE_ACCOUNT_APP_ID微信公众平台服务号 AppID(手机端授权)
WECHAT_SERVICE_ACCOUNT_APP_SECRET微信公众平台服务号 AppSecret
WECHAT_MINIPROGRAM_APP_ID微信小程序 AppID
WECHAT_MINIPROGRAM_APP_SECRET微信小程序 AppSecret

短信服务配置

根据你选择的短信服务商,配置对应的环境变量。通过 SMS_PROVIDER 指定服务商(默认 tencent)。

环境变量说明服务商
SMS_PROVIDER短信服务商:tencent / aliyun / twilio通用
TENCENT_CLOUD_SECRET_ID腾讯云 SecretId腾讯云
TENCENT_CLOUD_SECRET_KEY腾讯云 SecretKey腾讯云
TENCENT_CLOUD_REGION腾讯云地域(默认 ap-guangzhou腾讯云
TENCENT_SMS_SDK_APP_ID腾讯云短信 SDK AppID腾讯云
TENCENT_SMS_SIGN_NAME短信签名腾讯云
TENCENT_SMS_TEMPLATE_ID短信模板 ID腾讯云
ALIYUN_ACCESS_KEY_ID阿里云 AccessKeyId阿里云
ALIYUN_ACCESS_KEY_SECRET阿里云 AccessKeySecret阿里云
ALIYUN_SMS_SIGN_NAME短信签名阿里云
ALIYUN_SMS_TEMPLATE_CODE短信模板 Code阿里云
TWILIO_ACCOUNT_SIDTwilio Account SIDTwilio
TWILIO_AUTH_TOKENTwilio Auth TokenTwilio
TWILIO_FROM_PHONE_NUMBERTwilio 发送号码Twilio

邮件服务配置

环境变量说明
EMAIL_FROM发件人邮箱地址(用于验证邮件、Magic Link 等)
RESEND_API_KEYResend Email API 密钥(推荐)
ZEABUR_EMAIL_API_KEYZeabur Email API 密钥(备选)

所有环境变量都在 apps/01mvp-web/.env.local 中配置。复制 .env.example 作为起点:cp apps/01mvp-web/.env.example apps/01mvp-web/.env.local

双因素认证 (2FA)

项目支持基于 TOTP(如 Google Authenticator)的双因素认证,通过 Better Auth 的 twoFactor 插件实现。在 config.auth.enableTwoFactor 中控制开关(默认 true)。

详细的 2FA 配置、数据库模型和客户端集成说明,见 双因素认证

管理员角色

项目的管理员系统和细粒度权限(SUPER_ADMIN / OPERATION_ADMIN)是安全基础设施的一部分。角色定义、权限列表和代码用法详见 管理员系统

Better Auth 插件一览

项目预装了以下 Better Auth 插件,开箱即用:

插件说明
admin管理员角色和权限管理
magicLink邮件魔法链接登录(无密码)
openAPI自动生成 OpenAPI 文档
twoFactor双因素认证(TOTP)
username用户名登录支持
phoneNumber手机号 + 短信验证码登录
wechatOAuth(自定义)微信 OAuth 登录(自定义插件)

在代码中使用

客户端:获取当前用户

在 React 组件中使用 useSession hook 获取当前登录状态:

"use client";
import { useSession } from "@/modules/shared/auth/hooks/use-session";

export function UserProfile() {
  const { user, session, loaded } = useSession();

  if (!loaded) return <div>加载中...</div>;
  if (!user) return <div>未登录</div>;

  return (
    <div>
      <p>用户名: {user.name}</p>
      <p>邮箱: {user.email}</p>
    </div>
  );
}

客户端:调用登录 API

使用 authClient 发起各种认证操作:

import { authClient } from "@/lib/auth/client";

// 邮箱密码登录
await authClient.signIn.email({
  email: "user@example.com",
  password: "password123",
});

// GitHub 登录(跳转到 GitHub 授权页)
await authClient.signIn.social({ provider: "github" });

// 手机号验证码登录
await authClient.phoneNumber.sendOtp({ phoneNumber: "+8613800138000" });
await authClient.phoneNumber.verify({ phoneNumber: "+8613800138000", code: "123456" });

// Magic Link 登录
await authClient.signIn.magicLink({ email: "user@example.com" });

// 登出
await authClient.signOut();

服务端:在 API 路由和 Server Components 中

使用 getSession 获取服务端会话(已自动缓存):

import { getSession } from "@/modules/shared/auth/lib/server";

// 在 Server Component 中
export default async function DashboardPage() {
  const session = await getSession();

  if (!session) {
    redirect("/auth/login");
  }

  return <div>欢迎, {session.user.name}</div>;
}

// 在 API 路由中
export async function GET() {
  const session = await getSession();

  if (!session) {
    return Response.json({ error: "Unauthorized" }, { status: 401 });
  }

  // ... 业务逻辑
}

服务端:权限检查

import { isAdmin, hasPermission, AdminPermission } from "@01mvp/auth";
import { getSession } from "@/modules/shared/auth/lib/server";

export async function GET() {
  const session = await getSession();

  if (!session || !isAdmin(session.user)) {
    return Response.json({ error: "Forbidden" }, { status: 403 });
  }

  // 管理员逻辑
}

常见问题

各登录方式接入指南

相关资源