matrix-react-sdk
Version:
SDK for matrix.org using React
93 lines (90 loc) • 17.7 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.buildActivityScores = buildActivityScores;
exports.buildMemberScores = buildMemberScores;
exports.compareMembers = void 0;
var _lodash = require("lodash");
var _types = require("matrix-js-sdk/src/types");
var _DMRoomMap = _interopRequireDefault(require("./DMRoomMap"));
/*
Copyright 2024 New Vector Ltd.
Copyright 2022 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const compareMembers = (activityScores, memberScores) => (a, b) => {
const aActivityScore = activityScores[a.userId]?.score ?? 0;
const aMemberScore = memberScores[a.userId]?.score ?? 0;
const aScore = aActivityScore + aMemberScore;
const aNumRooms = memberScores[a.userId]?.numRooms ?? 0;
const bActivityScore = activityScores[b.userId]?.score ?? 0;
const bMemberScore = memberScores[b.userId]?.score ?? 0;
const bScore = bActivityScore + bMemberScore;
const bNumRooms = memberScores[b.userId]?.numRooms ?? 0;
if (aScore === bScore) {
if (aNumRooms === bNumRooms) {
// If there is no activity between members,
// keep the order received from the user directory search results
return 0;
}
return bNumRooms - aNumRooms;
}
return bScore - aScore;
};
exports.compareMembers = compareMembers;
function joinedRooms(cli) {
return cli.getRooms().filter(r => r.getMyMembership() === _types.KnownMembership.Join)
// Skip low priority rooms and DMs
.filter(r => !_DMRoomMap.default.shared().getUserIdForRoomId(r.roomId)).filter(r => !Object.keys(r.tags).includes("m.lowpriority"));
}
// Score people based on who have sent messages recently, as a way to improve the quality of suggestions.
// We do this by checking every room to see who has sent a message in the last few hours, and giving them
// a score which correlates to the freshness of their message. In theory, this results in suggestions
// which are closer to "continue this conversation" rather than "this person exists".
function buildActivityScores(cli) {
const now = new Date().getTime();
const earliestAgeConsidered = now - 60 * 60 * 1000; // 1 hour ago
const maxMessagesConsidered = 50; // so we don't iterate over a huge amount of traffic
const events = joinedRooms(cli).flatMap(room => (0, _lodash.takeRight)(room.getLiveTimeline().getEvents(), maxMessagesConsidered)).filter(ev => ev.getTs() > earliestAgeConsidered);
const senderEvents = (0, _lodash.groupBy)(events, ev => ev.getSender());
// If the iteratee in mapValues returns undefined that key will be removed from the resultant object
return (0, _lodash.mapValues)(senderEvents, events => {
if (!events.length) return;
const lastEvent = (0, _lodash.maxBy)(events, ev => ev.getTs());
const distanceFromNow = Math.abs(now - lastEvent.getTs()); // abs to account for slight future messages
const inverseTime = now - earliestAgeConsidered - distanceFromNow;
return {
lastSpoke: lastEvent.getTs(),
// Scores from being in a room give a 'good' score of about 1.0-1.5, so for our
// score we'll try and award at least 1.0 for making the list, with 4.0 being
// an approximate maximum for being selected.
score: Math.max(1, inverseTime / (15 * 60 * 1000)) // 15min segments to keep scores sane
};
});
}
function buildMemberScores(cli) {
const maxConsideredMembers = 200;
const consideredRooms = joinedRooms(cli).filter(room => room.getJoinedMemberCount() < maxConsideredMembers);
const memberPeerEntries = consideredRooms.flatMap(room => room.getJoinedMembers().map(member => ({
member,
roomSize: room.getJoinedMemberCount()
})));
const userMeta = (0, _lodash.groupBy)(memberPeerEntries, ({
member
}) => member.userId);
// If the iteratee in mapValues returns undefined that key will be removed from the resultant object
return (0, _lodash.mapValues)(userMeta, roomMemberships => {
if (!roomMemberships.length) return;
const maximumPeers = maxConsideredMembers * roomMemberships.length;
const totalPeers = (0, _lodash.sumBy)(roomMemberships, entry => entry.roomSize);
return {
member: (0, _lodash.minBy)(roomMemberships, entry => entry.roomSize).member,
numRooms: roomMemberships.length,
score: Math.max(0, Math.pow(1 - totalPeers / maximumPeers, 5))
};
});
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","require","_types","_DMRoomMap","_interopRequireDefault","compareMembers","activityScores","memberScores","a","b","aActivityScore","userId","score","aMemberScore","aScore","aNumRooms","numRooms","bActivityScore","bMemberScore","bScore","bNumRooms","exports","joinedRooms","cli","getRooms","filter","r","getMyMembership","KnownMembership","Join","DMRoomMap","shared","getUserIdForRoomId","roomId","Object","keys","tags","includes","buildActivityScores","now","Date","getTime","earliestAgeConsidered","maxMessagesConsidered","events","flatMap","room","takeRight","getLiveTimeline","getEvents","ev","getTs","senderEvents","groupBy","getSender","mapValues","length","lastEvent","maxBy","distanceFromNow","Math","abs","inverseTime","lastSpoke","max","buildMemberScores","maxConsideredMembers","consideredRooms","getJoinedMemberCount","memberPeerEntries","getJoinedMembers","map","member","roomSize","userMeta","roomMemberships","maximumPeers","totalPeers","sumBy","entry","minBy","pow"],"sources":["../../src/utils/SortMembers.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2022 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { groupBy, mapValues, maxBy, minBy, sumBy, takeRight } from \"lodash\";\nimport { MatrixClient, Room, RoomMember } from \"matrix-js-sdk/src/matrix\";\nimport { KnownMembership } from \"matrix-js-sdk/src/types\";\n\nimport { Member } from \"./direct-messages\";\nimport DMRoomMap from \"./DMRoomMap\";\n\nexport const compareMembers =\n    (\n        activityScores: Record<string, IActivityScore | undefined>,\n        memberScores: Record<string, IMemberScore | undefined>,\n    ) =>\n    (a: Member | RoomMember, b: Member | RoomMember): number => {\n        const aActivityScore = activityScores[a.userId]?.score ?? 0;\n        const aMemberScore = memberScores[a.userId]?.score ?? 0;\n        const aScore = aActivityScore + aMemberScore;\n        const aNumRooms = memberScores[a.userId]?.numRooms ?? 0;\n\n        const bActivityScore = activityScores[b.userId]?.score ?? 0;\n        const bMemberScore = memberScores[b.userId]?.score ?? 0;\n        const bScore = bActivityScore + bMemberScore;\n        const bNumRooms = memberScores[b.userId]?.numRooms ?? 0;\n\n        if (aScore === bScore) {\n            if (aNumRooms === bNumRooms) {\n                // If there is no activity between members,\n                // keep the order received from the user directory search results\n                return 0;\n            }\n\n            return bNumRooms - aNumRooms;\n        }\n        return bScore - aScore;\n    };\n\nfunction joinedRooms(cli: MatrixClient): Room[] {\n    return (\n        cli\n            .getRooms()\n            .filter((r) => r.getMyMembership() === KnownMembership.Join)\n            // Skip low priority rooms and DMs\n            .filter((r) => !DMRoomMap.shared().getUserIdForRoomId(r.roomId))\n            .filter((r) => !Object.keys(r.tags).includes(\"m.lowpriority\"))\n    );\n}\n\ninterface IActivityScore {\n    lastSpoke: number;\n    score: number;\n}\n\n// Score people based on who have sent messages recently, as a way to improve the quality of suggestions.\n// We do this by checking every room to see who has sent a message in the last few hours, and giving them\n// a score which correlates to the freshness of their message. In theory, this results in suggestions\n// which are closer to \"continue this conversation\" rather than \"this person exists\".\nexport function buildActivityScores(cli: MatrixClient): { [userId: string]: IActivityScore } {\n    const now = new Date().getTime();\n    const earliestAgeConsidered = now - 60 * 60 * 1000; // 1 hour ago\n    const maxMessagesConsidered = 50; // so we don't iterate over a huge amount of traffic\n    const events = joinedRooms(cli)\n        .flatMap((room) => takeRight(room.getLiveTimeline().getEvents(), maxMessagesConsidered))\n        .filter((ev) => ev.getTs() > earliestAgeConsidered);\n    const senderEvents = groupBy(events, (ev) => ev.getSender());\n    // If the iteratee in mapValues returns undefined that key will be removed from the resultant object\n    return mapValues(senderEvents, (events) => {\n        if (!events.length) return;\n        const lastEvent = maxBy(events, (ev) => ev.getTs())!;\n        const distanceFromNow = Math.abs(now - lastEvent.getTs()); // abs to account for slight future messages\n        const inverseTime = now - earliestAgeConsidered - distanceFromNow;\n        return {\n            lastSpoke: lastEvent.getTs(),\n            // Scores from being in a room give a 'good' score of about 1.0-1.5, so for our\n            // score we'll try and award at least 1.0 for making the list, with 4.0 being\n            // an approximate maximum for being selected.\n            score: Math.max(1, inverseTime / (15 * 60 * 1000)), // 15min segments to keep scores sane\n        };\n    }) as { [key: string]: IActivityScore };\n}\n\ninterface IMemberScore {\n    member: RoomMember;\n    score: number;\n    numRooms: number;\n}\n\nexport function buildMemberScores(cli: MatrixClient): { [userId: string]: IMemberScore } {\n    const maxConsideredMembers = 200;\n    const consideredRooms = joinedRooms(cli).filter((room) => room.getJoinedMemberCount() < maxConsideredMembers);\n    const memberPeerEntries = consideredRooms.flatMap((room) =>\n        room.getJoinedMembers().map((member) => ({ member, roomSize: room.getJoinedMemberCount() })),\n    );\n    const userMeta = groupBy(memberPeerEntries, ({ member }) => member.userId);\n    // If the iteratee in mapValues returns undefined that key will be removed from the resultant object\n    return mapValues(userMeta, (roomMemberships) => {\n        if (!roomMemberships.length) return;\n        const maximumPeers = maxConsideredMembers * roomMemberships.length;\n        const totalPeers = sumBy(roomMemberships, (entry) => entry.roomSize);\n        return {\n            member: minBy(roomMemberships, (entry) => entry.roomSize)!.member,\n            numRooms: roomMemberships.length,\n            score: Math.max(0, Math.pow(1 - totalPeers / maximumPeers, 5)),\n        };\n    }) as { [userId: string]: IMemberScore };\n}\n"],"mappings":"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","ignoreList":[]}