Files
swift-mirror/test/Interpreter/SDK/Foundation_test.swift

402 lines
12 KiB
Swift

// RUN: %target-run-simple-swift
// REQUIRES: executable_test
// XFAIL: interpret
// REQUIRES: objc_interop
import Foundation
import StdlibUnittest
// Also import modules which are used by StdlibUnittest internally. This
// workaround is needed to link all required libraries in case we compile
// StdlibUnittest with -sil-serialize-all.
import SwiftPrivate
#if _runtime(_ObjC)
import ObjectiveC
#endif
// rdar://problem/18884272
// Make sure that NSObject conforms to NSObjectProtocol. This
// particular bug is ridiculously hard to trigger without a complete
// SDK, so it sits here.
let objcProtocol: NSObjectProtocol = NSObject()
var FoundationTestSuite = TestSuite("Foundation")
func asNSString(s: String) -> NSString { return s as NSString }
func asString(ns: NSString) -> String { return ns as String }
//===----------------------------------------------------------------------===//
// Strings
//===----------------------------------------------------------------------===//
FoundationTestSuite.test("NSString") {
var str = "Hello"
var nsStr = str as NSString
assert(nsStr.compare(str).rawValue == NSComparisonResult.orderedSame.rawValue)
assert(nsStr.compare(str) == NSComparisonResult.orderedSame)
nsStr = "World"
str = nsStr as String
// FIXME: Shouldn't need coercion here to resolve ambiguity. <rdar://problem/14637688>
assert(str == asString(nsStr))
}
//===----------------------------------------------------------------------===//
// Numbers
//===----------------------------------------------------------------------===//
FoundationTestSuite.test("NSNumber") {
var i = 17
var d = 3.14159
var b = true
// Implicit boxing/explicit unboxing
var nsNum: NSNumber = i as NSNumber
expectEqual(i, Int(nsNum))
nsNum = d as NSNumber
expectEqual(d, Double(nsNum))
nsNum = b as NSNumber
expectEqual(b, Bool(nsNum))
// Literals
nsNum = 42
expectEqual(42, Int(nsNum))
nsNum = 3.14159
expectEqual(3.14159, Double(nsNum))
nsNum = false
expectEqual(false, Bool(nsNum))
}
//===----------------------------------------------------------------------===//
// Arrays
//===----------------------------------------------------------------------===//
FoundationTestSuite.test("NSArray") {
// Literals
var nsArr: NSArray = [ 1, 2.5, "Hello" ]
assert(nsArr.count == 3)
// Subscripting
expectEqual(1, Int(nsArr[0] as! NSNumber))
expectEqual(2.5, Double(nsArr[1] as! NSNumber))
expectTrue((nsArr[2] as! NSString).isEqual("Hello"))
// Iteration
var result = [String]()
for x: AnyObject in nsArr {
result.append((x as! NSObject).description)
}
expectEqualSequence([ "1", "2.5", "Hello" ], result)
}
FoundationTestSuite.test("NSMutableArray") {
let nsMutableArr: NSMutableArray = ["Constant", "Moon"]
nsMutableArr[0] = "Inconstant"
expectEqual(2, nsMutableArr.count)
expectEqual("Inconstant", nsMutableArr[0] as! NSString)
expectEqual("Moon", nsMutableArr[1] as! NSString)
}
FoundationTestSuite.test("NSArrayVariadicInit") {
let variadicArray = NSArray(objects: "A", "B", "C")
expectEqual(3, variadicArray.count)
}
FoundationTestSuite.test("arrayConversions") {
var nsa = NSArray()
var aoa: Array<AnyObject> = []
nsa as Array<AnyObject>
var nsa2 = NSArray()
var aoa2 = nsa2 as Array<AnyObject>
var nsaoa = aoa as NSArray
func nsArrayToAnyObjectArray(nsa: NSArray) -> [AnyObject] {
return nsa as [AnyObject]
}
nsArrayToAnyObjectArray(nsa)
nsArrayToAnyObjectArray(aoa as NSArray)
}
//===----------------------------------------------------------------------===//
// Dictionaries
//===----------------------------------------------------------------------===//
FoundationTestSuite.test("NSDictionary") {
var nsDict : NSDictionary = [1 : "Hello", 2 : "World"]
assert((nsDict[1]! as! NSString).isEqual("Hello"))
assert((nsDict[2]! as! NSString).isEqual("World"))
let nsMutableDict: NSMutableDictionary = ["Hello" : 1, "World" : 2]
assert(nsMutableDict["Hello"]!.isEqual(1))
assert(nsMutableDict["World"]!.isEqual(2))
}
//===----------------------------------------------------------------------===//
// Ranges
//===----------------------------------------------------------------------===//
FoundationTestSuite.test("NSRange") {
let nsRange = NSRange(1..<5)
expectEqual("{1, 4}", String(NSStringFromRange(nsRange)))
}
//===----------------------------------------------------------------------===//
// URLs
//===----------------------------------------------------------------------===//
FoundationTestSuite.test("NSURL") {
let nsURL = NSURL(string: "http://llvm.org")!
expectEqual("http://llvm.org", nsURL.description)
}
//===----------------------------------------------------------------------===//
// Pattern-matching
//===----------------------------------------------------------------------===//
func matchesEither(input: NSNumber, _ a: NSNumber, _ b: NSNumber) -> Bool {
switch input {
case a, b:
return true
default:
return false
}
}
FoundationTestSuite.test("patternMatching") {
var one, two, three, oneAgain : NSNumber
one = NSNumber(int: 1)
two = NSNumber(int: 2)
three = NSNumber(int: 3)
oneAgain = NSNumber(int: 1)
expectFalse(matchesEither(one, two, three))
expectTrue(matchesEither(one, oneAgain, three))
expectTrue(matchesEither(one, two, oneAgain))
}
//===----------------------------------------------------------------------===//
// Miscellaneous
//===----------------------------------------------------------------------===//
// <rdar://problem/14474701>
// Type checker used to crash on this.
class ClassWithDtor : NSObject {
deinit {
let noteCenter = NSNotificationCenter.defaultCenter()
noteCenter.removeObserver(self, name: "ReceivedContentNotification", object: nil)
}
}
FoundationTestSuite.test("rdar://17584531") {
// <rdar://problem/17584531>
// Type checker used to be confused by this.
var dict: NSDictionary = [ "status": 200, "people": [ [ "id": 255, "name": [ "first": "John", "last": "Appleseed" ] ] ] ]
var dict2 = dict["people"]?[0] as! NSDictionary
expectEqual("Optional(255)", String(dict2["id"]))
}
FoundationTestSuite.test("DarwinBoolean smoke test") {
expectFalse(CFEqual("abc" as NSString as CFString, "def" as NSString as CFString))
let _: CFArrayEqualCallBack = { DarwinBoolean($0 == $1) }
}
#if os(OSX)
FoundationTestSuite.test("NSRectEdge/constants") {
// Check that the following constants have the correct type and value.
//
// It is valid to hardcode the value in the test. The way they are currently
// defined in the SDK makes them ABI for Objective-C code.
expectEqual(0, CGRectEdge(rectEdge: NSRectEdge.minX).rawValue)
expectEqual(0, NSRectEdge(rectEdge: CGRectEdge.minXEdge).rawValue)
expectEqual(1, CGRectEdge(rectEdge: NSRectEdge.minY).rawValue)
expectEqual(1, NSRectEdge(rectEdge: CGRectEdge.minYEdge).rawValue)
expectEqual(2, CGRectEdge(rectEdge: NSRectEdge.maxX).rawValue)
expectEqual(2, NSRectEdge(rectEdge: CGRectEdge.maxXEdge).rawValue)
expectEqual(3, CGRectEdge(rectEdge: NSRectEdge.maxY).rawValue)
expectEqual(3, NSRectEdge(rectEdge: CGRectEdge.maxYEdge).rawValue)
}
#endif
if #available(OSX 10.11, iOS 9.0, *) {
FoundationTestSuite.test("NSUndoManager/ObjCClass") {
let UM = NSUndoManager()
// Confirm with an ObjC class.
class ObjCClass : NSObject {
var someProperty: String = ""
}
let f = ObjCClass()
UM.registerUndoWithTarget(f) { target in
target.someProperty = "expected"
}
UM.undo()
expectEqual(f.someProperty, "expected")
}
FoundationTestSuite.test("NSUndoManager/SwiftClass") {
let UM = NSUndoManager()
// Confirm with a Swift class.
class SwiftClass {
var someOtherProperty: String = ""
}
var b = SwiftClass()
UM.registerUndoWithTarget(b) { target in
target.someOtherProperty = "expected"
}
UM.undo()
expectEqual(b.someOtherProperty, "expected")
}
}
private let KEY = "some-key"
private func createTestArchive() -> NSData {
let mutableData = NSMutableData()
let KA = NSKeyedArchiver(forWritingWith: mutableData)
// Set up some fake data.
let obj = NSPredicate(value: true)
KA.encode(obj, forKey: KEY)
KA.encode(obj, forKey: NSKeyedArchiveRootObjectKey)
KA.finishEncoding()
return mutableData
}
FoundationTestSuite.test("NSKeyedUnarchiver/decodeObjectOfClass(_:forKey:)") {
let obj = NSPredicate(value: true)
let data = createTestArchive()
var KU = NSKeyedUnarchiver(forReadingWith: data)
var missing = KU.decodeObjectOfClass(NSPredicate.self, forKey: "Not there")
expectEmpty(missing)
expectType((NSPredicate?).self, &missing)
var decoded = KU.decodeObjectOfClass(NSPredicate.self, forKey: KEY)
expectNotEmpty(decoded)
expectType((NSPredicate?).self, &decoded)
expectCrashLater()
// Even though we're doing non-secure coding, this should still be checked in
// Swift.
var wrongType = KU.decodeObjectOfClass(NSDateFormatter.self, forKey: KEY)
expectType((NSDateFormatter?).self, &wrongType)
KU.finishDecoding()
}
if #available(OSX 10.11, iOS 9.0, *) {
FoundationTestSuite.test("NSKeyedUnarchiver/decodeTopLevel*") {
let obj = NSPredicate(value: true)
let data = createTestArchive()
// first confirm .decodeObjectWithClasses overlay requires NSSet
var KU = NSKeyedUnarchiver(forReadingWith: data)
var nonTopLevelResult = KU.decodeObjectOfClasses(NSSet(array: [NSPredicate.self]), forKey: KEY)
expectTrue(nonTopLevelResult != nil)
KU.finishDecoding()
KU = NSKeyedUnarchiver(forReadingWith: data)
do {
// decodeObjectForKey(_:) throws
let decoded1 = try KU.decodeTopLevelObjectForKey(KEY) as? NSPredicate
let missing1 = try KU.decodeTopLevelObjectForKey("Not there")
expectTrue(decoded1 != nil)
expectEqual(obj, decoded1)
expectTrue(missing1 == nil)
KU.finishDecoding()
// recreate so we can do the rest securely
KU = NSKeyedUnarchiver(forReadingWith: data)
KU.requiresSecureCoding = true
// decodeObjectOfClass(_:,forKey:) throws
let decoded2 = try KU.decodeTopLevelObjectOfClass(NSPredicate.self, forKey: KEY)
let missing2 = try KU.decodeTopLevelObjectOfClass(NSPredicate.self, forKey: "Not there")
expectTrue(decoded2 != nil)
expectEqual(obj, decoded2)
expectTrue(missing2 == nil)
// decodeObjectOfClasses(_:,forKey:) throws
let classes = NSSet(array: [NSPredicate.self])
let decoded3 = try KU.decodeTopLevelObjectOfClasses(classes, forKey: KEY) as? NSPredicate
let missing3 = try KU.decodeTopLevelObjectOfClasses(classes, forKey: "Not there")
expectTrue(decoded3 != nil)
expectEqual(obj, decoded3)
expectTrue(missing3 == nil)
}
catch {
expectTrue(false) // should have no errors
}
KU.finishDecoding()
KU = NSKeyedUnarchiver(forReadingWith: data)
KU.requiresSecureCoding = true
do {
// recreate so we avoid caches from above
let wrong = try KU.decodeTopLevelObjectOfClass(NSDateFormatter.self, forKey: KEY)
expectUnreachable() // should have error
}
catch {
// expected
}
KU.finishDecoding()
KU = NSKeyedUnarchiver(forReadingWith: data)
KU.requiresSecureCoding = true
do {
let wrongClasses = NSSet(array: [NSDateFormatter.self])
let wrong = try KU.decodeTopLevelObjectOfClasses(wrongClasses, forKey: KEY)
expectUnreachable() // should have error
}
catch {
// expected
}
KU.finishDecoding()
// confirm the that class function works
do {
let decoded = try NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(data) as? NSPredicate
expectEqual(obj, decoded)
}
catch {
expectUnreachableCatch(error)
}
}
FoundationTestSuite.test("NSKeyedUnarchiver/decodeTopLevelObjectOfClass(_:forKey:)/trap") {
let obj = NSPredicate(value: true)
let data = createTestArchive()
var KU = NSKeyedUnarchiver(forReadingWith: data)
expectCrashLater()
// Even though we're doing non-secure coding, this should still be checked
// in Swift.
do {
var wrongType = try KU.decodeTopLevelObjectOfClass(NSDateFormatter.self, forKey: KEY)
expectType((NSDateFormatter?).self, &wrongType)
} catch {
expectUnreachableCatch(error)
}
KU.finishDecoding()
}
}
runAllTests()