mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
238 lines
7.5 KiB
Swift
238 lines
7.5 KiB
Swift
// Copyright © 2024 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import Foundation
|
|
|
|
// MARK: - DataSource
|
|
|
|
public protocol DataSource: Equatable {
|
|
var data: Data { get }
|
|
var dataUrl: URL? { get }
|
|
|
|
/// The file path for the data, if it already exists on disk.
|
|
///
|
|
/// This method is safe to call as it will not do any expensive reads or writes.
|
|
///
|
|
/// May return nil if the data does not (yet) reside on disk.
|
|
///
|
|
/// Use `dataUrl` instead if you need to access the data; it will ensure the data is on disk and return a URL, barring an error.
|
|
var dataPathIfOnDisk: String? { get }
|
|
|
|
var dataLength: Int { get }
|
|
var sourceFilename: String? { get set }
|
|
var mimeType: String? { get }
|
|
var shouldDeleteOnDeinit: Bool { get }
|
|
|
|
// MARK: - Functions
|
|
|
|
func write(to path: String) throws
|
|
}
|
|
|
|
public extension DataSource {
|
|
var isValidImage: Bool {
|
|
guard let dataPath: String = self.dataPathIfOnDisk else {
|
|
return self.data.isValidImage
|
|
}
|
|
|
|
// if ows_isValidImage is given a file path, it will
|
|
// avoid loading most of the data into memory, which
|
|
// is considerably more performant, so try to do that.
|
|
return Data.isValidImage(at: dataPath, mimeType: mimeType)
|
|
}
|
|
|
|
var isValidVideo: Bool {
|
|
guard let dataUrl: URL = self.dataUrl else { return false }
|
|
|
|
return MediaUtils.isValidVideo(path: dataUrl.path)
|
|
}
|
|
}
|
|
|
|
// MARK: - DataSourceValue
|
|
|
|
public class DataSourceValue: DataSource {
|
|
public static let empty: DataSourceValue = DataSourceValue(
|
|
data: Data(),
|
|
fileExtension: MimeTypeUtil.FileExtension.syncMessage
|
|
)
|
|
|
|
public var data: Data
|
|
public var sourceFilename: String?
|
|
var fileExtension: String
|
|
var cachedFilePath: String?
|
|
public var shouldDeleteOnDeinit: Bool
|
|
|
|
public var dataUrl: URL? { dataPath.map { URL(fileURLWithPath: $0) } }
|
|
public var dataPathIfOnDisk: String? { cachedFilePath }
|
|
public var dataLength: Int { data.count }
|
|
public var mimeType: String? { MimeTypeUtil.mimeType(for: fileExtension) }
|
|
|
|
var dataPath: String? {
|
|
let fileExtension: String = self.fileExtension
|
|
|
|
return DataSourceValue.synced(self) { [weak self] in
|
|
guard let cachedFilePath: String = self?.cachedFilePath else {
|
|
let filePath: String = FileSystem.temporaryFilePath(fileExtension: fileExtension)
|
|
|
|
do { try self?.write(to: filePath) }
|
|
catch { return nil }
|
|
|
|
self?.cachedFilePath = filePath
|
|
return filePath
|
|
}
|
|
|
|
return cachedFilePath
|
|
}
|
|
}
|
|
|
|
// MARK: - Initialization
|
|
|
|
public init(data: Data, fileExtension: String) {
|
|
self.data = data
|
|
self.fileExtension = fileExtension
|
|
self.shouldDeleteOnDeinit = true
|
|
}
|
|
|
|
convenience init?(data: Data?, fileExtension: String) {
|
|
guard let data: Data = data else { return nil }
|
|
|
|
self.init(data: data, fileExtension: fileExtension)
|
|
}
|
|
|
|
public convenience init?(data: Data?, utiType: String) {
|
|
guard let fileExtension: String = MimeTypeUtil.fileExtension(forUtiType: utiType) else { return nil }
|
|
|
|
self.init(data: data, fileExtension: fileExtension)
|
|
}
|
|
|
|
public convenience init?(text: String?) {
|
|
guard
|
|
let text: String = text,
|
|
let data: Data = text.filteredForDisplay.data(using: .utf8)
|
|
else { return nil }
|
|
|
|
self.init(data: data, fileExtension: MimeTypeUtil.FileExtension.text)
|
|
}
|
|
|
|
convenience init(syncMessageData: Data) {
|
|
self.init(data: syncMessageData, fileExtension: MimeTypeUtil.FileExtension.syncMessage)
|
|
}
|
|
|
|
deinit {
|
|
guard
|
|
shouldDeleteOnDeinit,
|
|
let filePath: String = cachedFilePath
|
|
else { return }
|
|
|
|
DispatchQueue.global(qos: .default).async {
|
|
try? FileManager.default.removeItem(atPath: filePath)
|
|
}
|
|
}
|
|
|
|
// MARK: - Functions
|
|
|
|
@discardableResult private static func synced<T>(_ lock: Any, closure: () -> T) -> T {
|
|
objc_sync_enter(lock)
|
|
let result: T = closure()
|
|
objc_sync_exit(lock)
|
|
return result
|
|
}
|
|
|
|
public func write(to path: String) throws {
|
|
try data.write(to: URL(fileURLWithPath: path), options: .atomic)
|
|
}
|
|
|
|
public static func == (lhs: DataSourceValue, rhs: DataSourceValue) -> Bool {
|
|
return (
|
|
lhs.data == rhs.data &&
|
|
lhs.sourceFilename == rhs.sourceFilename &&
|
|
lhs.fileExtension == rhs.fileExtension &&
|
|
lhs.shouldDeleteOnDeinit == rhs.shouldDeleteOnDeinit
|
|
)
|
|
}
|
|
}
|
|
|
|
// MARK: - DataSourcePath
|
|
|
|
public class DataSourcePath: DataSource {
|
|
public var filePath: String
|
|
public var sourceFilename: String?
|
|
var cachedData: Data?
|
|
var cachedDataLength: Int?
|
|
public var shouldDeleteOnDeinit: Bool
|
|
|
|
public var data: Data {
|
|
let filePath: String = self.filePath
|
|
|
|
return DataSourcePath.synced(self) { [weak self] in
|
|
if let cachedData: Data = self?.cachedData {
|
|
return cachedData
|
|
}
|
|
|
|
let data: Data = ((try? Data(contentsOf: URL(fileURLWithPath: filePath))) ?? Data())
|
|
self?.cachedData = data
|
|
return data
|
|
}
|
|
}
|
|
|
|
public var dataUrl: URL? { URL(fileURLWithPath: filePath) }
|
|
public var dataPathIfOnDisk: String? { filePath }
|
|
|
|
public var dataLength: Int {
|
|
let filePath: String = self.filePath
|
|
|
|
return DataSourcePath.synced(self) { [weak self] in
|
|
if let cachedDataLength: Int = self?.cachedDataLength {
|
|
return cachedDataLength
|
|
}
|
|
|
|
let attrs: [FileAttributeKey: Any]? = try? FileManager.default.attributesOfItem(atPath: filePath)
|
|
let length: Int = ((attrs?[FileAttributeKey.size] as? Int) ?? 0)
|
|
self?.cachedDataLength = length
|
|
return length
|
|
}
|
|
}
|
|
|
|
public var mimeType: String? { MimeTypeUtil.mimeType(for: URL(fileURLWithPath: filePath).pathExtension) }
|
|
|
|
// MARK: - Initialization
|
|
|
|
public init(filePath: String, shouldDeleteOnDeinit: Bool) {
|
|
self.filePath = filePath
|
|
self.shouldDeleteOnDeinit = shouldDeleteOnDeinit
|
|
}
|
|
|
|
public convenience init?(fileUrl: URL?, shouldDeleteOnDeinit: Bool) {
|
|
guard let fileUrl: URL = fileUrl, fileUrl.isFileURL else { return nil }
|
|
|
|
self.init(filePath: fileUrl.path, shouldDeleteOnDeinit: shouldDeleteOnDeinit)
|
|
}
|
|
|
|
deinit {
|
|
guard shouldDeleteOnDeinit else { return }
|
|
|
|
DispatchQueue.global(qos: .default).async { [filePath] in
|
|
try? FileManager.default.removeItem(atPath: filePath)
|
|
}
|
|
}
|
|
|
|
// MARK: - Functions
|
|
|
|
@discardableResult private static func synced<T>(_ lock: Any, closure: () -> T) -> T {
|
|
objc_sync_enter(lock)
|
|
let result: T = closure()
|
|
objc_sync_exit(lock)
|
|
return result
|
|
}
|
|
|
|
public func write(to path: String) throws {
|
|
try FileManager.default.copyItem(atPath: filePath, toPath: path)
|
|
}
|
|
|
|
public static func == (lhs: DataSourcePath, rhs: DataSourcePath) -> Bool {
|
|
return (
|
|
lhs.filePath == rhs.filePath &&
|
|
lhs.sourceFilename == rhs.sourceFilename &&
|
|
lhs.shouldDeleteOnDeinit == rhs.shouldDeleteOnDeinit
|
|
)
|
|
}
|
|
}
|