Tiercel 2!完美支持原生级别后台下载

项目地址:https://github.com/Danie1s/Tiercel

Tiercel是一个简单易用且功能丰富的纯Swift下载框架,支持原生级别后台下载,拥有强大的任务管理功能,满足下载类APP的大部分需求。

Tiercel 2:

Tiercel 2 是全新的版本,下载实现基于URLSessionDownloadTask,支持原生的后台下载,功能更加强大,使用方式也有了一些改变,不兼容旧版本,请注意新版的使用方法。如果想了解后台下载的细节和注意事项,可以看这篇文章:iOS原生级别后台下载详解

旧版本下载实现基于URLSessionDataTask,不支持后台下载,已经移至dataTask分支,原则上不再更新,如果不需要后台下载功能,或者不想迁移到新版,可以直接下载dataTask分支的源码使用,也可以在Podfile里使用以下方式安装:

source 'https://github.com/CocoaPods/Specs.git'
platform :ios, '8.0'
use_frameworks!

target '<Your Target Name>' do
    pod 'Tiercel', :git => 'https://github.com/Danie1s/Tiercel.git', :branch => 'dataTask'
end

Features:

  • [x] 原生级别的后台下载
  • [x] 支持离线断点续传,无论crash还是手动Kill App都不会影响
  • [x] 精细的任务管理,每个下载任务都可以单独管理操作和状态回调
  • [x] 支持多个下载模块,每个模块拥有一个管理者,每个模块互不影响
  • [x] 下载模块的管理者拥有总任务的状态回调
  • [x] 内置了下载速度、剩余时间等常见的下载信息
  • [x] 链式语法调用
  • [x] 支持控制下载任务的最大并发数
  • [x] 支持文件校验
  • [x] 线程安全

Requirements

  • iOS 8.0+
  • Xcode 10.0+
  • Swift 4.2+

Installation

CocoaPods

CocoaPods is a dependency manager for Cocoa projects. You can install it with the following command:

$ gem install cocoapods

CocoaPods 1.1+ is required to build Tiercel.

To integrate Tiercel into your Xcode project using CocoaPods, specify it in your Podfile:

source 'https://github.com/CocoaPods/Specs.git'
platform :ios, '8.0'
use_frameworks!

target '<Your Target Name>' do
    pod 'Tiercel'
end

Then, run the following command:

$ pod install

Manually

If you prefer not to use any of the aforementioned dependency managers, you can integrate Tiercel into your project manually.

Example

To run the example project, clone the repo, and run Tiercel.xcodeproj .

批量下载
跨控制器下载

Usage

配置

Tiercel内置一个全局的TRManager.default单例,因为支持原生后台下载,所以需要在AppDelegate 文件里配置

以下为内置的default单例配置方法,如果需要使用多个下载模块,或者需要自定义TRManager,可参照Demo

// 在AppDelegate文件里

func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {

    // 如果有使用内置全局的default单例,必须在此方法内调用一次,否则不会在App启动的时候初始化
    
    // 在这里进行初始化的配置,也可以在任何地方,随时进行配置
    TRManager.default.configuration.allowsCellularAccess = true
    
    return true
}

// 必须实现此方法,并且把identifier对应的completionHandler保存起来
func application(_ application: UIApplication, handleEventsForBackgroundURLSession identifier: String, completionHandler: @escaping () -> Void) {

    if TRManager.default.identifier == identifier {
        TRManager.default.completionHandler = completionHandler
    }

}

基本用法

一行代码开启下载

// 创建下载任务并且开启下载,同时返回可选类型的TRDownloadTask实例,如果URLString无效,则返回nil
let task = TRManager.default.download("http://api.gfs100.cn/upload/20171219/201712191530562229.mp4")

// 批量创建下载任务并且开启下载,返回有效URLString对应的任务数组,URLStrings需要跟fileNames一一对应
let tasks = TRManager.default.multiDownload(URLStrings)

如果需要设置回调

// 回调闭包的参数是TRDownloadTask实例,可以得到所有相关的信息
// 回调闭包都是在主线程运行
// progress 闭包:如果任务正在下载,就会触发
// success 闭包:任务已经下载过,或者下载完成,都会出发,这时候task.status == .succeeded
// failure 闭包:只要task.status != .succeeded,就会触发:
//    1. 暂停任务,这时候task.status == .suspended
//    2. 任务下载失败,这时候task.status == .failed
//    3. 取消任务,这时候task.status == .canceled
//    4. 移除任务,这时候task.status == .removed
let task = TRManager.default.download("http://api.gfs100.cn/upload/20171219/201712191530562229.mp4")

task?.progress({ (task) in
    let progress = task.progress.fractionCompleted
    print("下载中, 进度:\(progress)")
}).success({ (task) in
    print("下载完成")
}).failure({ (task) in
    print("下载失败")
})

下载任务的管理和操作。在Tiercel中,URLString是下载任务的唯一标识,如果需要对下载任务进行操作,则使用TRManager实例对URLString进行操作。

let URLString = "http://api.gfs100.cn/upload/20171219/201712191530562229.mp4"

// 创建下载任务并且开启下载,同时返回可选类型的TRDownloadTask实例,如果URLString无效,则返回nil
let task = TRManager.default.download(URLString)
// 根据URLString查找下载任务,返回可选类型的TRTask实例,如果不存在,则返回nil
let task = TRManager.default.fetchTask(URLString)

// 开始下载
// 如果调用suspend暂停了下载,可以调用这个方法继续下载
TRManager.default.start(URLString)

// 暂停下载
TRManager.default.suspend(URLString)

// 取消下载,没有下载完成的任务会被移除,不保留缓存,已经下载完成的不受影响
TRManager.default.cancel(URLString)

// 移除下载,任何状态的任务都会被移除,没有下载完成的缓存文件会被删除,可以选择是否保留已经下载完成的文件
TRManager.default.remove(URLString, completely: false)

// 除了可以对单个任务进行操作,TRManager也提供了对所有任务同时操作的API
TRManager.default.totalStart()
TRManager.default.totalSuspend()
TRManager.default.totalCancel()
TRManager.default.totalRemove(completely: false)

后台下载

Tiercel 2 的下载实现基于URLSessionDownloadTask,支持原生的后台下载,按照苹果官方文档的要求,TRManager实例必须在App启动的时候创建,并且在AppDelegate 文件里实现以下方法

// 必须实现此方法,并且把identifier对应的completionHandler保存起来
func application(_ application: UIApplication, handleEventsForBackgroundURLSession identifier: String, completionHandler: @escaping () -> Void) {

    if TRManager.default.identifier == identifier {
        TRManager.default.completionHandler = completionHandler
    }
}

只要使用Tiercel 开启了下载任务:

  • 手动Kill App,任务会暂时,重启App后可以恢复进度,继续下载
  • 只要不是手动Kill App,任务会一直在下载
    • 把App退回后台,任务会一直在下载
    • 无论在前台还是后台,如果App崩溃或者被系统关闭,任务还是会一直在下载
    • 重启手机,任务会一直在下载

如果想了解后台下载的细节和注意事项,可以看这篇文章:iOS原生级别后台下载详解

文件校验

Tiercel提供了文件校验功能,可以根据需要添加,校验结果在回调的task.validation

// 回调闭包在主线程运行
let task = TRManager.default.download("http://dldir1.qq.com/qqfile/QQforMac/QQ_V4.2.4.dmg")

task?.validateFile(verificationCode: "9e2a3650530b563da297c9246acaad5c",
                   verificationType: .md5,
                   validateHandler: { (task) in
                    if task.validation == .correct {
                        // 文件正确
                    } else {
                        // 文件错误
                    }
})

TRChecksumHelper是文件校验的工具类,可以直接使用它对已经存在的文件进行校验

/// 对文件进行校验,是在子线程进行的
///
/// - Parameters:
///   - filePath: 文件路径
///   - verificationCode: 文件的Hash值
///   - verificationType: Hash类型
///   - completion: 完成回调
public class func validateFile(_ filePath: String, 
                               verificationCode: String, 
                               verificationType: TRVerificationType, 
                               completion: @escaping (Bool) -> ()) {
    
}

TRManager

TRManager是下载任务的管理者,管理当前模块所有下载任务,内置一个全局的default单例,如果需要多个下载模块,或者需要自定义TRManager,可以手动创建TRManager实例。

⚠️⚠️⚠️ 按照苹果官方文档的要求,TRManager实例必须在App启动的时候创建,即TRManager的生命周期跟App几乎一致,为方便使用,最好是作为AppDelegate的属性,或者是全局变量,具体请参照Demo

///  初始化方法
///
/// - Parameters:
///   - identifier: 设置TRManager实例的标识,区分不同的下载模块,同时为urlSession的标识,原生级别的后台下载必须要有唯一标识
public init(_ identifier: String) {
    // 实现的代码... 
}

TRManager作为所有下载任务的管理者,也可以设置回调

// 回调闭包的参数是TRManager实例,可以得到所有相关的信息
// 回调闭包都是在主线程运行
// progress 闭包:只要有一个任务正在下载,就会触发
// success 闭包:只有一种情况会触发:
//    所有任务都下载成功(取消和移除的任务会被移除然后销毁,不再被manager管理) ,这时候manager.status == .succeeded
// failure 闭包:只要manager.status != .succeeded,就会触发:
//    1. 调用全部暂停的方法,或者没有等待运行的任务,也没有正在运行的任务,这时候manager.status == .suspended
//    2. 所有任务都结束,但有一个或者多个是失败的,这时候manager.status == .failed
//    3. 调用全部取消的方法,或者剩下一个任务的时候把这个任务取消,这时候manager.status == .canceled
//    4. 调用全部移除的方法,或者剩下一个任务的时候把这个任务移除,这时候manager.status == .removed
TRManager.default.progress { (manager) in
    let progress = manager.progress.fractionCompleted
    print("downloadManager运行中, 总进度:\(progress)")
    }.success { (manager) in
         print("所有下载任务都成功了")
    }.failure { (manager) in
         if manager.status == .suspended {
            print("所有下载任务都暂停了")
        } else if manager.status == .failed {
            print("存在下载失败的任务")
        } else if manager.status == .canceled {
            print("所有下载任务都取消了")
        } else if manager.status == .removed {
            print("所有下载任务都移除了")
        }
}

TRManager的主要属性

// 内置的全局单例
public static let `default` = TRManager("default")
// 设置内置日志打印等级,如果为none则不打印
public static var logLevel: TRLogLevel = .detailed
// 是否需要对networkActivityIndicator进行管理
public static var isControlNetworkActivityIndicator = true
// TRManager的状态
public var status: TRStatus = .waiting
// TRManager的缓存管理实例
public var cache: TRCache
// TRManager的标识,区分不同的下载模块
public let identifier: String
// TRManager的进度
public var progress: Progress
// TRManager的配置,可以设置请求超时时间,最大并发数,是否允许蜂窝网络下载
public var configuration = TRConfiguration()
// 所有下载中的任务加起来的总速度
public private(set) var speed: Int64 = 0
// 所有下载中的任务需要的剩余时间
public private(set) var timeRemaining: Int64 = 0
// manager管理的下载任务,取消和移除的任务会被销毁,但操作是异步的,在回调闭包里面获取才能保证正确
public var tasks: [TRTask] = []

TRConfiguration

TRConfiguration是Tiercel中配置TRManager的结构体,可配置属性如下:

// 请求超时时间
public var timeoutIntervalForRequest = 30.0

// 最大并发数
public var maxConcurrentTasksLimit = Int.max

// 是否允许蜂窝网络下载
public var allowsCellularAccess = false

更改TRManager的配置

// 无论是否有下载任务正在运行,都可以在任何地方,随时配置
// 如果只是更改某一项,可以直接对TRManager属性设置
TRManager.default.configuration.allowsCellularAccess = true

// 如果是需要更改多项,最好是重新创建TRConfiguration
let configuration = TRConfiguration()
configuration.allowsCellularAccess = true
configuration.maxConcurrentTasksLimit = 2
configuration.timeoutIntervalForRequest = 60

TRManager.default.configuration = configuration

TRDownloadTask

TRDownloadTask是Tiercel中的下载任务类,继承自TRTask。在Tiercel中,URLString是下载任务的唯一标识,URLString代表着任务,如果需要对下载任务进行操作,则使用TRManager实例对URLString进行操作。所以TRDownloadTask实例都是由TRManager实例创建,单独创建没有意义。

主要属性

// 保存到沙盒的下载文件的文件名,如果在下载的时候没有设置,则默认为url的md5加上文件扩展名
public internal(set) var fileName: String
// 下载任务对应的URLString
public let URLString: String
// 下载任务的状态
public var status: TRStatus
// 下载文件的校验状态
public var validation: TRValidation
// 下载任务的进度
public var progress: Progress = Progress()
// 下载任务的开始日期
public var startDate: TimeInterval = 0
// 下载任务的结束日期
public var endDate: TimeInterval = Date().timeIntervalSince1970
// 下载任务的速度
public var speed: Int64 = 0
// 下载任务的剩余时间
public var timeRemaining: Int64 = 0
// 下载文件路径
public var filePath: String
// 下载文件的扩展名
public var pathExtension: String?

对下载任务操作,必须通过TRManager实例进行,不能用TRDownloadTask实例直接操作

  • 开启
  • 暂停
  • 取消,没有完成的任务从TRManager实例中的tasks中移除,不保留缓存,已经下载完成的任务不受影响
  • 移除,已经完成的任务也会被移除,没有下载完成的缓存文件会被删除,已经下载完成的文件可以选择是否保留

注意:对下载中的任务进行暂停、取消和移除操作,结果是异步回调的,在回调闭包里面获取状态才能保证正确

TRCache

TRCache是Tiercel中负责管理缓存下载任务信息和下载文件的类。TRCache实例一般作为TRManager实例的属性来使用,同样地,Tiercel内置一个全局的TRCache.default单例,对应TRManager.default

/// 初始化方法
///
/// - Parameters:
///   - name: 不同的name,代表不同的下载模块,对应的文件放在不同的地方,对应TRManager创建时传入的identifier
public init(_ name: String) {
    // 实现的代码...
}

主要属性

// 下载模块的目录路径
public let downloadPath: String

// 没有完成的下载文件缓存的目录路径
public let downloadTmpPath: String

// 下载完成的文件的目录路径
public let downloadFilePath: String

主要API分成几大类:

  • 检查沙盒是否存在文件
  • 移除跟下载任务相关的文件
  • 保存跟下载任务相关的文件
  • 读取下载任务相关的文件,获得下载任务相关的信息

License

Tiercel is available under the MIT license. See the LICENSE file for more info.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345