Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support parallel bundle imports in libraries #9156

Merged
merged 2 commits into from
Aug 1, 2023
Merged
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Next Next commit
Support parallel bundle imports in libraries
devongovett committed Jul 26, 2023

Unverified

This user has not yet uploaded their public signing key.
commit c61cb7d508de4a35c18eeea406bdeef00151d348
1 change: 1 addition & 0 deletions packages/bundlers/default/src/DefaultBundler.js
Original file line number Diff line number Diff line change
@@ -505,6 +505,7 @@ function createIdealGraph(
*/
let bundleGroupRootAsset = nullthrows(bundleGroup.mainEntryAsset);
if (
parentAsset.type !== childAsset.type &&
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

entries.has(bundleGroupRootAsset) &&
canMerge(bundleGroupRootAsset, childAsset) &&
dependency.bundleBehavior == null
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
{
"extends": "@parcel/config-default",
"resolvers": ["./ParallelResolver", "..."]
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
const {Resolver} = require('@parcel/plugin');

module.exports = new Resolver({
resolve({specifier}) {
if (specifier === './foo') {
return {
filePath: __dirname + '/foo.js',
priority: 'parallel'
};
}
}
});
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export const foo = 'foo';
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import {foo} from './foo';

export default foo + ' bar';
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
{
"module": "dist/out.js"
}
Empty file.
28 changes: 28 additions & 0 deletions packages/core/integration-tests/test/output-formats.js
Original file line number Diff line number Diff line change
@@ -1463,6 +1463,34 @@ describe('output formats', function () {
assert.deepEqual(calls, [[['a', 10]]]);
});

it('should support external parallel dependencies', async function () {
let b = await bundle(
path.join(__dirname, 'integration/library-parallel-deps/index.js'),
{
mode: 'production',
defaultTargetOptions: {
shouldOptimize: false,
},
},
);

assertBundles(b, [
{
name: 'out.js',
assets: ['index.js'],
},
{
assets: ['foo.js'],
},
]);

let res = await run(b);
assert.equal(res.default, 'foo bar');

let content = await outputFS.readFile(b.getBundles()[0].filePath, 'utf8');
assert(/import \{\s*foo.*\} from "\.\//.test(content));
});

describe('global', function () {
it.skip('should support split bundles between main script and workers', async function () {
let b = await bundle(
43 changes: 32 additions & 11 deletions packages/packagers/js/src/ScopeHoistingPackager.js
Original file line number Diff line number Diff line change
@@ -94,6 +94,7 @@ export class ScopeHoistingPackager {
hoistedRequires: Map<string, Map<string, string>> = new Map();
needsPrelude: boolean = false;
usedHelpers: Set<string> = new Set();
externalAssets: Set<Asset> = new Set();

constructor(
options: PluginOptions,
@@ -130,9 +131,19 @@ export class ScopeHoistingPackager {
this.bundle.env.isLibrary ||
this.bundle.env.outputFormat === 'commonjs'
) {
let bundles = this.bundleGraph.getReferencedBundles(this.bundle);
for (let b of bundles) {
this.externals.set(relativeBundlePath(this.bundle, b), new Map());
for (let b of this.bundleGraph.getReferencedBundles(this.bundle)) {
let entry = b.getMainEntry();
let symbols = new Map();
if (entry && !this.isAsyncBundle && entry.type === 'js') {
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

for now async bundles in libraries are still referenced with parcelRequire. Changing that would require additional changes to the runtimes, but we could fix it eventually.

this.externalAssets.add(entry);

let usedSymbols = this.bundleGraph.getUsedSymbols(entry) || new Set();
for (let s of usedSymbols) {
symbols.set(s, this.getSymbolResolution(entry, entry, s));
}
}

this.externals.set(relativeBundlePath(this.bundle, b), symbols);
}
}

@@ -756,6 +767,13 @@ ${code}
}
}

isWrapped(resolved: Asset, parentAsset: Asset): boolean {
return (
(!this.bundle.hasAsset(resolved) && !this.externalAssets.has(resolved)) ||
(this.wrappedAssets.has(resolved.id) && resolved !== parentAsset)
);
}

getSymbolResolution(
parentAsset: Asset,
resolved: Asset,
@@ -777,13 +795,18 @@ ${code}
return '{}';
}

let isWrapped =
!this.bundle.hasAsset(resolvedAsset) ||
(this.wrappedAssets.has(resolvedAsset.id) &&
resolvedAsset !== parentAsset);
let isWrapped = this.isWrapped(resolvedAsset, parentAsset);
let staticExports = resolvedAsset.meta.staticExports !== false;
let publicId = this.bundleGraph.getAssetPublicId(resolvedAsset);

// External CommonJS dependencies need to be accessed as an object property rather than imported
// directly to maintain live binding.
let isExternalCommonJS =
!isWrapped &&
this.bundle.env.isLibrary &&
this.bundle.env.outputFormat === 'commonjs' &&
!this.bundle.hasAsset(resolvedAsset);

// If the resolved asset is wrapped, but imported at the top-level by this asset,
// then we hoist parcelRequire calls to the top of this asset so side effects run immediately.
if (
@@ -849,7 +872,7 @@ ${code}
return obj;
}
} else if (
(!staticExports || isWrapped || !symbol) &&
(!staticExports || isWrapped || !symbol || isExternalCommonJS) &&
resolvedAsset !== parentAsset
) {
// If the resolved asset is wrapped or has non-static exports,
@@ -887,9 +910,7 @@ ${code}
let hoisted = this.hoistedRequires.get(dep.id);
let res = '';
let lineCount = 0;
let isWrapped =
!this.bundle.hasAsset(resolved) ||
(this.wrappedAssets.has(resolved.id) && resolved !== parentAsset);
let isWrapped = this.isWrapped(resolved, parentAsset);

// If the resolved asset is wrapped and is imported in the top-level by this asset,
// we need to run side effects when this asset runs. If the resolved asset is not