StreetMovement.java 9.51 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
32
33
34
35
36
37
38
39
40
41
42
43
44
/*
 * 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;

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

import de.tud.kom.p2psim.api.scenario.ConfigurationException;
import de.tud.kom.p2psim.api.topology.Topology;
import de.tud.kom.p2psim.api.topology.movement.MovementModel;
import de.tud.kom.p2psim.api.topology.movement.SimLocationActuator;
import de.tud.kom.p2psim.api.topology.placement.PlacementModel;
import de.tud.kom.p2psim.impl.simengine.Simulator;
import de.tud.kom.p2psim.impl.topology.PositionVector;
import de.tud.kom.p2psim.impl.topology.TopologyFactory;
import de.tud.kom.p2psim.impl.topology.movement.modularosm.GPSCalculation;
import de.tudarmstadt.maki.simonstrator.api.Binder;
import de.tudarmstadt.maki.simonstrator.api.Event;
import de.tudarmstadt.maki.simonstrator.api.EventHandler;
import de.tudarmstadt.maki.simonstrator.api.Randoms;
45
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.AttractionPoint;
46
47
48

/**
 * 
Julian Zobel's avatar
Julian Zobel committed
49
50
51
52
 * 
 * 
 * @author Julian Zobel
 * 
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
 * 
 */
public class StreetMovement implements MovementModel, EventHandler {

	private final int EVENT_MOVE = 1;

	private final int EVENT_INIT = 2;

	protected PositionVector worldDimensions;

	private Set<SimLocationActuator> moveableHosts = new LinkedHashSet<SimLocationActuator>();

	private Map<SimLocationActuator, PositionVector> positions = new LinkedHashMap<SimLocationActuator, PositionVector>();
	
	private Map<SimLocationActuator, AttractionPoint> attractionOfClients = new LinkedHashMap<SimLocationActuator, AttractionPoint>();
	
	List<AttractionPoint> attractionPoints = new LinkedList<AttractionPoint>();
	
	private boolean initialized = false;

	private long timeBetweenMoveOperation = Simulator.SECOND_UNIT;

	private Random rand;

	private double latLeft; //Values from -90 to 90; always smaller than latRight
	private double latRight; //Values from -90 to 90
	private double lonLeft; //Values from -180 to 180; Always smaller than lonRight
	private double lonRight; //Values from -180 to 180
	
	
	public StreetMovement() {
		this.worldDimensions = Binder.getComponentOrNull(Topology.class)
				.getWorldDimensions();
		this.rand = Randoms.getRandom(StreetMovement.class);
		
		latLeft = GPSCalculation.getLatLower();
		latRight = GPSCalculation.getLatUpper();
		lonLeft = GPSCalculation.getLonLeft();
		lonRight = GPSCalculation.getLonRight();
		
		System.out.println("Scheduling first event for Init");
		
		// 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() {

Julian Zobel's avatar
Julian Zobel committed
105
		if (!initialized) {	
106
107
			setTimeBetweenMoveOperations(timeBetweenMoveOperation);

108
109
			for (SimLocationActuator ms : moveableHosts) {				
				initializeStartingPosition(ms);							
110
111
112
113
114
115
116
117
			}			
			
			// initial move
			move();

			initialized = true;
		}
	}
Julian Zobel's avatar
Julian Zobel committed
118
119
120
121
122
123
124
125
126
	
	/**
	 * Assign the given node a random attraction point and place it 
	 * on a randomly selected position on the line between 
	 * the assigned attraction point and another adjacent attraction point.  
	 * 
	 * @param ms
	 */
	private void initializeStartingPosition(SimLocationActuator ms) {		
127
128
129
130
131
132
		// set an initial attraction point
		int index = rand.nextInt(attractionPoints.size());
		AttractionPoint a = attractionPoints.get(index);		
		attractionOfClients.put(ms, a);	
		
		AttractionPoint b = this.returnNextOrLastAttractionPoint(index);
Julian Zobel's avatar
Julian Zobel committed
133
				
134
135
		PositionVector startingPosition = this.returnRandomPositionBetweenPoints(a,b);
		
136
		ms.updateCurrentLocation(startingPosition);			
137
138
139
140
		
		positions.put(ms, ms.getRealPosition());	
		
	}
Julian Zobel's avatar
Julian Zobel committed
141
142
143
144
145
146
147
148
	
	/**
	 * Returns a randomly selected point on the line between the two given attraction points 
	 * 
	 * @param a Attraction Point A
	 * @param b Attraction Point B
	 * @return PositionVector of a randomly selected point on the connecting line of the attraction points
	 */
149
150
151
152
	private PositionVector returnRandomPositionBetweenPoints(AttractionPoint a,
			AttractionPoint b) {		
		double longMin, longMax, latMin, latMax, longNew, latNew;
		
Julian Zobel's avatar
Julian Zobel committed
153
		// Points have different longitude, so only search for random value for longitude
154
155
156
157
		if(a.getLongitudeOrX() != b.getLongitudeOrX()) {
			if(a.getLongitudeOrX() < b.getLongitudeOrX()) {
				longMin = a.getLongitudeOrX();
				longMax = b.getLongitudeOrX();
Julian Zobel's avatar
Julian Zobel committed
158
			} else {
159
160
				longMin = b.getLongitudeOrX();
				longMax = a.getLongitudeOrX();
161
162
163
164
165
166
167
			}
			
			do {
				longNew = rand.nextDouble()*longMax;				
			} while(longNew < longMin);
			
			assert longNew > longMin && longNew <= longMax;
168
			return new PositionVector(longNew, a.getLatitudeOrY());
169
170
		}
		
Julian Zobel's avatar
Julian Zobel committed
171
		// Points have different latitude, so only search for random value for latitude
172
173
174
175
		if(a.getLatitudeOrY() != b.getLatitudeOrY()) {		
			if(a.getLatitudeOrY() < b.getLatitudeOrY()) {
				latMin = a.getLatitudeOrY();
				latMax = b.getLatitudeOrY();
Julian Zobel's avatar
Julian Zobel committed
176
			} else {
177
178
				latMin = b.getLatitudeOrY();
				latMax = a.getLatitudeOrY();
179
180
181
182
183
184
185
			}		
			
			do{
				latNew = rand.nextDouble()*latMax;				
			} while(latNew < latMin);
			
			assert latNew > latMin && latNew <= latMax;
186
			return new PositionVector(a.getLongitudeOrX(), latNew);
187
188
189
190
191
		}
		
		return null;		
	}

192
193
194
195
196
197
	/**
	 * This default implementation relies on {@link PlacementModel}s to be
	 * configured in the {@link TopologyFactory}
	 */
	@Override
	public void placeComponent(SimLocationActuator actuator) {
198
		// not used here i guess
199
200
201
	}

	@Override
202
	public void changeTargetLocation(SimLocationActuator actuator, AttractionPoint ap) {
203
204
205
		/*
		 * Set a new target AP for the current actuator
		 */
206
		attractionOfClients.put(actuator, ap);
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
	}
	

	@Override
	public void addComponent(SimLocationActuator comp) {		
		moveableHosts.add(comp);		
		positions.put(comp, comp.getRealPosition());		
	}

	@Override
	public void setTimeBetweenMoveOperations(long time) {
		if (time > 0) {
			this.timeBetweenMoveOperation = time;
		} else {
			throw new ConfigurationException(
					"time is negative for the Move Operations");
		}
	}

Julian Zobel's avatar
Julian Zobel committed
226
227
228
229
	/**
	 * Move all nodes towards their assigned attraction point
	 */
	protected void move() {		
230
231
232
		for (Entry<SimLocationActuator, AttractionPoint> entry : attractionOfClients.entrySet()) {
			
			SimLocationActuator ms = entry.getKey();
233
			PositionVector attractionCenter = (PositionVector) entry.getValue();
234
235
			PositionVector destination = new PositionVector(attractionCenter);

Julian Zobel's avatar
Julian Zobel committed
236
			if (destination.distanceTo(ms.getRealPosition()) > ms.getMovementSpeed()) {				
237
				PositionVector newPosition = ms.getRealPosition().moveStep(destination, ms.getMovementSpeed());
238
				ms.updateCurrentLocation(newPosition);
Julian Zobel's avatar
Julian Zobel committed
239
				positions.put(ms, newPosition);				
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
			} else {
				assignNewAttractionPoint(ms);
			}			
			
			/*
			 * Check for negative or out of bound coordinates!
			 */
			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();
		}

		Event.scheduleWithDelay(timeBetweenMoveOperation, this, null,
				EVENT_MOVE);
	}
	
Julian Zobel's avatar
Julian Zobel committed
261
262
263
264
265
	/**
	 * Assign the node a random attraction point, that is adjacent to the last one
	 * 
	 * @param ms The given node
	 */
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
	private void assignNewAttractionPoint(SimLocationActuator ms) {
		
		if(attractionOfClients.containsKey(ms)) {
			boolean updownrand = rand.nextBoolean();
			
			int index = attractionPoints.indexOf(attractionOfClients.get(ms));
			
			if(updownrand) {
				index = ((index + 1) % attractionPoints.size());
			} else {				
				index = ((index - 1 + attractionPoints.size()) % attractionPoints.size());
			}
			
			attractionOfClients.put(ms, attractionPoints.get(index));		
		} else {
			attractionOfClients.put(ms, attractionPoints.get(rand.nextInt(attractionPoints.size())));	
		}
	}
284
	
Julian Zobel's avatar
Julian Zobel committed
285
286
287
288
289
290
	/**
	 * Returns the next or last attraction point of the list to the given list index 
	 * 
	 * @param index Index of the attraction point, for which another attraction point should be returned
	 * @return Attraction Point
	 */
291
292
293
294
295
296
297
298
299
300
301
	private AttractionPoint returnNextOrLastAttractionPoint(int index) {
		boolean updownrand = rand.nextBoolean();
				
		if(updownrand) {
			index = ((index + 1) % attractionPoints.size());
		} else {				
			index = ((index - 1 + attractionPoints.size()) % attractionPoints.size());
		}
		
		return attractionPoints.get(index);
	}
302
303
304
305
306
307
308
309
310

	@Override
	public void eventOccurred(Object content, int type) {
		if (type == EVENT_INIT) {
			initialize();
		} else if (type == EVENT_MOVE) {
			move();
		}
	}
Julian Zobel's avatar
Julian Zobel committed
311
	
312
	public List<AttractionPoint> getAttractionPoints() {
313
314
315
316
317
318
319
		return attractionPoints;
	}
	
	public void setAttractionPoint(AttractionPoint point) {
		this.attractionPoints.add(point);
	}
}