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