Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
signalapp
GitHub Repository: signalapp/Signal-iOS
Path: blob/main/SignalServiceKit/Devices/SyncMessages/LinkedDeviceReadReceipt.swift
1 views
//
// Copyright 2026 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
//

import Foundation
import LibSignalClient

@objc(OWSLinkedDeviceReadReceipt)
final class LinkedDeviceReadReceipt: NSObject, NSSecureCoding {

    let messageUniqueId: String? // Only nil if decoding old values
    let messageIdTimestamp: UInt64
    let readTimestamp: UInt64

    let senderPhoneNumber: String?
    let senderAci: Aci?

    init(
        senderAci: Aci,
        messageUniqueId: String?,
        messageIdTimestamp: UInt64,
        readTimestamp: UInt64,
    ) {
        owsAssertDebug(messageIdTimestamp > 0)
        self.senderPhoneNumber = nil
        self.senderAci = senderAci
        self.messageUniqueId = messageUniqueId
        self.messageIdTimestamp = messageIdTimestamp
        self.readTimestamp = readTimestamp
    }

    static var supportsSecureCoding: Bool { true }

    func encode(with coder: NSCoder) {
        coder.encode(NSNumber(value: 1), forKey: "linkedDeviceReadReceiptSchemaVersion")
        coder.encode(NSNumber(value: self.messageIdTimestamp), forKey: "messageIdTimestamp")
        if let messageUniqueId {
            coder.encode(messageUniqueId, forKey: "messageUniqueId")
        }
        coder.encode(NSNumber(value: self.readTimestamp), forKey: "readTimestamp")
        if let senderPhoneNumber {
            coder.encode(senderPhoneNumber, forKey: "senderPhoneNumber")
        }
        if let senderAci {
            coder.encode(senderAci.serviceIdUppercaseString, forKey: "senderUUID")
        }
    }

    init?(coder: NSCoder) {
        let schemaVersion = coder.decodeObject(of: NSNumber.self, forKey: "linkedDeviceReadReceiptSchemaVersion")?.uintValue ?? 0
        let messageUniqueId = coder.decodeObject(of: NSString.self, forKey: "messageUniqueId") as String?

        let senderAciString = coder.decodeObject(of: NSString.self, forKey: "senderUUID") as String?
        if let senderAciString {
            guard let senderAci = Aci.parseFrom(aciString: senderAciString) else {
                return nil
            }
            self.senderAci = senderAci
        } else {
            self.senderAci = nil
        }

        // renamed timestamp -> messageIdTimestamp
        let messageIdTimestamp = coder.decodeObject(of: NSNumber.self, forKey: "messageIdTimestamp") ?? coder.decodeObject(of: NSNumber.self, forKey: "timestamp")
        guard let messageIdTimestamp else {
            return nil
        }

        // For legacy objects, before we were tracking read time, use the original messages "sent" timestamp
        // as the local read time. This will always be at least a little bit earlier than the message was
        // actually read, which isn't ideal, but safer than persisting a disappearing message too long, especially
        // since we know they read it on their linked desktop.
        let readTimestamp = coder.decodeObject(of: NSNumber.self, forKey: "readTimestamp") ?? messageIdTimestamp

        let senderPhoneNumber: String?
        if schemaVersion < 1 {
            senderPhoneNumber = coder.decodeObject(of: NSString.self, forKey: "senderId") as String?
            owsAssertDebug(senderPhoneNumber != nil)
        } else {
            senderPhoneNumber = coder.decodeObject(of: NSString.self, forKey: "senderPhoneNumber") as String?
        }

        self.messageUniqueId = messageUniqueId
        self.messageIdTimestamp = messageIdTimestamp.uint64Value
        self.readTimestamp = readTimestamp.uint64Value
        self.senderPhoneNumber = senderPhoneNumber
    }

    override var hash: Int {
        var hasher = Hasher()
        hasher.combine(self.messageIdTimestamp)
        hasher.combine(self.messageUniqueId)
        hasher.combine(self.readTimestamp)
        hasher.combine(self.senderPhoneNumber)
        hasher.combine(self.senderAci)
        return hasher.finalize()
    }

    override func isEqual(_ object: Any?) -> Bool {
        guard let object = object as? Self else { return false }
        guard self.messageIdTimestamp == object.messageIdTimestamp else { return false }
        guard self.messageUniqueId == object.messageUniqueId else { return false }
        guard self.readTimestamp == object.readTimestamp else { return false }
        guard self.senderPhoneNumber == object.senderPhoneNumber else { return false }
        guard self.senderAci == object.senderAci else { return false }
        return true
    }

    var senderAddress: SignalServiceAddress {
        return SignalServiceAddress.legacyAddress(serviceId: self.senderAci, phoneNumber: self.senderPhoneNumber)
    }
}