Useful tools for remark
and rehype
.
npm i bityoungjae-remark-helper
Extract all image URLs from markdown content.
type getAllImageUrls = (markdown: string) => string[];
import { getAllImageUrls } from "bityoungjae-remark-helper";
const markdown = `data:image/s3,"s3://crabby-images/87b3b/87b3b1e5827ee6fd55ddbdf6267bc648c6850751" alt="image info"
data:image/s3,"s3://crabby-images/342fc/342fc5b66c9025c64f5850f6e8155f79857f83dd" alt="image info"
data:image/s3,"s3://crabby-images/a64cf/a64cfaaae2cc1cedc87b28954523858d576661a0" alt="image info"\n`;
test("getAllImageUrls", () => {
const urlList = getAllImageUrls(markdown);
expect(urlList).toContain("./pictures/image1.png");
expect(urlList).toContain("./pictures/image2.png");
expect(urlList).toContain("./pictures/image3.png");
});
Remark plugin for replacing image URLs in markdown
interface IURLMap {
[key: string]: string;
}
type remarkImageUrlReplacer = (urlMap: IURLMap) => Plugin;
import { unified } from "unified";
import remarkParse from "remark-parse";
import remarkStringify from "remark-stringify";
import { remarkImageUrlReplacer } from "bityoungjae-remark-helper";
const markdown = `data:image/s3,"s3://crabby-images/87b3b/87b3b1e5827ee6fd55ddbdf6267bc648c6850751" alt="image info"
data:image/s3,"s3://crabby-images/342fc/342fc5b66c9025c64f5850f6e8155f79857f83dd" alt="image info"
data:image/s3,"s3://crabby-images/a64cf/a64cfaaae2cc1cedc87b28954523858d576661a0" alt="image info"\n`;
const urlMap = {
"./pictures/image1.png": "https://image.com/image1.jpg",
"./pictures/image3.png": "https://image.com.image4.png",
};
const result = `data:image/s3,"s3://crabby-images/17180/17180893a58b1aefc587a7d7d0c7f1a414bc818d" alt="image info"
data:image/s3,"s3://crabby-images/342fc/342fc5b66c9025c64f5850f6e8155f79857f83dd" alt="image info"
data:image/s3,"s3://crabby-images/6c78a/6c78ac0d6804a2e444d57c4dd15a86a8d862f005" alt="image info"\n`;
test("remarkImageUrlReplacer", async () => {
expect(
String(
await unified()
.use(remarkParse)
.use(remarkImageUrlReplacer(urlMap))
.use(remarkStringify)
.process(markdown),
),
).toBe(result);
});
Extract all MediaUrlInfo from markdown content. MediaUrlInfo data has more specific pieces of information than just URL
interface IMediaUrlInfo {
type: "video" | "image" | "unknown";
url: string;
}
type getAllMediaUrlInfos = (markdown: string) => IMediaUrlInfo[];
import { getAllMediaUrlInfos } from "bityoungjae-remark-helper";
const markdown = `
data:image/s3,"s3://crabby-images/c8e8d/c8e8dd4e104f8d4ba227711fa3f62205715aefdf" alt="alt"
<video src="images/img_5.mp4" style="width:100%" muted autoplay loop playsinline></video>
<img src="images/img_3.png" />
<video
src="images/img_6.mp4"
style="width:100%" muted autoplay loop playsinline>
</video>
<video>
<unknownTag></unknownTag>\n`;
test("getAllMediaUrlInfos", () => {
const mediaUrlInfos = getAllMediaUrlInfos(markdown);
expect(mediaUrlInfos.length).toBe(4);
expect(mediaUrlInfos[0]).toEqual({ type: "image", url: "abc.jpg" });
expect(mediaUrlInfos[1]).toEqual({ type: "video", url: "images/img_5.mp4" });
expect(mediaUrlInfos[2]).toEqual({ type: "image", url: "images/img_3.png" });
expect(mediaUrlInfos[3]).toEqual({ type: "video", url: "images/img_6.mp4" });
});
Similar to getAllImageUrls
, but getAllMediaUrls
extracts html <image>
<video>
urls as well as markdown image urls.
type getAllMediaUrls = (markdown: string) => string[];
import { getAllMediaUrls } from "bityoungjae-remark-helper";
const markdown = `
data:image/s3,"s3://crabby-images/c8e8d/c8e8dd4e104f8d4ba227711fa3f62205715aefdf" alt="alt"
<video src="images/img_5.mp4" style="width:100%" muted autoplay loop playsinline></video>
<img src="images/img_3.png" />
<video
src="images/img_6.mp4"
style="width:100%" muted autoplay loop playsinline>
</video>
<video>
<unknownTag></unknownTag>\n`;
test("getAllMediaUrls", () => {
const urlList = getAllMediaUrls(markdown);
expect(urlList.length).toBe(4);
expect(urlList).toContain("abc.jpg");
expect(urlList).toContain("images/img_5.mp4");
expect(urlList).toContain("images/img_3.png");
expect(urlList).toContain("images/img_6.mp4");
});
interface IURLMap {
[key: string]: string;
}
type replaceAllMediaUrlsFromHTML = (urlMap: IURLMap, html: string) => string;
test("replaceAllMediaUrlsFromHTML", () => {
const html = `<video src="videos/old.mp4"></video>\n<img src="images/old.png" alt="old image">`;
const result = `<video src="videos/new.mp4"></video>\n<img src="images/new.png" alt="old image">`;
expect(
replaceAllMediaUrlsInHTML(
{
"videos/old.mp4": "videos/new.mp4",
"images/old.png": "images/new.png",
},
html,
),
).toBe(result);
});
Similar to remarkImageUrlReplacer
, but remarkMediaUrlReplacer
replace html <image>
<video>
urls as well as markdown image urls.
interface IURLMap {
[key: string]: string;
}
type remarkMediaUrlReplacer = (urlMap: IURLMap) => Plugin;
test("remarkMediaUrlReplacer", async () => {
const markdown = `
data:image/s3,"s3://crabby-images/98c3e/98c3e0a66513a253116716d61eea03325591d5db" alt="alt text"
<video src="videos/old-1.mp4" autoplay></video>
<img src="images/old-2.png">
<video src="videos/old-2.mp4"></video>`;
const result = `
data:image/s3,"s3://crabby-images/1d2e8/1d2e88528618dbe8a481c88c339f4a63a6efba09" alt="alt text"
<video src="videos/new-1.mp4" autoplay></video>
<img src="images/new-2.png">
<video src="videos/new-2.mp4"></video>`;
expect(
normalizeSpaces(
String(
await unified()
.use(remarkParse)
.use(
remarkMediaUrlReplacer({
"images/1.jpg": "images/new-1.jpg",
"images/2.png": "images/new-2.png",
"videos/1.mp4": "videos/new-1.mp4",
"videos/2.mp4": "videos/new-2.mp4",
}),
)
.use(remarkStringify)
.process(markdown),
),
),
).toBe(result);
});
interface IURLMap {
[key: string]: string;
}
type replaceAllMediaUrlsInMarkdown = (
urlMap: IURLMap,
markdown: string,
) => string;
test("replaceAllMediaUrlsInMarkdown", async () => {
const markdown = `
data:image/s3,"s3://crabby-images/98c3e/98c3e0a66513a253116716d61eea03325591d5db" alt="alt text"
<video src="videos/old-1.mp4" autoplay></video>
<img src="images/old-2.png">
<video src="videos/old-2.mp4"></video>`;
const result = `
data:image/s3,"s3://crabby-images/1d2e8/1d2e88528618dbe8a481c88c339f4a63a6efba09" alt="alt text"
<video src="videos/new-1.mp4" autoplay></video>
<img src="images/new-2.png">
<video src="videos/new-2.mp4"></video>`;
expect(
normalizeSpaces(
replaceAllMediaUrlsInMarkdown(
{
"images/1.jpg": "images/new-1.jpg",
"images/2.png": "images/new-2.png",
"videos/1.mp4": "videos/new-1.mp4",
"videos/2.mp4": "videos/new-2.mp4",
},
markdown,
),
),
).toBe(result);
});