/* * Copyright (c) 2019 TOYOTA MOTOR CORPORATION * * 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. */ #include "homescreen.h" #include "hs-proxy.h" struct closure_data { std::string appid; HS_ClientCtxt *clientCtx; struct hs_instance *hs_instance; }; const char _afm_main[] = "afm-main"; /** * the callback function * * #### Parameters * - closure : the user defined closure pointer 'closure' * - object : a JSON object returned (can be NULL) * - error : a string not NULL in case of error but NULL on success * - info : a string handling some info (can be NULL) * - api : the api * * #### Return * None * */ static void api_callback(void *closure, struct json_object *object, const char *error, const char *info, afb_api_t api) { AFB_INFO("asynchronous call, error=%s, info=%s, object=%s.", error, info, json_object_get_string(object)); struct closure_data *cdata = static_cast(closure); if (!cdata->hs_instance) { return; } struct HS_ClientManager *clientManager = cdata->hs_instance->client_manager; if (!clientManager) { return; } /* if we have an error then we couldn't start the application so we remove it */ if (error) { clientManager->removeClient(cdata->appid); } free(cdata); } /** * call api asynchronous * * #### Parameters * - api : the api serving the request * - service : the api name of service * - verb : the verb of service * - args : parameter * * #### Return * None * */ static void api_call(afb_api_t api, const char *service, const char *verb, struct json_object *args, struct closure_data *cdata) { AFB_INFO("service=%s verb=%s, args=%s.", service, verb, json_object_get_string(args)); afb_api_call(api, service, verb, args, api_callback, cdata); } /** * call api synchronous * * #### Parameters * - api : the api serving the request * - service : the api name of service * - verb : the verb of afm-main * - args : parameter * - object : return the details of application * * #### Return * 0 : success * 1 : fail * */ static int api_call_sync(afb_api_t api, const char *service, const char *verb, struct json_object *args, struct json_object **object) { char *error = nullptr, *info = nullptr; int ret = afb_api_call_sync(api, service, verb, args, object, &error, &info); AFB_INFO("synchronous call, error=%s, info=%s.", error, info); return ret; } /** * get runnables application list * * #### Parameters * - api : the api serving the request * - object : return the details of appid * * #### Return * 0 : success * 1 : fail * */ int HS_AfmMainProxy::runnables(afb_api_t api, struct json_object **object) { return api_call_sync(api, _afm_main, __FUNCTION__, nullptr, object); } /** * get details of application * * #### Parameters * - api : the api serving the request * - id : the id to get details,liked "dashboard@0.1" * - object : return the details of application * * #### Return * 0 : success * 1 : fail * */ int HS_AfmMainProxy::detail(afb_api_t api, const std::string &id, struct json_object **object) { struct json_object *args = json_object_new_string(id.c_str()); return api_call_sync(api, _afm_main, __FUNCTION__, args, object); } /** * start application * * #### Parameters * - request : the request * - id : the application id liked "dashboard@0.1" * * #### Return * None * */ void HS_AfmMainProxy::start(struct hs_instance *instance, afb_req_t request, const std::string &id) { struct closure_data *cdata; /* tentatively store the client and client context, as the afb_req_t * request will no longer be available in the async callback handler. This * is similar to that is done showWindow(), handleRequest() in * homescreen.cpp, but allows to fake the subscription here as well to * avoid clients create/install dummy event handlers as to 'register' (or * to keep track of applications started). * * In case api_callback() does return an error we'll remove then the client * and client context there. We pass the closure_data with the client context * and the application id to remove it. */ if (!instance) return; cdata = static_cast(calloc(1, sizeof(*cdata))); cdata->hs_instance = instance; cdata->appid = id; struct HS_ClientManager *clientManager = instance->client_manager; if (!clientManager) { return; } clientManager->addClient(request, id); api_call(request->api, _afm_main, __FUNCTION__, json_object_new_string(id.c_str()), cdata); }