2 * Copyright (C) 2015 "IoT.bzh"
3 * Author "Fulup Ar Foll"
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation, either version 3 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 #include "local-def.h"
22 // Dummy sample of Client Application Context
25 void *whateveryouwant;
26 } MyClientApplicationHandle;
29 // Request Creation of new context if it does not exist
30 STATIC json_object* clientContextCreate (AFB_request *request) {
34 AFB_clientCtx *client=request->client; // get client context from request
36 // check we do not already have a session
37 if ((client != NULL) && (client->ctx != NULL)) {
38 request->errcode=MHD_HTTP_FORBIDDEN;
39 return (jsonNewMessage(AFB_FAIL, "Token exist use refresh"));
42 // request a new client context token and check result
43 if (AFB_UNAUTH == ctxTokenCreate (request)) {
44 request->errcode=MHD_HTTP_UNAUTHORIZED;
45 jresp= jsonNewMessage(AFB_FAIL, "No/Invalid initial token provided [should match --token=xxxx]");
49 // request a new client context token and check result
50 if (AFB_SUCCESS != ctxTokenCreate (request)) {
51 request->errcode=MHD_HTTP_UNAUTHORIZED;
52 jresp= jsonNewMessage(AFB_FAIL, "Token Session Not Activated [restart with --token=xxxx]");
56 // add a client context to session
57 client->ctx = malloc (sizeof (MyClientApplicationHandle));
59 // Send response to UI
60 jresp = json_object_new_object();
61 json_object_object_add(jresp, "token", json_object_new_string (client->token));
66 // Renew an existing context
67 STATIC json_object* clientContextRefresh (AFB_request *request) {
70 // note: we do not need to parse the old token as clientContextRefresh doit for us
71 if (AFB_SUCCESS != ctxTokenRefresh (request)) {
72 request->errcode=MHD_HTTP_UNAUTHORIZED;
73 jresp= jsonNewMessage(AFB_FAIL, "Token Exchange Broken Refresh Refused");
75 jresp = json_object_new_object();
76 json_object_object_add(jresp, "token", json_object_new_string (request->client->token));
83 // Verify a context is still valid
84 STATIC json_object* clientContextCheck (AFB_request *request) {
86 json_object *jresp = json_object_new_object();
88 // add an error code to respond
89 if (AFB_SUCCESS != ctxTokenCheck (request)) {
90 request->errcode=MHD_HTTP_UNAUTHORIZED;
91 json_object_object_add(jresp, "isvalid", json_object_new_boolean (FALSE));
93 json_object_object_add(jresp, "isvalid", json_object_new_boolean (TRUE));
101 // Close and Free context
102 STATIC json_object* clientContextReset (AFB_request *request) {
105 // note: we do not need to parse the old token as clientContextRefresh doit for us
106 if (AFB_SUCCESS != ctxTokenReset (request)) {
107 request->errcode=MHD_HTTP_UNAUTHORIZED;
108 jresp= jsonNewMessage(AFB_FAIL, "No Token Client Context [use --token=xxx]");
110 jresp = json_object_new_object();
111 json_object_object_add(jresp, "uuid", json_object_new_string (request->client->uuid));
117 // In this case or handle is quite basic
122 // This function is call when PostForm processing is completed
123 STATIC void DonePostForm (AFB_request *request) {
124 AFB_PostHandle *postHandle = (AFB_PostHandle*)request->post->data;;
126 int fd = (int)postHandle->handle;
129 if (verbose) fprintf ("DonePostForm filename=%s upload done\n", form->filename);
133 // WARNING: PostForm callback are call one type for form value
134 STATIC AFB_error ProcessPostForm (AFB_request *request, AFB_PostItem *item) {
136 AFB_PostHandle *postHandle;
139 // When Post is fully processed the same callback is call with a item==NULL
141 return(jsonNewMessage(AFB_SUCESS,"File [%s] uploaded at [%s] error=\n", item->filename, request->config->sessiondir));
144 // Let's make sure this is a valid PostForm request
145 if (!request->post && request->post->type != AFB_POST_FORM) {
146 return(jsonNewMessage(AFB_FAIL,"This is not a valid PostForm request\n"));
148 // In AFB_POST_FORM case post->data is a PostForm handle
149 postHandle = (AFB_PostHandle*) request->post->data;
152 // Check this is a file element
153 if (0 != strcmp (item->key, "file")) {
154 request->errcode = MHD_HTTP_FORBIDDEN;
155 request.jresp = jsonNewMessage(AFB_FAIL,"No File within element key=%s\n", item->key);
159 // This is the 1st Item iteration let's open output file and allocate necessary resources
160 if (postHandle->handle == NULL) {
161 strncpy (filepath, request->config->sessiondir, sizeof(filepath));
162 strncat (filepath, "/", sizeof(filepath));
163 strncat (filepath, item->filename, sizeof(filepath));
165 if((fd = open(request->config->sessiondir, O_RDONLY)) < 0) {
166 request->errcode = MHD_HTTP_FORBIDDEN;
167 request->jresp = jsonNewMessage(AFB_FAIL,"Fail to Upload file [%s] at [%s] error=\n", item->filename, request->config->sessiondir, strerror(errno));
171 // keep track of file handle with item
172 appCtx = malloc (size(appPostCtx)); // May place anything here until post->completeCB handle resources liberation
173 postHandle->handle = malloc (size(appPostCtx)); // May place anything here until post->completeCB handle resources liberation
175 postHandle->completeCB = DonePostForm; // CallBack when Form Processing is finished
178 // this is not the call, FD is already open
179 fd = (int)post->handle;
182 // We have something to write
185 if (!write (fd, item->data, item->len)) {
186 request->errcode = MHD_HTTP_FORBIDDEN;
187 request->json = jsonNewMessage(AFB_FAIL,"Fail to write file [%s] at [%s] error=\n", item->filename, strerror(errno));
192 // every event should return Sucess or Form processing stop
196 // This function is call when Client Session Context is removed
197 // Note: when freeCtxCB==NULL standard free/malloc is called
198 STATIC void clientContextFree(AFB_clientCtx *client) {
199 fprintf (stderr,"Plugin[%s] Closing Session uuid=[%s]\n", client->plugin->prefix, client->uuid);
203 STATIC AFB_restapi pluginApis[]= {
204 {"ping" , (AFB_apiCB)apiPingTest ,"Ping Rest Test Service"},
205 {"token-create" , (AFB_apiCB)clientContextCreate ,"Request Client Context Creation"},
206 {"token-refresh" , (AFB_apiCB)clientContextRefresh,"Refresh Client Context Token"},
207 {"token-check" , (AFB_apiCB)clientContextCheck ,"Check Client Context Token"},
208 {"token-reset" , (AFB_apiCB)clientContextReset ,"Close Client Context and Free resources"},
209 {"file-upload" , (AFB_apiCB)ProcessPostForm ,"Demo for file upload"},
213 PUBLIC AFB_plugin *afsvRegister () {
214 AFB_plugin *plugin = malloc (sizeof (AFB_plugin));
215 plugin->type = AFB_PLUGIN_JSON;
216 plugin->info = "Application Framework Binder Service";
217 plugin->prefix= "afbs"; // url base
218 plugin->apis = pluginApis;
219 plugin->handle= (void*) "What ever you want";
220 plugin->freeCtxCB= (void*) clientContextFree;