summaryrefslogtreecommitdiff
path: root/src/services
diff options
context:
space:
mode:
authorsyuilo <Syuilotan@yahoo.co.jp>2020-07-26 11:04:07 +0900
committerGitHub <noreply@github.com>2020-07-26 11:04:07 +0900
commitf1ef85b63625edccd717dda8324078c232c7750e (patch)
tree2681f8f5057a9608dd9be943f486ebdfb81717e4 /src/services
parentUpdate CONTRIBUTING.md (diff)
downloadmisskey-f1ef85b63625edccd717dda8324078c232c7750e.tar.gz
misskey-f1ef85b63625edccd717dda8324078c232c7750e.tar.bz2
misskey-f1ef85b63625edccd717dda8324078c232c7750e.zip
feat(server): Fetch icon url of an instance (#6591)
* feat(server): Fetch icon url of an instance Resolve #6589 * chore: Rename the function
Diffstat (limited to 'src/services')
-rw-r--r--src/services/fetch-instance-metadata.ts135
-rw-r--r--src/services/fetch-nodeinfo.ts72
2 files changed, 135 insertions, 72 deletions
diff --git a/src/services/fetch-instance-metadata.ts b/src/services/fetch-instance-metadata.ts
new file mode 100644
index 0000000000..41fef859c9
--- /dev/null
+++ b/src/services/fetch-instance-metadata.ts
@@ -0,0 +1,135 @@
+import { JSDOM } from 'jsdom';
+import fetch from 'node-fetch';
+import { getJson, getHtml, getAgentByUrl } from '../misc/fetch';
+import { Instance } from '../models/entities/instance';
+import { Instances } from '../models';
+import { getFetchInstanceMetadataLock } from '../misc/app-lock';
+import Logger from './logger';
+import { URL } from 'url';
+
+const logger = new Logger('metadata', 'cyan');
+
+export async function fetchInstanceMetadata(instance: Instance): Promise<void> {
+ const unlock = await getFetchInstanceMetadataLock(instance.host);
+
+ const _instance = await Instances.findOne({ host: instance.host });
+ const now = Date.now();
+ if (_instance && _instance.infoUpdatedAt && (now - _instance.infoUpdatedAt.getTime() < 1000 * 60 * 60 * 24)) {
+ unlock();
+ return;
+ }
+
+ logger.info(`Fetching metadata of ${instance.host} ...`);
+
+ try {
+ const [info, icon] = await Promise.all([
+ fetchNodeinfo(instance).catch(() => null),
+ fetchIconUrl(instance).catch(() => null),
+ ]);
+
+ logger.succ(`Successfuly fetched metadata of ${instance.host}`);
+
+ const updates = {
+ infoUpdatedAt: new Date(),
+ } as Record<string, any>;
+
+ if (info) {
+ updates.softwareName = info.software.name.toLowerCase();
+ updates.softwareVersion = info.software.version;
+ updates.openRegistrations = info.openRegistrations;
+ updates.name = info.metadata ? (info.metadata.nodeName || info.metadata.name || null) : null;
+ updates.description = info.metadata ? (info.metadata.nodeDescription || info.metadata.description || null) : null;
+ updates.maintainerName = info.metadata ? info.metadata.maintainer ? (info.metadata.maintainer.name || null) : null : null;
+ updates.maintainerEmail = info.metadata ? info.metadata.maintainer ? (info.metadata.maintainer.email || null) : null : null;
+ }
+
+ if (icon) {
+ updates.iconUrl = icon;
+ }
+
+ await Instances.update(instance.id, updates);
+
+ logger.succ(`Successfuly updated metadata of ${instance.host}`);
+ } catch (e) {
+ logger.error(`Failed to update metadata of ${instance.host}: ${e}`);
+ } finally {
+ unlock();
+ }
+}
+
+async function fetchNodeinfo(instance: Instance): Promise<Record<string, any>> {
+ logger.info(`Fetching nodeinfo of ${instance.host} ...`);
+
+ try {
+ const wellknown = await getJson('https://' + instance.host + '/.well-known/nodeinfo')
+ .catch(e => {
+ if (e.statusCode === 404) {
+ throw 'No nodeinfo provided';
+ } else {
+ throw e.statusCode || e.message;
+ }
+ });
+
+ if (wellknown.links == null || !Array.isArray(wellknown.links)) {
+ throw 'No wellknown links';
+ }
+
+ const links = wellknown.links as any[];
+
+ const lnik1_0 = links.find(link => link.rel === 'http://nodeinfo.diaspora.software/ns/schema/1.0');
+ const lnik2_0 = links.find(link => link.rel === 'http://nodeinfo.diaspora.software/ns/schema/2.0');
+ const lnik2_1 = links.find(link => link.rel === 'http://nodeinfo.diaspora.software/ns/schema/2.1');
+ const link = lnik2_1 || lnik2_0 || lnik1_0;
+
+ if (link == null) {
+ throw 'No nodeinfo link provided';
+ }
+
+ const info = await getJson(link.href)
+ .catch(e => {
+ throw e.statusCode || e.message;
+ });
+
+ logger.succ(`Successfuly fetched nodeinfo of ${instance.host}`);
+
+ return info;
+ } catch (e) {
+ logger.error(`Failed to fetch nodeinfo of ${instance.host}: ${e}`);
+
+ throw e;
+ }
+}
+
+async function fetchIconUrl(instance: Instance): Promise<string | null> {
+ logger.info(`Fetching icon URL of ${instance.host} ...`);
+
+ const url = 'https://' + instance.host;
+
+ const html = await getHtml(url);
+
+ const { window } = new JSDOM(html);
+ const doc = window.document;
+
+ const hrefAppleTouchIconPrecomposed = doc.querySelector('link[rel="apple-touch-icon-precomposed"]')?.getAttribute('href');
+ const hrefAppleTouchIcon = doc.querySelector('link[rel="apple-touch-icon"]')?.getAttribute('href');
+ const hrefIcon = doc.querySelector('link[rel="icon"]')?.getAttribute('href');
+
+ const href = hrefAppleTouchIconPrecomposed || hrefAppleTouchIcon || hrefIcon;
+
+ if (href) {
+ return (new URL(href, url)).href;
+ }
+
+ const faviconUrl = url + '/favicon.ico';
+
+ const favicon = await fetch(faviconUrl, {
+ timeout: 10000,
+ agent: getAgentByUrl,
+ });
+
+ if (favicon.ok) {
+ return faviconUrl;
+ }
+
+ return null;
+}
diff --git a/src/services/fetch-nodeinfo.ts b/src/services/fetch-nodeinfo.ts
deleted file mode 100644
index 0cf51e3377..0000000000
--- a/src/services/fetch-nodeinfo.ts
+++ /dev/null
@@ -1,72 +0,0 @@
-import { getJson } from '../misc/fetch';
-import { Instance } from '../models/entities/instance';
-import { Instances } from '../models';
-import { getNodeinfoLock } from '../misc/app-lock';
-import Logger from '../services/logger';
-
-export const logger = new Logger('nodeinfo', 'cyan');
-
-export async function fetchNodeinfo(instance: Instance) {
- const unlock = await getNodeinfoLock(instance.host);
-
- const _instance = await Instances.findOne({ host: instance.host });
- const now = Date.now();
- if (_instance && _instance.infoUpdatedAt && (now - _instance.infoUpdatedAt.getTime() < 1000 * 60 * 60 * 24)) {
- unlock();
- return;
- }
-
- logger.info(`Fetching nodeinfo of ${instance.host} ...`);
-
- try {
- const wellknown = await getJson('https://' + instance.host + '/.well-known/nodeinfo')
- .catch(e => {
- if (e.statusCode === 404) {
- throw 'No nodeinfo provided';
- } else {
- throw e.statusCode || e.message;
- }
- });
-
- if (wellknown.links == null || !Array.isArray(wellknown.links)) {
- throw 'No wellknown links';
- }
-
- const links = wellknown.links as any[];
-
- const lnik1_0 = links.find(link => link.rel === 'http://nodeinfo.diaspora.software/ns/schema/1.0');
- const lnik2_0 = links.find(link => link.rel === 'http://nodeinfo.diaspora.software/ns/schema/2.0');
- const lnik2_1 = links.find(link => link.rel === 'http://nodeinfo.diaspora.software/ns/schema/2.1');
- const link = lnik2_1 || lnik2_0 || lnik1_0;
-
- if (link == null) {
- throw 'No nodeinfo link provided';
- }
-
- const info = await getJson(link.href)
- .catch(e => {
- throw e.statusCode || e.message;
- });
-
- await Instances.update(instance.id, {
- infoUpdatedAt: new Date(),
- softwareName: info.software.name.toLowerCase(),
- softwareVersion: info.software.version,
- openRegistrations: info.openRegistrations,
- name: info.metadata ? (info.metadata.nodeName || info.metadata.name || null) : null,
- description: info.metadata ? (info.metadata.nodeDescription || info.metadata.description || null) : null,
- maintainerName: info.metadata ? info.metadata.maintainer ? (info.metadata.maintainer.name || null) : null : null,
- maintainerEmail: info.metadata ? info.metadata.maintainer ? (info.metadata.maintainer.email || null) : null : null,
- });
-
- logger.succ(`Successfuly fetched nodeinfo of ${instance.host}`);
- } catch (e) {
- logger.error(`Failed to fetch nodeinfo of ${instance.host}: ${e}`);
-
- await Instances.update(instance.id, {
- infoUpdatedAt: new Date(),
- });
- } finally {
- unlock();
- }
-}