General Implementation Examples
This page gives an overview of the most common use cases and how to implement them with a Domain Service Project (Java).
Factory Commands
The main purpose of a factory command* is to set values to the properties of a root entity and then persist those changes to the database, thus, creating a new instance.
The factory command to create an instance of a root entity called MasterCard is the following:
@Service
public class MasterCardCommand extends MasterCardCommandBase {
@Override
public void createCreditCard(MasterCard instance, CreateCreditCardInput createCreditCardInput) {
// Fill instance with values from input entity
instance.setBankName(createCreditCardInput.getBankName());
// save entity to repository
repo.cc.masterCard.save(instance);
}
}
Instance Commands
The implementation of an instance command called ActivateCard which manipulates the state of an instance of the root entity MasterCard is shown below:
@Service
public class MasterCardCommand extends MasterCardCommandBase {
@Override
public void activateCard(MasterCard instance) {
// Change root entity data
instance.setActive(true);
// Save to repository
repo.cc.masterCard.save(instance);
}
Events
Within stub classes (commands, services, agents) that are modelled to publish events, a dedicated event publish function will be available from the base class.
@Service
public class MasterCardCommand extends MasterCardCommandBase {
@Override
public void activateCard(MasterCard instance) {
// Change root entity data
instance.setActive(true);
// Save to repository
repo.cc.masterCard.save(instance);
// Create payload entity
SuccessEventPayload payloadEntity = entityBuilder.cc.successEventPayload().build();
// Create domain event
SucessEvent event = eventBuilder.cc.sucessEvent().setPayload(payloadEntity).build();
// Publish event via publishSucessEvent() that exists defined in MasterCardCommandBase class.
publishSucessEvent(event);
}
An event can also be published by using the structure of the following code:
// Declare event builder
@Autowired
EventBuilder eventBuilder;
// Declare entity builder
@Autowired
EntityBuilder entityBuilder;
// Declare Event Prodcuer Service
@Autowired
EventProducerService eventProdcuer;
// Create payload entity
SuccessEventPayload payloadEntity = entityBuilder.cc.successEventPayload().build();
// Create domain event
SucessEvent event = eventBuilder.cc.sucessEvent().setPayload(payloadEntity).build();
// Publish a domain event.
eventProdcuer.publish(event);
Agents
When an event was triggered, it will call any agent that is bound to this event. This agent will receive the payload of the event as input and then it can call a service or publish another event.
public class SucessEventAgent extends SucessEventAgentBase {
private static Logger log = LoggerFactory.getLogger(SucessEventAgent.class);
@Override
public void onMessage(SucessEventPayload sucessEventPayload) {
log.info("Agent SucessEventAgent received event cc:SucessEvent");
// call service to notify customer passing the sucessEventPayload to the service as input
service.cc.notifyCustomer(sucessEventPayload);
}
}
Domain services
The implementation of a domain service called GetCustomer is shown below:
@Service
public class GetCustomer extends GetCustomerBase {
@NewSpan
@Override
public Customer execute(GetCustomerInputEntity input){
// Call repository to get all customers
return repo.cc.customer.findById(input.getCustomerId());
}
}
@NewSpan
annotation creates a new span which will be a child of the existing span via the underneath springframework.cloud.sleuth that helps with microservice tracing.External Entity Service
The external entity service comprises 3 functions: create, load and validate.
Create
The create function is used to construct a local instance of the External Entity and initialize the associated properties with certain pre-specified values that are provided as input parameters.
The input parameters are also used to identify and load external entity from external system.
@Service
public class AmexCardService extends AmexCardServiceBase {
@Override
public AmexCard create(String amexCardId, String customerId) {
// Call your external system and use parameters to identify and load external entity
// Create external entity instance and set its properties
AmexCard amexCard = new AmexCard(amexCardId, "Card type retrieved from external entity");
return amexCard;
}
Validate
The validate function gives information regarding the existence of a specific instance of an external entity within external system.
It usually returns
The external entity's instance if the external instance exists in the external system
null
if it does not exist
The validator can also be used for updating the local instance of an external entity if the update flag is set
to true
.
/**
* validate the existence of the external entity in the external system.
*
* @param externalEntity instance of AmexCard.
* @param update flag to indicate whether should an updated
* AmexCard instance be created and returned or not.
* @return Optional AmexCard instance to indicate whether
* external entity still exists in external system
*/
@Override
public Optional<AmexCard> validate(AmexCard externalEntity, boolean update) {
// Call external system to validate existence of external entity in external system
// Also you can update external entity properties
AmexCard amexCard = new AmexCard(externalEntity.getAmexCardId(),
"Updated Card type info retrieved from external entity");
return Optional.of(amexCard);
}
Load
The load function implements logic to call external system and loads the full external entity data.
A local external entity instance is used to provide identifier properties needed to call external systems and fully load the external entity's data. You can build an entity, fill it with data coming from external system and return that entity for further usages.
/**
* Load external Entity and return as an Entity
*
* @param externalEntity instance of AmexCard
* @return class that extends base type Entity
*/
@Override
public Entity load(AmexCard externalEntity) {
// Call external system to load external entity(for example make a REST call)
// Build a card entity and fill with data from external entity
Card card = entityBuilder.cc.card().build();
return card;
}