iOS-OAuth认证
OAuth授权
OAuth授权分四步:
第一步,应用向服务提供方申请请求令牌(Request Token),服务提供方验证通过后将令牌返回。这个步骤由于涉及到应用帐号密码,在应用的服务端发起,所以这个步骤对用户透明。
第二步,应用使用请求令牌让浏览器重定向到服务提供方进行登录验证和授权。服务提供方校验请求令牌,将第三方的资料显示给用户,提示用户选择同意或拒绝此次授权。如果用户同意授权,发放已授权令牌并将用户引导到当前应用的注册地址。这个步骤从重定向开始到引导回注册地址之前,应用方并不参与用户身份校验和授权过程,确保第三方不可获得用户的真实帐号密码。
第三步,用已授权令牌向服务提供方换取ATOK。第三方应用需在服务端发起请求,用帐号密码和上一步的令牌换取ATOK,这个步骤对用户而言也是透明的。如果前两步分别是让服务提供方认证应用和用户,那这步就是用户和服务提供方再次认证第三方应用。因为用户浏览器将第二步的结果重定向到第三步,除非用户DNS被劫持,否则就能确保重定向到的是合法的地址。曾经我很困惑在用户授权之后为何不直接返回ATOK而需要再次换取,估计是出于对ATOK的安全考虑,用户浏览器一端存在太多的可能性让ATOK泄漏,最安全的办法还是让第三方服务端来获取和保管ATOK。
第四步,用ATOK作为令牌访问受保护资源。很多时候,权限是有多种类别的。ATOK包含了某个用户对某个应用的授权凭据,准确的说,ATOK对应用户授权时所赋予的一系列权限的集合。所以在这一步,除了校验ATOK的合法性之外,服务提供方还需对该ATOK是否拥有足够的权限执行被保护操作进行判断。
为什么要OAuth认证?
(引述别人的话)
如果你开车去酒店赴宴,你经常会苦于找不到停车位而耽误很多时间。是否有好办法可以避免这个问题呢?有的,听说有一些豪车的车主就不担心这个问题。豪车一般配备两种钥匙:主钥匙和泊车钥匙。当你到酒店后,只需要将泊车钥匙交给服务生,停车的事情就由服务生去处理。与主钥匙相比,这种泊车钥匙的使用功能是受限制的:它只能启动发动机并让车行驶一段有限的距离,可以锁车,但无法打开后备箱,无法使用车内其他设备。这里就体现了一种简单的“开放授权”思想:通过一把泊车钥匙,车主便能将汽车的部分使用功能(如启动发动机、行驶一段有限的距离)授权给服务生。
新浪微博OAuth在iOS客户端中的应用
NSString *string = [NSString stringWithFormat:@"https://api.weibo.com/oauth2/authorize?client_id=%@&redirect_uri=%@&display=mobile",APPKEY,REDIRECT_URL]; NSURL *url = [NSURL URLWithString:string]; //用户授权请求 NSURLRequest *request = [NSURLRequest requestWithURL:url]; //加载请求界面 [_webView loadRequest:request];
执行上面的代码我们会调出下面的界面
输入账号信息,点击登录:
点击授权的时候,我们可以在代理方法中打印中请求对象的url.
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType { NSLog(@"网页加载请求的时候加载此方法"); //根据请求对象,获取请求url //absoluteString把url转化成字符串 NSString *string = request.URL.absoluteString; NSLog(@"%@",string); return YES; }
打印信息为:
2015-09-22 13:12:14.339 weibo[1907:60b] http://www.cnblogs.com/wangyaoguo?code=16cca5f05d27854d0341c76cb334390e
我们可以提取出code信息。
然后根据code,发起请求,获取access_token,然后根据access_token发起请求,获取微博数据。
下面的代码是发一条微博的代码段:
#import "ViewController.h"
#import "ASIFormDataRequest.h"
#import "sendViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
NSString *access = [[NSUserDefaults standardUserDefaults] objectForKey:@"access_token"];
if (access.length > 0)
{
UIWindow *window = [UIApplication sharedApplication].delegate.window;
sendViewController *send = [[sendViewController alloc]init];
window.rootViewController = send;
}
else
{
[self loadLoginScreen];
}
}
//载入登录界面
-(void)loadLoginScreen
{
NSString *string = [NSString stringWithFormat:@"https://api.weibo.com/oauth2/authorize?client_id=%@&redirect_uri=%@&display=mobile",APPKEY,REDIRECT_URL];
NSURL *url = [NSURL URLWithString:string];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
[_webView loadRequest:request];
_webView.delegate = self;
}
//webview代理方法页面载入时调用该方法
//根据相关参数,向服务器发起请求,获取access_token信息
- (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType
{
NSString *string = request.URL.absoluteString;
NSRange range = [string rangeOfString:@"code"];
if (range.location != NSNotFound)
{
NSString *code = [string componentsSeparatedByString:@"code="][1];
NSURL *url = [NSURL URLWithString:@"https://api.weibo.com/oauth2/access_token"];
ASIFormDataRequest *formRequest = [ASIFormDataRequest requestWithURL:url];
formRequest.tag = 10;
[formRequest setPostValue:APPKEY forKey:@"client_id"];
[formRequest setPostValue:APPSECRET forKey:@"client_secret"];
[formRequest setPostValue:@"authorization_code" forKey:@"grant_type"];
[formRequest setPostValue:code forKey:@"code"];
[formRequest setPostValue:REDIRECT_URL forKey:@"redirect_uri"];
formRequest.delegate = self;
[formRequest startAsynchronous];
}
return YES;
}
//请求完成是调用,获取到access_token
- (void)requestFinished:(ASIHTTPRequest *)request;
{
if (request.tag == 10)
{
NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:request.responseData options:NSJSONReadingMutableContainers error:nil];
//获取数据的通行证(令牌)
_accessToken = [dic objectForKey:@"access_token"];
NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
[userDefaults setObject:_accessToken forKey:@"access_token"];
}
sendViewController *send = [[sendViewController alloc]init];
[self presentViewController:send animated:YES completion:nil];
}
- (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
@end
点击发送按钮:
- (IBAction)sendWeibo:(id)sender { NSString *content = _textView.text; if (content.length > 0) { ASIFormDataRequest *sendRequest = [ASIFormDataRequest requestWithURL:[NSURL URLWithString:@"https://api.weibo.com/2/statuses/update.json"]]; NSString *accessToken = [[NSUserDefaults standardUserDefaults] objectForKey:@"access_token"]; [sendRequest setPostValue:accessToken forKey:@"access_token"]; [sendRequest setPostValue:_textView.text forKey:@"status"]; sendRequest.delegate = self; [sendRequest startAsynchronous]; } }