1 /*
2  * Copyright (c) 2023 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "vpn_monitor_ext.h"
17 
18 #include <cstddef>
19 #include <string>
20 
21 #include <napi/native_common.h>
22 #include <uv.h>
23 
24 #include "module_template.h"
25 #include "napi_utils.h"
26 #include "netmanager_ext_log.h"
27 #include "networkvpn_client.h"
28 #include "want.h"
29 #include "ability_manager_client.h"
30 #include "extension_ability_info.h"
31 
32 namespace OHOS {
33 namespace NetManagerStandard {
34 namespace {
35 constexpr const char *CONNECT = "connect";
36 constexpr int32_t PARAM_JUST_OPTIONS = 1;
37 constexpr int32_t PARAM_OPTIONS_AND_CALLBACK = 2;
38 
EventConnectCallback(uv_work_t * work,int status)39 void EventConnectCallback(uv_work_t *work, int status)
40 {
41     if (work == nullptr) {
42         NETMANAGER_EXT_LOGE("work is nullptr");
43         return;
44     }
45     auto workWrapper = reinterpret_cast<UvWorkWrapper *>(work->data);
46     if (workWrapper == nullptr) {
47         NETMANAGER_EXT_LOGE("workWrapper is nullptr");
48         delete work;
49         return;
50     }
51     bool *data = reinterpret_cast<bool *>(workWrapper->data);
52     if (data == nullptr) {
53         NETMANAGER_EXT_LOGE("isConnected is nullptr");
54         delete workWrapper;
55         delete work;
56         return;
57     }
58 
59     napi_env env = workWrapper->env;
60     napi_handle_scope scope = NapiUtils::OpenScope(env);
61     napi_value isConnected = NapiUtils::GetBoolean(env, *data);
62     napi_value result = NapiUtils::CreateObject(env);
63     NapiUtils::SetNamedProperty(env, result, "isConnected", isConnected);
64     workWrapper->manager->Emit(CONNECT, std::make_pair(NapiUtils::GetUndefined(env), result));
65     NapiUtils::CloseScope(env, scope);
66     delete data;
67     delete workWrapper;
68     delete work;
69 }
70 
CheckParamType(napi_env env,napi_value * params,size_t paramsCount)71 bool CheckParamType(napi_env env, napi_value *params, size_t paramsCount)
72 {
73     switch (paramsCount) {
74         case PARAM_JUST_OPTIONS:
75             return (NapiUtils::GetValueType(env, params[0]) == napi_string);
76         case PARAM_OPTIONS_AND_CALLBACK:
77             return ((NapiUtils::GetValueType(env, params[0]) == napi_string) &&
78                     (NapiUtils::GetValueType(env, params[1]) == napi_function));
79         default:
80             return false;
81     }
82 }
83 } // namespace
84 
OnVpnStateChanged(const bool & isConnected)85 void VpnEventCallback::OnVpnStateChanged(const bool &isConnected)
86 {
87     auto manager = VpnMonitor::GetInstance().GetManager();
88     bool *data = new bool(isConnected);
89     manager->EmitByUv(CONNECT, reinterpret_cast<void *>(data), EventConnectCallback);
90 }
91 
GetInstance()92 VpnMonitor &VpnMonitor::GetInstance()
93 {
94     static VpnMonitor instance;
95     return instance;
96 }
97 
On(napi_env env,napi_callback_info info)98 napi_value VpnMonitor::On(napi_env env, napi_callback_info info)
99 {
100     if (!ParseParams(env, info)) {
101         NETMANAGER_EXT_LOGE("parse failed");
102         NAPI_CALL(env, napi_throw_error(env, "0", "parse failed"));
103         return NapiUtils::GetUndefined(env);
104     }
105     Register(env);
106     return NapiUtils::GetUndefined(env);
107 }
108 
Off(napi_env env,napi_callback_info info)109 napi_value VpnMonitor::Off(napi_env env, napi_callback_info info)
110 {
111     if (!ParseParams(env, info)) {
112         NETMANAGER_EXT_LOGE("parse failed");
113         NAPI_CALL(env, napi_throw_error(env, "0", "parse failed"));
114         return NapiUtils::GetUndefined(env);
115     }
116     Unregister(env);
117     return NapiUtils::GetUndefined(env);
118 }
119 
ParseParams(napi_env env,napi_callback_info info)120 bool VpnMonitor::ParseParams(napi_env env, napi_callback_info info)
121 {
122     napi_value jsObject = nullptr;
123     size_t paramsCount = MAX_PARAM_NUM;
124     napi_value params[MAX_PARAM_NUM] = {nullptr};
125     NAPI_CALL_BASE(env, napi_get_cb_info(env, info, &paramsCount, params, &jsObject, nullptr), false);
126 
127     if (!CheckParamType(env, params, paramsCount)) {
128         return false;
129     }
130     if (!UnwrapManager(env, jsObject)) {
131         return false;
132     }
133     const std::string event = NapiUtils::GetStringFromValueUtf8(env, params[0]);
134     if (CONNECT != event) {
135         NETMANAGER_EXT_LOGE("%{public}s event is error", event.c_str());
136         return false;
137     }
138     if (paramsCount == PARAM_OPTIONS_AND_CALLBACK) {
139         callback_ = params[1];
140     }
141     return true;
142 }
143 
UnwrapManager(napi_env env,napi_value jsObject)144 bool VpnMonitor::UnwrapManager(napi_env env, napi_value jsObject)
145 {
146     NAPI_CALL_BASE(env, napi_unwrap(env, jsObject, reinterpret_cast<void **>(&manager_)), false);
147     if (manager_ == nullptr) {
148         return false;
149     }
150     return true;
151 }
152 
Register(napi_env env)153 void VpnMonitor::Register(napi_env env)
154 {
155     auto vpnClient = reinterpret_cast<NetworkVpnClient *>(manager_->GetData());
156     if (vpnClient == nullptr) {
157         NETMANAGER_EXT_LOGE("vpnClient is nullptr");
158         return;
159     }
160     manager_->AddListener(env, CONNECT, callback_, false, false);
161 
162     if (eventCallback_ != nullptr) {
163         vpnClient->UnregisterVpnEvent(eventCallback_);
164     }
165     eventCallback_ = new (std::nothrow) VpnEventCallback();
166     if (nullptr == eventCallback_) {
167         NETMANAGER_EXT_LOGE("eventCallback_ is nullptr");
168         return;
169     }
170     vpnClient->RegisterVpnEvent(eventCallback_);
171 }
172 
Unregister(napi_env env)173 void VpnMonitor::Unregister(napi_env env)
174 {
175     auto vpnClient = reinterpret_cast<NetworkVpnClient *>(manager_->GetData());
176     if (vpnClient == nullptr) {
177         NETMANAGER_EXT_LOGE("vpnClient is nullptr");
178         return;
179     }
180     manager_->DeleteListener(CONNECT);
181     vpnClient->UnregisterVpnEvent(eventCallback_);
182 }
183 
ShowVpnDialog(const std::string & bundleName,const std::string & abilityName,const std::string & appName)184 bool VpnMonitor::ShowVpnDialog(const std::string &bundleName, const std::string &abilityName,
185                                const std::string &appName)
186 {
187     auto abmc = AAFwk::AbilityManagerClient::GetInstance();
188     if (abmc == nullptr) {
189         NETMANAGER_EXT_LOGE("GetInstance failed");
190         return false;
191     }
192 
193     AAFwk::Want want;
194     want.SetElementName("com.ohos.vpndialog", "VpnServiceExtAbility");
195     want.SetParam("bundleName", bundleName);
196     want.SetParam("abilityName", abilityName + VPN_DIALOG_POSTFIX);
197     want.SetParam("appName", appName);
198 
199     sptr<VpnMonitor::VpnAbilityConn> vpnAbilityConn_ = new (std::nothrow) VpnMonitor::VpnAbilityConn();
200     auto ret = abmc->ConnectAbility(want, vpnAbilityConn_, -1);
201     if (ret != 0) {
202         NETMANAGER_EXT_LOGE("connectAbility failed %{public}d", ret);
203         return false;
204     }
205     /* Waiting for the user to click */
206     NETMANAGER_EXT_LOGI("click done");
207     return true;
208 }
209 } // namespace NetManagerStandard
210 } // namespace OHOS