1 /*
2 * Copyright (c) 2022-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 "devicestatus_event.h"
17
18 #include <js_native_api.h>
19 #include <map>
20 #include <uv.h>
21
22 #include "napi/native_api.h"
23 #include "napi/native_node_api.h"
24
25 #include "devicestatus_define.h"
26
27 namespace OHOS {
28 namespace Msdp {
29 namespace DeviceStatus {
30 namespace {
31 constexpr ::OHOS::HiviewDFX::HiLogLabel LABEL { LOG_CORE, MSDP_DOMAIN_ID, "DeviceStatusEvent" };
32 constexpr size_t EVENT_MAP_MAX { 20 };
33 constexpr size_t EVENT_LIST_MAX { 30 };
34 } // namespace
35
DeviceStatusEvent(napi_env env)36 DeviceStatusEvent::DeviceStatusEvent(napi_env env)
37 {
38 env_ = env;
39 }
40
~DeviceStatusEvent()41 DeviceStatusEvent::~DeviceStatusEvent()
42 {
43 eventOnces_.clear();
44 events_.clear();
45 }
46
On(int32_t eventType,napi_value handler,bool isOnce)47 bool DeviceStatusEvent::On(int32_t eventType, napi_value handler, bool isOnce)
48 {
49 FI_HILOGD("On for event:%{public}d, isOnce:%{public}d", eventType, isOnce);
50 std::lock_guard<std::mutex> guard(mutex_);
51 if ((events_.size() > EVENT_MAP_MAX) || (eventOnces_.size() > EVENT_MAP_MAX)) {
52 FI_HILOGE("events_ or eventOnces_ size over");
53 return false;
54 }
55 if (events_[eventType].size() > EVENT_LIST_MAX || eventOnces_[eventType].size() > EVENT_LIST_MAX) {
56 FI_HILOGE("list size over");
57 return false;
58 }
59 if (isOnce) {
60 if (!SaveCallbackByEvent(eventType, handler, isOnce, eventOnces_)) {
61 FI_HILOGE("Failed to save eventOnces_ callback");
62 return false;
63 }
64 } else {
65 if (!SaveCallbackByEvent(eventType, handler, isOnce, events_)) {
66 FI_HILOGE("Failed to save events_ callback");
67 return false;
68 }
69 }
70 return true;
71 }
72
SaveCallbackByEvent(int32_t eventType,napi_value handler,bool isOnce,std::map<int32_t,std::list<std::shared_ptr<DeviceStatusEventListener>>> events_)73 bool DeviceStatusEvent::SaveCallbackByEvent(int32_t eventType, napi_value handler, bool isOnce,
74 std::map<int32_t, std::list<std::shared_ptr<DeviceStatusEventListener>>> events_)
75 {
76 CALL_DEBUG_ENTER;
77 napi_ref onHandlerRef = nullptr;
78 napi_status status = napi_create_reference(env_, handler, 1, &onHandlerRef);
79 if (status != napi_ok) {
80 FI_HILOGE("Failed to napi_create_reference");
81 return false;
82 }
83 auto iter = events_.find(eventType);
84 if (iter == events_.end()) {
85 FI_HILOGE("eventType:%{public}d not exists", eventType);
86 events_[eventType] = std::list<std::shared_ptr<DeviceStatusEventListener>>();
87 }
88 if (events_[eventType].empty()) {
89 FI_HILOGE("events_ save callback");
90 SaveCallback(eventType, onHandlerRef, isOnce);
91 return true;
92 }
93 if (!IsNoExistCallback(events_[eventType], handler, eventType)) {
94 FI_HILOGE("Callback already exists");
95 return false;
96 }
97 SaveCallback(eventType, onHandlerRef, isOnce);
98 return true;
99 }
100
IsNoExistCallback(std::list<std::shared_ptr<DeviceStatusEventListener>>,napi_value handler,int32_t eventType)101 bool DeviceStatusEvent::IsNoExistCallback(std::list<std::shared_ptr<DeviceStatusEventListener>>,
102 napi_value handler, int32_t eventType)
103 {
104 CALL_DEBUG_ENTER;
105 napi_value result = nullptr;
106 bool equal = false;
107 for (const auto &item : events_[eventType]) {
108 napi_status napiStatus = napi_get_reference_value(env_, item->onHandlerRef, &result);
109 if (napiStatus != napi_ok) {
110 FI_HILOGE("Failed to napi_get_reference_value");
111 return false;
112 }
113 napiStatus = napi_strict_equals(env_, result, handler, &equal);
114 if (napiStatus != napi_ok) {
115 FI_HILOGE("Failed to napi_strict_equals");
116 return false;
117 }
118 if (equal) {
119 FI_HILOGE("Map callback is exist");
120 return false;
121 }
122 }
123 return true;
124 }
125
SaveCallback(int32_t eventType,napi_ref onHandlerRef,bool isOnce)126 void DeviceStatusEvent::SaveCallback(int32_t eventType, napi_ref onHandlerRef, bool isOnce)
127 {
128 auto listener = std::make_shared<DeviceStatusEventListener>();
129 listener->onHandlerRef = onHandlerRef;
130 if (isOnce) {
131 eventOnces_[eventType].push_back(listener);
132 } else {
133 events_[eventType].push_back(listener);
134 }
135 FI_HILOGD("Add handler to list %{public}d", eventType);
136 }
137
Off(int32_t eventType,napi_value handler)138 bool DeviceStatusEvent::Off(int32_t eventType, napi_value handler)
139 {
140 FI_HILOGD("DeviceStatusEvent off in for event:%{public}d", eventType);
141 std::lock_guard<std::mutex> guard(mutex_);
142 auto iter = events_.find(eventType);
143 if (iter == events_.end()) {
144 FI_HILOGE("eventType %{public}d not found", eventType);
145 return false;
146 }
147 bool equal = false;
148 napi_value result = nullptr;
149
150 for (const auto &listener : events_[eventType]) {
151 napi_status status = napi_get_reference_value(env_, listener->onHandlerRef, &result);
152 if (status != napi_ok) {
153 FI_HILOGE("Failed to napi_get_reference_value");
154 return false;
155 }
156 status = napi_strict_equals(env_, result, handler, &equal);
157 if (status != napi_ok) {
158 FI_HILOGE("Failed to napi_strict_equals");
159 return false;
160 }
161 if (equal) {
162 FI_HILOGI("Delete handler from list %{public}d", eventType);
163 status = napi_delete_reference(env_, listener->onHandlerRef);
164 if (status != napi_ok) {
165 FI_HILOGW("Failed to napi_delete_reference");
166 }
167 events_[eventType].remove(listener);
168 break;
169 }
170 }
171 FI_HILOGI("%{public}zu listeners in the list of %{public}d",
172 events_[eventType].size(), eventType);
173 return events_[eventType].empty();
174 }
175
OffOnce(int32_t eventType,napi_value handler)176 bool DeviceStatusEvent::OffOnce(int32_t eventType, napi_value handler)
177 {
178 FI_HILOGD("DeviceStatusEvent OffOnce in for event:%{public}d", eventType);
179 auto iter = eventOnces_.find(eventType);
180 if (iter == eventOnces_.end()) {
181 FI_HILOGE("eventType %{public}d not found", eventType);
182 return false;
183 }
184 bool equal = false;
185 napi_value result = nullptr;
186 for (const auto &listener : eventOnces_[eventType]) {
187 napi_get_reference_value(env_, listener->onHandlerRef, &result);
188 napi_strict_equals(env_, result, handler, &equal);
189 if (equal) {
190 FI_HILOGI("Delete once handler from list %{public}d", eventType);
191 napi_status status = napi_delete_reference(env_, listener->onHandlerRef);
192 if (status != napi_ok) {
193 FI_HILOGW("Failed to napi_delete_reference");
194 }
195 eventOnces_[eventType].remove(listener);
196 break;
197 }
198 }
199 FI_HILOGI("%{public}zu listeners in the once list of %{public}d",
200 eventOnces_[eventType].size(), eventType);
201 return events_[eventType].empty();
202 }
203
RemoveAllCallback(int32_t eventType)204 bool DeviceStatusEvent::RemoveAllCallback(int32_t eventType)
205 {
206 CALL_DEBUG_ENTER;
207 auto iter = events_.find(eventType);
208 if (iter == events_.end()) {
209 FI_HILOGE("evenType %{public}d not found", eventType);
210 return false;
211 }
212 events_.erase(eventType);
213 return true;
214 }
215
CheckRet(int32_t eventType,size_t argc,int32_t value,std::shared_ptr<DeviceStatusEventListener> & typeHandler)216 void DeviceStatusEvent::CheckRet(int32_t eventType, size_t argc, int32_t value,
217 std::shared_ptr<DeviceStatusEventListener> &typeHandler)
218 {
219 CHKPV(typeHandler);
220 napi_value handler = nullptr;
221 napi_status status = napi_ok;
222 status = napi_get_reference_value(env_, typeHandler->onHandlerRef, &handler);
223 if (status != napi_ok) {
224 FI_HILOGE("OnEvent handler for %{public}d failed, status:%{public}d", eventType, status);
225 return;
226 }
227 napi_value result = nullptr;
228 SendRet(eventType, value, result);
229 napi_value callResult = nullptr;
230 FI_HILOGD("Report to hap");
231 status = napi_call_function(env_, nullptr, handler, argc, &result, &callResult);
232 if (status != napi_ok) {
233 FI_HILOGE("CheckRet:napi_call_function for %{public}d failed, status:%{public}d", eventType, status);
234 return;
235 }
236 }
237
SendRet(int32_t eventType,int32_t value,napi_value & result)238 void DeviceStatusEvent::SendRet(int32_t eventType, int32_t value, napi_value &result)
239 {
240 napi_status status = napi_create_object(env_, &result);
241 if (status != napi_ok) {
242 FI_HILOGE("Failed to create object");
243 return;
244 }
245 napi_value tmpValue = nullptr;
246 status = napi_create_int32(env_, eventType, &tmpValue);
247 if (status != napi_ok) {
248 FI_HILOGE("Failed to create object");
249 return;
250 }
251 status = napi_set_named_property(env_, result, "type", tmpValue);
252 if (status != napi_ok) {
253 FI_HILOGE("Failed to set name");
254 return;
255 }
256
257 status = napi_create_int32(env_, value, &tmpValue);
258 if (status != napi_ok) {
259 FI_HILOGE("Failed to create value");
260 return;
261 }
262 status = napi_set_named_property(env_, result, "value", tmpValue);
263 if (status != napi_ok) {
264 FI_HILOGE("Failed to set_named");
265 return;
266 }
267 }
268
OnEvent(int32_t eventType,size_t argc,int32_t value,bool isOnce)269 void DeviceStatusEvent::OnEvent(int32_t eventType, size_t argc, int32_t value, bool isOnce)
270 {
271 CALL_DEBUG_ENTER;
272 FI_HILOGD("OnEvent for %{public}d, isOnce:%{public}d", eventType, isOnce);
273 std::map<int32_t, std::list<std::shared_ptr<DeviceStatusEventListener>>>::iterator typeHandler;
274 if (isOnce) {
275 typeHandler = eventOnces_.find(eventType);
276 if (typeHandler == eventOnces_.end()) {
277 FI_HILOGE("OnEvent eventType %{public}d not found", eventType);
278 return;
279 }
280 } else {
281 typeHandler = events_.find(eventType);
282 if (typeHandler == events_.end()) {
283 FI_HILOGE("OnEvent eventType %{public}d not found", eventType);
284 return;
285 }
286 }
287 FI_HILOGD("%{public}zu callbacks of eventType %{public}d are sent",
288 typeHandler->second.size(), eventType);
289 for (auto handler : typeHandler->second) {
290 CheckRet(eventType, argc, value, handler);
291 }
292 }
293
ClearEventMap()294 void DeviceStatusEvent::ClearEventMap()
295 {
296 for (const auto &iter : events_) {
297 for (const auto &eventListener : iter.second) {
298 napi_status status = napi_delete_reference(env_, eventListener->onHandlerRef);
299 if (status != napi_ok) {
300 FI_HILOGW("Failed to napi_delete_reference");
301 }
302 }
303 }
304 for (const auto &iter : eventOnces_) {
305 for (const auto &eventListener : iter.second) {
306 napi_status status = napi_delete_reference(env_, eventListener->onHandlerRef);
307 if (status != napi_ok) {
308 FI_HILOGW("Failed to napi_delete_reference");
309 napi_delete_reference(env_, eventListener->onHandlerRef);
310 }
311 }
312 }
313 events_.clear();
314 eventOnces_.clear();
315 }
316 } // namespace DeviceStatus
317 } // namespace Msdp
318 } // namespace OHOS
319