正如标题所示,我试图在用户登录时从用户那里获取特定数据并聚合所有数据,然后返回聚合数据。我尝试使用 $match
函数,但没有运气。令人沮丧的是我能够成功返回对象数组中的所有数据。更具体地说,下面的代码成功运行并返回一个对象数组:
const runs = await Run.find({ user: req.user.id })
但是此代码没有,它返回一个空数组:
const cumulativeTotals = await Run.aggregate([ { $match: { user: req.user.id } }, { $group: { _id: null, totalRunTime: { $sum: '$runTime' }, avgRunTime: { $avg: '$runTime' }, totalRunDistance: { $sum: '$runDistance' }, avgRunDistance: { $avg: '$runDistance' }, avgPace: { $avg: '$avgPace' }, totalHeartRate: { $avg: '$avgHeartRate' }, totalActiveCalories: { $sum: '$activeCalories' }, averageActiveCalories: { $avg: '$activeCalories' }, absoluteTotalCalories: { $sum: '$totalCalories' }, avgTotalCalories: { $avg: '$totalCalories' }, } } ])
我不确定为什么会出现这种情况。它应该返回的是一个带有对象的数组,该对象填充了其中的所有聚合数据。
我还知道,如果删除 $match
它将成功聚合运行集合中的所有数据,所以我认为这与此有关。
以下是所有相关代码:
const getRuns = asyncHandler(async (req, res) => { const runs = await Run.find({ user: req.user.id }) const cumulativeTotals = await Run.aggregate([ { $match: { user: req.user.id } }, { $group: { _id: null, totalRunTime: { $sum: '$runTime' }, avgRunTime: { $avg: '$runTime' }, totalRunDistance: { $sum: '$runDistance' }, avgRunDistance: { $avg: '$runDistance' }, avgPace: { $avg: '$avgPace' }, totalHeartRate: { $avg: '$avgHeartRate' }, totalActiveCalories: { $sum: '$activeCalories' }, averageActiveCalories: { $avg: '$activeCalories' }, absoluteTotalCalories: { $sum: '$totalCalories' }, avgTotalCalories: { $avg: '$totalCalories' }, } } ]) if (!runs) { res.status(400).json({ message: 'No Runs Found for This User' }) return } const response = { runs, cumulativeTotals } res.status(200).json(response) })
P粉7224099962024-03-23 10:52:36
因此,在 rickhg12hs 的帮助下,我们能够确定 {match: { user : req.user.id } }
是我试图将字符串与 ObjectId 进行比较,这是行不通的。这是因为我设置的 API 将用户 ID 作为字符串返回,而不是作为 ObjectID。为了确保您比较的是正确的值类型(在本例中为 ObjectID),您可以这样做:
{ $match: { user: new mongoose.Types.ObjectId(req.user.id) } }
它获取返回的字符串并将其转换为新的 ObjectID,并且因为它现在正在比较 ObjectID,所以它可以工作。
显然,它一直在文档中,我只是没有'看起来还不够仔细