It is a POC for adding a new WebApp to Doppler complementing the new WebApp (app.fromdoppler.com)
and the old MVC UI (app2.fromdoppler.com).
From the user point of view, this should be the same application. But, from ours, it is a different application deployed independently and loose coupled to the other ones.
By the moment, we will relay on Doppler user's JWT Token and GetUserData, and probably, in the
future we will try to use OAuth 2 in place.
On the other hand, we will use our Manifest Loader in order to take advantage of the cache and simplify the release process. CI/CD process will be very similar to our API microservices (for example Hello Microservice), but based on manifest files in our CDN in place of Docker images.
Since we are using the Manifest Loader, the packages are independent from the hosting service, you can see an example of how to use it in index.html.
As we successful learned with the "original" Doppler WebApp, having clear abstractions for the services is really useful for team work because it helps to create doubles and to have a clear separation of concerns and to simplify testing.
For that reason, we will follow this architecture:
Since TypeScript/JavaScript does not have real types to resolve dependencies during execution time, we will use the
AppServices interface to access the services by name. The problem with this approach,
is that all abstractions should be in the same component and to be shared with all the application.
By default, the implementation of AppServices is SingletonLazyAppServicesContainer.
It uses a dictionary of the factories of all services and resolves the singleton instances of the services in a lazy way
when they are required by a component or other service.
We will use AppServicesContext to store a instance of AppServices in the React's
Context.
  sequenceDiagram
    participant index_tsx
    participant AppServicesProvider
    participant AppServicesContext
    participant composition root
    participant implementations
    index_tsx->>+composition root: configureApp(window["editor-webapp-configuration])
    composition root->>+implementations: build concrete services
    implementations-->>-composition root: appServices
    composition root-->>-index_tsx: appServices
    index_tsx->>+AppServicesProvider: appServices={appServices}
    AppServicesProvider->>+AppServicesContext: Provider value={appServices}
    AppServicesContext-->>-AppServicesProvider: .
    AppServicesProvider-->>-index_tsx: .
    Then, AppServices will be injected in the desired components using
InjectAppServices HOC or useAppServices hook:
  sequenceDiagram
    participant index_tsx
    participant components
    participant AppServicesProvider
    participant AppServicesContext
    participant implementations
    index_tsx->>+components: render
    components->>+AppServicesProvider: InjectAppServices / useAppServices
    AppServicesProvider->>+AppServicesContext: AppServicesContext.Consumer
    AppServicesContext-->>-AppServicesProvider: appServices
    AppServicesProvider-->>-components: appServices
    components->>+implementations: do work
    implementations-->>-components: work results
    components-->>-index_tsx: html
    This is the resulting directory structure:
  + /
  |
  +--- index.tsx
  |
  +--- composition-root.ts
  |
  +--+ abstractions
  |  |
  |  +--- services.ts
  |  |
  |  +--+ common
  |  |  |
  |  |  \--- {shared types}
  |  |
  |  \--- {a folder for each abstraction domain}
  |
  +--+ implementations
  |  |
  |  +--- SingletonLazyAppServicesContainer.ts
  |  |
  |  \--- {a folder for each implementation (without
  |        dependencies between them)}
  |
  \--+ utils (abstract utilities, domain agnostic)
Another drawback of this approach is that we will need a little boilerplate each time that we add a new service:
- 
Define the abstraction in abstractionsdirectory
- 
Define the implementations in implementationsdirectory
- 
Add the entry in AppServicesinterface
- 
Add the entry in SingletonLazyAppServicesContainerfile
- 
Add the factory of the new service in composition-rootfile
Apps sometimes store config as constants in the code. This is a violation of twelve-factor, which requires strict separation of config from code. Config varies substantially across deploys, code does not. The Twelve-Factor App - III. Config
We know that this application is not a micro-service, but we believe that this principle stills apply.
For that reason, we want to generate a bundle and share it as it is in all our different environments.
It requires to inject the configuration from the outside, in our case we choose using an object in
the global window scope with the name editors-webapp-configuration.
Production's index.html example:
<!doctype html>
<html lang="en">
  <!-- . . . -->
  <body>
    <!-- . . . -->
    <script src="https://cdn.fromdoppler.com/mfe-loader/loader-v2.0.0.js"></script>
    <script type="text/javascript">
      const scriptUrl = "https://cdn.fromdoppler.com/editors-webapp/asset-manifest-v1.json`;
      window["editors-webapp-configuration"] = {
        basename: "editors",
        dopplerLegacyBaseUrl: "https://app2.fromdoppler.com",
        htmlEditorApiBaseUrl: "https://apis.fromdoppler.com/html-editor",
        keepAliveMilliseconds: 300000
      };
      assetServices.load({ manifestURL: scriptUrl });
    </script>
  </body>
</html>This configuration object will be merged with the defaultAppConfiguration and will be available as a service to be injected in any component or service.
Example of a configuration injected into a component:
export const DemoComponent = InjectAppServices(
  ({ appServices: { appConfiguration } }: AppServices) => (
    <code>
      <pre>{JSON.stringify(appConfiguration)}</pre>
    </code>
  ),
);
