/* * Copyright 2016 IoT.bzh * Author: José Bollo * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define _GNU_SOURCE #include #include #include #include #include #include #include "afb-ws.h" #include "afb-ws-json1.h" #include "afb-msg-json.h" #include "session.h" #include "afb-req-itf.h" #include "afb-apis.h" #include "afb-context.h" static void aws_on_hangup(struct afb_ws_json1 *ws); static void aws_on_text(struct afb_ws_json1 *ws, char *text, size_t size); static struct afb_ws_itf aws_itf = { .on_hangup = (void*)aws_on_hangup, .on_text = (void*)aws_on_text }; struct afb_wsreq; struct afb_ws_json1 { void (*cleanup)(void*); void *cleanup_closure; struct afb_wsreq *requests; struct AFB_clientCtx *session; struct json_tokener *tokener; struct afb_ws *ws; }; static void aws_send_event(struct afb_ws_json1 *ws, const char *event, struct json_object *object); static const struct afb_event_listener_itf event_listener_itf = { .send = (void*)aws_send_event, .expects = NULL }; static inline struct afb_event_listener listener_for(struct afb_ws_json1 *aws) { return (struct afb_event_listener){ .itf = &event_listener_itf, .closure = aws }; } struct afb_ws_json1 *afb_ws_json1_create(int fd, struct AFB_clientCtx *session, void (*cleanup)(void*), void *cleanup_closure) { struct afb_ws_json1 *result; assert(fd >= 0); assert(session != NULL); result = malloc(sizeof * result); if (result == NULL) goto error; result->cleanup = cleanup; result->cleanup_closure = cleanup_closure; result->requests = NULL; result->session = ctxClientAddRef(session); if (result->session == NULL) goto error2; result->tokener = json_tokener_new(); if (result->tokener == NULL) goto error3; result->ws = afb_ws_create(fd, &aws_itf, result); if (result->ws == NULL) goto error4; if (0 > ctxClientEventListenerAdd(result->session, listener_for(result))) goto error5; return result; error5: afb_ws_destroy(result->ws); error4: json_tokener_free(result->tokener); error3: ctxClientUnref(result->session); error2: free(result); error: close(fd); return NULL; } static void aws_on_hangup(struct afb_ws_json1 *ws) { ctxClientEventListenerRemove(ws->session, listener_for(ws)); afb_ws_destroy(ws->ws); json_tokener_free(ws->tokener); if (ws->cleanup != NULL) ws->cleanup(ws->cleanup_closure); ctxClientUnref(ws->session); free(ws); } #define CALL 2 #define RETOK 3 #define RETERR 4 #define EVENT 5 struct afb_wsreq { struct afb_context context; int refcount; struct afb_ws_json1 *aws; struct afb_wsreq *next; char *text; size_t size; int code; char *id; size_t idlen; char *api; size_t apilen; char *verb; size_t verblen; char *obj; size_t objlen; char *tok; size_t toklen; struct json_object *root; }; static void wsreq_addref(struct afb_wsreq *wsreq); static void wsreq_unref(struct afb_wsreq *wsreq); static struct json_object *wsreq_json(struct afb_wsreq *wsreq); static struct afb_arg wsreq_get(struct afb_wsreq *wsreq, const char *name); static void wsreq_fail(struct afb_wsreq *wsreq, const char *status, const char *info); static void wsreq_success(struct afb_wsreq *wsreq, struct json_object *obj, const char *info); static const char *wsreq_raw(struct afb_wsreq *wsreq, size_t *size); static void wsreq_send(struct afb_wsreq *wsreq, const char *buffer, size_t size); static const struct afb_req_itf wsreq_itf = { .json = (void*)wsreq_json, .get = (void*)wsreq_get, .success = (void*)wsreq_success, .fail = (void*)wsreq_fail, .raw = (void*)wsreq_raw, .send = (void*)wsreq_send, .context_get = (void*)afb_context_get, .context_set = (void*)afb_context_set, .addref = (void*)wsreq_addref, .unref = (void*)wsreq_unref }; static int aws_wsreq_parse(struct afb_wsreq *r, char *text, size_t size) { char *pos, *end, c; int aux; /* scan */ pos = text; end = text + size; /* scans: [ */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != '[') goto bad_header; /* scans code: 2|3|4 */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; switch (*pos++) { case '2': r->code = CALL; break; case '3': r->code = RETOK; break; case '4': r->code = RETERR; break; default: goto bad_header; } /* scans: , */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != ',') goto bad_header; /* scans id: "id" */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != '"') goto bad_header; r->id = pos; while(pos < end && *pos != '"') pos++; if (pos == end) goto bad_header; r->idlen = (size_t)(pos++ - r->id); /* scans: , */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != ',') goto bad_header; /* scans the method if needed */ if (r->code == CALL) { /* scans: " */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != '"') goto bad_header; /* scans: api/ */ r->api = pos; while(pos < end && *pos != '"' && *pos != '/') pos++; if (pos == end) goto bad_header; if (*pos != '/') goto bad_header; r->apilen = (size_t)(pos++ - r->api); if (r->apilen && r->api[r->apilen - 1] == '\\') r->apilen--; /* scans: verb" */ r->verb = pos; while(pos < end && *pos != '"') pos++; if (pos == end) goto bad_header; r->verblen = (size_t)(pos++ - r->verb); /* scans: , */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != ',') goto bad_header; } /* scan obj */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; aux = 0; r->obj = pos; while (pos < end && (aux != 0 || (*pos != ',' && *pos != ']'))) { if (pos == end) goto bad_header; switch(*pos) { case '{': case '[': aux++; break; case '}': case ']': if (!aux--) goto bad_header; break; case '"': do { pos += 1 + (*pos == '\\'); } while(pos < end && *pos != '"'); default: break; } pos++; } if (pos > end) goto bad_header; if (pos == end && aux != 0) goto bad_header; c = *pos; r->objlen = (size_t)(pos++ - r->obj); while (r->objlen && r->obj[r->objlen - 1] == ' ') r->objlen--; /* scan the token (if any) */ if (c == ',') { /* scans token: "token" */ while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; if (*pos++ != '"') goto bad_header; r->tok = pos; while(pos < end && *pos != '"') pos++; if (pos == end) goto bad_header; r->toklen = (size_t)(pos++ - r->tok); while(pos < end && *pos == ' ') pos++; if (pos == end) goto bad_header; c = *pos++; } /* scan: ] */ if (c != ']') goto bad_header; while(pos < end && *pos == ' ') pos++; if (pos != end) goto bad_header; /* done */ r->text = text; r->size = size; return 1; bad_header: return 0; } static void aws_on_text(struct afb_ws_json1 *ws, char *text, size_t size) { struct afb_req r; struct afb_wsreq *wsreq; /* allocate */ wsreq = calloc(1, sizeof *wsreq); if (wsreq == NULL) goto alloc_error; /* init */ if (!aws_wsreq_parse(wsreq, text, size)) goto bad_header; /* fill and record the request */ if (wsreq->tok != NULL) wsreq->tok[wsreq->toklen] = 0; afb_context_init(&wsreq->context, ws->session, wsreq->tok); if (!wsreq->context.invalidated) wsreq->context.validated = 1; wsreq->refcount = 1; wsreq->aws = ws; wsreq->next = ws->requests; ws->requests = wsreq; r.closure = wsreq; r.itf = &wsreq_itf; afb_apis_call(r, &wsreq->context, wsreq->api, wsreq->apilen, wsreq->verb, wsreq->verblen); wsreq_unref(wsreq); return; bad_header: free(wsreq); alloc_error: free(text); afb_ws_close(ws->ws, 1008, NULL); return; } static void wsreq_addref(struct afb_wsreq *wsreq) { wsreq->refcount++; } static void wsreq_unref(struct afb_wsreq *wsreq) { if (--wsreq->refcount == 0) { afb_context_disconnect(&wsreq->context); free(wsreq->text); free(wsreq); } } static struct json_object *wsreq_json(struct afb_wsreq *wsreq) { struct json_object *root = wsreq->root; if (root == NULL) { json_tokener_reset(wsreq->aws->tokener); root = json_tokener_parse_ex(wsreq->aws->tokener, wsreq->obj, (int)wsreq->objlen); if (root == NULL) { /* lazy error detection of json request. Is it to improve? */ root = json_object_new_string_len(wsreq->obj, (int)wsreq->objlen); } wsreq->root = root; } return root; } static struct afb_arg wsreq_get(struct afb_wsreq *wsreq, const char *name) { struct afb_arg arg; struct json_object *value, *root; root = wsreq_json(wsreq); if (json_object_object_get_ex(root, name, &value)) { arg.name = name; arg.value = json_object_get_string(value); } else { arg.name = NULL; arg.value = NULL; } arg.path = NULL; return arg; } static void aws_emit(struct afb_ws_json1 *aws, int code, const char *id, size_t idlen, struct json_object *data, const char *token) { json_object *msg; const char *txt; /* pack the message */ msg = json_object_new_array(); json_object_array_add(msg, json_object_new_int(code)); json_object_array_add(msg, json_object_new_string_len(id, (int)idlen)); json_object_array_add(msg, data); if (token) json_object_array_add(msg, json_object_new_string(token)); /* emits the reply */ txt = json_object_to_json_string(msg); afb_ws_text(aws->ws, txt, strlen(txt)); json_object_put(msg); } static void wsreq_reply(struct afb_wsreq *wsreq, int retcode, const char *status, const char *info, json_object *resp) { const char *token = afb_context_sent_token(&wsreq->context); aws_emit(wsreq->aws, retcode, wsreq->id, wsreq->idlen, afb_msg_json_reply(status, info, resp, token, NULL), token); } static void wsreq_fail(struct afb_wsreq *wsreq, const char *status, const char *info) { wsreq_reply(wsreq, RETERR, status, info, NULL); } static void wsreq_success(struct afb_wsreq *wsreq, json_object *obj, const char *info) { wsreq_reply(wsreq, RETOK, "success", info, obj); } static const char *wsreq_raw(struct afb_wsreq *wsreq, size_t *size) { *size = wsreq->objlen; return wsreq->obj; } static void wsreq_send(struct afb_wsreq *wsreq, const char *buffer, size_t size) { afb_ws_text(wsreq->aws->ws, buffer, size); } static void aws_send_event(struct afb_ws_json1 *aws, const char *event, struct json_object *object) { aws_emit(aws, EVENT, event, strlen(event), afb_msg_json_event(event, object), NULL); }