从 Stripe 获取 webhook 异常
Getting exception on webhook from Stripe
我正在尝试从 Stripe 设置一个 webhook 来处理 payment_intent.succeeded
事件,但我遇到了异常。这是我来自 Node 后端的代码(我已经提取了我希望的所有相关部分。如果需要其他任何内容,请告诉我):
const stripeWebHookSecret = 'whsec_WA0Rh4vAD3z0rMWy4kv2p6XXXXXXXXXX';
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);
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 });
}
});
}
到目前为止 good.When 我从 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"
我正在返回带有错误消息的签名以及您在上面看到的签名,签名如下所示:
"t=1557911017,v1=bebf499bcb35198b8bfaf22a68b8879574298f9f424e57ef292752e3ce21914d,v0=23402bb405bfd6bd2a13c310cfecda7ae1905609923d801fa4e8b872a4f82894"
据我从文档中了解到,获取上述原始请求正文所需的是bodyParser.raw({type: 'application/json'})
路由器的参数,我已经在那里了。
谁能看到缺失的部分?
这是因为您已经将 Express 应用设置为使用 bodyParser.json()
中间件,这与您在 webhook 路由中设置的 bodyParser.raw()
中间件冲突。
如果您删除 app.use(bodyParser.json());
行,您的 webhook 将按预期工作,但您的其余路由将没有已解析的正文,这并不理想。
我建议添加自定义中间件以根据路由选择 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);
}
});
然后在 webhook 路由上显式使用 bodyParser.raw()
中间件。
我正在尝试从 Stripe 设置一个 webhook 来处理 payment_intent.succeeded
事件,但我遇到了异常。这是我来自 Node 后端的代码(我已经提取了我希望的所有相关部分。如果需要其他任何内容,请告诉我):
const stripeWebHookSecret = 'whsec_WA0Rh4vAD3z0rMWy4kv2p6XXXXXXXXXX';
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);
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 });
}
});
}
到目前为止 good.When 我从 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"
我正在返回带有错误消息的签名以及您在上面看到的签名,签名如下所示:
"t=1557911017,v1=bebf499bcb35198b8bfaf22a68b8879574298f9f424e57ef292752e3ce21914d,v0=23402bb405bfd6bd2a13c310cfecda7ae1905609923d801fa4e8b872a4f82894"
据我从文档中了解到,获取上述原始请求正文所需的是bodyParser.raw({type: 'application/json'})
路由器的参数,我已经在那里了。
谁能看到缺失的部分?
这是因为您已经将 Express 应用设置为使用 bodyParser.json()
中间件,这与您在 webhook 路由中设置的 bodyParser.raw()
中间件冲突。
如果您删除 app.use(bodyParser.json());
行,您的 webhook 将按预期工作,但您的其余路由将没有已解析的正文,这并不理想。
我建议添加自定义中间件以根据路由选择 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);
}
});
然后在 webhook 路由上显式使用 bodyParser.raw()
中间件。