Lazy Loading
Note: Spartacus 4.x is no longer maintained. Please upgrade to the latest version.
Note: Spartacus 4.x was tested with SAP Commerce Cloud versions 1905 to 2205. Spartacus 4.x has not been verified to work with (and is not guaranteed to work with) SAP Commerce Cloud 2211 or later releases.
Lazy loading, also known as code splitting, lets you divide your JavaScript code into multiple chunks. The result is that you do not have to load all the JavaScript of the full application when a user accesses the first page. Instead, only the chunks that are required for the given page are loaded. While navigating the storefront, additional chunks are loaded when needed.
Such an approach can substantially improve “Time To Interactive”, especially in the case of complex web applications being accessed by low-end mobile devices.
Table of Contents
- Spartacus Approach to Lazy Loading
- General Concepts
- Lazy Loading of CMS Components
- Lazy Loading of Modules
- Customizing Lazy Loaded Modules
- Preparing Libraries to Work with Lazy Loading
Spartacus Approach to Lazy Loading
Code splitting is a technique that has to be done at application build time. Code splitting provided by Angular is typically route-based, which means there is a chunk for the landing page, another chunk for the product page, and so on. Since Spartacus is mostly CMS driven, the actual application logic for each route cannot be decided at build time. Business users will eventually alter the page structure by introducing or removing components. This is why an alternative approach to lazy loading is required, which Spartacus provides in the following ways:
- Lazy loading of CMS components
- CMS-driven lazy loading of feature modules
General Concepts
The following sections offer some important information about how lazy loading works in Spartacus.
Defining Dynamic Imports Only in the Main Application
Dynamic imports, a technique that is used to facilitate lazy loading and that also allows code splitting, can only be used in the main application. It is not possible to define dynamic imports in the prebuilt libraries.
This is an unfortunate limitation, which results in some application code that must be added by customers. Although the amount of custom code is limited to the bare minimum, we’ll add a feature in a future version of the schematics library to automatically add lazy-loaded modules.
Avoiding Static Imports for Lazy-Loaded Code
To make code spitting possible, your static JavaScript code (the main app bundle) should not have any static imports to code that you want to lazy load. The builder will notice that the code is already included, and as a result, will not generate a separate chunk for it. This is especially important in the case of importing symbols from libraries.
At the time of writing (Angular 9 and Angular 10), mixing static imports with dynamic imports for the same library entry point, even for distinct symbols, will break lazy loading and tree shaking for this library entry point. If you were to do this, it would statically include the entire entry point in the build. For this reason, it is highly recommended that you create specific entry point for the code that has to be loaded statically, and separate entry points for the code that can be loaded lazily.
Configuration in Lazy-Loaded Modules
Note: This functionality is introduced with version 3.0 of the Spartacus libraries.
If additional configuration is provided inside the lazy-loaded module, Spartacus merges it into the global application configuration to support lazy loading scenarios for existing components and services. In most cases, especially when lazy-loaded modules provide mostly default configurations, this works reliably. However, it can cause problems if it is overused, especially when two modules provide different configurations for the same part of the config. Scenarios such as these can be fixed by providing the necessary overrides in the main app.
This merging functionality is made possible by a compatibility mechanism that is enabled by default, but you can disable it with the disableConfigUpdates
feature flag. If you are developing new modules that have to hook into configurations from lazy-loaded modules, you should use the ConfigurationService.unifiedConfig$
instead. This functionality is described in the following section.
Unified Configuration
Unified configuration provides a way to get a global configuration that includes both the root configuration and the configuration from already-loaded lazy-loaded modules.
The ConfigurationService.unifiedConfig$
exposes the unified configuration as an observable that emits the new configuration each time it changes. This happens, for example, every time a lazy-loaded module with a provided configuration is loaded and instantiated.
All configuration parts are merged in a strict order, where the actual configuration always overrides the default one, and the configuration defined in the root module (that is, the app shell) has precedence.
The following is an example that shows the order in which different configurations provided in the root app and in two lazy-loaded modules are merged, where each subsequent item in the list can override the previous one:
- default root configuration
- default configuration from lazy-loaded module 1
- default configuration from lazy-loaded module 2
- configuration from lazy-loaded module 1
- configuration from lazy-loaded module 2
- root configuration (always takes precedence)
Providers in Lazy-Loaded Modules
Injection tokens provided in lazy-loaded modules are not visible to services provided in the root application. This applies especially to multi-provided tokens, such as HttpInterceptors
, various handlers, and so on.
To mitigate this drawback, some Spartacus features, such as the PageMetaService
(which consumes PageMetaResolver
tokens), or the ConverterService
(which mostly consumes adapter serializers and normalizers), use the unified injector under the hood. In doing so, they have access to lazy-loaded tokens and can leverage them for global features.
For mechanisms that do not rely on the unified injector (for example, functionality from most non-Spartacus libraries, such as the core Angular libraries), it is recommended that you always eager load modules with these kinds of tokens.
For more information about eager loading, see the Angular documentation glossary.
Unified Injector
The unified injector provides a way to inject a token, or multi-provided tokens, while taking into account the root injector and the injectors from lazy-loaded features. The injector exposes an observable that emits a new set of injectables for a specified token each time the status of the unified injector changes.
Avoiding Importing the HttpClientModule in Your Lazy-Loaded Modules
In general, the HttpClientModule
should be imported in the root application, not in the library. For example, if you import it inside a lazy-loaded library, all injectors from the root library will be invisible to HTTP calls originating from the lazy-loaded module.
Although technically it is possible to import the HttpClientModule
in the library, in most cases it is not something that is expected, and it might cause errors that are difficult to explain, so please keep this in mind.
Lazy Loading of CMS Components
Note: This feature is introduced with version 2.0 of the Spartacus libraries.
Configuration of Lazy Loading CMS Components
Lazy loading of CMS code is achieved by specifying a dynamic import in place of a statically-referenced components class in the CMS Mapping configuration. The following is an example:
{
cmsComponents: {
SimpleResponsiveBannerComponent: {
component: () => import('./lazy/lazy.component').then(m => m.LazyComponent)
}
}
}
For more information on CMS mapping, see Customizing CMS Components.
Technical Details
Support for dynamic imports in CMS component mapping is implemented using customizable component handlers (specifically, the LazyComponentHandler
).
It is possible to extend this handler to customize its behavior, to add special hooks, or different triggers, or to implement a completely new handler that can optionally reuse existing ones.
Lazy Loading of Modules
Note: This feature is introduced with version 2.1 of the Spartacus libraries.
CMS-driven lazy loading of feature modules allows the following:
- Lazy loading not only component code, but also the core part (including NgRx state)
- Loading a feature only once, when first needed
- Providing shared, lazy-loaded dependency modules
Lazy loading of the feature module is triggered by the CMS requesting a component when the implementation is covered by the relevant feature configuration.
Configuration of Lazy-Loaded Modules
Configuration of the feature involves the following aspects:
- The dynamic import of the feature module (which has to be defined in main application)
-
The information about what CMS components are covered by a specific feature (which can become part of the library and imported statically). This information is provided as an array of strings under the
cmsComponents
key. The following is an example:{ featureModules: { organization: { module: () => import('@spartacus/my-account/organization').then( (m) => m.OrganizationModule ), cmsComponents: [ 'OrderApprovalListComponent', 'ManageBudgetsListComponent', 'ManageCostCentersListComponent', 'ManagePermissionsListComponent', 'ManageUnitsListComponent', 'ManageUserGroupsListComponent', 'ManageUsersListComponent', ], }, }, }
Component Mapping Configuration in Lazy-Loaded Modules
Default CMS mapping configuration in lazy-loaded modules should be defined in exactly the same fashion as in statically-imported ones.
Spartacus is able to extract a CMS component mapping configuration from a lazy-loaded feature and use it to resolve component classes and factories covered by the feature. That is why it is possible and recommended to use the standard way of providing the default CMS mapping configuration inside a lazy-loaded module. As a result, the exact same module and library entry point can be imported dynamically or statically, depending on the needs, and it is still possible to overwrite a lazy-loaded CMS configuration from the application level by providing configuration overrides in the application.
Defining Shared-Dependency Modules
It is possible to extract some logic to a shared, lazy-loaded module that can be defined as a lazy-loaded dependency for a feature module by providing an array of dynamic imports in the dependencies
property of the feature configuration. The following is an example:
{
featureModules: {
organization: {
module: () =>
import('@spartacus/my-account/organization').then(
(m) => m.OrganizationModule
),
dependencies: [
() =>
import('@spartacus/storefinder/core').then(
(m) => m.OrganizationModule
),
// ,,
],
},
},
}
Such unnamed dependency modules are instantiated only once, when lazy loading the first feature that depends on it. Its providers contribute to the combined injector that is passed to the feature module, and as a result, all feature services and components have access to the services provided by the dependency modules.
Dependency Aliases
For greater flexibility, it is possible to use string aliases for both dependencies and for whole features. If one feature depends on another one, you can reference it as a dependency, as shown in the following example:
{
featureModules: {
featureOne: {
module: () =>
import('./feature/feature-one.module.ts').then(
(m) => m.FeatureOneModule
),
},
featureTwo: {
module: () =>
import('./feature/feature-two.module.ts').then(
(m) => m.FeatureTwoModule
),
dependencies: ['featureOne']
},
},
}
You can also alias the entire feature directly, as shown in the following example:
{
featureModules: {
featureOne: {
module: () =>
import('./feature/feature-one.module.ts').then(
(m) => m.FeatureOneModule
),
},
featureTwo: 'featureOne'
},
}
In this case, featureTwo
is just an alias of featureOne
, so the same code is loaded and initialized only once, whenever either of these features is requested. This mechanism is used by default in Spartacus libraries to expose components and the core in one lazy-loaded chunk, without sacrificing the possibility to easily split them up through simple configuration.
The following two examples are effectively equivalent, with the difference being the number of dynamic imports (and as a result, the number of lazy loaded chunks), and the ability to customize or replace selected parts by configuration only.
The following is an example of the default approach:
{
featureModules: {
[USER_ACCOUNT_FEATURE]: {
module: () =>
import('@spartacus/user/account').then((m) => m.UserAccountModule),
},
// cmsComponents is a part of default config, provided by UserAccountRootModule
// and usually can be ommited
cmsComponents: [
'LoginComponent',
'ReturningCustomerLoginComponent',
'ReturningCustomerRegisterComponent',
],
},
// Core to Feature alias is a part of default config, provided by UserAccountRootModule
// and usually can be ommited
[USER_ACCOUNT_CORE_FEATURE]: USER_ACCOUNT_FEATURE, // by default core is bundled together with components
}
The following is an example of a decoupled approach that is easier to customize:
{
featureModules: {
[USER_ACCOUNT_FEATURE]: {
module: () =>
import('@spartacus/user/account/components').then((m) => m.UserAccountComponentsModule),
},
// cmsComponents is a part of default config, bundled with UserAccountRootModule
// and usually can be ommited
cmsComponents: [
'LoginComponent',
'ReturningCustomerLoginComponent',
'ReturningCustomerRegisterComponent',
],
},
[USER_ACCOUNT_CORE_FEATURE]: {
module: () =>
import('@spartacus/user/account/core').then((m) => m.UserAccountCoreModule),
dependencies: [
() =>
import('@spartacus/user/account/occ').then(
(m) => m.UserAccountOccModule
),
]
},
}
Exposing Smart Proxy Facades From lazy loaded Features
Proxy facades offer a flexible way to expose core functionality from lazy-loaded feature modules in such a way that components consuming those facades do not have to know if it is lazy loaded and needs to be initialized, or not.
Any access to a method or property of a proxy facade triggers the lazy loading and initialization of all related features. For more information, see Proxy Facades.
Combined Injector
Any lazy-loaded module can inject (that is, have access to) services and tokens from the root application injector and the dependency modules injectors. This is possible because of the CombinedInjector
that is created each time the feature module with dependencies is instantiated.
When a CMS component that is covered by a lazy-loaded module is instantiated, it can inject (that is, have access to) services from the following:
- The
ModuleInjector
hierarchy, starting from the feature module injector, and including dependency modules and the root injector - The
ElementInjector
hierarchy, which is created implicitly at each DOM element
Initializing Lazy Loaded Modules
Note: This functionality is introduced with version 3.2 of the Spartacus libraries.
Spartacus provides a MODULE_INITIALIZER
that should be used instead of the Angular APP_INITIALIZER
for initializing lazy-loaded modules. The APP_INITIALIZER
mechanism finishes initializing the application before any lazy loading occurs, so lazy-loaded features that may need to run initialization logic when they are loaded are unable to do so.
The MODULE_INITIALIZER
injection token can be used to provide initialization functions in modules that are intended to be lazy loaded. The MODULE_INITIALIZER
is supported by the Spartacus lazy loading mechanism, and as a result, initialization functions that are provided using the MODULE_INITIALIZER
will run just before the module they are defined in is lazy loaded.
You configure a MODULE_INITIALIZER
in the same way that you would configure an APP_INITIALIZER
. The following is an example:
...
import { MODULE_INITIALIZER } from '@spartacus/core';
...
export function myFactoryFunction(
dependencyOne: DependencyOne
) {
const result = () => {
// add initialization logic here
};
return result;
}
@NgModule({
providers: [
{
provide: MODULE_INITIALIZER,
useFactory: myFactoryFunction,
deps: [DependencyOne],
multi: true,
},
],
})
export class MyLazyLoadedModule {}
Although the primary use case for the MODULE_INITIALIZER
is to run when a module is lazy loaded, the MODULE_INITIALIZER
function still runs if the module it is defined in is configured to be loaded eagerly (that is, loaded when the app starts, which is the default method for importing modules). If an eager-loading configuration is applied on a module that is designed for lazy loading, the MODULE_INITIALIZER
functions are run during the Angular app bootstrap sequence.
An initialization function that is provided by the MODULE_INITIALIZER
only runs when the module is loaded. This means that if a module is lazy loaded, the initialization function will run before the module is loaded, but it will not run when the app is initialized. If a module is configured for eager loading, the initialization function will only run when the app is initialized.
If any of the initialization functions returns a promise, app initialization or module loading does not complete until the promise is resolved. If one promise is rejected, the app initialization or the module loading is interrupted. If an initialization function throws an error, the app initialization or module loading is also interrupted.
If a feature needs to apply initialization logic at the moment the app is loaded, feature libraries can still use the regular APP_INITIALIZER
in their @spartacus/{featurename}/root
entry point, which by convention is an entry point that is always eager loaded.
Note: The MODULE_INITIALIZER
is a feature of the Spartacus lazy loading mechanism. It will not work for other lazy loading mechanisms, such as the default route-based lazy loading from Angular.
Customizing Lazy Loaded Modules
To customize a lazy loaded module, you start by creating a custom feature module in your application code.
In the implementation of this custom feature module, you statically import the original Spartacus feature module (which used to be lazy loaded), and then you import or provide all the customizations (for example, providing a custom service there). The following is an example:
// custom-rulebased-configurator.module.ts
import { RulebasedConfiguratorModule } from '@spartacus/product-configurator/rulebased';
@NgModule({
imports: [RulebasedConfiguratorModule], // import the original Spartacus module
providers: [
// provide customizations of classes from the original module here, such as the following:
{ provide: ConfiguratorCartService, useClass: CustomConfiguratorCartService }
]
})
export class CustomRulebasedConfiguratorModule {}
You then provide a custom Spartacus configuration that contains the featureModules
key for SOME_FEATURE_NAME. You point SOME_FEATURE_NAME to your custom feature module by referencing it with a dynamic import, such as import(./local/custom-feature.module.ts).then(m) => m.CustomFeatureModule)
.
You provide the configuration in a static module, such as the app module. The following is an example:
provideConfig({
featureModules: {
[RULEBASED_PRODUCT_CONFIGURATOR_FEATURE]: {
module: () =>
import('../custom-rulebased-configurator.module').then(
(m) => m.CustomRulebasedConfiguratorModule
),
},
},
},
Once this is done, during the build, Webpack bundles a separate JS chunk for your custom feature module, which includes all the things it statically imports, as well as the customizations the custom feature module contains.
Preparing Libraries to Work with Lazy Loading
Providing Fine-Grained Entry Points in Your Library
Mixing static and dynamic imports from the same entry points breaks lazy loading and affects tree shaking, so any library that will be used directly in dynamic imports should expose fine-grained secondary entry points to optimize code splitting.
As a convention, Spartacus exposes root entry points for features, such as @spartacus/orgainzation/administration/root
. This type of entry point contains all of the code that should not or cannot be lazy loaded. Modules from the root entry point should be imported statically in the root application, which means they will be eager loaded and will be available in the main application chunk.
For more information about support for secondary entry points in the Angular libraries, see Secondary Entry Points in the ng-packagr documentation on GitHub.
Separating Static Code from Lazy-Loaded Code
When you work with Angular Dependency Injection, the list of providers in the injector should not change after the injector is initialized. This paradigm specifically applies to any multi-provided tokens, to handlers, and especially to any Angular-native multi-provided tokens, such as HTTP_INTERCEPTOR
, APP_INITIALIZER
, and so on.
The result is that any multi-provided token in a lazy-loaded module will not be visible to modules and services that are provided in the root, or in other lazy-loaded chunks, with the exception of multi-provided tokens that are injected using the unified injector.
Some Spartacus features, such as PageMetaService
or ConverterService
, use the UnifiedInjector
to be aware of tokens that can be lazy loaded, so that the global logic (such as SEO features) can work reliably even if the logic is lazy loaded with the feature. For example, the store locator page meta resolver can be lazy loaded with the store locator feature.
The Spartacus configuration, which is also defined by providing configuration chunks, is treated a bit differently because of a compatibility mechanism that contributes configurations from lazy loading features to the global configuration. This mechanism can be disabled by a feature flag, and will be turned off by default in the future, in favor of the unified configuration feature.
If there are issues with lazy-loaded providers not being visible by the root services, it is always possible to include this kind of code in a statically-linked module that is available upfront. It is recommended to create a separate entry point in your library (by convention, named root
, such as my-library/root
) that contains minimal code, that will be included in the main bundle, and that will be available from the beginning.
Simple Strategies for Optimized and Balanced Code Splitting
Each business feature is a bit different, but in general, for a balanced approach you can consider the following:
- Most features offer a range of UI components, and if you access one component, it is likely that other components will soon be needed as well. To minimize network traffic and the overhead of unnecessary granularity, you can consider bundling more than one component in the chunk.
- Most UI components for a feature usually require the core logic, such as facade services and adapters. In this case, you can consider lazy loading the core with the component.
- Some core services of a feature may also be used frequently by other features. In this case, you can consider splitting the core code from the UI components.
- Some parts of a feature are used very frequently (such as the cart icon component, which is used on every page), while some are used only in certain specific scenarios (such as the Cart Summary). To achieve an optimal experience, it is recommended to split the feature into logical parts based on the usage, and create separate entry points (and features) for the minimal, most needed parts, and another entry point for less frequently used code.
Wrapping Library Code in a Lazy-Loaded Module
To address some specific customizations on top of a lazy-loaded library, it is possible to wrap static imports from one or more libraries into a single module that will be imported dynamically instead. This technique can also be an effective optimization strategy, because the builder will be able to use tree shaking for all of the static imports inside the module.