2024-05-20 18:15:39 +00:00
|
|
|
// Copyright 2024 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import type {
|
|
|
|
GroupSendCombinedEndorsementRecord,
|
|
|
|
GroupSendEndorsementsData,
|
|
|
|
GroupSendMemberEndorsementRecord,
|
|
|
|
} from '../../types/GroupSendEndorsements';
|
|
|
|
import { groupSendEndorsementExpirationSchema } from '../../types/GroupSendEndorsements';
|
2024-07-22 18:16:33 +00:00
|
|
|
import { prepare } from '../Server';
|
|
|
|
import type { ReadableDB, WritableDB } from '../Interface';
|
2024-05-20 18:15:39 +00:00
|
|
|
import { sql } from '../util';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* We don't need to store more than one endorsement per group or per member.
|
|
|
|
*/
|
2024-07-22 18:16:33 +00:00
|
|
|
export function replaceAllEndorsementsForGroup(
|
|
|
|
db: WritableDB,
|
2024-05-20 18:15:39 +00:00
|
|
|
data: GroupSendEndorsementsData
|
2024-07-22 18:16:33 +00:00
|
|
|
): void {
|
2024-05-20 18:15:39 +00:00
|
|
|
db.transaction(() => {
|
|
|
|
const { combinedEndorsement, memberEndorsements } = data;
|
|
|
|
_replaceCombinedEndorsement(db, combinedEndorsement);
|
|
|
|
_replaceMemberEndorsements(db, memberEndorsements);
|
|
|
|
})();
|
|
|
|
}
|
|
|
|
|
|
|
|
function _replaceCombinedEndorsement(
|
2024-07-22 18:16:33 +00:00
|
|
|
db: WritableDB,
|
2024-05-20 18:15:39 +00:00
|
|
|
combinedEndorsement: GroupSendCombinedEndorsementRecord
|
|
|
|
): void {
|
|
|
|
const { groupId, expiration, endorsement } = combinedEndorsement;
|
|
|
|
const [insertCombined, insertCombinedParams] = sql`
|
|
|
|
INSERT OR REPLACE INTO groupSendCombinedEndorsement
|
|
|
|
(groupId, expiration, endorsement)
|
|
|
|
VALUES (${groupId}, ${expiration}, ${endorsement});
|
|
|
|
`;
|
|
|
|
prepare<Array<unknown>>(db, insertCombined).run(insertCombinedParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
function _replaceMemberEndorsements(
|
2024-07-22 18:16:33 +00:00
|
|
|
db: WritableDB,
|
2024-05-20 18:15:39 +00:00
|
|
|
memberEndorsements: ReadonlyArray<GroupSendMemberEndorsementRecord>
|
|
|
|
) {
|
|
|
|
for (const memberEndorsement of memberEndorsements) {
|
|
|
|
const { groupId, memberAci, expiration, endorsement } = memberEndorsement;
|
|
|
|
const [replaceMember, replaceMemberParams] = sql`
|
|
|
|
INSERT OR REPLACE INTO groupSendMemberEndorsement
|
|
|
|
(groupId, memberAci, expiration, endorsement)
|
|
|
|
VALUES (${groupId}, ${memberAci}, ${expiration}, ${endorsement});
|
|
|
|
`;
|
|
|
|
prepare<Array<unknown>>(db, replaceMember).run(replaceMemberParams);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-22 18:16:33 +00:00
|
|
|
export function deleteAllEndorsementsForGroup(
|
|
|
|
db: WritableDB,
|
2024-05-20 18:15:39 +00:00
|
|
|
groupId: string
|
2024-07-22 18:16:33 +00:00
|
|
|
): void {
|
2024-05-20 18:15:39 +00:00
|
|
|
db.transaction(() => {
|
|
|
|
const [deleteCombined, deleteCombinedParams] = sql`
|
|
|
|
DELETE FROM groupSendCombinedEndorsement
|
|
|
|
WHERE groupId = ${groupId};
|
|
|
|
`;
|
|
|
|
const [deleteMembers, deleteMembersParams] = sql`
|
|
|
|
DELETE FROM groupSendMemberEndorsement
|
|
|
|
WHERE groupId = ${groupId};
|
|
|
|
`;
|
|
|
|
prepare<Array<unknown>>(db, deleteCombined).run(deleteCombinedParams);
|
|
|
|
prepare<Array<unknown>>(db, deleteMembers).run(deleteMembersParams);
|
|
|
|
})();
|
|
|
|
}
|
|
|
|
|
2024-07-22 18:16:33 +00:00
|
|
|
export function getGroupSendCombinedEndorsementExpiration(
|
|
|
|
db: ReadableDB,
|
2024-05-20 18:15:39 +00:00
|
|
|
groupId: string
|
2024-07-22 18:16:33 +00:00
|
|
|
): number | null {
|
2024-05-20 18:15:39 +00:00
|
|
|
const [selectGroup, selectGroupParams] = sql`
|
|
|
|
SELECT expiration FROM groupSendCombinedEndorsement
|
|
|
|
WHERE groupId = ${groupId};
|
|
|
|
`;
|
2024-05-22 16:24:27 +00:00
|
|
|
const value = prepare<Array<unknown>>(db, selectGroup)
|
|
|
|
.pluck()
|
|
|
|
.get(selectGroupParams);
|
2024-05-20 18:15:39 +00:00
|
|
|
if (value == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return groupSendEndorsementExpirationSchema.parse(value);
|
|
|
|
}
|