07-Access Token

Access Token

课程目标

  • 自定义对象
  • 构造函数
  • 归档 & 接档

接口定义

文档地址

http://open.weibo.com/wiki/OAuth2/access_token

接口地址

https://api.weibo.com/oauth2/access_token

HTTP 请求方式

  • POST

请求参数

参数 描述
client_id 申请应用时分配的AppKey
client_secret 申请应用时分配的AppSecret
grant_type 请求的类型,填写 authorization_code
code 调用authorize获得的code值
redirect_uri 回调地址,需需与注册应用里的回调地址一致

返回数据

返回值字段 字段说明
access_token 用于调用access_token,接口获取授权后的access token
expires_in access_token的生命周期,单位是秒数
remind_in access_token的生命周期(该参数即将废弃,开发者请使用expires_in)
uid 当前授权用户的UID

用户账户模型

加载AccessToken

  • HMOAuthViewController 中增加函数加载 AccessToken
/// 加载AccessToken的方法
///
/// - parameter code: 授权码
private func loadAccessToken(code: String){
    let urlString = "https://api.weibo.com/oauth2/access_token"

    // 定义参数
    let params = [
        "client_id": WB_APPKEY,
        "client_secret": WB_APPSECRET,
        "grant_type": "authorization_code",
        "code": code,
        "redirect_uri": WB_REDIRECTURI]

    NetworkTools.shareTools.request(.POST, url: urlString, params: params) { (result, error) -> () in
        print(result)
    }
}
  • 在获取授权码成功之后调用方法
// 判断是否包含 'code=' 字样
if let query = request.URL?.query where query.containsString("code=") {
    // 截取授权码
    let code = query.substringFromIndex("code=".endIndex)
    print("请求码:\(code)")
    // 请求AccessToken
    loadAccessToken(code)
}else{
    // 用户点击了取消授权,直接关闭页面
    self.close()
}

运行测试

  • 返回错误信息
Error Domain=com.alamofire.error.serialization.response Code=-1016 "Request failed: unacceptable content-type: text/plain"
  • HMNetworkTools 单例初始化时设置反序列化数据格式
static let shareTools: HMNetworkTools = {
    let tools = HMNetworkTools()
    tools.responseSerializer.acceptableContentTypes?.insert("text/plain")
    return tools
}()

定义 HMUserAcount 模型

  • Model 目录下添加 HMUserAccount
  • 定义模型属性
/// 用户帐号模型
/// - see: [http://open.weibo.com/wiki/OAuth2/access_token](http://open.weibo.com/wiki/OAuth2/access_token)
class HMUserAccount: NSObject {

    // 用于调用access_token,接口获取授权后的access token
    var access_token: String?
    // access_token的生命周期,单位是秒数
    var expires_in: NSTimeInterval = 0
    // 当前授权用户的UID
    var uid: String?

    init(dict: [String: AnyObject]) {
        super.init()

        setValuesForKeysWithDictionary(dict)
    }

    override func setValue(value: AnyObject?, forUndefinedKey key: String) {}
}
  • 创建用户帐号模型,在 HMOAuthViewController 的网络回调代码中添加如下代码
let account = HMUserAccount(dict: result as! [String: AnyObject])
print(account)
  • 调试模型信息

  • 与 OC 不同,如果要在 Swift 1.2 中调试模型信息,需要遵守 Printable 协议,并且重写 descriptiongetter 方法,在 Swift 2.0 中,description 属性定义在 CustomStringConvertible 协议中

override var description: String {
    let keys = ["access_token", "expires_in", "uid"]
    return dictionaryWithValuesForKeys(keys).description
}

建议description此代码抽取到Xcode的代码块中,方便使用

设置过期日期

过期日期

  • 在新浪微博返回的数据中,过期日期是以当前系统时间加上秒数计算的,为了方便后续使用,增加过期日期属性

  • 定义属性

/// token过期日期
var expiresDate: NSDate?
  • expires_indidSet 方法里面给 expiresDate 赋值
var expires_in: NSTimeInterval = 0 {
    didSet{
        expiresDate = NSDate(timeIntervalSinceNow: expires_in)
    }
}
  • 修改 description
let keys = ["access_token", "expires_in", "expiresDate", "uid"]

加载用户信息

课程目标

  • 通过 AccessToken 获取新浪微博网络数据

接口定义

文档地址

http://open.weibo.com/wiki/2/users/show

接口地址

https://api.weibo.com/2/users/show.json

HTTP 请求方式

  • GET

请求参数

参数 描述
access_token 采用OAuth授权方式为必填参数,其他授权方式不需要此参数,OAuth授权后获得
uid 需要查询的用户ID

返回数据

返回值字段 字段说明
name 友好显示名称
avatar_large 用户头像地址(大图),180×180像素

测试 URL

https://api.weibo.com/2/users/show.json?access_token=2.00ml8IrF0qLZ9W5bc20850c50w9hi9&uid=5365823342

代码实现

  • HMOAuthViewController 中添加 loadUserInfo 方法
/// 加载用户数据
///
/// - parameter userAccount: 用户账户模型
private func loadUserInfo(userAccount: HMUserAccount) {

    let urlString = "https://api.weibo.com/2/users/show.json"
    // 定义参数
    let params = [
        "uid": userAccount.uid!,
        "access_token": userAccount.access_token!
    ]

    HMNetworkTools.shareTools.request(.GET, url: urlString, params: params) { (result, error) -> () in
        if error != nil {
            print("请求失败:\(error)")
            return
        }
        print(result)
    }
}
  • loadAccessToken 方法中请求accessToken成功后调用该方法
// 将返回字典转成模型
let account = HMUserAccount(dict: result as! [String: AnyObject])
print(account)
// 请求用户数据
self.loadUserInfo(account)

注意:在 Swift 中,闭包中输入代码的智能提示非常不好,因此新建一个函数单独处理加载用户功能

扩展用户模型

  • HMUserAccount 中增加用户名和头像属性
/// 友好显示名称
var screen_name: String?
/// 用户头像地址(大图),180×180像素
var avatar_large: String?
  • 扩展 description 中的属性
override var description: String {
    let keys = ["access_token", "expires_in", "expiresDate", "uid", "screen_name", "avatar_large"]

    return dictionaryWithValuesForKeys(keys).description
}
  • 在用户信息请求成功之后获取昵称与头像
// 发送get请求
HMNetworkTools.shareTools.request(.GET, url: urlString, params: params) { (result, error) -> () in
    if error != nil {
        print("请求失败:\(error)")
        return
    }
    guard let dict = result as? [String: AnyObject] else {
        return
    }
    // 设置请求回来的用户昵称和头像
    userAccount.screen_name = dict["screen_name"] as? String
    userAccount.avatar_large = dict["avatar_large"] as? String
}

每一个令牌授权一个 特定的网站特定的时段内 访问 特定的资源

归档 & 解档

课程目标

  • 对比 OC 的归档 & 解档实现

  • 利用归档 & 解档保存用户信息

  • 遵守协议

class HMUserAccount: NSObject, NSCoding
  • 实现协议方法
// MARK: - NSCoding
/// 归档 - 将对象以二进制形式保存至磁盘前被调用,与网络的序列化类似
func encodeWithCoder(aCoder: NSCoder) {
    aCoder.encodeObject(access_token, forKey: "access_token")
    aCoder.encodeObject(expiresDate, forKey: "expiresDate")
    aCoder.encodeObject(uid, forKey: "uid")
    aCoder.encodeObject(name, forKey: "name")
    aCoder.encodeObject(avatar_large, forKey: "avatar_large")
}

/// 解档 - 将二进制数据从磁盘读取并且转换成对象时被调用,与网络的反序列化类似
required init?(coder aDecoder: NSCoder) {
    access_token = aDecoder.decodeObjectForKey("access_token") as? String
    expiresDate = aDecoder.decodeObjectForKey("expiresDate") as? NSDate
    uid = aDecoder.decodeObjectForKey("uid") as? String
    name = aDecoder.decodeObjectForKey("name") as? String
    avatar_large = aDecoder.decodeObjectForKey("avatar_large") as? String
}
  • 实现将当前对象归档保存的函数
/// 归档保存当前对象
func saveUserAccount(){
    let path = (NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true).last! as NSString).stringByAppendingPathExtension("userAccount.archive")!
    print("归档保存路径:\(path)")
    // 归档
    NSKeyedArchiver.archiveRootObject(self, toFile: path)
}

视图模型

目标一:建立视图模型,抽取网络请求代码

  • 建立用户账户的视图模型
/// 用户账户视图模型
class HMUserAccountViewModel: NSObject {

    /// 单例
    static let sharedUserAccount = HMUserAccountViewModel()

    /// 用户账户模型
    var userAccount: HMUserAccount?
}
  • 抽取网络请求代码到视图模型中
/// 加载AccessToken并且加载用户信息
///
/// - parameter code:     授权码
/// - parameter complete: 完成回调
func loadAccessToken(code: String, complete:(isSuccessed: Bool)->()){

    let urlString = "https://api.weibo.com/oauth2/access_token"

    // 定义参数
    let params = [
        "client_id": WB_APPKEY,
        "client_secret": WB_APPSECRET,
        "grant_type": "authorization_code",
        "code": code,
        "redirect_uri": WB_REDIRECTURI]

    HMNetworkTools.sharedTools.request(.POST, urlString: urlString, parameters: params) { (response, error) -> () in

        if error != nil {
            print(error)
            return
        }

        let account = HMUserAccount(dict: response as! [String: AnyObject])
        print(account)
        self.userAccount = account
        // 请求用户数据
        self.loadUserInfo(account.uid!, accessToken: account.access_token!, complete: complete)
    }
}

/// 加载用户信息
///
/// - parameter uid:         用户uid
/// - parameter accessToken: accessToken
/// - parameter complete:    完成回调
private func loadUserInfo(uid: String, accessToken: String, complete:(isSuccessed: Bool)->()){

    let urlString = "https://api.weibo.com/2/users/show.json"
    // 组织参数
    let params = [
        "access_token": accessToken,
        "uid": uid
    ]
    // 发送请求
    HMNetworkTools.sharedTools.request(.GET, urlString: urlString, parameters: params) { (response, error) -> () in

        if error != nil {
            print("请求失败\(error)")
            complete(isSuccessed: false)
            return
        }
        print(response)

        // 在 if let 或者 guard let 里面 使用 as 都使用 ?
        guard let responseDic = response as? [String: AnyObject] else {
            print("返回数据不是一个字典")
            complete(isSuccessed: false)
            return
        }
        // 赋值用户的头像与昵称
        self.userAccount?.screen_name = responseDic["screen_name"] as? String
        self.userAccount?.avatar_large = responseDic["avatar_large"] as? String

        complete(isSuccessed: true)
    }
}
  • 修改 HMOAuthViewController 中的代码调用
// 请求  AccessToken以及用户数据
HMUserAccountViewModel.sharedUserAccount.loadAccessToken(code, complete: { (isSuccessed) -> () in
    if isSuccessed {
        print("个人信息请求成功,跳转界面")
    }else{
        print("个人信息请求失败")
    }
})
  • 删除控制器里面 loadAccessTokenloadUserInfo 两个方法

控制器的代码简单了!

目标二:加载归档保存的 token,避免重复登录

  • 抽取归档代码到 ViewModel中,并且实现解档方法
// MARK: - 保存 & 加载
/// 解档归档路径
    // 归档与解档的路径
    private var archivePath: String = (NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true).last! as NSString).stringByAppendingPathComponent("useraccount.archive")

// 归档当前对象
func saveAccount(account: HMUserAccount) {
    // 归档
    NSKeyedArchiver.archiveRootObject(account, toFile: archivePath)
}

// 解档
func loadUserAccount() -> HMUserAccount? {
    // 解档
    let result = NSKeyedUnarchiver.unarchiveObjectWithFile(archivePath) as? HMUserAccount
    return result
}
  • 在 AppDelegate 中添加以下代码测试
printLog(HMUserAccountViewModel.sharedUserAccount.loadUserAccount())
  • 代码优化,在视图模型初始化方法里面添加解档的逻辑
private override init() {
    super.init()
    userAccount = loadUserAccount()
}

private init 可以避免外部调用构造函数创建对象

  • view model 中添加 accessToken 计算型属性
// 提供给外界accessToken的值
var accessToken: String? {
    return userAccount?.access_token
}
  • 添加 isExpires 判断帐号是否过期
// 帐号是否过期
var isExpires: Bool {
    if userAccount?.expiresDate?.compare(NSDate()) == NSComparisonResult.OrderedDescending {
        return false
    }
    return true
}
  • 更改 accessToken 的 get 方法
// 访问令牌
var accessToken: String? {
    // 其内部判断 accessToken 是否有
    if !isExpires {
        return userAccount?.access_token
    }
    return nil
}
  • 修改 HMVisitorViewController 中的用户登录标记
var userLogon = HMUserAccountViewModel.sharedUserAccount.accessToken != nil
  • 在视图模型中增加 userLogon 计算型属性
/// 用户登录标记
var userLogon: Bool {
    return userAccount?.access_token != nil
}
  • 再次修改 HMVisitorViewController 中的用户登录标记
var userLogon = HMUserAccountViewModel.sharedUserAccount.userLogon
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,716评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,558评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,431评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,127评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,511评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,692评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,915评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,664评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,412评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,616评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,105评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,424评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,098评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,096评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,869评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,748评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,641评论 2 271

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,100评论 18 139
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,617评论 4 59
  • 5 分享内容到你的网站 上一章中,你在网站中构建了用户注册和认证。你学会了如何为用户创建自定义的个人资料模型,并添...
    lakerszhy阅读 1,602评论 5 16
  • 自学写下第一篇产品体验报告,诚邀互联网前辈们给出意见~ 产品体验报告框架产品体验报告框架图 1.市场分析(战略层)...
    叮咚酱阅读 2,726评论 4 10
  • 一眨眼来澳洲四年多了,辗转了两个城市,换了四个工作,突然想分享一下自己的一些经验和感受,或许能对新来的或者准备来的...
    aussieweever阅读 3,726评论 2 4