mirror of
https://github.com/xtool-org/xtool.git
synced 2026-02-04 11:53:30 +01:00
109 lines
3.7 KiB
Swift
109 lines
3.7 KiB
Swift
//
|
|
// DeveloperServicesFetchProfileOperation.swift
|
|
// XKit
|
|
//
|
|
// Created by Kabir Oberai on 13/10/19.
|
|
// Copyright © 2019 Kabir Oberai. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
import Superutils
|
|
import DeveloperAPI
|
|
|
|
public struct DeveloperServicesFetchProfileOperation: DeveloperServicesOperation {
|
|
|
|
public enum Errors: Error {
|
|
case bundleIDNotFound
|
|
case tooManyMatchingBundleIDs
|
|
case invalidProfileData
|
|
}
|
|
|
|
public let context: SigningContext
|
|
public let bundleID: String
|
|
public let signingInfo: SigningInfo
|
|
public init(context: SigningContext, bundleID: String, signingInfo: SigningInfo) {
|
|
self.context = context
|
|
self.bundleID = bundleID
|
|
self.signingInfo = signingInfo
|
|
}
|
|
|
|
public func perform() async throws -> Mobileprovision {
|
|
let bundleIDs = try await context.developerAPIClient
|
|
.bundleIdsGetCollection(query: .init(
|
|
filter_lbrack_identifier_rbrack_: [bundleID],
|
|
fields_lbrack_profiles_rbrack_: [.bundleId],
|
|
include: [.profiles]
|
|
))
|
|
.ok.body.json
|
|
|
|
// filter[identifier] is a prefix filter so we need to manually upgrade to equality
|
|
let filtered = bundleIDs.data.filter { $0.attributes?.identifier == self.bundleID }
|
|
|
|
let bundleID: Components.Schemas.BundleId
|
|
switch filtered.count {
|
|
case 0:
|
|
throw Errors.bundleIDNotFound
|
|
case 1:
|
|
bundleID = filtered[0]
|
|
default:
|
|
throw Errors.tooManyMatchingBundleIDs
|
|
}
|
|
|
|
let profiles = bundleID.relationships?.profiles?.data ?? []
|
|
switch profiles.count {
|
|
case 0:
|
|
// we're good
|
|
break
|
|
case 1:
|
|
_ = try await context.developerAPIClient.profilesDeleteInstance(path: .init(id: profiles[0].id)).noContent
|
|
default:
|
|
// if the user has >1 profile, it's probably okay to add another one (acceptable for non-free accounts?)
|
|
break
|
|
}
|
|
|
|
let serialNumber = signingInfo.certificate.serialNumber()
|
|
let certs = try await context.developerAPIClient.certificatesGetCollection(
|
|
query: .init(
|
|
filter_lbrack_serialNumber_rbrack_: [serialNumber]
|
|
)
|
|
)
|
|
.ok.body.json.data
|
|
|
|
let allDevices = try await context.developerAPIClient.devicesGetCollection()
|
|
.ok.body.json.data
|
|
|
|
let response = try await context.developerAPIClient.profilesCreateInstance(
|
|
body: .json(
|
|
.init(
|
|
data: .init(
|
|
_type: .profiles,
|
|
attributes: .init(
|
|
name: "SC profile \(bundleID.id)",
|
|
profileType: .iosAppDevelopment
|
|
),
|
|
relationships: .init(
|
|
bundleId: .init(
|
|
data: .init(_type: .bundleIds, id: bundleID.id)
|
|
),
|
|
devices: .init(data: allDevices.map {
|
|
.init(_type: .devices, id: $0.id)
|
|
}),
|
|
certificates: .init(data: certs.map {
|
|
.init(_type: .certificates, id: $0.id)
|
|
})
|
|
)
|
|
)
|
|
)
|
|
)
|
|
)
|
|
.created.body.json.data
|
|
|
|
guard let contentString = response.attributes?.profileContent,
|
|
let contentData = Data(base64Encoded: contentString)
|
|
else { throw Errors.invalidProfileData }
|
|
|
|
return try Mobileprovision(data: contentData)
|
|
}
|
|
|
|
}
|