| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552 | ////  PQAliOssUtil.swift//  PQSpeed////  Created by SanW on 2020/12/9.//  Copyright © 2020 BytesFlow. All rights reserved.//import UIKitimport BFCommonKitimport BFNetRequestKitimport BFAnalyzeKit// MARK: - 阿里OSS工具类/// 阿里OSS工具类public class PQAliOssUtil: NSObject {    static public let shared = PQAliOssUtil()    public var client: OSSClient?    // 文件类型:materialType (1:PICTURE, 2:VIDEO, 3:VOICE, 4:FILE, 5:GIF)    public  var aliOssHander: ((_ isMatarialUpload: Bool, _ materialType: StickerType, _ fileExtensions: String, _ code: Int, _ objectKey: String?, _ contentMD5: String, _ width: CGFloat, _ height: CGFloat, _ duration: CGFloat, _ frameNumber: Int, _ netResourceUrl: String?, _ fileURL: URL?, _ data: Data?, _ msg: String?) -> Void)?    public  var aliOssProgressHander: ((_ bytesSent: Int64, _ totalBytesSent: Int64, _ totalBytesExpectedToSend: Int64, _ resttime: Int64, _ uploadSpeed: String?) -> Void)?    public  var lastInterfaceBytes: Int64 = 0 // 上次网速    public var oldTime: Int = 0 // 上次进度时间S    public var oloaded: Int64 = 0    public var dics: [String: Any] = [:]    public  var allTasks: [String: OSSMultipartUploadRequest] = [:] // add by ak 保存当前所有任务 用于取消某个任务使用    public  func startClient(accessKeyId: String, secretKeyId: String, securityToken: String, endpoint: String) -> PQAliOssUtil {        if endpoint.count == 0 {            debugPrint("endpoint is nil xxxxxx \(endpoint)")        }        let credential = OSSStsTokenCredentialProvider(accessKeyId: accessKeyId, secretKeyId: secretKeyId, securityToken: securityToken)        let conf = OSSClientConfiguration()        conf.timeoutIntervalForRequest = 60 // 连接超时,默认15秒        conf.maxRetryCount = 3        conf.maxConcurrentRequestCount = 5 // 最大并发请求书,默认5个        client = OSSClient(endpoint: endpoint, credentialProvider: credential, clientConfiguration: conf)        return .shared    }    /// 图片上传    /// - Parameters:    ///   - bucketName: <#bucketName description#>    ///   - objectKey: <#objectKey description#>    ///   - data: <#data description#>    /// - Returns: <#description#>    public  func uploadObjectAsync(bucketName: String, objectKey: String, data: Data, materialType: StickerType = .IMAGE, fileExtensions: String, isMatarialUpload: Bool = false, contentMD5: String = "", width: CGFloat = 0, height: CGFloat = 0, imageUploadBlock: @escaping (_ osstask: OSSTask<AnyObject>?, _ code: Int, _ objectKey: String, _ fileExtensions: String) -> Void) -> PQAliOssUtil {        let putRequest: OSSPutObjectRequest = OSSPutObjectRequest()        putRequest.bucketName = bucketName        putRequest.objectKey = objectKey        putRequest.uploadingData = data        putRequest.uploadProgress = { [weak self] _, totalByteSent, totalBytesExpectedToSend in            debugPrint("文件上传进度:totalByteSent = \(totalByteSent),totalBytesExpectedToSend = \(totalBytesExpectedToSend)")            if totalBytesExpectedToSend > 0, totalByteSent == totalBytesExpectedToSend {                if isMatarialUpload, self?.aliOssHander != nil {                    self?.aliOssHander!(isMatarialUpload, materialType, fileExtensions, 1, objectKey, contentMD5, width, height, 0, 0, nil, nil, data, "上传成功")                }            }        }        putRequest.contentType = "application/octet-stream"        //        putRequest.contentMd5 = contentMD5        putRequest.contentEncoding = ""        putRequest.contentDisposition = ""        let putTask: OSSTask = (client?.putObject(putRequest))!        putTask.continue(successBlock: { [weak self] (osstask) -> Any? in            if osstask.error == nil {                let task = self?.client?.presignPublicURL(withBucketName: putRequest.bucketName, withObjectKey: putRequest.objectKey)                debugPrint("图片原方法上传完成=\(objectKey)")                debugPrint("url == \(task?.result ?? "" as AnyObject)")                imageUploadBlock(osstask, 1, objectKey, fileExtensions)            } else {                if self?.aliOssHander != nil {                    self?.aliOssHander!(isMatarialUpload, materialType, fileExtensions, 0, objectKey, contentMD5, width, height, 0, 0, nil, nil, data, "上传失败")                }                debugPrint("图片原方法上传失败=\(objectKey),osstask.error = \(osstask.error ?? PQError(msg: "失败"))")                imageUploadBlock(osstask, 0, objectKey, fileExtensions)            }            return nil        }).waitUntilFinished()        return .shared    }    /// 快速上传视频    /// - Parameters:    ///   - localPath: <#localPath description#>    ///   - response: <#response description#>    /// - Returns: <#description#>    class public  func multipartUpload(localPath: String, response: [String: Any]?,videoSource:String? = nil) {        let accessKeyId: String = "\(response?["AccessKeyId"] ?? "")"        let secretKeyId: String = "\(response?["AccessKeySecret"] ?? "")"        let securityToken: String = "\(response?["SecurityToken"] ?? "")"        let endpoint: String = "\(response?["Host"] ?? "")"        let bucketName: String = "\(response?["Bucket"] ?? "")"        let FileName: String = "\(response?["FileName"] ?? "")"        let uploadID: String = "\(response?["Upload"] ?? "")"        var endpoints: [String] = Array<String>.init()        if response?.keys.contains("Hosts") ?? false {            endpoints = response?["Hosts"] as! [String]            endpoints.append(endpoint)        }        PQAliOssUtil.shared.PQOSSMultipartUpload(accessKeyId: accessKeyId, secretKeyId: secretKeyId, securityToken: securityToken, bucketName: bucketName, endpoints: endpoints, FileName: FileName, fileURL: URL(fileURLWithPath: localPath.replacingOccurrences(of: "file:///", with: "")), ossUploadID: uploadID, videoSource: videoSource)    }    /// 重新上传    /// - Parameters:    ///   - response: <#response description#>    ///   - localPath: <#localPath description#>    /// - Returns: <#description#>    public  func reloadTask(response: [String: Any]?, localPath: String?,videoSource:String? = nil) {        if response == nil {            return        }        let accessKeyId: String = "\(response?["AccessKeyId"] ?? "")"        let secretKeyId: String = "\(response?["AccessKeySecret"] ?? "")"        let securityToken: String = "\(response?["SecurityToken"] ?? "")"        let endpoint: String = "\(response?["Host"] ?? "")"        let bucketName: String = "\(response?["Bucket"] ?? "")"        let FileName: String = "\(response?["FileName"] ?? "")"        let uploadID: String = "\(response?["Upload"] ?? "")"        var endpoints: [String] = Array<String>.init()        if response?.keys.contains("Hosts") ?? false {            endpoints = response?["Hosts"] as! [String]            endpoints.append(endpoint)        }        debugPrint("取我方服务器STS 返回数据 \(String(describing: response))")        PQAliOssUtil.shared.PQOSSMultipartUpload(accessKeyId: accessKeyId, secretKeyId: secretKeyId, securityToken: securityToken, bucketName: bucketName, endpoints: endpoints, FileName: FileName, fileURL: URL(fileURLWithPath: (localPath ?? "").replacingOccurrences(of: "file:///", with: "")), ossUploadID: uploadID,videoSource: videoSource)    }    /// 分片上传    /// - Parameters:    ///   - accessKeyId: <#accessKeyId description#>    ///   - secretKeyId: <#secretKeyId description#>    ///   - securityToken: <#securityToken description#>    ///   - bucketName: <#bucketName description#>    ///   - endpoints: oss host域名    ///   - FileName: oss 资源地址    ///   - fileURL: 文件本地地址    ///   - ossUploadID: oss 上传ID    ///   - contentType: 文件类型    ///   - materialType: 素材类型    ///   - isMatarialUpload: 是否是素材上传    ///   - contentMD5: 素材内容MD5值 isMatarialUpload = true时传    ///   - width: 视频宽 isMatarialUpload = true时传    ///   - height: 视频高 isMatarialUpload = true时传    /// - Returns: <#description#>    public  func PQOSSMultipartUpload(accessKeyId: String, secretKeyId: String, securityToken: String, bucketName: String, endpoints: [String], FileName: String, fileURL: URL, ossUploadID: String, materialType: StickerType = .VIDEO, isMatarialUpload: Bool = false, contentMD5: String = "", width: CGFloat = 0, height: CGFloat = 0,videoSource:String? = nil) -> PQAliOssUtil {        debugPrint("上传数据 参数\n accessKeyId:\(accessKeyId)\n secretKeyId:\(secretKeyId)\n  securityToken:\(securityToken) \n bucketName:\(bucketName)\n endpoint:\(endpoints)\n FileName:\(FileName)")        if endpoints.count <= 0 || (endpoints.first?.count ?? 0) <= 0 {            debugPrint("endpoints 为空")            return .shared        }        #if DEBUG            // 打开 oss 日志            OSSLog.enable()        #endif        // 1,设置鉴权        let url = PQENVUtil.shared.longvideoapi + (isMatarialUpload ? materialUploadStsTokenUrl : getStsTokenUrl)                let authServerUrl = url + "?appType=\(commonParams()["appType"] as? String ?? "")" + "&machineCode=" + getMachineCode()            + "&token=" + (BFConfig.shared.token ?? "") + "&loginUid" + (BFConfig.shared.uid ?? "") + "&fileType=2" + "&uploadId=\(ossUploadID)"        debugPrint("authServerUrl is: \(authServerUrl)")        debugPrint("当前上传authServerUrl线程:\(Thread.isMainThread) ")        let provider = OSSAuthCredentialProvider(authServerUrl: authServerUrl) { (data) -> Data? in            debugPrint("当前上传provider线程:\(Thread.isMainThread) ")            // 在 OSSModel 代码中解析有自己的格式 所以接收到我方服务器后的数据要进行二次处理 AccessKeyId 等信息            let str = String(data: data, encoding: .utf8)            let jsonDic = jsonStringToDictionary(str!)            var respDic: [String: Any] = [:]            if jsonDic?.keys.contains("code") ?? false, "\(jsonDic?["code"] ?? "")" == "0" {                respDic = [                    "StatusCode": "200",                    "AccessKeyId": (jsonDic?["data"] as? [String: Any])?["AccessKeyId"] ?? accessKeyId,                    "AccessKeySecret": (jsonDic?["data"] as? [String: Any])?["AccessKeySecret"] ?? secretKeyId,                    "SecurityToken": (jsonDic?["data"] as? [String: Any])?["SecurityToken"] ?? securityToken,                    "Expiration": (jsonDic?["data"] as? [String: Any])?["Expiration"] ?? securityToken,                ]            }            let proStr = dictionaryToJsonString(respDic as [String: Any])            debugPrint("处理后准备给 OSS SDK数据 \(String(describing: proStr))")            let decodedData = proStr?.data(using: .utf8)            if decodedData != nil {                return decodedData            }            return data        }        // 2 拼装 request        let request = OSSMultipartUploadRequest()        // add by ak  set  contect type . doc https://help.aliyun.com/knowledge_detail/39522.html        request.contentType = fileURL.absoluteString.mimeType() // 媒体类型        request.uploadingFileURL = fileURL        request.bucketName = bucketName        request.objectKey = FileName        // 根据文件大小 设置 part size        var fileSize: UInt64 = 0        do {            let attr = try FileManager.default.attributesOfItem(atPath: fileURL.relativePath)            fileSize = attr[FileAttributeKey.size] as! UInt64        } catch {            debugPrint("取文件大小 Error: \(error)")            if aliOssHander != nil {                DispatchQueue.main.async { [weak self] in                    self?.aliOssHander!(isMatarialUpload, materialType, fileURL.absoluteString.pathExtension, 260, FileName, contentMD5, width, height, 0, 0, nil, fileURL, nil, "文件已丢失")                }            }        }        debugPrint("文件已经存在  \(fileURL) 文件大小 \(fileSize)")        var partSize: Int = 0        // sdk中规定kClientMaximumOfChunks = 5000;        let defaultChunkSize: UInt64 = 1024 * 1024        if ceil(Float(fileSize / defaultChunkSize)) <= 1 {            partSize = 500 * 1024        } else if ceil(Float(fileSize / defaultChunkSize)) < 5000 {            partSize = Int(fileSize / UInt64(ceil(Float(fileSize / defaultChunkSize))))        } else {            partSize = Int(ceil(Float(fileSize / 5000)))        }        debugPrint("partSize \(partSize)")        // 除最后一片外 不能小于 102400(100kb)        request.partSize = UInt(partSize)        request.uploadId = ossUploadID        request.uploadProgress = { [weak self] (bytesSent: Int64, totalBytesSent: Int64, totalBytesExpectedToSend: Int64) -> Void in            debugPrint("当前上传uploadProgress线程:\(Thread.isMainThread) ")            self?.paraseSpeedAndRestTime(bytesSent: bytesSent, totalBytesSent: totalBytesSent, totalBytesExpectedToSend: totalBytesExpectedToSend) { uploadSpeed, resttime in                if self?.aliOssProgressHander != nil {                    DispatchQueue.main.async { [weak self] in                        self?.aliOssProgressHander!(bytesSent, totalBytesSent, totalBytesExpectedToSend, resttime, uploadSpeed)                    }                }            }            // 多个界面都要处理上传进度 所以使用通知 ? 通知有回到主线?            // postNotification(name: cOSSUploadFileProgress,userInfo: ["bytesSent":bytesSent,"totalBytesSent":totalBytesSent,"totalBytesExpectedToSend":totalBytesExpectedToSend,"resttime":resttime])        }        // 3,设置 config        let conf = OSSClientConfiguration()        conf.timeoutIntervalForRequest = 30 // 连接超时,默认15秒        conf.maxConcurrentRequestCount = 5 // 最大并发请求书,默认5个        conf.maxRetryCount = 3 // 失败后最大重试次数,默认2次        // 打开后台上传        conf.enableBackgroundTransmitService = true        conf.backgroundSesseionIdentifier = getUniqueId(desc: "\(ossUploadID)\(fileURL.absoluteString)\(FileName)")        // 4,设置 clinet        let client = OSSClient(endpoint: endpoints[0], credentialProvider: provider, clientConfiguration: conf)        if !isMatarialUpload {            allTasks[FileName] = request        }        // 5,发起任务        let task = client.multipartUpload(request)        task.continue ({ [weak self] (osstask) -> Any? in            debugPrint("当前上传线程:\(Thread.isMainThread) ")            if osstask.error == nil {                debugPrint("上传成功")                // 文件URL的格式为:BucketName.Endpoint/ObjectName。                debugPrint(" 上传成功注意使用时拼接域名 url == \(FileName)")                request.callbackParam = ["code": 1, "objectKey": FileName, "msg": "上传成功"]                DispatchQueue.main.async { [weak self] in                    if self?.aliOssHander != nil {                        self?.aliOssHander!(isMatarialUpload, materialType, fileURL.absoluteString.pathExtension, 1, FileName, contentMD5, width, height, 0, 0, nil, fileURL, nil, "上传成功")                    }                }                // 如果在后台发送上传成功的本地通知                if !isMatarialUpload {                    DispatchQueue.main.async {                        if UIApplication.shared.applicationState == .background {                            sendUploadNotification(isSuccess: true)                        }                    }                    postNotification(name: cUploadSuccessKey, userInfo: ["code": 1, "objectKey": FileName, "msg": "上传成功"])                }                // 上传完成                var extParams:Dictionary<String,Any>?                if videoSource != nil && (videoSource?.count ?? 0) > 0 {                    extParams = ["source":videoSource ?? ""]                }                BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_success, pageSource: nil,extParams: extParams, commonParams: commonParams())            } else {                debugPrint("上传失败 \(osstask.error!)")                request.callbackParam = ["code": (osstask.error! as NSError).code, "objectKey": FileName, "msg": osstask.error?.localizedDescription ?? ""]                if self?.aliOssHander != nil {                    DispatchQueue.main.async { [weak self] in                        self?.aliOssHander!(isMatarialUpload, materialType, fileURL.absoluteString.pathExtension, (osstask.error! as NSError).code, FileName, contentMD5, width, height, 0, 0, nil, fileURL, nil, osstask.error?.localizedDescription)                    }                }                // 如果在后台发送上传失败的本地通知                if !isMatarialUpload {                    DispatchQueue.main.async {                        if UIApplication.shared.applicationState == .background {                            sendUploadNotification(isSuccess: false)                        }                    }                }                // 上传失败                var extParams:Dictionary<String,Any>?                if videoSource != nil && (videoSource?.count ?? 0) > 0 {                    extParams = ["source":videoSource ?? ""]                }                BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_fail, pageSource: nil,extParams: extParams, commonParams: commonParams())            }            return nil        })        // 开始上传        var extParams:Dictionary<String,Any>?        if videoSource != nil && (videoSource?.count ?? 0) > 0 {            extParams = ["source":videoSource ?? ""]        }        BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_start, pageSource: nil,extParams: extParams, commonParams: commonParams())        return .shared    }    /// add by ak  简单上传方式 本方法支持后台运行 设置支持后台时会把 DATA    /// - Parameters:    ///   - accessKeyId: <#accessKeyId description#>    ///   - secretKeyId: <#secretKeyId description#>    ///   - securityToken: <#securityToken description#>    ///   - bucketName: <#bucketName description#>    ///   - endpoint: oss host域名    ///   - objectKey: oss 资源地址    ///   - fileURL: 文件本地地址    ///   - data: 文件数据    ///   - fileExtensions: 文件后缀名 mp4/mp3    ///   - enableBackground: 是否支持后台下载    ///   - materialType: 素材类型    ///   - isMatarialUpload: 是否是素材上传    ///   - contentMD5: 素材内容MD5值 isMatarialUpload = true时传    ///   - width: 视频宽 isMatarialUpload = true时传    ///   - height: 视频高 isMatarialUpload = true时传    ///   - duration: 素材时长 isMatarialUpload = true时传    ///   - frameNumber: gif素材帧数 isMatarialUpload = true时传    ///   - netResourceUrl: 网络素材地址 isMatarialUpload = true时传    /// - Returns: <#description#>    public  func putObjectAsync(accessKeyId: String, secretKeyId: String, securityToken: String, bucketName: String, endpoint: [String], objectKey: String, fileURL: URL?, data: Data?, fileExtensions: String, enableBackground: Bool, materialType: StickerType = .VIDEO, isMatarialUpload: Bool = false, contentMD5: String = "", width: CGFloat = 0, height: CGFloat = 0, duration: CGFloat, frameNumber: Int, netResourceUrl: String? = nil,videoSource:String? = nil) -> PQAliOssUtil {        #if DEBUG            OSSLog.enable()        #endif        debugPrint("普通上传数据 参数 accessKeyId:\(accessKeyId) secretKeyId:\(secretKeyId) securityToken:\(securityToken) bucketName:\(bucketName) endpoint:\(endpoint) objectKey:\(objectKey) enableBackground:\(enableBackground) fileURL : \(fileURL!)")        if endpoint.count <= 0 || (endpoint.first?.count ?? 0) <= 0 {            debugPrint("endpoints 为空")            return .shared        }        let provider = OSSStsTokenCredentialProvider(accessKeyId: accessKeyId, secretKeyId: secretKeyId, securityToken: securityToken)//            let configuration = OSSClientConfiguration()//            if enableBackground {//                configuration.enableBackgroundTransmitService = true//                configuration.backgroundSesseionIdentifier = objectKey//            }//            configuration.timeoutIntervalForRequest = 60 // 连接超时,默认15秒//            configuration.maxConcurrentRequestCount = 20 // 最大并发请求书,默认5个//            configuration.maxRetryCount = 3 // 失败后最大重试次数,默认2次        startClient(accessKeyId: accessKeyId, secretKeyId: secretKeyId, securityToken: securityToken, endpoint: endpoint.first ?? "")        let request = OSSPutObjectRequest()        if fileURL != nil {            request.uploadingFileURL = fileURL!            request.contentType = fileURL!.absoluteString.mimeType() // 媒体类型        } else if data != nil {            request.uploadingData = data!            request.contentType = materialType.mimeType() // 媒体类型        }        request.bucketName = bucketName        request.objectKey = objectKey        // request.contentMd5 = contentMD5        request.uploadProgress = { (bytesSent: Int64, totalBytesSent: Int64, totalBytesExpectedToSend: Int64) -> Void in            var resttime: Int64 = 0            if self.oldTime == 0 {                self.oldTime = Int(Date().timeIntervalSince1970)            }            let nowTime = Int(Date().timeIntervalSince1970)            let pertime = Int(nowTime - self.oldTime)            if pertime != 0 {                var speed = totalBytesSent / Int64(pertime) // 单位b/s                if speed != 0 {                    resttime = ((totalBytesExpectedToSend - totalBytesSent) / speed)                    var units = "b/s" // 单位名称                    if speed / 1024 > 1 {                        speed = speed / 1024                        units = "k/s"                    }                    if speed / 1024 > 1 {                        speed = speed / 1024                        units = "M/s"                    }                    debugPrint("上传速度: \(speed)\(units)   还剩时间 \(resttime)")                }            }            // 多个界面都要处理上传进度 所以使用通知 ? 通知有回到主线?            postNotification(name: cOSSUploadFileProgress, userInfo: ["bytesSent": bytesSent, "totalBytesSent": totalBytesSent, "totalBytesExpectedToSend": totalBytesExpectedToSend, "resttime": resttime])        }        let task = client?.putObject(request)        if fileURL?.absoluteString.contains("_noise_") ?? false {            task?.continue ({ [weak self] (osstask) -> Any? in                debugPrint("当前上传线程:\(Thread.isMainThread) ")                if osstask.error == nil {                    debugPrint("上传成功")                    // 文件URL的格式为:BucketName.Endpoint/ObjectName。                    debugPrint(" 上传成功注意使用时拼接域名 url == \(objectKey)")                    request.callbackParam = ["code": 1, "objectKey": objectKey, "msg": "上传成功"]                    DispatchQueue.main.async { [weak self] in                        if self?.aliOssHander != nil {                            self?.aliOssHander!(isMatarialUpload, materialType, fileExtensions, 1, objectKey, contentMD5, width, height, duration, frameNumber, netResourceUrl, fileURL, data, "上传成功")                        }                    }                    // 如果在后台发送上传成功的本地通知                    if !isMatarialUpload {                        DispatchQueue.main.async {                            if UIApplication.shared.applicationState == .background {                                sendUploadNotification(isSuccess: true)                            }                        }                        postNotification(name: cUploadSuccessKey, userInfo: ["code": 1, "objectKey": objectKey, "msg": "上传成功"])                    }                    // 上传完成                    var extParams:Dictionary<String,Any>?                    if videoSource != nil && (videoSource?.count ?? 0) > 0 {                        extParams = ["source":videoSource ?? ""]                    }                    // 上传完成                    BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_success, pageSource: nil,extParams: extParams, commonParams: commonParams())                } else {                    debugPrint("上传失败 \(osstask.error!)")                    request.callbackParam = ["code": (osstask.error! as NSError).code, "objectKey": objectKey, "msg": osstask.error?.localizedDescription ?? ""]                    if self?.aliOssHander != nil {                        DispatchQueue.main.async { [weak self] in                            self?.aliOssHander!(isMatarialUpload, materialType, fileExtensions, (osstask.error! as NSError).code, objectKey, contentMD5, width, height, duration, frameNumber, netResourceUrl, fileURL, data, osstask.error?.localizedDescription)                        }                    }                    // 如果在后台发送上传失败的本地通知                    if !isMatarialUpload {                        DispatchQueue.main.async {                            if UIApplication.shared.applicationState == .background {                                sendUploadNotification(isSuccess: false)                            }                        }                    }                    // 上传失败                    var extParams:Dictionary<String,Any>?                    if videoSource != nil && (videoSource?.count ?? 0) > 0 {                        extParams = ["source":videoSource ?? ""]                    }                    BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_fail, pageSource: nil,extParams: extParams, commonParams: commonParams())                }                return nil            }).waitUntilFinished()        } else {            task?.continue ({ [weak self] (osstask) -> Any? in                debugPrint("当前上传线程:\(Thread.isMainThread) ")                if osstask.error == nil {                    debugPrint("上传成功")                    // 文件URL的格式为:BucketName.Endpoint/ObjectName。                    debugPrint(" 上传成功注意使用时拼接域名 url == \(objectKey)")                    request.callbackParam = ["code": 1, "objectKey": objectKey, "msg": "上传成功"]                    DispatchQueue.main.async { [weak self] in                        if self?.aliOssHander != nil {                            self?.aliOssHander!(isMatarialUpload, materialType, fileExtensions, 1, objectKey, contentMD5, width, height, duration, frameNumber, netResourceUrl, fileURL, data, "上传成功")                        }                    }                    // 如果在后台发送上传成功的本地通知                    if !isMatarialUpload {                        DispatchQueue.main.async {                            if UIApplication.shared.applicationState == .background {                                sendUploadNotification(isSuccess: true)                            }                        }                        postNotification(name: cUploadSuccessKey, userInfo: ["code": 1, "objectKey": objectKey, "msg": "上传成功"])                    }                    // 上传完成                    var extParams:Dictionary<String,Any>?                    if videoSource != nil && (videoSource?.count ?? 0) > 0 {                        extParams = ["source":videoSource ?? ""]                    }                    BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_success, pageSource: nil,extParams: extParams, commonParams: commonParams())                } else {                    debugPrint("上传失败 \(osstask.error!)")                    request.callbackParam = ["code": (osstask.error! as NSError).code, "objectKey": objectKey, "msg": osstask.error?.localizedDescription ?? ""]                    if self?.aliOssHander != nil {                        DispatchQueue.main.async { [weak self] in                            self?.aliOssHander!(isMatarialUpload, materialType, fileExtensions, (osstask.error! as NSError).code, objectKey, contentMD5, width, height, duration, frameNumber, netResourceUrl, fileURL, data, osstask.error?.localizedDescription)                        }                    }                    // 如果在后台发送上传失败的本地通知                    if !isMatarialUpload {                        DispatchQueue.main.async {                            if UIApplication.shared.applicationState == .background {                                sendUploadNotification(isSuccess: false)                            }                        }                    }                    // 上传失败                    var extParams:Dictionary<String,Any>?                    if videoSource != nil && (videoSource?.count ?? 0) > 0 {                        extParams = ["source":videoSource ?? ""]                    }                    BFEventTrackAdaptor.baseReportUpload(businessType: .bt_up_process, objectType: .ot_up_fail, pageSource: nil,extParams: extParams, commonParams: commonParams())                }                return nil            })        }        return .shared    }    /// 取消某个任务    /// - Parameter objectKey: 任务唯一标识    public  func putObjectCancel(objectKey: String) {        for key in allTasks.keys {            if key == objectKey {                allTasks[key]!.cancel()                allTasks.removeValue(forKey: key)            }        }    }    /// 解析上传速度及剩余时间    /// - Parameters:    ///   - bytesSent: <#bytesSent description#>    ///   - totalBytesSent: <#totalBytesSent description#>    ///   - totalBytesExpectedToSend: <#totalBytesExpectedToSend description#>    ///   - complateHandle: <#complateHandle description#>    /// - Returns: <#description#>    public  func paraseSpeedAndRestTime(bytesSent _: Int64, totalBytesSent: Int64, totalBytesExpectedToSend: Int64, complateHandle: (_ uploadSpeed: String, _ resttime: Int64) -> Void) {        let newInterfaceBytes = PQBridgeObject.getInterfaceBytes()        var interfaceBytes = abs(newInterfaceBytes - lastInterfaceBytes)        if interfaceBytes <= 0 {            interfaceBytes = 1        }        lastInterfaceBytes = newInterfaceBytes        let resttime = (totalBytesExpectedToSend - totalBytesSent) / interfaceBytes        debugPrint("interfaceBytes = \(interfaceBytes),totalBytesExpectedToSend = \(totalBytesExpectedToSend),totalBytesSent = \(totalBytesSent),")        complateHandle("\(PQBridgeObject.formatNetWork(interfaceBytes))", resttime)    }    override private init() {        super.init()    }    public override func copy() -> Any {        return self    }    public override func mutableCopy() -> Any {        return self    }}
 |