2019-05-09 15:29:54 -07:00
|
|
|
//
|
2019-07-31 14:07:00 -07:00
|
|
|
// StoreApp.swift
|
2019-05-09 15:29:54 -07:00
|
|
|
// AltStore
|
|
|
|
|
//
|
2019-05-20 21:24:53 +02:00
|
|
|
// Created by Riley Testut on 5/20/19.
|
2019-05-09 15:29:54 -07:00
|
|
|
// Copyright © 2019 Riley Testut. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import Foundation
|
2019-05-20 21:24:53 +02:00
|
|
|
import CoreData
|
2019-05-09 15:29:54 -07:00
|
|
|
|
2019-05-30 17:10:50 -07:00
|
|
|
import Roxas
|
2019-07-28 15:08:13 -07:00
|
|
|
import AltSign
|
2019-05-30 17:10:50 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
public extension StoreApp
|
2019-06-17 16:31:10 -07:00
|
|
|
{
|
2020-04-01 11:51:00 -07:00
|
|
|
#if ALPHA
|
2022-12-30 16:51:36 -05:00
|
|
|
static let altstoreAppID = Bundle.Info.appbundleIdentifier
|
2020-04-01 11:51:00 -07:00
|
|
|
#elseif BETA
|
2022-12-30 16:51:36 -05:00
|
|
|
static let altstoreAppID = Bundle.Info.appbundleIdentifier
|
2019-09-12 13:04:15 -07:00
|
|
|
#else
|
2022-12-30 16:51:36 -05:00
|
|
|
static let altstoreAppID = Bundle.Info.appbundleIdentifier
|
2019-09-12 13:04:15 -07:00
|
|
|
#endif
|
2020-05-08 11:45:23 -07:00
|
|
|
|
|
|
|
|
static let dolphinAppID = "me.oatmealdome.dolphinios-njb"
|
2019-06-17 16:31:10 -07:00
|
|
|
}
|
|
|
|
|
|
2021-09-15 20:55:41 -04:00
|
|
|
@objc
|
|
|
|
|
public enum Platform: UInt {
|
|
|
|
|
case ios
|
|
|
|
|
case tvos
|
|
|
|
|
case macos
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
extension Platform: Decodable {}
|
|
|
|
|
|
|
|
|
|
@objc
|
|
|
|
|
public final class PlatformURL: NSManagedObject, Decodable {
|
|
|
|
|
/* Properties */
|
|
|
|
|
@NSManaged public private(set) var platform: Platform
|
|
|
|
|
@NSManaged public private(set) var downloadURL: URL
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private enum CodingKeys: String, CodingKey
|
|
|
|
|
{
|
|
|
|
|
case platform
|
|
|
|
|
case downloadURL
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public init(from decoder: Decoder) throws
|
|
|
|
|
{
|
|
|
|
|
guard let context = decoder.managedObjectContext else { preconditionFailure("Decoder must have non-nil NSManagedObjectContext.") }
|
|
|
|
|
|
|
|
|
|
// Must initialize with context in order for child context saves to work correctly.
|
|
|
|
|
super.init(entity: PlatformURL.entity(), insertInto: context)
|
|
|
|
|
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
let container = try decoder.container(keyedBy: CodingKeys.self)
|
|
|
|
|
self.platform = try container.decode(Platform.self, forKey: .platform)
|
|
|
|
|
self.downloadURL = try container.decode(URL.self, forKey: .downloadURL)
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
if let context = self.managedObjectContext
|
|
|
|
|
{
|
|
|
|
|
context.delete(self)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw error
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
extension PlatformURL: Comparable {
|
|
|
|
|
public static func < (lhs: PlatformURL, rhs: PlatformURL) -> Bool {
|
|
|
|
|
return lhs.platform.rawValue < rhs.platform.rawValue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static func > (lhs: PlatformURL, rhs: PlatformURL) -> Bool {
|
|
|
|
|
return lhs.platform.rawValue > rhs.platform.rawValue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static func <= (lhs: PlatformURL, rhs: PlatformURL) -> Bool {
|
|
|
|
|
return lhs.platform.rawValue <= rhs.platform.rawValue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static func >= (lhs: PlatformURL, rhs: PlatformURL) -> Bool {
|
|
|
|
|
return lhs.platform.rawValue >= rhs.platform.rawValue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public typealias PlatformURLs = [PlatformURL]
|
|
|
|
|
|
2019-07-31 14:07:00 -07:00
|
|
|
@objc(StoreApp)
|
2020-09-03 16:39:08 -07:00
|
|
|
public class StoreApp: NSManagedObject, Decodable, Fetchable
|
2019-05-09 15:29:54 -07:00
|
|
|
{
|
2019-05-20 21:24:53 +02:00
|
|
|
/* Properties */
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public private(set) var name: String
|
|
|
|
|
@NSManaged public private(set) var bundleIdentifier: String
|
|
|
|
|
@NSManaged public private(set) var subtitle: String?
|
2019-05-09 17:21:55 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public private(set) var developerName: String
|
|
|
|
|
@NSManaged public private(set) var localizedDescription: String
|
|
|
|
|
@NSManaged public private(set) var size: Int32
|
2019-05-09 15:29:54 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public private(set) var iconURL: URL
|
|
|
|
|
@NSManaged public private(set) var screenshotURLs: [URL]
|
2019-05-20 21:24:53 +02:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public var version: String
|
|
|
|
|
@NSManaged public private(set) var versionDate: Date
|
|
|
|
|
@NSManaged public private(set) var versionDescription: String?
|
2019-05-20 22:24:16 +02:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public private(set) var downloadURL: URL
|
2021-09-15 20:55:41 -04:00
|
|
|
@NSManaged public private(set) var platformURLs: PlatformURLs?
|
|
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public private(set) var tintColor: UIColor?
|
|
|
|
|
@NSManaged public private(set) var isBeta: Bool
|
2019-05-31 18:24:08 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public var sourceIdentifier: String?
|
2020-03-24 13:27:44 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public var sortIndex: Int32
|
2019-07-30 17:00:04 -07:00
|
|
|
|
2019-05-20 21:26:01 +02:00
|
|
|
/* Relationships */
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged public var installedApp: InstalledApp?
|
|
|
|
|
@NSManaged public var newsItems: Set<NewsItem>
|
2020-03-24 13:27:44 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@NSManaged @objc(source) public var _source: Source?
|
|
|
|
|
@NSManaged @objc(permissions) public var _permissions: NSOrderedSet
|
2020-03-24 13:27:44 -07:00
|
|
|
|
2022-09-08 16:14:28 -05:00
|
|
|
@NSManaged public private(set) var loggedErrors: NSSet /* Set<LoggedError> */ // Use NSSet to avoid eagerly fetching values.
|
|
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@nonobjc public var source: Source? {
|
2020-03-24 13:27:44 -07:00
|
|
|
set {
|
|
|
|
|
self._source = newValue
|
|
|
|
|
self.sourceIdentifier = newValue?.identifier
|
|
|
|
|
}
|
|
|
|
|
get {
|
|
|
|
|
return self._source
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-24 12:23:54 -07:00
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
@nonobjc public var permissions: [AppPermission] {
|
2019-07-24 12:23:54 -07:00
|
|
|
return self._permissions.array as! [AppPermission]
|
|
|
|
|
}
|
2019-05-20 21:26:01 +02:00
|
|
|
|
2019-05-20 21:24:53 +02:00
|
|
|
private override init(entity: NSEntityDescription, insertInto context: NSManagedObjectContext?)
|
|
|
|
|
{
|
|
|
|
|
super.init(entity: entity, insertInto: context)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private enum CodingKeys: String, CodingKey
|
|
|
|
|
{
|
|
|
|
|
case name
|
2019-07-28 15:08:13 -07:00
|
|
|
case bundleIdentifier
|
2019-05-20 21:24:53 +02:00
|
|
|
case developerName
|
|
|
|
|
case localizedDescription
|
2019-05-20 22:24:16 +02:00
|
|
|
case version
|
|
|
|
|
case versionDescription
|
|
|
|
|
case versionDate
|
2019-08-20 19:06:03 -05:00
|
|
|
case iconURL
|
|
|
|
|
case screenshotURLs
|
2019-05-31 18:24:08 -07:00
|
|
|
case downloadURL
|
2021-09-15 20:55:41 -04:00
|
|
|
case platformURLs
|
2019-07-16 14:25:09 -07:00
|
|
|
case tintColor
|
|
|
|
|
case subtitle
|
2019-07-24 12:23:54 -07:00
|
|
|
case permissions
|
2019-07-29 16:02:15 -07:00
|
|
|
case size
|
2019-08-28 11:13:22 -07:00
|
|
|
case isBeta = "beta"
|
2019-05-20 21:24:53 +02:00
|
|
|
}
|
|
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
public required init(from decoder: Decoder) throws
|
2019-05-20 21:24:53 +02:00
|
|
|
{
|
|
|
|
|
guard let context = decoder.managedObjectContext else { preconditionFailure("Decoder must have non-nil NSManagedObjectContext.") }
|
|
|
|
|
|
2020-08-27 16:23:50 -07:00
|
|
|
// Must initialize with context in order for child context saves to work correctly.
|
|
|
|
|
super.init(entity: StoreApp.entity(), insertInto: context)
|
2019-05-20 21:24:53 +02:00
|
|
|
|
2020-08-27 16:23:50 -07:00
|
|
|
do
|
2019-07-16 14:25:09 -07:00
|
|
|
{
|
2020-08-27 16:23:50 -07:00
|
|
|
let container = try decoder.container(keyedBy: CodingKeys.self)
|
|
|
|
|
self.name = try container.decode(String.self, forKey: .name)
|
|
|
|
|
self.bundleIdentifier = try container.decode(String.self, forKey: .bundleIdentifier)
|
|
|
|
|
self.developerName = try container.decode(String.self, forKey: .developerName)
|
|
|
|
|
self.localizedDescription = try container.decode(String.self, forKey: .localizedDescription)
|
|
|
|
|
|
|
|
|
|
self.subtitle = try container.decodeIfPresent(String.self, forKey: .subtitle)
|
|
|
|
|
|
|
|
|
|
self.version = try container.decode(String.self, forKey: .version)
|
|
|
|
|
self.versionDate = try container.decode(Date.self, forKey: .versionDate)
|
|
|
|
|
self.versionDescription = try container.decodeIfPresent(String.self, forKey: .versionDescription)
|
|
|
|
|
|
|
|
|
|
self.iconURL = try container.decode(URL.self, forKey: .iconURL)
|
|
|
|
|
self.screenshotURLs = try container.decodeIfPresent([URL].self, forKey: .screenshotURLs) ?? []
|
|
|
|
|
|
2021-09-15 20:55:41 -04:00
|
|
|
let downloadURL = try container.decodeIfPresent(URL.self, forKey: .downloadURL)
|
|
|
|
|
let platformURLs = try container.decodeIfPresent(PlatformURLs.self.self, forKey: .platformURLs)
|
|
|
|
|
if let platformURLs = platformURLs {
|
|
|
|
|
self.platformURLs = platformURLs
|
|
|
|
|
// Backwards compatibility, use the fiirst (iOS will be first since sorted that way)
|
|
|
|
|
if let first = platformURLs.sorted().first {
|
|
|
|
|
self.downloadURL = first.downloadURL
|
|
|
|
|
} else {
|
|
|
|
|
throw DecodingError.dataCorruptedError(forKey: .platformURLs, in: container, debugDescription: "platformURLs has no entries")
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else if let downloadURL = downloadURL {
|
|
|
|
|
self.downloadURL = downloadURL
|
|
|
|
|
} else {
|
|
|
|
|
throw DecodingError.dataCorruptedError(forKey: .downloadURL, in: container, debugDescription: "E downloadURL:String or downloadURLs:[[Platform:URL]] key required.")
|
|
|
|
|
}
|
2020-08-27 16:23:50 -07:00
|
|
|
|
|
|
|
|
if let tintColorHex = try container.decodeIfPresent(String.self, forKey: .tintColor)
|
|
|
|
|
{
|
|
|
|
|
guard let tintColor = UIColor(hexString: tintColorHex) else {
|
|
|
|
|
throw DecodingError.dataCorruptedError(forKey: .tintColor, in: container, debugDescription: "Hex code is invalid.")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.tintColor = tintColor
|
2019-07-16 14:25:09 -07:00
|
|
|
}
|
|
|
|
|
|
2020-08-27 16:23:50 -07:00
|
|
|
self.size = try container.decode(Int32.self, forKey: .size)
|
|
|
|
|
self.isBeta = try container.decodeIfPresent(Bool.self, forKey: .isBeta) ?? false
|
|
|
|
|
|
|
|
|
|
let permissions = try container.decodeIfPresent([AppPermission].self, forKey: .permissions) ?? []
|
|
|
|
|
self._permissions = NSOrderedSet(array: permissions)
|
|
|
|
|
}
|
|
|
|
|
catch
|
|
|
|
|
{
|
|
|
|
|
if let context = self.managedObjectContext
|
|
|
|
|
{
|
|
|
|
|
context.delete(self)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw error
|
2019-07-16 14:25:09 -07:00
|
|
|
}
|
2019-05-20 21:24:53 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-03 16:39:08 -07:00
|
|
|
public extension StoreApp
|
2019-05-20 21:24:53 +02:00
|
|
|
{
|
2019-07-31 14:07:00 -07:00
|
|
|
@nonobjc class func fetchRequest() -> NSFetchRequest<StoreApp>
|
2019-05-20 21:24:53 +02:00
|
|
|
{
|
2019-07-31 14:07:00 -07:00
|
|
|
return NSFetchRequest<StoreApp>(entityName: "StoreApp")
|
2019-05-20 21:24:53 +02:00
|
|
|
}
|
2019-06-17 16:31:10 -07:00
|
|
|
|
2019-07-31 14:07:00 -07:00
|
|
|
class func makeAltStoreApp(in context: NSManagedObjectContext) -> StoreApp
|
2019-06-17 16:31:10 -07:00
|
|
|
{
|
2019-07-31 14:07:00 -07:00
|
|
|
let app = StoreApp(context: context)
|
2022-11-05 23:50:07 -07:00
|
|
|
app.name = "SideStore"
|
2019-07-31 14:07:00 -07:00
|
|
|
app.bundleIdentifier = StoreApp.altstoreAppID
|
2022-10-18 01:10:18 -07:00
|
|
|
app.developerName = "Side Team"
|
2022-11-05 23:50:07 -07:00
|
|
|
app.localizedDescription = "SideStore is an alternative App Store."
|
2019-08-20 19:06:03 -05:00
|
|
|
app.iconURL = URL(string: "https://user-images.githubusercontent.com/705880/63392210-540c5980-c37b-11e9-968c-8742fc68ab2e.png")!
|
|
|
|
|
app.screenshotURLs = []
|
2019-06-17 16:31:10 -07:00
|
|
|
app.version = "1.0"
|
|
|
|
|
app.versionDate = Date()
|
|
|
|
|
app.downloadURL = URL(string: "http://rileytestut.com")!
|
|
|
|
|
|
2019-09-12 13:04:15 -07:00
|
|
|
#if BETA
|
|
|
|
|
app.isBeta = true
|
|
|
|
|
#endif
|
|
|
|
|
|
2019-06-17 16:31:10 -07:00
|
|
|
return app
|
|
|
|
|
}
|
2019-05-09 15:29:54 -07:00
|
|
|
}
|