From ba4658d58eddec409dfefbb43b09e8979909228c Mon Sep 17 00:00:00 2001 From: Lawrence Forooghian Date: Thu, 28 Nov 2024 10:00:20 -0300 Subject: [PATCH] Implement CHA-PR3h and similar Resolves #151. --- Sources/AblyChat/Errors.swift | 45 ++++++++----------- Sources/AblyChat/RoomFeature.swift | 7 ++- Sources/AblyChat/RoomLifecycleManager.swift | 9 ++-- .../DefaultRoomLifecycleManagerTests.swift | 45 +++++-------------- 4 files changed, 34 insertions(+), 72 deletions(-) diff --git a/Sources/AblyChat/Errors.swift b/Sources/AblyChat/Errors.swift index b9027c7b..ba036c46 100644 --- a/Sources/AblyChat/Errors.swift +++ b/Sources/AblyChat/Errors.swift @@ -11,8 +11,6 @@ public let errorDomain = "AblyChatErrorDomain" The error codes for errors in the ``errorDomain`` error domain. */ public enum ErrorCode: Int { - case nonspecific = 40000 - /// ``Rooms.get(roomID:options:)`` was called with a different set of room options than was used on a previous call. You must first release the existing room instance using ``Rooms.release(roomID:)``. /// /// TODO this code is a guess, revisit in https://github.com/ably-labs/ably-chat-swift/issues/32 @@ -38,7 +36,6 @@ public enum ErrorCode: Int { /// Has a case for each of the ``ErrorCode`` cases that imply a fixed status code. internal enum CaseThatImpliesFixedStatusCode { - case nonspecific case inconsistentRoomOptions case messagesAttachmentFailed case presenceAttachmentFailed @@ -53,12 +50,9 @@ public enum ErrorCode: Int { case roomInFailedState case roomIsReleasing case roomIsReleased - case roomInInvalidState internal var toNumericErrorCode: ErrorCode { switch self { - case .nonspecific: - .nonspecific case .inconsistentRoomOptions: .inconsistentRoomOptions case .messagesAttachmentFailed: @@ -87,8 +81,6 @@ public enum ErrorCode: Int { .roomIsReleasing case .roomIsReleased: .roomIsReleased - case .roomInInvalidState: - .roomInInvalidState } } @@ -96,8 +88,7 @@ public enum ErrorCode: Int { internal var statusCode: Int { // These status codes are taken from the "Chat-specific Error Codes" section of the spec. switch self { - case .nonspecific, - .inconsistentRoomOptions, + case .inconsistentRoomOptions, .roomInFailedState, .roomIsReleasing, .roomIsReleased: @@ -112,9 +103,7 @@ public enum ErrorCode: Int { .presenceDetachmentFailed, .reactionsDetachmentFailed, .occupancyDetachmentFailed, - .typingDetachmentFailed, - // CHA-RL9c - .roomInInvalidState: + .typingDetachmentFailed: 500 } } @@ -122,8 +111,13 @@ public enum ErrorCode: Int { /// Has a case for each of the ``ErrorCode`` cases that do not imply a fixed status code. internal enum CaseThatImpliesVariableStatusCode { + case roomInInvalidState + internal var toNumericErrorCode: ErrorCode { - switch self {} + switch self { + case .roomInInvalidState: + .roomInInvalidState + } } } } @@ -169,8 +163,7 @@ internal enum ChatError { case roomIsReleasing case roomIsReleased case presenceOperationRequiresRoomAttach(feature: RoomFeature) - case presenceOperationDisallowedForCurrentRoomStatus(feature: RoomFeature) - case roomInInvalidState(cause: ARTErrorInfo?) + case roomTransitionedToInvalidStateForPresenceOperation(cause: ARTErrorInfo?) internal var codeAndStatusCode: ErrorCodeAndStatusCode { switch self { @@ -208,11 +201,12 @@ internal enum ChatError { .fixedStatusCode(.roomIsReleasing) case .roomIsReleased: .fixedStatusCode(.roomIsReleased) - case .roomInInvalidState: - .fixedStatusCode(.roomInInvalidState) - case .presenceOperationRequiresRoomAttach, - .presenceOperationDisallowedForCurrentRoomStatus: - .fixedStatusCode(.nonspecific) + case .roomTransitionedToInvalidStateForPresenceOperation: + // CHA-RL9c + .variableStatusCode(.roomInInvalidState, statusCode: 500) + case .presenceOperationRequiresRoomAttach: + // CHA-PR3h, CHA-PR10h, CHA-PR6h, CHA-T2g + .variableStatusCode(.roomInInvalidState, statusCode: 400) } } @@ -268,9 +262,7 @@ internal enum ChatError { "Cannot perform operation because the room is in a released state." case let .presenceOperationRequiresRoomAttach(feature): "To perform this \(Self.descriptionOfFeature(feature)) operation, you must first attach the room." - case let .presenceOperationDisallowedForCurrentRoomStatus(feature): - "This \(Self.descriptionOfFeature(feature)) operation can not be performed given the current room status." - case .roomInInvalidState: + case .roomTransitionedToInvalidStateForPresenceOperation: "The room operation failed because the room was in an invalid state." } } @@ -282,14 +274,13 @@ internal enum ChatError { underlyingError case let .detachmentFailed(_, underlyingError): underlyingError - case let .roomInInvalidState(cause): + case let .roomTransitionedToInvalidStateForPresenceOperation(cause): cause case .inconsistentRoomOptions, .roomInFailedState, .roomIsReleasing, .roomIsReleased, - .presenceOperationRequiresRoomAttach, - .presenceOperationDisallowedForCurrentRoomStatus: + .presenceOperationRequiresRoomAttach: nil } } diff --git a/Sources/AblyChat/RoomFeature.swift b/Sources/AblyChat/RoomFeature.swift index 0aee6b7a..5383b8ff 100644 --- a/Sources/AblyChat/RoomFeature.swift +++ b/Sources/AblyChat/RoomFeature.swift @@ -41,12 +41,11 @@ internal protocol FeatureChannel: Sendable, EmitsDiscontinuities { /// Waits until we can perform presence operations on the contributors of this room without triggering an implicit attach. /// - /// Implements the checks described by CHA-PR3d, CHA-PR3e, CHA-PR3f, and CHA-PR3g (and similar ones described by other functionality that performs contributor presence operations). Namely: + /// Implements the checks described by CHA-PR3d, CHA-PR3e, and CHA-PR3h (and similar ones described by other functionality that performs contributor presence operations). Namely: /// - /// - CHA-RL9, which is invoked by CHA-PR3d, CHA-PR10d, CHA-PR6c, CHA-T2c: If the room is in the ATTACHING status, it waits for the next room status change. If the new status is ATTACHED, it returns. Else, it throws an `ARTErrorInfo` derived from ``ChatError.roomInInvalidState(cause:)``. + /// - CHA-RL9, which is invoked by CHA-PR3d, CHA-PR10d, CHA-PR6c, CHA-T2c: If the room is in the ATTACHING status, it waits for the next room status change. If the new status is ATTACHED, it returns. Else, it throws an `ARTErrorInfo` derived from ``ChatError.roomTransitionedToInvalidStateForPresenceOperation(cause:)``. /// - CHA-PR3e, CHA-PR10e, CHA-PR6d, CHA-T2d: If the room is in the ATTACHED status, it returns immediately. - /// - CHA-PR3f, CHA-PR10f, CHA-PR6e, CHA-T2e: If the room is in the DETACHED status, it throws an `ARTErrorInfo` derived from ``ChatError.presenceOperationRequiresRoomAttach(feature:)``. - /// - // CHA-PR3g, CHA-PR10g, CHA-PR6f, CHA-T2f: If the room is in any other status, it throws an `ARTErrorInfo` derived from ``ChatError.presenceOperationDisallowedForCurrentRoomStatus(feature:)``. + /// - CHA-PR3h, CHA-PR10h, CHA-PR6h, CHA-T2g: If the room is in any other status, it throws an `ARTErrorInfo` derived from ``ChatError.presenceOperationRequiresRoomAttach(feature:)``. /// /// - Parameters: /// - requester: The room feature that wishes to perform a presence operation. This is only used for customising the message of the thrown error. diff --git a/Sources/AblyChat/RoomLifecycleManager.swift b/Sources/AblyChat/RoomLifecycleManager.swift index 472e4e6d..8c61522e 100644 --- a/Sources/AblyChat/RoomLifecycleManager.swift +++ b/Sources/AblyChat/RoomLifecycleManager.swift @@ -1225,17 +1225,14 @@ internal actor DefaultRoomLifecycleManager