ApiService.swift 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580
  1. // Copyright 2023 Google LLC
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. import Foundation
  15. import FirebaseInstallations
  16. import FirebaseCore
  17. import UIKit
  18. // Avoids exposing internal APIs to Swift users
  19. @_implementationOnly import FirebaseCoreInternal
  20. enum Strings {
  21. static let errorDomain = "com.firebase.appdistribution.api"
  22. static let errorDetailsKey = "details"
  23. static let httpGet = "GET"
  24. static let httpPost = "POST"
  25. static let releaseEndpointUrlTemplate =
  26. "https://firebaseapptesters.googleapis.com/v1alpha/devices/-/testerApps/%@/installations/%@/releases"
  27. static let findReleaseEndpointUrlTemplate =
  28. "https://firebaseapptesters.googleapis.com/v1alpha/projects/%@/installations/%@/releases:find"
  29. static let createFeedbackEndpointUrlTemplate =
  30. "https://firebaseapptesters.googleapis.com/v1alpha/%@/feedbackReports"
  31. static let uploadImageEndpointUrlTemplate =
  32. "https://firebaseapptesters.googleapis.com/v1alpha/%@/feedbackReports"
  33. static let commitFeedbackEndpointUrlTemplate =
  34. "https://firebaseapptesters.googleapis.com/v1alpha/%@:commit"
  35. static let installationsAuthHeader = "X-Goog-Firebase-Installations-Auth"
  36. static let apiHeaderKey = "X-Goog-Api-Key"
  37. static let apiBundleKey = "X-Ios-Bundle-Identifier"
  38. static let responseReleaseKey = "releases"
  39. static let compositeBinaryIdQueryParamName = "compositeBinaryId"
  40. static let uploadArtifactTypeQueryParamName = "type"
  41. static let uploadArtifactScreenshotType = "SCREENSHOT"
  42. static let GoogleUploadProtocolHeader = "X-Goog-Upload-Protocol"
  43. static let GoogleUploadProtocolRaw = "raw"
  44. static let GoogleUploadFileNameHeader = "X-Goog-Upload-File-Name"
  45. static let GoogleUploadFileName = "screenshot.png"
  46. }
  47. enum AppDistributionApiError: NSInteger {
  48. case ApiErrorTimeout = 0
  49. case ApiTokenGenerationFailure = 1
  50. case ApiInstallationIdentifierError = 2
  51. case ApiErrorUnauthenticated = 3
  52. case ApiErrorUnauthorized = 4
  53. case ApiErrorNotFound = 5
  54. case ApiErrorUnknownFailure = 6
  55. case ApiErrorParseFailure = 7
  56. }
  57. struct CompositeBinaryId: Codable {
  58. var displayVersion: String
  59. var buildVersion: String
  60. var codeHash: String
  61. }
  62. struct FindReleaseResponse: Codable {
  63. var release: String
  64. }
  65. struct FeedbackReport: Codable {
  66. var name: String?
  67. var text: String?
  68. }
  69. struct CreateFeedbackReportRequest: Codable {
  70. var feedbackReport: FeedbackReport
  71. }
  72. @objc(FIRFADApiServiceSwift) open class ApiService: NSObject {
  73. @objc(generateAuthTokenWithCompletion:) public static func generateAuthToken(completion: @escaping (_ identifier: String?,
  74. _ authTokenResult: InstallationsAuthTokenResult?,
  75. _ error: Error?)
  76. -> Void) {
  77. generateAuthToken(installations: Installations.installations(), completion: completion)
  78. }
  79. static func generateAuthToken(installations: InstallationsProtocol,
  80. completion: @escaping (_ identifier: String?,
  81. _ authTokenResult: InstallationsAuthTokenResult?,
  82. _ error: Error?)
  83. -> Void) {
  84. installations.authToken(completion: { authTokenResult, error in
  85. var fadError: Error? = error
  86. if self.handleError(
  87. error: &fadError,
  88. description: "Failed to generate Firebase installation auth token",
  89. code: .ApiTokenGenerationFailure
  90. ) {
  91. Logger
  92. .logError(String(format: "Error getting auth token. Error: %@",
  93. error?.localizedDescription ?? ""))
  94. completion(nil, nil, fadError)
  95. return
  96. }
  97. installations.installationID(completion: { identifier, error in
  98. var fadError: Error? = error
  99. if self.handleError(
  100. error: &fadError,
  101. description: "Failed to generate Firebase installation id",
  102. code: .ApiInstallationIdentifierError
  103. ) {
  104. Logger
  105. .logError(String(format: "Error getting installation ID. Error: %@",
  106. error?.localizedDescription ?? ""))
  107. completion(nil, nil, fadError)
  108. return
  109. }
  110. completion(identifier, authTokenResult, nil)
  111. })
  112. })
  113. }
  114. @objc(fetchReleasesWithCompletion:) public static func fetchReleases(completion: @escaping (_ releases: [Any]?,
  115. _ error: Error?)
  116. -> Void) {
  117. guard let app = FirebaseApp.app() else {
  118. return
  119. }
  120. fetchReleases(
  121. app: app,
  122. installations: Installations.installations(),
  123. urlSession: URLSession.shared,
  124. completion: completion
  125. )
  126. }
  127. static func fetchReleases(app: FirebaseApp, installations: InstallationsProtocol,
  128. urlSession: URLSession, completion: @escaping (_ releases: [Any]?,
  129. _ error: Error?)
  130. -> Void) {
  131. Logger.logInfo(String(
  132. format: "Requesting release for app id - %@",
  133. app.options.googleAppID
  134. ))
  135. generateAuthToken(installations: installations) { identifier, authTokenResult, error in
  136. let urlString = String(
  137. format: Strings.releaseEndpointUrlTemplate,
  138. app.options.googleAppID,
  139. identifier!
  140. )
  141. let request = self.createHttpRequest(
  142. app: app,
  143. method: Strings.httpGet,
  144. url: urlString,
  145. authTokenResult: authTokenResult!
  146. )
  147. Logger.logInfo(String(
  148. format: "Url: %@ Auth token: %@ Api Key: %@",
  149. urlString,
  150. authTokenResult?.authToken ?? "",
  151. app.options.apiKey ?? ""
  152. ))
  153. let listReleaseDataTask = urlSession
  154. .dataTask(with: request as URLRequest) { data, response, error in
  155. var fadError = error
  156. let releasesResponse = self.handleResponse(data: data,
  157. response: response,
  158. error: &fadError,
  159. returnType: [String: [Any]].self)
  160. DispatchQueue.main.async {
  161. completion(releasesResponse?[Strings.responseReleaseKey], fadError)
  162. }
  163. }
  164. listReleaseDataTask.resume()
  165. }
  166. }
  167. @objc(findReleaseWithDisplayVersion:buildVersion:codeHash:completion:)
  168. public static func findRelease(displayVersion: String, buildVersion: String, codeHash: String,
  169. completion: @escaping (_ releaseName: String?, _ error: Error?)
  170. -> Void) {
  171. findRelease(
  172. app: FirebaseApp.app()!,
  173. installations: Installations.installations(),
  174. urlSession: URLSession.shared,
  175. displayVersion: displayVersion,
  176. buildVersion: buildVersion,
  177. codeHash: codeHash,
  178. completion: completion
  179. )
  180. }
  181. static func findRelease(app: FirebaseApp, installations: InstallationsProtocol,
  182. urlSession: URLSession, displayVersion: String,
  183. buildVersion: String, codeHash: String,
  184. completion: @escaping (_ releaseName: String?, _ error: Error?)
  185. -> Void) {
  186. generateAuthToken(installations: installations) { identifier, authTokenResult, error in
  187. // TODO(tundeagboola) The backend may not accept project ID here in which case
  188. // we'll have to figure out a way to get the project number
  189. let urlString = String(
  190. format: Strings.findReleaseEndpointUrlTemplate,
  191. app,
  192. identifier!
  193. )
  194. guard var urlComponents = URLComponents(string: urlString) else {
  195. // TODO(tundeagboola) We should throw exceptions here insead of piping errors
  196. Logger.logError("Unable to build URL for findRelease request")
  197. return
  198. }
  199. let compositeBinaryId = CompositeBinaryId(
  200. displayVersion: displayVersion,
  201. buildVersion: buildVersion,
  202. codeHash: codeHash
  203. )
  204. guard let compositeBinaryIdData = try? JSONEncoder().encode(compositeBinaryId) else {
  205. // TODO(tundeagboola) We should throw exceptions here insead of piping errors
  206. Logger.logError("Unable to build URL for findRelease request")
  207. return
  208. }
  209. urlComponents.queryItems = [URLQueryItem(
  210. name: Strings.compositeBinaryIdQueryParamName,
  211. value: String(data: compositeBinaryIdData, encoding: .utf8)
  212. )]
  213. guard let url = urlComponents.url else {
  214. // TODO(tundeagboola) We should throw exceptions here insead of piping errors
  215. Logger.logError("Unable to build URL for findRelease request")
  216. return
  217. }
  218. let request = self.createHttpRequest(
  219. app: app,
  220. method: Strings.httpGet,
  221. url: url,
  222. authTokenResult: authTokenResult!
  223. )
  224. let findReleaseDataTask = urlSession
  225. .dataTask(with: request as URLRequest) { data, response, error in
  226. var fadError = error
  227. let findReleaseResponse = self.handleCodableResponse(
  228. data: data,
  229. response: response,
  230. error: &fadError,
  231. returnType: FindReleaseResponse.self
  232. )
  233. DispatchQueue.main.async {
  234. completion(findReleaseResponse?.release, fadError)
  235. }
  236. }
  237. findReleaseDataTask.resume()
  238. }
  239. }
  240. @objc(createFeedbackWithReleaseName:feedbackText:completion:)
  241. public static func createFeedback(releaseName: String,
  242. feedbackText: String,
  243. completion: @escaping (_ feedbackName: String?, _ error: Error?)
  244. -> Void) {
  245. createFeedback(
  246. app: FirebaseApp.app()!,
  247. installations: Installations.installations(),
  248. urlSession: URLSession.shared,
  249. releaseName: releaseName,
  250. feedbackText: feedbackText,
  251. completion: completion
  252. )
  253. }
  254. static func createFeedback(app: FirebaseApp, installations: InstallationsProtocol,
  255. urlSession: URLSession, releaseName: String,
  256. feedbackText: String,
  257. completion: @escaping (_ feedbackName: String?, _ error: Error?)
  258. -> Void) {
  259. generateAuthToken(installations: installations) { identifier, authTokenResult, error in
  260. guard let authTokenResult = authTokenResult else {
  261. completion(nil, error)
  262. return
  263. }
  264. let urlString = String(
  265. format: Strings.createFeedbackEndpointUrlTemplate,
  266. releaseName
  267. )
  268. let request = createHttpRequest(
  269. app: app,
  270. method: Strings.httpPost,
  271. url: urlString,
  272. authTokenResult: authTokenResult
  273. )
  274. let createFeedbackRequest =
  275. CreateFeedbackReportRequest(feedbackReport: FeedbackReport(text: feedbackText))
  276. request.httpBody = try? JSONEncoder().encode(createFeedbackRequest)
  277. let createFeedbackTask = urlSession
  278. .dataTask(with: request as URLRequest) { data, response, error in
  279. var fadError = error
  280. let feedback = self.handleCodableResponse(
  281. data: data,
  282. response: response,
  283. error: &fadError,
  284. returnType: FeedbackReport.self
  285. )
  286. DispatchQueue.main.async {
  287. completion(feedback?.name, fadError)
  288. }
  289. }
  290. createFeedbackTask.resume()
  291. }
  292. }
  293. @objc(uploadImageWithFeedbackName:image:completion:)
  294. public static func uploadImage(feedbackName: String,
  295. image: UIImage,
  296. completion: @escaping (_ error: Error?)
  297. -> Void) {
  298. uploadImage(
  299. app: FirebaseApp.app()!,
  300. installations: Installations.installations(),
  301. urlSession: URLSession.shared,
  302. feedbackName: feedbackName,
  303. image: image,
  304. completion: completion
  305. )
  306. }
  307. static func uploadImage(app: FirebaseApp, installations: InstallationsProtocol,
  308. urlSession: URLSession, feedbackName: String,
  309. image: UIImage,
  310. completion: @escaping (_ error: Error?)
  311. -> Void) {
  312. generateAuthToken(installations: installations) { identifier, authTokenResult, error in
  313. guard let authTokenResult = authTokenResult else {
  314. completion(error)
  315. return
  316. }
  317. let urlString = String(
  318. format: Strings.uploadImageEndpointUrlTemplate,
  319. feedbackName
  320. )
  321. guard var urlComponents = URLComponents(string: urlString) else {
  322. // TODO(tundeagboola) We should throw exceptions here insead of piping errors
  323. Logger.logError("Unable to build URL for uploadArtifact request")
  324. return
  325. }
  326. urlComponents.queryItems = [URLQueryItem(
  327. name: Strings.uploadArtifactTypeQueryParamName,
  328. value: Strings.uploadArtifactScreenshotType
  329. )]
  330. guard let url = urlComponents.url else {
  331. // TODO(tundeagboola) We should throw exceptions here insead of piping errors
  332. Logger.logError("Unable to build URL for uploadArtifact request")
  333. return
  334. }
  335. let request = createHttpRequest(
  336. app: app,
  337. method: Strings.httpPost,
  338. url: url,
  339. authTokenResult: authTokenResult
  340. )
  341. request.setValue(
  342. Strings.GoogleUploadProtocolHeader,
  343. forHTTPHeaderField: Strings.GoogleUploadProtocolRaw
  344. )
  345. request.setValue(
  346. Strings.GoogleUploadFileNameHeader,
  347. forHTTPHeaderField: Strings.GoogleUploadFileName
  348. )
  349. let uploadImageTask = urlSession
  350. .dataTask(with: request as URLRequest) { data, response, error in
  351. var fadError = error
  352. self.handleError(httpResponse: response as? HTTPURLResponse, error: &fadError)
  353. DispatchQueue.main.async {
  354. completion(fadError)
  355. }
  356. }
  357. uploadImageTask.resume()
  358. }
  359. }
  360. @objc(commitFeedbackWithFeedbackName:completion:)
  361. public static func commitFeedback(feedbackName: String,
  362. completion: @escaping (_ error: Error?)
  363. -> Void) {
  364. commitFeedback(
  365. app: FirebaseApp.app()!,
  366. installations: Installations.installations(),
  367. urlSession: URLSession.shared,
  368. feedbackName: feedbackName,
  369. completion: completion
  370. )
  371. }
  372. static func commitFeedback(app: FirebaseApp, installations: InstallationsProtocol,
  373. urlSession: URLSession,
  374. feedbackName: String,
  375. completion: @escaping (_ error: Error?)
  376. -> Void) {
  377. generateAuthToken(installations: installations) { identifier, authTokenResult, error in
  378. guard let authTokenResult = authTokenResult else {
  379. completion(error)
  380. return
  381. }
  382. let urlString = String(
  383. format: Strings.commitFeedbackEndpointUrlTemplate,
  384. feedbackName
  385. )
  386. let request = createHttpRequest(
  387. app: app,
  388. method: Strings.httpPost,
  389. url: urlString,
  390. authTokenResult: authTokenResult
  391. )
  392. let commitFeedbackTask = urlSession
  393. .dataTask(with: request as URLRequest) { data, response, error in
  394. var fadError = error
  395. self.handleError(httpResponse: response as? HTTPURLResponse, error: &fadError)
  396. DispatchQueue.main.async {
  397. completion(fadError)
  398. }
  399. }
  400. commitFeedbackTask.resume()
  401. }
  402. }
  403. @discardableResult
  404. static func handleError(httpResponse: HTTPURLResponse?, error: inout Error?) -> Bool {
  405. // TODO(tundeagboola) We should be throwing errors instead of piping the error object through
  406. if error != nil || httpResponse == nil {
  407. return handleError(
  408. error: &error,
  409. description: "Unknown http error occurred",
  410. code: .ApiErrorUnknownFailure
  411. )
  412. } else if httpResponse?.statusCode != 200 {
  413. error = createError(statusCode: httpResponse!.statusCode)
  414. return true
  415. }
  416. return false
  417. }
  418. @discardableResult
  419. static func handleError(error: inout Error?, description: String?,
  420. code: AppDistributionApiError) -> Bool {
  421. // TODO(tundeagboola) We should be throwing errors instead of piping the error object through
  422. if error != nil {
  423. error = createError(description: description!, code: code)
  424. return true
  425. }
  426. return false
  427. }
  428. static func createError(description: String, code: AppDistributionApiError) -> Error {
  429. let userInfo = [NSLocalizedDescriptionKey: description]
  430. return NSError(domain: Strings.errorDomain, code: code.rawValue, userInfo: userInfo)
  431. }
  432. static func createError(statusCode: NSInteger) -> Error {
  433. switch statusCode {
  434. case 401:
  435. return createError(description: "Tester not authenticated", code: .ApiErrorUnauthenticated)
  436. case 403, 400:
  437. return createError(description: "Tester not authorized", code: .ApiErrorUnauthorized)
  438. case 404:
  439. return createError(description: "Tester or releases not found", code: .ApiErrorUnauthorized)
  440. case 408, 504:
  441. return createError(description: "Request timeout", code: .ApiErrorTimeout)
  442. default:
  443. print("Unknown status code")
  444. }
  445. let description = String(format: "Unknown status code: %ld", statusCode)
  446. return createError(description: description, code: .ApiErrorUnknownFailure)
  447. }
  448. static func createHttpRequest(app: FirebaseApp, method: String, url: String,
  449. authTokenResult: InstallationsAuthTokenResult)
  450. -> NSMutableURLRequest {
  451. return createHttpRequest(
  452. app: app,
  453. method: method,
  454. url: URL(string: url),
  455. authTokenResult: authTokenResult
  456. )
  457. }
  458. static func createHttpRequest(app: FirebaseApp, method: String, url: URL?,
  459. authTokenResult: InstallationsAuthTokenResult)
  460. -> NSMutableURLRequest {
  461. let request = NSMutableURLRequest()
  462. request.url = url
  463. request.httpMethod = method
  464. request.setValue(authTokenResult.authToken, forHTTPHeaderField: Strings.installationsAuthHeader)
  465. request.setValue(
  466. app.options.apiKey,
  467. forHTTPHeaderField: Strings.apiHeaderKey
  468. )
  469. request.setValue(Bundle.main.bundleIdentifier, forHTTPHeaderField: Strings.apiBundleKey)
  470. return request
  471. }
  472. static func validateResponse(response: URLResponse?, error: inout Error?) -> Bool {
  473. guard let response = response else {
  474. handleError(
  475. error: &error,
  476. description: "URLResponse is nil",
  477. code: .ApiErrorUnknownFailure
  478. )
  479. return false
  480. }
  481. let httpResponse = response as! HTTPURLResponse
  482. Logger
  483. .logInfo(String(format: "HTTPResponse status code %ld, response %@", httpResponse.statusCode,
  484. httpResponse))
  485. if handleError(httpResponse: httpResponse, error: &error) {
  486. Logger
  487. .logError(String(format: "App tester API service error: %@",
  488. error?.localizedDescription ?? ""))
  489. return false
  490. }
  491. return true
  492. }
  493. static func handleCodableResponse<T: Codable>(data: Data?, response: URLResponse?,
  494. error: inout Error?,
  495. returnType: T.Type) -> T? {
  496. if !validateResponse(response: response, error: &error) {
  497. return nil
  498. }
  499. guard let data = data else {
  500. return nil
  501. }
  502. do {
  503. return try JSONDecoder().decode(T.self, from: data)
  504. } catch let thrownError {
  505. handleApiParserErorr(thrownError, &error)
  506. return nil
  507. }
  508. }
  509. static func handleResponse<T>(data: Data?, response: URLResponse?,
  510. error: inout Error?, returnType: T.Type) -> T? {
  511. if !validateResponse(response: response, error: &error) {
  512. return nil
  513. }
  514. guard let data = data else {
  515. return nil
  516. }
  517. do {
  518. return try JSONSerialization.jsonObject(
  519. with: data,
  520. options: JSONSerialization.ReadingOptions(rawValue: 0)
  521. ) as? T
  522. } catch let thrownError {
  523. handleApiParserErorr(thrownError, &error)
  524. return nil
  525. }
  526. }
  527. static func handleApiParserErorr(_ thrownError: Error, _ error: inout Error?) {
  528. let description: String = (thrownError as NSError)
  529. .userInfo[NSLocalizedDescriptionKey] as? String ?? "Failed to parse response"
  530. error = thrownError
  531. handleError(error: &error, description: description, code: .ApiErrorParseFailure)
  532. Logger.logError("Tester API - Error deserializing json response")
  533. }
  534. }