1 /*
2 * Copyright (c) 2021-2024 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 #include "dataobs_mgr_inner_pref.h"
16
17 #include "data_ability_observer_stub.h"
18 #include "dataobs_mgr_errors.h"
19 #include "hilog_tag_wrapper.h"
20 #include "common_utils.h"
21
22 namespace OHOS {
23 namespace AAFwk {
24
DataObsMgrInnerPref()25 DataObsMgrInnerPref::DataObsMgrInnerPref() {}
26
~DataObsMgrInnerPref()27 DataObsMgrInnerPref::~DataObsMgrInnerPref() {}
28
HandleRegisterObserver(const Uri & uri,sptr<IDataAbilityObserver> dataObserver)29 int DataObsMgrInnerPref::HandleRegisterObserver(const Uri &uri, sptr<IDataAbilityObserver> dataObserver)
30 {
31 std::lock_guard<std::mutex> lock(preferenceMutex_);
32
33 auto [obsPair, flag] = observers_.try_emplace(uri.ToString(), std::list<sptr<IDataAbilityObserver>>());
34 if (!flag && obsPair->second.size() > OBS_NUM_MAX) {
35 TAG_LOGE(AAFwkTag::DBOBSMGR,
36 "subscribers num:%{public}s num maxed",
37 CommonUtils::Anonymous(uri.ToString()).c_str());
38 return DATAOBS_SERVICE_OBS_LIMMIT;
39 }
40
41 for (auto obs = obsPair->second.begin(); obs != obsPair->second.end(); obs++) {
42 if ((*obs)->AsObject() == dataObserver->AsObject()) {
43 TAG_LOGE(AAFwkTag::DBOBSMGR, "registered obs:%{public}s",
44 CommonUtils::Anonymous(uri.ToString()).c_str());
45 return OBS_EXIST;
46 }
47 }
48
49 obsPair->second.push_back(dataObserver);
50
51 AddObsDeathRecipient(dataObserver);
52 return NO_ERROR;
53 }
54
HandleUnregisterObserver(const Uri & uri,sptr<IDataAbilityObserver> dataObserver)55 int DataObsMgrInnerPref::HandleUnregisterObserver(const Uri &uri, sptr<IDataAbilityObserver> dataObserver)
56 {
57 std::lock_guard<std::mutex> lock(preferenceMutex_);
58
59 auto obsPair = observers_.find(uri.ToString());
60 if (obsPair == observers_.end()) {
61 TAG_LOGW(
62 AAFwkTag::DBOBSMGR, "uris no obs:%{public}s", CommonUtils::Anonymous(uri.ToString()).c_str());
63 return NO_OBS_FOR_URI;
64 }
65
66 TAG_LOGD(AAFwkTag::DBOBSMGR, "obs num:%{public}zu:%{public}s", obsPair->second.size(),
67 CommonUtils::Anonymous(uri.ToString()).c_str());
68 auto obs = obsPair->second.begin();
69 for (; obs != obsPair->second.end(); obs++) {
70 if ((*obs)->AsObject() == dataObserver->AsObject()) {
71 break;
72 }
73 }
74 if (obs == obsPair->second.end()) {
75 TAG_LOGW(
76 AAFwkTag::DBOBSMGR, "uris no obs:%{public}s", CommonUtils::Anonymous(uri.ToString()).c_str());
77 return NO_OBS_FOR_URI;
78 }
79 obsPair->second.remove(*obs);
80 if (obsPair->second.empty()) {
81 observers_.erase(obsPair);
82 }
83
84 if (!HaveRegistered(dataObserver)) {
85 RemoveObsDeathRecipient(dataObserver->AsObject());
86 }
87 return NO_ERROR;
88 }
89
HandleNotifyChange(const Uri & uri)90 int DataObsMgrInnerPref::HandleNotifyChange(const Uri &uri)
91 {
92 std::list<sptr<IDataAbilityObserver>> obsList;
93 std::lock_guard<std::mutex> lock(preferenceMutex_);
94 {
95 std::string uriStr = uri.ToString();
96 size_t pos = uriStr.find('?');
97 if (pos == std::string::npos) {
98 TAG_LOGW(AAFwkTag::DBOBSMGR, "uri missing query:%{public}s",
99 CommonUtils::Anonymous(uriStr).c_str());
100 return INVALID_PARAM;
101 }
102 std::string observerKey = uriStr.substr(0, pos);
103 auto obsPair = observers_.find(observerKey);
104 if (obsPair == observers_.end()) {
105 TAG_LOGD(AAFwkTag::DBOBSMGR, "uri no obs:%{public}s",
106 CommonUtils::Anonymous(uri.ToString()).c_str());
107 return NO_OBS_FOR_URI;
108 }
109 obsList = obsPair->second;
110 }
111
112 for (auto &obs : obsList) {
113 if (obs != nullptr) {
114 obs->OnChangePreferences(const_cast<Uri &>(uri).GetQuery());
115 }
116 }
117
118 TAG_LOGD(AAFwkTag::DBOBSMGR, "uri end:%{public}s,obs num:%{public}zu",
119 CommonUtils::Anonymous(uri.ToString()).c_str(), obsList.size());
120 return NO_ERROR;
121 }
122
AddObsDeathRecipient(sptr<IDataAbilityObserver> dataObserver)123 void DataObsMgrInnerPref::AddObsDeathRecipient(sptr<IDataAbilityObserver> dataObserver)
124 {
125 if ((dataObserver == nullptr) || dataObserver->AsObject() == nullptr) {
126 return;
127 }
128
129 auto it = obsRecipient_.find(dataObserver->AsObject());
130 if (it != obsRecipient_.end()) {
131 TAG_LOGW(AAFwkTag::DBOBSMGR, "called");
132 return;
133 } else {
134 std::weak_ptr<DataObsMgrInnerPref> thisWeakPtr(shared_from_this());
135 sptr<IRemoteObject::DeathRecipient> deathRecipient =
136 new DataObsCallbackRecipient([thisWeakPtr](const wptr<IRemoteObject> &remote) {
137 auto dataObsMgrInner = thisWeakPtr.lock();
138 if (dataObsMgrInner) {
139 dataObsMgrInner->OnCallBackDied(remote);
140 }
141 });
142 if (!dataObserver->AsObject()->AddDeathRecipient(deathRecipient)) {
143 TAG_LOGE(AAFwkTag::DBOBSMGR, "failed");
144 }
145 obsRecipient_.emplace(dataObserver->AsObject(), deathRecipient);
146 }
147 }
148
RemoveObsDeathRecipient(sptr<IRemoteObject> dataObserver)149 void DataObsMgrInnerPref::RemoveObsDeathRecipient(sptr<IRemoteObject> dataObserver)
150 {
151 if (dataObserver == nullptr) {
152 return;
153 }
154
155 auto it = obsRecipient_.find(dataObserver);
156 if (it != obsRecipient_.end()) {
157 it->first->RemoveDeathRecipient(it->second);
158 obsRecipient_.erase(it);
159 return;
160 }
161 }
162
OnCallBackDied(const wptr<IRemoteObject> & remote)163 void DataObsMgrInnerPref::OnCallBackDied(const wptr<IRemoteObject> &remote)
164 {
165 auto dataObserver = remote.promote();
166 if (dataObserver == nullptr) {
167 return;
168 }
169 std::lock_guard<std::mutex> lock(preferenceMutex_);
170
171 if (dataObserver == nullptr) {
172 TAG_LOGE(AAFwkTag::DBOBSMGR, "null dataObserver");
173 return;
174 }
175
176 RemoveObs(dataObserver);
177 }
178
RemoveObs(sptr<IRemoteObject> dataObserver)179 void DataObsMgrInnerPref::RemoveObs(sptr<IRemoteObject> dataObserver)
180 {
181 for (auto iter = observers_.begin(); iter != observers_.end();) {
182 auto &obsList = iter->second;
183 for (auto it = obsList.begin(); it != obsList.end(); it++) {
184 if ((*it)->AsObject() == dataObserver) {
185 TAG_LOGD(AAFwkTag::DBOBSMGR, "erase");
186 obsList.erase(it);
187 break;
188 }
189 }
190 if (obsList.size() == 0) {
191 iter = observers_.erase(iter);
192 } else {
193 iter++;
194 }
195 }
196 RemoveObsDeathRecipient(dataObserver);
197 }
198
HaveRegistered(sptr<IDataAbilityObserver> dataObserver)199 bool DataObsMgrInnerPref::HaveRegistered(sptr<IDataAbilityObserver> dataObserver)
200 {
201 for (auto &[key, value] : observers_) {
202 auto obs = std::find(value.begin(), value.end(), dataObserver);
203 if (obs != value.end()) {
204 return true;
205 }
206 }
207 return false;
208 }
209 } // namespace AAFwk
210 } // namespace OHOS
211