main.swift 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. import JavaScriptCore
  2. import Alamofire
  3. import Alamofire
  4. import Foundation
  5. //// 扩展SessionManager以支持多个Cookies
  6. //extension SessionManager {
  7. // static let multipleCookiesSessionManager: SessionManager = {
  8. // let configuration = URLSessionConfiguration.default
  9. // configuration.httpAdditionalHeaders = SessionManager.defaultHTTPHeaders
  10. //
  11. // let manager = SessionManager(configuration: configuration)
  12. //
  13. // // 自定义处理Cookies的方式
  14. // manager.delegate.taskWillPerformHTTPRedirection = { session, task, response, request in
  15. // if let headers = response.allHeaderFields as? [String: String],
  16. // let setCookies = HTTPCookie.cookies(withResponseHeaderFields: headers, for: request.url!) {
  17. // for cookie in setCookies {
  18. // HTTPCookieStorage.shared.setCookie(cookie)
  19. // }
  20. // }
  21. // return request
  22. // }
  23. //
  24. // return manager
  25. // }()
  26. //}
  27. //
  28. //// 使用扩展的SessionManager发送请求
  29. //SessionManager.multipleCookiesSessionManager.request("https://www.youtube.com/watch?v=5GJWxDKyk3A").response { response in
  30. // debugPrint(response)
  31. //}
  32. func downloadJSFile(url: URL, completion: @escaping (Result<String, Error>) -> Void) {
  33. AF.download(url).responseData { (response) in
  34. switch response.result {
  35. case .success(let data):
  36. if let jsString = String(data: data, encoding: .utf8) {
  37. completion(.success(jsString))
  38. } else {
  39. completion(.failure(NSError(domain: "Download Error", code: -1, userInfo: nil)))
  40. }
  41. case .failure(let error):
  42. completion(.failure(error))
  43. }
  44. }
  45. }
  46. func createJSContext() -> JSContext {
  47. let ctx = JSContext()
  48. // 注入 console.log
  49. ctx?.evaluateScript("var console = { log: function(message) { _consoleLog(message) } }")
  50. let consoleLog: @convention(block) (String) -> Void = { message in
  51. print("JS 打印: \(message)")
  52. }
  53. ctx?.setObject(unsafeBitCast(consoleLog, to: AnyObject.self), forKeyedSubscript: "_consoleLog" as (NSCopying & NSObjectProtocol)?)
  54. // 注入AF
  55. ctx?.evaluateScript("var AF = { request: function(url, method, data, headers, callback) { _request(url, method, data, headers, callback) } }")
  56. let af: @convention(block) (String, String, String?, [String: String]?, JSValue?) -> Void = { url, method, data, headers, callback in
  57. var request = URLRequest(url: URL(string: url)!)
  58. request.httpMethod = method
  59. if method == "POST" {
  60. if let data = data {
  61. request.setValue("application/json", forHTTPHeaderField: "Content-Type")
  62. request.httpBody = data.data(using: .utf8)
  63. }
  64. }
  65. if let headers = headers {
  66. request.headers = HTTPHeaders(headers)
  67. }
  68. AF.request(request).response { response in
  69. if let data = response.data {
  70. var headerJsonString = ""
  71. if let headers = response.response?.allHeaderFields as? [String: String] {
  72. do {
  73. let jsonData = try JSONSerialization.data(withJSONObject: headers, options: [])
  74. if let jsonString = String(data: jsonData, encoding: .utf8) {
  75. headerJsonString = jsonString
  76. }
  77. } catch {
  78. }
  79. }
  80. callback?.call(withArguments: [String(data: data, encoding: .utf8), headerJsonString, nil])
  81. }
  82. if let error = response.error {
  83. debugPrint(response)
  84. callback?.call(withArguments: [nil, nil, error.localizedDescription])
  85. }
  86. }
  87. }
  88. ctx?.setObject(unsafeBitCast(af, to: AnyObject.self), forKeyedSubscript: "_request" as (NSCopying & NSObjectProtocol)?)
  89. // 捕捉JS运行异常
  90. ctx?.exceptionHandler = { context, exception in
  91. if let exception = exception {
  92. debugPrint(exception)
  93. print("JS 执行异常: \(exception)")
  94. }
  95. }
  96. return ctx!
  97. }
  98. func testDetail(url: String, ctx: JSContext) -> Void {
  99. let startTime = DispatchTime.now()
  100. if let detailFunction = ctx.objectForKeyedSubscript("detail") {
  101. let result = detailFunction.call(withArguments: [url])
  102. let completionHandler: @convention(block) (JSValue) -> Void = { result in
  103. print("详情结果!!: \(result.toDictionary())")
  104. let endTime = DispatchTime.now()
  105. let nanoTime = endTime.uptimeNanoseconds - startTime.uptimeNanoseconds
  106. let timeInterval = Double(nanoTime) / 1_000_000_000
  107. print("耗时: \(timeInterval) seconds")
  108. }
  109. let completionFunction = unsafeBitCast(completionHandler, to: AnyObject.self)
  110. result?.invokeMethod("then", withArguments: [completionFunction])
  111. }
  112. }
  113. func testSearch(keyword: String, ctx: JSContext) -> Void {
  114. if let search = ctx.objectForKeyedSubscript("search") {
  115. let result = search.call(withArguments: [keyword])
  116. let completionHandler: @convention(block) (JSValue) -> Void = { result in
  117. print("搜索结果!!: \(result.toDictionary())")
  118. }
  119. let completionFunction = unsafeBitCast(completionHandler, to: AnyObject.self)
  120. result?.invokeMethod("then", withArguments: [completionFunction])
  121. }
  122. }
  123. let ctx = createJSContext()
  124. let remote = "https://d3crpuooyqht8f.cloudfront.net/e76b32a6-42a0-4682-a30c-0b5c2d75e5b9"
  125. let local = "file:///Users/ben/Desktop/app/be/be-ytb/js/bundle.js"
  126. if let url = URL(string: remote) {
  127. downloadJSFile(url: url) { result in
  128. switch result {
  129. case .success(let jsString):
  130. print("下载远程JS成功")
  131. ctx.evaluateScript(jsString)
  132. testDetail(url: "https://www.youtube.com/watch?v=JByDbPn6A1o", ctx: ctx)
  133. // testSearch(keyword: "周杰伦", ctx: ctx)
  134. case .failure(let error):
  135. print("Download Error: \(error)")
  136. }
  137. }
  138. }
  139. RunLoop.main.run()