Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >在iOS/Swift中创建并导出为base64的RSA公钥

在iOS/Swift中创建并导出为base64的RSA公钥
EN

Stack Overflow用户
提问于 2018-12-23 10:50:04
回答 2查看 3.7K关注 0票数 9

TL;DR:在iOS中生成并存储在密钥链中的RSA公钥,导出为base64并发送到java后端,是不被识别的。

我正在iOS应用程序中实现聊天加密功能,我使用对称+非对称密钥来处理它。

在后端,我使用用户的公钥加密用于加密和解密消息的对称密钥,而不需要过多的细节。

我创建了两个框架,分别用Swift和Java (后端)来处理密钥生成、加密、解密等。我还为它们进行了测试,所以我100%地按照预期的方式工作。

但是,后端似乎无法识别从iOS传递的公钥的格式。使用RSA双方,这是我在Swift中用来生成密钥的代码:

代码语言:javascript
运行
AI代码解释
复制
// private key parameters
static let privateKeyParams: [String : Any] = [
        kSecAttrIsPermanent as String: true,
        kSecAttrApplicationTag as String: "..." // I have a proper unique tag here
]

// public  key parameters
static let publicKeyParams: [String : Any] = [
        kSecAttrIsPermanent as String: true,
        kSecAttrApplicationTag as String: "..." // I have a proper unique tag here
]

// global parameters for our key generation
static let keyCreationParameters: [String : Any] = [
        kSecAttrKeyType as String: kSecAttrKeyTypeRSA,
        kSecAttrKeySizeInBits as String: 2048,
        kSecPublicKeyAttrs as String: publicKeyParams,
        kSecPrivateKeyAttrs as String: privateKeyParams
]

...

var publicKey, privateKey: SecKey?
let status = SecKeyGeneratePair(Constants.keyCreationParameters as CFDictionary, &publicKey, &privateKey)

我用镜面密码来读取钥匙链上的钥匙。

这是一段代码,用于将公钥导出为base64字符串:

代码语言:javascript
运行
AI代码解释
复制
extension SecKey {
  func asBase64() throws -> String {
    var dataPtr: CFTypeRef?
    let query: [String:Any] = [
      kSecClass as String: kSecClassKey,
      kSecAttrApplicationTag as String: "...", // Same unique tag here
      kSecAttrKeyType as String: kSecAttrKeyTypeRSA,
      kSecReturnData as String: kCFBooleanTrue
    ]
    let result = SecItemCopyMatching(query as CFDictionary, &dataPtr)

    switch (result, dataPtr) {
    case (errSecSuccess, .some(let data)):
      // convert to Base64 string
      let base64PublicKey = data.base64EncodedString(options: [])
      return base64PublicKey
    default:
      throw CryptoError.keyConversionError
    }
  }
}

在后端级别,我使用这段Java代码将base64字符串转换为公钥:

代码语言:javascript
运行
AI代码解释
复制
public PublicKey publicKeyFrom(String data) throws NoSuchAlgorithmException, InvalidKeySpecException {
    byte[] publicBytes = Base64.decodeBase64(data);
    X509EncodedKeySpec keySpec = new X509EncodedKeySpec(publicBytes);
    KeyFactory keyFactory = KeyFactory.getInstance("RSA");
    return keyFactory.generatePublic(keySpec);
}

但是,这在最后一行失败了,但有以下例外:

代码语言:javascript
运行
AI代码解释
复制
java.security.spec.InvalidKeySpecException: java.security.InvalidKeyException: IOException: algid parse error, not a sequence

在进行一些手动调试时,我注意到公钥的格式是不同的--当我在iOS中生成一个密钥,然后导出为base 64时,如下所示:

代码语言:javascript
运行
AI代码解释
复制
MIIBCgKCAQEA4M/bRDdH0f6qFIXxOg13RHka+g4Yv8u9PpPp1IR6pSwrM1aq8B6cyKRwnLe/MOkvODvDfJzvGXGQ01zSTxYWAW1B4uc/NCEemCmZqMosSB/VUJdNxxWtt2hJxpz06hAawqV+6HmweAB2dUn9tDEsQLsNHdwYouOKpyRZGimcF9qRFn1RjR0Q54sUh1tQAj/EwmgY2S2bI5TqtZnZw7X7Waji7wWi6Gz88IkuzLAzB9VBNDeV1cfJFiWsZ/MIixSvhpW3dMNCrJShvBouIG8nS+vykBlbFVRGy3gJr8+OcmIq5vuHVhqrWwHNOs+WR87K/qTFO/CB7MiyiIV1b1x5DQIDAQAB

对于总共360个字符,而在Java (仍在使用RSA)中进行相同操作时,如下所示:

代码语言:javascript
运行
AI代码解释
复制
MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCAAnWO4BXUGP0qM3Op36YXkWNxb4I2pPZuZ7jJtfUO7v+IO1mq43WzNaxLqqLPkTnMrv2ACRDK55vin+leQlL1z0LzVxjtZ9F6pajQo1r7PqBlL5N8bzBFKpagEf0QfyHPw0/0kG9DMnvQ+Im881QyN2zdl33wp5Fi+jRT7cunFQIDAQAB

长度为216个字符。

我不知道出了什么问题--显然,如果iOS在另一个键中处理密钥,并且需要特殊处理才能与其他人交谈,我就不会感到惊讶了。

有什么想法吗?

EN

回答 2

Stack Overflow用户

发布于 2019-06-03 00:37:44

在将iOS应用程序连接到Java后端时,我们遇到了完全相同的问题。pedrofb提到的CryptoExportImportManager也帮助了我们,这太棒了。但是,CryptoExportImportManager类中的代码有点详细,可能很难维护。这是因为在向DER编码中添加新组件时使用的是自顶向下的方法。因此,必须预先计算长度字段所包含的数字(即在定义适用长度的内容之前)。因此,我创建了一个新的类,我们现在使用它来转换RSA公钥的DER编码:

代码语言:javascript
运行
AI代码解释
复制
class RSAKeyEncoding: NSObject {

  // ASN.1 identifiers
  private let bitStringIdentifier: UInt8 = 0x03
  private let sequenceIdentifier: UInt8 = 0x30

  // ASN.1 AlgorithmIdentfier for RSA encryption: OID 1 2 840 113549 1 1 1 and NULL
  private let algorithmIdentifierForRSAEncryption: [UInt8] = [0x30, 0x0d, 0x06,
    0x09, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x01, 0x01, 0x01, 0x05, 0x00]

  /// Converts the DER encoding of an RSA public key that is either fetched from the
  /// keychain (e.g. by using `SecItemCopyMatching(_:_:)`) or retrieved in another way
  /// (e.g. by using `SecKeyCopyExternalRepresentation(_:_:)`), to a format typically
  /// used by tools and programming languages outside the Apple ecosystem (such as
  /// OpenSSL, Java, PHP and Perl). The DER encoding of an RSA public key created by
  /// iOS is represented with the ASN.1 RSAPublicKey type as defined by PKCS #1.
  /// However, many systems outside the Apple ecosystem expect the DER encoding of a
  /// key to be represented with the ASN.1 SubjectPublicKeyInfo type as defined by
  /// X.509. The two types are related in a way that if the SubjectPublicKeyInfo’s
  /// algorithm field contains the rsaEncryption object identifier as defined by
  /// PKCS #1, the subjectPublicKey field shall contain the DER encoding of an
  /// RSAPublicKey type.
  ///
  /// - Parameter rsaPublicKeyData: A data object containing the DER encoding of an
  ///     RSA public key, which is represented with the ASN.1 RSAPublicKey type.
  /// - Returns: A data object containing the DER encoding of an RSA public key, which
  ///     is represented with the ASN.1 SubjectPublicKeyInfo type.
  func convertToX509EncodedKey(_ rsaPublicKeyData: Data) -> Data {
    var derEncodedKeyBytes = [UInt8](rsaPublicKeyData)

    // Insert ASN.1 BIT STRING bytes at the beginning of the array
    derEncodedKeyBytes.insert(0x00, at: 0)
    derEncodedKeyBytes.insert(contentsOf: lengthField(of: derEncodedKeyBytes), at: 0)
    derEncodedKeyBytes.insert(bitStringIdentifier, at: 0)

    // Insert ASN.1 AlgorithmIdentifier bytes at the beginning of the array
    derEncodedKeyBytes.insert(contentsOf: algorithmIdentifierForRSAEncryption, at: 0)

    // Insert ASN.1 SEQUENCE bytes at the beginning of the array
    derEncodedKeyBytes.insert(contentsOf: lengthField(of: derEncodedKeyBytes), at: 0)
    derEncodedKeyBytes.insert(sequenceIdentifier, at: 0)

    return Data(derEncodedKeyBytes)
  }

  private func lengthField(of valueField: [UInt8]) -> [UInt8] {
    var length = valueField.count

    if length < 128 {
      return [ UInt8(length) ]
    }

    // Number of bytes needed to encode the length
    let lengthBytesCount = Int((log2(Double(length)) / 8) + 1)

    // First byte encodes the number of remaining bytes in this field
    let firstLengthFieldByte = UInt8(128 + lengthBytesCount)

    var lengthField: [UInt8] = []
    for _ in 0..<lengthBytesCount {
      // Take the last 8 bits of length
      let lengthByte = UInt8(length & 0xff)
      // Insert them at the beginning of the array
      lengthField.insert(lengthByte, at: 0)
      // Delete the last 8 bits of length
      length = length >> 8
    }

    // Insert firstLengthFieldByte at the beginning of the array
    lengthField.insert(firstLengthFieldByte, at: 0)

    return lengthField
  }
}

用法

您可以在函数asBase64()中使用这个类,如下所示:

代码语言:javascript
运行
AI代码解释
复制
extension SecKey {
  func asBase64() throws -> String {
    var dataPtr: CFTypeRef?
    let query: [String:Any] = [
      kSecClass as String: kSecClassKey,
      kSecAttrApplicationTag as String: "...", // Same unique tag here
      kSecAttrKeyType as String: kSecAttrKeyTypeRSA,
      kSecReturnData as String: kCFBooleanTrue
    ]
    let result = SecItemCopyMatching(query as CFDictionary, &dataPtr)

    switch (result, dataPtr) {
    case (errSecSuccess, .some(let data)):

      // convert to X509 encoded key
      let convertedData = RSAKeyEncoding().convertToX509EncodedKey(data)

      // convert to Base64 string
      let base64PublicKey = convertedData.base64EncodedString(options: [])
      return base64PublicKey
    default:
      throw CryptoError.keyConversionError
    }
  }
}

更新-其他问题

在使用上述课程一段时间后,我们偶然发现了另一个问题。有时候,从密钥链中获取的公钥似乎是无效的,因为由于某种原因,它的大小增加了。这种行为与问题中描述的结果相匹配(尽管在我们的例子中,Base64编码密钥已经增长到392个字符,而不是360个字符)。不幸的是,我们没有找到这种奇怪行为的确切原因,但我们找到了两个解决方案。第一个解决方案是在定义查询时指定kSecAttrKeySizeInBitskSecAttrEffectiveKeySize,如下代码段所示:

代码语言:javascript
运行
AI代码解释
复制
let keySize = ... // Key size specified when storing the key, for example: 2048

let query: [String: Any] = [
    kSecAttrKeySizeInBits as String: keySize,
    kSecAttrEffectiveKeySize as String: keySize,
    ... // More attributes
]

var dataPtr: CFTypeRef?

let result = SecItemCopyMatching(query as CFDictionary, &dataPtr)

第二种解决方案是,在添加具有相同标记的新键之前,始终从密钥链(如果有的话)中删除旧键。

更新-替代解决方案

我在GitHub上发布了GitHub,它可以作为上述类的替代。

参考文献

ASN.1、BER和DER子集的Layman指南

RFC 5280 (X.509 v3)

RFC 8017 (PKCS #1 v2.2)

我发现的一些代码在创建这里函数时给了我灵感。

票数 7
EN

Stack Overflow用户

发布于 2018-12-23 11:26:30

Java需要一个以DER格式编码的公钥。不幸的是,iOS不支持这种标准格式,而且还需要额外的转换(我不知道这在最新版本的needed中是否会有所改进)

参见我的答案这里您可以使用CryptoExportImportManager转换密钥

代码语言:javascript
运行
AI代码解释
复制
func exportPublicKeyToDER(keyId:String) -> NSData?{

    let publicKey = loadKeyStringFromKeyChainAsNSData(PUBLIC_KEY + keyId)
    let keyType = kSecAttrKeyTypeRSA
    let keySize = 2048
    let exportImportManager = CryptoExportImportManager()
    if let exportableDERKey = exportImportManager.exportPublicKeyToDER(publicKey, keyType: keyType as String, keySize: keySize) {
        return exportableDERKey
    } else {
        return nil
    }
}
票数 5
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/53906275

复制
相关文章
Python27 反射
图中输入choice的内容是一个字符串,正常调用d.eat()这可不是一个字符串。 报错提示Dog中不存在attribute choice(字符串)
py3study
2020/01/06
4850
Python27 反射
【编程基础】C语言从源程序到可执行程序
一.0,1、文本信息和字符编码 所有的信息在计算机中都是以0、1及其组合形式存在。文本信息也不例外。文本信息是以人类容易理解的方式来呈现信息。 计算机是在美国诞生的,英文26个字母加上其他符号只有128个,只用7个bit便可以完全表示所有符号。用8个bit,及一个byte来表示一个符号的方式就叫做ascii编码。对应的有ascii码表。 比如说要在计算机中表示"i love you"这个信息,采用ascii编码方式,那么在计算机中,那就是69 20 6c 6f 76 65 20 79 6f
程序员互动联盟
2018/03/13
1.3K0
.NET框架设计(高级框架架构模式)—钝化程序、逻辑冻结、冻结程序的延续、瞬间转移
本文介绍了如何利用领域驱动设计思想设计可恢复性语句,通过领域事件、领域模型、领域服务、领域事件监听器以及可恢复语句管理器,实现了一套可恢复性语句的框架。同时,本文还提供了一个示例,演示了如何使用该框架来实现可恢复性语句。
王清培
2018/01/08
8890
.NET框架设计(高级框架架构模式)—钝化程序、逻辑冻结、冻结程序的延续、瞬间转移
进程冻结
进程冻结技术(freezing of tasks)是指在系统hibernate或者suspend的时候,将用户进程和部分内核线程置于“可控”的暂停状态。
233333
2023/03/08
1.3K0
进程冻结
冻结计划
大多数SQL语句都有一个关联的查询计划。查询计划是在准备SQL语句时创建的。默认情况下,添加索引和重新编译类等操作会清除此查询计划。下次调用查询时,将重新准备查询并创建新的查询计划。冻结计划使可以跨编译保留(冻结)现有查询计划。查询执行使用冻结的计划,而不是执行新的优化并生成新的查询计划。
用户7741497
2022/06/08
1.9K0
账户冻结说明
开通按量计费(后付费)资源时,会冻结您账户上的部分资金(余额小于冻结费用则无法开通);在您主动释放资源或由于您账户欠费导致资源释放后会解冻该部分资金;冻结的金额没有实际消耗,在冻结状态下无法使用,解冻后恢复使用。
☆~oo~☆
2022/09/21
1.8K0
ETH被冻结_微信冻结显示什么界面
如果我们的电脑在启动挖矿软件的时候,发现界面有停顿,Miner都会冻结。有时矿工会随机冻结,直到按下任何键。
全栈程序员站长
2022/11/04
1.3K0
ETH被冻结_微信冻结显示什么界面
SAP SD初阶VKM3 对销售订单执行信用冻结释放
保存的时候,系统提醒说:static credit check: credit limit exceeded.
SAP虾客
2022/06/14
8300
Python27中Json对中文的处理
应用场景如下:从api下载数据,json解析,存入字典,定期保存。重启程序需要加载保存的文本。 问题1:json中都是unicode串,存到文本里都是些\u*** 解决:关闭ensure_ascii开关 json.dump(pub.listData,fp,ensure_ascii=False) 问题2:字典关键字用的数字,从文本load后变为unicode串 解决: 走了一点弯路,网上的解决方法,都是转换,把串转回utf-8,方法是 def byteify(input):     if isinsta
用户1075292
2018/01/23
1.7K0
获取HTML冻结窗口元素
在写爬虫时会获取页面的元素去定位,但是有些页面元素只在鼠标悬停时才会出现,鼠标离开后就会消失,这种情况很不利于去查看想要获取的元素。
sugarbeet
2022/10/04
2.8K0
程序执行的本质
这一过程需要计算机的管控。下面我们着重介绍对代码区和动态数据区的管控。CPU中有三个寄存器,分别是eip、ebp和esp,情景如图所示。
一个会写诗的程序员
2021/06/17
8820
程序执行的本质
从受限的代码执行到任意代码执行
看到信安之路发了一篇关于某 CMS 的审计,之前对这个 CMS 也算是有一点了解吧,看到里面的一处 RCE 提起了我一点兴趣,于是有了下文。
信安之路
2020/04/22
9870
从受限的代码执行到任意代码执行
QQ永久冻结解封方法
说到QQ永久封禁,说多了都是泪呀!五年前在从王者峡谷出来后就再也登录不上了,多次在腾讯客服小程序上申诉,每次都是那吊样,同样的理由同样的话术,有人就说找人工客服啊,在理的人都知道,腾讯没有人工客服。不对!是钱冲到位了就有人工客服。
浩瀚博客
2022/08/15
6.6K0
QQ永久冻结解封方法
SAP SD 解除订单信贷冻结处理
销售订单超出信贷额度后被冻结,不能再用于发货,需解除订单的信贷冻结才能继续执行相关业务。
用户5495712
2020/02/16
10.2K0
SAP SD  解除订单信贷冻结处理
SAP最佳业务实践:MM–库存处理:报废、冻结库存(131)-4冻结
4.3 MIGO冻结物料 – 将非限制物料库存调拨到冻结物料 需要冻结物料以防止进一步使用。这意味着不能将库存用于后勤,系统会将库存从 MRP 计算中排除。 1. 在初始屏幕上,确保在屏幕左上
SAP最佳业务实践
2018/03/28
8.7K0
SAP最佳业务实践:MM–库存处理:报废、冻结库存(131)-4冻结
python程序执行的原理
python是一门解释性语言,不是编译性语言。解释性语言的特点是需要解释器进行逐行解释代码,没有编译阶段。编译是指将代码转换成计算机可识别的二进制文件,然后将这些二进制文件链接成可执行文件。
西西嘛呦
2020/08/26
4440
ABAP 防止程序重复执行
有的程序需要现在同一时间只能运行一个,这种需求可以用锁的方式来实现,在START-OF-SELECTION 事件后加锁,如果是加锁成功,说明还没有程序在执行,如果是加锁失败,则说明已经有同名程序在运行,报错即可。
matinal
2020/11/04
9320
ABAP 防止程序重复执行
Perl 程序后台执行示例
自己写示例发现这种方法可以使程序进入后台执行状态,大概原理是 fork 子进程,退出主进程,使得程序被 1 号父进程接管,在终端表现则是进入了后台执行状态。
宋天伦
2023/10/18
1420
Perl 程序后台执行示例
SAP MM 冻结库存的公司间STO
公司间STO一般都是针对可用库存来进行的。冻结库存的公司间STO流程,在项目实践中很少出现。笔者从业十多年,从未遇到哪个企业有这个流程。
SAP虾客
2023/04/02
9510
点击加载更多

相似问题

<a>单元内的垂直中心图像

35

在另一个以中心为中心的div内垂直地居中2项

48

在TD内垂直对齐跨度

10

在跨度内垂直对齐文本

23

在div内具有背景的中心跨度

31
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文