/* * Copyright (c) 2022-2023 Huawei Device Co., Ltd. * 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. */ #ifndef COMMUNICATIONNETMANAGER_BASE_NETMANAGER_BASE_MODULE_TEMPLATE_H #define COMMUNICATIONNETMANAGER_BASE_NETMANAGER_BASE_MODULE_TEMPLATE_H #include <initializer_list> #include <napi/native_api.h> #include <napi/native_common.h> #include "base_context.h" #include "napi_utils.h" #include "netmanager_base_log.h" #define MAX_PARAM_NUM 64 namespace OHOS { namespace NetManagerStandard { namespace ModuleTemplate { using Finalizer = void (*)(napi_env env, void *data, void *); template <class Context> napi_value InterfaceWithoutManager(napi_env env, napi_callback_info info, const std::string &asyncWorkName, bool (*Work)(napi_env, napi_value, Context *), AsyncWorkExecutor executor, AsyncWorkCallback callback) { static_assert(std::is_base_of<BaseContext, Context>::value); napi_value thisVal = nullptr; size_t paramsCount = MAX_PARAM_NUM; napi_value params[MAX_PARAM_NUM] = {nullptr}; NAPI_CALL(env, napi_get_cb_info(env, info, ¶msCount, params, &thisVal, nullptr)); auto context = new Context(env, nullptr); context->ParseParams(params, paramsCount); if (context->IsNeedThrowException()) { // only api9 or later need throw exception. napi_throw_error(env, std::to_string(context->GetErrorCode()).c_str(), context->GetErrorMessage().c_str()); delete context; context = nullptr; return NapiUtils::GetUndefined(env); } if (Work != nullptr) { if (!Work(env, thisVal, context)) { NETMANAGER_BASE_LOGE("work failed error code = %{public}d", context->GetErrorCode()); } } context->CreateAsyncWork(asyncWorkName, executor, callback); if (NapiUtils::GetValueType(env, context->GetCallback()) != napi_function && context->IsNeedPromise()) { return context->CreatePromise(); } return NapiUtils::GetUndefined(env); } template <class Context> napi_value Interface(napi_env env, napi_callback_info info, const std::string &asyncWorkName, bool (*Work)(napi_env, napi_value, Context *), AsyncWorkExecutor executor, AsyncWorkCallback callback) { static_assert(std::is_base_of<BaseContext, Context>::value); napi_value thisVal = nullptr; size_t paramsCount = MAX_PARAM_NUM; napi_value params[MAX_PARAM_NUM] = {nullptr}; NAPI_CALL(env, napi_get_cb_info(env, info, ¶msCount, params, &thisVal, nullptr)); EventManager *manager = nullptr; napi_unwrap(env, thisVal, reinterpret_cast<void **>(&manager)); auto context = new Context(env, manager); context->ParseParams(params, paramsCount); if (context->IsNeedThrowException()) { // only api9 or later need throw exception. napi_throw_error(env, std::to_string(context->GetErrorCode()).c_str(), context->GetErrorMessage().c_str()); delete context; context = nullptr; return NapiUtils::GetUndefined(env); } if (Work != nullptr) { if (!Work(env, thisVal, context)) { NETMANAGER_BASE_LOGE("work failed error code = %{public}d", context->GetErrorCode()); } } context->CreateAsyncWork(asyncWorkName, executor, callback); if (NapiUtils::GetValueType(env, context->GetCallback()) != napi_function && context->IsNeedPromise()) { return context->CreatePromise(); } return NapiUtils::GetUndefined(env); } template <class Context> napi_value InterfaceSync(napi_env env, napi_callback_info info, const std::string &asyncWorkName, bool (*Work)(napi_env, napi_value, Context *), bool (*executor)(Context *), napi_value (*callback)(Context *)) { static_assert(std::is_base_of<BaseContext, Context>::value); napi_value thisVal = nullptr; size_t paramsCount = MAX_PARAM_NUM; napi_value params[MAX_PARAM_NUM] = {nullptr}; NAPI_CALL(env, napi_get_cb_info(env, info, ¶msCount, params, &thisVal, nullptr)); EventManager *manager = nullptr; napi_unwrap(env, thisVal, reinterpret_cast<void **>(&manager)); auto deleter = [](Context *context) { delete context; }; auto text = new Context(env, manager); std::unique_ptr<Context, decltype(deleter)> context(text, deleter); if (!context) { return NapiUtils::GetUndefined(env); } context->ParseParams(params, paramsCount); if (!context->IsParseOK()) { napi_throw_error(env, std::to_string(context->GetErrorCode()).c_str(), context->GetErrorMessage().c_str()); return NapiUtils::GetUndefined(env); } if (Work != nullptr) { if (!Work(env, thisVal, context.get())) { NETMANAGER_BASE_LOGE("work failed error code = %{public}d", context->GetErrorCode()); } } if (!executor || !callback) { NETMANAGER_BASE_LOGE("executor or callback is null"); return NapiUtils::GetUndefined(context->GetEnv()); } if (!executor(context.get())) { NETMANAGER_BASE_LOGE("executor is fail, errorcode= %{public}d", context->GetErrorCode()); napi_throw_error(env, std::to_string(context->GetErrorCode()).c_str(), context->GetErrorMessage().c_str()); return NapiUtils::GetUndefined(env); } return callback(context.get()); } template <class Context> napi_value InterfaceWithOutAsyncWork(napi_env env, napi_callback_info info, bool (*Work)(napi_env, napi_value, Context *)) { static_assert(std::is_base_of<BaseContext, Context>::value); napi_value thisVal = nullptr; size_t paramsCount = MAX_PARAM_NUM; napi_value params[MAX_PARAM_NUM] = {nullptr}; NAPI_CALL(env, napi_get_cb_info(env, info, ¶msCount, params, &thisVal, nullptr)); EventManager *manager = nullptr; napi_unwrap(env, thisVal, reinterpret_cast<void **>(&manager)); auto context = new Context(env, manager); context->ParseParams(params, paramsCount); if (Work != nullptr) { if (!Work(env, thisVal, context)) { NETMANAGER_BASE_LOGE("work failed error code = %{public}d", context->GetErrorCode()); } } if (NapiUtils::GetValueType(env, context->GetCallback()) != napi_function && context->IsNeedPromise()) { return context->CreatePromise(); } return NapiUtils::GetUndefined(env); } napi_value On(napi_env env, napi_callback_info info, const std::initializer_list<std::string> &events, bool asyncCallback); napi_value Once(napi_env env, napi_callback_info info, const std::initializer_list<std::string> &events, bool asyncCallback); napi_value Off(napi_env env, napi_callback_info info, const std::initializer_list<std::string> &events); void DefineClass(napi_env env, napi_value exports, const std::initializer_list<napi_property_descriptor> &properties, const std::string &className); napi_value NewInstance(napi_env env, napi_callback_info info, const std::string &className, void *(*MakeData)(napi_env, size_t, napi_value *, EventManager *), Finalizer finalizer); } // namespace ModuleTemplate } // namespace NetManagerStandard } // namespace OHOS #endif // COMMUNICATIONNETMANAGER_BASE_NETMANAGER_BASE_MODULE_TEMPLATE_H