session_shared: relay facets for requests
[metaproxy-moved-to-github.git] / src / filter_cgi.cpp
1 /* This file is part of Metaproxy.
2    Copyright (C) 2005-2013 Index Data
3
4 Metaproxy is free software; you can redistribute it and/or modify it under
5 the terms of the GNU General Public License as published by the Free
6 Software Foundation; either version 2, or (at your option) any later
7 version.
8
9 Metaproxy is distributed in the hope that it will be useful, but WITHOUT ANY
10 WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12 for more details.
13
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
17 */
18
19 #include "filter_cgi.hpp"
20 #include <metaproxy/package.hpp>
21 #include <metaproxy/util.hpp>
22 #include "gduutil.hpp"
23 #include <yaz/zgdu.h>
24 #include <yaz/log.h>
25
26 #include <unistd.h>
27 #include <signal.h>
28 #include <sys/wait.h>
29 #include <sstream>
30
31 #include "config.hpp"
32
33 namespace mp = metaproxy_1;
34 namespace yf = mp::filter;
35
36 namespace metaproxy_1 {
37     namespace filter {
38         class CGI::Exec {
39             friend class Rep;
40             friend class CGI;
41             std::string path;
42             std::string program;
43         };
44         class CGI::Rep {
45             friend class CGI;
46             std::list<CGI::Exec> exec_map;
47             std::map<pid_t,pid_t> children;
48             boost::mutex m_mutex;
49         public:
50             ~Rep();
51         };
52     }
53 }
54
55 yf::CGI::CGI() : m_p(new Rep)
56 {
57
58 }
59
60 yf::CGI::Rep::~Rep()
61 {
62     std::map<pid_t,pid_t>::const_iterator it;
63     boost::mutex::scoped_lock lock(m_mutex);
64
65     for (it = children.begin(); it != children.end(); it++)
66         kill(it->second, SIGTERM);
67 }
68
69 yf::CGI::~CGI()
70 {
71 }
72
73 void yf::CGI::process(mp::Package &package) const
74 {
75     Z_GDU *zgdu_req = package.request().get();
76     Z_GDU *zgdu_res = 0;
77
78     if (!zgdu_req)
79         return;
80
81     if (zgdu_req->which != Z_GDU_HTTP_Request)
82     {
83         package.move();
84         return;
85     }
86     std::string path_info;
87     std::string query_string;
88     const char *path = zgdu_req->u.HTTP_Request->path;
89     yaz_log(YLOG_LOG, "path=%s", path);
90     const char *p_cp = strchr(path, '?');
91     if (p_cp)
92     {
93         path_info.assign(path, p_cp - path);
94         query_string.assign(p_cp+1);
95     }
96     else
97         path_info.assign(path);
98
99     std::list<CGI::Exec>::const_iterator it;
100     metaproxy_1::odr odr;
101     for (it = m_p->exec_map.begin(); it != m_p->exec_map.end(); it++)
102     {
103         if (it->path.compare(path_info) == 0)
104         {
105             int r;
106             pid_t pid;
107             int status;
108
109             pid = ::fork();
110             switch (pid)
111             {
112             case 0: /* child */
113                 setenv("PATH_INFO", path_info.c_str(), 1);
114                 setenv("QUERY_STRING", query_string.c_str(), 1);
115                 r = execl(it->program.c_str(), it->program.c_str(), (char *) 0);
116                 if (r == -1)
117                     exit(1);
118                 exit(0);
119                 break;
120             case -1: /* error */
121                 zgdu_res = odr.create_HTTP_Response(
122                     package.session(), zgdu_req->u.HTTP_Request, 400);
123                 package.response() = zgdu_res;
124                 break;
125             default: /* parent */
126                 if (pid)
127                 {
128                     boost::mutex::scoped_lock lock(m_p->m_mutex);
129                     m_p->children[pid] = pid;
130                 }
131                 waitpid(pid, &status, 0);
132
133                 if (pid)
134                 {
135                     boost::mutex::scoped_lock lock(m_p->m_mutex);
136                     m_p->children.erase(pid);
137                 }
138                 zgdu_res = odr.create_HTTP_Response(
139                     package.session(), zgdu_req->u.HTTP_Request, 200);
140                 package.response() = zgdu_res;
141                 break;
142             }
143             return;
144         }
145     }
146     package.move();
147 }
148
149 void yf::CGI::configure(const xmlNode *ptr, bool test_only, const char *path)
150 {
151     for (ptr = ptr->children; ptr; ptr = ptr->next)
152     {
153         if (ptr->type != XML_ELEMENT_NODE)
154             continue;
155         if (!strcmp((const char *) ptr->name, "map"))
156         {
157             CGI::Exec exec;
158
159             const struct _xmlAttr *attr;
160             for (attr = ptr->properties; attr; attr = attr->next)
161             {
162                 if (!strcmp((const char *) attr->name,  "path"))
163                     exec.path = mp::xml::get_text(attr->children);
164                 else if (!strcmp((const char *) attr->name, "exec"))
165                     exec.program = mp::xml::get_text(attr->children);
166                 else
167                     throw mp::filter::FilterException
168                         ("Bad attribute "
169                          + std::string((const char *) attr->name)
170                          + " in cgi section");
171             }
172             m_p->exec_map.push_back(exec);
173         }
174         else
175         {
176             throw mp::filter::FilterException("Bad element "
177                                                + std::string((const char *)
178                                                              ptr->name));
179         }
180     }
181 }
182
183 static mp::filter::Base* filter_creator()
184 {
185     return new mp::filter::CGI;
186 }
187
188 extern "C" {
189     struct metaproxy_1_filter_struct metaproxy_1_filter_cgi = {
190         0,
191         "cgi",
192         filter_creator
193     };
194 }
195
196
197 /*
198  * Local variables:
199  * c-basic-offset: 4
200  * c-file-style: "Stroustrup"
201  * indent-tabs-mode: nil
202  * End:
203  * vim: shiftwidth=4 tabstop=8 expandtab
204  */
205