首页 > 解决方案 > MongoDB $ifNull 空数组仍在添加数据

问题描述

我有 2 个集合、用户和曲目。当我获取用户个人资料时,我想获取他的曲目。用户对象具有tracks轨道 ID 数组,第二个集合是轨道。

现在我在下面运行这段代码:

users
    .aggregate([
      {
        $match: { _id: ObjectId(userId) },
      },
      {
        $lookup: {
          from: "tracks",
          localField: "tracks",
          foreignField: "_id",
          as: "tracks",
        },
      },
      {
        $unwind: {
          path: "$tracks",
          preserveNullAndEmptyArrays: true,
        },
      },
      {
        $group: {
          _id: ObjectId(userId),
          username: { $first: "$username" },
          profileImg: { $first: "$profileImg" },
          socialLinks: { $first: "$socialLinks" },
          tracks: {
            $push: {
              _id: "$tracks._id",
              name: "$tracks.name",
              categoryId: "$tracks.categoryId",
              links: "$tracks.links",
              mediaId: isLogged ? "$tracks.mediaId" : null,
              thumbnailId: "$tracks.thumbnailId",
              views: { $size: { $ifNull: ["$tracks.views", []] } },
              downloads: { $size: { $ifNull: ["$tracks.downloads", []] } },
              uploadedDate: "$tracks.uploadedDate",
            },
          },
        },
      },
    ])

如果用户没有曲目或没有曲目,则该$ifNull语句返回一个仅包含这些字段的对象,因此它看起来像这样:

user: {
    // user data
    tracks: [{ views: 0, downloads: 0, mediaId: null }]
}

没有找到轨道所以"tracks.views"无法读取所以我添加了 $ifNull 语句,如何避免它返回空数据?此外,API 调用知道用户是否已登录(isLogged),我将其设置mediaIdnull. 如果没有找到轨道,为什么代码仍然只添加这 3 个字段?没有轨道可以通过它们...

编辑:任何曲目都有downloadsviews包含下载/查看曲目的用户 ID,曲目看起来像这样

{
    "name": "New Track #2227",
    "categoryId": "61695d57893f048528d049e5",
    "links": {
        "youtube": "https://www.youtube.com",
        "soundcloud": null,
        "spotify": null
    },
    "_id": "616c90651ab67bbd0b0a1172",
    "creatorId": "61695b5986ed44e5c1e1d29d",
    "mediaId": "616c90651ab67bbd0b0a1171",
    "thumbnailId": "616c90651ab67bbd0b0a1170",
    "plays": [],
    "status": "pending",
    "downloads": [],
    "uploadedDate": 1634504805
}

当获取的用户还没有任何跟踪帖子时,我会收到一个数组,其中包含一个包含上述字段的对象。

当用户没有轨道时,我期望得到的是一个空数组,正如您在上面看到的响应,轨道对象未满并且仅包含具有零值和空值的条件键。底线我想要的长度views并且downloads只有当有一个或更多的轨道时mediaId,我也想要隐藏它以防用户没有登录。如果没有曲目,我不明白为什么它会返回这 3 个字段

用户拥有一首或多首曲目时的预期结果

user: {
  // user data
  tracks: [
    {
      name: "New Track #2227",
      categoryId: "61695d57893f048528d049e5",
      links: {
        youtube: "https://www.youtube.com",
        soundcloud: null,
        spotify: null,
      },
      _id: "616c90651ab67bbd0b0a1172",
      creatorId: "61695b5986ed44e5c1e1d29d",
      mediaId: "616c90651ab67bbd0b0a1171",
      thumbnailId: "616c90651ab67bbd0b0a1170",
      plays: 0,
      status: "pending",
      downloads: 0,
      uploadedDate: 1634504805,
    },
  ];
}

用户没有曲目时的预期结果

user: {
  // user data
  tracks: [];
}

标签: node.jsmongodbmongodb-queryaggregateaggregation

解决方案


将此阶段附加到您的管道中:

{
  $set: {
    tracks: {
      $filter: {
        input: "$tracks",
        cond: { $ne: [ { $type: "$$this._id" }, "missing" ] } 
      }
    }
  }
}

推荐阅读