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, ¶msCount, 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