aboutsummaryrefslogtreecommitdiff
path: root/UnitTesting
diff options
context:
space:
mode:
authorGravatar gtm.daemon <gtm.daemon@7dc7ac4e-7543-0410-b95c-c1676fc8e2a3>2010-11-19 21:30:22 +0000
committerGravatar gtm.daemon <gtm.daemon@7dc7ac4e-7543-0410-b95c-c1676fc8e2a3>2010-11-19 21:30:22 +0000
commitb4ed53890ab5454ae0d6926d5acb7d1d898917e2 (patch)
tree45a71cd9323f10dc9354d56f01f561274de6d471 /UnitTesting
parent59218cf3ff9c6d290de327220758eb84f9889bd0 (diff)
[Author: thomasvl]
Move the test for abstract unittest support out into a file that can be shared with iOS. Add the unittest to the iOS project. Fix up the iOS unittesting support to handle abstract tests. R=dmaclach DELTA=156 (105 added, 20 deleted, 31 changed)
Diffstat (limited to 'UnitTesting')
-rw-r--r--UnitTesting/GTMSenTestCase.m92
-rw-r--r--UnitTesting/GTMSenTestCaseTest.m72
-rw-r--r--UnitTesting/GTMUnitTestingTest.m113
3 files changed, 177 insertions, 100 deletions
diff --git a/UnitTesting/GTMSenTestCase.m b/UnitTesting/GTMSenTestCase.m
index 5c90ae9..2b0a8f7 100644
--- a/UnitTesting/GTMSenTestCase.m
+++ b/UnitTesting/GTMSenTestCase.m
@@ -329,53 +329,69 @@ NSString *const SenTestLineNumberKey = @"SenTestLineNumberKey";
}
// Used for sorting methods below
-static int MethodSort(const void *a, const void *b) {
- const char *nameA = sel_getName(method_getName(*(Method*)a));
- const char *nameB = sel_getName(method_getName(*(Method*)b));
+static int MethodSort(id a, id b, void *context) {
+ NSInvocation *invocationA = a;
+ NSInvocation *invocationB = b;
+ const char *nameA = sel_getName([invocationA selector]);
+ const char *nameB = sel_getName([invocationB selector]);
return strcmp(nameA, nameB);
}
+
+ (NSArray *)testInvocations {
NSMutableArray *invocations = nil;
- unsigned int methodCount;
- Method *methods = class_copyMethodList(self, &methodCount);
- if (methods) {
- // This handles disposing of methods for us even if an exception should fly.
- [NSData dataWithBytesNoCopy:methods
- length:sizeof(Method) * methodCount];
- // Sort our methods so they are called in Alphabetical order just
- // because we can.
- qsort(methods, methodCount, sizeof(Method), MethodSort);
- invocations = [NSMutableArray arrayWithCapacity:methodCount];
- for (size_t i = 0; i < methodCount; ++i) {
- Method currMethod = methods[i];
- SEL sel = method_getName(currMethod);
- char *returnType = NULL;
- const char *name = sel_getName(sel);
- // If it starts with test, takes 2 args (target and sel) and returns
- // void run it.
- if (strstr(name, "test") == name) {
- returnType = method_copyReturnType(currMethod);
- if (returnType) {
- // This handles disposing of returnType for us even if an
- // exception should fly. Length +1 for the terminator, not that
- // the length really matters here, as we never reference inside
- // the data block.
- [NSData dataWithBytesNoCopy:returnType
- length:strlen(returnType) + 1];
- }
+ // Need to walk all the way up the parent classes collecting methods (in case
+ // a test is a subclass of another test).
+ Class senTestCaseClass = [SenTestCase class];
+ for (Class currentClass = self;
+ currentClass && (currentClass != senTestCaseClass);
+ currentClass = class_getSuperclass(currentClass)) {
+ unsigned int methodCount;
+ Method *methods = class_copyMethodList(currentClass, &methodCount);
+ if (methods) {
+ // This handles disposing of methods for us even if an exception should fly.
+ [NSData dataWithBytesNoCopy:methods
+ length:sizeof(Method) * methodCount];
+ if (!invocations) {
+ invocations = [NSMutableArray arrayWithCapacity:methodCount];
}
- if (returnType // True if name starts with "test"
- && strcmp(returnType, @encode(void)) == 0
- && method_getNumberOfArguments(currMethod) == 2) {
- NSMethodSignature *sig = [self instanceMethodSignatureForSelector:sel];
- NSInvocation *invocation
- = [NSInvocation invocationWithMethodSignature:sig];
- [invocation setSelector:sel];
- [invocations addObject:invocation];
+ for (size_t i = 0; i < methodCount; ++i) {
+ Method currMethod = methods[i];
+ SEL sel = method_getName(currMethod);
+ char *returnType = NULL;
+ const char *name = sel_getName(sel);
+ // If it starts with test, takes 2 args (target and sel) and returns
+ // void run it.
+ if (strstr(name, "test") == name) {
+ returnType = method_copyReturnType(currMethod);
+ if (returnType) {
+ // This handles disposing of returnType for us even if an
+ // exception should fly. Length +1 for the terminator, not that
+ // the length really matters here, as we never reference inside
+ // the data block.
+ [NSData dataWithBytesNoCopy:returnType
+ length:strlen(returnType) + 1];
+ }
+ }
+ // TODO: If a test class is a subclass of another, and they reuse the
+ // same selector name (ie-subclass overrides it), this current loop
+ // and test here will cause cause it to get invoked twice. To fix this
+ // the selector would have to be checked against all the ones already
+ // added, so it only gets done once.
+ if (returnType // True if name starts with "test"
+ && strcmp(returnType, @encode(void)) == 0
+ && method_getNumberOfArguments(currMethod) == 2) {
+ NSMethodSignature *sig = [self instanceMethodSignatureForSelector:sel];
+ NSInvocation *invocation
+ = [NSInvocation invocationWithMethodSignature:sig];
+ [invocation setSelector:sel];
+ [invocations addObject:invocation];
+ }
}
}
}
+ // Match SenTestKit and run everything in alphbetical order.
+ [invocations sortUsingFunction:MethodSort context:nil];
return invocations;
}
diff --git a/UnitTesting/GTMSenTestCaseTest.m b/UnitTesting/GTMSenTestCaseTest.m
new file mode 100644
index 0000000..9d9a594
--- /dev/null
+++ b/UnitTesting/GTMSenTestCaseTest.m
@@ -0,0 +1,72 @@
+//
+// GTMSenTestCaseTest.m
+//
+// Copyright 2010 Google Inc.
+//
+// Licensed under the Apache License, Version 2.0 (the "License"); you may not
+// use this file except in compliance with the License. You may obtain a copy
+// of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+// License for the specific language governing permissions and limitations under
+// the License.
+//
+
+#import "GTMSenTestCase.h"
+
+// These make use of the fact that methods are run in alphebetical order
+// to have one test check that a previous one was run. If that order ever
+// changes, there is a good chance things will break.
+
+static int gAbstractCalls_ = 0;
+static int gZzCheckCalls_ = 0;
+
+@interface GTMTestingAbstractTest : GTMTestCase
+@end
+
+@interface GTMTestingTestOne : GTMTestingAbstractTest {
+ BOOL zzCheckCalled_;
+}
+@end
+
+@interface GTMTestingTestTwo : GTMTestingTestOne
+@end
+
+@implementation GTMTestingAbstractTest
+
+- (void)testAbstractUnitTest {
+ STAssertFalse([self isMemberOfClass:[GTMTestingAbstractTest class]],
+ @"test should not run on the abstract class");
+ ++gAbstractCalls_;
+}
+
+@end
+
+@implementation GTMTestingTestOne
+
+- (void)testZZCheck {
+ ++gZzCheckCalls_;
+ if ([self isMemberOfClass:[GTMTestingTestOne class]]) {
+ STAssertEquals(gAbstractCalls_, 1,
+ @"wrong number of abstract calls at this point");
+ } else {
+ STAssertTrue([self isMemberOfClass:[GTMTestingTestTwo class]], nil);
+ STAssertEquals(gAbstractCalls_, 2,
+ @"wrong number of abstract calls at this point");
+ }
+}
+
+@end
+
+@implementation GTMTestingTestTwo
+
+- (void)testZZZCheck {
+ // Test defined at this leaf, it should always run, check on the other methods.
+ STAssertEquals(gZzCheckCalls_, 2, @"the parent class method wasn't called");
+}
+
+@end
diff --git a/UnitTesting/GTMUnitTestingTest.m b/UnitTesting/GTMUnitTestingTest.m
index 8ee2244..d14795f 100644
--- a/UnitTesting/GTMUnitTestingTest.m
+++ b/UnitTesting/GTMUnitTestingTest.m
@@ -6,9 +6,9 @@
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy
// of the License at
-//
+//
// http://www.apache.org/licenses/LICENSE-2.0
-//
+//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
@@ -16,17 +16,14 @@
// the License.
//
-#import <SenTestingKit/SenTestingKit.h>
+#import "GTMSenTestCase.h"
#import "GTMUnitTestingTest.h"
#import "GTMAppKit+UnitTesting.h"
NSString *const kGTMWindowNibName = @"GTMUnitTestingTest";
NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
-@interface GTMUnitTestingAbstractTest : GTMTestCase
-@end
-
-@interface GTMUnitTestingTest : GTMUnitTestingAbstractTest {
+@interface GTMUnitTestingTest : GTMTestCase {
int expectedFailureCount_;
}
@end
@@ -47,17 +44,9 @@ NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
@interface GTMUnitTestingProxyTest : NSProxy
@end
-@implementation GTMUnitTestingAbstractTest
-- (void)testAbstractUnitTest {
- static int testCount = 0;
- testCount += 1;
- STAssertEquals(testCount, 1, @"testAbstractUnitTest should only fire once");
-}
-@end
-
@implementation GTMUnitTestingTest
-
-// Brings up the window defined in the nib and takes a snapshot of it.
+
+// Brings up the window defined in the nib and takes a snapshot of it.
// We use the "empty" GTMUnitTestingTestController controller so that
// initWithWindowNibName can find the appropriate bundle to load our nib from.
// For some reason when running unit tests, with all the injecting going on
@@ -69,24 +58,24 @@ NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
// set up our delegates so we can test delegate handling
GTMUnitTestingDelegate *appDelegate = [[GTMUnitTestingDelegate alloc] init];
[NSApp setDelegate:appDelegate];
-
+
// Get our window
- GTMUnitTestingTestController *testWindowController
+ GTMUnitTestingTestController *testWindowController
= [[GTMUnitTestingTestController alloc] initWithWindowNibName:kGTMWindowNibName];
NSWindow *window = [testWindowController window];
// Test the app state. This will cover windows and menus
- GTMAssertObjectStateEqualToStateNamed(NSApp,
- @"GTMUnitTestingTestApp",
+ GTMAssertObjectStateEqualToStateNamed(NSApp,
+ @"GTMUnitTestingTestApp",
@"Testing the app state");
-
- // Test the window image and state
- GTMAssertObjectEqualToStateAndImageNamed(window,
- kGTMWindowSaveFileName,
+
+ // Test the window image and state
+ GTMAssertObjectEqualToStateAndImageNamed(window,
+ kGTMWindowSaveFileName,
@"Testing the window image and state");
-
+
// Verify that all of our delegate encoders got called
STAssertTrue([appDelegate didEncode], @"app delegate didn't get called?");
-
+
// Clean up
[NSApp setDelegate:nil];
[appDelegate release];
@@ -95,10 +84,10 @@ NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
- (void)testViewUnitTesting {
GTMUnitTestingView *unitTestingView = [[GTMUnitTestingView alloc] init];
- GTMAssertDrawingEqualToImageNamed(unitTestingView,
- NSMakeSize(200,200),
- @"GTMUnitTestingView",
- NSApp,
+ GTMAssertDrawingEqualToImageNamed(unitTestingView,
+ NSMakeSize(200,200),
+ @"GTMUnitTestingView",
+ NSApp,
@"Testing view drawing");
STAssertTrue([unitTestingView hadGoodContext], @"bad context?");
[unitTestingView release];
@@ -123,33 +112,33 @@ NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
NSString *originalPath = [NSObject gtm_getUnitTestSaveToDirectory];
STAssertNotNil(originalPath, @"No save dir?");
[NSObject gtm_setUnitTestSaveToDirectory:tempDir];
- STAssertEqualObjects(tempDir, [NSObject gtm_getUnitTestSaveToDirectory],
+ STAssertEqualObjects(tempDir, [NSObject gtm_getUnitTestSaveToDirectory],
@"Save to dir not set?");
NSString *statePath = [self gtm_saveToPathForStateNamed:bogusTestName];
STAssertNotNil(statePath, @"no state path?");
NSString *imagePath = [self gtm_saveToPathForImageNamed:bogusTestName];
STAssertNotNil(imagePath, @"no image path?");
- GTMUnitTestingTestController *testWindowController
+ GTMUnitTestingTestController *testWindowController
= [[GTMUnitTestingTestController alloc] initWithWindowNibName:kGTMWindowNibName];
NSWindow *window = [testWindowController window];
-
+
// Test against a golden master filename that doesn't exist
expectedFailureCount_ = 2;
GTMAssertObjectEqualToStateAndImageNamed(window,
bogusTestName,
@"Creating image and state files");
- STAssertEquals(expectedFailureCount_, 0,
+ STAssertEquals(expectedFailureCount_, 0,
@"Didn't get expected failures creating files");
-
+
// Change our image and state and verify failures
[[testWindowController textField] setStringValue:@"Foo"];
expectedFailureCount_ = 2;
- GTMAssertObjectEqualToStateAndImageNamed(window,
- kGTMWindowSaveFileName,
+ GTMAssertObjectEqualToStateAndImageNamed(window,
+ kGTMWindowSaveFileName,
@"Testing the window image and state");
- STAssertEquals(expectedFailureCount_, 0,
+ STAssertEquals(expectedFailureCount_, 0,
@"Didn't get expected failures testing files");
-
+
// Now change the size of our image and verify failures
NSRect oldFrame = [window frame];
NSRect newFrame = oldFrame;
@@ -157,45 +146,45 @@ NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
[window setFrame:newFrame display:YES];
expectedFailureCount_ = 1;
GTMAssertObjectImageEqualToImageNamed(window,
- kGTMWindowSaveFileName,
+ kGTMWindowSaveFileName,
@"Testing the changed window size");
- [window setFrame:oldFrame display:YES];
-
+ [window setFrame:oldFrame display:YES];
+
// Set our unit test save dir to a bogus directory and
// run the tests again.
[NSObject gtm_setUnitTestSaveToDirectory:@"/zim/blatz/foo/bob/bar"];
expectedFailureCount_ = 2;
- GTMAssertObjectEqualToStateAndImageNamed(window,
- kGTMWindowSaveFileName,
+ GTMAssertObjectEqualToStateAndImageNamed(window,
+ kGTMWindowSaveFileName,
@"Testing the window image and state");
- STAssertEquals(expectedFailureCount_, 0,
- @"Didn't get expected failures testing files");
+ STAssertEquals(expectedFailureCount_, 0,
+ @"Didn't get expected failures testing files");
expectedFailureCount_ = 2;
- GTMAssertObjectEqualToStateAndImageNamed(window,
- @"GTMUnitTestingWindowDoesntExist",
+ GTMAssertObjectEqualToStateAndImageNamed(window,
+ @"GTMUnitTestingWindowDoesntExist",
@"Testing the window image and state");
- STAssertEquals(expectedFailureCount_, 0,
- @"Didn't get expected failures testing files");
-
+ STAssertEquals(expectedFailureCount_, 0,
+ @"Didn't get expected failures testing files");
+
// Reset our unit test save dir
[NSObject gtm_setUnitTestSaveToDirectory:nil];
-
+
// Test against something that doesn't have an image
expectedFailureCount_ = 1;
- GTMAssertObjectImageEqualToImageNamed(@"a string",
- @"GTMStringsDontHaveImages",
+ GTMAssertObjectImageEqualToImageNamed(@"a string",
+ @"GTMStringsDontHaveImages",
@"Testing that strings should fail");
- STAssertEquals(expectedFailureCount_, 0, @"Didn't get expected failures testing files");
-
+ STAssertEquals(expectedFailureCount_, 0, @"Didn't get expected failures testing files");
+
// Test against something that doesn't implement our support
expectedFailureCount_ = 1;
GTMUnitTestingProxyTest *proxy = [[GTMUnitTestingProxyTest alloc] init];
- GTMAssertObjectStateEqualToStateNamed(proxy,
- @"NSProxiesDontDoState",
+ GTMAssertObjectStateEqualToStateNamed(proxy,
+ @"NSProxiesDontDoState",
@"Testing that NSProxy should fail");
- STAssertEquals(expectedFailureCount_, 0, @"Didn't get expected failures testing proxy");
+ STAssertEquals(expectedFailureCount_, 0, @"Didn't get expected failures testing proxy");
[proxy release];
-
+
[window close];
}
@@ -235,7 +224,7 @@ NSString *const kGTMWindowSaveFileName = @"GTMUnitTestingWindow";
return didEncode_;
}
@end
-
+
@implementation GTMUnitTestingView
- (void)gtm_unitTestViewDrawRect:(NSRect)rect contextInfo:(void*)contextInfo {