5

输入dictionary(fromTXTRecord:)来自网络,可能来自应用程序外部,甚至是设备。但是,Apple 的文档说:

... 如果 txtData 不能表示为 NSDictionary 对象,则断言失败。

断言失败会使程序员(我)无法处理错误,这对于处理外部数据的方法来说似乎不合逻辑。

如果我在 Mac 上的终端中运行它:

dns-sd -R 'My Service Name' _myservice._tcp local 4567 asdf asdf

我在 iPhone 上运行的应用程序崩溃了。

dictionary(fromTXTRecord:)期望 TXT 记录数据 ( asdf asdf) 在key=val格式中。如果像上面一样,一个单词不包含任何内容=,则该方法将无法解析它并使断言失败。

除了根本不使用该方法并实现我自己的解析之外,我认为没有其他方法可以解决这个问题,这感觉是错误的。

我错过了什么吗?

4

3 回答 3

3

这是 Swift 4.2 中的一个解决方案,假设 TXT 记录只有字符串:

/// Decode the TXT record as a string dictionary, or [:] if the data is malformed
public func dictionary(fromTXTRecord txtData: Data) -> [String: String] {

    var result = [String: String]()
    var data = txtData

    while !data.isEmpty {
        // The first byte of each record is its length, so prefix that much data
        let recordLength = Int(data.removeFirst())
        guard data.count >= recordLength else { return [:] }
        let recordData = data[..<(data.startIndex + recordLength)]
        data = data.dropFirst(recordLength)

        guard let record = String(bytes: recordData, encoding: .utf8) else { return [:] }
        // The format of the entry is "key=value"
        // (According to the reference implementation, = is optional if there is no value,
        // and any equals signs after the first are part of the value.)
        // `ommittingEmptySubsequences` is necessary otherwise an empty string will crash the next line
        let keyValue = record.split(separator: "=", maxSplits: 1, omittingEmptySubsequences: false)
        let key = String(keyValue[0])
        // If there's no value, make the value the empty string
        switch keyValue.count {
        case 1:
            result[key] = ""
        case 2:
            result[key] = String(keyValue[1])
        default:
            fatalError()
        }
    }

    return result
}
于 2018-09-07T10:36:35.763 回答
1

我刚刚使用 Swift 3 遇到了这个问题。就我而言,问题仅在我使用时发生,NetService.dictionary(fromTXTRecord:)但在我切换到 Objective-C 并调用NSNetService dictionaryFromTXTRecord:. 当 Objective-C 调用遇到一个没有等号的条目时,它会创建一个包含数据的键并将其推入带有NSNull值的字典中。据我所知,Swift 版本会枚举该字典并在看到NSNull. 我的解决方案是添加一个 Objective-C 文件和一个实用程序函数,在将结果返回给我的 Swift 代码之前调用dictionaryFromTXTRecord:和清理结果。

于 2016-12-29T01:17:34.090 回答
1

我仍然希望这里缺少一些东西,但与此同时,我最终检查了数据的正确性,然后才调用 Apple 自己的方法。

这是我的解决方法:

func dictionaryFromTXTRecordData(data: NSData) -> [String:NSData] {
    let buffer = UnsafeBufferPointer<UInt8>(start: UnsafePointer(data.bytes), count: data.length)
    var pos = 0
    while pos < buffer.count {
        let len = Int(buffer[pos])
        if len > (buffer.count - pos + 1) {
            return [:]
        }
        let subdata = data.subdataWithRange(NSRange(location: pos + 1, length: len))
        guard let substring = String(data: subdata, encoding: NSUTF8StringEncoding) else {
            return [:]
        }
        if !substring.containsString("=") {
            return [:]
        }
        pos = pos + len + 1
    }
    return NSNetService.dictionaryFromTXTRecordData(data)
}

我在这里使用 Swift 2。欢迎所有贡献。Swift 3 版本、Objective-C 版本、改进、更正。

于 2016-11-07T19:51:00.983 回答