This is xnu-8019. See this file in:
//
//  Tests.swift
//
//  Some of these tests here verify that kdd is able to parse old
//  kcdata files and generate the correct output. To do so, we include
//  compressed versions of the raw kcdata and as well as the expected
//  plist output.
//
//  NOTE: If you're adding sample data/plist files, you'll need to first
//        add them to the project and then make sure each is part of the
//        tests target.
//
//  Other tests verify the expected behavior of libkdd for certain
//  situations.
//
//

import XCTest
import Foundation

// Swift's bridging to uuid_t is awkward.

func nsuuid2uuid_t(_ nsuuid : NSUUID) -> uuid_t {
    let dat = nsuuid2array(nsuuid)
    return nsarray2uuid(dat)
}

func nsarray2uuid(_ a : [Int]) -> uuid_t {
    return uuid_t(UInt8(a[0]),
                  UInt8(a[1]),
                  UInt8(a[2]),
                  UInt8(a[3]),
                  UInt8(a[4]),
                  UInt8(a[5]),
                  UInt8(a[6]),
                  UInt8(a[7]),
                  UInt8(a[8]),
                  UInt8(a[9]),
                  UInt8(a[10]),
                  UInt8(a[11]),
                  UInt8(a[12]),
                  UInt8(a[13]),
                  UInt8(a[14]),
                  UInt8(a[15]))
}

func nsuuid2array(_ uuid: NSUUID) -> [Int] {
    var ret = [Int]()
    let ptr = UnsafeMutablePointer<UInt8>.allocate(capacity: 16)
    
    defer { ptr.deallocate(capacity:16) }

    uuid.getBytes(ptr)
    for i in 0..<16 {
        ret.append(Int(ptr[i]))
    }
    return ret
}

func decompress(_ data:NSData) throws -> NSData {
    var stream = z_stream(next_in: nil, avail_in: 0, total_in: 0, next_out: nil, avail_out: 0, total_out: 0, msg: nil, state: nil, zalloc: nil, zfree: nil, opaque: nil, data_type: 0, adler: 0, reserved: 0)

    let bufsize : Int = 1000
    let buffer = UnsafeMutablePointer<UInt8>.allocate(capacity: bufsize)
    defer { buffer.deallocate(capacity:bufsize) }
    let output = NSMutableData()
    stream.next_out = buffer
    stream.avail_out = UInt32(bufsize)
    stream.next_in = UnsafeMutablePointer(mutating:data.bytes.assumingMemoryBound(to:Bytef.self))
    stream.avail_in = UInt32(data.length)
    inflateInit2_(&stream, 16+MAX_WBITS, ZLIB_VERSION, Int32(MemoryLayout<z_stream>.size))

    while (true) {
        let z = inflate(&stream, Z_NO_FLUSH);
        if (z == Z_OK || z == Z_STREAM_END) {
            output.append(buffer, length: bufsize - Int(stream.avail_out))
            stream.avail_out = UInt32(bufsize)
            stream.next_out = buffer
            if (z == Z_STREAM_END) {
                return output;
            }
        } else {
            throw NSError(domain: "zlib", code: Int(z), userInfo: nil)
        }
    }
}


extension Dictionary {
    func value(forKeyPath s:String) -> Any? {
        return (self as NSDictionary).value(forKeyPath:s)
    }
}


class Tests: XCTestCase {
    
    override func setUp() {
        super.setUp()
        continueAfterFailure = false
    }
    
    override func tearDown() {
        // Put teardown code here. This method is called after the invocation of each test method in the class.
        super.tearDown()
    }
    
    func parseBuffer(_ buffer:NSData) throws -> [AnyHashable:Any] {
        var error : NSError?
        guard let dict = parseKCDataBuffer(UnsafeMutablePointer(mutating:buffer.bytes.assumingMemoryBound(to:UInt8.self)), UInt32(buffer.length), &error)
        else {
                XCTAssert(error != nil)
                throw error!
        }
        return dict
    }

    func testPaddingFlags(_ pad : Int) {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_LIBRARY_LOADINFO)
        item.flags = UInt64(pad)
        item.size = UInt32(MemoryLayout<dyld_uuid_info_32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!

        var payload = dyld_uuid_info_32(imageLoadAddress: 42, imageUUID: nsuuid2uuid_t(uuid))
        buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return; }

        var uuidarray = nsuuid2array(uuid)
        for _ in 0..<pad {
            uuidarray.removeLast()
        }

        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageLoadAddress") as? Int == 42)
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageUUID") as! [Int] == uuidarray)
    }

    func testPaddingFlags() {
        for i in 0..<15 {
            testPaddingFlags(i)
        }
    }
    func testBootArgs() {
        let s = "hello, I am some boot args"

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(STACKSHOT_KCTYPE_BOOTARGS)
        item.flags = 0
        item.size = UInt32(s.utf8.count + 1)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        s.utf8CString.withUnsafeBufferPointer({
            buffer.append($0.baseAddress!, length:s.utf8.count + 1)
        })
        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer) else { XCTFail(); return; }
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.boot_args") as? String == s)
    }

    func testBootArgsMissingNul() {
        let s = "hello, I am some boot args"

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(STACKSHOT_KCTYPE_BOOTARGS)
        item.flags = 0
        item.size = UInt32(s.utf8.count)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        s.utf8CString.withUnsafeBufferPointer({
            buffer.append($0.baseAddress!, length:s.utf8.count)
        })

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }

    func testLoadInfo() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_LIBRARY_LOADINFO)
        item.flags = 0
        item.size = UInt32(MemoryLayout<dyld_uuid_info_32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!

        var payload = dyld_uuid_info_32(imageLoadAddress: 42, imageUUID: nsuuid2uuid_t(uuid))
        buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }

        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageLoadAddress") as? Int == 42)
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageUUID") as! [Int] == nsuuid2array(uuid))
    }

    func testLoadInfoWrongSize() {
        // test what happens when a struct size is short

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_LIBRARY_LOADINFO)
        item.flags = 0
        item.size = UInt32(MemoryLayout<dyld_uuid_info_32>.size) - 1
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!

        var payload = dyld_uuid_info_32(imageLoadAddress: 42, imageUUID: nsuuid2uuid_t(uuid))
        buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size - 1)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageLoadAddress") as? Int == 42)
        var uuidarray = nsuuid2array(uuid)
        uuidarray.removeLast()
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageUUID") as! [Int] == uuidarray)
    }

    func testLoadInfoWayWrongSize() {
        // test what happens when a struct size is short

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_LIBRARY_LOADINFO)
        item.flags = 0
        item.size = UInt32(MemoryLayout<dyld_uuid_info_32>.size) - 16
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!

        var payload = dyld_uuid_info_32(imageLoadAddress: 42, imageUUID: nsuuid2uuid_t(uuid))
        buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size - 16)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageLoadAddress") as? Int == 42)
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageUUID") == nil)
    }

    func testLoadInfoPreposterousWrongSize() {
        // test what happens when a struct size is short

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_LIBRARY_LOADINFO)
        item.flags = 0
        item.size = UInt32(1)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload = UInt8(42)
        buffer.append(&payload, length:1)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageLoadAddress") == nil)
        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info.imageUUID") == nil)
    }


    func testNewArray(n : Int, pad : Int) {
        let buffer = NSMutableData(capacity:1000)!
        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY_PAD0) + UInt32(pad)
        item.flags = UInt64(STACKSHOT_KCTYPE_DONATING_PIDS) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<UInt32>.size + pad)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload = UInt32(42 * i)
            buffer.append(&payload, length:MemoryLayout<UInt32>.size)
        }

        for i in 0..<pad {
            var payload = UInt8(42-i)
            buffer.append(&payload, length:MemoryLayout<UInt8>.size)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }
        XCTAssert((dict.value(forKeyPath:"kcdata_crashinfo.donating_pids") as! [Any]).count == n)
        for i in 0..<n {
            let x = dict["kcdata_crashinfo"] as? NSDictionary
            let y = x?["donating_pids"] as? NSArray
            XCTAssert((y?[i]) as? Int == 42 * i)
        }
    }

    func testNewArrays() {
        self.testNewArray(n:0,pad:0)
        for i in 1..<20 {
            for pad in 0..<16 {
                self.testNewArray(n:i, pad:pad)
            }
        }
    }


    func testArrayLoadInfo(n : Int) {
        let buffer = NSMutableData(capacity:1000)!
        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY_PAD0)
        item.flags = UInt64(KCDATA_TYPE_LIBRARY_LOADINFO) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<dyld_uuid_info_32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!


        for i in 0..<n {
            var payload = dyld_uuid_info_32(imageLoadAddress:UInt32(i+42), imageUUID: nsuuid2uuid_t(uuid))

            buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }
        XCTAssert((dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as! [Any]).count == n)
        for i in 0..<n {
            guard let loadinfo = dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as? [Any]
                else { XCTFail(); return; }
            guard let loadinfo_i = loadinfo[i] as? [AnyHashable:Any]
                else { XCTFail(); return; }
            XCTAssert(loadinfo_i["imageLoadAddress"] as? Int == 42 + i)
            XCTAssert(loadinfo_i["imageUUID"] as! [Int] == nsuuid2array(uuid))
        }
    }

    func testArrayLoadInfo() {
        for n in 0..<20 {
            testArrayLoadInfo(n: n)
        }
    }

    func testArrayLoadInfoWrongSize() {
        // test what happens when array element sizes are too short

        let n = 7
        let wrong = 1
        let buffer = NSMutableData(capacity:1000)!
        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY_PAD0)
        item.flags = UInt64(KCDATA_TYPE_LIBRARY_LOADINFO) << 32 | UInt64(n)
        item.size = UInt32(n * (MemoryLayout<dyld_uuid_info_32>.size - wrong))
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!

        for i in 0..<n {
            var payload = dyld_uuid_info_32(imageLoadAddress:UInt32(i+42), imageUUID: nsuuid2uuid_t(uuid))
            buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size-wrong)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        var uuidarray = nsuuid2array(uuid)
        uuidarray.removeLast()

        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return; }
        XCTAssert((dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as! [Any]).count == n)
        for i in 0..<n {
            guard let loadinfo = dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as? [Any]
                else { XCTFail(); return; }
            guard let loadinfo_i = loadinfo[i] as? [AnyHashable:Any]
                else { XCTFail(); return; }
            XCTAssert(loadinfo_i["imageLoadAddress"] as? Int == 42 + i)
            XCTAssert(loadinfo_i["imageUUID"] as! [Int] == uuidarray)
        }

    }


    func testArrayLoadInfoWayWrongSize() {
        // test what happens when array element sizes are too short

        let n = 7
        let wrong = 16
        let buffer = NSMutableData(capacity:1000)!
        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY_PAD0)
        item.flags = UInt64(KCDATA_TYPE_LIBRARY_LOADINFO) << 32 | UInt64(n)
        item.size = UInt32(n * (MemoryLayout<dyld_uuid_info_32>.size - wrong))
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let uuid = NSUUID(uuidString: "de305d54-75b4-431b-adb2-eb6b9e546014")!

        for i in 0..<n {
            var payload = dyld_uuid_info_32(imageLoadAddress:UInt32(i+42), imageUUID: nsuuid2uuid_t(uuid))
            buffer.append(&payload, length:MemoryLayout<dyld_uuid_info_32>.size-wrong)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)


        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return; }
        XCTAssert((dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as! [Any]).count == n)
        for i in 0..<n {
            guard let loadinfo = dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as? [Any]
                else { XCTFail(); return; }
            guard let loadinfo_i = loadinfo[i] as? [AnyHashable:Any]
                else { XCTFail(); return; }
            XCTAssert(loadinfo_i["imageLoadAddress"] as? Int == 42 + i)
            XCTAssert(loadinfo_i["imageUUID"] == nil)
        }
    }

    func testArrayLoadInfoPreposterouslyWrongSize() {
        // test what happens when array element sizes are too short

        let n = 7
        let wrong = 19
        let buffer = NSMutableData(capacity:1000)!
        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY_PAD0)
        item.flags = UInt64(KCDATA_TYPE_LIBRARY_LOADINFO) << 32 | UInt64(n)
        item.size = UInt32(n * (MemoryLayout<dyld_uuid_info_32>.size - wrong))
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload = UInt8(42*i)
            buffer.append(&payload, length:1)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)


        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return; }
        XCTAssert((dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as! [Any]).count == n)
        for i in 0..<n {
            guard let loadinfo = dict.value(forKeyPath:"kcdata_crashinfo.dyld_load_info") as? [Any]
                else { XCTFail(); return; }
            guard let loadinfo_i = loadinfo[i] as? [AnyHashable:Any]
                else { XCTFail(); return; }
            XCTAssert(loadinfo_i["imageLoadAddress"] == nil)
            XCTAssert(loadinfo_i["imageUUID"] == nil)
        }
    }


    func testNested() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt64 = 42
        buffer.append(&payload, length:MemoryLayout<UInt64>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        let buffer2 = NSMutableData(capacity:1000)!

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer2.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_NESTED_KCDATA)
        item.flags = 0
        item.size = UInt32(buffer.length)
        buffer2.append(&item, length: MemoryLayout<kcdata_item>.size)
        buffer2.append(buffer as Data)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer2.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict2 = try? self.parseBuffer(buffer2)
            else { XCTFail(); return; }

        XCTAssert(dict2.value(forKeyPath:"kcdata_crashinfo.kcdata_crashinfo.crashed_threadid") as? Int == 42)
    }


    func testReadThreadid() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt64 = 42
        buffer.append(&payload, length:MemoryLayout<UInt64>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }

        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.crashed_threadid") as? Int == 42)
    }


    func testRepeatedKey() {
        // test a repeated item of the same key causes error

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt64 = 42
        buffer.append(&payload, length:MemoryLayout<UInt64>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        payload = 42
        buffer.append(&payload, length:MemoryLayout<UInt64>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }


    func testContainer() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()
        var payload64 : UInt64
        var payload32 : UInt32

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload64 = 42
        buffer.append(&payload64, length:MemoryLayout<UInt64>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_END)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)


        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return; }

        XCTAssert(dict.value(forKeyPath: "kcdata_crashinfo.task_snapshots.0.crashed_threadid")  as? Int == 42)
    }

    func testDispatchQueueLabel() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()
        let dql = "houston.we.had.a.problem"
        var payload32 : UInt32

        item.type = KCDATA_BUFFER_BEGIN_STACKSHOT
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_THREAD)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(STACKSHOT_KCTYPE_THREAD_DISPATCH_QUEUE_LABEL)
        item.flags = 0
        item.size = UInt32(dql.utf8.count + 1)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        dql.utf8CString.withUnsafeBufferPointer({
            buffer.append($0.baseAddress!, length:dql.utf8.count + 1)
        })

        item.type = UInt32(KCDATA_TYPE_CONTAINER_END)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_THREAD)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_END)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)


        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return; }

        XCTAssert(dict.value(forKeyPath: "kcdata_stackshot.task_snapshots.0.thread_snapshots.0.dispatch_queue_label")  as? String == dql)
    }

    func testRepeatedContainer() {
        //repeated container of same name and key shoudl fail

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()
        var payload64 : UInt64
        var payload32 : UInt32

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload64 = 42
        buffer.append(&payload64, length:MemoryLayout<UInt64>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_END)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)


        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload64 = 42
        buffer.append(&payload64, length:MemoryLayout<UInt64>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_END)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }


    func testContainerNoEnd() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()
        var payload64 : UInt64
        var payload32 : UInt32

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload64 = 42
        buffer.append(&payload64, length:MemoryLayout<UInt64>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }

    func testContainerNoEndNoEnd() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()
        var payload64 : UInt64
        var payload32 : UInt32

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_CONTAINER_BEGIN)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt32>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload32 = UInt32(STACKSHOT_KCCONTAINER_TASK)
        buffer.append(&payload32, length:MemoryLayout<UInt32>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)
        payload64 = 42
        buffer.append(&payload64, length:MemoryLayout<UInt64>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }



    func testNoEnd() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = UInt32(MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt64 = 42
        buffer.append(&payload, length:MemoryLayout<UInt64>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }


    func  testCrazySize() {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = 99999
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt64 = 42
        buffer.append(&payload, length:MemoryLayout<UInt64>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }

    func testReadRepeatedArray() {
        // repeated arrays should be concatenated
        let n = 10

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload : UInt64 = UInt64(i)
            buffer.append(&payload, length:MemoryLayout<UInt64>.size)
        }

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<UInt64>.size)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload : UInt64 = UInt64(i)
            buffer.append(&payload, length:MemoryLayout<UInt64>.size)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
            else { XCTFail(); return }

        XCTAssert( 2*n == (dict.value(forKeyPath:"kcdata_crashinfo.crashed_threadid") as! [Any]).count)
        for i in 0..<2*n {
            let x = dict["kcdata_crashinfo"] as? NSDictionary
            let y = x?["crashed_threadid"] as? NSArray
            XCTAssert((y?[i]) as? Int == i % n)
        }
    }

    func testReadThreadidArray(n : Int, pad : Int) {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<UInt64>.size + pad)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload : UInt64 = UInt64(i)
            buffer.append(&payload, length:MemoryLayout<UInt64>.size)
        }

        for _ in 0..<pad {
            var payload : UInt8 = 0
            buffer.append(&payload, length:1)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }

        XCTAssert( n == (dict.value(forKeyPath:"kcdata_crashinfo.crashed_threadid") as! [Any]).count)

        for i in 0..<n {
            let x = dict["kcdata_crashinfo"] as? NSDictionary
            let y = x?["crashed_threadid"] as? NSArray
            XCTAssert((y?[i]) as? Int == i)
        }

    }

    func testReadThreadidArray() {
        // test that we can correctly read old arrays with a variety of sizes and paddings
        self.testReadThreadidArray(n: 0, pad:0)
        for n in 1..<100 {
            for pad in 0..<16 {
                self.testReadThreadidArray(n: n, pad:pad)
            }
        }
    }

    func testReadThreadidArrayWrongSize1() {
        /// for old style arrays, if the element size is determined by the type.   If the array of that size element at the given count doesn't fit, then parsing should fail

        let n = 1

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(4)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt32 = UInt32(42)
        buffer.append(&payload, length:MemoryLayout<UInt32>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }

    func testReadThreadidArrayWrongSize5() {
        /// if the count is bigger than the buffer, parsing will just fail

        let n = 5

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(4)
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt32 = UInt32(42)
        buffer.append(&payload, length:MemoryLayout<UInt32>.size)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        XCTAssert( (try? self.parseBuffer(buffer)) == nil )
    }


    func testReadThreadidArrayPaddedSize() {
        // test that we can tolerate a little padding at the end of an array
        let n = 5

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<UInt64>.size) + 1
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload : UInt64 = UInt64(i)
            buffer.append(&payload, length:MemoryLayout<UInt64>.size)
        }
        var payload : UInt8 = 0
        buffer.append(&payload, length:1)

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }

        XCTAssert( n == (dict.value(forKeyPath:"kcdata_crashinfo.crashed_threadid") as! [Any]).count)
        for i in 0..<n {
            let x = dict["kcdata_crashinfo"] as? NSDictionary
            let y = x?["crashed_threadid"] as? NSArray
            XCTAssert((y?[i]) as? Int == i)
        }
    }

    func testReadThreadidArrayPaddedSize15() {
        // test that we can tolerate a little padding at the end of an array
        let n = 5

        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(KCDATA_TYPE_ARRAY)
        item.flags = UInt64(TASK_CRASHINFO_CRASHED_THREADID) << 32 | UInt64(n)
        item.size = UInt32(n * MemoryLayout<UInt64>.size) + 15
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        for i in 0..<n {
            var payload : UInt64 = UInt64(i)
            buffer.append(&payload, length:MemoryLayout<UInt64>.size)
        }
        for _ in 0..<15 {
            var payload : UInt8 = 0
            buffer.append(&payload, length:1)
        }

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }

        XCTAssert( n == (dict.value(forKeyPath:"kcdata_crashinfo.crashed_threadid") as! [Any]).count)
        for i in 0..<n {
            let x = dict["kcdata_crashinfo"] as? NSDictionary
            let y = x?["crashed_threadid"] as? NSArray
            XCTAssert((y?[i]) as? Int == i)
        }
    }


    func testReadThreadidWrongSize(size : UInt32) {
        let buffer = NSMutableData(capacity:1000)!

        var item = kcdata_item()

        item.type = KCDATA_BUFFER_BEGIN_CRASHINFO
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        item.type = UInt32(TASK_CRASHINFO_CRASHED_THREADID)
        item.flags = 0
        item.size = size
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        var payload : UInt64 = 42
        buffer.append(&payload, length:Int(size))

        item.type = KCDATA_TYPE_BUFFER_END
        item.flags = 0
        item.size = 0
        buffer.append(&item, length: MemoryLayout<kcdata_item>.size)

        guard let dict = try? self.parseBuffer(buffer)
        else { XCTFail(); return; }

        XCTAssert(dict.value(forKeyPath:"kcdata_crashinfo.crashed_threadid") == nil)
    }

    func testReadThreadidWrongSize0() {
        self.testReadThreadidWrongSize(size: 0)
    }

    func testReadThreadidWrongSize7() {
        self.testReadThreadidWrongSize(size: 7)
    }

    func dataWithResource(_ name:String) -> NSData? {
        guard let filename =  Bundle(for: self.classForCoder).path(forResource: name, ofType: nil)
        else { return nil }
        return NSData(contentsOfFile:filename)!
    }

    func testSampleStackshot(_ name : String) {
        // check that we agree with sample file

        guard let sampledata = self.dataWithResource(name)
            else { XCTFail("failed to open bundle resource named " + name); return }
        var dict : NSDictionary?

        dict = try? self.parseBuffer(sampledata) as NSDictionary

        if (dict == nil) {
            if let decoded = NSData(base64Encoded: sampledata as Data, options:.ignoreUnknownCharacters) {
                dict = try? self.parseBuffer(decoded) as NSDictionary
            }
        }

        if (dict == nil) {
            if let decompressed = try? decompress(sampledata) {
                dict = try? self.parseBuffer(decompressed) as NSDictionary
            }
        }

        if (dict == nil) {
            XCTFail(String(format:"Failed to decode %@", name)); return;
        }

        guard let plistdata = self.dataWithResource(name + ".plist.gz") ??
                              self.dataWithResource(name + ".plist")
            else {XCTFail(); return}

        var opt_dict2 = try? PropertyListSerialization.propertyList(from: plistdata as Data, options: [], format: nil)
        if opt_dict2 == nil {
            opt_dict2 = try? PropertyListSerialization.propertyList(from:decompress(plistdata) as Data, options:[], format: nil)
        }
        guard let dict2 = opt_dict2
            else { XCTFail(); return}

        XCTAssertEqual(dict, dict2 as! NSDictionary);

        //XCTAssert(dict == dict2 as? NSDictionary)

        // check that we agree with python

        #if os(OSX)
        // default python based on #! is python3 in kcdata.py
        let kcdatapy = Bundle(for: self.classForCoder).path(forResource: "kcdata.py", ofType: nil)
        let pyargs = ["-p", Bundle(for:self.classForCoder).path(forResource: name, ofType: nil)!]
        let task = Process()
        task.launchPath = kcdatapy
        task.arguments = pyargs
        let pipe = Pipe()
        task.standardOutput = pipe
        let cli_invocation = task.arguments!.reduce(task.launchPath!) {$0 + " " + $1}
        print(cli_invocation)
        task.launch()

        let data = pipe.fileHandleForReading.readDataToEndOfFile()

            guard let dict3 = try? PropertyListSerialization.propertyList(from:data, options:[], format: nil) as? NSDictionary
            else { XCTFail(); return }

        XCTAssert(dict == dict3)

        // check for python2 if present
        let py2path = "/usr/bin/python2"
        if FileManager.default.fileExists(atPath: py2path) {
            let task = Process()
            task.launchPath = py2path
            task.arguments = [kcdatapy!] + pyargs
            let pipe = Pipe()
            task.standardOutput = pipe
            let cli_invocation = task.arguments!.reduce(task.launchPath!) {$0 + " " + $1}
            print(cli_invocation)
            task.launch()

            let data = pipe.fileHandleForReading.readDataToEndOfFile()

            guard let dict4 = try? PropertyListSerialization.propertyList(from:data, options:[], format: nil) as? NSDictionary
            else { XCTFail(); return }

            XCTAssert(dict == dict4)

        }
        #endif
    }

    func testSampleStackshot() {
        self.testSampleStackshot("stackshot-sample")
    }

    func testSampleStackshotOldArrays() {
        self.testSampleStackshot("stackshot-sample-old-arrays")
    }

    func testSampleStackshotNewArrays() {
        self.testSampleStackshot("stackshot-sample-new-arrays")
    }

    func testSampleDeltaStackshotOldArrays() {
        self.testSampleStackshot("delta-stackshot-sample-old-arrays")
    }

    func testSampleDeltaStackshotNewArrays() {
        self.testSampleStackshot("delta-stackshot-sample-new-arrays")
    }

    func testSampleCorpse() {
        self.testSampleStackshot("corpse-sample")
    }

    func testSampleStackshotTailspin() {
        self.testSampleStackshot("stackshot-sample-tailspin")
    }

    func testSampleStackshotTailspin2() {
        self.testSampleStackshot("stackshot-sample-tailspin-2")
    }

    func testSampleExitReason() {
        self.testSampleStackshot("exitreason-sample")
    }

    func testSampleThreadT() {
        self.testSampleStackshot("stackshot-sample-ths-thread-t")
    }

    func testSampleCpuTimes() {
        self.testSampleStackshot("stackshot-sample-cputime")
    }

    func testSampleDuration() {
        self.testSampleStackshot("stackshot-sample-duration")
    }

    func testSampleNested() {
        self.testSampleStackshot("nested-sample")
    }

    func testSampleTermWithReason() {
        self.testSampleStackshot("test-twr-sample")
    }

    func testSampleCorpseTermWithReason() {
        self.testSampleStackshot("corpse-twr-sample")
    }

    func testSampleCorpseTermWithReasonV2() {
        self.testSampleStackshot("corpse-twr-sample-v2")
    }

    func testSampleCodesigningExitReason() {
        self.testSampleStackshot("exitreason-codesigning")
    }

    func testSampleThreadGroups() {
        self.testSampleStackshot("stackshot-sample-thread-groups")
    }

    func testSampleThreadGroupsFlags() {
        self.testSampleStackshot("stackshot-sample-thread-groups-flags")
    }

    func testSampleCoalitions() {
        self.testSampleStackshot("stackshot-sample-coalitions")
    }

    func testSampleTurnstileInfo() {
        self.testSampleStackshot("stackshot-sample-turnstileinfo")
    }

    func testStackshotSharedcacheV2() {
        self.testSampleStackshot("stackshot-sample-sharedcachev2")
    }

    func testStackshotFaultStats() {
        self.testSampleStackshot("stackshot-fault-stats")
    }

    func testStackshotwithKCID() {
        self.testSampleStackshot("stackshot-with-kcid")
    }

    func testXNUPostTestConfig() {
        self.testSampleStackshot("xnupost_testconfig-sample")
    }

    func testStackshotWithWaitinfo() {
        self.testSampleStackshot("stackshot-with-waitinfo")
    }

    func testStackshotWithAsyncInfo() {
        self.testSampleStackshot("stackshot-with-asyncinfo")
    }

    func testStackshotWithThreadPolicy() {
        self.testSampleStackshot("stackshot-sample-thread-policy")
    }

    func testDeltaStackshotWithThreadPolicy() {
        self.testSampleStackshot("stackshot-sample-delta-thread-policy")
    }

    func testStackshotWithInstrsCycles() {
        self.testSampleStackshot("stackshot-sample-instrs-cycles")
    }

    func testStackshotWithStacktop() {
        self.testSampleStackshot("stackshot-sample-stacktop")
    }

    func testStackshotWithAOT() {
        self.testSampleStackshot("stackshot-sample-aot")
    }

    func testStackshotWithASID() {
        self.testSampleStackshot("stackshot-sample-asid")
    }

    func testStackshotWithPageTables() {
        self.testSampleStackshot("stackshot-sample-asid-pagetable")
    }

    func testStackshotCPUTimes() {
        self.testSampleStackshot("stackshot-sample-cpu-times")
    }
    
    func testStackshotWithSharedCacheLayout() {
        self.testSampleStackshot("stackshot-with-shared-cache-layout")
    }

    func testStackshotDispatchQueueLabel() {
        self.testSampleStackshot("stackshot-sample-dispatch-queue-label")
    }

    func testTrivial() {
    }
}