Integrating iPhone Application with Shibboleth

后端 未结 3 1477
忘了有多久
忘了有多久 2020-12-28 22:41

Has anyone integrated an iPhone application with a Shibboleth Identity Provider? Googling did not come up with anything so I am asking the gurus directly.

If it has

3条回答
  •  时光说笑
    2020-12-28 22:50

    The answer to both is "Yes."

    I'm a Java guy, so being asked two weeks ago to:

    • Learn Objective-C
    • Write an native iPhone App
    • Authenticated programmatically with Shibboleth
    • Download an display Shibboleth protected datafile

    ...Was a little daunting. Compound that with the absence of any forum posts to help out has prompted me to share my experience.

    Here's an overview followed by some hopefully very helpful sample code. Please vote for my answer if this helps! It worth a few weeks of my time :)

    For an application on the iPhone to download Shibbolized resources, the following needs to happen:

    1. Use the URL API's in Cocoa to submit the HTTP request for the resource in question.
    2. Implement a delegate class for the request to:
    3. Respond to the SP re-direct to the IdP (automatic courtesy of Cocoa)
    4. Respond to server certificate trust challenges
    5. Respond to user credential challenges
    6. Respond to errors (if needed)
    7. Receive IdP's "binding template" for the authenticated user, an HTML form which re-directs the user back to the SP with two parameters
    8. Programmatically HTTP POST the two parameters from the IdP back to the SP.
    9. Cookies are automatically stored and forwards courtesy of Cocoa again
    10. Implement a second URL Request delegate to receive the originally request data.

    Here are some useful references from Apple and Shibboleth:

    • http://developer.apple.com/mac/library/DOCUMENTATION/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html
    • https://spaces.internet2.edu/display/SHIB2/IdPSPLocalTestInstall

    And hopefully I can include all the source for a quick demonstration.

    ApplicationDelegate.h
    ----------
    #import 
    #import "ConsoleViewController.h"
    
    /*
     The application delegate will hold references to the application's UIWindow and a ConsoleViewController.
     The console does all of the interesting Shibboleth activities.
    */
    @interface ApplicationDelegate : NSObject  {
    
     UIWindow *window;
     ConsoleViewController *consoleViewController;
    }
    
    
    @end
    
    ApplicationDelegate.m
    ----------
    #import "ApplicationDelegate.h"
    #import "ConsoleViewController.h"
    
    /*
     The implementation for the ApplicationDelegate initializes the console view controller and assembles everything.
     The console does all of the interesting Shibboleth activities.
     */
    @implementation ApplicationDelegate
    
    
    - (void)applicationDidFinishLaunching:(UIApplication *)application {    
    
     // Initialize the console.
     consoleViewController = [[ConsoleViewController alloc] init];
    
     window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
     [window setBackgroundColor:[UIColor lightGrayColor]];
     [window addSubview:[consoleViewController view]];
    
     [window makeKeyAndVisible];
    }
    
    
    - (void)dealloc {
        [window release];
     [ConsoleViewController release];
        [super dealloc];
    }
    
    
    @end
    
    ConsoleController.h
    ----------
    #import 
    #import 
    
    /*
     The ConsoleViewController's interface declares references to the network data used in negotiating with Shibboleth
     and a UITextView used to display the final result or errors.
     */
    @interface ConsoleViewController : UIViewController {
    
     NSMutableData *responseData;
     NSString *responseString;
     UITextView *console;
    }
    
    @end
    
    ConsoleController.m
    ----------
    #import "ApplicationDelegate.h"
    #import "ConsoleViewController.h"
    
    
    /*
     This delegate is used when making the second HTTP request with Shibboleth.  If you're just getting here, start
     by reading the comments for ConsoleViewController below.
    
     All we need to do now is receive the response from the SP and display it.
     If all goes well, this should be the secured page originally requested.
     */
    @interface AuthenticationRedirectDelegate : NSObject {
    
     NSMutableData *authResponseData;
     NSString *authResponseString;
     UITextView *console;
    }
    
    @property (nonatomic retain) UITextView *console;
    
    @end
    
    
    /*
     Refer to the comments for the interface above.
     */
    @implementation AuthenticationRedirectDelegate
    
    @synthesize console;
    
    -(id)init {
     authResponseData = [[NSMutableData alloc] retain];
     return self;
    }
    
    
    - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
     [authResponseData setLength:0];
    }
    
    
    - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
     [authResponseData appendData:data];
    }
    
    
    - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
     [console setText:[error localizedDescription]]; 
    }
    
    
    /*
     Once the data is received from Shibboleth's SP, display it.
     */
    - (void)connectionDidFinishLoading:(NSURLConnection *)connection {  
    
     authResponseString = [[NSString alloc] initWithData:authResponseData encoding:NSUTF8StringEncoding]; 
     [console setText:authResponseString]; 
     [connection release];
    }
    
    
    @end
    
    
    /*
     The implementation of the ConsoleViewController, and AuthenticationRedirectDelegate above, contain the real logic of
     this Shibboleth exercise.  The ConsoleViewController performs the following:
     1. Prepare the initial HTTP request to a Shibboleth protected resource.
     2. Act as the delegate whilst Cocoa's URL Loading API receives the HTTP Response.
     NOTE: We instruct Cocoa in advance to take care of the SP redirecting to the IdP, accepting the server certificate,
     and submitting the user credentials
     3. Once the HTTP Response is finished loading, parse the 
    "]; NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:12.0]; [[NSURLConnection alloc] initWithRequest:request delegate:self]; /* Control flows to the delegate methods below */ } /* Refer to Apple's docs on the URL Loading System for details. http://developer.apple.com/mac/library/DOCUMENTATION/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html */ - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response { [responseData setLength:0]; } /* Refer to Apple's docs on the URL Loading System for details. http://developer.apple.com/mac/library/DOCUMENTATION/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html */ - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data { [responseData appendData:data]; } /* This implementation in the delegate let's Cocoa trust my SP Web Server's self-signed certificate. TODO: You will want to harden this for production use. Refer to Apple's docs on the URL Loading System for details. http://developer.apple.com/mac/library/DOCUMENTATION/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html */ - (BOOL)connection:(NSURLConnection *)connection canAuthenticateAgainstProtectionSpace:(NSURLProtectionSpace *)protectionSpace { return [protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust] || [protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodHTTPBasic]; } /* This implementation for the delegate does two things: 1. Respond to challenges for my server's self-signed certificate 2. Respond to the IdP's challenge for the username and password. TODO: Enter your own username and password here. Refer to Apple's docs on the URL Loading System for details. http://developer.apple.com/mac/library/DOCUMENTATION/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html */ - (void)connection:(NSURLConnection *)connection didReceiveAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge { // TODO: Enter the correct username and password below. /* WARNING: Using an incorrect user name and password will result in your application being re-challenged by the IdP. Cocoa will return to this function in a never-ending loop. This can result in the message "NSPosixErrorDomain Too many open files". You'll need to perform additional coding to handle this. */ if ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodServerTrust]) [challenge.sender useCredential:[NSURLCredential credentialForTrust:challenge.protectionSpace.serverTrust] forAuthenticationChallenge:challenge]; else if ([challenge.protectionSpace.authenticationMethod isEqualToString:NSURLAuthenticationMethodHTTPBasic]) [challenge.sender useCredential:[NSURLCredential credentialWithUser:@"" password:@"" persistence:NSURLCredentialPersistenceNone] forAuthenticationChallenge:challenge]; else [challenge.sender continueWithoutCredentialForAuthenticationChallenge:challenge]; } /* You may wish to add more code here to log errors. Refer to Apple's docs on the URL Loading System for details. http://developer.apple.com/mac/library/DOCUMENTATION/Cocoa/Conceptual/URLLoadingSystem/URLLoadingSystem.html */ - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error { [console setText:[error localizedDescription]]; } /* Once Cocoa has received a (hopefully) authenticated response from the IdP, we parse out the relevant pieces and prepare to HTTP POST them back to the SP as specified by the IdP in the "]; NSString *SAMLResponse = [ConsoleViewController substringFromString:responseString BetweenOpenToken:@"SAMLResponse\" value=\"" AndCloseToken:@"\"/>"]; NSString *formAction = [ConsoleViewController substringFromString:responseString BetweenOpenToken:@"

提交回复
热议问题