From c101763b330dff3f358e88cfd0c86d5761ba0e1e Mon Sep 17 00:00:00 2001 From: Alex Potsides Date: Thu, 15 Jul 2021 04:37:44 +0100 Subject: [PATCH] fix: work with browserify (#13) Browserify [does not support](https://github.com/browserify/resolve/pull/224) `"exports"` in a `package.json`, and nor can you use `"browser"` in `package.json` [as a hint to Browserify to look in a certain place for a file](https://github.com/browserify/resolve/issues/250#issuecomment-880029677). This means the output of `ipjs` is not compatible with Browserify since it expects the runtime/bundler to use the `"exports"` field to look up files paths. If Browserify finds a file path, it will then use the `"browser"` values to apply any overrides, which `ipjs` uses to direct it to the `/cjs` folder. The problem is if it can't find the file in the first place it won't use the `"browser"` map to get the overrides. Handily we're generating the `"browser"` field values from the `"exports"` values so we know we have the complete set of files that the user wants to expose to the outside world, and the paths we want people to use to access them. The change in this PR is to use the `"browser"` field values to [mimc the `"exports"` structure in a CJS-compatible directory structure](https://github.com/browserify/resolve/issues/250#issuecomment-879241002) as per @ljharb's suggestion. For any file that we are overriding with `"browser"` values, we create an empty file (where a resolvable file does not already exist) a the path Browserify expects it to be at, then it'll dutifully use the `"browser"` field to pull the actual file in. --- src/package/index.js | 36 +++++++++++++++++-- .../pkg-kitchensink/output-main/index.js | 0 .../pkg-kitchensink/output-main/secondary | 0 .../pkg-kitchensink/output-notests/index.js | 0 .../pkg-kitchensink/output-notests/secondary | 0 .../pkg-kitchensink/output-tests/index.js | 0 .../pkg-kitchensink/output-tests/secondary | 0 7 files changed, 34 insertions(+), 2 deletions(-) create mode 100644 test/fixtures/pkg-kitchensink/output-main/index.js create mode 100644 test/fixtures/pkg-kitchensink/output-main/secondary create mode 100644 test/fixtures/pkg-kitchensink/output-notests/index.js create mode 100644 test/fixtures/pkg-kitchensink/output-notests/secondary create mode 100644 test/fixtures/pkg-kitchensink/output-tests/index.js create mode 100644 test/fixtures/pkg-kitchensink/output-tests/secondary diff --git a/src/package/index.js b/src/package/index.js index e3928a1..48bf3e8 100644 --- a/src/package/index.js +++ b/src/package/index.js @@ -1,10 +1,10 @@ -import { promises as fs } from 'fs' +import { promises as fs, existsSync } from 'fs' import { rollup } from 'rollup' import file from './file.js' import testFile from './testFile.js' import path from '../path-to-url.js' import { fileURLToPath } from 'url' -import { join } from 'path' +import { join, dirname } from 'path' import rmtree from '@tgrajewski/rmtree' import preserveShebangs from 'rollup-plugin-preserve-shebangs' @@ -163,6 +163,37 @@ class Package { await unlink(new URL(dist + '/cjs/_ipjsInput.js')) } + async stubFiles (dist, files) { + await Promise.all( + files.map(async (file) => { + if (file === '.') { + file = 'index.js' + } + if (file.startsWith('./')) { + file = file.substring(2) + } + const dir = dirname(file) + if (dir !== '.') { + try { + await mkdir(new URL(dist + '/' + dir), { + recursive: true + }) + } catch (err) { + if (err.code !== 'EEXIST') { + throw err + } + } + } + + if (existsSync(new URL(dist + '/' + file))) { + return + } + + await writeFile(new URL(dist + '/' + file), '') + }) + ) + } + async deflate (dist) { if (!(dist instanceof URL)) dist = path(dist) rmtree(fileURLToPath(dist)) @@ -210,6 +241,7 @@ class Package { json.exports = json.exports.import json.browser = json.browser.import } + await this.stubFiles(dist, Object.keys(json.browser)) let files = Promise.all(pending) pending.push(writeFile(new URL(dist + '/package.json'), JSON.stringify(json, null, 2))) const typeModule = { diff --git a/test/fixtures/pkg-kitchensink/output-main/index.js b/test/fixtures/pkg-kitchensink/output-main/index.js new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/pkg-kitchensink/output-main/secondary b/test/fixtures/pkg-kitchensink/output-main/secondary new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/pkg-kitchensink/output-notests/index.js b/test/fixtures/pkg-kitchensink/output-notests/index.js new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/pkg-kitchensink/output-notests/secondary b/test/fixtures/pkg-kitchensink/output-notests/secondary new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/pkg-kitchensink/output-tests/index.js b/test/fixtures/pkg-kitchensink/output-tests/index.js new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/pkg-kitchensink/output-tests/secondary b/test/fixtures/pkg-kitchensink/output-tests/secondary new file mode 100644 index 0000000..e69de29