IBR-DTNSuite  0.12
DeliveryPredictabilityMap.cpp
Go to the documentation of this file.
1 /*
2  * DeliveryPredictabilityMap.cpp
3  *
4  * Created on: 08.01.2013
5  * Author: morgenro
6  */
7 
9 #include "core/BundleCore.h"
10 #include <ibrdtn/utils/Clock.h>
11 #include <ibrcommon/Logger.h>
12 #include <vector>
13 
14 namespace dtn
15 {
16  namespace routing
17  {
19 
21  : NeighborDataSetImpl(DeliveryPredictabilityMap::identifier), _beta(0.0), _gamma(0.0), _lastAgingTime(0), _time_unit(0)
22  {
23  }
24 
25  DeliveryPredictabilityMap::DeliveryPredictabilityMap(const size_t &time_unit, const float &beta, const float &gamma)
26  : NeighborDataSetImpl(DeliveryPredictabilityMap::identifier), _beta(beta), _gamma(gamma), _lastAgingTime(0), _time_unit(time_unit)
27  {
28  }
29 
31  }
32 
34  {
35  return identifier;
36  }
37 
39  {
40  dtn::data::Length len = 0;
41  for(predictmap::const_iterator it = _predictmap.begin(); it != _predictmap.end(); ++it)
42  {
43  /* calculate length of the EID */
44  const std::string eid = it->first.getString();
45  dtn::data::Length eid_len = eid.length();
46  len += data::Number(eid_len).getLength() + eid_len;
47 
48  /* calculate length of the float in fixed notation */
49  const float& f = it->second;
50  std::stringstream ss;
51  ss << f << std::flush;
52 
53  dtn::data::Length float_len = ss.str().length();
54  len += data::Number(float_len).getLength() + float_len;
55  }
56  return data::Number(_predictmap.size()).getLength() + len;
57  }
58 
59  std::ostream& DeliveryPredictabilityMap::serialize(std::ostream& stream) const
60  {
61  stream << data::Number(_predictmap.size());
62  for(predictmap::const_iterator it = _predictmap.begin(); it != _predictmap.end(); ++it)
63  {
64  const std::string eid = it->first.getString();
65  stream << data::Number(eid.length()) << eid;
66 
67  const float& f = it->second;
68  /* write f into a stringstream to get final length */
69  std::stringstream ss;
70  ss << f << std::flush;
71 
72  stream << data::Number(ss.str().length());
73  stream << ss.str();
74  }
75  IBRCOMMON_LOGGER_DEBUG_TAG("DeliveryPredictabilityMap", 20) << "Serialized with " << _predictmap.size() << " items." << IBRCOMMON_LOGGER_ENDL;
76  IBRCOMMON_LOGGER_DEBUG_TAG("DeliveryPredictabilityMap", 60) << *this << IBRCOMMON_LOGGER_ENDL;
77  return stream;
78  }
79 
80  std::istream& DeliveryPredictabilityMap::deserialize(std::istream& stream)
81  {
82  data::Number elements_read(0);
83  data::Number map_size;
84  stream >> map_size;
85 
86  while(elements_read < map_size)
87  {
88  /* read the EID */
89  data::Number eid_len;
90  stream >> eid_len;
91 
92  // create a buffer for the EID
93  std::vector<char> eid_cstr(eid_len.get<size_t>());
94 
95  // read the EID string
96  stream.read(&eid_cstr[0], eid_cstr.size());
97 
98  // convert the string into an EID object
99  dtn::data::EID eid(std::string(eid_cstr.begin(), eid_cstr.end()));
100 
101  if(eid == data::EID())
102  throw dtn::InvalidDataException("EID could not be casted, while parsing a dp_map.");
103 
104  /* read the probability (float) */
105  float f;
106  dtn::data::Number float_len;
107  stream >> float_len;
108 
109  // create a buffer for the data string
110  std::vector<char> f_cstr(float_len.get<size_t>());
111 
112  // read the data string
113  stream.read(&f_cstr[0], f_cstr.size());
114 
115  // convert string data into a stringstream
116  std::stringstream ss(std::string(f_cstr.begin(), f_cstr.end()));
117 
118  // convert string data into a float
119  ss >> f;
120  if(ss.fail())
121  throw dtn::InvalidDataException("Float could not be casted, while parsing a dp_map.");
122 
123  /* check if f is in a proper range */
124  if(f < 0 || f > 1)
125  continue;
126 
127  /* insert the data into the map */
128  _predictmap[eid] = f;
129 
130  elements_read += 1;
131  }
132 
133  IBRCOMMON_LOGGER_DEBUG_TAG("DeliveryPredictabilityMap", 20) << "Deserialized with " << _predictmap.size() << " items." << IBRCOMMON_LOGGER_ENDL;
134  IBRCOMMON_LOGGER_DEBUG_TAG("DeliveryPredictabilityMap", 60) << *this << IBRCOMMON_LOGGER_ENDL;
135  return stream;
136  }
137 
139  {
140  predictmap::const_iterator it;
141  if ((it = _predictmap.find(neighbor)) != _predictmap.end())
142  {
143  return it->second;
144  }
145 
146  throw ValueNotFoundException();
147  }
148 
149  void DeliveryPredictabilityMap::set(const dtn::data::EID &neighbor, float value)
150  {
151  _predictmap[neighbor] = value;
152  }
153 
155  {
156  _predictmap.clear();
157  }
158 
159  void DeliveryPredictabilityMap::update(const dtn::data::EID &host_b, const DeliveryPredictabilityMap &dpm, const float &p_encounter_first)
160  {
161  float p_ab = 0.0f;
162 
163  try {
164  p_ab = get(host_b);
166  p_ab = p_encounter_first;
167  }
168 
172  for (predictmap::const_iterator it = dpm._predictmap.begin(); it != dpm._predictmap.end(); ++it)
173  {
174  const dtn::data::EID &host_c = it->first;
175  const float &p_bc = it->second;
176 
177  // do not update values for the origin host
178  if (host_b.sameHost(host_c)) continue;
179 
180  // do not process values with our own EID
181  if (dtn::core::BundleCore::local.sameHost(host_c)) continue;
182 
183  predictmap::iterator dp_it;
184  if ((dp_it = _predictmap.find(host_c)) != _predictmap.end()) {
185  dp_it->second = max(dp_it->second, p_ab * p_bc * _beta);
186  } else {
187  _predictmap[host_c] = p_ab * p_bc * _beta;
188  }
189  }
190  }
191 
192  void DeliveryPredictabilityMap::age(const float &p_first_threshold)
193  {
195 
196  // prevent double aging
197  if (current_time <= _lastAgingTime) return;
198 
199  const dtn::data::Timestamp k = (current_time - _lastAgingTime) / _time_unit;
200 
201  predictmap::iterator it;
202  for(it = _predictmap.begin(); it != _predictmap.end();)
203  {
204  if(it->first == dtn::core::BundleCore::local)
205  {
206  ++it;
207  continue;
208  }
209 
210  it->second *= pow(_gamma, k.get<int>());
211 
212  if(it->second < p_first_threshold)
213  {
214  _predictmap.erase(it++);
215  } else {
216  ++it;
217  }
218  }
219 
220  _lastAgingTime = current_time;
221  }
222 
223  void DeliveryPredictabilityMap::toString(std::ostream &stream) const
224  {
225  predictmap::const_iterator it;
226  for (it = _predictmap.begin(); it != _predictmap.end(); ++it)
227  {
228  stream << it->first.getString() << ": " << it->second << std::endl;
229  }
230  }
231 
232  std::ostream& operator<<(std::ostream& stream, const DeliveryPredictabilityMap& map)
233  {
234  map.toString(stream);
235  return stream;
236  }
237  } /* namespace routing */
238 } /* namespace dtn */