demo添加获取私钥功能

This commit is contained in:
peng fan 2024-08-07 18:15:39 +08:00
parent 61ccbc125c
commit 4a2394d57f
3 changed files with 174 additions and 24 deletions

23
package-lock.json generated
View File

@ -6,8 +6,10 @@
"": { "": {
"dependencies": { "dependencies": {
"crc": "^4.3.2", "crc": "^4.3.2",
"js-md5": "^0.8.3",
"pinia": "^2.2.0", "pinia": "^2.2.0",
"pinia-plugin-unistorage": "^0.1.2", "pinia-plugin-unistorage": "^0.1.2",
"sm-crypto": "^0.3.13",
"uview-plus": "^3.3.12" "uview-plus": "^3.3.12"
} }
}, },
@ -134,6 +136,18 @@
"optional": true, "optional": true,
"peer": true "peer": true
}, },
"node_modules/js-md5": {
"version": "0.8.3",
"resolved": "https://registry.npmjs.org/js-md5/-/js-md5-0.8.3.tgz",
"integrity": "sha512-qR0HB5uP6wCuRMrWPTrkMaev7MJZwJuuw4fnwAzRgP4J4/F8RwtodOKpGp4XpqsLBFzzgqIO42efFAyz2Et6KQ==",
"license": "MIT"
},
"node_modules/jsbn": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/jsbn/-/jsbn-1.1.0.tgz",
"integrity": "sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A==",
"license": "MIT"
},
"node_modules/pinia": { "node_modules/pinia": {
"version": "2.2.0", "version": "2.2.0",
"resolved": "https://registry.npmjs.org/pinia/-/pinia-2.2.0.tgz", "resolved": "https://registry.npmjs.org/pinia/-/pinia-2.2.0.tgz",
@ -172,6 +186,15 @@
"integrity": "sha512-OwpTSOfy6xSs1+pwcNrv0RBMOzI39Lp3qQKUTPVVPRjCdNa5JH/oPRiqsesIskK8TVgmRiHwO4KXlV2Li9dANA==", "integrity": "sha512-OwpTSOfy6xSs1+pwcNrv0RBMOzI39Lp3qQKUTPVVPRjCdNa5JH/oPRiqsesIskK8TVgmRiHwO4KXlV2Li9dANA==",
"license": "MIT" "license": "MIT"
}, },
"node_modules/sm-crypto": {
"version": "0.3.13",
"resolved": "https://registry.npmjs.org/sm-crypto/-/sm-crypto-0.3.13.tgz",
"integrity": "sha512-ztNF+pZq6viCPMA1A6KKu3bgpkmYti5avykRHbcFIdSipFdkVmfUw2CnpM2kBJyppIalqvczLNM3wR8OQ0pT5w==",
"license": "MIT",
"dependencies": {
"jsbn": "^1.1.0"
}
},
"node_modules/tiny-emitter": { "node_modules/tiny-emitter": {
"version": "2.1.0", "version": "2.1.0",
"resolved": "https://registry.npmjs.org/tiny-emitter/-/tiny-emitter-2.1.0.tgz", "resolved": "https://registry.npmjs.org/tiny-emitter/-/tiny-emitter-2.1.0.tgz",

View File

@ -1,8 +1,10 @@
{ {
"dependencies": { "dependencies": {
"crc": "^4.3.2", "crc": "^4.3.2",
"js-md5": "^0.8.3",
"pinia": "^2.2.0", "pinia": "^2.2.0",
"pinia-plugin-unistorage": "^0.1.2", "pinia-plugin-unistorage": "^0.1.2",
"sm-crypto": "^0.3.13",
"uview-plus": "^3.3.12" "uview-plus": "^3.3.12"
} }
} }

View File

@ -1,7 +1,6 @@
<template> <template>
<view> <view>
<button @click="getList">获取设备列表</button> <button @click="getList">获取设备列表</button>
<button @click="stopGetList">停止获取</button>
<view v-if="showList"> <view v-if="showList">
<button class="device" v-for="item in list" <button class="device" v-for="item in list"
:key="item.deviceId" @click="connect(item)">{{item.name}} :key="item.deviceId" @click="connect(item)">{{item.name}}
@ -10,28 +9,27 @@
</view> </view>
<view v-else> <view v-else>
<button @click="getPublicKey">获取锁公钥</button> <button @click="getPublicKey">获取锁公钥</button>
<button @click="getTest">获取锁私钥</button> <button @click="getCommKey">获取锁私钥</button>
</view> </view>
</view> </view>
</template> </template>
<script> <script>
import crc from 'crc' import crc from 'crc'
import { md5 } from 'js-md5'
import { sm4 } from 'sm-crypto'
export default { export default {
data() { data() {
return { return {
list: [], list: [],
getting: false, getting: false,
// deviceId: '6C3CD148-F456-1C61-F35E-C176B0CC644A',
// serviceId: '0000FFF0-0000-1000-8000-00805F9B34FB',
// characteristicId1: '0000FFF1-0000-1000-8000-00805F9B34FB',
// characteristicId2: '0000FFF2-0000-1000-8000-00805F9B34FB',
deviceId: '', deviceId: '',
serviceId: '', serviceId: '',
characteristicId1: '', characteristicId1: '',
characteristicId2: '', characteristicId2: '',
lockId: '', lockId: '',
publicKey: new Uint8Array(16),
showList: true, showList: true,
number: 1 number: 1
} }
@ -61,10 +59,54 @@ export default {
}) })
}, },
methods: { methods: {
GetCommKey() { // Uint8Array
const userId = 294 hexToUint8Array(hex) {
if (hex.length % 2 !== 0) {
throw new Error('Invalid hex string')
}
const array = new Uint8Array(hex.length / 2)
for (let i = 0; i < hex.length; i += 2) {
array[i / 2] = parseInt(hex.substr(i, 2), 16)
}
return array
},
//
getCommKey() {
const cmdId = 0x3091
const keyId = '0'
const authUserId = '294'
const nowTime = parseInt(new Date().getTime() / 1000)
let buffer = new ArrayBuffer(54) const originAuthCodeLen = keyId.length + authUserId.length + 4 + 16
let authCodeBuffer = new ArrayBuffer(originAuthCodeLen)
let authCodeBinaryData = new Uint8Array(authCodeBuffer)
for(let i=0; i < keyId.length; i++){
authCodeBinaryData[i] = keyId.charCodeAt(i)
}
for(let i=0; i < authUserId.length; i++){
authCodeBinaryData[keyId.length + i] = authUserId.charCodeAt(i)
}
const number = keyId.length + authUserId.length
authCodeBinaryData[number + 0] = (nowTime & 0xff000000) >> 24
authCodeBinaryData[number + 1] = (nowTime & 0xff0000) >> 16
authCodeBinaryData[number + 2] = (nowTime & 0xff00) >> 8
authCodeBinaryData[number + 3] = (nowTime & 0xff)
for(let i=0; i < 16; i++){
authCodeBinaryData[number + 4 + i] = this.publicKey[i]
}
console.log('md5前的数据', Array.from(authCodeBinaryData, byte =>
byte.toString(10)).join(','))
const codeMd5 = md5(authCodeBinaryData)
console.log('md5后的数据', codeMd5, codeMd5.length)
const bufferLength = codeMd5.length + 12 + 107
let buffer = new ArrayBuffer(bufferLength)
let binaryData = new Uint8Array(buffer) let binaryData = new Uint8Array(buffer)
// //
@ -82,8 +124,82 @@ export default {
this.number++ this.number++
// //
binaryData[7] = 0x22
//
binaryData[10] = (codeMd5.length + 107) / 256
binaryData[11] = (codeMd5.length + 107) % 256
//
binaryData[12] = 0x3091 / 256
binaryData[13] = 0x3091 % 256
for(let i=0; i < this.lockId.length; i++){
binaryData[14 + i] = this.lockId.charCodeAt(i)
}
let flagNumber = 14 + 40
for (let i = 0; i < keyId.length; i++) {
binaryData[flagNumber + i] = keyId.charCodeAt(i)
}
flagNumber += 40
for (let i = 0; i < authUserId.length; i++) {
binaryData[flagNumber + i] = authUserId.charCodeAt(i)
}
flagNumber += 20
binaryData[flagNumber + 0] = (nowTime & 0xff000000) >> 24
binaryData[flagNumber + 1] = (nowTime & 0xff0000) >> 16
binaryData[flagNumber + 2] = (nowTime & 0xff00) >> 8
binaryData[flagNumber + 3] = (nowTime & 0xff)
flagNumber += 4
binaryData[flagNumber] = codeMd5.length
flagNumber += 1
for (let i = 0; i < codeMd5.length; i++) {
binaryData[flagNumber + i] = codeMd5.charCodeAt(i)
}
binaryData[8] = Math.ceil((bufferLength - 12)/16) * 16 / 256
binaryData[9] = Math.ceil((bufferLength - 12)/16) * 16 % 256
let cebBuffer = new ArrayBuffer(bufferLength - 12)
let cebBinaryData = new Uint8Array(cebBuffer)
for(let i = 0; i < bufferLength - 12; i++){
cebBinaryData[i] = binaryData[i + 12]
}
const key = new Uint8Array(16)
for (let i = 0; i < this.lockId.length; i++) {
key[i] = this.lockId.charCodeAt(i)
}
console.log('未加密的数据', Array.from(cebBinaryData))
const encrypted = sm4.encrypt(cebBinaryData, key, { mode: 'ecb' })
const newBinaryData = this.hexToUint8Array(encrypted)
console.log('ecb加密后的数据', Array.from(newBinaryData))
const encrypteBuffer = new ArrayBuffer(12 + newBinaryData.length)
const encrypteBinaryData = new Uint8Array(encrypteBuffer)
for(let i = 0; i< 12; i++){
encrypteBinaryData[i] = binaryData[i]
}
for(let i = 12; i < encrypteBinaryData.length; i++){
encrypteBinaryData[i] = newBinaryData[i - 12]
}
console.log('crc前的数据', Array.from(encrypteBinaryData))
const resultBuffer = new ArrayBuffer(encrypteBinaryData.length + 2)
const resultBinaryData = new Uint8Array(resultBuffer)
for(let i = 0; i< encrypteBinaryData.length; i++){
resultBinaryData[i] = encrypteBinaryData[i]
}
resultBinaryData[encrypteBinaryData.length] = crc.crc16kermit(encrypteBuffer) / 256
resultBinaryData[1 + encrypteBinaryData.length] = crc.crc16kermit(encrypteBuffer) % 256
this.writeBLECharacteristicValue(resultBuffer)
}, },
//
getPublicKey() { getPublicKey() {
let buffer = new ArrayBuffer(54) let buffer = new ArrayBuffer(54)
let binaryData = new Uint8Array(buffer) let binaryData = new Uint8Array(buffer)
@ -131,6 +247,7 @@ export default {
newBinaryData[55] = crc.crc16kermit(buffer) % 256 newBinaryData[55] = crc.crc16kermit(buffer) % 256
this.writeBLECharacteristicValue(newBuffer) this.writeBLECharacteristicValue(newBuffer)
}, },
//
writeBLECharacteristicValue(data) { writeBLECharacteristicValue(data) {
const that = this const that = this
console.log('写入设备的设备 ID',that.deviceId) console.log('写入设备的设备 ID',that.deviceId)
@ -140,9 +257,7 @@ export default {
let Uint8ArrayData = new Uint8Array(data) let Uint8ArrayData = new Uint8Array(data)
const hexString = Array.from(Uint8ArrayData, byte => console.log('未加工的数据', Array.from(Uint8ArrayData))
byte.toString(10)).join(',')
console.log('未加工的数据', hexString)
for(let i = 0; i < Math.ceil(data.byteLength / 20); i++){ for(let i = 0; i < Math.ceil(data.byteLength / 20); i++){
let buffer = new ArrayBuffer(20) let buffer = new ArrayBuffer(20)
@ -170,6 +285,7 @@ export default {
}) })
} }
}, },
//
notifyBLECharacteristicValueChange() { notifyBLECharacteristicValueChange() {
const that = this const that = this
uni.notifyBLECharacteristicValueChange({ uni.notifyBLECharacteristicValueChange({
@ -184,13 +300,29 @@ export default {
console.log('设备特征值改变') console.log('设备特征值改变')
console.log(res) console.log(res)
let binaryData = new Uint8Array(res.value) let binaryData = new Uint8Array(res.value)
const hexString = Array.from(binaryData, byte => console.log('ecb解密前的数据', Array.from(binaryData))
byte.toString(10)).join(',') //
console.log(hexString) if(binaryData[7] === 0x20) {
if(binaryData[12] === 48 && binaryData[13] === 144){
const publicKey = binaryData.slice(15, 31)
console.log('公钥', Array.from(publicKey))
that.publicKey = publicKey
}
} else if(binaryData[7] === 0x22) {
const cebBinaryData = binaryData.slice(12, 28)
console.log('sm4返回参数', Array.from(cebBinaryData))
const key = new Uint8Array(16)
for (let i = 0; i < that.lockId.length; i++) {
key[i] = that.lockId.charCodeAt(i)
}
const decrypted = sm4.decrypt(cebBinaryData, key, { mode: 'ecb', output: 'array' })
console.log('ecb解密后的数据', decrypted)
}
}) })
} }
}) })
}, },
//
connect(item) { connect(item) {
const that = this const that = this
if(item.pair) { if(item.pair) {
@ -251,6 +383,7 @@ export default {
} }
}) })
}, },
//
startGetList() { startGetList() {
const that = this const that = this
this.getting = true this.getting = true
@ -265,15 +398,7 @@ export default {
} }
}) })
}, },
stopGetList(){ //
const that = this
uni.stopBluetoothDevicesDiscovery({
success: function () {
that.getting = false
console.log('停止获取设备列表成功')
}
})
},
getList() { getList() {
const that = this const that = this
uni.getBluetoothDevices({ uni.getBluetoothDevices({