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