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
|
|
|
|
|
|
|
// 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: '',
|
|
|
|
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 */
|
|
|
|
const normalizeWidth = (card: ImmutableMap<string, any>) => {
|
|
|
|
const width = card.get('width') || card.getIn(['pleroma', 'opengraph', 'width']) || 0;
|
|
|
|
return card.set('width', width);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Fall back to Pleroma's OG data */
|
|
|
|
const normalizeHeight = (card: ImmutableMap<string, any>) => {
|
|
|
|
const height = card.get('height') || card.getIn(['pleroma', 'opengraph', 'height']) || 0;
|
|
|
|
return card.set('height', height);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Fall back to Pleroma's OG data */
|
|
|
|
const normalizeHtml = (card: ImmutableMap<string, any>) => {
|
|
|
|
const html = card.get('html') || card.getIn(['pleroma', 'opengraph', 'html']) || '';
|
|
|
|
return card.set('html', html);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Fall back to Pleroma's OG data */
|
|
|
|
const normalizeImage = (card: ImmutableMap<string, any>) => {
|
|
|
|
const image = card.get('image') || card.getIn(['pleroma', 'opengraph', 'thumbnail_url']) || null;
|
|
|
|
return card.set('image', image);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** 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);
|
|
|
|
};
|
|
|
|
|
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 => {
|
|
|
|
normalizeWidth(card);
|
|
|
|
normalizeHeight(card);
|
|
|
|
normalizeHtml(card);
|
|
|
|
normalizeImage(card);
|
|
|
|
normalizeProviderName(card);
|
|
|
|
}),
|
2022-03-16 19:33:09 -07:00
|
|
|
);
|
2022-03-12 13:54:28 -08:00
|
|
|
};
|