Fixed a few more bugs

• Fixed an issue where excessive job failures could result in jobs not running onLaunch or onActive as expected (extended network issues resulted in the default communities not getting fetched)
• Fixed an issue where the PathVC could end up in a weird state after the IP2Country cache gets updated
• Fixed an issue where LinkPreview caching was incorrectly case sensitive
• Fixed an issue where the IP2Country might load it's cache on the wrong thread
• Added the ability to copy the file path to the log file when exporting on the simulator
• Updated the Request type to correctly encode a body of 'Data' type when given (would previously incorrectly encode it as JSON)
pull/986/head
Morgan Pretty 9 months ago
parent 25150f931e
commit a8d9200ef5

@ -129,7 +129,10 @@ final class PathVC: BaseVC {
// Register for path country updates
cacheUpdateId = IP2Country.onCacheLoaded { [weak self] in
DispatchQueue.main.async {
self?.update(paths: (self?.lastPath.map { [$0] } ?? []), force: true)
switch (self?.lastPath, self?.lastPath.isEmpty == true) {
case (.none, _), (_, true): self?.update(paths: [], force: true)
case (.some(let lastPath), _): self?.update(paths: [lastPath], force: true)
}
}
}
}

@ -178,6 +178,50 @@ class HelpViewModel: SessionTableViewModel, NavigatableStateHolder, ObservableTa
targetView: UIView? = nil,
animated: Bool = true,
onShareComplete: (() -> ())? = nil
) {
guard
let latestLogFilePath: String = Log.logFilePath(),
Singleton.hasAppContext,
let viewController: UIViewController = Singleton.appContext.frontmostViewController
else { return }
#if targetEnvironment(simulator)
let modal: ConfirmationModal = ConfirmationModal(
info: ConfirmationModal.Info(
title: "Export Logs", // stringlint:disable
body: .text(
"How would you like to export the logs?\n\n(This modal only appears on the Simulator)" // stringlint:disable
),
confirmTitle: "Copy Path", // stringlint:disable
cancelTitle: "Share", // stringlint:disable
cancelStyle: .alert_text,
onConfirm: { _ in UIPasteboard.general.string = latestLogFilePath },
onCancel: { _ in
HelpViewModel.shareLogsInternal(
viewControllerToDismiss: viewControllerToDismiss,
targetView: targetView,
animated: animated,
onShareComplete: onShareComplete
)
}
)
)
viewController.present(modal, animated: animated, completion: nil)
#else
HelpViewModel.shareLogsInternal(
viewControllerToDismiss: viewControllerToDismiss,
targetView: targetView,
animated: animated,
onShareComplete: onShareComplete
)
#endif
}
private static func shareLogsInternal(
viewControllerToDismiss: UIViewController? = nil,
targetView: UIView? = nil,
animated: Bool = true,
onShareComplete: (() -> ())? = nil
) {
Log.info("[Version] \(SessionApp.versionInfo)")
Log.flush()

@ -46,6 +46,10 @@ public enum IP2Country {
static func populateCacheIfNeededAsync() {
DispatchQueue.global(qos: .utility).async {
// Ensure the caches get loaded in the background
_ = ipv4Table
_ = countryNamesCache
pathsChangedCallbackId.mutate { pathsChangedCallbackId in
guard pathsChangedCallbackId == nil else { return }

@ -299,7 +299,10 @@ public extension LinkPreview {
return Fail(error: LinkPreviewError.featureDisabled)
.eraseToAnyPublisher()
}
guard let previewUrl: String = previewUrl else {
// Force the url to lowercase to ensure we casing doesn't result in redownloading the
// details
guard let previewUrl: String = previewUrl?.lowercased() else {
return Fail(error: LinkPreviewError.invalidInput)
.eraseToAnyPublisher()
}

@ -491,7 +491,14 @@ public final class JobRunner: JobRunnerType {
// Add and start any blocking jobs
blockingQueue.wrappedValue?.appDidFinishLaunching(
with: jobsToRun.blocking,
with: jobsToRun.blocking.map { job -> Job in
guard job.behaviour == .recurringOnLaunch else { return job }
// If the job is a `recurringOnLaunch` job then we reset the `nextRunTimestamp`
// value on the instance because the assumption is that `recurringOnLaunch` will
// run a job regardless of how many times it previously failed
return job.with(nextRunTimestamp: 0)
},
canStart: true,
using: dependencies
)
@ -503,7 +510,14 @@ public final class JobRunner: JobRunnerType {
jobsByVariant.forEach { variant, jobs in
jobQueues[variant]?.appDidFinishLaunching(
with: jobs,
with: jobs.map { job -> Job in
guard job.behaviour == .recurringOnLaunch else { return job }
// If the job is a `recurringOnLaunch` job then we reset the `nextRunTimestamp`
// value on the instance because the assumption is that `recurringOnLaunch` will
// run a job regardless of how many times it previously failed
return job.with(nextRunTimestamp: 0)
},
canStart: false,
using: dependencies
)
@ -562,7 +576,12 @@ public final class JobRunner: JobRunnerType {
}
.forEach { queue, jobs in
queue.appDidBecomeActive(
with: jobs,
with: jobs.map { job -> Job in
// We reset the `nextRunTimestamp` value on the instance because the
// assumption is that `recurringOnActive` will run a job regardless
// of how many times it previously failed
job.with(nextRunTimestamp: 0)
},
canStart: !blockingQueueIsRunning,
using: dependencies
)

@ -70,6 +70,9 @@ public struct Request<T: Encodable, Endpoint: EndpointType> {
case let bodyBytes as [UInt8]:
return Data(bodyBytes)
case let bodyDirectData as Data:
return bodyDirectData
default:
// Having no body is fine so just return nil

@ -749,6 +749,33 @@ class JobRunnerSpec: QuickSpec {
jobRunner.appDidFinishLaunching(using: dependencies)
expect(jobRunner.allJobInfo()).to(beEmpty())
}
// MARK: -------- runs the job regardless of the nextRunTimestamp value it has
it("runs the job regardless of the nextRunTimestamp value it has") {
job1 = Job(
id: 100,
failureCount: 0,
variant: .messageSend,
behaviour: .recurringOnLaunch,
shouldBlock: false,
shouldBeUnique: false,
shouldSkipLaunchBecomeActive: false,
nextRunTimestamp: Date.distantFuture.timeIntervalSince1970,
threadId: nil,
interactionId: nil,
details: try? JSONEncoder()
.with(outputFormatting: .sortedKeys)
.encode(TestDetails(completeTime: 1))
)
mockStorage.write { db in
try job1.upsert(db)
}
jobRunner.appDidFinishLaunching(using: dependencies)
jobRunner.appDidBecomeActive(using: dependencies)
expect(jobRunner.isCurrentlyRunning(job1)).to(beTrue())
}
}
// MARK: ------ by being notified of app becoming active
@ -1058,6 +1085,33 @@ class JobRunnerSpec: QuickSpec {
expect(jobRunner.isCurrentlyRunning(job1)).to(beTrue())
expect(jobRunner.isCurrentlyRunning(job2)).to(beTrue())
}
// MARK: -------- runs the job regardless of the nextRunTimestamp value it has
it("runs the job regardless of the nextRunTimestamp value it has") {
job1 = Job(
id: 100,
failureCount: 0,
variant: .messageSend,
behaviour: .recurringOnActive,
shouldBlock: false,
shouldBeUnique: false,
shouldSkipLaunchBecomeActive: false,
nextRunTimestamp: Date.distantFuture.timeIntervalSince1970,
threadId: nil,
interactionId: nil,
details: try? JSONEncoder()
.with(outputFormatting: .sortedKeys)
.encode(TestDetails(completeTime: 1))
)
mockStorage.write { db in
try job1.upsert(db)
}
jobRunner.appDidFinishLaunching(using: dependencies)
jobRunner.appDidBecomeActive(using: dependencies)
expect(jobRunner.isCurrentlyRunning(job1)).to(beTrue())
}
}
// MARK: ------ by checking if a job can be added to the queue

Loading…
Cancel
Save