从 Stripe 获取 webhook 异常

Øyv*_*hen 1 webhooks node.js stripe-payments

我正在尝试从 Stripe 设置一个 webhook 来处理该payment_intent.succeeded事件,但出现异常。这是我来自 Node 后端的代码(我已经提取了我希望的所有相关部分。如果还需要其他任何东西,请告诉我):

const stripeWebHookSecret = 'whsec_WA0Rh4vAD3z0rMWy4kv2p6XXXXXXXXXXX';

import express from 'express';
const app = express();
app.use(bodyParser.urlencoded({ extended:true }));
app.use(bodyParser.json());
app.use(session({ <some params here> }));

const openRouter = express.Router();

registerOpenPaymentRoutes(openRouter);

app.use('/open', openRouter);
Run Code Online (Sandbox Code Playgroud)

的实现registerOpenPaymentRoutes如下所示:

export const registerOpenPaymentRoutes = (router) => {
    router.post('/payment/intent/webhook', bodyParser.raw({type: 'application/json'}), (req, res) => {
        let signature = req.headers['stripe-signature'];
        try {
            let event = stripe.webhooks.constructEvent(req.body, signature, stripeWebHookSecret);
            switch(event.type){
                case 'payment_intent.succeeded':
                    let intent = event.data.object;
                    res.json({ message: 'Everything went smooth!', intent });
                default:
                    res.status(400).json({ error: 'Event type not supported' });
            }
        }
        catch (error){
            res.status(400).json({ message: `Wrong signature`, signature, body: req.body, error });
        }
    });
}
Run Code Online (Sandbox Code Playgroud)

到目前为止一切顺利。当我从 Stripe 仪表板触发测试 webhook 事件时,我击中了端点,但从 catch 块中获得了结果。错误信息如下:

No signatures found matching the expected signature for payload. Are you passing the raw request body you received from Stripe? https://github.com/stripe/stripe-node#webhook-signing"
Run Code Online (Sandbox Code Playgroud)

我正在返回带有错误消息的签名以及您在上面看到的签名,并且签名如下所示:

"t=1557911017,v1=bebf499bcb35198b8bfaf22a68b8879574298f9f424e57ef292752e3ce21914d,v0=23402bb405bfd6bd2a13c310cfecda7ae1905609923d801fa4e8b872a4f82894"

据我从文档中了解到,获取上述原始请求正文所需bodyParser.raw({type: 'application/json'})的是我已经拥有的路由器的参数。

有人能看到缺失的部分吗?

Pau*_*jes 7

这是因为您已经将您的 express 应用程序设置为使用bodyParser.json()中间件,这与bodyParser.raw()您在 webhook 路由中设置的中间件发生冲突。

如果您删除该app.use(bodyParser.json());行,您的 webhooks 将按预期工作,但是您的其余路由将没有解析的正文,这并不理想。

我建议添加一个自定义中间件来bodyParser根据路由选择版本。就像是:

// only use the raw bodyParser for webhooks
app.use((req, res, next) => {
  if (req.originalUrl === '/payment/intent/webhook') {
    next();
  } else {
    bodyParser.json()(req, res, next);
  }
});
Run Code Online (Sandbox Code Playgroud)

然后bodyParser.raw()在 webhook 路由上显式使用中间件。