如何覆盖 Alamofire SessionManager 以修改响应以添加标头?

Posted

技术标签:

【中文标题】如何覆盖 Alamofire SessionManager 以修改响应以添加标头?【英文标题】:How to override Alamofire SessionManager to modify response in order to add headers? 【发布时间】:2018-07-09 19:43:53 【问题描述】:

所以,我正在使用 Alamofire 发出一些 HTTP 请求,并且我正在尝试使用缓存策略,但服务器的标头中没有“Cache-Control”标志。所以我想基于这个answer here.使用会话管理器delegate methoddataTaskWillCacheResponse添加这个标志

为了做到这一点,我想创建一个具有 SessionManager 的静态共享实例的类(我不想在 AppDelegate 上创建全局变量。我认为这不是一个好习惯)并在这个类中覆盖委托方法,但我不能这样做。

这是我的代码:

public class Manager: SessionDelegate 
    var manager: SessionManager?
    weak var sessionDelegate: SessionDelegate?
    override init() 
        super.init()
        let configuration = URLSessionConfiguration.default
        configuration.urlCache = URLCache(memoryCapacity: 20 * 1024 * 1024, diskCapacity: 100 * 1024 * 1024, diskPath: nil)
        configuration.requestCachePolicy = .returnCacheDataElseLoad
        configuration.timeoutIntervalForRequest = 35
        self.manager = SessionManager(configuration: configuration)
        self.manager = Alamofire.SessionManager(configuration: configuration)
        self.manager?.delegate = self
    

self.manager.delegate = self 我得到错误:无法分配给属性:'delegate' is a 'let' constant

我在 Swift 和 Alamofire 方面相对较新。我究竟做错了什么?我怎样才能实现我想要做的事情?

编辑: 我看到我无法创建单例,但我能够使用共享实例并覆盖这些方法。看看下面的答案

【问题讨论】:

单身有什么特别的原因吗?你不需要一个。 SessionManager 在自己的 init() 期间创建并分配其委托 没有人。我认为这将是一个很好的解决方案,我将能够覆盖单例类中的委托方法。 您已经可以覆盖 dataTaskWillCacheResponse 它是一个开放的变量。请参阅 Alamofire 文档中的 Overriding Closures 是的,这就是我正在玩的。我想我可以做到,但我认为我无法编辑 cachedResponse。 【参考方案1】:

经过更多研究,我发现了一些东西: 1 - Advanced usage override closures 2 - Found this answer that creates a Session Manager 3 - Found this solution 在dataTaskWillCacheResponse 委托方法中为我的响应添加标题。

这是我的解决方案:

class MyService 
    static let shared = MyService()

    //The manager with the cache policy
    private let manager: Alamofire.SessionManager = 
        let configuration = URLSessionConfiguration.default
        configuration.requestCachePolicy = .useProtocolCachePolicy
        configuration.timeoutIntervalForRequest = 35
        return Alamofire.SessionManager(configuration: configuration)
    ()

    init() 
        let delegate: Alamofire.SessionDelegate = manager.delegate
        //Overriding delegate to add headers
        delegate.dataTaskWillCacheResponseWithCompletion =  session, datatask, cachedResponse, completion in
            let response = cachedResponse.response  as! HTTPURLResponse
            var headers = response.allHeaderFields as! [String: String]
            print(headers.keys.contains("Cache-Control"))
            headers["Cache-Control"] = "max-age=30"
            let modifiedResponse = HTTPURLResponse(
                url: response.url!,
                statusCode: response.statusCode,
                httpVersion: "HTTP/1.1",
                headerFields: headers)

            let modifiedCachedResponse = CachedURLResponse(
                response: modifiedResponse!,
                data: cachedResponse.data,
                userInfo: cachedResponse.userInfo,
                storagePolicy: cachedResponse.storagePolicy)
            completion(modifiedCachedResponse)
        
    

【讨论】:

是的,确实如此。只需调用相同的 URL,如果响应被缓存,它将返回它。 如果 max-age 没有过期,会返回缓存的响应

以上是关于如何覆盖 Alamofire SessionManager 以修改响应以添加标头?的主要内容,如果未能解决你的问题,请参考以下文章

Alamofire 多参数字典

Alamofire 是不是支持 HTTP 流水线?

Alamofire 服务器信任策略对我不起作用

Alamofire - 有没有办法设置自定义默认队列来处理所有请求回调

如何在 alamofire 中使用自定义动词?

Alamofire 完成工作后如何加载视图?