【问题标题】:Express/React with CORS - Setting HTTP-Only Secure Cookie for React SPAExpress/React with CORS - 为 React SPA 设置仅 HTTP 安全 Cookie
【发布时间】:2019-07-17 14:36:18
【问题描述】:

我正在尝试在 AWS ElasticBeanstalk/S3 部署上使用 Express 作为 API (api.mysite.co) 和 React 作为 SPA (app.mysite.co) 设置一个简单的 API/SPA。对于身份验证,我正在尝试在仅限 HTTP 的安全 cookie 中设置 JWT。但是由于某种原因,cookie 永远不会到达客户端。

出于安全考虑,我已将域更改为 mysite.co。

这是我在 express 中的 app.js 文件:

const dotenv = require('dotenv');
const express = require('express');
const cookieParser = require('cookie-parser');
const logger = require('morgan');
const helmet = require('helmet');
const passport = require('passport');
const cors = require('cors');

const usersRouter = require('./routes/users');

const env = process.env.NODE_ENV || 'development';
const port = process.env.PORT || '3000';
const app = express();

// Setup dev env variables and modules
if (env === 'development') {
  dotenv.config();
  app.use(logger('dev'));
}

app.use(helmet());

// Setup CORS requests based on environment
if (env === 'development') {
  app.use(cors());
} else {
  app.use(
    cors({
      origin: 'https://app.mysite.co',
      credentials: true,
    }),
  );
}

app.use(cookieParser());

// Handle CORS pre-flight reqests
app.options('*', cors());

app.use(passport.initialize());
require('./lib/auth/passport')(passport); // eslint-disable-line global-require

app.use(express.json());
app.use(express.urlencoded({ extended: false }));

app.use('/users', usersRouter);

// Catch all route that masks 404 for security
app.get('*', (req, res) => {
  res.status(401).json({ error: 'Unauthorized access' });
});

app.listen(port);

还有我的 ./routes/users 文件和登录路径。我已经确认它正在工作,因为响应已发送并且用户已登录到应用程序。但由于某种原因,cookie 从未在 chrome 中设置。

const express = require('express');
const bcrypt = require('bcryptjs');
const jwt = require('jsonwebtoken');

const router = express.Router();

const mongodb = require('../db/mongodb');
const validateRegisterInput = require('../lib/validation/register');
const validateLoginInput = require('../lib/validation/login');

// // POST /users/login - User Login
router.post('/login', (req, res) => {
  mongodb.initDb().then((db) => {
    const collection = db.collection('users');
    const { errors, isValid } = validateLoginInput(req.body);

    if (!isValid) {
      return res.status(400).json(errors);
    }

    const { email, password } = req.body;

    return collection
      .find({ email })
      .limit(1)
      .toArray((connectErr, users) => {
        if (connectErr) {
          errors.email = 'Could not verify email. Please try again later.';
          return res.status(500).json(errors);
        }

        // Pull user out of array
        const user = users[0];

        if (!user) {
          errors.email = 'Username and Password do not match.';
          return res.status(400).json(errors);
        }

        return bcrypt.compare(password, user.password).then((isMatch) => {
          if (isMatch) {
            const payload = {
              id: user._id, // eslint-disable-line no-underscore-dangle
              expire_date: Date.now() + parseInt(process.env.JWT_EXPIRATION_MS, 10),
            };

            return jwt.sign(JSON.stringify(payload), process.env.PASSPORT_SECRET, (err, token) => {
              if (err) {
                return res.status(500).json({
                  password: 'There was an internal error. Please try again later',
                });
              }

              res.cookie('jwt', token, {
                domain: 'app.mysite.co',
                httpOnly: true,
                secure: true,
                maxAge: parseInt(process.env.JWT_EXPIRATION_MS, 10),
              });
              return res.status(200).json({
                name: user.name,
              });
            });
          }

          errors.password = 'Username and Password do not match.';
          return res.status(401).json(errors);
        });
      });
  });
});

module.exports = router;

在 React SPA 中,我使用 Axios 进行 POST 调用。在我正在设置的 app.js 文件中:

axios.defaults.baseURL = 'https://api.mysite.co';
axios.defaults.withCredentials = true;

然后像这样拨打电话:

/**
 * @description Logins in new user and sets their token in localStorage
 * @param {object} user Input from the login form.
 */
export const loginUser = user => dispatch => axios
  .post('/users/login', user)
  .then((res) => {
    dispatch(setCurrentUser(res.data));
  })
  .catch((err) => {
    dispatch(handleError(err));
  });

需要注意的一点是,当我发送请求时,飞行前的 OPTIONS 和 POST 调用对于请求标头都有这个:

Provisional headers are shown
Access-Control-Request-Headers: content-type
Access-Control-Request-Method: POST
Origin: https://app.mysite.co
Referer: https://app.mysite.co/login
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/72.0.3626.119 Safari/537.36

我做了一些研究,似乎当显示Provisional headers are shown 警告时,这意味着 API 调用无法正常工作。但我在 POST 调用中收到来自 API 的以下响应:

Request URL: https://api.mysite.co/users/login
Request Method: POST
Status Code: 200 
Remote Address: 52.201.140.183:443
Referrer Policy: no-referrer-when-downgrade

access-control-allow-credentials: true
access-control-allow-origin: https://app.mysite.co
content-length: 16
content-type: application/json; charset=utf-8
date: Sat, 23 Feb 2019 15:22:33 GMT
etag: W/"10-pyeidcXwb9ByfCNz+iqLTARQNP8"
server: nginx/1.14.1
status: 200
vary: Origin
x-powered-by: Express

Provisional headers are shown
Accept: application/json, text/plain, */*
Content-Type: application/json;charset=UTF-8
Origin: https://app.mysite.co
Referer: https://app.mysite.co/login
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/72.0.3626.119 Safari/537.36

显示成功,然后用户就登录了。

任何帮助都将不胜感激,因为我似乎已经超出了我的深度。我没有记录任何服务器或客户端错误,从我读过的所有内容来看,在 Express 和 Axios 上设置 credentials: true 标志应该可以解决问题。

谢谢!

【问题讨论】:

  • 你是从本地主机尝试吗?如果是这样,请尝试将安全设置为 false。
  • 好问题。请求从使用 CloudFront 的 S3 托管静态站点发送到使用 Application Load Balancer 的 Node.js AWS Elastic Beanstalk 部署。两者都使用 AWS SSL 证书加密。
  • 嗯,很奇怪。尝试将 app.use(cookieParser()); 移动到 server.js 中的 cors 内容之后
  • 只是这样做并更新了上面的代码以反映更改。不用找了。你认为这可能与 Chrome 给Provisional headers are shown 的请求中的警告有关吗?这是否表明浏览器正在使用缓存响应,即使我在 Chrome 开发工具中选中了“禁用缓存”标志?
  • 老实说不确定,谷歌搜索似乎表明存在 SSL 问题。在不使用 CloudFront 的情况下尝试一下,如果是这样,您可以隔离。

标签: reactjs express cookies axios cross-domain


【解决方案1】:

所以在尝试了一些不同的事情之后,问题在于将res.cookie 中的域设置为子域。我将该代码更改为:

              res.cookie('jwt', token, {
                domain: 'mysite.co',
                secure: true,
                httpOnly: true,
                maxAge: parseInt(process.env.JWT_EXPIRATION_MS, 10),
              });

现在正在设置 cookie。

【讨论】:

  • 遇到和你一样的问题。您能否概述一下您如何在 access-control-allow-origin 和前端设置域。
  • access-control-allow-origin 由 CORS 包设置。 npmjs.com/package/cors
猜你喜欢
  • 2020-10-25
  • 2021-04-04
  • 2015-02-08
  • 1970-01-01
  • 2021-08-27
  • 2021-09-05
  • 1970-01-01
  • 2021-09-05
  • 2013-01-06
相关资源
最近更新 更多