I have a "traditional" enterprise iPad application that needs to make many different web service calls over its lifetime using NTLM authentication. Upon start up of the application, I anticipate getting the user name and password off of a keychain (which the app saves the first time its used since the keychain does not have the user name and subsequently is updated when the password fails to work due to updates).
On startup, various web service calls are needed to get initial data for the application. The user then will be presented with a tabbed controller to choose the functionality they want which in turn will, of course, do more web service calls.
I believe I have a tactic for dealing with each class receiving data through a custom data delegate as presented in this StackOverflow answer (How do you return from an asynchronous NSURLConnection to the calling class?). However, I'm still a bit confused as to how to properly use the -(void)useCredential:(NSURLCredential *)credential forAuthenticationChallenge:(NSURLAuthenticationChallenge *)challenge
functionality.
In didReceiveAuthenticationChallenge, I have code like this
[[challenge sender] useCredential:[NSURLCredential credentialWithUser:@"myusername"
password:@"mypassword"
persistence:NSURLCredentialPersistencePermanent]
forAuthenticationChallenge:challenge];
Since I'm setting permanent persistence, I would expect to not to have to constantly pass in the user name and password in the functionality. Is there a pattern that's used to initially setup the user's NTLM credentials (and/or check to see if they're already there) and then just use the "permanent" credentials for subsequent web service calls?
Also, as a secondary question/part to this. What is the appropriate/elegant approach to passing around a username/password throughout an Objective-C application? I'm thinking either global var's or a singleton instance (which seems a bit overkill for just a couple of needed var's).
It has been awhile since we've tackled this issue and successfully solved it. I thought it was time to put up the answer here. The code below belongs in its own class and will not work quite out of the box but should get you a long ways towards what you need. For the most part, it all should work fine, but you'll just need to make sure the various areas such as alert views, data stores, etc. are all set up the way you need.
A major stumbling block in our understanding of the way that Objective-C & iOS deals with NTLM communications is figuring out its normal process of communicating with a URL.
First contact with a URL is done anonymously. Of course, in a Windows secure environment this will fail. This is when the application will attempt to contact the URL again but this time with any credentials it has for that URL already on the keychain and utilize the willSendRequestForAuthenticationChallenge method. This was very confusing for us since this method didn't fire until AFTER the first call failed. It finally dawned on us what was happening with that first call being anonymous.
Part of the pattern you'll see here is that the connection will be attempted with any credentials already on the keychain. If those fail/missing, then we will popup a view that requests the user to enter the username and password and then we retry with that.
There's a number of idiosyncrasies that we needed to account for as you'll see throughout the code. It took many iterations and lots of testing to get this stable. Much of this was based on patterns that have been posted all over the internet for doing pretty much what we were trying to do but didn't quite take us all the way there.
The code we did generalizes GET/POST calls. This is my first major code post to StackOverflow and my apologies if I'm missing some conventions and I'll correct what I need to when brought to my attention.