Concept - B2B Order Approval Service

Introduction

Info

This concept is valid from Intershop Commerce Mangement 7.10.20.1. 


In a B2B scenario, orders may need to be approved by certain users of a customer before they are placed. The order approval workflow can vary in complexity and could also be handled by an external system. To ensure that the order approval is flexible enough, an Order Approval Service is implemented using the Approval Service interfaces from the standard Intershop 7.

References

Workflow

The following diagram shows the approval workflow:

Implementation

The implementation of the service is based on the order approval workflow of the previous version of the B2B Extension Module and supports approval rules as before (see Cookbook - B2B Order Approval for more details).

Currently a generic REST approval callback is not part of the standard implementation. A callback has to be implemented separately when integrating with an external approval system.

Order Approval Handler Chains

With ICM version 7.10.20.1 the approval process which was mainly implemented with pipelines has been re-implemented using customizable handler chains. There are separate handler chains for the Approve and Reject process.
These handler chains are called via a Business Object extension of the BasketBO business object called BasketBOOrderApprovalExtension.

BasketBOOrderApprovalExtension
    /**
     * Starts the approval chain for the extended basket and given approver
     * @param approver The approver
     * @return The result of the approval as {@link OrderApprovalResult}
     */
    OrderApprovalResult approve(UserBO approver);

    /**
     * Starts the rejection chain for extended basket and given approver
     * @param approver The approver
     * @param approvalComment The approval comment
     * @return The result of the rejection as {@link OrderApprovalResult}
     */
    OrderApprovalResult reject(UserBO approver, String approvalComment);

The return value of the methods calls is an OrderApprovalResult. It contains the list of possible failures that might have occurred during the approval process. It also contains an optional OrderBO (only in case of positive approval).

Approve via BasketBOOrderApprovalExtension:

Example code for "Approve"
        BasketBOOrderApprovalExtension extension = basketBO.getExtension(BasketBOOrderApprovalExtension.EXTENSION_ID);
        OrderApprovalResult approvalResult = extension.approve(approver);
        if (approvalResult.isFailure()) 
        {
            // error handling
        }
        if (approvalResult.getOrderBO().isPresent()) 
        {
            // approval was the last one -> so order was created
            OrderBO orderBO = approvalResult.getOrderBO().get();
            // ... do further processing
        }

Reject via BasketBOOrderApprovalExtension:

Example code for "Reject"
        BasketBOOrderApprovalExtension extension = basketBO.getExtension(BasketBOOrderApprovalExtension.EXTENSION_ID);
        OrderApprovalResult approvalResult = extension.reject(approver, "Reject comment...");
        if (approvalResult.isFailure()) 
        {
            // error handling
        }

The new REST API for requisition approval also uses these BO layer methods for the approval and rejection process.

As mentioned before there are separate handler chains for Approve and Reject. It is possible to visualize the list of registered handler chains by opening the following back office URL manually (adapt host, port to your local environment):

List of Registered Handler Chains
https://<host:port>/INTERSHOP/web/WFS/inSPIRED-Site/en_US/-/USD/InspectHandlerChains-Start

The two chains will be explained in detail in the following sections.

Approve Handler Chain

The OrderApprovalApproveChain which handles the positive case Approve consists of two sub-chains: A PreOrderApprovalApproveChain and the OrderApprovalApproveChain itsself.

The PreOrderApprovalApproveChain does some general checks before the approval process can start, e.g., if the approval feature is generally enabled or the requisition (basket) has to be approved by the user. If any of the checks fails, the execution of the whole handler chain will be stopped (Stop On Failure).

The OrderApprovalApproveChain contains all necessary steps for an approval like adjusting the approval steps, setting the basket status, creating emails etc. If any of the handlers fails, the process will be continued anyway (Continue On Failure).

Reject Handler Chain

The OrderApprovaRejectChain which handles the negative case Reject also consists of two sub-chains: A PreOrderApprovalRejectChain and the OrderApprovalRejectChain itsself.

Again the PreOrderApprovalRejectChain does some general checks before the approval process can start, e.g. if the approval feature is generally enabled or the requisition (basket) has to be approved by the user. If any of the checks will fail, the execution of the whole handler chain will be stopped (Stop On Failure).

The OrderApprovalRejectChain contains all necessary steps for an approval reject like adjusting the approval steps, setting the basket status, creating emails etc. If any of the handlers fails, the process will be continued anyway (Continue On Failure).

Dependencies

Classes

The following diagram describes the main classes used by the Intershop Commerce Management approval service implementation. For more information on the approval service itself, refer to Concept - Approval Managed Service.

  • ApprovalRequiredRule - A rule that can determine whether approval is required for a given buyer and requisition. For example a rule could say approval is required if the requisition total is greater than the buyer's budget. In general the rules should perform simple checks.
  • ApprovalRequiredRuleProvider - A provider for ApprovalRequiredRule instances.
  • ApprovalStepBO - A unit that needs to be approved by a single approver. Multiple steps can be associated with the same requisition. In general each step of the same requisition should require approval by a different set of users (e.g., any of the users with the permission to approve orders, the manager of a cost center, etc.). Different ApprovalRequiredRule instances could require different approval steps. Once all applicable approval steps for a requisition are set to approved, then the requisition is considered as approved and the order creation process could start. If any of the approval steps for a requisition is set to Rejected, then the entire requisition is considered to be refused. In case the same user is responsible for more than one approval step of a requisition, he could set the approval decision for all steps in one action(Approve or Reject).
  • BusinessObjectApprovalStepExtension - Manages the approval steps. It is also aware of which steps need to be created for a requisition based on the ApprovalRequiredRule instances.
  • EligibleApproverPredicate - A predicate that can be used to determine whether a user is eligible to approve an approval step. For example a predicate could check whether the user has the permission to approve orders. The predicate is used internally by the default implementation of the approval step BO.

Components

The ApprovalRequiredRule and EligibleApproverPredicate are available as contract components and can be used to add new rules or trigger the creation of new steps. For more details refer to Cookbook - B2B Order Approval.

The following rules are available and wired out of the box:

  • BudgetThresholdApprovalRule - requires approval if the requisition total exceeds the buyer's budget. Applicable for one time purchase baskets, but not for baskets automatically created from approved subscription.
  • SingleOrderThresholdApprovalRule - requires approval if the requisition total exceeds the buyer's limit for a single order. Applicable for one time purchase baskets, but not for baskets automatically created from an approved subscription.
  • CostCenterApprovalRule - requires approval if the requisition is assigned to a cost center. In case the buyer is assigned as buyer for at least one cost center, a cost center must be selected and it is assigned to the basket. Once the basket has cost center assigned it requires approval, unless it is automatically created from approved subscription.
  • SubscriptionApprovalRule - requires approval if the requisition is a subscription. Once the subscription is approved, all subsequent baskets created from it are also considered as automatically approved.

From ICM 12.3.0 on, BudgetThresholdApprovalRule and SingleOrderThresholdApprovalRule can check the buyer's budget/limit against the requisition total gross or requisition total net. The price type being used depends on the value set at the CustomerBOBudgetExtension of the user's customer. Before, the requisition total gross has always been used and is now the default, until the customer changes it.

The following predicates are available and wired out of the box:

  • ApprovalPermissionEligibleApproverPredicate - users with the permission to approve orders can approve a step. This is set as default. Steps are created using this predicate when a rule does not have a specific assignment.
  • CostCenterEligibleApproverPredicate - the manager of the cost center to which the requisition is assigned. A step using this predicate is created if approval is required by the CostCenterApprovalRule.

Basket State Changes

The following diagram illustrates possible basket state changes related to the approval process:

  • BASKET_OPEN - Basket is open for every kind of manipulation, such as adding products, warranties and coupons or removing line items.
    Furthermore a basket being in this state is ready for checkout.
  • BASKET_ORDERED - Checkout of basket has been successfully finished.

One time purchase requisitions has basket type REQUISITION(10). Subscriptions has basket type RECURRINGBASKET(11). Before to start the Checkout process it is possible to switch basket type from REQUISITION to RECURRINGBASKET and vise versa.

Disclaimer
The information provided in the Knowledge Base may not be applicable to all systems and situations. Intershop Communications will not be liable to any party for any direct or indirect damages resulting from the use of the Customer Support section of the Intershop Corporate Web site, including, without limitation, any lost profits, business interruption, loss of programs or other data on your information handling system.
The Intershop Knowledge Portal uses only technically necessary cookies. We do not track visitors or have visitors tracked by 3rd parties. Please find further information on privacy in the Intershop Privacy Policy and Legal Notice.
Home
Knowledge Base
Product Releases
Log on to continue
This Knowledge Base document is reserved for registered customers.
Log on with your Intershop Entra ID to continue.
Write an email to supportadmin@intershop.de if you experience login issues,
or if you want to register as customer.