在支付系统的构建与管理中,返回码的设计和映射是确保交易顺畅进行的关键环节。本文将深入探讨支付系统返回码的重要性、常见问题及其解决方案,旨在帮助开发者和支付系统管理者避免潜在的陷阱,提升用户体验,并确保交易的准确性和安全性。
有些人喜欢用“返回码”,有些人喜欢用“错误码”。两者本质相同,都是用于标识通信或交易的状态。但我更倾向于使用“返回码”,因为它不仅涵盖错误状态,还包括成功状态,而成功并非错误,所以使用“返回码”更为合适。
我很喜欢探寻事物的本质,那么返回码的本质是什么?我觉得是解决2个问题:
返回码最核心的关注点也只有2个:
踩过的坑很多,大致可以归为以下几类:
支付同步请求渠道响应还没有回来,发起了查询,查询返回“订单不存在”,直接推进失败,但最后银行扣款成功。
退款同步请求渠道响应返回“系统异常”,直接推进到失败,但最后银行退款成功。
3. 外部渠道有双层返回码,没有做完整判断。比如第1层只表示接口是否成功(通信层面),第2层才是表示业务是否成功,但是只判断了接口层面,就推进了内部订单的业务状态。
4. 返回码制定过于笼统或太细。
外部商户对接支付平台,支付平台内部有自己的业务处理,同时还对接了外部的很多渠道,所以需要管理三套返回码:
为什么需要三层?主要有3个原因:
这部分建议直接参考微信支付、支付宝或者ISO20022标准,这几家代表了行业的最高水准。
一般来说最少有两个字段:resultCode和message,一个表示码,一个表示码的描述。
也可以增加一个参数result,使用S,F,U表示业务状态的成功、失败、未知。
如果是查询类接口,一定要明确说明是接口成功,还是业务成功。
支付平台内部也分了不同域,建议使用一个共同的规范,比如:RS+子系统编号+错误级别+具体返回码。具体如下图所示:
说明:
public interface IResultCode {
String getResultCode();
String getMessage();
}
public enum PaymentResultCode implements IResultCode {
SUCCESS( "0000", "success"),
FAIL("2998", "fail"),
SYSTEM_ERROR("2999","system error"),
// 额度相关 11XX
INSUFFICIENT_FUND("1101", "insufficient fund"),
// 风控相关 12XX
RISK_REJECTED("1201", "risk rejected"),
// DB相关 21XX ;
private static final String PREFIX = "RS";
private static final String SYSTEM_CODE = "101";
private String codeNumber;
private String message;
@Override
public String getResultCode() {
return PREFIX + SYSTEM_CODE + codeNumber;
}
@Override
public String getMessage() {
return message;
}
PaymentResultCode(String codeNumber, String message) {
this.codeNumber = codeNumber;
this.message = message;
}
}
每个渠道的返回码都是不一样的,所以需要设计外部渠道返回码映射到内部标准返回码。需要遵守几个原则:
具体的技术实现,通过使用映射表就足够,加到缓存中,增加运算速度。如果找不到映射关系,就全部转到一个默认的返回码上面,同时对这个默认返回码做监控,定期把这些没有做映射的返回码映射到正确的返回码上面去。避免应该把类似“余额不足”映射成了“系统异常请重试”的场景。
大部分团队都会监控成功率,只有少数团队会监控返回码或定期分析返回码。然而当交易量足够大时,成功率的波动可能只有0.5%,很难看出异常,而如果去分析返回码,则可以快速看出并定位问题。
一般来说,有几个建议:
卷用户体验和成功率时,往往需要于细微处见真章,而返回码的设计和映射就是如此。做得不好,轻则影响用户体验,重则资损。
希望对大家在设计标准返回码及映射时有所启发,也欢迎点赞转发。
这是《图解支付系统设计与实现》专栏系列文章中的第(48)篇。欢迎和我一起深入解码支付系统的方方面面。
作者:隐墨星辰,公众号:隐墨星辰
本文由 @隐墨星辰 原创发布于人人都是产品经理。未经作者许可,禁止转载
题图来自Unsplash,基于CC0协议
该文观点仅代表作者本人,人人都是产品经理平台仅提供信息存储空间服务