ModularMovementModel.java 11.3 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/*
 * Copyright (c) 2005-2010 KOM – Multimedia Communications Lab
 *
 * This file is part of PeerfactSim.KOM.
 * 
 * PeerfactSim.KOM is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * any later version.
 * 
 * PeerfactSim.KOM is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with PeerfactSim.KOM.  If not, see <http://www.gnu.org/licenses/>.
 *
 */

package de.tud.kom.p2psim.impl.topology.movement.modularosm;

import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Random;
import java.util.Set;
import java.util.Vector;

32
33
import javax.swing.JComponent;

34
import de.tud.kom.p2psim.api.scenario.ConfigurationException;
35
import de.tud.kom.p2psim.api.topology.Topology;
36
import de.tud.kom.p2psim.api.topology.movement.MovementModel;
37
import de.tud.kom.p2psim.api.topology.movement.SimLocationActuator;
38
import de.tud.kom.p2psim.api.topology.movement.local.LocalMovementStrategy;
39
import de.tud.kom.p2psim.api.topology.placement.PlacementModel;
40
41
import de.tud.kom.p2psim.impl.simengine.Simulator;
import de.tud.kom.p2psim.impl.topology.PositionVector;
42
import de.tud.kom.p2psim.impl.topology.TopologyFactory;
43
import de.tud.kom.p2psim.impl.topology.movement.modularosm.attraction.IAttractionGenerator;
44
import de.tud.kom.p2psim.impl.topology.movement.modularosm.mapvisualization.IMapVisualization;
45
import de.tud.kom.p2psim.impl.topology.movement.modularosm.transition.ITransitionStrategy;
46
import de.tud.kom.p2psim.impl.topology.movement.modularosm.transition.ITransitionStrategy.AttractionAssignmentListener;
47
48
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView.VisualizationInjector;
import de.tud.kom.p2psim.impl.util.Either;
49
import de.tudarmstadt.maki.simonstrator.api.Binder;
50
51
52
import de.tudarmstadt.maki.simonstrator.api.Event;
import de.tudarmstadt.maki.simonstrator.api.EventHandler;
import de.tudarmstadt.maki.simonstrator.api.Randoms;
53
import de.tudarmstadt.maki.simonstrator.api.Time;
54
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.AttractionPoint;
55
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.LocationActuator;
56
57
58

/**
 * Modular Movement Model uses different models/strategies to create a movement
59
 * model. In this implementation, it has 3 different models/strategies.
60
61
 * <p>
 * M0: AttractionGenerator -> Generates the {@link AttractionPoint}s and place
62
63
 * them on the map. The {@link AttractionPoint}s can't be moved, because they
 * are static POIs from real-world data!
64
 * <p>
65
66
 * M1: A general {@link MovementModel} is not used, because we use static
 * attraction points.
67
68
69
70
71
72
73
74
75
76
77
78
 * <p>
 * M2: The {@link TransitionStrategy}! It takes the Hosts, which should be moved
 * around, but calculates only the assignment to the {@link AttractionPoint}s.
 * It doesn't move the Hosts! It will be only assignment a new AttractionPoint!
 * 
 * <p>
 * M3: The {@link LocalMovementStrategy} is responsible for the movement of the
 * Hosts. It moves the hosts to the assigned AttractionPoint, and if the
 * AttractionPoint has moved, then will be followed. The
 * {@link LocalMovementStrategy} will be called from the
 * {@link ModularMovementModel} to do a Movement!
 * <p>
79
 * This class contains all three components and manage the data exchange.
80
81
82
83
84
85
86
87
88
89
 * Additionally it contains an periodic operation, which handle the movement of
 * all hosts. This mean, that it will be call the {@link LocalMovementStrategy}
 * with the destination. Please take care, that the handling of the movement of
 * the AttractionPoints will be handled by the movement model in M1! <br>
 * Further it contains an offset for every Host, which will be added to the
 * destination point (AttractionPoint), so that not all hosts, which are
 * assigned to one {@link AttractionPoint}, lies on the same point.<br>
 * 
 * 
 * 
90
 * @author Martin Hellwig, Christoph Muenker
91
 * @version 1.0, 07.07.2015
92
 */
93
public class ModularMovementModel implements MovementModel, EventHandler, AttractionAssignmentListener {
94
95
96
97

	private final int EVENT_MOVE = 1;

	private final int EVENT_INIT = 2;
98

99
100
101
102
103
104
105
	protected PositionVector worldDimensions;

	protected ITransitionStrategy transition;

	protected IAttractionGenerator attractionGenerator;

	protected LocalMovementStrategy localMovementStrategy;
106

107
	protected IMapVisualization mapVisualization;
108

109
	private Set<SimLocationActuator> moveableHosts = new LinkedHashSet<SimLocationActuator>();
110
111
112
	
	private Map<SimLocationActuator, PositionVector> currentTarget = new LinkedHashMap<>();
	
113
114
115
116
117
118
119
	private boolean initialized = false;

	private long timeBetweenMoveOperation = Simulator.SECOND_UNIT;

	private Random rand;

	public ModularMovementModel() {
120
121
		this.worldDimensions = Binder.getComponentOrNull(Topology.class)
				.getWorldDimensions();
122
123
124
125
126
127
128
129
130
131
132
		this.rand = Randoms.getRandom(ModularMovementModel.class);

		// scheduling initalization!
		Event.scheduleImmediately(this, null, EVENT_INIT);
	}

	/**
	 * This Method will be not called from the Components. So we call this
	 * manually!
	 */
	public void initialize() {
133

134
135
136
		if (!initialized) {
			VisualizationInjector
					.injectComponent(new ModularMovementModelViz(this));
137
138
139
			if (mapVisualization != null) {
				VisualizationInjector.injectComponent(mapVisualization);
			}
140
141
142
143

			checkConfiguration();

			// setWayPointModel
144
145
146
147
			localMovementStrategy.setObstacleModel(Binder
					.getComponentOrNull(Topology.class).getObstacleModel());
			localMovementStrategy.setWaypointModel(Binder
					.getComponentOrNull(Topology.class).getWaypointModel());
148

149
150
151
152
153
154
			/*
			 * Scale depending on calculation interval, if interval != 1 Second.
			 */
			localMovementStrategy
					.setScaleFactor(timeBetweenMoveOperation / (double) Time.SECOND);

155
156
			List<AttractionPoint> attractionPoints = attractionGenerator
					.getAttractionPoints();
157
			transition.setAttractionPoints(attractionPoints);
158
			transition.addAttractionAssignmentListener(this);
159
160
161

			// This adds the mobile hosts (smartphones/users) to the transition
			// strategy
162
			for (SimLocationActuator ms : moveableHosts) {
163
164
165
166
167
168
169
170
171
172
173
				transition.addComponent(ms);
			}

			setTimeBetweenMoveOperations(timeBetweenMoveOperation);

			// initial move
			move();

			initialized = true;
		}
	}
174

175
176
177
178
179
180
181
182
	/**
	 * This default implementation relies on {@link PlacementModel}s to be
	 * configured in the {@link TopologyFactory}
	 */
	@Override
	public void placeComponent(SimLocationActuator actuator) {
		// not supported
	}
183

184
	@Override
185
186
187
188
189
190
	public void changeTargetLocation(SimLocationActuator actuator, AttractionPoint ap) {
		transition.updateTargetAttractionPoint(actuator, ap);
	}
	
	@Override
	public AttractionPoint getTargetLocation(SimLocationActuator actuator) {
191
		return transition.getAssignment(actuator);
192
	}
193
194
195
196
197
198
	
	@Override
	public Set<AttractionPoint> getAllAttractionPoints()
			throws UnsupportedOperationException {
		return transition.getAllAttractionPoints();
	}
199

200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
	private void checkConfiguration() {
		if (localMovementStrategy == null) {
			throw new ConfigurationException(
					"LocalMovementStrategy is missing in ModularMovementModel!");
		}
		if (transition == null) {
			throw new ConfigurationException(
					"TransitionStrategy is missing in ModularMovementModel!");
		}
		if (attractionGenerator == null) {
			throw new ConfigurationException(
					"AttractionGenerator is missing in ModularMovementModel!");
		}
	}

	@Override
216
	public void addComponent(SimLocationActuator comp) {
217
218
		moveableHosts.add(comp);
	}
219
220
221
222
	
	public Set<SimLocationActuator> getAllLocationActuators() {
		return moveableHosts;
	}
223
224
225
226
227
228
229
230
231
232

	@Override
	public void setTimeBetweenMoveOperations(long time) {
		if (time > 0) {
			this.timeBetweenMoveOperation = time;
		} else {
			throw new ConfigurationException(
					"time is negative for the Move Operations");
		}
	}
233
234
235
236
237
238
239
240
241
	
	@Override
	public void updatedAttractionAssignment(SimLocationActuator component,
			AttractionPoint newAssignment) {
		/*
		 * Use this method to calculate the offset and target location for a
		 * host.
		 */
		PositionVector attractionCenter = (PositionVector) newAssignment;
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
		PositionVector destination = null;
		
		int tries = 0;
		do {
			destination = new PositionVector(attractionCenter);
			// Gaussian with std = 1 --> >99% of nodes
			PositionVector offset = new PositionVector(
					rand.nextGaussian() * newAssignment.getRadius() / 3,
					rand.nextGaussian() * newAssignment.getRadius() / 3);
			destination.add(offset);
			// Check constraints
			if (destination.getX() < 0.0
					|| destination.getX() > Binder
							.getComponentOrNull(Topology.class)
							.getWorldDimensions().getX() || destination.getY() < 0.0
					|| destination.getY() > Binder
							.getComponentOrNull(Topology.class)
							.getWorldDimensions().getY()) {
				destination = null;
				if (tries > 100) {
					throw new AssertionError("Unable to find a valid target destination within <100 tries.");
				}
			}
			tries++;
		} while (destination == null);
		
268
		currentTarget.put(component, destination);
269
270
271
	}

	protected void move() {
272
273
274
		for (SimLocationActuator component : moveableHosts) {
			assert currentTarget.containsKey(component);
			doLocalMovement(component, currentTarget.get(component));
275
276
277
278
279
280
281
282
283
284
285
286
287
288
		}
		Event.scheduleWithDelay(timeBetweenMoveOperation, this, null,
				EVENT_MOVE);
	}

	/**
	 * 
	 * Ask the local movement strategy for the next position. It may return the
	 * next position or a boolean with true to notify the movement model that it
	 * can't get any closer to the current way point.
	 * 
	 * @param ms
	 * @param destination
	 */
289
	private void doLocalMovement(SimLocationActuator ms,
290
291
292
293
294
			PositionVector destination) {

		Either<PositionVector, Boolean> either = localMovementStrategy
				.nextPosition(ms, destination);
		if (either.hasLeft()) {
295
			ms.updateCurrentLocation(either.getLeft());
296
297
298
			/*
			 * Check for negative or out of bound coordinates!
			 */
299
300
301
302
303
304
305
306
			assert ms.getRealPosition().getX() >= 0.0
					&& ms.getRealPosition().getX() <= Binder
							.getComponentOrNull(Topology.class)
							.getWorldDimensions().getX();
			assert ms.getRealPosition().getY() >= 0.0
					&& ms.getRealPosition().getY() <= Binder
							.getComponentOrNull(Topology.class)
							.getWorldDimensions().getY();
Björn Richerzhagen's avatar
Björn Richerzhagen committed
307
308
		} else {
			transition.reachedAttractionPoint(ms);
309
310
311
		}
	}

312
313
	public void setIAttractionGenerator(
			IAttractionGenerator attractionGenerator) {
314
315
316
317
318
319
320
321
322
323
324
		this.attractionGenerator = attractionGenerator;
	}

	public void setLocalMovementStrategy(
			LocalMovementStrategy localMovementStrategy) {
		this.localMovementStrategy = localMovementStrategy;
	}

	public void setITransitionStrategy(ITransitionStrategy transition) {
		this.transition = transition;
	}
325

326
327
328
	public void setIMapVisualization(IMapVisualization mapVisualization) {
		this.mapVisualization = mapVisualization;
	}
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343

	@Override
	public void eventOccurred(Object content, int type) {
		if (type == EVENT_INIT) {
			initialize();
		} else if (type == EVENT_MOVE) {
			move();
		}
	}

	/**
	 * Only for visualization!
	 * 
	 * @return
	 */
344
345
346
	public List<AttractionPoint> getAttractionPoints() {
		return new Vector<AttractionPoint>(transition.getAllAttractionPoints());
	}
347
}