【加密专栏】OpenHarmony应用开发-加解密之AES128_CBC_PKCS5

简介

该文主要提供AES128|CBC|PKCS5加解密的最佳实践示例。

开发环境

DevEco Studio: DevEco Studio 5.1.0 Release(Build Version: 5.1.0.128)

系统: OpenHarmony 5.0.3.135

设备: DAYU200(rk3568)

最佳实践示例

ts 复制代码
// AES128|CBC|PKCS5
import { cryptoFramework } from '@kit.CryptoArchitectureKit';
import { buffer, util } from '@kit.ArkTS';

@Entry
@Component
struct CryptoArchitectureTest2 {

  encryptText: string = ''
  iv = this.genIvParamsSpec();
  base64Helper = new util.Base64Helper()
  @State originMessage: string = '123123123123131312313123'
  @State finalMessage: string = ''

  generateRandom(len: number): cryptoFramework.DataBlob {
    let rand = cryptoFramework.createRandom();
    let generateRandSync = rand.generateRandomSync(len);
    return generateRandSync;
  }

  genIvParamsSpec() {
    let ivBlob = this.generateRandom(16);
    let ivParamsSpec: cryptoFramework.IvParamsSpec = {
      algName: "IvParamsSpec",
      iv: ivBlob
    };
    return ivParamsSpec;
  }

  async encryptMessagePromise(symKey: cryptoFramework.SymKey, plainText: cryptoFramework.DataBlob) {
    let cipher = cryptoFramework.createCipher('AES128|CBC|PKCS5');
    await cipher.init(cryptoFramework.CryptoMode.ENCRYPT_MODE, symKey, this.iv);
    let cipherData = await cipher.doFinal(plainText);
    return cipherData;
  }

  async decryptMessagePromise(symKey: cryptoFramework.SymKey, cipherText: cryptoFramework.DataBlob) {
    let decoder = cryptoFramework.createCipher('AES128|CBC|PKCS5');
    await decoder.init(cryptoFramework.CryptoMode.DECRYPT_MODE, symKey, this.iv);
    let decryptData = await decoder.doFinal(cipherText);
    return decryptData;
  }

  async genSymKeyByData(symKeyData: Uint8Array) {
    let symKeyBlob: cryptoFramework.DataBlob = { data: symKeyData };
    let aesGenerator = cryptoFramework.createSymKeyGenerator('AES128');
    let symKey = await aesGenerator.convertKey(symKeyBlob);
    return symKey;
  }

  // Uint8Array转化Base64
  uint8ArrayToBase64(fileData: Uint8Array): string {
    return this.base64Helper.encodeToStringSync(fileData);
  }

  // Base64转化Uint8Array
  base64ToUint8Array(fileData: string): Uint8Array {
    return this.base64Helper.decodeSync(fileData);
  }

  // 将UTF-8编码转换成Unicode编码
  uint8ArrayToString(array: Uint8Array) {
    let out: string = '';
    let index: number = 0;
    let len: number = array.length;
    while (index < len) {
      let character = array[index++];
      switch(character >> 4) {
        case 0:
        case 1:
        case 2:
        case 3:
        case 4:
        case 5:
        case 6:
        case 7:
          out += String.fromCharCode(character);
          break;
        case 12:
        case 13:
          out += String.fromCharCode(((character & 0x1F) << 6) | (array[index++] & 0x3F));
          break;
        case 14:
          out += String.fromCharCode(((character & 0x0F) << 12) | ((array[index++] & 0x3F) << 6) | ((array[index++] & 0x3F) << 0));
          break;
        default:
          break;
      }
    }
    return out;
  }

  build() {
    Column({ space: 20 }) {
      Text('AES128|CBC|PKCS5')
        .fontSize(30)

      Button('加密')
        .fontSize(50)
        .onClick(async () => {
          try {
            let keyData = new Uint8Array([83, 217, 231, 76, 28, 113, 23, 219, 250, 71, 209, 210, 205, 97, 32, 159]);
            let symKey = await this.genSymKeyByData(keyData);
            console.info(`originMessage: ${this.originMessage}`);
            let plainText: cryptoFramework.DataBlob = { data: new Uint8Array(buffer.from(this.originMessage, 'utf-8').buffer) };
            console.info(`plainText: ${this.uint8ArrayToBase64(plainText.data)}`);
            let encryptText = await this.encryptMessagePromise(symKey, plainText);
            this.encryptText = this.uint8ArrayToBase64(encryptText.data);
            console.info(`encryptText: ${this.encryptText}`);
          } catch (error) {
            console.error(`AES CBC "${error}", error code: ${error.code}`);
          }
        })

      Text(`originMessage: ${this.originMessage}`)
        .fontSize(30)

      Button('解密')
        .fontSize(50)
        .onClick(async () => {
          try {
            let keyData = new Uint8Array([83, 217, 231, 76, 28, 113, 23, 219, 250, 71, 209, 210, 205, 97, 32, 159]);
            let symKey = await this.genSymKeyByData(keyData);
            let decryptText = await this.decryptMessagePromise(symKey, { data: this.base64ToUint8Array(this.encryptText) });
            console.info(`decryptText: ${JSON.stringify(this.uint8ArrayToBase64(decryptText.data))}`);
            this.finalMessage = this.uint8ArrayToString(decryptText.data)
            console.info(`finalMessage: ${this.finalMessage}`);
          } catch (error) {
            console.error(`AES CBC "${error}", error code: ${error.code}`);
          }
        })

      Text(`finalMessage: ${this.finalMessage}`)
        .fontSize(30)
    }
    .justifyContent(FlexAlign.Center)
    .alignItems(HorizontalAlign.Center)
    .height('100%')
    .width('100%')
  }
}
相关推荐
fanruitian3 小时前
uniapp android开发 测试板本与发行版本
前端·javascript·uni-app
rayufo3 小时前
【工具】列出指定文件夹下所有的目录和文件
开发语言·前端·python
RANCE_atttackkk3 小时前
[Java]实现使用邮箱找回密码的功能
java·开发语言·前端·spring boot·intellij-idea·idea
2501_944525545 小时前
Flutter for OpenHarmony 个人理财管理App实战 - 支出分析页面
android·开发语言·前端·javascript·flutter
李白你好5 小时前
Burp Suite插件用于自动检测Web应用程序中的未授权访问漏洞
前端
刘一说7 小时前
Vue 组件不必要的重新渲染问题解析:为什么子组件总在“无故”刷新?
前端·javascript·vue.js
徐同保7 小时前
React useRef 完全指南:在异步回调中访问最新的 props/state引言
前端·javascript·react.js
刘一说8 小时前
Vue 导航守卫未生效问题解析:为什么路由守卫不执行或逻辑失效?
前端·javascript·vue.js
一周七喜h8 小时前
在Vue3和TypeScripts中使用pinia
前端·javascript·vue.js
weixin_395448919 小时前
main.c_cursor_0202
前端·网络·算法