]> git.stg.codes - stg.git/blob - stglibs/srvconf.lib/parsers/get_tariff.cpp
0b96bec3b8e234aced6461e2647fe8ea590d7600
[stg.git] / stglibs / srvconf.lib / parsers / get_tariff.cpp
1 /*
2  *    This program is free software; you can redistribute it and/or modify
3  *    it under the terms of the GNU General Public License as published by
4  *    the Free Software Foundation; either version 2 of the License, or
5  *    (at your option) any later version.
6  *
7  *    This program is distributed in the hope that it will be useful,
8  *    but WITHOUT ANY WARRANTY; without even the implied warranty of
9  *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10  *    GNU General Public License for more details.
11  *
12  *    You should have received a copy of the GNU General Public License
13  *    along with this program; if not, write to the Free Software
14  *    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
15  */
16
17 /*
18  *    Author : Maxim Mamontov <faust@stargazer.dp.ua>
19  */
20
21 #include "get_tariff.h"
22
23 #include "parsers/property.h"
24
25 #include "stg/common.h"
26
27 #include <utility>
28
29 #include <strings.h>
30
31 using namespace STG;
32
33 namespace
34 {
35
36 template <typename A, typename T>
37 class AOS_PARSER : public BASE_PROPERTY_PARSER
38 {
39     public:
40         typedef bool (* FUNC)(const char **, A &, T A::value_type:: *);
41         AOS_PARSER(A & a, T A::value_type:: * fld, FUNC f) : array(a), field(fld), func(f) {}
42         virtual bool Parse(const char ** attr, const std::string & /*attrName*/, const std::string & /*fromEncoding*/) { return func(attr, array, field); }
43     private:
44         A & array;
45         T A::value_type:: * field;
46         FUNC func;
47 };
48
49 template <typename A, typename T>
50 inline
51 void AddAOSParser(PROPERTY_PARSERS & parsers, const std::string & name, A & array, T A::value_type:: * field, const typename AOS_PARSER<A, T>::FUNC & func)
52 {
53     parsers.insert(std::make_pair(ToLower(name), new AOS_PARSER<A, T>(array, field, func)));
54 }
55
56 bool GetTimeSpan(const char ** attr, DIRPRICE_DATA & value, const std::string & attrName)
57 {
58 if (CheckValue(attr, attrName))
59     {
60     int hb = 0;
61     int mb = 0;
62     int he = 0;
63     int me = 0;
64     if (ParseTariffTimeStr(attr[1], hb, mb, he, me) == 0)
65         {
66         value.hDay = hb;
67         value.mDay = mb;
68         value.hNight = he;
69         value.mNight = me;
70         return true;
71         }
72     }
73 return false;
74 }
75
76 template <typename T>
77 bool GetTraffType(const char ** attr, T & value, const std::string & attrName)
78 {
79 if (!CheckValue(attr, attrName))
80     return false;
81 value = TARIFF::StringToTraffType(attr[1]);
82 return true;
83 }
84
85 template <typename T>
86 bool GetPeriod(const char ** attr, T & value, const std::string & attrName)
87 {
88 if (!CheckValue(attr, attrName))
89     return false;
90 std::string type(attr[1]);
91 if (type == "day")
92     value = TARIFF::DAY;
93 else if (type == "month")
94     value = TARIFF::MONTH;
95 else
96     return false;
97 return true;
98 }
99
100 template <typename T>
101 bool GetChangePolicy(const char ** attr, T & value, const std::string & attrName)
102 {
103 if (!CheckValue(attr, attrName))
104     return false;
105 std::string type(attr[1]);
106 if (type == "allow")
107         value = TARIFF::ALLOW;
108 else if (type == "to_cheap")
109         value = TARIFF::TO_CHEAP;
110 else if (type == "to_expensive")
111         value = TARIFF::TO_EXPENSIVE;
112 else if (type == "deny")
113         value = TARIFF::DENY;
114 else
115     return false;
116 return true;
117 }
118
119 template <typename T>
120 bool GetChangePolicyTimeout(const char ** attr, T & value, const std::string & attrName)
121 {
122 if (!CheckValue(attr, attrName))
123     return false;
124 value = readTime(attr[1]);
125 return true;
126 }
127
128 template <typename A, typename T>
129 bool GetSlashedValue(const char ** attr, A & array, T A::value_type:: * field)
130 {
131 if (!CheckValue(attr, "value"))
132     return false;
133 const char * start = attr[1];
134 size_t item = 0;
135 const char * pos = NULL;
136 while ((pos = strchr(start, '/')) && item < array.size())
137     {
138     if (str2x(std::string(start, pos), array[item++].*field))
139             return false;
140     start = pos + 1;
141     }
142 if (item < array.size())
143     if (str2x(start, array[item].*field))
144         return false;
145 return true;
146 }
147
148 } // namespace anonymous
149
150 GET_TARIFF::PARSER::PARSER(CALLBACK f, void * d, const std::string & e)
151     : callback(f),
152       data(d),
153       encoding(e),
154       depth(0),
155       parsingAnswer(false)
156 {
157     AddParser(propertyParsers, "fee", info.tariffConf.fee);
158     AddParser(propertyParsers, "passiveCost", info.tariffConf.passiveCost);
159     AddParser(propertyParsers, "free", info.tariffConf.free);
160     AddParser(propertyParsers, "traffType", info.tariffConf.traffType, GetTraffType);
161     AddParser(propertyParsers, "period", info.tariffConf.period, GetPeriod);
162     AddParser(propertyParsers, "changePolicy", info.tariffConf.changePolicy, GetChangePolicy);
163     for (size_t i = 0; i < DIR_NUM; ++i)
164         AddParser(propertyParsers, "time" + unsigned2str(i), info.dirPrice[i], GetTimeSpan);
165     AddAOSParser(propertyParsers, "priceDayA", info.dirPrice, &DIRPRICE_DATA::priceDayA, GetSlashedValue);
166     AddAOSParser(propertyParsers, "priceDayB", info.dirPrice, &DIRPRICE_DATA::priceDayB, GetSlashedValue);
167     AddAOSParser(propertyParsers, "priceNightA", info.dirPrice, &DIRPRICE_DATA::priceNightA, GetSlashedValue);
168     AddAOSParser(propertyParsers, "priceNightB", info.dirPrice, &DIRPRICE_DATA::priceNightB, GetSlashedValue);
169     AddAOSParser(propertyParsers, "singlePrice", info.dirPrice, &DIRPRICE_DATA::singlePrice, GetSlashedValue);
170     AddAOSParser(propertyParsers, "noDiscount", info.dirPrice, &DIRPRICE_DATA::noDiscount, GetSlashedValue);
171     AddAOSParser(propertyParsers, "threshold", info.dirPrice, &DIRPRICE_DATA::threshold, GetSlashedValue);
172 }
173 //-----------------------------------------------------------------------------
174 GET_TARIFF::PARSER::~PARSER()
175 {
176     PROPERTY_PARSERS::iterator it(propertyParsers.begin());
177     while (it != propertyParsers.end())
178         delete (it++)->second;
179 }
180 //-----------------------------------------------------------------------------
181 int GET_TARIFF::PARSER::ParseStart(const char * el, const char ** attr)
182 {
183 depth++;
184 if (depth == 1)
185     ParseTariff(el, attr);
186
187 if (depth == 2 && parsingAnswer)
188     ParseTariffParams(el, attr);
189
190 return 0;
191 }
192 //-----------------------------------------------------------------------------
193 void GET_TARIFF::PARSER::ParseEnd(const char * /*el*/)
194 {
195 depth--;
196 if (depth == 0 && parsingAnswer)
197     {
198     if (callback)
199         callback(error.empty(), error, info, data);
200     error.clear();
201     parsingAnswer = false;
202     }
203 }
204 //-----------------------------------------------------------------------------
205 void GET_TARIFF::PARSER::ParseTariff(const char * el, const char ** attr)
206 {
207 if (strcasecmp(el, "tariff") == 0)
208     {
209     if (attr && attr[0] && attr[1])
210         {
211         if (strcasecmp(attr[1], "error") == 0)
212             {
213             if (attr[2] && attr[3])
214                 error = attr[3];
215             else
216                 error = "Tariff not found.";
217             }
218         else
219             {
220             parsingAnswer = true;
221             if (strcasecmp(attr[0], "name") == 0)
222                 info.tariffConf.name = attr[1];
223             }
224         }
225     else
226         parsingAnswer = true;
227     }
228 }
229 //-----------------------------------------------------------------------------
230 void GET_TARIFF::PARSER::ParseTariffParams(const char * el, const char ** attr)
231 {
232 if (!TryParse(propertyParsers, ToLower(el), attr, encoding))
233     error = std::string("Invalid parameter '") + el + "'.";
234 }