/MicroservicesConcepts

Various extension and dev use cases. Proving ground for concepts.

Primary LanguageJava

Microservice Extensibility Concepts

This project contains several example concepts demonstrating various extension and backend commerce API developer use-cases.

Terminology

  • Concept: A sub-project within this overall project that showcases a discrete customization goal. The goal will be clearly stated in each concept project. Some concept projects will be deeper riffs on concepts from other projects (e.g. a simple domain extension in one project may be built on top of to add additional repository extension in another project).

  • Flex Package: A deployable unit of Broadleaf Commerce Microservices. This may contain a single microservice (e.g. the Catalog Service), or it may contain several services bundled together. The advantages of flexpackage bundling are: reduced devops complexity with fewer discrete units running in an environment, smaller deployment footprint with multiple services running under the same JVM/heap allocation (saves $), and a sane local development footprint suitable for developer laptops to run the entire stack. Disadvantages include: less granular scale tuning, and possibly decreased team contribution isolation. It’s notable that flexpackage setup is a configuration only concern, and changing later on to a more granular deployment is entirely feasible. The Balanced flexpackage is generally considered the best mix when weighed against all these concerns at the same time. More details can be found here

  • Auto Projection: Broadleaf Commerce Microservices operate with two versions of the domain class representation. The repository domain is generally represented as a JPA annotated POJO that interacts with an ORM to facilitate input/output from database storage. The projection domain is a version of the repository domain that is exposed outside of the service layer and represents the structure that is serialized into JSON as output from the microservice endpoints. Projection and repository domains can share the same structure, but are not required to do so, and Broadleaf provides mapping facilities to customize how data flows between these two representations of the same data. While Broadleaf does allow complexity here, it is not required that extensions of Broadleaf repository domain (or introduction of new repository domain) explicitly declare a projection with explicit mapping. If the needs are simple and direct representation of the data to the business layer is acceptable, then Broadleaf can auto generate a projection for you. Henceforth, you will be able to reference the projection and its fields using the facilities in Broadleaf (demonstrated in several of the concept examples). This serves to simplify configuration and development for standard use cases.

  • Data Routing: This is the technology that primarily facilitates the operation of flexpackage bundling. Data routing is used to detect the datasource that should be utilized for the current flow. As a result, multiple service endpoints housed in the same JVM/classloader can leverage different datasources (and different backing database/schema) to enable their data interactions. Beyond datasources, spring beans can be grouped and filtered by a route to produce interesting behavior for service flows as well. This all supports proper bounded context per service, while still allowing more interesting deployment options.

  • Data Tracking: This is the technology that supports entity discrimination, sandboxing, and catalog propagation. There are different tracking types based on the maintenance needs of the data. Types of tracking include: catalog, application, sandbox, customercontext, and tenant. These types include interesting grouping, discrimination, preview, audit, and workflow capabilities. Based on your requirements, you can also add these capabilities in a simple way to your own unique domain.

  • Fragment (or Repository Fragment): This is a Spring Data concept for contributing repository data persistence behavior via interfaces and concrete implementations. The resulting spring component is callable at runtime and exposes the aggregate contract of all the contributed interfaces. Broadleaf provides facilities to contribute additional fragments (interfaces and optional concrete impls) to existing repositories. This is the primary extension mechanism for adding behavior to Broadleaf declared repositories.

Project Guidelines

  • The concept project structure will be simple. The Min flexpackage alone will be generally favored, avoiding balanced and granular bloat.

  • Each concept project will consume framework and demo artifacts from the latest relevant develop branches.

  • Each concept project is constructed using the minimal amount of configuration and code for optimum clarity.

Project Structure

Directory Description

concepts

Group of runnable projects that demonstrate a key dev or extension experience using the most streamlined approach available.

docker

Basic docker configuration to support backend of the tutorial runtime.

script

core build script code that supports all instances of reset-and-run and stop-docker.

What Do I Need?

In order to get these concepts running locally, you will want to obtain and install the following:

Credentials

Access to Broadleaf’s Docker Registry and Maven Nexus is needed in order to pull down the appropriate resources.

Once you have obtained these credentials, you will need to configure them for the dependencies below.

ℹ️

Contact info@broadleafcommerce.com for information on how to receive these credentials

Java

You will need Java 17 installed on your machine.

Docker

You will need to have Docker Engine & Docker Compose installed locally

💡

Docker Desktop for both Mac and Windows already includes compose along with other docker apps.

Once you have docker installed, you will want to authenticate with Broadleaf’s docker registry.

Type the following into your CLI:

docker login repository.broadleafcommerce.com:5001

When prompted, type in the username and password you received above.

Maven

You’ll need to have Maven installed locally as well. Maven 3.5 or later is recommended.

Once you have maven installed, another step you need to do before you can build the project is to configure the authentication. Maven requires it to be specified in a file called settings.xml that has to be in the .m2 subdirectory in a user’s home directory.

  1. Create a folder called .m2 in your home directory (if one does not already exist)

  2. Create a file called settings.xml in the .m2 folder (if not there already)

  3. Copy the following contents to your ~/.m2/settings.xml servers section, making sure to replace the credentials with the ones you received above:

<settings xsi:schemaLocation="http://maven.apache.org/SETTINGS/1.1.0 http://maven.apache.org/xsd/settings-1.1.0.xsd"
    xmlns="http://maven.apache.org/SETTINGS/1.1.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <servers>
    <server>
      <id>broadleaf-microservices</id>
      <username>username_here</username>
      <password>password_here</password>
    </server>
  </servers>
</settings>
💡

More info about authenticating with private repositories can be found here and here

Running a Concept Project

  • In the specific concept module, execute the platform specific reset-and-run script.

💡

ctrl-c will terminate the process.

  • The stop-docker script may be used to take down any running container resulting from the reset-and-run script.

  • The admin application will be accessible at https://localhost:8446 after all components have completed startup.

💡

The browser may complain about the self-signed cert and you may have to make an exception for the cert in your browser.

How Does This Work?

  • Each concept project creates a jar that is contributed to a standard Broadleaf Min flexpackage demo

  • The jar is contributed via Spring Boot auto-configuration

  • When necessary, @AutoConfigureBefore and @AutoConfigureAfter are employed to favor concept configuration over Broadleaf configuration

  • Integration tests are generally employed in each concept to exercise the service API and demonstrate the customization.

  • The customizations are generally Catalog service customizations - specifically riffs on Product.

Scope of Applicability

  • The Java customization examples in this project should all be applicable to existing and new Broadleaf Microservices projects.

  • The maven pom inheritance, maven profiles, run commands, etc…​ are not currently applicable to existing or new Broadleaf Microservices projects - although advances here will eventually be brought over to real dev projects in the future.

The concept projects require recent versions of several Broadleaf Framework libraries

<dependencyManagement>
    <dependencies>
        <!-- ↓ These should appear first before the release train ↓ -->
        <dependency>
            <groupId>org.broadleafcommerce</groupId>
            <artifactId>spring-frameworkmapping</artifactId>
            <version>0.9.1-GA</version>
        </dependency>
        <dependency>
            <groupId>com.broadleafcommerce.microservices</groupId>
            <artifactId>broadleaf-common-extension</artifactId>
            <version>1.4.13-GA</version>
        </dependency>
        <dependency>
            <groupId>com.broadleafcommerce.microservices</groupId>
            <artifactId>broadleaf-common-jpa</artifactId>
            <version>1.5.7-GA</version>
        </dependency>
        <dependency>
            <groupId>com.broadleafcommerce.microservices</groupId>
            <artifactId>broadleaf-data-tracking-dependencies</artifactId>
            <version>1.7.10-GA</version>
            <type>pom</type>
            <scope>import</scope>
        </dependency>
        ...
    </dependencies>
</dependencyManagement>

Creating integration tests in your own projects using the patterns demonstrated in integration tests in this project requires an additional library

<dependencyManagement>
    <dependencies>
        ...
        <!-- ↓ This should appear before the release train, and after the deps above ↓ -->
        <dependency>
            <groupId>com.broadleafcommerce.microservices</groupId>
            <artifactId>broadleaf-core-test</artifactId>
            <version>1.0.0-SNAPSHOT</version> <!-- preliminary version - GA available soon -->
            <scope>test</scope>
        </dependency>
        ...
    </dependencies>
</dependencyManagement>
<dependencies>
    ...
    <dependency>
        <groupId>com.broadleafcommerce.microservices</groupId>
        <artifactId>broadleaf-core-test</artifactId>
        <scope>test</scope>
    </dependency>
</dependencies>
  • Leveraging the code patterns and components demonstrated in the concept projects requires recent versions of several Broadleaf Framework libraries. You should set these versions (or newer) at the top of the dependencyManagement section of your root pom in your own project before attempting to reproduce or compile against the patterns shown here. If your release train reference is advanced enough to bring these versions (or newer) in by default, then you can skip explicitly declaring them here.

Roadmap

  • This project represents an ongoing effort to document customization use cases and patterns.

  • We will continue to add new concepts as they come up.

  • If a new concept requires a framework change or enhancement (e.g. a change to one or more of the libraries above), we will increment the version of this concept project along with noting the version change(s) to the associated common framework libraries.

  • Training will also be developed based on these concept materials and will be available separately.

Concept Directory

Simple Product Extension

Simple extension of JpaProduct adding only a basic field type. Also leverages auto projection, rather than opting for an explicit extended projection.

  • Demonstrate the simplest type of extension

  • Introduce the Projection interface

  • Show full lifecycle support (json in/out) for the endpoint API

  • Show supporting admin customization

  • Show automated testing and the use of @TestCatalogRouted (et al.) to handle datarouting requirements during the test

Product Extension with JSON Field

Simple extension of JpaProduct adding more complex field types, including collections and maps. The complex types use JPA converters to persist the complex structure as JSON. This example still leverages auto projection and does not declare an explicit extending projection type.

  • Demonstrate more complex field type

  • Demonstrate interaction with Projection interface to expose complex structures for editing

  • Show full lifecycle support (json in/out) for the endpoint API

  • Show supporting admin customization

  • Builds On : 00100-productExtensionOnly

Explicit Product Projection

Continues with the complex field example persisted as JSON. However, in this case, an explicit projection type is declared.

  • Demonstrate custom mapping to/from projection

  • Demonstrate response only projection field

  • Show supporting admin customization

  • Demonstrate mapping to synthetic fields

  • Builds On : 00200-productExtensionComplexFieldJson

Product Extension with New Table Relationship

Alters the complex field example to leverage a traditional JPA OneToMany associated collection. The relates to a new table in the database, rather than serializing to JSON.

  • Show table based complex field support in the JpaProduct extension

  • Demonstrate custom mapping to/from projection

  • Demonstrate special @ProjectionPostConvert support for setting bi-directional references

  • Show supporting admin customization

  • Builds On : 00200-productExtensionComplexFieldJson

Product with Nested JSON Collection

Extends nested structures that appear arbitrarily deep in the object graph of JpaProduct. The structures appear in various embedded collections and are persisted as JSON.

Product with New Nested Table Relationship

Extends nested structures that appear arbitrarily deep in the object graph of JpaProduct. The structures appear in OneToMany table based collections.

Overriding a Repository

Adds a new repository implementation fragment overriding out-of-the-box behavior of JpaTrackableRepository

  • Show concrete fragment contribution example overriding JpaTrackableRepository methods for JpaProductRepository

  • Demonstrate the use of JpaTrackableRepositoryDelegateSupplier to use in the fragment for extension via composition

  • Builds On : 00200-productExtensionComplexFieldJson

New Repository

Introduces new repository methods that contribute new persistence related behavior. This take the form of either dynamic query method fragments, or concrete implementation fragments.

  • Demonstrate new query method fragment contribution (interface only)

  • Demonstrate new concrete method implementation fragment contribution

  • Show concrete fragment contribution example overriding JpaTrackableRepository methods for JpaProductRepository

  • Demonstrate the use of JpaTrackableRepositoryDelegateSupplier to use in the fragment for extension via composition

  • Builds On : 00200-productExtensionComplexFieldJson

Customizing Business Logic

Uses a simple customization of the DefaultProductService.

  • Show a minor customization of the business logic of DefaultProductService

Customized Business Logic with Auto Projection

Business logic customization that leverages a customized repository and extended domain with auto projection

  • Show DefaultProductService call the customized repository to search by a new extended field

  • Demonstrate how to use the Projection interface to interact with the service API

  • Builds On : 00800-repositoryCustomizationContribution

Customized Business Logic with Explicit Projection

Business logic customization that leverages a customized repository and extended domain with explicit projection

Endpoint Customization

Simple customization of out-of-the-box ProductEndpoint

  • Demonstrate a behavior tweak of a single endpoint method

Endpoint Customization using AutoProjection

Customization of an endpoint method in ProductEndpoint leveraging a customized service, repository, auto-projection, and domain

  • Demonstrate a behavior tweak of a single endpoint method

  • Show leveraging a completely customized flow through to persistence

  • Demonstrate working with an auto projection in the endpoint

  • Builds On : 01000-businessLogicCustomizationAutoProjection

Endpoint Customization using Explicit Projection

Customization of an endpoint method in ProductEndpoint using an extended explicit projection and domain

Brand New Entity

Concept: 01500-newDomain

Introduction of new domain without explicit projection or any other explicit plumbing like repository, service, or endpoint

  • Demonstrate the simplest type of domain introduction

  • Show full lifecycle support (json in/out) for the endpoint API

  • Builds On : 00100-productExtensionOnly

Brand New Entity with Complex Fields

Introduction of new domain including complex field structures

  • Demonstrate domain introduction with embedded json collection fields

  • Demonstrate domain introduction with nested JPA OneToMany collection fields

  • Builds On : 01500-newDomain

Tuning Auto Projection with New Entity

Introduction of new domain with auto projection output fine tuned through customization

  • Demonstrate customization of auto projection with the ExplicitProjectionFieldConfiguration annotation

  • Demonstrate removing a field from the projection

  • Demonstrate limiting a field to response only during update/replace

  • Demonstrate altering deserialization/serialization (e.g. to/from MonetaryAmount for a BigDecimal field)

  • Builds On : 01600-newDomainComplexField

New Entity with Explicit Projection

Introduction of new domain including explicit projection declaration