• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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