2022-02-27 03:07:39 +01:00
|
|
|
import { ApObject, getApIds } from './type.js';
|
|
|
|
import Resolver from './resolver.js';
|
|
|
|
import { resolvePerson } from './models/person.js';
|
|
|
|
import { unique, concat } from '@/prelude/array.js';
|
2022-02-28 17:26:32 +01:00
|
|
|
import promiseLimit from 'promise-limit';
|
2022-03-25 08:27:41 +01:00
|
|
|
import { User, CacheableRemoteUser, CacheableUser } from '@/models/entities/user.js';
|
2020-01-30 10:58:13 +01:00
|
|
|
|
|
|
|
type Visibility = 'public' | 'home' | 'followers' | 'specified';
|
|
|
|
|
|
|
|
type AudienceInfo = {
|
|
|
|
visibility: Visibility,
|
2022-03-25 08:27:41 +01:00
|
|
|
mentionedUsers: CacheableUser[],
|
|
|
|
visibleUsers: CacheableUser[],
|
2020-01-30 10:58:13 +01:00
|
|
|
};
|
|
|
|
|
2022-03-25 08:27:41 +01:00
|
|
|
export async function parseAudience(actor: CacheableRemoteUser, to?: ApObject, cc?: ApObject, resolver?: Resolver): Promise<AudienceInfo> {
|
2020-01-30 10:58:13 +01:00
|
|
|
const toGroups = groupingAudience(getApIds(to), actor);
|
|
|
|
const ccGroups = groupingAudience(getApIds(cc), actor);
|
|
|
|
|
|
|
|
const others = unique(concat([toGroups.other, ccGroups.other]));
|
|
|
|
|
2022-03-25 08:27:41 +01:00
|
|
|
const limit = promiseLimit<CacheableUser | null>(2);
|
2020-01-30 10:58:13 +01:00
|
|
|
const mentionedUsers = (await Promise.all(
|
|
|
|
others.map(id => limit(() => resolvePerson(id, resolver).catch(() => null)))
|
2022-03-25 08:27:41 +01:00
|
|
|
)).filter((x): x is CacheableUser => x != null);
|
2020-01-30 10:58:13 +01:00
|
|
|
|
|
|
|
if (toGroups.public.length > 0) {
|
|
|
|
return {
|
|
|
|
visibility: 'public',
|
|
|
|
mentionedUsers,
|
2021-12-09 15:58:30 +01:00
|
|
|
visibleUsers: [],
|
2020-01-30 10:58:13 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ccGroups.public.length > 0) {
|
|
|
|
return {
|
|
|
|
visibility: 'home',
|
|
|
|
mentionedUsers,
|
2021-12-09 15:58:30 +01:00
|
|
|
visibleUsers: [],
|
2020-01-30 10:58:13 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (toGroups.followers.length > 0) {
|
|
|
|
return {
|
|
|
|
visibility: 'followers',
|
|
|
|
mentionedUsers,
|
2021-12-09 15:58:30 +01:00
|
|
|
visibleUsers: [],
|
2020-01-30 10:58:13 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
visibility: 'specified',
|
|
|
|
mentionedUsers,
|
2021-12-09 15:58:30 +01:00
|
|
|
visibleUsers: mentionedUsers,
|
2020-01-30 10:58:13 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-03-25 08:27:41 +01:00
|
|
|
function groupingAudience(ids: string[], actor: CacheableRemoteUser) {
|
2020-01-30 10:58:13 +01:00
|
|
|
const groups = {
|
|
|
|
public: [] as string[],
|
|
|
|
followers: [] as string[],
|
|
|
|
other: [] as string[],
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const id of ids) {
|
|
|
|
if (isPublic(id)) {
|
|
|
|
groups.public.push(id);
|
|
|
|
} else if (isFollowers(id, actor)) {
|
|
|
|
groups.followers.push(id);
|
|
|
|
} else {
|
|
|
|
groups.other.push(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
groups.other = unique(groups.other);
|
|
|
|
|
|
|
|
return groups;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isPublic(id: string) {
|
|
|
|
return [
|
|
|
|
'https://www.w3.org/ns/activitystreams#Public',
|
|
|
|
'as#Public',
|
|
|
|
'Public',
|
|
|
|
].includes(id);
|
|
|
|
}
|
|
|
|
|
2022-03-25 08:27:41 +01:00
|
|
|
function isFollowers(id: string, actor: CacheableRemoteUser) {
|
2021-02-06 03:50:33 +01:00
|
|
|
return (
|
|
|
|
id === (actor.followersUri || `${actor.uri}/followers`)
|
|
|
|
);
|
2020-01-30 10:58:13 +01:00
|
|
|
}
|