Skip to content

Commit

Permalink
1
Browse files Browse the repository at this point in the history
  • Loading branch information
SunWuyuan committed Nov 16, 2024
1 parent a8dd296 commit 7a8d057
Show file tree
Hide file tree
Showing 4 changed files with 55 additions and 46 deletions.
12 changes: 6 additions & 6 deletions prisma/schema.prisma
Original file line number Diff line number Diff line change
Expand Up @@ -162,7 +162,7 @@ model ow_users {
updatedAt DateTime? @default(dbgenerated("'2000-03-31 16:00:00'")) @db.Timestamp(0)
label String? @db.VarChar(255)
@@id([id, username])
@@id([id, username, email])
}

/// This model or at least one of its fields has comments in the database, and requires an additional setup for migrations: Read more: https://pris.ly/d/database-comments
Expand Down Expand Up @@ -359,11 +359,11 @@ model ow_users_totp {
updated_at DateTime? @default(now()) @db.Timestamp(0)
}

model MagicLinkToken {
id Int @id @default(autoincrement())
userId Int
token String @unique
expiresAt DateTime
model magiclinktoken {
id Int @id @default(autoincrement())
userId Int
token String @unique(map: "token") @db.VarChar(255)
expiresAt DateTime @db.DateTime(0)
}

/// The underlying view does not contain a valid unique identifier and can therefore currently not be handled by Prisma Client.
Expand Down
7 changes: 5 additions & 2 deletions server/configManager.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,14 +27,17 @@ class ConfigManager {
// Configuration information
global.configinfo = configs;

console.log(global.configinfo); // Log the updated config info
//console.log(global.configinfo); // Log the updated config info
}

async getConfig(key) {
// Check if the value is already cached
if (global.config && global.config[key]) {
if (global.config && global.config[key]!=null) {
return global.config[key];
}
var config = await this.prisma.ow_config.findFirst({ where: { key: key } });
console.log(config);
return config.value;
// If not cached, fetch from the database
await this.loadConfigsFromDB();
// If not cached, fetch from the database
Expand Down
37 changes: 19 additions & 18 deletions server/router_account.js
Original file line number Diff line number Diff line change
Expand Up @@ -416,64 +416,65 @@ router.post("/magiclink/generate", async (req, res) => {
try {
const { email } = req.body;
if (!email || !I.emailTest(email)) {
return res.status(400).json({ message: "无效的邮箱地址" });
return res.status(200).json({ status: "error",message: "无效的邮箱地址" });
}

const user = await I.prisma.ow_users.findUnique({ where: { email } });
const user = await I.prisma.ow_users.findFirst({ where: { email } });
if (!user) {
return res.status(404).json({ message: "用户不存在" });
//用户不存在
return res.status(404).json({ status: "error",message: "无效的邮箱地址" });
}

const token = jwt.sign(
{ userId: user.id },
await configManager.getConfig("MAGICLINK_SECRET"),
{ expiresIn: await configManager.getConfig("MAGICLINK_EXPIRATION") }
{ id: user.id },
await configManager.getConfig("security.jwttoken"),
{ expiresIn: 60 * 10 }
);

await I.prisma.magicLinkToken.create({
await I.prisma.magiclinktoken.create({
data: {
userId: user.id,
token,
expiresAt: new Date(Date.now() + parseInt(await configManager.getConfig("MAGICLINK_EXPIRATION")) * 1000),
expiresAt: new Date(Date.now() + 60 * 1000),
},
});

const magicLink = `${await configManager.getConfig("MAGICLINK_BASE_URL")}/magiclink/validate?token=${token}`;
const magicLink = `${await configManager.getConfig("urls.frontend")}/account/magiclink/validate?token=${token}`;
await sendEmail(
email,
"Magic Link 登录",
`点击以下链接登录:<a href="${magicLink}">${magicLink}</a>`
);

res.status(200).json({ message: "Magic Link 已发送到您的邮箱" });
res.status(200).json({status: "success", message: "Magic Link 已发送到您的邮箱" });
} catch (error) {
console.error("生成 Magic Link 时出错:", error);
res.status(500).json({ message: "生成 Magic Link 失败" });
res.status(200).json({ status: "error", message: "生成 Magic Link 失败" });
}
});

router.get("/magiclink/validate", async (req, res) => {
try {
const { token } = req.query;
if (!token) {
return res.status(400).json({ message: "无效的 Magic Link" });
return res.status(200).json({status: "error", message: "无效的 Magic Link" });
}

const decoded = jwt.verify(token, await configManager.getConfig("MAGICLINK_SECRET"));
const magicLinkToken = await I.prisma.magicLinkToken.findUnique({
const decoded = jwt.verify(token, await configManager.getConfig("security.jwttoken"));
const magicLinkToken = await I.prisma.magiclinktoken.findUnique({
where: { token },
});

if (!magicLinkToken || magicLinkToken.expiresAt < new Date()) {
return res.status(400).json({ message: "Magic Link 已过期" });
return res.status(200).json({ status: "error",message: "Magic Link 已过期" });
}

const user = await I.prisma.ow_users.findUnique({
where: { id: magicLinkToken.userId },
});

if (!user) {
return res.status(404).json({ message: "用户不存在" });
return res.status(404).json({ status: "error",message: "用户不存在" });
}

const jwtToken = await I.GenerateJwt({
Expand All @@ -485,7 +486,7 @@ router.get("/magiclink/validate", async (req, res) => {
});

res.status(200).json({
status: "OK",
status: "success",
message: "登录成功",
userid: user.id,
email: user.email,
Expand All @@ -496,7 +497,7 @@ router.get("/magiclink/validate", async (req, res) => {
});
} catch (error) {
console.error("验证 Magic Link 时出错:", error);
res.status(500).json({ message: "验证 Magic Link 失败" });
res.status(200).json({ status: "error",message: "验证 Magic Link 失败" });
}
});

Expand Down
45 changes: 25 additions & 20 deletions server/services/emailService.js
Original file line number Diff line number Diff line change
@@ -1,35 +1,40 @@
const configManager = require("../configManager");

const nodemailer = require('nodemailer');
let service, user, pass
configManager.getConfig('mail.service').then((res) => {
service = res
});
configManager.getConfig('mail.user').then((res) => {
user = res
});
configManager.getConfig('mail.pass').then((res) => {
pass = res
});
const transporter = nodemailer.createTransport({
service: service,
secure: true,
auth: {
user: user,
pass: pass,
},
const nodemailer = require("nodemailer");
let service, user, pass, transporter;
configManager.getConfig("mail.service").then((res) => {
service = res;

configManager.getConfig("mail.user").then((res) => {
user = res;

configManager.getConfig("mail.pass").then((res) => {
pass = res;
//console.log(service, user, pass);
transporter = nodemailer.createTransport({
service: service,
secure: true,
auth: {
user: user,
pass: pass,
},
});
});
});
});

const sendEmail = async (to, subject, html) => {
try {
await transporter.sendMail({
from: `${await configManager.getConfig('site.name')} <${await configManager.getConfig('mail.from')}>`,
from: `${await configManager.getConfig(
"site.name"
)} <${await configManager.getConfig("mail.from")}>`,
to,
subject,
html,
});
} catch (error) {
console.error('Error sending email:', error);
console.error("Error sending email:", error);
}
};

Expand Down

0 comments on commit 7a8d057

Please sign in to comment.