2023-01-13 04:40:33 +00:00
|
|
|
import { Brackets } from "typeorm";
|
|
|
|
import { fetchMeta } from "@/misc/fetch-meta.js";
|
|
|
|
import { Instances } from "@/models/index.js";
|
|
|
|
import type { Instance } from "@/models/entities/instance.js";
|
|
|
|
import { DAY } from "@/const.js";
|
|
|
|
import { shouldBlockInstance } from "./should-block-instance.js";
|
2022-11-09 17:47:28 +00:00
|
|
|
|
|
|
|
// Threshold from last contact after which an instance will be considered
|
|
|
|
// "dead" and should no longer get activities delivered to it.
|
|
|
|
const deadThreshold = 7 * DAY;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the subset of hosts which should be skipped.
|
2023-01-13 04:40:33 +00:00
|
|
|
*
|
2022-11-09 17:47:28 +00:00
|
|
|
* @param hosts array of punycoded instance hosts
|
|
|
|
* @returns array of punycoed instance hosts that should be skipped (subset of hosts parameter)
|
|
|
|
*/
|
2023-01-13 04:40:33 +00:00
|
|
|
export async function skippedInstances(
|
|
|
|
hosts: Instance["host"][],
|
|
|
|
): Promise<Instance["host"][]> {
|
2022-11-09 17:47:28 +00:00
|
|
|
// first check for blocked instances since that info may already be in memory
|
2022-12-24 19:39:54 +00:00
|
|
|
const meta = await fetchMeta();
|
2023-01-13 04:40:33 +00:00
|
|
|
const shouldSkip = await Promise.all(
|
|
|
|
hosts.map((host) => shouldBlockInstance(host, meta)),
|
|
|
|
);
|
2022-12-24 19:39:54 +00:00
|
|
|
const skipped = hosts.filter((_, i) => shouldSkip[i]);
|
2023-01-13 04:40:33 +00:00
|
|
|
|
2022-11-09 17:47:28 +00:00
|
|
|
// if possible return early and skip accessing the database
|
2023-01-13 04:40:33 +00:00
|
|
|
if (skipped.length === hosts.length) return hosts;
|
2022-11-09 17:47:28 +00:00
|
|
|
|
|
|
|
const deadTime = new Date(Date.now() - deadThreshold);
|
|
|
|
|
|
|
|
return skipped.concat(
|
2023-01-13 04:40:33 +00:00
|
|
|
await Instances.createQueryBuilder("instance")
|
|
|
|
.where("instance.host in (:...hosts)", {
|
2022-11-09 17:47:28 +00:00
|
|
|
// don't check hosts again that we already know are suspended
|
|
|
|
// also avoids adding duplicates to the list
|
2023-01-13 04:40:33 +00:00
|
|
|
hosts: hosts.filter((host) => !skipped.includes(host)),
|
2022-11-09 17:47:28 +00:00
|
|
|
})
|
2023-01-13 04:40:33 +00:00
|
|
|
.andWhere(
|
|
|
|
new Brackets((qb) => {
|
|
|
|
qb.where("instance.isSuspended");
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
.select("host")
|
2023-01-03 17:12:26 +00:00
|
|
|
.getRawMany(),
|
2022-11-09 17:47:28 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether a specific host (punycoded) should be skipped.
|
|
|
|
* Convenience wrapper around skippedInstances which should only be used if there is a single host to check.
|
|
|
|
* If you have multiple hosts, consider using skippedInstances instead to do a bulk check.
|
|
|
|
*
|
|
|
|
* @param host punycoded instance host
|
|
|
|
* @returns whether the given host should be skipped
|
|
|
|
*/
|
2023-01-13 04:40:33 +00:00
|
|
|
export async function shouldSkipInstance(
|
|
|
|
host: Instance["host"],
|
|
|
|
): Promise<boolean> {
|
2022-11-09 17:47:28 +00:00
|
|
|
const skipped = await skippedInstances([host]);
|
|
|
|
return skipped.length > 0;
|
|
|
|
}
|