I wrote an idempotent article earlier, but just thinking about it is definitely not enough in a production environment. It just so happens that a colleague has been discussing how to do idempotent. So how can idempotent design be productively usable? Take a look at the following code:
Idempotent processing interface:
Public interface IdempotentProcessor {/** * handle idempotent execution code block ** @param optType the opttype
* @param optId the opt id
* @param initStatus the init status
* @param expireMs the expire ms
* @returnthe status enum */ StatusEnum process(String optType, String optId, StatusEnum initStatus, long expireMs); /** * Commit idempotent * @param optType the opttype
* @param optId the opt id
* @returnthe boolean */ boolean confirm(String optType, String optId); /** ** cancel idempotent * @param optType the opttype
* @param optId the opt id
* @return the boolean
*/
boolean cancel(String optType, String optId);
}
Copy the code
So if you look at the interface design above, you can see that when the process method is executed, it is idempotent and needs to confim once, and you can call Cancel to cancel the idempotent. Why? Because if it’s idempotent before a method is executed it’s a big deal if the subsequent method fails. Confirm and cancel are required to ensure success and failure. (VOICEover: Is there any TCC transaction in this design?)
Idempotent implementation class
public class ItemProcessor implements IdempotentProcessor {
private static final String CACHE_KEY = "dew:idempotent:item:";
@Override
public StatusEnum process(String optType, String optId, StatusEnum initStatus, long expireMs) {
if (Dew.cluster.cache.setnx(CACHE_KEY + optType + ":"+ optId, initStatus.toString(), expireMs / 1000)) {// If the value is set successfully, it does not exist beforereturn StatusEnum.NOT_EXIST;
} elseString status = dew.cluster.cache. get(CACHE_KEY + optType +)":" + optId);
if(status = = null | | status. IsEmpty ()) {/ / set the success, said before does not existreturn StatusEnum.NOT_EXIST;
} else {
return StatusEnum.valueOf(status);
}
}
}
@Override
public boolean confirm(String optType, String optId) {
long ttl = Dew.cluster.cache.ttl(CACHE_KEY + optType + ":" + optId);
if (ttl > 0) {
Dew.cluster.cache.setex(CACHE_KEY + optType + ":" + optId, StatusEnum.CONFIRMED.toString(), ttl);
}
return true;
}
@Override
public boolean cancel(String optType, String optId) {
Dew.cluster.cache.del(CACHE_KEY + optType + ":" + optId);
return true; }}Copy the code
Idempotent interceptor
public class IdempotentHandlerInterceptor extends HandlerInterceptorAdapter {
private DewIdempotentConfig dewIdempotentConfig;
/**
* Instantiates a new Idempotent handler interceptor.
*
* @param dewIdempotentConfig the dew idempotent config
*/
public IdempotentHandlerInterceptor(DewIdempotentConfig dewIdempotentConfig) {
this.dewIdempotentConfig = dewIdempotentConfig;
}
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
Idempotent idempotent = ((HandlerMethod) handler).getMethod().getAnnotation(Idempotent.class);
if (idempotent == null) {
returnsuper.preHandle(request, response, handler); } // Parameter Settings String optType ="[" + request.getMethod() + "]" + Dew.Info.name + "/" + request.getRequestURI();
String optIdFlag = StringUtils.isEmpty(idempotent.optIdFlag()) ? dewIdempotentConfig.getDefaultOptIdFlag() : idempotent.optIdFlag();
String optId = request.getHeader(optIdFlag);
if (StringUtils.isEmpty(optId)) {
optId = request.getParameter(optIdFlag);
}
if(stringutils.isempty (optId)) {// optId does not exist, indicating that the idempotent check is ignored and executed forciblyreturn super.preHandle(request, response, handler);
}
if(! DewIdempotent.existOptTypeInfo(optType)) { long expireMs = idempotent.expireMs() == -1 ? dewIdempotentConfig.getDefaultExpireMs() : idempotent.expireMs(); boolean needConfirm = idempotent.needConfirm(); StrategyEnum strategy = idempotent.strategy() == StrategyEnum.AUTO ? dewIdempotentConfig.getDefaultStrategy() : idempotent.strategy(); DewIdempotent.initOptTypeInfo(optType, needConfirm, expireMs, strategy); } switch (DewIdempotent.process(optType, optId)) {case NOT_EXIST:
return super.preHandle(request, response, handler);
case UN_CONFIRM:
ErrorController.error(request, response, 409,
"The last operation was still going on, please wait.", IdempotentException.class.getName());
return false;
case CONFIRMED:
ErrorController.error(request, response, 423,
"Resources have been processed, can't repeat the request.", IdempotentException.class.getName());
return false;
default:
return false; }}}Copy the code
The test case
Test Scenario 1
Manually cancel idempotent when the first request fails, and the second attempt succeeds
Code:
@GetMapping(value = "manual-confirm")
@Idempotent(expireMs = 5000)
public Resp<String> testManualConfirm(@RequestParam("str") String str) {
try {
if ("dew-test1".equals(str)){
throw new RuntimeException("Processing idempotent failure");
}
DewIdempotent.confirm();
} catch (Exception e) {
DewIdempotent.cancel();
return Resp.serverError(str + "Processing idempotent failure");
}
return Resp.success(str);
}
Copy the code
@Test
public void testConfirm() throws IOException, InterruptedException {// Idempotent unique key value // HashMap<String, String> header = new HashMap<String, String>() { { put(DewIdempotentConfig.DEFAULT_OPT_ID_FLAG,"0001"); }}; // String equals dew-test1 idempotent failure Resp<String> error = resp.generic ($.http.get(urlPre +)"manual-confirm? str=dew-test1", header), String.class);
System.out.println("Idempotent failure [errorCode=" + error.getCode() + "-errorMsg=" + error.getMessage() + "]"); Dew-test2 idempotent success Resp<String> success = resp.generic ($.http.get(urlPre +)"manual-confirm? str=dew-test2", header), String.class);
System.out.println("Idempotent success [code=" + success.getCode() + "-body=" + success.getBody() + "]");
}
Copy the code
Results:
Idempotent failed [errorCode= 500-errormsg =dew-test1 failed to handle idempotent] Idempotent succeeded [code=200-body=dew-test2]Copy the code
Test Scenario 2
The first request was idempotent successfully and the manual submission of Confim failed
Code:
@GetMapping(value = "manual-confirm")
@Idempotent(expireMs = 5000)
public Resp<String> testManualConfirm(@RequestParam("str") String str) {
try {
DewIdempotent.confirm();
} catch (Exception e) {
DewIdempotent.cancel();
return Resp.serverError(str + "Processing idempotent failure");
}
return Resp.success(str);
}
Copy the code
@Test
public void testConfirm() throws IOException, InterruptedException {// Idempotent unique key value // HashMap<String, String> header = new HashMap<String, String>() { { put(DewIdempotentConfig.DEFAULT_OPT_ID_FLAG,"0001"); }}; Dew-test1 idempotent success Resp<String> success = resp.generic ($.http.get(urlPre +)"manual-confirm? str=dew-test1", header), String.class);
System.out.println("Idempotent success [code=" + success.getCode() + "-body=" + success.getBody() + "]"); Thread.sleep(500); // String equals dew-test2 idempotent failure Resp<String> error = resp.generic ($.http.get(urlPre +)"manual-confirm? str=dew-test2", header), String.class);
System.out.println("Idempotent failure [errorCode=" + error.getCode() + "-errorMsg=" + error.getMessage() + "]");
}
Copy the code
Results:
[code=200-body=dew-test1] 2019-04-23 00:14:17.123 ERROR 8600 -- [nio-8080-exec-2] Ms. Developed. Core. Web. Error. ErrorController: Request the GET - / idempotent/manual - confirm 169.254.156.20, error 423: Resources have been processed, can[errorCode=423-errorMsg=[]Resources have been processed, can't repeat the request.]
Copy the code
The last
The project for this code and idea is called DEW and is a one-stop solution for microservices, providing: architecture guidelines, container first/compliant Frameworks for Spring Cloud and Istio, best practices, and Devops standardization processes. dew.ms
Source code Github address: github.com/gudaoxuri/d…
Friends can give this big guy’s project STAR wave