|
@@ -1,8 +1,12 @@
|
|
|
import dbConnect from "../../lib/dbConnect";
|
|
|
import Prediction from "../../models/Prediction";
|
|
|
+import Match from "../../models/Match";
|
|
|
+import User from "../../models/User";
|
|
|
import { NextResponse } from "next/server";
|
|
|
import { setCORSHeaders, handleError } from "../../lib/apiUtils";
|
|
|
import { withAuth } from "../../middleware/authMiddleware";
|
|
|
+import mongoose from "mongoose";
|
|
|
+
|
|
|
|
|
|
export const GET = withAuth(async (request) => {
|
|
|
await dbConnect();
|
|
@@ -13,176 +17,272 @@ export const GET = withAuth(async (request) => {
|
|
|
const pageSize = parseInt(searchParams.get("pageSize") || "10");
|
|
|
const homeTeam = searchParams.get("homeTeam");
|
|
|
const awayTeam = searchParams.get("awayTeam");
|
|
|
- const username = searchParams.get("username");
|
|
|
+ const userId = searchParams.get("userId");
|
|
|
const type = searchParams.get("type"); // 新增type参数来筛选运动类型
|
|
|
+ const action = searchParams.get("action"); // 新增type参数来筛选运动类型
|
|
|
+ const matchIds = searchParams.get("matchIds"); // 新增type参数来筛选运动类型
|
|
|
+
|
|
|
+ if (action == 'app') {
|
|
|
+ // 构建基础查询条件
|
|
|
+ let matchStage = {};
|
|
|
+ if (userId) {
|
|
|
+ matchStage.user = new mongoose.Types.ObjectId(userId);
|
|
|
+ }
|
|
|
+ if (matchIds) {
|
|
|
+ const matchIdsArray = matchIds.split(",").map((id) => new mongoose.Types.ObjectId(id.trim()));
|
|
|
+ matchStage.match = { $in: matchIdsArray };
|
|
|
+ }
|
|
|
|
|
|
- let matchStage = {};
|
|
|
-
|
|
|
- // 处理基础筛选条件
|
|
|
- if (username) {
|
|
|
- matchStage.username = { $regex: username, $options: "i" };
|
|
|
- }
|
|
|
+ const predictions = await Prediction.find(matchStage).lean();
|
|
|
|
|
|
- if (type) {
|
|
|
- matchStage.type = type;
|
|
|
- }
|
|
|
+ // 如果没有匹配的预测记录,直接返回空数组
|
|
|
+ if (!predictions.length) {
|
|
|
+ const response = NextResponse.json({
|
|
|
+ success: true,
|
|
|
+ total: 0,
|
|
|
+ data: [],
|
|
|
+ });
|
|
|
|
|
|
- // 处理其他搜索参数
|
|
|
- for (const [key, value] of searchParams.entries()) {
|
|
|
- if (
|
|
|
- ["current", "pageSize", "homeTeam", "awayTeam", "username", "type"].includes(key)
|
|
|
- )
|
|
|
- continue;
|
|
|
-
|
|
|
- switch (key) {
|
|
|
- case "user":
|
|
|
- try {
|
|
|
- matchStage[key] = new mongoose.Types.ObjectId(value);
|
|
|
- } catch (error) {
|
|
|
- console.error(`Invalid ObjectId for user: ${value}`);
|
|
|
- }
|
|
|
- break;
|
|
|
- // 足球预测筛选
|
|
|
- case "whoWillWin":
|
|
|
- matchStage["football.whoWillWin.prediction"] = value;
|
|
|
- break;
|
|
|
- case "firstTeamToScore":
|
|
|
- matchStage["football.firstTeamToScore.prediction"] = value;
|
|
|
- break;
|
|
|
- case "totalGoals":
|
|
|
- matchStage["football.totalGoals.prediction"] = parseInt(value);
|
|
|
- break;
|
|
|
- // 篮球预测筛选
|
|
|
- case "spread":
|
|
|
- matchStage["basketball.spread.prediction"] = value;
|
|
|
- break;
|
|
|
- case "totalPoints":
|
|
|
- matchStage["basketball.totalPoints.prediction"] = value;
|
|
|
- break;
|
|
|
- default:
|
|
|
- matchStage[key] = { $regex: value, $options: "i" };
|
|
|
+ return setCORSHeaders(response);
|
|
|
}
|
|
|
- }
|
|
|
|
|
|
- let pipeline = [
|
|
|
- {
|
|
|
- $lookup: {
|
|
|
- from: "users",
|
|
|
- localField: "user",
|
|
|
- foreignField: "_id",
|
|
|
- as: "userInfo",
|
|
|
- },
|
|
|
- },
|
|
|
- { $unwind: { path: "$userInfo", preserveNullAndEmptyArrays: true } },
|
|
|
- {
|
|
|
- $addFields: {
|
|
|
- username: "$userInfo.username",
|
|
|
- },
|
|
|
- },
|
|
|
- { $match: matchStage },
|
|
|
- {
|
|
|
- $lookup: {
|
|
|
- from: "matches",
|
|
|
- localField: "match",
|
|
|
- foreignField: "_id",
|
|
|
- as: "matchDetails",
|
|
|
- },
|
|
|
- },
|
|
|
- { $unwind: { path: "$matchDetails", preserveNullAndEmptyArrays: true } },
|
|
|
- {
|
|
|
- $addFields: {
|
|
|
- matchTime: {
|
|
|
- $concat: [
|
|
|
- {
|
|
|
- $dateToString: {
|
|
|
- format: "%Y-%m-%d",
|
|
|
- date: "$matchDetails.date",
|
|
|
- },
|
|
|
- },
|
|
|
- " ",
|
|
|
- { $ifNull: ["$matchDetails.time", "00:00"] },
|
|
|
- ],
|
|
|
- },
|
|
|
- },
|
|
|
- },
|
|
|
- {
|
|
|
- $project: {
|
|
|
- userInfo: 0,
|
|
|
- },
|
|
|
- },
|
|
|
- { $sort: { matchTime: -1 } },
|
|
|
- ];
|
|
|
-
|
|
|
- if (homeTeam) {
|
|
|
- pipeline.push({
|
|
|
- $match: {
|
|
|
- "matchDetails.homeTeam.name": homeTeam,
|
|
|
- },
|
|
|
- });
|
|
|
- }
|
|
|
- if (awayTeam) {
|
|
|
- pipeline.push({
|
|
|
- $match: {
|
|
|
- "matchDetails.awayTeam.name": awayTeam,
|
|
|
- },
|
|
|
- });
|
|
|
- }
|
|
|
+ const matchIdsToQuery = predictions.map((prediction) => prediction.match);
|
|
|
+ const matches = await Match.find({ _id: { $in: matchIdsToQuery } }).lean();
|
|
|
+ // 将 matches 转换为一个映射,方便快速查找
|
|
|
+ const matchMap = matches.reduce((map, match) => {
|
|
|
+ map[match._id.toString()] = match;
|
|
|
+ return map;
|
|
|
+ }, {});
|
|
|
+
|
|
|
+ // 合并结果
|
|
|
+ const formattedPredictions = predictions.map((prediction) => {
|
|
|
+ const matchDetails = matchMap[prediction.match.toString()] || {};
|
|
|
|
|
|
+ const formattedPrediction = {
|
|
|
+ matchId: prediction.match,
|
|
|
+ user: prediction.user,
|
|
|
+ username: prediction.username,
|
|
|
+ type: prediction.type,
|
|
|
+ ...prediction,
|
|
|
+ pointsEarned: prediction.pointsEarned,
|
|
|
+ matchTime: matchDetails.date
|
|
|
+ ? `${new Date(matchDetails.date).toISOString().split("T")[0]} ${matchDetails.time || "00:00"}`
|
|
|
+ : null,
|
|
|
+ homeTeam: matchDetails.homeTeam?.name || "N/A",
|
|
|
+ awayTeam: matchDetails.awayTeam?.name || "N/A",
|
|
|
+ score: "",
|
|
|
+ };
|
|
|
|
|
|
- const countPipeline = [...pipeline, { $count: "total" }];
|
|
|
- const totalResult = await Prediction.aggregate(countPipeline);
|
|
|
- const totalCount = totalResult.length > 0 ? totalResult[0].total : 0;
|
|
|
+ // 根据运动类型添加特定信息
|
|
|
+ if (prediction.type === "football") {
|
|
|
+ formattedPrediction.score = `${matchDetails.homeTeamScore || 0}:${matchDetails.awayTeamScore || 0}`;
|
|
|
+ } else if (prediction.type === "basketball") {
|
|
|
+ formattedPrediction.score = `${matchDetails.homeTeamScore || 0}-${matchDetails.awayTeamScore || 0}`;
|
|
|
+ }
|
|
|
|
|
|
- pipeline.push({ $skip: (current - 1) * pageSize });
|
|
|
- pipeline.push({ $limit: pageSize });
|
|
|
+ return formattedPrediction;
|
|
|
+ });
|
|
|
+ const response = NextResponse.json({
|
|
|
+ success: true,
|
|
|
+ total: 0,
|
|
|
+ data: formattedPredictions,
|
|
|
+ });
|
|
|
|
|
|
- const predictions = await Prediction.aggregate(pipeline);
|
|
|
+ return setCORSHeaders(response);
|
|
|
+ } else {
|
|
|
+ let matchStage = {};
|
|
|
+ // 处理基础筛选条件
|
|
|
+ if (userId) {
|
|
|
+ matchStage.userId = { $regex: userId, $options: "i" };
|
|
|
+ }
|
|
|
|
|
|
- const validPredictions = [];
|
|
|
- const invalidPredictionIds = [];
|
|
|
+ if (type) {
|
|
|
+ matchStage.type = type;
|
|
|
+ }
|
|
|
+ if (homeTeam) {
|
|
|
+ pipeline.push({
|
|
|
+ $match: {
|
|
|
+ "matchDetails.homeTeam.name": homeTeam,
|
|
|
+ },
|
|
|
+ });
|
|
|
+ }
|
|
|
+ if (awayTeam) {
|
|
|
+ pipeline.push({
|
|
|
+ $match: {
|
|
|
+ "matchDetails.awayTeam.name": awayTeam,
|
|
|
+ },
|
|
|
+ });
|
|
|
+ }
|
|
|
+ // 处理其他搜索参数
|
|
|
+ for (const [key, value] of searchParams.entries()) {
|
|
|
+ if (
|
|
|
+ ["current", "pageSize", "homeTeam", "awayTeam", "username", "type"].includes(key)
|
|
|
+ )
|
|
|
+ continue;
|
|
|
+
|
|
|
+ switch (key) {
|
|
|
+ case "user":
|
|
|
+ try {
|
|
|
+ matchStage[key] = new mongoose.Types.ObjectId(value);
|
|
|
+ } catch (error) {
|
|
|
+ console.error(`Invalid ObjectId for user: ${value}`);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ // 足球预测筛选
|
|
|
+ case "whoWillWin":
|
|
|
+ matchStage["football.whoWillWin.prediction"] = value;
|
|
|
+ break;
|
|
|
+ case "firstTeamToScore":
|
|
|
+ matchStage["football.firstTeamToScore.prediction"] = value;
|
|
|
+ break;
|
|
|
+ case "totalGoals":
|
|
|
+ matchStage["football.totalGoals.prediction"] = parseInt(value);
|
|
|
+ break;
|
|
|
+ // 篮球预测筛选
|
|
|
+ case "spread":
|
|
|
+ matchStage["basketball.spread.prediction"] = value;
|
|
|
+ break;
|
|
|
+ case "totalPoints":
|
|
|
+ matchStage["basketball.totalPoints.prediction"] = value;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ matchStage[key] = { $regex: value, $options: "i" };
|
|
|
+ }
|
|
|
+ }
|
|
|
+ let basePipeline = [{ $match: matchStage }];
|
|
|
+
|
|
|
+ const countPipeline = [...basePipeline, { $count: "total" }];
|
|
|
+ const totalResult = await Prediction.aggregate(countPipeline);
|
|
|
+ const totalCount = totalResult.length > 0 ? totalResult[0].total : 0;
|
|
|
+
|
|
|
+ let pipeline = [
|
|
|
+ { $match: matchStage },
|
|
|
+ // {
|
|
|
+ // $lookup: {
|
|
|
+ // from: "matches",
|
|
|
+ // localField: "match",
|
|
|
+ // foreignField: "_id",
|
|
|
+ // as: "matchDetails",
|
|
|
+ // },
|
|
|
+ // },
|
|
|
+ // { $unwind: { path: "$matchDetails", preserveNullAndEmptyArrays: true } },
|
|
|
+ // {
|
|
|
+ // $addFields: {
|
|
|
+ // matchTime: {
|
|
|
+ // $concat: [
|
|
|
+ // {
|
|
|
+ // $dateToString: {
|
|
|
+ // format: "%Y-%m-%d",
|
|
|
+ // date: "$matchDetails.date",
|
|
|
+ // },
|
|
|
+ // },
|
|
|
+ // " ",
|
|
|
+ // { $ifNull: ["$matchDetails.time", "00:00"] },
|
|
|
+ // ],
|
|
|
+ // },
|
|
|
+ // },
|
|
|
+ // },
|
|
|
+ // {
|
|
|
+ // $project: {
|
|
|
+ // userInfo: 0,
|
|
|
+ // },
|
|
|
+ // },
|
|
|
+ { $sort: { createdAt: -1 } },
|
|
|
+ ];
|
|
|
+
|
|
|
+ pipeline.push({ $skip: (current - 1) * pageSize });
|
|
|
+ pipeline.push({ $limit: pageSize });
|
|
|
+
|
|
|
+ const predictions = await Prediction.aggregate(pipeline);
|
|
|
+
|
|
|
+ // predictions.forEach((prediction) => {
|
|
|
+ // if (prediction.matchDetails) {
|
|
|
+ // const formattedPrediction = {
|
|
|
+ // matchId: prediction.matchDetails._id,
|
|
|
+ // ...prediction,
|
|
|
+ // match: undefined,
|
|
|
+ // matchDetails: undefined,
|
|
|
+ // homeTeam: prediction.matchDetails.homeTeam.name,
|
|
|
+ // awayTeam: prediction.matchDetails.awayTeam.name
|
|
|
+ // };
|
|
|
+
|
|
|
+ // // 根据运动类型添加特定信息
|
|
|
+ // if (prediction.type === "football") {
|
|
|
+ // formattedPrediction.score = `${prediction.matchDetails.homeTeamScore}:${prediction.matchDetails.awayTeamScore}`;
|
|
|
+ // } else if (prediction.type === "basketball") {
|
|
|
+ // formattedPrediction.score = `${prediction.matchDetails.homeTeamScore}-${prediction.matchDetails.awayTeamScore}`;
|
|
|
+ // }
|
|
|
+
|
|
|
+ // validPredictions.push(formattedPrediction);
|
|
|
+ // } else {
|
|
|
+ // invalidPredictionIds.push(prediction._id);
|
|
|
+ // }
|
|
|
+ // });
|
|
|
+
|
|
|
+ const matchIdsToQuery = predictions.map((prediction) => prediction.match);
|
|
|
+ const matches = await Match.find({ _id: { $in: matchIdsToQuery } }).lean();
|
|
|
+ // 将 matches 转换为一个映射,方便快速查找
|
|
|
+ const matchMap = matches.reduce((map, match) => {
|
|
|
+ map[match._id.toString()] = match;
|
|
|
+ return map;
|
|
|
+ }, {});
|
|
|
+
|
|
|
+ const validPredictions = [];
|
|
|
+ const invalidPredictionIds = [];
|
|
|
+
|
|
|
+ // 合并结果
|
|
|
+ predictions.map((prediction) => {
|
|
|
+ const matchDetails = matchMap[prediction.match.toString()] || {};
|
|
|
|
|
|
- predictions.forEach((prediction) => {
|
|
|
- if (prediction.matchDetails) {
|
|
|
const formattedPrediction = {
|
|
|
- matchId: prediction.matchDetails._id,
|
|
|
+ matchId: prediction.match,
|
|
|
+ user: prediction.user,
|
|
|
+ username: prediction.username,
|
|
|
+ type: prediction.type,
|
|
|
...prediction,
|
|
|
- match: undefined,
|
|
|
- matchDetails: undefined,
|
|
|
- homeTeam: prediction.matchDetails.homeTeam.name,
|
|
|
- awayTeam: prediction.matchDetails.awayTeam.name
|
|
|
+ pointsEarned: prediction.pointsEarned,
|
|
|
+ matchTime: matchDetails.date
|
|
|
+ ? `${new Date(matchDetails.date).toISOString().split("T")[0]} ${matchDetails.time || "00:00"}`
|
|
|
+ : null,
|
|
|
+ homeTeam: matchDetails.homeTeam?.name || "N/A",
|
|
|
+ awayTeam: matchDetails.awayTeam?.name || "N/A",
|
|
|
+ score: "",
|
|
|
};
|
|
|
|
|
|
// 根据运动类型添加特定信息
|
|
|
if (prediction.type === "football") {
|
|
|
- formattedPrediction.score = `${prediction.matchDetails.homeTeamScore}:${prediction.matchDetails.awayTeamScore}`;
|
|
|
+ formattedPrediction.score = `${matchDetails.homeTeamScore || 0}:${matchDetails.awayTeamScore || 0}`;
|
|
|
} else if (prediction.type === "basketball") {
|
|
|
- formattedPrediction.score = `${prediction.matchDetails.homeTeamScore}-${prediction.matchDetails.awayTeamScore}`;
|
|
|
+ formattedPrediction.score = `${matchDetails.homeTeamScore || 0}-${matchDetails.awayTeamScore || 0}`;
|
|
|
+ }
|
|
|
+ if (matchDetails) {
|
|
|
+ validPredictions.push(formattedPrediction);
|
|
|
+ } else {
|
|
|
+ invalidPredictionIds.push(prediction._id);
|
|
|
}
|
|
|
+ // return formattedPrediction;
|
|
|
+ });
|
|
|
|
|
|
- validPredictions.push(formattedPrediction);
|
|
|
- } else {
|
|
|
- invalidPredictionIds.push(prediction._id);
|
|
|
+ if (invalidPredictionIds.length > 0) {
|
|
|
+ const deleteResult = await Prediction.deleteMany({
|
|
|
+ _id: { $in: invalidPredictionIds },
|
|
|
+ });
|
|
|
+ console.log(`Deleted ${deleteResult.deletedCount} invalid predictions.`);
|
|
|
}
|
|
|
- });
|
|
|
|
|
|
- if (invalidPredictionIds.length > 0) {
|
|
|
- const deleteResult = await Prediction.deleteMany({
|
|
|
- _id: { $in: invalidPredictionIds },
|
|
|
+ const response = NextResponse.json({
|
|
|
+ success: true,
|
|
|
+ total: totalCount - invalidPredictionIds.length,
|
|
|
+ data: validPredictions,
|
|
|
});
|
|
|
- console.log(`Deleted ${deleteResult.deletedCount} invalid predictions.`);
|
|
|
- }
|
|
|
|
|
|
- const response = NextResponse.json({
|
|
|
- success: true,
|
|
|
- total: totalCount - invalidPredictionIds.length,
|
|
|
- data: validPredictions,
|
|
|
- });
|
|
|
-
|
|
|
- return setCORSHeaders(response);
|
|
|
+ return setCORSHeaders(response);
|
|
|
+ }
|
|
|
} catch (error) {
|
|
|
console.error("Error in GET request:", error);
|
|
|
return handleError(error);
|
|
|
}
|
|
|
+
|
|
|
});
|
|
|
|
|
|
export const POST = withAuth(async (request) => {
|
|
@@ -198,6 +298,16 @@ export const POST = withAuth(async (request) => {
|
|
|
);
|
|
|
}
|
|
|
|
|
|
+ // 查找用户
|
|
|
+ const userInfo = await User.findOne({ _id: userId });
|
|
|
+ if (!userInfo) {
|
|
|
+ response = NextResponse.json(
|
|
|
+ { success: false, error: "无效用户" },
|
|
|
+ { status: 401 }
|
|
|
+ );
|
|
|
+ return setCORSHeaders(response);
|
|
|
+ }
|
|
|
+
|
|
|
if (
|
|
|
!predictions ||
|
|
|
!Array.isArray(predictions) ||
|
|
@@ -226,6 +336,7 @@ export const POST = withAuth(async (request) => {
|
|
|
|
|
|
|
|
|
if (existingPrediction) {
|
|
|
+ existingPrediction.username = userInfo.username;
|
|
|
// 更新预测
|
|
|
if (type === "football" && football) {
|
|
|
if (football.whoWillWin) {
|
|
@@ -274,6 +385,7 @@ export const POST = withAuth(async (request) => {
|
|
|
// 创建新预测
|
|
|
const newPrediction = new Prediction({
|
|
|
user: userId,
|
|
|
+ username: userInfo.username,
|
|
|
match: matchId,
|
|
|
type,
|
|
|
...(type === "football" &&
|