xiejun
2024-11-01 80b6cbfc9c861469146318d0b3dd5f8b8b525b8a
Source/BladeX-Tool/blade-starter-api-crypto/src/main/java/org/springblade/core/api/crypto/core/ApiDecryptRequestBodyAdvice.java
¶Ô±ÈÐÂÎļþ
@@ -0,0 +1,87 @@
package org.springblade.core.api.crypto.core;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springblade.core.api.crypto.annotation.decrypt.ApiDecrypt;
import org.springblade.core.api.crypto.bean.CryptoInfoBean;
import org.springblade.core.api.crypto.bean.DecryptHttpInputMessage;
import org.springblade.core.api.crypto.config.ApiCryptoProperties;
import org.springblade.core.api.crypto.exception.DecryptBodyFailException;
import org.springblade.core.api.crypto.util.ApiCryptoUtil;
import org.springblade.core.tool.utils.ClassUtil;
import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.core.MethodParameter;
import org.springframework.core.annotation.Order;
import org.springframework.http.HttpInputMessage;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.lang.NonNull;
import org.springframework.util.StreamUtils;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.servlet.mvc.method.annotation.RequestBodyAdvice;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Type;
/**
 * è¯·æ±‚数据的加密信息解密处理<br>
 * æœ¬ç±»åªå¯¹æŽ§åˆ¶å™¨å‚数中含有<strong>{@link org.springframework.web.bind.annotation.RequestBody}</strong>
 * ä»¥åŠpackage为<strong><code>org.springblade.core.api.signature.annotation.decrypt</code></strong>下的注解有效
 *
 * @author licoy.cn, L.cm
 * @see RequestBodyAdvice
 */
@Slf4j
@Order(1)
@AutoConfiguration
@ControllerAdvice
@RequiredArgsConstructor
@ConditionalOnProperty(value = ApiCryptoProperties.PREFIX + ".enabled", havingValue = "true", matchIfMissing = true)
public class ApiDecryptRequestBodyAdvice implements RequestBodyAdvice {
   private final ApiCryptoProperties properties;
   @Override
   public boolean supports(MethodParameter methodParameter, @NonNull Type targetType, @NonNull Class<? extends HttpMessageConverter<?>> converterType) {
      return ClassUtil.isAnnotated(methodParameter.getMethod(), ApiDecrypt.class);
   }
   @Override
   public Object handleEmptyBody(Object body, @NonNull HttpInputMessage inputMessage, @NonNull MethodParameter parameter,
                          @NonNull Type targetType, @NonNull Class<? extends HttpMessageConverter<?>> converterType) {
      return body;
   }
   @NonNull
   @Override
   public HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, @NonNull MethodParameter parameter,
                                 @NonNull Type targetType, @NonNull Class<? extends HttpMessageConverter<?>> converterType) throws IOException {
      // åˆ¤æ–­ body æ˜¯å¦ä¸ºç©º
      InputStream messageBody = inputMessage.getBody();
      if (messageBody.available() <= 0) {
         return inputMessage;
      }
      byte[] decryptedBody = null;
      CryptoInfoBean cryptoInfoBean = ApiCryptoUtil.getDecryptInfo(parameter);
      if (cryptoInfoBean != null) {
         // base64 byte array
         byte[] bodyByteArray = StreamUtils.copyToByteArray(messageBody);
         decryptedBody = ApiCryptoUtil.decryptData(properties, bodyByteArray, cryptoInfoBean);
      }
      if (decryptedBody == null) {
         throw new DecryptBodyFailException("Decryption error, " +
            "please check if the selected source data is encrypted correctly." +
            " (解密错误,请检查选择的源数据的加密方式是否正确。)");
      }
      InputStream inputStream = new ByteArrayInputStream(decryptedBody);
      return new DecryptHttpInputMessage(inputStream, inputMessage.getHeaders());
   }
   @NonNull
   @Override
   public Object afterBodyRead(@NonNull Object body, @NonNull HttpInputMessage inputMessage, @NonNull MethodParameter parameter, @NonNull Type targetType, @NonNull Class<? extends HttpMessageConverter<?>> converterType) {
      return body;
   }
}