blob: 86cf206d96575761937c133dbf4c93166df0ea45 [file] [log] [blame]
/*
* Copyright 2011 Marc Grue.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied.
*
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.zest.sample.dcicargo.sample_b.context.interaction.handling.inspection.event;
import java.util.Date;
import org.apache.zest.api.injection.scope.This;
import org.apache.zest.api.mixin.Mixins;
import org.apache.zest.api.value.ValueBuilder;
import org.apache.zest.sample.dcicargo.sample_b.context.interaction.handling.inspection.exception.InspectionException;
import org.apache.zest.sample.dcicargo.sample_b.context.interaction.handling.inspection.exception.InspectionFailedException;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.cargo.Cargo;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.cargo.RouteSpecification;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.delivery.Delivery;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.delivery.RoutingStatus;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.delivery.TransportStatus;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.handling.HandlingEvent;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.itinerary.Itinerary;
import org.apache.zest.sample.dcicargo.sample_b.infrastructure.dci.Context;
import org.apache.zest.sample.dcicargo.sample_b.infrastructure.dci.RoleMixin;
import static org.apache.zest.sample.dcicargo.sample_b.data.structure.delivery.RoutingStatus.NOT_ROUTED;
import static org.apache.zest.sample.dcicargo.sample_b.data.structure.delivery.TransportStatus.IN_PORT;
import static org.apache.zest.sample.dcicargo.sample_b.data.structure.delivery.TransportStatus.ONBOARD_CARRIER;
import static org.apache.zest.sample.dcicargo.sample_b.data.structure.handling.HandlingEventType.CUSTOMS;
/**
* Inspect Cargo In Customs (subfunction use case)
*
* This is one the variations of the {@link org.apache.zest.sample.dcicargo.sample_b.context.interaction.handling.inspection.InspectCargoDeliveryStatus} use case.
*
* Can the cargo get handled by customs only in the current port location?! Nothing now prevents
* an unexpected cargo in customs in some random location. A domain expert is needed to explain
* how this goes.
*
* For now the location doesn't affect the misdirection status, and we presume (hope) that the
* cargo will only get handled by customs in the port it's currently in. Should we safeguard
* against unexpected in-customs-locations?
*/
public class InspectCargoInCustoms extends Context
{
DeliveryInspectorRole deliveryInspector;
HandlingEvent customsEvent;
RouteSpecification routeSpecification;
Itinerary itinerary;
Integer itineraryProgressIndex;
TransportStatus transportStatus;
public InspectCargoInCustoms( Cargo cargo, HandlingEvent handlingEvent )
{
deliveryInspector = rolePlayer( DeliveryInspectorRole.class, cargo );
customsEvent = handlingEvent;
routeSpecification = cargo.routeSpecification().get();
itinerary = cargo.itinerary().get();
itineraryProgressIndex = cargo.delivery().get().itineraryProgressIndex().get();
// Transport status before customs handling
transportStatus = cargo.delivery().get().transportStatus().get();
}
public void inspect()
throws InspectionException
{
// Pre-conditions
if( customsEvent == null || !customsEvent.handlingEventType().get().equals( CUSTOMS ) )
{
throw new InspectionFailedException( "Can only inspect cargo in customs." );
}
if( transportStatus.equals( ONBOARD_CARRIER ) )
{
throw new InspectionFailedException( "Cannot handle cargo in customs on board a carrier." );
}
deliveryInspector.inspectCargoInCustoms();
}
@Mixins( DeliveryInspectorRole.Mixin.class )
public interface DeliveryInspectorRole
{
void setContext( InspectCargoInCustoms context );
void inspectCargoInCustoms()
throws InspectionException;
class Mixin
extends RoleMixin<InspectCargoInCustoms>
implements DeliveryInspectorRole
{
@This
Cargo cargo;
Delivery newDelivery;
public void inspectCargoInCustoms()
throws InspectionException
{
// Step 1 - Collect known delivery data
ValueBuilder<Delivery> newDeliveryBuilder = vbf.newValueBuilder( Delivery.class );
newDelivery = newDeliveryBuilder.prototype();
newDelivery.timestamp().set( new Date() );
newDelivery.lastHandlingEvent().set( c.customsEvent );
newDelivery.transportStatus().set( IN_PORT );
newDelivery.isUnloadedAtDestination().set( false );
// Customs handling location doesn't affect direction
newDelivery.isMisdirected().set( false );
// We can't predict the next handling event from the customs event
newDelivery.nextHandlingEvent().set( null );
// Step 2 - Verify cargo is routed
if( c.itinerary == null )
{
newDelivery.routingStatus().set( NOT_ROUTED );
newDelivery.itineraryProgressIndex().set( 0 );
newDelivery.eta().set( null );
}
else if( !c.routeSpecification.isSatisfiedBy( c.itinerary ) )
{
newDelivery.routingStatus().set( RoutingStatus.MISROUTED );
newDelivery.itineraryProgressIndex().set( 0 );
newDelivery.eta().set( null );
}
else
{
newDelivery.routingStatus().set( RoutingStatus.ROUTED );
newDelivery.itineraryProgressIndex().set( c.itineraryProgressIndex );
newDelivery.eta().set( c.itinerary.eta() );
}
// Step 3 - Save cargo delivery snapshot
cargo.delivery().set( newDeliveryBuilder.newInstance() );
}
}
}
}