]> git.stg.codes - stg.git/blob - stglibs/srvconf.lib/parsers/get_container.h
31bb425032ad892ef63cb33e9f7a79e0bc825fb4
[stg.git] / stglibs / srvconf.lib / parsers / get_container.h
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 #ifndef __STG_STGLIBS_SRVCONF_PARSER_GET_CONTAINER_H__
22 #define __STG_STGLIBS_SRVCONF_PARSER_GET_CONTAINER_H__
23
24 #include "base.h"
25
26 #include <string>
27
28 namespace STG
29 {
30 namespace GET_CONTAINER
31 {
32
33 template <typename ELEMENT_PARSER>
34 class PARSER: public STG::PARSER
35 {
36 public:
37     typedef std::vector<typename ELEMENT_PARSER::INFO> INFO;
38     typedef void (* CALLBACK)(bool result, const std::string & reason, const INFO & info, void * data);
39     PARSER(const std::string & t, CALLBACK f, void * d)
40         : tag(t), callback(f), data(d),
41           elementParser(&PARSER<ELEMENT_PARSER>::ElementCallback, this),
42           depth(0), parsingAnswer(false)
43     {}
44     int  ParseStart(const char * el, const char ** attr)
45     {
46     depth++;
47     if (depth == 1 && strcasecmp(el, tag.c_str()) == 0)
48         parsingAnswer = true;
49
50     if (depth > 1 && parsingAnswer)
51         elementParser.ParseStart(el, attr);
52
53     return 0;
54     }
55     void ParseEnd(const char * el)
56     {
57     depth--;
58     if (depth > 0 && parsingAnswer)
59         elementParser.ParseEnd(el);
60
61     if (depth == 0 && parsingAnswer)
62         {
63         if (callback)
64             callback(error.empty(), error, info, data);
65         error.clear();
66         info.clear();
67         parsingAnswer = false;
68         }
69     }
70
71 private:
72     std::string tag;
73     CALLBACK callback;
74     void * data;
75     ELEMENT_PARSER elementParser;
76     INFO info;
77     int depth;
78     bool parsingAnswer;
79     std::string error;
80
81     void AddElement(const typename ELEMENT_PARSER::INFO & elementInfo)
82     {
83     info.push_back(elementInfo);
84     }
85     void SetError(const std::string & e) { error = e; }
86
87     static void ElementCallback(bool result, const std::string& reason, const typename ELEMENT_PARSER::INFO & info, void * data)
88     {
89     PARSER<ELEMENT_PARSER> * parser = static_cast<PARSER<ELEMENT_PARSER> *>(data);
90     if (!result)
91         parser->SetError(reason);
92     else
93         parser->AddElement(info);
94     }
95 };
96
97 } // namespace GET_CONTAINER
98 } // namespace STG
99
100 #endif