Экспресс Mongoose использует Dot Notated Field в качестве поля для обновления - PullRequest
0 голосов
/ 12 декабря 2018

Я пытаюсь использовать Express для динамического обновления полей модели Mongoose.

Данные, отправленные в обработчик маршрута из вызова ajax, выглядят так:

{ value: true, name: 'settings.follow.state' }

Как вы можете видеть, значение представляет собой строку с точечной нотацией?и я хотел бы использовать это как селектор поля для обновления моей модели Mongoose.

Так что мой клиентский вызов ajax выглядит так:

$("[name='settings.follow.state']").on("change",function () {
   $.ajax({
       url : window.location.pathname +"/update",
       type : "POST",
       contentType: "application/json",
       data : JSON.stringify({
           value : $(this).is(":checked"),
           name : "settings.follow.state"
       }),
       success:function (response) {
           console.log(response);
       }
   });
});

Кроме того, мне нужно JSON.stringifyмои данные при использовании Express?

Это мой обработчик маршрута Express:

router.post("/:id/update",[
    oneOf([
        check("value","Field cannot be empty.").isBoolean(),
        check("value","Field cannot be empty.").not().isEmpty()
    ]),
    check("name","Missing name field.").not().isEmpty(),
],function (req,res,next) {
    let errors = validationResult(req);
    if(!errors.isEmpty()){
        return res.status(400).send(errors.array()[0].msg);
    }

    Profile.findOne({_id:req.params.id,user:req.user._id}, function (err,doc) {
        if(err) return next(err);
        if(doc !== null){
            TwitterSettings.findOne({_id:doc.settings},function (err,doc) {
                if(err) next(err);
                if(doc !== null){
                    //HERE is where I would like to update the dynamic fields/values
                    doc.set(req.body.name,req.body.value);
                    //Tried with no avail :-(
                    // doc.set([req.body.name],req.body.value);
                    // doc.set(`${req.body.name}`,req.body.value);
                    doc.save(function (err,doc) {
                        if(err) return next(err);
                        if(doc !== null){
                            return res.status(200).json(doc);
                        }
                        return res.status(400).send("Could not update profile.");
                    });
                }else{
                    return res.status(400).send("Could find profile settings to update.");
                }
            });

            // TRIED findOneAndUpdate too
            // TwitterSettings.findOneAndUpdate({_id:doc.settings},{$set:{[req.body.name] : req.body.value}},{"new":true},function (err,doc) {
            //     if(err) return next(err);
            //     if(doc !== null){
            //         return res.status(200).json(doc);
            //     }else{
            //         return res.status(400).send("Could not update profile.");
            //     }
            // });
        }else{
            return res.status(400).send("Could find profile to update.");
        }
    });
});

Моя схема режима Mongoose (только если она вам нужна):

const mongoose = require('mongoose');
const Schema = mongoose.Schema;
const schema = new Schema({
    followers : {type:Number,default:0},
    followings : {type:Number,default:0},
    follow:{
        state : {type:Boolean,default:false},

    },
    un_follow:{
        state : {type:Boolean,default:false}

    },
    follow_back:{
        state : {type:Boolean,default:false}

    },

});
const model = mongoose.model('TwitterSettings', schema);
module.exports = model;

IЯ уверен, что есть кто-то умнее меня, который может указать мне правильное направление.

Спасибо, Карл

1 Ответ

0 голосов
/ 12 декабря 2018

Хорошо, у вас есть коллекция с именем TwitterSettings, и вы хотите обновить значение по пути follow.state.

Я бы обычно использовал для этого модуль nessy .

const nessy = require( "nessy" );

...

Profile.findOne({_id:req.params.id,user:req.user._id}, function (err,doc) {
   ...
   TwitterSettings.findOne({_id:doc.settings},function (err,doc) {
       ...
       nessy( req.body.name, req.body.value, doc );
       doc.save( function( err, doc ) {
           ...
       } );
       ...
   } ); 
   ...
} );

Следует иметь в виду, что req.body.name на самом деле содержит settings.follow.state, поэтому вы должны удалить это из req.body.name:

nessy( req.body.name.replace(/^settings\./, ""), req.body.value, doc );

РЕДАКТИРОВАТЬ

На самом деле на основе комментария вы можете просто использовать model.set() и точечную запись без settings. части в нем.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...