diff --git a/src/index.mjs b/src/index.mjs index bc4f308..fe2178c 100644 --- a/src/index.mjs +++ b/src/index.mjs @@ -109,6 +109,13 @@ function astroRehypeRelativeMarkdownLinks(opts = {}) { const collectionName = path .dirname(relativeToContentPath) .split(FILE_PATH_SEPARATOR)[0]; + // if linked file is outside of a collection directory + if ( + collectionName === ".." || + (collectionPathMode !== "root" && collectionName === ".") + ) { + return; + } const collectionPathSegment = collectionPathMode === "root" ? PATH_SEGMENT_EMPTY : collectionName; // determine the path of the target file relative to the collection diff --git a/src/index.test.mjs b/src/index.test.mjs index 7acb3d9..c2038c0 100644 --- a/src/index.test.mjs +++ b/src/index.test.mjs @@ -31,10 +31,13 @@ import astroRehypeRelativeMarkdownLinks from "./index.mjs"; - https://github.com/nodejs/node/issues/51164#issuecomment-2034518078 */ +/** @param {Record { - visit(tree, "element", (node) => { - const fileInHistory = path.resolve(__dirname, __filename); + visit(tree, "element", () => { + const fileInHistory = options.currentFilePath + ? path.resolve(options.currentFilePath) + : path.resolve(__dirname, __filename); if (!file.history.includes(fileInHistory)) { file.history.push(fileInHistory); @@ -373,6 +376,38 @@ describe("astroRehypeRelativeMarkdownLinks", () => { }); }); + describe("collection names", () => { + test("should not transform path outside of a content collection directory and the content directory", async () => { + const input = 'foo'; + const { value: actual } = await rehype() + .use(testSetupRehype) + .use(astroRehypeRelativeMarkdownLinks, { + contentPath: "src/fixtures/dir-test", + }) + .process(input); + + const expected = + 'foo'; + + assert.equal(actual, expected); + }); + + test("should not transform path outside of a content collection directory and in the content directory", async () => { + const input = 'foo'; + const { value: actual } = await rehype() + .use(testSetupRehype, { + currentFilePath: "./src/fixtures/dir-test/index.md", + }) + .use(astroRehypeRelativeMarkdownLinks, { contentPath: "src/fixtures" }) + .process(input); + + const expected = + 'foo'; + + assert.equal(actual, expected); + }); + }); + describe("config option validation", () => { const runValidationTest = async (context, options) => { const validateOptionsMock = context.mock.fn(validateOptionsOriginal); @@ -549,6 +584,24 @@ describe("astroRehypeRelativeMarkdownLinks", () => { assert.equal(actual, expected); }); + + test("should transform path in content directory when content path same as collection path", async () => { + const input = 'foo'; + const { value: actual } = await rehype() + .use(testSetupRehype, { + currentFilePath: "./src/fixtures/dir-test/index.md", + }) + .use(astroRehypeRelativeMarkdownLinks, { + contentPath: "src/fixtures", + collectionPathMode: "root", + }) + .process(input); + + const expected = + 'foo'; + + assert.equal(actual, expected); + }); }); describe("config option - trailingSlash", () => {