38e6793ff31a5c95618b9a20e463ffedf198ec74
[gssweb.git] / json_gssapi / src / GSSRequest.cpp
1 /*
2  * Copyright (c) 2014, 2015 JANET(UK)
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  *
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  *
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * 3. Neither the name of JANET(UK) nor the names of its contributors
17  *    may be used to endorse or promote products derived from this software
18  *    without specific prior written permission.
19  *
20  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
24  * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
25  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
26  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
27  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
29  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
30  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
31  * OF THE POSSIBILITY OF SUCH DAMAGE.
32  *
33  */
34
35 #include <cstddef>
36 #include <stdexcept>
37 #include <string.h>
38
39 #include "commands/GSSAcquireCred.h"
40 #include "commands/GSSInitSecContext.h"
41 #include "commands/GSSImportName.h"
42 #include "commands/GSSDisplayName.h"
43 #include "GSSRequest.h"
44 #include "GSSException.h"
45
46 using std::bad_alloc;
47
48 GSSRequest::GSSRequest ( string jsonString )
49 {
50   /* Local variables */
51   /* Error checking */
52   /* Setup */
53   /* Main processing */
54   response = JSONObject();
55   cmd = NULL;
56   requestString = jsonString;
57   
58   /* Cleanup */
59   /* Return */
60 }
61
62 void GSSRequest::execute()
63 {
64   try {
65     /* variables */
66     /* Error checking */
67     /* Setup */
68     parseJSON();
69     getCommand();
70     
71     /* Main processing */
72     if (NULL != cmd)
73       cmd->execute();
74   }
75   catch (GSSException e)
76   {
77     /* Cleanup */
78     if ( NULL != cmd )
79     {
80       delete(cmd);
81       cmd = NULL;
82     }
83
84     JSONObject return_values;
85     return_values.set("major_status", e.getMajor());
86     return_values.set("minor_status", e.getMinor());
87     return_values.set("what", e.what());
88     response.set("return_values", return_values);
89   }
90   catch (std::invalid_argument e)
91   {
92     /* Cleanup */
93     if ( NULL != cmd )
94     {
95       delete(cmd);
96       cmd = NULL;
97     }
98
99     JSONObject return_values, errors;
100     errors.set("major_status_message", "An error occurred in parsing the JSON arguments.\0");
101     errors.set("minor_status_message", e.what());
102     return_values.set("errors", errors);
103     return_values.set("major_status", -1);
104     return_values.set("minor_status", -1);
105     response.set("return_values", return_values);
106   }
107
108   /* return */
109 }
110
111
112
113 void GSSRequest::parseJSON()
114 {
115   /* variables */
116   json_error_t jsonErr;
117   
118   try {
119     JSONObject cookies;
120     request = JSONObject::load(requestString.c_str(), 0, &jsonErr);
121     cookies = request.get("cookies");
122     response.set("cookies", cookies );
123     response.set("method", request.get("method").string());
124   }
125   /* bad_alloc is thrown when JSONObject can't parse the input string as JSON */
126   catch ( bad_alloc& ) 
127   {
128     // Top-level response
129     response.set("error_message", "Could not parse the input JSON.");
130     response.set("original_message", requestString.c_str());
131   }
132 }
133
134
135 void GSSRequest::getCommand()
136 {
137   string method;
138   JSONObject arguments = request.get("arguments");
139   
140   /* Error checking */
141   /* Setup */
142   if (request.get("method").isNull() )
143     method = "";
144   else
145     method = string( request.get("method").string() );
146   
147   if ( "gss_import_name" == method ) 
148   {
149     cmd = new GSSImportName ( &arguments );
150   } 
151   else if ( "gss_init_sec_context" == method ) 
152   {
153     cmd = new GSSInitSecContext ( &arguments );
154   } 
155   else if ( "gss_acquire_cred" == method ) 
156   {
157     cmd = new GSSAcquireCred ( &arguments );
158   } 
159   else if ( "gss_display_name" == method )
160   {
161     cmd = new GSSDisplayName ( &arguments );
162   }
163   else 
164   {
165     string error_message = string("Unrecognized command: ") + method;
166     response.set("error_message", error_message.c_str() );
167     response.set("original_message", requestString.c_str());
168     cmd = NULL;
169   }
170 }
171
172 string GSSRequest::getResponse()
173 {
174   /* Variables */
175   JSONObject *return_values;
176   string gssResponse;
177   
178   /* Main processing */
179   // Put the return values into the response, assuming that the command
180   // was actually executed.
181   if (NULL != cmd)
182   {
183     return_values = cmd->toJSON();
184     response.set("return_values", *return_values);
185   }
186   
187   // Convert the response into a string to return.
188   gssResponse = string( response.dump() );
189   
190   /* Return */
191   return(gssResponse);
192 }
193
194
195
196 char *gss_request(char *json_string)
197 {
198   /* Variables */
199   GSSRequest *req = new GSSRequest(string(json_string));
200   
201   /* Error checking */
202   // An empty json_string could be an error, but GSSRequest does
203   // a good job of handling it.
204   
205   /* Setup */
206   /* Main processing */
207   req->execute();
208   return strdup(req->getResponse().c_str());
209 }
210
211 void deallocate_reply(char *reply)
212 {
213   delete(reply);
214 }