Rumah > Peranti teknologi > AI > Penyulitan dan penyahsulitan data antara muka Spring Boot harus direka bentuk seperti ini~

Penyulitan dan penyahsulitan data antara muka Spring Boot harus direka bentuk seperti ini~

王林
Lepaskan: 2023-05-13 10:58:05
ke hadapan
972 orang telah melayarinya

Artikel hari ini membincangkan isu keselamatan antara muka, yang melibatkan penyulitan antara muka dan penyahsulitan.

Spring Boot 接口数据加解密就该这样设计~

Selepas berkomunikasi dengan produk dan pelajar hadapan tentang permintaan luar, kami menyusun penyelesaian teknikal yang berkaitan adalah seperti berikut:

  • Minimumkan perubahan sebanyak mungkin Mempengaruhi logik perniagaan sebelumnya , memandangkan keselamatan kunci storan pada bahagian H5 agak rendah Ia lebih rendah, jadi dua set kunci diperuntukkan untuk H5 dan Android dan IOS
  • mestilah serasi dengan versi antara muka yang lebih rendah , dan antara muka yang baru dibangunkan kemudiannya tidak perlu serasi;
  • Pelayan, klien dan H5 memintas seragam dan penyahsulitan Terdapat penyelesaian matang dalam talian, dan Anda boleh mengikuti proses penyulitan dan penyahsulitan yang dilaksanakan dalam perkhidmatan lain
  • Gunakan AES untuk melonggarkan penyulitan . Memandangkan keselamatan kunci storan sisi H5 agak rendah, dua diperuntukkan untuk kunci H5 dan Android dan IOS
Kali ini melibatkan transformasi keseluruhan klien dan pelayan , antara muka baharu disatukan dengan awalan /rahsia/ untuk membezakannya

    Pemulihan mudah mengikut keperluan ini Soalan, tentukan dua objek, yang akan digunakan kemudian,
  • Kelas pengguna :
  • @Data
    public class User {
    private Integer id;
    private String name;
    private UserType userType = UserType.COMMON;
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime registerTime;
    }
    Salin selepas log masuk
  • Kelas penghitungan jenis pengguna:
  • @Getter
    @JsonFormat(shape = JsonFormat.Shape.OBJECT)
    public enum UserType {
    VIP("VIP用户"),
    COMMON("普通用户");
    private String code;
    private String type;
    
    UserType(String type) {
    this.code = name();
    this.type = type;
    }
    }
    Salin selepas log masuk
  • Bina contoh pertanyaan senarai pengguna yang mudah:
@RestController
@RequestMapping(value = {"/user", "/secret/user"})
public class UserController {
@RequestMapping("/list")
ResponseEntity<List<User>> listUser() {
List<User> users = new ArrayList<>();
User u = new User();
u.setId(1);
u.setName("boyka");
u.setRegisterTime(LocalDateTime.now());
u.setUserType(UserType.COMMON);
users.add(u);
ResponseEntity<List<User>> response = new ResponseEntity<>();
response.setCode(200);
response.setData(users);
response.setMsg("用户列表查询成功");
return response;
}
}
Salin selepas log masuk

panggilan: localhost:8080/user/list

Keputusan pertanyaan adalah seperti berikut, tidak salah:

{
 "code": 200,
 "data": [{
"id": 1,
"name": "boyka",
"userType": {
 "code": "COMMON",
 "type": "普通用户"
},
"registerTime": "2022-03-24 23:58:39"
 }],
 "msg": "用户列表查询成功"
}
Salin selepas log masuk

Pada masa ini, ControllerAdvice digunakan terutamanya untuk memintas permintaan dan badan tindak balas Ia mentakrifkan SecretRequestAdvice untuk menyulitkan permintaan dan SecretResponseAdvice untuk menyulitkan respons (situasi sebenar ialah. sedikit lebih rumit. Terdapat juga permintaan jenis GET dalam projek, dan Penapis disesuaikan untuk pemprosesan penyahsulitan permintaan yang berbeza).

Baiklah, terdapat banyak contoh penggunaan ControllerAdvice di Internet Saya akan menunjukkan kepada anda dua kaedah teras Saya percaya orang besar akan memahaminya dengan pantas. Kod di atas:

Penyahsulitan permintaan SecretRequestAdvice:

@ControllerAdvice
@Order(Ordered.HIGHEST_PRECEDENCE)
@Slf4j
public class SecretRequestAdvice extends RequestBodyAdviceAdapter {
@Override
public boolean supports(MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass){
return true;
}

@Override
public HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType) throws IOException {
//如果支持加密消息,进行消息解密。
String httpBody;
if (Boolean.TRUE.equals(SecretFilter.secretThreadLocal.get())) {
httpBody = decryptBody(inputMessage);
} else {
httpBody = StreamUtils.copyToString(inputMessage.getBody(), Charset.defaultCharset());
}
//返回处理后的消息体给messageConvert
return new SecretHttpMessage(new ByteArrayInputStream(httpBody.getBytes()), inputMessage.getHeaders());
}

/**
 * 解密消息体
 *
 * @param inputMessage 消息体
 * @return 明文
 */
private String decryptBody(HttpInputMessage inputMessage) throws IOException {
InputStream encryptStream = inputMessage.getBody();
String requestBody = StreamUtils.copyToString(encryptStream, Charset.defaultCharset());
// 验签过程
HttpHeaders headers = inputMessage.getHeaders();
if (CollectionUtils.isEmpty(headers.get("clientType"))
|| CollectionUtils.isEmpty(headers.get("timestamp"))
|| CollectionUtils.isEmpty(headers.get("salt"))
|| CollectionUtils.isEmpty(headers.get("signature"))) {
throw new ResultException(SECRET_API_ERROR, "请求解密参数错误,clientType、timestamp、salt、signature等参数传递是否正确传递");
}

String timestamp = String.valueOf(Objects.requireNonNull(headers.get("timestamp")).get(0));
String salt = String.valueOf(Objects.requireNonNull(headers.get("salt")).get(0));
String signature = String.valueOf(Objects.requireNonNull(headers.get("signature")).get(0));
String privateKey = SecretFilter.clientPrivateKeyThreadLocal.get();
ReqSecret reqSecret = JSON.parseObject(requestBody, ReqSecret.class);
String data = reqSecret.getData();
String newSignature = "";
if (!StringUtils.isEmpty(privateKey)) {
newSignature = Md5Utils.genSignature(timestamp + salt + data + privateKey);
}
if (!newSignature.equals(signature)) {
// 验签失败
throw new ResultException(SECRET_API_ERROR, "验签失败,请确认加密方式是否正确");
}

try {
String decrypt = EncryptUtils.aesDecrypt(data, privateKey);
if (StringUtils.isEmpty(decrypt)) {
decrypt = "{}";
}
return decrypt;
} catch (Exception e) {
log.error("error: ", e);
}
throw new ResultException(SECRET_API_ERROR, "解密失败");
}
}
Salin selepas log masuk

Penyulitan respons SecretResponseAdvice:

@ControllerAdvice
public class SecretResponseAdvice implements ResponseBodyAdvice {
private Logger logger = LoggerFactory.getLogger(SecretResponseAdvice.class);

@Override
public boolean supports(MethodParameter methodParameter, Class aClass){
return true;
}

@Override
public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse){
// 判断是否需要加密
Boolean respSecret = SecretFilter.secretThreadLocal.get();
String secretKey = SecretFilter.clientPrivateKeyThreadLocal.get();
// 清理本地缓存
SecretFilter.secretThreadLocal.remove();
SecretFilter.clientPrivateKeyThreadLocal.remove();
if (null != respSecret && respSecret) {
if (o instanceof ResponseBasic) {
// 外层加密级异常
if (SECRET_API_ERROR == ((ResponseBasic) o).getCode()) {
return SecretResponseBasic.fail(((ResponseBasic) o).getCode(), ((ResponseBasic) o).getData(), ((ResponseBasic) o).getMsg());
}
// 业务逻辑
try {
String data = EncryptUtils.aesEncrypt(JSON.toJSONString(o), secretKey);
// 增加签名
long timestamp = System.currentTimeMillis() / 1000;
int salt = EncryptUtils.genSalt();
String dataNew = timestamp + "" + salt + "" + data + secretKey;
String newSignature = Md5Utils.genSignature(dataNew);
return SecretResponseBasic.success(data, timestamp, salt, newSignature);
} catch (Exception e) {
logger.error("beforeBodyWrite error:", e);
return SecretResponseBasic.fail(SECRET_API_ERROR, "", "服务端处理结果数据异常");
}
}
}
return o;
}
}
Salin selepas log masuk

OK, demo kod sudah sedia, mari cuba:

请求方法:
localhost:8080/secret/user/list

header:
Content-Type:application/json
signature:55efb04a83ca083dd1e6003cde127c45
timestamp:1648308048
salt:123456
clientType:ANDORID

body体:
// 原始请求体
{
 "page": 1,
 "size": 10
}
// 加密后的请求体
{
 "data": "1ZBecdnDuMocxAiW9UtBrJzlvVbueP9K0MsIxQccmU3OPG92oRinVm0GxBwdlXXJ"
}

// 加密响应体:
{
"data": "fxHYvnIE54eAXDbErdrDryEsIYNvsOOkyEKYB1iBcre/QU1wMowHE2BNX/je6OP3NlsCtAeDqcp7J1N332el8q2FokixLvdxAPyW5Un9JiT0LQ3MB8p+nN23pTSIvh9VS92lCA8KULWg2nViSFL5X1VwKrF0K/dcVVZnpw5h227UywP6ezSHjHdA+Q0eKZFGTEv3IzNXWqq/otx5fl1gKQ==",
"code": 200,
"signature": "aa61f19da0eb5d99f13c145a40a7746b",
"msg": "",
"timestamp": 1648480034,
"salt": 632648
}

// 解密后的响应体:
{
 "code": 200,
 "data": [{
"id": 1,
"name": "boyka",
"registerTime": "2022-03-27T00:19:43.699",
"userType": "COMMON"
 }],
 "msg": "用户列表查询成功",
 "salt": 0
}
Salin selepas log masuk

OK, klien meminta penyulitan -> Mulakan permintaan -> Penyahsulitan pelayan -> Pemprosesan perniagaan -> Penyulitan tindak balas pelayan -> Paparan penyahsulitan pelanggan, nampaknya tiada masalah, tetapi sebenarnya, saya menghabiskan 2 jam untuk permintaan itu. Petang sebelum itu. Ia mengambil masa hampir sejam untuk menulis ujian tunjuk cara, dan kemudian memproses semua antara muka dengan cara yang bersatu Ia sepatutnya cukup untuk menyelesaikan sepanjang petang, dan memberitahu pelajar H5 dan Android untuk bersama-sama menyahpepijat pagi esok perkara, semua orang telah mendapati bahawa tiada apa-apa yang mencurigakan pada masa ini) , saya memang cuai pada masa itu dan terbalik...)

Keesokan harinya, pihak Android melaporkan bahawa terdapat masalah dengan penyulitan anda dan penyahsulitan. Format data yang dinyahsulit adalah berbeza daripada sebelumnya. Lihat lebih dekat, gosok, ada sesuatu yang tidak kena dengan jenis pengguna dan registerTime ini, dan saya mula berfikir: Apakah masalahnya? Selepas 1s, kedudukan awal adalah bahawa ia sepatutnya menjadi masalah dengan JSON.toJSONString dalam badan respons:

String data = EncryptUtils.aesEncrypt(JSON.toJSONString(o)),
Salin selepas log masuk

Nyahpepijat titik putus nyahpepijat Sudah pasti, terdapat masalah dengan penukaran JSON.toJSONString(o) . Kemudian apabila menukar JSON Adakah terdapat sifat lanjutan yang boleh dikonfigurasikan untuk menjana format bersiri yang dikehendaki? FastJson menyediakan kaedah lebihan semasa bersiri Cari salah satu daripada parameter "SerializerFeature" dan fikirkan tentangnya Parameter ini boleh dikonfigurasikan untuk bersiri, antaranya saya rasa ini agak relevan:

WriteEnumUsingToString,
WriteEnumUsingName,
UseISO8601DateFormat
Salin selepas log masuk
<🎜. >Untuk jenis penghitungan, lalainya ialah menggunakan WriteEnumUsingName (nama penghitungan). Satu lagi jenis WriteEnumUsingToString ialah kaedah re-toString, yang secara teorinya boleh ditukar kepada rupa yang diingini, iaitu seperti ini:

@Getter
@JsonFormat(shape = JsonFormat.Shape.OBJECT)
public enum UserType {
VIP("VIP用户"),
COMMON("普通用户");
private String code;
private String type;

UserType(String type) {
this.code = name();
this.type = type;
}

@Override
public String toString(){
return "{" +
""code":"" + name() + '"' +
", "type":"" + type + '"' +
'}';
}
}
Salin selepas log masuk

Data yang ditukar adalah daripada jenis rentetan "{"code":"COMMON", "type":"Normal User"}". Selepas memikirkannya, saya melihat kelas Pengguna dan UserType yang ditakrifkan pada permulaan artikel dan menandakan format siri data @JsonFormat Kemudian saya tiba-tiba teringat beberapa artikel yang saya lihat sebelum ini lalai. Baik, gunakan Jackson untuk melaksanakannya Baik, gantikan kaedah bersiri dalam SecretResponseAdvice:

String data = EncryptUtils.aesEncrypt(JSON.toJSONString(o), secretKey);
 换为:
String data =EncryptUtils.aesEncrypt(new ObjectMapper().writeValueAsString(o), secretKey);
Salin selepas log masuk

Jalankan semula dan mulakan:

{
 "code": 200,
 "data": [{
"id": 1,
"name": "boyka",
"userType": {
 "code": "COMMON",
 "type": "普通用户"
},
"registerTime": {
 "month": "MARCH",
 "year": 2022,
 "dayOfMonth": 29,
 "dayOfWeek": "TUESDAY",
 "dayOfYear": 88,
 "monthValue": 3,
 "hour": 22,
 "minute": 30,
 "nano": 453000000,
 "second": 36,
 "chronology": {
"id": "ISO",
"calendarType": "iso8601"
 }
}
 }],
 "msg": "用户列表查询成功"
}
Salin selepas log masuk

Jenis penghitungan jenis pengguna yang didekripsi adalah sama dengan versi tidak disulitkan, yang selesa , == Nampaknya tidak betul Mengapa registerTime menjadi seperti ini? Ia pada asalnya dalam format "2022-03-24 23:58:39". Terdapat banyak penyelesaian di Internet, tetapi apabila digunakan dalam keperluan semasa kami, ia adalah pengubahsuaian yang rugi, yang tidak digalakkan, jadi kami pergi. ke tapak web rasmi Jackson Cari dokumen yang berkaitan Sudah tentu, Jackson juga menyediakan konfigurasi bersiri ObjectMapper mulakan semula dan konfigurasikan objek ObjectMpper:

String DATE_TIME_FORMATTER = "yyyy-MM-dd HH:mm:ss";
ObjectMapper objectMapper = new Jackson2ObjectMapperBuilder()
.findModulesViaServiceLoader(true)
.serializerByType(LocalDateTime.class, new LocalDateTimeSerializer(
DateTimeFormatter.ofPattern(DATE_TIME_FORMATTER)))
.deserializerByType(LocalDateTime.class, new LocalDateTimeDeserializer(
DateTimeFormatter.ofPattern(DATE_TIME_FORMATTER)))
.build();
Salin selepas log masuk

Hasil penukaran:

.

{
 "code": 200,
 "data": [{
"id": 1,
"name": "boyka",
"userType": {
 "code": "COMMON",
 "type": "普通用户"
},
"registerTime": "2022-03-29 22:57:33"
 }],
 "msg": "用户列表查询成功"
}
Salin selepas log masuk

OK,和非加密版的终于一致了,完了吗?感觉还是可能存在些什么问题,首先业务代码的时间序列化需求不一样,有"yyyy-MM-dd hh:mm:ss"的,也有"yyyy-MM-dd"的,还可能其他配置思考不到位的,导致和之前非加密版返回数据不一致的问题,到时候联调测出来了也麻烦,有没有一劳永逸的办法呢?哎,这个时候如果你看过 Spring 源码的话,就应该知道spring框架自身是怎么序列化的,照着配置应该就行嘛,好像有点道理,我这里不从0开始分析源码了。

跟着执行链路,找到具体的响应序列化,重点就是RequestResponseBodyMethodProcessor,

protected <T> void writeWithMessageConverters(@Nullable T value, MethodParameter returnType, ServletServerHttpRequest inputMessage, ServletServerHttpResponse outputMessage) throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {
// 获取响应的拦截器链并执行beforeBodyWrite方法,也就是执行了我们自定义的SecretResponseAdvice中的beforeBodyWrite啦
body = this.getAdvice().beforeBodyWrite(body, returnType, selectedMediaType, converter.getClass(), inputMessage, outputMessage);
if (body != null) {
// 执行响应体序列化工作
 if (genericConverter != null) {
genericConverter.write(body, (Type)targetType, selectedMediaType, outputMessage);
 } else {
converter.write(body, selectedMediaType, outputMessage);
 }
}
Salin selepas log masuk

进而通过实例化的AbstractJackson2HttpMessageConverter对象找到执行序列化的核心方法

-> AbstractGenericHttpMessageConverter:
 
 public final void write(T t, @Nullable Type type, @Nullable MediaType contentType, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException {
...
this.writeInternal(t, type, outputMessage);
outputMessage.getBody().flush();
 
}
 -> 找到Jackson序列化 AbstractJackson2HttpMessageConverter:
 // 从spring容器中获取并设置的ObjectMapper实例
 protected ObjectMapper objectMapper;
 
 protected void writeInternal(Object object, @Nullable Type type, HttpOutputMessage outputMessage) throws IOException, HttpMessageNotWritableException {
MediaType contentType = outputMessage.getHeaders().getContentType();
JsonEncoding encoding = this.getJsonEncoding(contentType);
JsonGenerator generator = this.objectMapper.getFactory().createGenerator(outputMessage.getBody(), encoding);

this.writePrefix(generator, object);
Object value = object;
Class<?> serializationView = null;
FilterProvider filters = null;
JavaType javaType = null;
if (object instanceof MappingJacksonValue) {
 MappingJacksonValue container = (MappingJacksonValue)object;
 value = container.getValue();
 serializationView = container.getSerializationView();
 filters = container.getFilters();
}

if (type != null && TypeUtils.isAssignable(type, value.getClass())) {
 javaType = this.getJavaType(type, (Class)null);
}

ObjectWriter objectWriter = serializationView != null ? this.objectMapper.writerWithView(serializationView) : this.objectMapper.writer();
if (filters != null) {
 objectWriter = objectWriter.with(filters);
}

if (javaType != null && javaType.isContainerType()) {
 objectWriter = objectWriter.forType(javaType);
}

SerializationConfig config = objectWriter.getConfig();
if (contentType != null && contentType.isCompatibleWith(MediaType.TEXT_EVENT_STREAM) && config.isEnabled(SerializationFeature.INDENT_OUTPUT)) {
 objectWriter = objectWriter.with(this.ssePrettyPrinter);
}
// 重点进行序列化
objectWriter.writeValue(generator, value);
this.writeSuffix(generator, object);
generator.flush();
}
Salin selepas log masuk

那么,可以看出SpringMVC在进行响应序列化的时候是从容器中获取的ObjectMapper实例对象,并会根据不同的默认配置条件进行序列化,那处理方法就简单了,我也可以从Spring容器拿数据进行序列化啊。SecretResponseAdvice进行如下进一步改造:

@ControllerAdvice
public class SecretResponseAdvice implements ResponseBodyAdvice {

@Autowired
private ObjectMapper objectMapper;
 
@Override
public Object beforeBodyWrite(....){
.....
String dataStr =objectMapper.writeValueAsString(o);
String data = EncryptUtils.aesEncrypt(dataStr, secretKey);
.....
}
 }
Salin selepas log masuk

经测试,响应数据和非加密版万全一致啦,还有GET部分的请求加密,以及后面加解密惨遭跨域问题,后面有空再和大家聊聊。

Atas ialah kandungan terperinci Penyulitan dan penyahsulitan data antara muka Spring Boot harus direka bentuk seperti ini~. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Label berkaitan:
sumber:51cto.com
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan