@ecency/render-helper 2.4.15 → 2.4.16
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/browser/index.d.ts +12 -2
- package/dist/browser/index.js +49 -48
- package/dist/browser/index.js.map +1 -1
- package/dist/node/index.cjs +49 -48
- package/dist/node/index.cjs.map +1 -1
- package/dist/node/index.mjs +49 -48
- package/dist/node/index.mjs.map +1 -1
- package/package.json +1 -1
package/dist/node/index.cjs
CHANGED
|
@@ -107,8 +107,8 @@ var LBRY_REGEX = /^(https?:)?\/\/lbry.tv\/\$\/embed\/[^?#]+(?:$|[?#])/i;
|
|
|
107
107
|
var ODYSEE_REGEX = /^(https?:)?\/\/odysee\.com\/(?:\$|%24)\/embed\/[^/?#]+(?:$|[?#])/i;
|
|
108
108
|
var SKATEHIVE_IPFS_REGEX = /^https?:\/\/ipfs\.skatehive\.app\/ipfs\/([^/?#]+)/i;
|
|
109
109
|
var ARCH_REGEX = /^(https?:)?\/\/archive.org\/embed\/[^/?#]+(?:$|[?#])/i;
|
|
110
|
-
var SPEAK_REGEX = /(?:https?:\/\/(?:(?:play\.)?3speak
|
|
111
|
-
var SPEAK_EMBED_REGEX = /^(https?:)?\/\/(?:play\.)?3speak
|
|
110
|
+
var SPEAK_REGEX = /(?:https?:\/\/(?:(?:play\.)?3speak\.([a-z]+)\/watch\?v=)|(?:(?:play\.)?3speak\.([a-z]+)\/embed\?v=))([A-Za-z0-9_\-\.\/]+)(&.*)?/i;
|
|
111
|
+
var SPEAK_EMBED_REGEX = /^(https?:)?\/\/(?:play\.)?3speak\.([a-z]+)\/(?:embed|watch)\?.+$/i;
|
|
112
112
|
var TWITTER_REGEX = /(?:https?:\/\/(?:(?:twitter\.com\/(.*?)\/status\/(.*))))/gi;
|
|
113
113
|
var SPOTIFY_REGEX = /^https:\/\/open\.spotify\.com\/playlist\/(.*)?$/gi;
|
|
114
114
|
var RUMBLE_REGEX = /^https:\/\/rumble.com\/embed\/([a-zA-Z0-9-]+)\/\?pub=\w+/;
|
|
@@ -316,15 +316,13 @@ function sanitizeHtml(html) {
|
|
|
316
316
|
});
|
|
317
317
|
}
|
|
318
318
|
var proxyBase = "https://images.ecency.com";
|
|
319
|
-
var fileExtension = true;
|
|
320
319
|
function setProxyBase(p2) {
|
|
321
320
|
proxyBase = p2;
|
|
322
|
-
fileExtension = proxyBase == "https://images.ecency.com";
|
|
323
321
|
}
|
|
324
322
|
function extractPHash(url) {
|
|
325
323
|
if (url.startsWith(`${proxyBase}/p/`)) {
|
|
326
324
|
const [hash] = url.split("/p/")[1].split("?");
|
|
327
|
-
return hash.replace(
|
|
325
|
+
return hash.replace(/\.(webp|png)$/, "");
|
|
328
326
|
}
|
|
329
327
|
return null;
|
|
330
328
|
}
|
|
@@ -339,7 +337,7 @@ function getLatestUrl(str) {
|
|
|
339
337
|
const [last] = [...str.replace(/https?:\/\//g, "\n$&").trim().split("\n")].reverse();
|
|
340
338
|
return last;
|
|
341
339
|
}
|
|
342
|
-
function proxifyImageSrc(url, width = 0, height = 0,
|
|
340
|
+
function proxifyImageSrc(url, width = 0, height = 0, _format = "match") {
|
|
343
341
|
if (!url || typeof url !== "string" || !isValidUrl(url)) {
|
|
344
342
|
return "";
|
|
345
343
|
}
|
|
@@ -352,7 +350,7 @@ function proxifyImageSrc(url, width = 0, height = 0, format = "match") {
|
|
|
352
350
|
const realUrl = getLatestUrl(url);
|
|
353
351
|
const pHash = extractPHash(realUrl);
|
|
354
352
|
const options = {
|
|
355
|
-
format,
|
|
353
|
+
format: "match",
|
|
356
354
|
mode: "fit"
|
|
357
355
|
};
|
|
358
356
|
if (width > 0) {
|
|
@@ -363,31 +361,29 @@ function proxifyImageSrc(url, width = 0, height = 0, format = "match") {
|
|
|
363
361
|
}
|
|
364
362
|
const qs = querystring__default.default.stringify(options);
|
|
365
363
|
if (pHash) {
|
|
366
|
-
|
|
367
|
-
return `${proxyBase}/p/${pHash}${format === "webp" ? ".webp" : ".png"}?${qs}`;
|
|
368
|
-
} else {
|
|
369
|
-
return `${proxyBase}/p/${pHash}?${qs}`;
|
|
370
|
-
}
|
|
364
|
+
return `${proxyBase}/p/${pHash}?${qs}`;
|
|
371
365
|
}
|
|
372
366
|
const b58url = multihash__default.default.toB58String(Buffer.from(realUrl.toString()));
|
|
373
|
-
return `${proxyBase}/p/${b58url}
|
|
367
|
+
return `${proxyBase}/p/${b58url}?${qs}`;
|
|
374
368
|
}
|
|
375
369
|
|
|
376
370
|
// src/methods/img.method.ts
|
|
377
|
-
function img(el,
|
|
378
|
-
|
|
371
|
+
function img(el, state) {
|
|
372
|
+
const src = el.getAttribute("src") || "";
|
|
379
373
|
const decodedSrc = decodeURIComponent(
|
|
380
374
|
src.replace(/&#(\d+);/g, (_, dec) => String.fromCharCode(dec)).replace(/&#x([0-9a-f]+);/gi, (_, hex) => String.fromCharCode(parseInt(hex, 16)))
|
|
381
375
|
).trim();
|
|
376
|
+
["onerror", "dynsrc", "lowsrc", "width", "height"].forEach((attr) => el.removeAttribute(attr));
|
|
382
377
|
const isInvalid = !src || decodedSrc.startsWith("javascript") || decodedSrc.startsWith("vbscript") || decodedSrc === "x";
|
|
383
378
|
if (isInvalid) {
|
|
384
|
-
src
|
|
379
|
+
el.removeAttribute("src");
|
|
380
|
+
return;
|
|
385
381
|
}
|
|
386
382
|
const isRelative = !/^https?:\/\//i.test(decodedSrc) && !decodedSrc.startsWith("/");
|
|
387
383
|
if (isRelative) {
|
|
388
|
-
src
|
|
384
|
+
el.removeAttribute("src");
|
|
385
|
+
return;
|
|
389
386
|
}
|
|
390
|
-
["onerror", "dynsrc", "lowsrc", "width", "height"].forEach((attr) => el.removeAttribute(attr));
|
|
391
387
|
el.setAttribute("itemprop", "image");
|
|
392
388
|
const isLCP = state && !state.firstImageFound;
|
|
393
389
|
if (isLCP) {
|
|
@@ -402,14 +398,17 @@ function img(el, webp, state) {
|
|
|
402
398
|
const shouldReplace = !cls.includes("no-replace");
|
|
403
399
|
const hasAlreadyProxied = src.startsWith("https://images.ecency.com");
|
|
404
400
|
if (shouldReplace && !hasAlreadyProxied) {
|
|
405
|
-
const proxified = proxifyImageSrc(
|
|
406
|
-
|
|
401
|
+
const proxified = proxifyImageSrc(decodedSrc);
|
|
402
|
+
if (proxified) {
|
|
403
|
+
el.setAttribute("src", proxified);
|
|
404
|
+
}
|
|
407
405
|
}
|
|
408
406
|
}
|
|
409
|
-
function createImageHTML(src, isLCP
|
|
407
|
+
function createImageHTML(src, isLCP) {
|
|
408
|
+
const proxified = proxifyImageSrc(src);
|
|
409
|
+
if (!proxified) return "";
|
|
410
410
|
const loading = isLCP ? "eager" : "lazy";
|
|
411
411
|
const fetch = isLCP ? 'fetchpriority="high"' : 'decoding="async"';
|
|
412
|
-
const proxified = proxifyImageSrc(src, 0, 0, webp ? "webp" : "match");
|
|
413
412
|
return `<img
|
|
414
413
|
class="markdown-img-link"
|
|
415
414
|
src="${proxified}"
|
|
@@ -461,7 +460,7 @@ var addLineBreakBeforePostLink = (el, forApp, isInline) => {
|
|
|
461
460
|
el.parentNode.insertBefore(br, el);
|
|
462
461
|
}
|
|
463
462
|
};
|
|
464
|
-
function a(el, forApp,
|
|
463
|
+
function a(el, forApp, parentDomain = "ecency.com", seoContext) {
|
|
465
464
|
if (!el || !el.parentNode) {
|
|
466
465
|
return;
|
|
467
466
|
}
|
|
@@ -479,7 +478,7 @@ function a(el, forApp, webp, parentDomain = "ecency.com", seoContext) {
|
|
|
479
478
|
}
|
|
480
479
|
if (href.match(IMG_REGEX) && href.trim().replace(/&/g, "&") === getSerializedInnerHTML(el).trim().replace(/&/g, "&")) {
|
|
481
480
|
const isLCP = false;
|
|
482
|
-
const imgHTML = createImageHTML(href, isLCP
|
|
481
|
+
const imgHTML = createImageHTML(href, isLCP);
|
|
483
482
|
const doc = DOMParser.parseFromString(imgHTML, "text/html");
|
|
484
483
|
const replaceNode = doc.body?.firstChild || doc.firstChild;
|
|
485
484
|
if (replaceNode) {
|
|
@@ -801,7 +800,7 @@ function a(el, forApp, webp, parentDomain = "ecency.com", seoContext) {
|
|
|
801
800
|
el.setAttribute("class", "markdown-video-link markdown-video-link-youtube");
|
|
802
801
|
el.removeAttribute("href");
|
|
803
802
|
const vid = match[1];
|
|
804
|
-
const thumbnail = proxifyImageSrc(`https://img.youtube.com/vi/${vid.split("?")[0]}/hqdefault.jpg`, 0, 0,
|
|
803
|
+
const thumbnail = proxifyImageSrc(`https://img.youtube.com/vi/${vid.split("?")[0]}/hqdefault.jpg`, 0, 0, "match");
|
|
805
804
|
const embedSrc = `https://www.youtube.com/embed/${vid}?autoplay=1`;
|
|
806
805
|
el.textContent = "";
|
|
807
806
|
el.setAttribute("data-embed-src", embedSrc);
|
|
@@ -897,7 +896,7 @@ function a(el, forApp, webp, parentDomain = "ecency.com", seoContext) {
|
|
|
897
896
|
if (imgEls.length === 1) {
|
|
898
897
|
const src = imgEls[0].getAttribute("src");
|
|
899
898
|
if (src) {
|
|
900
|
-
const thumbnail = proxifyImageSrc(src.replace(/\s+/g, ""), 0, 0,
|
|
899
|
+
const thumbnail = proxifyImageSrc(src.replace(/\s+/g, ""), 0, 0, "match");
|
|
901
900
|
const thumbImg = el.ownerDocument.createElement("img");
|
|
902
901
|
thumbImg.setAttribute("class", "no-replace video-thumbnail");
|
|
903
902
|
thumbImg.setAttribute("itemprop", "thumbnailUrl");
|
|
@@ -932,7 +931,7 @@ function a(el, forApp, webp, parentDomain = "ecency.com", seoContext) {
|
|
|
932
931
|
const imgEls2 = el.getElementsByTagName("img");
|
|
933
932
|
if (imgEls2.length === 1 || el.textContent.trim() === href) {
|
|
934
933
|
if ((match[1] || match[2]) && match[3]) {
|
|
935
|
-
const videoHref = `https://play.3speak.tv/
|
|
934
|
+
const videoHref = `https://play.3speak.tv/embed?v=${match[3]}&mode=iframe`;
|
|
936
935
|
el.setAttribute("class", "markdown-video-link markdown-video-link-speak");
|
|
937
936
|
el.removeAttribute("href");
|
|
938
937
|
el.setAttribute("data-embed-src", videoHref);
|
|
@@ -942,7 +941,7 @@ function a(el, forApp, webp, parentDomain = "ecency.com", seoContext) {
|
|
|
942
941
|
if (imgEls2.length === 1) {
|
|
943
942
|
const src = imgEls2[0].getAttribute("src");
|
|
944
943
|
if (src) {
|
|
945
|
-
const thumbnail = proxifyImageSrc(src.replace(/\s+/g, ""), 0, 0,
|
|
944
|
+
const thumbnail = proxifyImageSrc(src.replace(/\s+/g, ""), 0, 0, "match");
|
|
946
945
|
const thumbImg = el.ownerDocument.createElement("img");
|
|
947
946
|
thumbImg.setAttribute("class", "no-replace video-thumbnail");
|
|
948
947
|
thumbImg.setAttribute("itemprop", "thumbnailUrl");
|
|
@@ -1065,7 +1064,8 @@ function iframe(el, parentDomain = "ecency.com") {
|
|
|
1065
1064
|
return;
|
|
1066
1065
|
}
|
|
1067
1066
|
if (src.match(SPEAK_EMBED_REGEX)) {
|
|
1068
|
-
let normalizedSrc = src.replace(/3speak\.[a-z]+/i, "play.3speak.tv");
|
|
1067
|
+
let normalizedSrc = src.replace(/(?:play\.)?3speak\.[a-z]+/i, "play.3speak.tv");
|
|
1068
|
+
normalizedSrc = normalizedSrc.replace(/\/watch\?/, "/embed?");
|
|
1069
1069
|
const hasMode = /[?&]mode=/.test(normalizedSrc);
|
|
1070
1070
|
if (!hasMode) {
|
|
1071
1071
|
normalizedSrc = `${normalizedSrc}&mode=iframe`;
|
|
@@ -1073,6 +1073,7 @@ function iframe(el, parentDomain = "ecency.com") {
|
|
|
1073
1073
|
const hasAutoplay = /[?&]autoplay=/.test(normalizedSrc);
|
|
1074
1074
|
const s = hasAutoplay ? normalizedSrc : `${normalizedSrc}&autoplay=true`;
|
|
1075
1075
|
el.setAttribute("src", s);
|
|
1076
|
+
el.setAttribute("class", "speak-iframe");
|
|
1076
1077
|
return;
|
|
1077
1078
|
}
|
|
1078
1079
|
if (src.match(SPOTIFY_EMBED_REGEX)) {
|
|
@@ -1181,7 +1182,7 @@ function p(el) {
|
|
|
1181
1182
|
}
|
|
1182
1183
|
|
|
1183
1184
|
// src/methods/linkify.method.ts
|
|
1184
|
-
function linkify(content, forApp
|
|
1185
|
+
function linkify(content, forApp) {
|
|
1185
1186
|
content = content.replace(/(^|\s|>)(#[-a-z\d]+)/gi, (tag) => {
|
|
1186
1187
|
if (/#[\d]+$/.test(tag)) return tag;
|
|
1187
1188
|
const preceding = /^\s|>/.test(tag) ? tag[0] : "";
|
|
@@ -1223,21 +1224,21 @@ function linkify(content, forApp, webp) {
|
|
|
1223
1224
|
content = content.replace(IMG_REGEX, (imglink) => {
|
|
1224
1225
|
const isLCP = !firstImageUsed;
|
|
1225
1226
|
firstImageUsed = true;
|
|
1226
|
-
return createImageHTML(imglink, isLCP
|
|
1227
|
+
return createImageHTML(imglink, isLCP);
|
|
1227
1228
|
});
|
|
1228
1229
|
return content;
|
|
1229
1230
|
}
|
|
1230
1231
|
|
|
1231
1232
|
// src/methods/text.method.ts
|
|
1232
|
-
function text(node, forApp
|
|
1233
|
+
function text(node, forApp) {
|
|
1233
1234
|
if (!node || !node.parentNode) {
|
|
1234
1235
|
return;
|
|
1235
1236
|
}
|
|
1236
|
-
if (
|
|
1237
|
+
if (["a", "code"].includes(node.parentNode.nodeName.toLowerCase())) {
|
|
1237
1238
|
return;
|
|
1238
1239
|
}
|
|
1239
1240
|
const nodeValue = node.nodeValue || "";
|
|
1240
|
-
const linkified = linkify(nodeValue, forApp
|
|
1241
|
+
const linkified = linkify(nodeValue, forApp);
|
|
1241
1242
|
if (linkified !== nodeValue) {
|
|
1242
1243
|
const doc = DOMParser.parseFromString(
|
|
1243
1244
|
`<span class="wr">${linkified}</span>`,
|
|
@@ -1252,7 +1253,7 @@ function text(node, forApp, webp) {
|
|
|
1252
1253
|
}
|
|
1253
1254
|
if (nodeValue.match(IMG_REGEX)) {
|
|
1254
1255
|
const isLCP = false;
|
|
1255
|
-
const imageHTML = createImageHTML(nodeValue, isLCP
|
|
1256
|
+
const imageHTML = createImageHTML(nodeValue, isLCP);
|
|
1256
1257
|
const doc = DOMParser.parseFromString(imageHTML, "text/html");
|
|
1257
1258
|
const replaceNode = doc.body?.firstChild || doc.firstChild;
|
|
1258
1259
|
if (replaceNode) {
|
|
@@ -1264,7 +1265,7 @@ function text(node, forApp, webp) {
|
|
|
1264
1265
|
const e = YOUTUBE_REGEX.exec(nodeValue);
|
|
1265
1266
|
if (e && e[1]) {
|
|
1266
1267
|
const vid = e[1];
|
|
1267
|
-
const thumbnail = proxifyImageSrc(`https://img.youtube.com/vi/${vid.split("?")[0]}/hqdefault.jpg`, 0, 0,
|
|
1268
|
+
const thumbnail = proxifyImageSrc(`https://img.youtube.com/vi/${vid.split("?")[0]}/hqdefault.jpg`, 0, 0, "match");
|
|
1268
1269
|
const embedSrc = `https://www.youtube.com/embed/${vid}?autoplay=1`;
|
|
1269
1270
|
const startTime = extractYtStartTime(nodeValue);
|
|
1270
1271
|
const container = node.ownerDocument.createElement("p");
|
|
@@ -1310,7 +1311,7 @@ function text(node, forApp, webp) {
|
|
|
1310
1311
|
}
|
|
1311
1312
|
|
|
1312
1313
|
// src/methods/traverse.method.ts
|
|
1313
|
-
function traverse(node, forApp, depth = 0,
|
|
1314
|
+
function traverse(node, forApp, depth = 0, state = { firstImageFound: false }, parentDomain = "ecency.com", seoContext) {
|
|
1314
1315
|
if (!node || !node.childNodes) {
|
|
1315
1316
|
return;
|
|
1316
1317
|
}
|
|
@@ -1318,23 +1319,23 @@ function traverse(node, forApp, depth = 0, webp = false, state = { firstImageFou
|
|
|
1318
1319
|
const child = node.childNodes[i];
|
|
1319
1320
|
if (!child) return;
|
|
1320
1321
|
if (child.nodeName.toLowerCase() === "a") {
|
|
1321
|
-
a(child, forApp,
|
|
1322
|
+
a(child, forApp, parentDomain, seoContext);
|
|
1322
1323
|
}
|
|
1323
1324
|
if (child.nodeName.toLowerCase() === "iframe") {
|
|
1324
1325
|
iframe(child, parentDomain);
|
|
1325
1326
|
}
|
|
1326
1327
|
if (child.nodeName === "#text") {
|
|
1327
|
-
text(child, forApp
|
|
1328
|
+
text(child, forApp);
|
|
1328
1329
|
}
|
|
1329
1330
|
if (child.nodeName.toLowerCase() === "img") {
|
|
1330
|
-
img(child,
|
|
1331
|
+
img(child, state);
|
|
1331
1332
|
}
|
|
1332
1333
|
if (child.nodeName.toLowerCase() === "p") {
|
|
1333
1334
|
p(child);
|
|
1334
1335
|
}
|
|
1335
1336
|
const currentChild = node.childNodes[i];
|
|
1336
1337
|
if (currentChild) {
|
|
1337
|
-
traverse(currentChild, forApp, depth + 1,
|
|
1338
|
+
traverse(currentChild, forApp, depth + 1, state, parentDomain, seoContext);
|
|
1338
1339
|
}
|
|
1339
1340
|
});
|
|
1340
1341
|
}
|
|
@@ -1385,7 +1386,7 @@ function fixBlockLevelTagsInParagraphs(html) {
|
|
|
1385
1386
|
html = html.replace(/<p><br>\s*<\/p>/g, "");
|
|
1386
1387
|
return html;
|
|
1387
1388
|
}
|
|
1388
|
-
function markdownToHTML(input, forApp,
|
|
1389
|
+
function markdownToHTML(input, forApp, parentDomain = "ecency.com", seoContext) {
|
|
1389
1390
|
input = input.replace(new RegExp("https://leofinance.io/threads/view/", "g"), "/@");
|
|
1390
1391
|
input = input.replace(new RegExp("https://leofinance.io/posts/", "g"), "/@");
|
|
1391
1392
|
input = input.replace(new RegExp("https://leofinance.io/threads/", "g"), "/@");
|
|
@@ -1445,7 +1446,7 @@ function markdownToHTML(input, forApp, webp, parentDomain = "ecency.com", seoCon
|
|
|
1445
1446
|
output = md.render(input);
|
|
1446
1447
|
output = fixBlockLevelTagsInParagraphs(output);
|
|
1447
1448
|
const doc = DOMParser.parseFromString(`<body id="root">${removeDuplicateAttributes(output)}</body>`, "text/html");
|
|
1448
|
-
traverse(doc, forApp, 0,
|
|
1449
|
+
traverse(doc, forApp, 0, { firstImageFound: false }, parentDomain, seoContext);
|
|
1449
1450
|
output = serializer.serializeToString(doc);
|
|
1450
1451
|
} catch (error) {
|
|
1451
1452
|
try {
|
|
@@ -1458,7 +1459,7 @@ function markdownToHTML(input, forApp, webp, parentDomain = "ecency.com", seoCon
|
|
|
1458
1459
|
});
|
|
1459
1460
|
const repairedHtml = domSerializer(dom.children);
|
|
1460
1461
|
const doc = DOMParser.parseFromString(`<body id="root">${removeDuplicateAttributes(repairedHtml)}</body>`, "text/html");
|
|
1461
|
-
traverse(doc, forApp, 0,
|
|
1462
|
+
traverse(doc, forApp, 0, { firstImageFound: false }, parentDomain, seoContext);
|
|
1462
1463
|
output = serializer.serializeToString(doc);
|
|
1463
1464
|
} catch (fallbackError) {
|
|
1464
1465
|
const escapedContent = he2__default.default.encode(output || md.render(input));
|
|
@@ -1486,18 +1487,18 @@ function cacheSet(key, value) {
|
|
|
1486
1487
|
}
|
|
1487
1488
|
|
|
1488
1489
|
// src/markdown-2-html.ts
|
|
1489
|
-
function markdown2Html(obj, forApp = true,
|
|
1490
|
+
function markdown2Html(obj, forApp = true, _webp = false, parentDomain = "ecency.com", seoContext) {
|
|
1490
1491
|
if (typeof obj === "string") {
|
|
1491
1492
|
const cleanedStr = cleanReply(obj);
|
|
1492
|
-
return markdownToHTML(cleanedStr, forApp,
|
|
1493
|
+
return markdownToHTML(cleanedStr, forApp, parentDomain, seoContext);
|
|
1493
1494
|
}
|
|
1494
|
-
const key = `${makeEntryCacheKey(obj)}-md
|
|
1495
|
+
const key = `${makeEntryCacheKey(obj)}-md-${forApp ? "app" : "site"}-${parentDomain}${seoContext ? `-seo${seoContext.authorReputation ?? ""}-${seoContext.postPayout ?? ""}` : ""}`;
|
|
1495
1496
|
const item = cacheGet(key);
|
|
1496
1497
|
if (item) {
|
|
1497
1498
|
return item;
|
|
1498
1499
|
}
|
|
1499
1500
|
const cleanBody = cleanReply(obj.body);
|
|
1500
|
-
const res = markdownToHTML(cleanBody, forApp,
|
|
1501
|
+
const res = markdownToHTML(cleanBody, forApp, parentDomain, seoContext);
|
|
1501
1502
|
cacheSet(key, res);
|
|
1502
1503
|
return res;
|
|
1503
1504
|
}
|