Skip to content

Commit

Permalink
feat: adds fallback image
Browse files Browse the repository at this point in the history
  • Loading branch information
akoenig committed May 31, 2024
1 parent fb3c306 commit 306cb64
Show file tree
Hide file tree
Showing 7 changed files with 55 additions and 19 deletions.
9 changes: 9 additions & 0 deletions __snapshots__/mod_test.ts.snap
Original file line number Diff line number Diff line change
Expand Up @@ -8,3 +8,12 @@ snapshot[`should archive a markdown file with all images 1`] = `
<img src="" alt="Google Logo 2006" />
'
`;

snapshot[`should use a fallback image if downloading the image failed 1`] = `
'# Testing
![]()
<img src="" />
'
`;
4 changes: 2 additions & 2 deletions deno.json
Original file line number Diff line number Diff line change
@@ -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",
Expand All @@ -21,4 +21,4 @@
"__snapshots__"
]
}
}
}
21 changes: 18 additions & 3 deletions lib/image.ts
Original file line number Diff line number Diff line change
Expand Up @@ -28,16 +28,18 @@ const ImageDataUri = Brand.refined<ImageDataUri>(
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,
}),
});

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}.`,
Expand Down Expand Up @@ -76,10 +78,22 @@ export class ImageService extends Context.Tag("ImageService")<
readonly fetch: (
url: string,
) => Effect.Effect<ImageDataUri, FetchImageError>;
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);
Expand All @@ -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);
Expand Down
10 changes: 3 additions & 7 deletions lib/mod.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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<string, FetchImageError, never>;

/**
* 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<string> {
const run = Effect.gen(function* () {
const markdownService = yield* MarkdownService;

Expand Down
13 changes: 9 additions & 4 deletions lib/transformer.ts
Original file line number Diff line number Diff line change
@@ -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";
Expand All @@ -11,15 +10,18 @@ export class TransformerService extends Context.Tag("TransformerService")<
{
readonly embedImages: (
ast: Root,
) => Effect.Effect<void, FetchImageError, ImageService>;
) => Effect.Effect<void, never, ImageService>;
}
>() {}

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;
});
Expand All @@ -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);
}
Expand Down
3 changes: 0 additions & 3 deletions mod.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1 @@
export type { FetchImageError } from "./lib/image.ts";
export type { Archiver } from "./lib/mod.ts";

export { archive } from "./lib/mod.ts";
14 changes: 14 additions & 0 deletions mod_test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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)
<img src="https://example.com/not-found.jpg" />
`);

const result = await Effect.runPromise(program);

await assertSnapshot(t, result);
});

0 comments on commit 306cb64

Please sign in to comment.