/** * Card normalizer: * Converts API cards into our internal format. * @see {@link https://docs.joinmastodon.org/entities/card/} */ import punycode from 'punycode'; import { Record as ImmutableRecord, Map as ImmutableMap, fromJS } from 'immutable'; // https://docs.joinmastodon.org/entities/card/ export const CardRecord = ImmutableRecord({ author_name: '', author_url: '', blurhash: null as string | null, description: '', embed_url: '', height: 0, html: '', image: null as string | null, provider_name: '', provider_url: '', title: '', type: 'link', url: '', width: 0, }); 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) => { 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) => { 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) => { 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) => { 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) => { const providerName = card.get('provider_name') || decodeIDNA(getHostname(card.get('url'))); return card.set('provider_name', providerName); }; export const normalizeCard = (card: Record) => { return CardRecord( ImmutableMap(fromJS(card)).withMutations(card => { normalizeWidth(card); normalizeHeight(card); normalizeHtml(card); normalizeImage(card); normalizeProviderName(card); }), ); };