最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - How `transform` in mongoose schema works? - Stack Overflow

programmeradmin2浏览0评论

Picture of my user model : In my user model I have this as second argument to my user model, to delete __v and replace _id by id:

  {
    toJSON: {
      transform: function (doc, ret) {
        ret.id = ret._id;
        delete ret._id;
        delete ret.password;
        delete ret.__v;
      },
    },
  }

In my signin router I have something like this :

const existingUser = await User.findOne({email});
console.log("existingUser*", existingUser)
res.status(200).send(existingUser);

I got this from my console.log

 {
  _id: 5fe81e29fdd22a00546b05e3,
  email: '[email protected]',
  password: '0636b425ef0add0056ec85a5596eacf9ff0c71f8c2a1d4bad068a8679398e11870df12262722b911502eacb5fca23cef0cdd3b740481102ead50c58756d14a34.3f82d856ad93bc99',
  __v: 0
}

But in postman I received this :

{
    "email": "[email protected]",
    "id": "5fe81e29fdd22a00546b05e3"
}

I know that with transform, "if set, mongoose will call this function to allow you to transform the returned object".
But could someone explain to me when the 'transform' occurs to justify the difference between the console.log and the data I received in postman ?
Does this have something to do with asynchronous ?

Picture of my user model : In my user model I have this as second argument to my user model, to delete __v and replace _id by id:

  {
    toJSON: {
      transform: function (doc, ret) {
        ret.id = ret._id;
        delete ret._id;
        delete ret.password;
        delete ret.__v;
      },
    },
  }

In my signin router I have something like this :

const existingUser = await User.findOne({email});
console.log("existingUser*", existingUser)
res.status(200).send(existingUser);

I got this from my console.log

 {
  _id: 5fe81e29fdd22a00546b05e3,
  email: '[email protected]',
  password: '0636b425ef0add0056ec85a5596eacf9ff0c71f8c2a1d4bad068a8679398e11870df12262722b911502eacb5fca23cef0cdd3b740481102ead50c58756d14a34.3f82d856ad93bc99',
  __v: 0
}

But in postman I received this :

{
    "email": "[email protected]",
    "id": "5fe81e29fdd22a00546b05e3"
}

I know that with transform, "if set, mongoose will call this function to allow you to transform the returned object".
But could someone explain to me when the 'transform' occurs to justify the difference between the console.log and the data I received in postman ?
Does this have something to do with asynchronous ?

Share edited Dec 27, 2020 at 7:26 Ibra asked Dec 27, 2020 at 7:17 IbraIbra 3484 silver badges9 bronze badges 4
  • i didnt undertand your problem, do you want to replace id with _id and remove __v ? – Amir Doreh Commented Dec 27, 2020 at 8:34
  • Everythink works fine. What I don't understand is why different output between the console.log(existingUser) and res.status(200).send(existingUser); Same variable but outputs are different as explained above. – Ibra Commented Dec 27, 2020 at 10:21
  • Are you sure that is happening ? I don't believe that your server would respond differenly for postman and console.log; Most probably your console.log argument and res.send argument would be different – d_bhatnagar Commented Feb 14, 2021 at 19:52
  • Also, try using toObject() method with transform property like userSchema.options.toObject.transform = function(doc, output, options) { delete output.__v; return output; } That is a very helpful functionality. Read more here – d_bhatnagar Commented Feb 14, 2021 at 19:55
Add a ment  | 

1 Answer 1

Reset to default 11

res.status(200).send(existingUser); looks like expressjs (or look-alike) controller code, so i'll assume it is Express.

.send(body) method sends response to the client browser as a string (well, technically). So, before actual transmission, the body argument is converted to string if it isn't a string already. existingUser in your code isn't a string, it's a mongoose object, so express casts it to a string, effectively this will be similar to the following code:

res.status(200)
  .send(
    existingUser.toString() // .toString() here is the key
  );

Under the hood, mongoose object's .toString() is proxied to .toJSON() method, so your code bees equivalent to following:

  ...
  .send(
    existingUser.toJSON() // .toJSON() here
  );

...and .toJSON() method is what takes into account the transform(doc, ret) option, that you specified for the mongoose schema.

console.log() on the other hand, does not use underlying .toString()/.toJSON() methods of the arguments. If you want to print to console the result, that would be received by the end consumer (postman, f.e.), then you should call the transform manually:

console.log(existingUser.toJSON()); // like this, transformed, but not stringified
console.log(existingUser.toString()); // or like this, but already stringified
console.log(JSON.stringify(existingUser, null, 3)); // or transform and then stringify with custom formatting (3-space tabulated instead of default single-line formatting)

The whole transform chain looks like this:

Model
  -> `.toJSON()` 
    -> Mongoose transforms model internally into POJO
      if custom transform is defined
      -> Mongoose passes POJO to user defined `transform(doc, ret)`, 
         where `doc` - original document, `ret` - internally transformed POJO
        -> `ret` is returned
      else
      -> POJO is returned 
发布评论

评论列表(0)

  1. 暂无评论