blob: b437ceca19e4320506b0f25431055bef75acc8fd [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.CargoMisdirectedException;
import org.apache.zest.sample.dcicargo.sample_b.context.interaction.handling.inspection.exception.CargoMisroutedException;
import org.apache.zest.sample.dcicargo.sample_b.context.interaction.handling.inspection.exception.CargoNotRoutedException;
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.NextHandlingEvent;
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.data.structure.itinerary.Leg;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.location.Location;
import org.apache.zest.sample.dcicargo.sample_b.data.structure.voyage.Voyage;
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.*;
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.handling.HandlingEventType.LOAD;
import static org.apache.zest.sample.dcicargo.sample_b.data.structure.handling.HandlingEventType.UNLOAD;
/**
* Inspect Unloaded Cargo (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.
*/
public class InspectUnloadedCargo extends Context
{
DeliveryInspectorRole deliveryInspector;
HandlingEvent unloadEvent;
Location unloadLocation;
Voyage voyage;
RouteSpecification routeSpecification;
Location destination;
Itinerary itinerary;
Integer itineraryProgressIndex;
boolean wasMisdirected;
public InspectUnloadedCargo( Cargo cargo, HandlingEvent handlingEvent )
{
deliveryInspector = rolePlayer( DeliveryInspectorRole.class, cargo );
unloadEvent = handlingEvent;
unloadLocation = unloadEvent.location().get();
voyage = unloadEvent.voyage().get();
routeSpecification = cargo.routeSpecification().get();
destination = routeSpecification.destination().get();
itinerary = cargo.itinerary().get();
wasMisdirected = cargo.delivery().get().isMisdirected().get();
// Before inspection
itineraryProgressIndex = cargo.delivery().get().itineraryProgressIndex().get();
}
public void inspect()
throws InspectionException
{
// Pre-conditions
if( unloadEvent == null || !unloadEvent.handlingEventType()
.get()
.equals( UNLOAD ) || unloadLocation.equals( destination ) )
{
throw new InspectionFailedException( "Can only inspect unloaded cargo that hasn't arrived at destination." );
}
deliveryInspector.inspectUnloadedCargo();
}
@Mixins( DeliveryInspectorRole.Mixin.class )
public interface DeliveryInspectorRole
{
void setContext( InspectUnloadedCargo context );
void inspectUnloadedCargo()
throws InspectionException;
class Mixin
extends RoleMixin<InspectUnloadedCargo>
implements DeliveryInspectorRole
{
@This
Cargo cargo;
Delivery newDelivery;
public void inspectUnloadedCargo()
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.unloadEvent );
newDelivery.transportStatus().set( IN_PORT );
newDelivery.isUnloadedAtDestination().set( false );
// Step 2 - Verify cargo is routed
if( c.itinerary == null )
{
newDelivery.routingStatus().set( NOT_ROUTED );
newDelivery.itineraryProgressIndex().set( 0 );
cargo.delivery().set( newDeliveryBuilder.newInstance() );
throw new CargoNotRoutedException( c.unloadEvent );
}
if( !c.routeSpecification.isSatisfiedBy( c.itinerary ) )
{
newDelivery.routingStatus().set( MISROUTED );
newDelivery.itineraryProgressIndex().set( 0 );
cargo.delivery().set( newDeliveryBuilder.newInstance() );
throw new CargoMisroutedException( c.unloadEvent, c.routeSpecification, c.itinerary );
}
newDelivery.routingStatus().set( ROUTED );
newDelivery.eta().set( c.itinerary.eta() );
// Current itinerary progress
newDelivery.itineraryProgressIndex().set( c.itineraryProgressIndex );
// Step 3 - Verify cargo is on track
Leg plannedCarrierMovement = c.itinerary.leg( c.itineraryProgressIndex );
if( plannedCarrierMovement == null )
{
throw new InspectionFailedException( "Itinerary progress index '" + c.itineraryProgressIndex + "' is invalid!" );
}
Integer itineraryProgressIndex;
// if (c.wasMisdirected && c.unloadLocation.equals( c.routeSpecification.origin().get() ))
if( c.unloadLocation.equals( c.routeSpecification.origin().get() ) )
// if (c.itineraryProgressIndex == -1)
{
/**
* Unloading in the origin of a route specification of a misdirected cargo
* tells us that the cargo has been re-routed (re-routing while on board a
* carrier sets new origin of route specification to arrival location of
* current carrier movement).
*
* Since the current unload was related to the old itinerary, we don't verify
* the misdirection status against the new itinerary.
*
* The itinerary index starts over from the first leg of the new itinerary
* */
itineraryProgressIndex = 0;
}
else if( !plannedCarrierMovement.unloadLocation().get().equals( c.unloadLocation ) )
{
newDelivery.isMisdirected().set( true );
cargo.delivery().set( newDeliveryBuilder.newInstance() );
throw new CargoMisdirectedException( c.unloadEvent, "Itinerary expected unload in "
+ plannedCarrierMovement.unloadLocation()
.get() );
}
else if( !plannedCarrierMovement.voyage().get().equals( c.voyage ) )
{
// Do we care if cargo unloads from an unexpected carrier?
itineraryProgressIndex = c.itineraryProgressIndex + 1;
}
else
{
// Cargo delivery has progressed and we expect a load in next itinerary leg load location
itineraryProgressIndex = c.itineraryProgressIndex + 1;
}
newDelivery.isMisdirected().set( false );
// Modify itinerary progress index according to misdirection status
newDelivery.itineraryProgressIndex().set( itineraryProgressIndex );
// Step 4 - Determine next expected handling event
Leg nextCarrierMovement = c.itinerary.leg( itineraryProgressIndex );
ValueBuilder<NextHandlingEvent> nextHandlingEvent = vbf.newValueBuilder( NextHandlingEvent.class );
nextHandlingEvent.prototype().handlingEventType().set( LOAD );
nextHandlingEvent.prototype().location().set( nextCarrierMovement.loadLocation().get() );
nextHandlingEvent.prototype().time().set( nextCarrierMovement.loadTime().get() );
nextHandlingEvent.prototype().voyage().set( nextCarrierMovement.voyage().get() );
newDelivery.nextHandlingEvent().set( nextHandlingEvent.newInstance() );
// Step 5 - Save cargo delivery snapshot
cargo.delivery().set( newDeliveryBuilder.newInstance() );
}
}
}
}