This moved to an ember-addon. Please check ember-cli-bundle-loader instead.
This is a demo and reference implementation for a way of building Ember Apps prior by breaking app.[js|css] into smaller packages that can be lazy loaded. For now, the demo only does the different packages, but will be expanded soon to also do lazy loading.
Note: there is a ton of overlap with the work being done for * ember engines. While you might find this interesting, I strongly recommend you NOT to use these techniques and instead wait for ember-engines unless you're really desperate (like I am) to get lazy loading.
The typical output of an Ember App looks something like:
- dist/
- index.html
- assets/ app.js app.css vendor.js vendor.css
- dist/
- index.html
- assets/ boot.js boot.css package1.js package1.css package2.js package2.css vedor.js* vendor.csss
* In the future we might also break vendor into different packages. For example:
- vendor-stable, to get better caching of vendor libs rarely revved, like Ember or jQuery.
- vendor-boot, the minimum necessary to boot the app. Hopefully, this file would be empty and boot only requires vendor-stable.
- vendor, everything else.
The vendor split is important as well since is common to have an internal addon that is really volatile and would invalidate your cache during daily deployments.
Note this demo doesn't do the vendor split yet.
This demo tweaks ember-cli-build to create multiple EmberApps for each package, each of them will have its output. For each package, it also changes certain to avoid unnecessary processing of addons to get faster builds for packages.
At the top level, this is the traditional ember-app
which is responsible for booting the app. Only put here anything that is strictly required to get to the first page or maybe even just the shell of your app or whatever is more important for a first time load. For example, your login page, the navbar, the default landing page. Each app will have different requirements, but the guidance is to keep it small. Initializers also have to be here since packages won't be available when we create the app. We could, in theory, lazy load initializers, but that doesn't make a lot of sense, so I have not tested them.
At the top level, we have a new "packages" folder. In the future, they might become engines and the idea is to move away from this structure into whatever the engine structure ends up being. Each package is an ember-app, with the exception of initializers.
Tests. Currently, all tests live under the traditional folders. However, in the future, tests for a package will move under each package folder. (Future: packages/my-package/tests
and packages/my-package/app
).
The config is at the top level and it's shared. All packages can access it by using {modulePrefix}/config
.
All the JS modules are namespaced with the package name. For example, the route defined in package1/routes/package1.js
is defined as define('package1/routes/package1'
in the output code. This is because that's simply the name of the app when building. That gives us the advantage of being explicit about the way we consume and refer in code. However, this also requires a new resolver that can lookup for routes under the packages see app/resolvers/packages.js
for more info. Please be aware of this namespace, it's especially important for utils
or any other code that you import, for a hypothetical date-formatter utility under package1/utils/date-formatter
will be imported as import dateFormatter from 'package1/utils/date-formatter'
. Please be aware of this for tests.
The config/environment lives under [modulePrefix]/config/environment
, which means it has the same path for every package. This is fine in most cases except if you're planning to share your packages across applications.
This demo uses ember-cli-sass, but the same technique would work with vanilla CSS (but please use a pre-processor) or Less. Is not tested with postCSS, but it isn't tested.
This is just a demo and I won't get to write a generator, but if this was a real thing we would be able to ember generate package new-package-name
. Engines will do that and that's the path forward, so I won't invest on it for packages. For now, just create a folder and start writing components, routes, etc. Packages get auto-resolved.
[TODO:] this will be implemented soon. I recommend splitting the packages based on functional areas of your application or volatility. While that makes sense when distributing work across teams, it might not be the best way to serve the assets. There's an additional bundle step that will grab the output of two packages and concatenate them together. There is a configuration file, config/bundles.js
which also generates metadata for the lazy-loader to know what bundle to load for each route. Each bundle has a name, the list of packages that will be concatenated together, a list of routes that it handles and a list of dependent bundles. See cofig/bundles.js
for more details about how to use them. For now, we only support a single package per bundle (we don't concatenate yet) and we don't auto-load dependencies.
git clone <repository-url>
this repository- change into the new directory
npm install
bower install
ember server
- Visit your app at http://localhost:4200.
ember test
ember test --server
ember build
(development)ember build --environment production
(production)
This is just a demo for packaging ember apps, for now, it's not deployed anywhere, please run it locally.