_http_outgoing.js: 470 выбрасывать новый ERR_HTTP_HEADERS_SENT ('set');Ошибка: невозможно установить заголовки после их отправки клиенту: веб-приложение Nodejs - PullRequest
0 голосов
/ 20 октября 2019

Код в Users.js получает ошибку во фрагменте по адресу: qrcode.toDataURL (secret.otpauth_url, (err, data_url) => {.

Я пыталсядобавив оператор return, чтобы убедиться, что я не отправляю ответ несколько раз. Я вижу, что data_url при преобразовании в онлайн-изображение показывает мне QR-код, но я не могу увидеть это, когда использую Postman.

router.post(
  "/",
  [
    check("name", "Name is required")
      .not().isEmpty(),
    check("email", "Please include a valid email").isEmail(),
    check(
      "password",
      "Please enter a password with 6 or more characters"
    ).isLength({ min: 6 })
  ],
async (req, res) => {
    console.log("hi");
    console.log(JSON.stringify(req.body));
    const errors = validationResult(req);
    if (!errors.isEmpty()) {
      //  return res.status(400).json({ errors: errors.array() });
    }

    const {
      name,
      email,
      password,
      type_of_user,
      question1,
      answer1,
    question2,
    answer2
    } = req.body;

    try {
      let user = await User.findOne({ email }); // await User.findOne({ email });



      user = new User({
        name,
        email,
        avatar,
        password,
        type_of_user,
        question1,
        answer1,
        question2,
        answer2
      });

      const salt = await bcrypt.genSalt(10); //await

      user.password = await bcrypt.hash(password, salt); // await

      user
        .save()
        .then(result => {
          // MFAOptions & secret will generate a secret
          const MFAOptions = {
            issuer: "xyz",
            user: req.body.email,
            length: 64
          };
          const secret = speakEasy.generateSecret(MFAOptions);

          const token = jwt.sign(
            {
              name: user.name,
              email: user.email,
              twofactor: false
            },

            config.get("jwtSecret"), // chnaged from process env jwt
            {
              expiresIn: "1h"
            }
          );

          // update the user that is just created:
          user
            .update(
              { email: req.body.email },
              {
                $set: { twoFASecret: secret.base32 }
              }
            )
            .exec()
            .then(result => {
              console.log(result);
              qrcode.toDataURL(secret.otpauth_url, (err, data_url) => {
                console.log(data_url);
                res.status(200).json({
                  img: data_url,
                  token: token
                });
              });
              return;
            })
            //if anything wrong, throws an error
            .catch(err => {
              console.log(err);
              //  res.status(500).json({ error: err });
            });
        })
        // originaly this will end here, but now it should redirect to twoFA route,
        // if something wrong, shows an error
        .catch(err => {
          console.log(err);
          //  res.status(500).json({ error: err });
        });

      // user with an id, primise which returns an id

      const payload = {
        user: {
          id: user.id
        }
      };

      jwt.sign(
        payload,
        config.get("jwtSecret"),
        { expiresIn: 3600 }, 
        (err, token) => {
          if (err) throw err;
          res.json({ token });
        }
      );
      //  } //else end
    } catch (err) {
      console.error(err.message);
      res.status(500).send("Server error");
    }
  }
);



module.exports = router;

1 Ответ

0 голосов
/ 21 октября 2019

Я думаю, что ваша проблема с выполнением этой строки qrcode.toDataURL(secret.otpauth_url, (err, data_url) => { у этого вызова есть обратный вызов, что означает, что вы продолжите выполнять остальную часть кода и отправите ответ, используя res.json, затем, после завершения выполнения qrcode, войдут в обратный вызов иотправить другой ответ, который не разрешен. у вас есть многократное выполнение для res.json, вам нужно удалить один из них и выполнить рефакторинг вашего кода. Я попытался провести рефакторинг вашего кода:

const validation = [check('name', 'Name is required').not().isEmpty(),
check('email', 'Please include a valid email').isEmail(),
check('password', 'Please enter a password with 6 or more characters').isLength({ min: 6 })]

const toDataURL = (otpauth_url) => new Promise((resolve, reject) => {
  qrcode.toDataURL(secret.otpauth_url, (err, data_url) => {
    if(err)reject(err)
    resolve(data_url)
      res.status(200).json({
      img: data_url,
      token,
    })
  })
});
const signJwt = (payload)=>new Promise((resolve,reject)=>{
 return  jwt.sign(
      payload,
      config.get('jwtSecret'),
      { expiresIn: 3600 },
      (err, token) => {
        if (err) reject(err)
        resolve(token)
      }
    )
})
const postRequest = async (req, res) => {
  const errors = validationResult(req)
  if (!errors.isEmpty()) {
    return res.status(400).json({ errors: errors.array() })
  }
  const { name, email, password, type_of_user, question1, answer1, question2, answer2, } = req.body

  try {
    let user = await User.findOne({ email })
    user = new User({
      name,
      email,
      avatar,
      password,
      type_of_user,
      question1,
      answer1,
      question2,
      answer2,
    })
    const salt = await bcrypt.genSalt(10) // await
    user.password = await bcrypt.hash(password, salt) // await
    await user.save()
    // MFAOptions & secret will generate a secret
    const MFAOptions = {
      issuer: 'xyz', user: req.body.email, length: 64,
    }
    const secret = speakEasy.generateSecret(MFAOptions)

    const token = jwt.sign(
      {
        name: user.name,
        email: user.email,
        twofactor: false,
      },
      config.get('jwtSecret'), { expiresIn: '1h', })

    // update the user that is just created:
    await user.update({ email: req.body.email },
      { $set: { twoFASecret: secret.base32 }, }).exec()
      const  data_url= await toDataURL(secret.otpauth_url)
      if(data_url)    return   res.status(200).json({
        img: data_url,
        token,
      })
    const payload = {
      user: {
        id: user.id,
      },
    }
  const  token= await signJwt(payload)
  return res.json({token})

  } catch (err) {
    console.error(err.message)
  return  res.status(500).send('Server error')
  }
}
router.post('/', validation, postRequest)


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