First commit
[anna.git] / source / comm / IndexedDelivery.cpp
1 // ANNA - Anna is Not 'N' Anymore
2 //
3 // (c) Copyright 2005-2014 Eduardo Ramos Testillano & Francisco Ruiz Rayo
4 //
5 // https://bitbucket.org/testillano/anna
6 //
7 // Redistribution and use in source and binary forms, with or without
8 // modification, are permitted provided that the following conditions
9 // are met:
10 //
11 //     * Redistributions of source code must retain the above copyright
12 // notice, this list of conditions and the following disclaimer.
13 //     * Redistributions in binary form must reproduce the above
14 // copyright notice, this list of conditions and the following disclaimer
15 // in the documentation and/or other materials provided with the
16 // distribution.
17 //     * Neither the name of Google Inc. nor the names of its
18 // contributors may be used to endorse or promote products derived from
19 // this software without specific prior written permission.
20 //
21 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 //
33 // Authors: eduardo.ramos.testillano@gmail.com
34 //          cisco.tierra@gmail.com
35
36
37 #include <algorithm>
38
39 #include <anna/core/tracing/Logger.hpp>
40 #include <anna/core/tracing/TraceMethod.hpp>
41
42 #include <anna/xml/Node.hpp>
43 #include <anna/xml/Attribute.hpp>
44
45 #include <anna/app/functions.hpp>
46
47 #include <anna/comm/Network.hpp>
48 #include <anna/comm/Host.hpp>
49 #include <anna/comm/Server.hpp>
50
51 #include <anna/comm/Communicator.hpp>
52 #include <anna/comm/IndexedDelivery.hpp>
53 #include <anna/comm/functions.hpp>
54
55 using namespace std;
56 using namespace anna;
57
58 void comm::IndexedDelivery::prepare(const int key)
59 throw(RuntimeException) {
60   const int size = comm::Delivery::size();
61
62   if(size == 0) {
63     string msg(asString());
64     msg += " | No resource has been attached";
65     throw RuntimeException(msg, ANNA_FILE_LOCATION);
66   }
67
68   const int index = key % size;
69
70   a_iikey = begin() + index;
71
72   LOGDEBUG(
73     string msg("anna::comm::IndexedDelivery::prepare | Key: ");
74     msg += functions::asString(key);
75     msg += " | ";
76     msg += resource(a_iikey)->asString();
77     Logger::debug(msg, ANNA_FILE_LOCATION);
78   );
79 }
80
81 comm::Resource* comm::IndexedDelivery::do_apply()
82 throw(RuntimeException) {
83   comm::Resource* result = NULL;
84
85   if(a_iikey == comm::Delivery::end())
86     return NULL;
87
88   comm::Resource* w = result = comm::Delivery::resource(a_iikey);
89
90   if(w->isAvailable() == false || w->isEnabled() == false) {
91     LOGWARNING(
92       string msg(asString());
93       msg += " | ";
94       msg += w->asString();
95       msg += " | Unavailable";
96       Logger::warning(msg, ANNA_FILE_LOCATION);
97     );
98     result = NULL;
99   }
100
101   if(a_mode == Mode::Flexible) {
102     iterator end;
103     iterator ii;
104     end = a_iikey;
105
106     if((ii = a_iikey + 1) == comm::Delivery::end())
107       ii = comm::Delivery::begin();
108
109     while(ii != end) {
110       w = comm::Delivery::resource(ii);
111
112       if(w->isAvailable() == true && w->isEnabled() == true) {
113         result = w;
114         break;
115       }
116
117       ii ++;
118
119       if(ii == comm::Delivery::end())
120         ii = comm::Delivery::begin();
121     }
122   }
123
124   return result;
125 }
126
127 string comm::IndexedDelivery::asString() const
128 throw() {
129   string result = className();
130   result += " { ";
131   result += comm::Delivery::asString();
132   result += " | Mode: ";
133   result += (a_mode == Mode::Strict) ? "Strict" : "Flexible";
134   return result += " }";
135 }
136
137 xml::Node* comm::IndexedDelivery::asXML(xml::Node* parent) const
138 throw() {
139   xml::Node* node = parent->createChild("comm.IndexedDelivery");
140   node->createAttribute("Mode", (a_mode == Mode::Strict) ? "Strict" : "Flexible");
141   comm::Service::asXML(node);
142   return node;
143 }
144