Merge branch 'main' into mint-cashu

This commit is contained in:
P. Reis 2025-03-13 10:41:17 -03:00
commit 921f478279
3 changed files with 64 additions and 40 deletions

View file

@ -1,8 +1,7 @@
import { DittoConf } from '@ditto/conf'; import { DittoConf } from '@ditto/conf';
import { assertEquals } from '@std/assert'; import { assertEquals } from '@std/assert';
import { eventFixture } from '@/test.ts'; import { contentToHtml, getCardUrl, getMediaLinks, removeTrailingTokens } from '@/utils/note.ts';
import { contentToHtml, getCardUrl, getMediaLinks, stripMediaUrls } from '@/utils/note.ts';
import { genEvent } from '@nostrify/nostrify/test'; import { genEvent } from '@nostrify/nostrify/test';
Deno.test('contentToHtml', () => { Deno.test('contentToHtml', () => {
@ -125,24 +124,53 @@ Deno.test('getMediaLinks', () => {
]]); ]]);
}); });
Deno.test('stripMediaUrls', async () => { Deno.test('removeTrailingTokens with spaces', () => {
const { content, tags } = await eventFixture('event-imeta'); const urls = new Set<string>([
'https://ditto.pub/a.png',
'https://ditto.pub/b.jpg',
]);
const media: string[][][] = tags const result = removeTrailingTokens(
.filter(([name]) => name === 'imeta') 'hey!\n\nthis is cool https://ditto.pub/a.png https://ditto.pub/b.jpg',
.map(([_, ...entries]) => urls,
entries.map((entry) => { );
const split = entry.split(' ');
return [split[0], split.splice(1).join(' ')];
})
);
const stripped = stripMediaUrls(content, media); assertEquals(result, 'hey!\n\nthis is cool');
});
const expected = Deno.test('removeTrailingTokens with newlines', () => {
`Today we were made aware of multiple Fediverse blog posts incorrectly attributing “vote Trump” spam on Bluesky to the Mostr.pub Bridge. \n\nThis spam is NOT coming from Mostr. From the screenshots used in these blogs, it's clear the spam is coming from an entirely different bridge called momostr.pink. This bridge is not affiliated with Mostr, and is not even a fork of Mostr. We appreciate that the authors of these posts responded quickly to us and have since corrected the blogs. \n\nMostr.pub uses stirfry policies for anti-spam filtering. This includes an anti-duplication policy that prevents spam like the recent “vote Trump” posts weve seen repeated over and over. \n\nIt is important to note WHY there are multiple bridges, though. \n\nWhen Mostr.pub launched, multiple major servers immediately blocked Mostr, including Mastodon.social. The moderators of Mastodon.social claimed that this was because Nostr was unregulated, and suggested to one user that if they want to bridge their account they should host their own bridge.\n\nThat is exactly what momostr.pink, the source of this spam, has done. \n\nThe obvious response to the censorship of the Mostr Bridge is to build more bridges. \n\nWhile we have opted for pro-social policies that aim to reduce spam and build better connections between decentralized platforms, other bridges built to get around censorship of the Mostr Bridge may not — as were already seeing.\n\nThere will inevitably be multiple bridges, and were working on creating solutions to the problems that arise from that. In the meantime, if the Fediverse could do itself a favor and chill with the censorship for two seconds, we might not have so many problems. `; const urls = new Set<string>([
'https://ditto.pub/a.png',
'https://ditto.pub/b.jpg',
]);
assertEquals(stripped, expected); const result = removeTrailingTokens(
'Hey!\n\nthis is cool \n\nhttps://ditto.pub/a.png\nhttps://ditto.pub/b.jpg\n ',
urls,
);
assertEquals(result, 'Hey!\n\nthis is cool');
});
Deno.test('removeTrailingTokens with only URLs', () => {
const urls = new Set<string>([
'https://ditto.pub/a.png',
'https://ditto.pub/b.jpg',
]);
const result = removeTrailingTokens(
'https://ditto.pub/a.png https://ditto.pub/b.jpg',
urls,
);
assertEquals(result, '');
});
Deno.test('removeTrailingTokens with just one URL', () => {
const urls = new Set<string>(['https://ditto.pub/a.png']);
const result = removeTrailingTokens('https://ditto.pub/a.png', urls);
assertEquals(result, '');
}); });
Deno.test('getCardUrl', async (t) => { Deno.test('getCardUrl', async (t) => {

View file

@ -69,34 +69,20 @@ export function contentToHtml(content: string, mentions: MastodonMention[], opts
}).replace(/\n+$/, ''); }).replace(/\n+$/, '');
} }
/** Remove media URLs from content. */ /** Remove the tokens from the _end_ of the content. */
export function stripMediaUrls(content: string, media: string[][][]): string { export function removeTrailingTokens(text: string, tokens: Set<string>): string {
if (!media.length) { let trimmedText = text;
return content;
}
const urls = new Set<string>(); while (true) {
const match = trimmedText.match(/([^\s]+)(?:\s+)?$/);
for (const tags of media) { if (match && tokens.has(match[1])) {
for (const [name, value] of tags) { trimmedText = trimmedText.slice(0, match.index).replace(/\s+$/, '');
if (name === 'url') {
urls.add(value);
break;
}
}
}
const lines = content.split('\n').reverse();
for (const line of [...lines]) {
if (line === '' || urls.has(line)) {
lines.splice(0, 1);
} else { } else {
break; break;
} }
} }
return lines.reverse().join('\n'); return trimmedText;
} }
export function getLinks(content: string) { export function getLinks(content: string) {

View file

@ -4,7 +4,7 @@ import { nip19 } from 'nostr-tools';
import { Conf } from '@/config.ts'; import { Conf } from '@/config.ts';
import { type DittoEvent } from '@/interfaces/DittoEvent.ts'; import { type DittoEvent } from '@/interfaces/DittoEvent.ts';
import { nostrDate } from '@/utils.ts'; import { nostrDate } from '@/utils.ts';
import { contentToHtml, getLinks, getMediaLinks, stripMediaUrls } from '@/utils/note.ts'; import { contentToHtml, getLinks, getMediaLinks, removeTrailingTokens } from '@/utils/note.ts';
import { findReplyTag } from '@/utils/tags.ts'; import { findReplyTag } from '@/utils/tags.ts';
import { accountFromPubkey, renderAccount } from '@/views/mastodon/accounts.ts'; import { accountFromPubkey, renderAccount } from '@/views/mastodon/accounts.ts';
import { renderAttachment } from '@/views/mastodon/attachments.ts'; import { renderAttachment } from '@/views/mastodon/attachments.ts';
@ -52,8 +52,18 @@ async function renderStatus(
); );
const media = imeta.length ? imeta : getMediaLinks(links); const media = imeta.length ? imeta : getMediaLinks(links);
const mediaUrls = new Set<string>();
const html = contentToHtml(stripMediaUrls(event.content, media), mentions, { conf: Conf }); for (const tags of media) {
for (const [name, value] of tags) {
if (name === 'url') {
mediaUrls.add(value);
break;
}
}
}
const html = contentToHtml(removeTrailingTokens(event.content, mediaUrls), mentions, { conf: Conf });
const relatedEvents = viewerPubkey const relatedEvents = viewerPubkey
? await store.query([ ? await store.query([