mirror of
https://github.com/ZeroCatDev/ClassworksKV.git
synced 2025-10-24 03:13:12 +00:00
Compare commits
6 Commits
d9a610ed71
...
60c52caeca
Author | SHA1 | Date | |
---|---|---|---|
![]() |
60c52caeca | ||
![]() |
da77018509 | ||
![]() |
f8f63d1ddd | ||
![]() |
5435814cb6 | ||
![]() |
d5fc9e2dae | ||
![]() |
6c05d71506 |
29
README.md
29
README.md
@ -1,4 +1,5 @@
|
|||||||
# Classworks KV
|
# Classworks KV
|
||||||
|
|
||||||
[Classworks](https://cs.houlangs.com)用于班级大屏的作业板小工具
|
[Classworks](https://cs.houlangs.com)用于班级大屏的作业板小工具
|
||||||
|
|
||||||
|
|
||||||
@ -20,3 +21,31 @@ This project is licensed under the **GNU AGPL v3.0**.
|
|||||||
Copyright (C) 2025 **Sunwuyuan** (<https://wuyuan.dev>)
|
Copyright (C) 2025 **Sunwuyuan** (<https://wuyuan.dev>)
|
||||||
See [LICENSE](./LICENSE) for details.
|
See [LICENSE](./LICENSE) for details.
|
||||||
|
|
||||||
|
## 配置(OAuth / JWT)
|
||||||
|
|
||||||
|
在根目录创建或编辑 `.env`:
|
||||||
|
|
||||||
|
- 基础地址(用于回调):
|
||||||
|
- `BASE_URL`: `http://localhost:3030`
|
||||||
|
- `FRONTEND_URL`: `http://localhost:5173`
|
||||||
|
|
||||||
|
- STCN(Casdoor)OIDC:
|
||||||
|
- `STCN_CLIENT_ID`: `53e65cfd81232e729730`
|
||||||
|
- `STCN_CLIENT_SECRET`: `e1b1277f8906e5df162b1d2f2eb3692182dd2920`
|
||||||
|
- 回调地址:`${BASE_URL}/accounts/oauth/stcn/callback`
|
||||||
|
|
||||||
|
- 其他可选提供者:GitHub、ZeroCat、厚浪云(Logto)
|
||||||
|
|
||||||
|
- JWT:
|
||||||
|
- 默认 HS256(提供 `JWT_SECRET`)
|
||||||
|
- 如需 RS256,请设置:
|
||||||
|
- `JWT_ALG=RS256`
|
||||||
|
- `JWT_PRIVATE_KEY`(PEM,\n 转义)
|
||||||
|
- `JWT_PUBLIC_KEY`(PEM,\n 转义)
|
||||||
|
- `JWT_EXPIRES_IN=7d`
|
||||||
|
|
||||||
|
完成后启动服务并访问:
|
||||||
|
|
||||||
|
- GET /accounts/oauth/providers 列出可用登录方式
|
||||||
|
- 浏览器打开 /accounts/oauth/stcn 发起 STCN 登录
|
||||||
|
|
||||||
|
@ -7,10 +7,13 @@ export const oauthProviders = {
|
|||||||
tokenURL: "https://github.com/login/oauth/access_token",
|
tokenURL: "https://github.com/login/oauth/access_token",
|
||||||
userInfoURL: "https://api.github.com/user",
|
userInfoURL: "https://api.github.com/user",
|
||||||
scope: "read:user user:email",
|
scope: "read:user user:email",
|
||||||
|
// 展示相关
|
||||||
name: "GitHub",
|
name: "GitHub",
|
||||||
|
displayName: "GitHub",
|
||||||
icon: "github",
|
icon: "github",
|
||||||
color: "#24292e",
|
color: "#24292e",
|
||||||
description: "使用 GitHub 账号登录",
|
description: "使用 GitHub 账号登录",
|
||||||
|
website: "https://github.com",
|
||||||
},
|
},
|
||||||
zerocat: {
|
zerocat: {
|
||||||
clientId: process.env.ZEROCAT_CLIENT_ID,
|
clientId: process.env.ZEROCAT_CLIENT_ID,
|
||||||
@ -19,23 +22,49 @@ export const oauthProviders = {
|
|||||||
tokenURL: "https://zerocat-api.houlangs.com/oauth/token",
|
tokenURL: "https://zerocat-api.houlangs.com/oauth/token",
|
||||||
userInfoURL: "https://zerocat-api.houlangs.com/oauth/userinfo",
|
userInfoURL: "https://zerocat-api.houlangs.com/oauth/userinfo",
|
||||||
scope: "user:basic user:email",
|
scope: "user:basic user:email",
|
||||||
|
// 展示相关
|
||||||
name: "ZeroCat",
|
name: "ZeroCat",
|
||||||
|
displayName: "ZeroCat",
|
||||||
icon: "zerocat",
|
icon: "zerocat",
|
||||||
color: "#6366f1",
|
color: "#415f91",
|
||||||
description: "使用 ZeroCat 账号登录",
|
description: "使用 ZeroCat 账号登录",
|
||||||
|
website: "https://zerocat.dev",
|
||||||
|
},
|
||||||
|
stcn: {
|
||||||
|
// STCN(Casdoor)- 标准 OIDC Provider
|
||||||
|
clientId: process.env.STCN_CLIENT_ID,
|
||||||
|
clientSecret: process.env.STCN_CLIENT_SECRET,
|
||||||
|
// Casdoor 标准端点
|
||||||
|
authorizationURL: "https://auth.smart-teach.cn/login/oauth/authorize",
|
||||||
|
tokenURL: "https://auth.smart-teach.cn/api/login/oauth/access_token",
|
||||||
|
userInfoURL: "https://auth.smart-teach.cn/api/userinfo",
|
||||||
|
scope: "openid profile email offline_access",
|
||||||
|
// 展示相关
|
||||||
|
name: "stcn",
|
||||||
|
displayName: "智教联盟账户",
|
||||||
|
icon: "casdoor",
|
||||||
|
color: "#1068af",
|
||||||
|
description: "使用智教联盟账户登录",
|
||||||
|
website: "https://auth.smart-teach.cn",
|
||||||
|
tokenRequestFormat: "json", // Casdoor 推荐 JSON 提交
|
||||||
},
|
},
|
||||||
hly: {
|
hly: {
|
||||||
// 厚浪云(Logto) - OIDC Provider
|
// 厚浪云(Logto) - OIDC Provider
|
||||||
clientId: process.env.HLY_CLIENT_ID,
|
clientId: process.env.HLY_CLIENT_ID,
|
||||||
clientSecret: process.env.HLY_CLIENT_SECRET, // 可选:若使用PKCE且应用为Public,可不配置
|
clientSecret: process.env.HLY_CLIENT_SECRET,
|
||||||
authorizationURL: "https://oauth.houlang.cloud/oidc/auth",
|
authorizationURL: "https://oauth.houlang.cloud/oidc/auth",
|
||||||
tokenURL: "https://oauth.houlang.cloud/oidc/token",
|
tokenURL: "https://oauth.houlang.cloud/oidc/token",
|
||||||
userInfoURL: "https://oauth.houlang.cloud/oidc/me",
|
userInfoURL: "https://oauth.houlang.cloud/oidc/me",
|
||||||
scope: "openid profile email offline_access",
|
scope: "openid profile email offline_access",
|
||||||
|
// 展示相关
|
||||||
name: "厚浪云",
|
name: "厚浪云",
|
||||||
|
displayName: "厚浪云",
|
||||||
icon: "logto",
|
icon: "logto",
|
||||||
color: "#0ea5e9",
|
color: "#2d53f8",
|
||||||
|
textColor: "#ffffff",
|
||||||
|
order: 40,
|
||||||
description: "使用厚浪云账号登录",
|
description: "使用厚浪云账号登录",
|
||||||
|
website: "https://houlang.cloud",
|
||||||
pkce: true, // 启用PKCE支持
|
pkce: true, // 启用PKCE支持
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "ClassworksKV",
|
"name": "ClassworksKV",
|
||||||
"version": "1.0.9",
|
"version": "1.1.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node ./bin/www",
|
"start": "node ./bin/www",
|
||||||
@ -16,7 +16,7 @@
|
|||||||
"@opentelemetry/sdk-node": "^0.201.1",
|
"@opentelemetry/sdk-node": "^0.201.1",
|
||||||
"@opentelemetry/sdk-trace-base": "^2.0.1",
|
"@opentelemetry/sdk-trace-base": "^2.0.1",
|
||||||
"@opentelemetry/semantic-conventions": "^1.34.0",
|
"@opentelemetry/semantic-conventions": "^1.34.0",
|
||||||
"@prisma/client": "6.16.2",
|
"@prisma/client": "6.16.3",
|
||||||
"axios": "^1.9.0",
|
"axios": "^1.9.0",
|
||||||
"bcrypt": "^6.0.0",
|
"bcrypt": "^6.0.0",
|
||||||
"body-parser": "^2.2.0",
|
"body-parser": "^2.2.0",
|
||||||
|
1958
pnpm-lock.yaml
generated
1958
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,2 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE `Account` MODIFY `refreshToken` TEXT NULL;
|
5
prisma/migrations/20251007041015_update/migration.sql
Normal file
5
prisma/migrations/20251007041015_update/migration.sql
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
-- DropIndex
|
||||||
|
DROP INDEX `Account_accessToken_key` ON `Account`;
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE `Account` MODIFY `accessToken` TEXT NOT NULL;
|
@ -29,8 +29,8 @@ model Account {
|
|||||||
name String? // 用户名称
|
name String? // 用户名称
|
||||||
avatarUrl String? // 用户头像URL
|
avatarUrl String? // 用户头像URL
|
||||||
providerData Json? // OAuth提供者返回的完整信息
|
providerData Json? // OAuth提供者返回的完整信息
|
||||||
accessToken String @unique // 账户访问令牌
|
accessToken String @db.Text // 账户访问令牌
|
||||||
refreshToken String? // OAuth refresh token (如果提供者支持)
|
refreshToken String? @db.Text // OAuth refresh token (如果提供者支持) - 可能很长,使用 TEXT
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
@ -41,7 +41,7 @@ function generateAccessToken() {
|
|||||||
* GET /accounts/oauth/providers
|
* GET /accounts/oauth/providers
|
||||||
*/
|
*/
|
||||||
router.get("/oauth/providers", (req, res) => {
|
router.get("/oauth/providers", (req, res) => {
|
||||||
const providers = [];
|
let providers = [];
|
||||||
|
|
||||||
for (const [key, config] of Object.entries(oauthProviders)) {
|
for (const [key, config] of Object.entries(oauthProviders)) {
|
||||||
// 只返回已配置的提供者
|
// 只返回已配置的提供者
|
||||||
@ -50,14 +50,22 @@ router.get("/oauth/providers", (req, res) => {
|
|||||||
providers.push({
|
providers.push({
|
||||||
id: key,
|
id: key,
|
||||||
name: config.name,
|
name: config.name,
|
||||||
|
displayName: config.displayName || config.name,
|
||||||
icon: config.icon,
|
icon: config.icon,
|
||||||
color: config.color,
|
color: config.color, // 向后兼容
|
||||||
|
brandColor: config.brandColor || config.color,
|
||||||
|
textColor: config.textColor || "#ffffff",
|
||||||
description: config.description,
|
description: config.description,
|
||||||
|
order: typeof config.order === 'number' ? config.order : 9999,
|
||||||
authUrl: `/accounts/oauth/${key}`, // 前端用于发起认证的URL
|
authUrl: `/accounts/oauth/${key}`, // 前端用于发起认证的URL
|
||||||
|
website: config.website,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 按 order 排序(从小到大)
|
||||||
|
providers = providers.sort((a, b) => a.order - b.order);
|
||||||
|
|
||||||
res.json({
|
res.json({
|
||||||
success: true,
|
success: true,
|
||||||
data: providers,
|
data: providers,
|
||||||
@ -179,7 +187,26 @@ router.get("/oauth/:provider/callback", async (req, res) => {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
// 1. 使用授权码换取访问令牌
|
// 1. 使用授权码换取访问令牌
|
||||||
const tokenResponse = await fetch(providerConfig.tokenURL, {
|
let tokenResponse;
|
||||||
|
if (providerConfig.tokenRequestFormat === 'json') {
|
||||||
|
tokenResponse = await fetch(providerConfig.tokenURL, {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Accept": "application/json",
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
client_id: providerConfig.clientId,
|
||||||
|
...(providerConfig.clientSecret ? { client_secret: providerConfig.clientSecret } : {}),
|
||||||
|
code: code,
|
||||||
|
grant_type: "authorization_code",
|
||||||
|
redirect_uri: getCallbackURL(provider),
|
||||||
|
// PKCE: 携带code_verifier
|
||||||
|
...(stateData?.codeVerifier ? { code_verifier: stateData.codeVerifier } : {}),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
tokenResponse = await fetch(providerConfig.tokenURL, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
"Accept": "application/json",
|
"Accept": "application/json",
|
||||||
@ -195,6 +222,7 @@ router.get("/oauth/:provider/callback", async (req, res) => {
|
|||||||
...(stateData?.codeVerifier ? { code_verifier: stateData.codeVerifier } : {}),
|
...(stateData?.codeVerifier ? { code_verifier: stateData.codeVerifier } : {}),
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const tokenData = await tokenResponse.json();
|
const tokenData = await tokenResponse.json();
|
||||||
|
|
||||||
@ -203,12 +231,20 @@ router.get("/oauth/:provider/callback", async (req, res) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 2. 使用访问令牌获取用户信息
|
// 2. 使用访问令牌获取用户信息
|
||||||
const userResponse = await fetch(providerConfig.userInfoURL, {
|
let userResponse;
|
||||||
|
// Casdoor 支持两种方式:Authorization Bearer 或 accessToken 查询参数
|
||||||
|
if (provider === 'stcn') {
|
||||||
|
const url = new URL(providerConfig.userInfoURL);
|
||||||
|
url.searchParams.set('accessToken', tokenData.access_token);
|
||||||
|
userResponse = await fetch(url, { headers: { "Accept": "application/json" } });
|
||||||
|
} else {
|
||||||
|
userResponse = await fetch(providerConfig.userInfoURL, {
|
||||||
headers: {
|
headers: {
|
||||||
"Authorization": `Bearer ${tokenData.access_token}`,
|
"Authorization": `Bearer ${tokenData.access_token}`,
|
||||||
"Accept": "application/json",
|
"Accept": "application/json",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const userData = await userResponse.json();
|
const userData = await userResponse.json();
|
||||||
|
|
||||||
@ -237,6 +273,14 @@ router.get("/oauth/:provider/callback", async (req, res) => {
|
|||||||
name: userData.name || userData.preferred_username || userData.nickname,
|
name: userData.name || userData.preferred_username || userData.nickname,
|
||||||
avatarUrl: userData.picture,
|
avatarUrl: userData.picture,
|
||||||
};
|
};
|
||||||
|
} else if (provider === "stcn") {
|
||||||
|
// STCN(Casdoor)标准OIDC用户信息
|
||||||
|
normalizedUser = {
|
||||||
|
providerId: userData.sub,
|
||||||
|
email: userData.email_verified ? userData.email : userData.email || null,
|
||||||
|
name: userData.name || userData.preferred_username || userData.nickname,
|
||||||
|
avatarUrl: userData.picture,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// 名称为空时,用邮箱@前部分回填(若邮箱可用)
|
// 名称为空时,用邮箱@前部分回填(若邮箱可用)
|
||||||
@ -295,6 +339,12 @@ router.get("/oauth/:provider/callback", async (req, res) => {
|
|||||||
const callbackUrl = new URL(frontendBaseUrl);
|
const callbackUrl = new URL(frontendBaseUrl);
|
||||||
callbackUrl.searchParams.append("token", jwtToken);
|
callbackUrl.searchParams.append("token", jwtToken);
|
||||||
callbackUrl.searchParams.append("provider", provider);
|
callbackUrl.searchParams.append("provider", provider);
|
||||||
|
// 附带展示信息,便于前端显示品牌与名称
|
||||||
|
const pconf = oauthProviders[provider] || {};
|
||||||
|
callbackUrl.searchParams.append("providerName", pconf.displayName || pconf.name || provider);
|
||||||
|
if (pconf.brandColor || pconf.color) {
|
||||||
|
callbackUrl.searchParams.append("providerColor", pconf.brandColor || pconf.color);
|
||||||
|
}
|
||||||
callbackUrl.searchParams.append("success", "true");
|
callbackUrl.searchParams.append("success", "true");
|
||||||
|
|
||||||
res.redirect(callbackUrl.toString());
|
res.redirect(callbackUrl.toString());
|
||||||
@ -338,11 +388,27 @@ router.get("/profile", jwtAuth, async (req, res, next) => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 组装 provider 展示信息
|
||||||
|
const pconf = (account?.provider && oauthProviders[account.provider]) || {};
|
||||||
|
const providerInfo = {
|
||||||
|
id: account?.provider || undefined,
|
||||||
|
name: pconf.name,
|
||||||
|
displayName: pconf.displayName || pconf.name || account?.provider,
|
||||||
|
icon: pconf.icon,
|
||||||
|
color: pconf.color, // 兼容字段
|
||||||
|
brandColor: pconf.brandColor || pconf.color,
|
||||||
|
textColor: pconf.textColor || "#ffffff",
|
||||||
|
description: pconf.description,
|
||||||
|
order: typeof pconf.order === 'number' ? pconf.order : undefined,
|
||||||
|
website: pconf.website,
|
||||||
|
};
|
||||||
|
|
||||||
res.json({
|
res.json({
|
||||||
success: true,
|
success: true,
|
||||||
data: {
|
data: {
|
||||||
id: account.id,
|
id: account.id,
|
||||||
provider: account.provider,
|
provider: account.provider,
|
||||||
|
providerInfo,
|
||||||
email: account.email,
|
email: account.email,
|
||||||
name: account.name,
|
name: account.name,
|
||||||
avatarUrl: account.avatarUrl,
|
avatarUrl: account.avatarUrl,
|
||||||
|
@ -140,7 +140,6 @@ router.post(
|
|||||||
errors.catchAsync(async (req, res, next) => {
|
errors.catchAsync(async (req, res, next) => {
|
||||||
const { uuid } = req.params;
|
const { uuid } = req.params;
|
||||||
const newPassword = req.query.newPassword || req.body.newPassword;
|
const newPassword = req.query.newPassword || req.body.newPassword;
|
||||||
const passwordHint = req.query.passwordHint || req.body.passwordHint;
|
|
||||||
|
|
||||||
if (!newPassword) {
|
if (!newPassword) {
|
||||||
return next(errors.createError(400, "新密码是必需的"));
|
return next(errors.createError(400, "新密码是必需的"));
|
||||||
@ -166,7 +165,6 @@ router.post(
|
|||||||
where: { id: device.id },
|
where: { id: device.id },
|
||||||
data: {
|
data: {
|
||||||
password: hashedPassword,
|
password: hashedPassword,
|
||||||
passwordHint: passwordHint || null,
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
35
utils/jwt.js
35
utils/jwt.js
@ -1,33 +1,48 @@
|
|||||||
import jwt from 'jsonwebtoken';
|
import jwt from 'jsonwebtoken';
|
||||||
|
|
||||||
// JWT密钥 - 生产环境应该从环境变量读取
|
// JWT 配置(支持 HS256 与 RS256)
|
||||||
|
const JWT_ALG = (process.env.JWT_ALG || 'HS256').toUpperCase();
|
||||||
|
const JWT_EXPIRES_IN = process.env.JWT_EXPIRES_IN || '7d';
|
||||||
|
|
||||||
|
// HS256 密钥
|
||||||
const JWT_SECRET = process.env.JWT_SECRET || 'your-secret-key-change-this-in-production';
|
const JWT_SECRET = process.env.JWT_SECRET || 'your-secret-key-change-this-in-production';
|
||||||
const JWT_EXPIRES_IN = process.env.JWT_EXPIRES_IN || '7d'; // 默认7天过期
|
|
||||||
|
// RS256 密钥对(PEM 格式字符串)
|
||||||
|
const JWT_PRIVATE_KEY = process.env.JWT_PRIVATE_KEY?.replace(/\\n/g, '\n');
|
||||||
|
const JWT_PUBLIC_KEY = process.env.JWT_PUBLIC_KEY?.replace(/\\n/g, '\n');
|
||||||
|
|
||||||
|
function getSignVerifyKeys() {
|
||||||
|
if (JWT_ALG === 'RS256') {
|
||||||
|
if (!JWT_PRIVATE_KEY || !JWT_PUBLIC_KEY) {
|
||||||
|
throw new Error('RS256 需要同时提供 JWT_PRIVATE_KEY 与 JWT_PUBLIC_KEY');
|
||||||
|
}
|
||||||
|
return { signKey: JWT_PRIVATE_KEY, verifyKey: JWT_PUBLIC_KEY };
|
||||||
|
}
|
||||||
|
// 默认 HS256
|
||||||
|
return { signKey: JWT_SECRET, verifyKey: JWT_SECRET };
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 签发JWT token
|
* 签发JWT token
|
||||||
* @param {Object} payload - 要编码的数据
|
|
||||||
* @returns {string} JWT token
|
|
||||||
*/
|
*/
|
||||||
export function signToken(payload) {
|
export function signToken(payload) {
|
||||||
return jwt.sign(payload, JWT_SECRET, {
|
const { signKey } = getSignVerifyKeys();
|
||||||
|
return jwt.sign(payload, signKey, {
|
||||||
expiresIn: JWT_EXPIRES_IN,
|
expiresIn: JWT_EXPIRES_IN,
|
||||||
|
algorithm: JWT_ALG,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 验证JWT token
|
* 验证JWT token
|
||||||
* @param {string} token - JWT token
|
|
||||||
* @returns {Object} 解码后的payload
|
|
||||||
*/
|
*/
|
||||||
export function verifyToken(token) {
|
export function verifyToken(token) {
|
||||||
return jwt.verify(token, JWT_SECRET);
|
const { verifyKey } = getSignVerifyKeys();
|
||||||
|
return jwt.verify(token, verifyKey, { algorithms: [JWT_ALG] });
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 为账户生成JWT token
|
* 为账户生成JWT token
|
||||||
* @param {Object} account - 账户对象
|
|
||||||
* @returns {string} JWT token
|
|
||||||
*/
|
*/
|
||||||
export function generateAccountToken(account) {
|
export function generateAccountToken(account) {
|
||||||
return signToken({
|
return signToken({
|
||||||
|
Loading…
x
Reference in New Issue
Block a user