diff --git a/__snapshots__/mod_test.ts.snap b/__snapshots__/mod_test.ts.snap
index 37a3f94..c71ab9b 100644
--- a/__snapshots__/mod_test.ts.snap
+++ b/__snapshots__/mod_test.ts.snap
@@ -8,3 +8,12 @@ snapshot[`should archive a markdown file with all images 1`] = `
'
`;
+
+snapshot[`should use a fallback image if downloading the image failed 1`] = `
+'# Testing
+
+![]()
+
+
+'
+`;
diff --git a/deno.json b/deno.json
index 671293c..775905f 100644
--- a/deno.json
+++ b/deno.json
@@ -1,7 +1,7 @@
{
"name": "@openformation/markdown-archiver",
"description": "Library for creating self-contained markdown files by embedding images.",
- "version": "1.0.3",
+ "version": "1.0.4",
"exports": "./mod.ts",
"imports": {
"@std/encoding": "jsr:@std/encoding@^0.224.2",
@@ -21,4 +21,4 @@
"__snapshots__"
]
}
-}
+}
\ No newline at end of file
diff --git a/lib/image.ts b/lib/image.ts
index eba6858..7f20bb9 100644
--- a/lib/image.ts
+++ b/lib/image.ts
@@ -28,9 +28,7 @@ const ImageDataUri = Brand.refined(
export function fetchImage(url: string) {
return Effect.gen(function* () {
const response = yield* Effect.tryPromise({
- try: () => {
- return fetch(url);
- },
+ try: () => fetch(url),
catch: (cause: unknown) =>
new FetchImageError(`Failed to fetch image at ${url}: ${cause}`, {
cause,
@@ -38,6 +36,10 @@ export function fetchImage(url: string) {
});
if (!response.ok) {
+ if (response.body) {
+ yield* Effect.promise(response.body.cancel.bind(response.body));
+ }
+
return yield* Effect.fail(
new FetchImageError(
`Failed to fetch image at ${url} as the server responded with a status code of ${response.status}.`,
@@ -76,10 +78,22 @@ export class ImageService extends Context.Tag("ImageService")<
readonly fetch: (
url: string,
) => Effect.Effect;
+ readonly getFallbackImageDataUri: () => Effect.Effect<
+ ImageDataUri
+ >;
}
>() {}
+function getFallbackImageDataUri() {
+ return Effect.succeed(
+ ImageDataUri(
+ "",
+ ),
+ );
+}
+
const ImageServiceServer = ImageService.of({
+ getFallbackImageDataUri,
fetch(url: string) {
return Effect.gen(function* () {
const { buffer, mimeType } = yield* fetchImage(url);
@@ -92,6 +106,7 @@ const ImageServiceServer = ImageService.of({
});
const ImageServiceBrowser = ImageService.of({
+ getFallbackImageDataUri,
fetch(url: string) {
return Effect.gen(function* () {
const { buffer } = yield* fetchImage(url);
diff --git a/lib/mod.ts b/lib/mod.ts
index 23babb3..a3a86cf 100644
--- a/lib/mod.ts
+++ b/lib/mod.ts
@@ -20,19 +20,15 @@ function makeServices() {
return Layer.mergeAll(imageService, markdownService, transformerService);
}
-/**
- * The archiver result type. The effect contains either a `string` with
- * the archived markdown or a `FetchImageError` if fetching an image failed.
- */
-export type Archiver = Effect.Effect;
-
/**
* Takes source markdown contents and makes it self-contained by embedding images.
*
* @param markdown {string} The markdown to archive.
* @returns {string} The markdown with embedded images.
*/
-export function archive(markdown: string): Archiver {
+export function archive(
+ markdown: string,
+): Effect.Effect {
const run = Effect.gen(function* () {
const markdownService = yield* MarkdownService;
diff --git a/lib/transformer.ts b/lib/transformer.ts
index 564567a..04d2f22 100644
--- a/lib/transformer.ts
+++ b/lib/transformer.ts
@@ -1,5 +1,4 @@
import type { Html, Image, Root } from "mdast";
-import type { FetchImageError } from "./image.ts";
import { Context, Effect, Layer } from "effect";
import { visit } from "unist-util-visit";
@@ -11,7 +10,7 @@ export class TransformerService extends Context.Tag("TransformerService")<
{
readonly embedImages: (
ast: Root,
- ) => Effect.Effect;
+ ) => Effect.Effect;
}
>() {}
@@ -19,7 +18,10 @@ function processImage(image: Image) {
return Effect.gen(function* () {
const imageService = yield* ImageService;
- const dataUri = yield* imageService.fetch(image.url);
+ const dataUri = yield* Effect.orElse(
+ imageService.fetch(image.url),
+ () => imageService.getFallbackImageDataUri(),
+ );
image.url = dataUri;
});
@@ -37,7 +39,10 @@ function processHtmlImage({
if (match) {
const url = match[1];
- const dataUri = yield* imageService.fetch(url);
+ const dataUri = yield* Effect.orElse(
+ imageService.fetch(url),
+ () => imageService.getFallbackImageDataUri(),
+ );
node.value = node.value.replace(url, dataUri);
}
diff --git a/mod.ts b/mod.ts
index 5a076bd..1dfdb5a 100644
--- a/mod.ts
+++ b/mod.ts
@@ -1,4 +1 @@
-export type { FetchImageError } from "./lib/image.ts";
-export type { Archiver } from "./lib/mod.ts";
-
export { archive } from "./lib/mod.ts";
diff --git a/mod_test.ts b/mod_test.ts
index f5dd446..ce70df8 100644
--- a/mod_test.ts
+++ b/mod_test.ts
@@ -15,3 +15,17 @@ Deno.test("should archive a markdown file with all images", async (t) => {
await assertSnapshot(t, result);
});
+
+Deno.test("should use a fallback image if downloading the image failed", async (t) => {
+ const program = archive(`
+# Testing
+
+![](https://example.com/not-found.jpg)
+
+
+`);
+
+ const result = await Effect.runPromise(program);
+
+ await assertSnapshot(t, result);
+});