signal-desktop/ts/sql/server/groupEndorsements.ts

90 lines
2.9 KiB
TypeScript
Raw Normal View History

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);
}