2022-03-12 13:54:28 -08:00
|
|
|
/**
|
|
|
|
* Card normalizer:
|
|
|
|
* Converts API cards into our internal format.
|
|
|
|
* @see {@link https://docs.joinmastodon.org/entities/card/}
|
|
|
|
*/
|
2022-04-16 10:48:05 -07:00
|
|
|
import punycode from 'punycode';
|
|
|
|
|
2022-03-16 19:33:09 -07:00
|
|
|
import { Record as ImmutableRecord, Map as ImmutableMap, fromJS } from 'immutable';
|
2022-03-12 13:54:28 -08:00
|
|
|
|
2023-04-17 13:34:48 -07:00
|
|
|
import { groupSchema, type Group } from 'soapbox/schemas';
|
2022-04-16 10:57:17 -07:00
|
|
|
import { mergeDefined } from 'soapbox/utils/normalizers';
|
|
|
|
|
2022-03-12 13:54:28 -08:00
|
|
|
// https://docs.joinmastodon.org/entities/card/
|
2022-03-16 19:15:38 -07:00
|
|
|
export const CardRecord = ImmutableRecord({
|
2022-03-12 13:54:28 -08:00
|
|
|
author_name: '',
|
|
|
|
author_url: '',
|
2022-04-16 10:48:05 -07:00
|
|
|
blurhash: null as string | null,
|
2022-03-12 13:54:28 -08:00
|
|
|
description: '',
|
|
|
|
embed_url: '',
|
2023-04-17 13:34:48 -07:00
|
|
|
group: null as null | Group,
|
2022-03-12 13:54:28 -08:00
|
|
|
height: 0,
|
|
|
|
html: '',
|
2022-04-16 10:48:05 -07:00
|
|
|
image: null as string | null,
|
2022-03-12 13:54:28 -08:00
|
|
|
provider_name: '',
|
|
|
|
provider_url: '',
|
|
|
|
title: '',
|
|
|
|
type: 'link',
|
|
|
|
url: '',
|
|
|
|
width: 0,
|
|
|
|
});
|
|
|
|
|
2022-04-16 10:48:05 -07:00
|
|
|
const IDNA_PREFIX = 'xn--';
|
|
|
|
|
|
|
|
const decodeIDNA = (domain: string): string => {
|
|
|
|
return domain
|
|
|
|
.split('.')
|
|
|
|
.map(part => part.indexOf(IDNA_PREFIX) === 0 ? punycode.decode(part.slice(IDNA_PREFIX.length)) : part)
|
|
|
|
.join('.');
|
|
|
|
};
|
|
|
|
|
|
|
|
const getHostname = (url: string): string => {
|
|
|
|
const parser = document.createElement('a');
|
|
|
|
parser.href = url;
|
|
|
|
return parser.hostname;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Fall back to Pleroma's OG data */
|
2022-04-16 10:57:17 -07:00
|
|
|
const normalizePleromaOpengraph = (card: ImmutableMap<string, any>) => {
|
|
|
|
const opengraph = ImmutableMap({
|
|
|
|
width: card.getIn(['pleroma', 'opengraph', 'width']),
|
|
|
|
height: card.getIn(['pleroma', 'opengraph', 'height']),
|
|
|
|
html: card.getIn(['pleroma', 'opengraph', 'html']),
|
|
|
|
image: card.getIn(['pleroma', 'opengraph', 'thumbnail_url']),
|
|
|
|
});
|
2022-04-16 10:48:05 -07:00
|
|
|
|
2022-04-16 10:57:17 -07:00
|
|
|
return card.mergeWith(mergeDefined, opengraph);
|
2022-04-16 10:48:05 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/** Set provider from URL if not found */
|
|
|
|
const normalizeProviderName = (card: ImmutableMap<string, any>) => {
|
|
|
|
const providerName = card.get('provider_name') || decodeIDNA(getHostname(card.get('url')));
|
|
|
|
return card.set('provider_name', providerName);
|
|
|
|
};
|
|
|
|
|
2023-04-17 13:34:48 -07:00
|
|
|
const normalizeGroup = (card: ImmutableMap<string, any>) => {
|
|
|
|
try {
|
|
|
|
const group = groupSchema.parse(card.get('group').toJS());
|
|
|
|
return card.set('group', group);
|
|
|
|
} catch (_e) {
|
|
|
|
return card.set('group', null);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-03-16 19:33:09 -07:00
|
|
|
export const normalizeCard = (card: Record<string, any>) => {
|
|
|
|
return CardRecord(
|
2022-04-16 10:48:05 -07:00
|
|
|
ImmutableMap(fromJS(card)).withMutations(card => {
|
2022-04-16 10:57:17 -07:00
|
|
|
normalizePleromaOpengraph(card);
|
2022-04-16 10:48:05 -07:00
|
|
|
normalizeProviderName(card);
|
2023-04-17 13:34:48 -07:00
|
|
|
normalizeGroup(card);
|
2022-04-16 10:48:05 -07:00
|
|
|
}),
|
2022-03-16 19:33:09 -07:00
|
|
|
);
|
2022-03-12 13:54:28 -08:00
|
|
|
};
|