Fixed a few issues found during QA

• Fixed an issue where incoming legacy group messages were failing to decrypt
• Fixed an issue where decoding push notifications could result in an infinite loop
• Fixed an issue where the extensions would incorrectly try to append extension logs (only want the main app to do this)
• Updated the accessibility ids for the switches and radios on the privacy and disappearing message settings screens
pull/1005/head
Morgan Pretty 8 months ago
parent 5da842a109
commit 027ce1604d

@ -1 +1 @@
Subproject commit ba7919d304bfe44a75a77d418638bf82ac0b7f93 Subproject commit af0ab996da46bcbeeea5e70831cda7e23c955243

@ -7673,7 +7673,7 @@
CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES; CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
CODE_SIGN_IDENTITY = "iPhone Developer"; CODE_SIGN_IDENTITY = "iPhone Developer";
CURRENT_PROJECT_VERSION = 474; CURRENT_PROJECT_VERSION = 475;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES; ENABLE_STRICT_OBJC_MSGSEND = YES;
ENABLE_TESTABILITY = YES; ENABLE_TESTABILITY = YES;
@ -7751,7 +7751,7 @@
CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES; CLANG_WARN__ARC_BRIDGE_CAST_NONARC = YES;
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
CODE_SIGN_IDENTITY = "iPhone Distribution"; CODE_SIGN_IDENTITY = "iPhone Distribution";
CURRENT_PROJECT_VERSION = 474; CURRENT_PROJECT_VERSION = 475;
ENABLE_BITCODE = NO; ENABLE_BITCODE = NO;
ENABLE_STRICT_OBJC_MSGSEND = YES; ENABLE_STRICT_OBJC_MSGSEND = YES;
GCC_NO_COMMON_BLOCKS = YES; GCC_NO_COMMON_BLOCKS = YES;

@ -130,7 +130,10 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
id: "DISAPPEARING_MESSAGES_OFF".localized(), id: "DISAPPEARING_MESSAGES_OFF".localized(),
title: "DISAPPEARING_MESSAGES_OFF".localized(), title: "DISAPPEARING_MESSAGES_OFF".localized(),
rightAccessory: .radio( rightAccessory: .radio(
isSelected: { (self?.currentSelection.value.isEnabled == false) } isSelected: { (self?.currentSelection.value.isEnabled == false) },
accessibility: Accessibility(
identifier: "Off - Radio"
)
), ),
accessibility: Accessibility( accessibility: Accessibility(
identifier: "Disable disappearing messages (Off option)", identifier: "Disable disappearing messages (Off option)",
@ -154,7 +157,10 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
isSelected: { isSelected: {
(self?.currentSelection.value.isEnabled == true) && (self?.currentSelection.value.isEnabled == true) &&
(self?.currentSelection.value.type == .disappearAfterRead) (self?.currentSelection.value.type == .disappearAfterRead)
} },
accessibility: Accessibility(
identifier: "Disappear After Read - Radio"
)
), ),
accessibility: Accessibility( accessibility: Accessibility(
identifier: "Disappear after read option", identifier: "Disappear after read option",
@ -184,7 +190,10 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
isSelected: { isSelected: {
(self?.currentSelection.value.isEnabled == true) && (self?.currentSelection.value.isEnabled == true) &&
(self?.currentSelection.value.type == .disappearAfterSend) (self?.currentSelection.value.type == .disappearAfterSend)
} },
accessibility: Accessibility(
identifier: "Disappear After Read - Radio"
)
), ),
accessibility: Accessibility( accessibility: Accessibility(
identifier: "Disappear after send option", identifier: "Disappear after send option",
@ -226,7 +235,10 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
isSelected: { isSelected: {
(self?.currentSelection.value.isEnabled == true) && (self?.currentSelection.value.isEnabled == true) &&
(self?.currentSelection.value.durationSeconds == duration) (self?.currentSelection.value.durationSeconds == duration)
} },
accessibility: Accessibility(
identifier: "\(title) - Radio"
)
), ),
accessibility: Accessibility( accessibility: Accessibility(
identifier: "Time option", identifier: "Time option",
@ -252,7 +264,10 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
id: "DISAPPEARING_MESSAGES_OFF".localized(), id: "DISAPPEARING_MESSAGES_OFF".localized(),
title: "DISAPPEARING_MESSAGES_OFF".localized(), title: "DISAPPEARING_MESSAGES_OFF".localized(),
rightAccessory: .radio( rightAccessory: .radio(
isSelected: { (self?.currentSelection.value.isEnabled == false) } isSelected: { (self?.currentSelection.value.isEnabled == false) },
accessibility: Accessibility(
identifier: "Off - Radio"
)
), ),
isEnabled: ( isEnabled: (
isNoteToSelf || isNoteToSelf ||
@ -287,7 +302,10 @@ class ThreadDisappearingMessagesSettingsViewModel: SessionTableViewModel, Naviga
isSelected: { isSelected: {
(self?.currentSelection.value.isEnabled == true) && (self?.currentSelection.value.isEnabled == true) &&
(self?.currentSelection.value.durationSeconds == duration) (self?.currentSelection.value.durationSeconds == duration)
} },
accessibility: Accessibility(
identifier: "\(title) - Radio"
)
), ),
isEnabled: (isNoteToSelf || currentUserIsClosedGroupAdmin == true), isEnabled: (isNoteToSelf || currentUserIsClosedGroupAdmin == true),
accessibility: Accessibility( accessibility: Accessibility(

@ -114,6 +114,9 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
key: .isScreenLockEnabled, key: .isScreenLockEnabled,
value: current.isScreenLockEnabled, value: current.isScreenLockEnabled,
oldValue: (previous ?? current).isScreenLockEnabled oldValue: (previous ?? current).isScreenLockEnabled
),
accessibility: Accessibility(
identifier: "Lock App - Switch"
) )
), ),
onTap: { [weak self] in onTap: { [weak self] in
@ -152,6 +155,9 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
key: .checkForCommunityMessageRequests, key: .checkForCommunityMessageRequests,
value: current.checkForCommunityMessageRequests, value: current.checkForCommunityMessageRequests,
oldValue: (previous ?? current).checkForCommunityMessageRequests oldValue: (previous ?? current).checkForCommunityMessageRequests
),
accessibility: Accessibility(
identifier: "Community Message Requests - Switch"
) )
), ),
onTap: { [weak self] in onTap: { [weak self] in
@ -177,6 +183,9 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
key: .areReadReceiptsEnabled, key: .areReadReceiptsEnabled,
value: current.areReadReceiptsEnabled, value: current.areReadReceiptsEnabled,
oldValue: (previous ?? current).areReadReceiptsEnabled oldValue: (previous ?? current).areReadReceiptsEnabled
),
accessibility: Accessibility(
identifier: "Read Receipts - Switch"
) )
), ),
onTap: { onTap: {
@ -234,6 +243,9 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
key: .typingIndicatorsEnabled, key: .typingIndicatorsEnabled,
value: current.typingIndicatorsEnabled, value: current.typingIndicatorsEnabled,
oldValue: (previous ?? current).typingIndicatorsEnabled oldValue: (previous ?? current).typingIndicatorsEnabled
),
accessibility: Accessibility(
identifier: "Typing Indicators - Switch"
) )
), ),
onTap: { onTap: {
@ -256,6 +268,9 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
key: .areLinkPreviewsEnabled, key: .areLinkPreviewsEnabled,
value: current.areLinkPreviewsEnabled, value: current.areLinkPreviewsEnabled,
oldValue: (previous ?? current).areLinkPreviewsEnabled oldValue: (previous ?? current).areLinkPreviewsEnabled
),
accessibility: Accessibility(
identifier: "Send Link Previews - Switch"
) )
), ),
onTap: { onTap: {
@ -278,6 +293,9 @@ class PrivacySettingsViewModel: SessionTableViewModel, NavigationItemSource, Nav
key: .areCallsEnabled, key: .areCallsEnabled,
value: current.areCallsEnabled, value: current.areCallsEnabled,
oldValue: (previous ?? current).areCallsEnabled oldValue: (previous ?? current).areCallsEnabled
),
accessibility: Accessibility(
identifier: "Voice and Video Calls - Switch"
) )
), ),
accessibility: Accessibility( accessibility: Accessibility(

@ -322,7 +322,7 @@ extension SessionCell.Accessory {
// MARK: - .toggle Variants // MARK: - .toggle Variants
public static func toggle(_ dataSource: DataSource) -> SessionCell.Accessory { public static func toggle(_ dataSource: DataSource) -> SessionCell.Accessory {
return .toggle(dataSource, accessibility: nil) return .toggle(dataSource, accessibility: Accessibility(identifier: "Switch"))
} }
// MARK: - .dropDown Variants // MARK: - .dropDown Variants
@ -334,11 +334,30 @@ extension SessionCell.Accessory {
// MARK: - .radio Variants // MARK: - .radio Variants
public static func radio(isSelected: @escaping () -> Bool) -> SessionCell.Accessory { public static func radio(isSelected: @escaping () -> Bool) -> SessionCell.Accessory {
return .radio(size: .medium, isSelected: isSelected, storedSelection: false, accessibility: nil) return .radio(
size: .medium,
isSelected: isSelected,
storedSelection: false,
accessibility: Accessibility(identifier: "Radio")
)
}
public static func radio(isSelected: @escaping () -> Bool, accessibility: Accessibility) -> SessionCell.Accessory {
return .radio(
size: .medium,
isSelected: isSelected,
storedSelection: false,
accessibility: accessibility
)
} }
public static func radio(isSelected: @escaping () -> Bool, storedSelection: Bool) -> SessionCell.Accessory { public static func radio(isSelected: @escaping () -> Bool, storedSelection: Bool) -> SessionCell.Accessory {
return .radio(size: .medium, isSelected: isSelected, storedSelection: storedSelection, accessibility: nil) return .radio(
size: .medium,
isSelected: isSelected,
storedSelection: storedSelection,
accessibility: Accessibility(identifier: "Radio")
)
} }
// MARK: - .highlightingBackgroundLabel Variants // MARK: - .highlightingBackgroundLabel Variants

@ -380,14 +380,16 @@ extension SessionCell {
let isSelected: Bool = isSelectedRetriever() let isSelected: Bool = isSelectedRetriever()
let wasOldSelection: Bool = (!isSelected && storedSelection) let wasOldSelection: Bool = (!isSelected && storedSelection)
radioView.isAccessibilityElement = true radioBorderView.isAccessibilityElement = true
radioBorderView.accessibilityIdentifier = accessibility?.identifier
radioBorderView.accessibilityLabel = accessibility?.label
if isSelected || wasOldSelection { if isSelected || wasOldSelection {
radioView.accessibilityTraits.insert(.selected) radioBorderView.accessibilityTraits.insert(.selected)
radioView.accessibilityValue = "selected" radioBorderView.accessibilityValue = "selected"
} else { } else {
radioView.accessibilityTraits.remove(.selected) radioBorderView.accessibilityTraits.remove(.selected)
radioView.accessibilityValue = nil radioBorderView.accessibilityValue = nil
} }
radioBorderView.isHidden = false radioBorderView.isHidden = false
@ -402,8 +404,6 @@ extension SessionCell {
radioBorderView.layer.cornerRadius = (size.borderSize / 2) radioBorderView.layer.cornerRadius = (size.borderSize / 2)
radioView.accessibilityIdentifier = accessibility?.identifier
radioView.accessibilityLabel = accessibility?.label
radioView.alpha = (wasOldSelection ? 0.3 : 1) radioView.alpha = (wasOldSelection ? 0.3 : 1)
radioView.isHidden = (!isSelected && !storedSelection) radioView.isHidden = (!isSelected && !storedSelection)
radioView.themeBackgroundColor = { radioView.themeBackgroundColor = {

@ -170,7 +170,7 @@ public extension Crypto.Generator {
guard guard
cX25519Pubkey.count == 32, cX25519Pubkey.count == 32,
cX25519Seckey.count == 32, cX25519Seckey.count == 64,
session_decrypt_incoming_legacy_group( session_decrypt_incoming_legacy_group(
&cCiphertext, &cCiphertext,
cCiphertext.count, cCiphertext.count,

@ -376,6 +376,13 @@ public class Logger {
return return
} }
// We only want to append extension logs to the main app logs (so just early out if this isn't
// the main app)
guard Singleton.hasAppContext && Singleton.appContext.isMainApp else {
self?.completeResumeLogging()
return
}
DDLog.loggingQueue.async { DDLog.loggingQueue.async {
let extensionInfo: [(dir: String, type: ExtensionType)] = [ let extensionInfo: [(dir: String, type: ExtensionType)] = [
("\(FileManager.default.appSharedDataDirectoryPath)/Logs/NotificationExtension", .notification), ("\(FileManager.default.appSharedDataDirectoryPath)/Logs/NotificationExtension", .notification),

@ -536,7 +536,46 @@ extension _BencodeDecoder.SingleValueContainer: SingleValueDecodingContainer {
) )
} }
func decode(_ type: Int.Type) throws -> Int { func decode(_ type: String.Type) throws -> String {
guard
let decodedData = _BencodeDecoder.decodeString(data),
let result: String = decodedData.value.value
else {
throw DecodingError.dataCorrupted(
DecodingError.Context(
codingPath: codingPath,
debugDescription: "failed to decode String"
)
)
}
return result
}
func decode(_ type: Double.Type) throws -> Double {
let intValue: Int = try decodeFixedInt(Int.self)
return Double(intValue)
}
func decode(_ type: Float.Type) throws -> Float {
let intValue: Int = try decodeFixedInt(Int.self)
return Float(intValue)
}
func decode(_ type: Int.Type) throws -> Int { return try decodeFixedInt(type) }
func decode(_ type: Int8.Type) throws -> Int8 { return try decodeFixedInt(type) }
func decode(_ type: Int16.Type) throws -> Int16 { return try decodeFixedInt(type) }
func decode(_ type: Int32.Type) throws -> Int32 { return try decodeFixedInt(type) }
func decode(_ type: Int64.Type) throws -> Int64 { return try decodeFixedInt(type) }
func decode(_ type: UInt.Type) throws -> UInt { return try decodeFixedInt(type) }
func decode(_ type: UInt8.Type) throws -> UInt8 { return try decodeFixedInt(type) }
func decode(_ type: UInt16.Type) throws -> UInt16 { return try decodeFixedInt(type) }
func decode(_ type: UInt32.Type) throws -> UInt32 { return try decodeFixedInt(type) }
func decode(_ type: UInt64.Type) throws -> UInt64 { return try decodeFixedInt(type) }
func decodeFixedInt<T: FixedWidthInteger>(_ type: T.Type) throws -> T {
var mutableData: Data = data var mutableData: Data = data
var intData: [UInt8] = [] var intData: [UInt8] = []
_ = mutableData.popFirst() // drop `i` _ = mutableData.popFirst() // drop `i`
@ -548,7 +587,7 @@ extension _BencodeDecoder.SingleValueContainer: SingleValueDecodingContainer {
guard guard
let intString: String = String(data: Data(intData), encoding: .ascii), let intString: String = String(data: Data(intData), encoding: .ascii),
let result: Int = Int(intString, radix: 10) let result: T = T(intString, radix: 10)
else { else {
throw DecodingError.dataCorrupted( throw DecodingError.dataCorrupted(
DecodingError.Context( DecodingError.Context(
@ -561,23 +600,18 @@ extension _BencodeDecoder.SingleValueContainer: SingleValueDecodingContainer {
return result return result
} }
func decode(_ type: String.Type) throws -> String { func decode<T>(_ type: T.Type) throws -> T where T: Decodable {
guard if T.self is any FixedWidthInteger.Type {
let decodedData = _BencodeDecoder.decodeString(data), // This will be handled by the integer-specific decode function
let result: String = decodedData.value.value throw DecodingError.typeMismatch(
else { T.self,
throw DecodingError.dataCorrupted(
DecodingError.Context( DecodingError.Context(
codingPath: codingPath, codingPath: codingPath,
debugDescription: "failed to decode String" debugDescription: "attempted to use generic decode function instead of integer-specific one for integer type"
) )
) )
} }
return result
}
func decode<T>(_ type: T.Type) throws -> T where T: Decodable {
let decoder = _BencodeDecoder(data: data, userInfo: userInfo) let decoder = _BencodeDecoder(data: data, userInfo: userInfo)
let value = try T(from: decoder) let value = try T(from: decoder)

@ -99,6 +99,30 @@ class BencodeDecoderSpec: QuickSpec {
try BencodeDecoder().decode(TestType2.self, from: data) try BencodeDecoder().decode(TestType2.self, from: data)
}.to(throwError(DecodingError.typeMismatch(Bool.self, DecodingError.Context(codingPath: [], debugDescription: "Bencode doesn't support Bool values, use an Int and custom Encode/Decode functions isntead")))) }.to(throwError(DecodingError.typeMismatch(Bool.self, DecodingError.Context(codingPath: [], debugDescription: "Bencode doesn't support Bool values, use an Int and custom Encode/Decode functions isntead"))))
} }
// MARK: ---- does not end up in an infinite loop when decoding Int64 types
it("does not end up in an infinite loop when decoding Int64 types") {
let basicIntListData: Data = "li1ei2ee".data(using: .utf8)!
let result = try? BencodeDecoder().decode([Int64].self, from: basicIntListData)
expect(result).to(equal([1, 2]))
}
// MARK: ---- does not end up in an infinite loop when decoding Double types
it("does not end up in an infinite loop when decoding Double types") {
let basicIntListData: Data = "li1ei2ee".data(using: .utf8)!
let result = try? BencodeDecoder().decode([Double].self, from: basicIntListData)
expect(result).to(equal([1, 2]))
}
// MARK: ---- does not end up in an infinite loop when decoding Float types
it("does not end up in an infinite loop when decoding Float types") {
let basicIntListData: Data = "li1ei2ee".data(using: .utf8)!
let result = try? BencodeDecoder().decode([Float].self, from: basicIntListData)
expect(result).to(equal([1, 2]))
}
} }
} }
} }

Loading…
Cancel
Save