TM-SGNL-iOS/SignalServiceKit/Groups/StorageService+GroupsV2.swift
TeleMessage developers dde0620daf initial commit
2025-05-03 12:28:28 -07:00

187 lines
6.6 KiB
Swift

//
// Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
//
import Foundation
public import LibSignalClient
public struct GroupsV2Request {
let urlString: String
let method: HTTPMethod
let bodyData: Data?
let headers = OWSHttpHeaders()
func addHeader(_ header: String, value: String) {
headers.addHeader(header, value: value, overwriteOnConflict: true)
}
}
// MARK: -
public extension StorageService {
static func buildNewGroupRequest(
groupProto: GroupsProtoGroup,
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni
) throws -> GroupsV2Request {
let protoData = try groupProto.serializedData()
return try buildGroupV2Request(
protoData: protoData,
urlString: "v2/groups",
method: .put,
groupV2Params: groupV2Params,
authCredential: authCredential
)
}
static func buildUpdateGroupRequest(
groupChangeProto: GroupsProtoGroupChangeActions,
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni,
groupInviteLinkPassword: Data?
) throws -> GroupsV2Request {
var urlString = "v2/groups"
if let groupInviteLinkPassword {
urlString += "?inviteLinkPassword=\(groupInviteLinkPassword.asBase64Url)"
}
let protoData = try groupChangeProto.serializedData()
return try buildGroupV2Request(
protoData: protoData,
urlString: urlString,
method: .patch,
groupV2Params: groupV2Params,
authCredential: authCredential
)
}
static func buildFetchCurrentGroupV2SnapshotRequest(
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni
) throws -> GroupsV2Request {
return try buildGroupV2Request(
protoData: nil,
urlString: "v2/groups",
method: .get,
groupV2Params: groupV2Params,
authCredential: authCredential
)
}
static func buildFetchGroupChangeActionsRequest(
groupV2Params: GroupV2Params,
fromRevision: UInt32,
requireSnapshotForFirstChange: Bool,
authCredential: AuthCredentialWithPni
) throws -> GroupsV2Request {
let urlPath = "/v1/groups/logs/\(fromRevision)?includeFirstState=\(requireSnapshotForFirstChange)&maxSupportedChangeEpoch=\(GroupManager.changeProtoEpoch)"
return try buildGroupV2Request(
protoData: nil,
urlString: urlPath,
method: .get,
groupV2Params: groupV2Params,
authCredential: authCredential
)
}
static func buildGetJoinedAtRevisionRequest(
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni
) throws -> GroupsV2Request {
return try buildGroupV2Request(
protoData: nil,
urlString: "/v1/groups/joined_at_version/",
method: .get,
groupV2Params: groupV2Params,
authCredential: authCredential
)
}
static func buildGroupAvatarUploadFormRequest(groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni) throws -> GroupsV2Request {
let urlPath = "/v1/groups/avatar/form"
return try buildGroupV2Request(protoData: nil,
urlString: urlPath,
method: .get,
groupV2Params: groupV2Params,
authCredential: authCredential)
}
// inviteLinkPassword is not necessary if we're already a member or have a pending request.
static func buildFetchGroupInviteLinkPreviewRequest(inviteLinkPassword: Data?,
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni) throws -> GroupsV2Request {
var urlPath = "/v1/groups/join/"
if let inviteLinkPassword = inviteLinkPassword {
urlPath += "\(inviteLinkPassword.asBase64Url)"
}
return try buildGroupV2Request(protoData: nil,
urlString: urlPath,
method: .get,
groupV2Params: groupV2Params,
authCredential: authCredential)
}
static func buildFetchGroupExternalCredentials(
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni
) throws -> GroupsV2Request {
return try buildGroupV2Request(
protoData: nil,
urlString: "/v1/groups/token",
method: .get,
groupV2Params: groupV2Params,
authCredential: authCredential
)
}
private static func buildGroupV2Request(
protoData: Data?,
urlString: String,
method: HTTPMethod,
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni
) throws -> GroupsV2Request {
let request = GroupsV2Request(urlString: urlString, method: method, bodyData: protoData)
// The censorship circumvention reflectors require a Content-Type
// even if the body is empty.
request.addHeader("Content-Type", value: MimeType.applicationXProtobuf.rawValue)
try self.addAuthorizationHeader(
to: request,
groupV2Params: groupV2Params,
authCredential: authCredential
)
return request
}
// MARK: - Authorization Headers
private static func addAuthorizationHeader(
to request: GroupsV2Request,
groupV2Params: GroupV2Params,
authCredential: AuthCredentialWithPni
) throws {
let serverPublicParams = GroupsV2Protos.serverPublicParams()
let clientZkAuthOperations = ClientZkAuthOperations(serverPublicParams: serverPublicParams)
let authCredentialPresentation = try clientZkAuthOperations.createAuthCredentialPresentation(groupSecretParams: groupV2Params.groupSecretParams, authCredential: authCredential)
let authCredentialPresentationData = authCredentialPresentation.serialize().asData
let username: String = groupV2Params.groupPublicParamsData.hexadecimalString
let password: String = authCredentialPresentationData.hexadecimalString
request.addHeader(
OWSHttpHeaders.authHeaderKey,
value: try OWSHttpHeaders.authHeaderValue(username: username, password: password)
)
}
}