// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
//
// stringlint:disable

import Foundation
import SessionUtilitiesKit

extension SnodeAPI {
    public class UpdateExpiryRequest: SnodeAuthenticatedRequestBody {
        enum CodingKeys: String, CodingKey {
            case messageHashes = "messages"
            case expiryMs = "expiry"
            case shorten
            case extend
        }
        
        /// Array of message hash strings (as provided by the storage server) to update. Messages can be from any namespace(s)
        let messageHashes: [String]
        
        /// The new expiry timestamp (milliseconds since unix epoch).  Must be >= 60s ago.  The new expiry can be anywhere from
        /// current time up to the maximum TTL (30 days) from now; specifying a later timestamp will be truncated to the maximum
        let expiryMs: UInt64
        
        /// If provided and set to true then the expiry is only shortened, but not extended. If the expiry is already at or before the given
        /// `expiry` timestamp then expiry will not be changed
        ///
        /// **Note:** This option is only supported starting at network version 19.3).  This option is not permitted when using
        /// subaccount authentication
        let shorten: Bool?
        
        /// If provided and set to true then the expiry is only extended, but not shortened.  If the expiry is already at or beyond
        /// the given `expiry` timestamp then expiry will not be changed
        ///
        /// **Note:** This option is only supported starting at network version 19.3.  This option is mutually exclusive of "shorten"
        let extend: Bool?
        
        override var verificationBytes: [UInt8] {
            /// Ed25519 signature of `("expire" || ShortenOrExtend || expiry || messages[0] || ...`
            /// ` || messages[N])` where `expiry` is the expiry timestamp expressed as a string.
            /// `ShortenOrExtend` is string signature must be base64 "shorten" if the shorten option is given (and true),
            /// "extend" if `extend` is true, and empty otherwise. The signature must be base64 encoded (json) or bytes (bt).
            SnodeAPI.Endpoint.expire.path.bytes
                .appending(contentsOf: (shorten == true ? "shorten".bytes : []))
                .appending(contentsOf: (extend == true ? "extend".bytes : []))
                .appending(contentsOf: "\(expiryMs)".data(using: .ascii)?.bytes)
                .appending(contentsOf: messageHashes.joined().bytes)
        }
        
        // MARK: - Init
        
        public init(
            messageHashes: [String],
            expiryMs: UInt64,
            shorten: Bool? = nil,
            extend: Bool? = nil,
            authMethod: AuthenticationMethod
        ) {
            self.messageHashes = messageHashes
            self.expiryMs = expiryMs
            self.shorten = shorten
            self.extend = extend
            
            super.init(authMethod: authMethod)
        }
        
        // MARK: - Coding
        
        override public func encode(to encoder: Encoder) throws {
            var container: KeyedEncodingContainer<CodingKeys> = encoder.container(keyedBy: CodingKeys.self)
            
            try container.encode(messageHashes, forKey: .messageHashes)
            try container.encode(expiryMs, forKey: .expiryMs)
            try container.encodeIfPresent(shorten, forKey: .shorten)
            try container.encodeIfPresent(extend, forKey: .extend)
            
            try super.encode(to: encoder)
        }
    }
}