1 /*
2  * Copyright (c) 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 
16 #include "oh_preferences.h"
17 
18 #include "application_context.h"
19 #include "convertor_error_code.h"
20 #include "log_print.h"
21 #include "oh_preferences_err_code.h"
22 #include "oh_preferences_impl.h"
23 #include "oh_preferences_value_impl.h"
24 #include "oh_preferences_value.h"
25 #include "preferences_file_operation.h"
26 #include "preferences_helper.h"
27 #include "securec.h"
28 
29 using namespace OHOS::PreferencesNdk;
30 using namespace OHOS::AbilityRuntime;
31 
OH_PreferencesImpl(std::shared_ptr<OHOS::NativePreferences::Preferences> preferences)32 OH_PreferencesImpl::OH_PreferencesImpl
33     (std::shared_ptr<OHOS::NativePreferences::Preferences> preferences) : preferences_(preferences)
34 {
35 }
36 
PreferencesStructValidCheck(int64_t originCid,int64_t targetCid)37 bool NDKPreferencesUtils::PreferencesStructValidCheck(int64_t originCid, int64_t targetCid)
38 {
39     if (originCid != targetCid) {
40         LOG_ERROR("cid check failed, ori cid: %{public}ld, target cid: %{public}ld", static_cast<long>(originCid),
41             static_cast<long>(targetCid));
42         return false;
43     }
44     return true;
45 }
46 
GetPreferencesDir(OH_PreferencesOption * options)47 std::pair<int, std::string> GetPreferencesDir(OH_PreferencesOption *options)
48 {
49     auto context = OHOS::AbilityRuntime::Context::GetApplicationContext();
50     if (context == nullptr) {
51         LOG_ERROR("get application context go wrong");
52         return { OH_Preferences_ErrCode::PREFERENCES_ERROR_STORAGE, "" };
53     }
54     if (options->GetDataGroupId().empty()) {
55         return { OH_Preferences_ErrCode::PREFERENCES_OK, context->GetPreferencesDir() };
56     }
57     std::string stagePreferencesDir = "";
58     int err = context->GetSystemPreferencesDir(options->GetDataGroupId(), false, stagePreferencesDir);
59     if (err != 0) {
60         LOG_ERROR("get system preferences dir failed, err: %{public}d", err);
61         return { OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM, stagePreferencesDir };
62     }
63     return { OH_Preferences_ErrCode::PREFERENCES_OK, stagePreferencesDir };
64 }
65 
OH_Preferences_Open(OH_PreferencesOption * option,int * errCode)66 OH_Preferences *OH_Preferences_Open(OH_PreferencesOption *option, int *errCode)
67 {
68     int err = OH_Preferences_ErrCode::PREFERENCES_OK;
69     if (option == nullptr || option->fileName.empty() ||
70         !NDKPreferencesUtils::PreferencesStructValidCheck(
71             option->cid, PreferencesNdkStructId::PREFERENCES_OH_OPTION_CID) ||
72         errCode == nullptr) {
73         LOG_ERROR("open preference cfg error, option is null: %{public}d, fileName is null: %{public}d, "
74             "errCode is null: %{public}d, err:%{public}d",
75             (option == nullptr), (option == nullptr) ? 1 : option->fileName.empty(), (errCode == nullptr),
76             OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
77         if (errCode != nullptr) {
78             *errCode = OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
79         }
80         return nullptr;
81     }
82 
83     auto dirRes = GetPreferencesDir(option);
84     if (dirRes.first != OH_Preferences_ErrCode::PREFERENCES_OK) {
85         *errCode = dirRes.first;
86         return nullptr;
87     }
88     std::string filePath = dirRes.second + "/" + option->GetFileName();
89 
90     OHOS::NativePreferences::Options nativeOptions(filePath, option->GetBundleName(),
91         option->GetDataGroupId(), true);
92 
93     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences=
94         OHOS::NativePreferences::PreferencesHelper::GetPreferences(nativeOptions, err);
95     err = ConvertorErrorCode::NativeErrToNdk(err);
96     *errCode = err;
97     if (innerPreferences== nullptr || err != OH_Preferences_ErrCode::PREFERENCES_OK) {
98         LOG_ERROR("Get native Preferences failed: %{public}s, errcode: %{public}d",
99             OHOS::NativePreferences::ExtractFileName(nativeOptions.filePath).c_str(), err);
100         return nullptr;
101     }
102     OH_PreferencesImpl *preferenceImpl = new (std::nothrow) OH_PreferencesImpl(innerPreferences);
103     if (preferenceImpl == nullptr) {
104         LOG_ERROR("new impl object failed");
105         *errCode = OH_Preferences_ErrCode::PREFERENCES_ERROR_MALLOC;
106         return nullptr;
107     }
108     preferenceImpl->SetPreferencesStoreFilePath(filePath);
109     preferenceImpl->cid = PreferencesNdkStructId::PREFERENCES_OH_PREFERENCES_CID;
110     return static_cast<OH_Preferences *>(preferenceImpl);
111 }
112 
GetPreferencesImpl(OH_Preferences * preference)113 static OH_PreferencesImpl *GetPreferencesImpl(OH_Preferences *preference)
114 {
115     if (preference == nullptr ||
116         !NDKPreferencesUtils::PreferencesStructValidCheck(
117             preference->cid, PreferencesNdkStructId::PREFERENCES_OH_PREFERENCES_CID)) {
118         LOG_ERROR("preference invalid, is null: %{public}d", preference == nullptr);
119         return nullptr;
120     }
121     return static_cast<OH_PreferencesImpl *>(preference);
122 }
123 
GetNativePreferencesFromOH(OH_Preferences * preference)124 static std::shared_ptr<OHOS::NativePreferences::Preferences> GetNativePreferencesFromOH(OH_Preferences *preference)
125 {
126     auto preferencesImpl = GetPreferencesImpl(preference);
127     if (preferencesImpl == nullptr ||
128         !NDKPreferencesUtils::PreferencesStructValidCheck(
129             preference->cid, PreferencesNdkStructId::PREFERENCES_OH_PREFERENCES_CID)) {
130         LOG_ERROR("preferences is null: %{public}d when get native preferences from ohPreferences",
131             (preferencesImpl == nullptr));
132         return nullptr;
133     }
134     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= preferencesImpl->GetNativePreferences();
135     if (innerPreferences== nullptr) {
136         LOG_ERROR("preference not open yet");
137         return nullptr;
138     }
139     return innerPreferences;
140 }
141 
SetPreferencesStoreFilePath(const std::string & filePath)142 void OH_PreferencesImpl::SetPreferencesStoreFilePath(const std::string &filePath)
143 {
144     std::unique_lock<std::shared_mutex> writeLock(mutex_);
145     filePath_ = filePath;
146 }
147 
GetPreferencesStoreFilePath()148 std::string OH_PreferencesImpl::GetPreferencesStoreFilePath()
149 {
150     std::shared_lock<std::shared_mutex> readLock(mutex_);
151     return filePath_;
152 }
153 
OH_Preferences_Close(OH_Preferences * preference)154 int OH_Preferences_Close(OH_Preferences *preference)
155 {
156     auto preferencesImpl = GetPreferencesImpl(preference);
157     if (preferencesImpl == nullptr) {
158         LOG_ERROR("preferences close failed, preferences is null");
159         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
160     }
161     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= preferencesImpl->GetNativePreferences();
162     if (innerPreferences== nullptr) {
163         LOG_ERROR("preference not open yet");
164         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
165     }
166 
167     innerPreferences->FlushSync();
168 
169     int errCode = OHOS::NativePreferences::PreferencesHelper::RemovePreferencesFromCache(
170         preferencesImpl->GetPreferencesStoreFilePath());
171     if (errCode != OHOS::NativePreferences::E_OK) {
172         LOG_ERROR("preference close failed: %{public}d", errCode);
173         return ConvertorErrorCode::NativeErrToNdk(errCode);
174     }
175     delete preferencesImpl;
176     return OH_Preferences_ErrCode::PREFERENCES_OK;
177 }
178 
OH_Preferences_GetInt(OH_Preferences * preference,const char * key,int * value)179 int OH_Preferences_GetInt(OH_Preferences *preference, const char *key, int *value)
180 {
181     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
182     if (innerPreferences== nullptr || key == nullptr || value == nullptr) {
183         LOG_ERROR("get int failed, preference not open yet: %{public}d, key is null: %{public}d, "
184             "value is null: %{public}d, err: %{public}d", (innerPreferences== nullptr), (key == nullptr),
185             (value == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
186         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
187     }
188 
189     auto res = innerPreferences->GetValue(key, OHOS::NativePreferences::PreferencesValue());
190     if (res.first != OHOS::NativePreferences::E_OK) {
191         LOG_ERROR("Get Int failed, %{public}d", res.first);
192         return ConvertorErrorCode::NativeErrToNdk(res.first);
193     }
194 
195     if (res.second.IsInt()) {
196         *value = (int)(res.second);
197     } else {
198         LOG_ERROR("Get Int failed, value's type is not int");
199         if (res.first == OHOS::NativePreferences::E_OK) {
200             return OH_Preferences_ErrCode::PREFERENCES_ERROR_KEY_NOT_FOUND;
201         }
202     }
203 
204     return ConvertorErrorCode::NativeErrToNdk(res.first);
205 }
206 
OH_Preferences_GetString(OH_Preferences * preference,const char * key,char ** value,uint32_t * valueLen)207 int OH_Preferences_GetString(OH_Preferences *preference, const char *key, char **value,
208     uint32_t *valueLen)
209 {
210     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
211     if (innerPreferences== nullptr || key == nullptr || value == nullptr || valueLen == nullptr) {
212         LOG_ERROR("get str failed, preference not open yet: %{public}d, key is null: %{public}d, "
213             "value is null: %{public}d, valueLen is null: %{public}d, err: %{public}d",
214             (innerPreferences== nullptr), (key == nullptr), (value == nullptr), (valueLen == nullptr),
215             OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
216         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
217     }
218 
219     auto res = innerPreferences->GetValue(key, OHOS::NativePreferences::PreferencesValue());
220     if (res.first != OHOS::NativePreferences::E_OK) {
221         LOG_ERROR("Get string failed, %{public}d", res.first);
222         return ConvertorErrorCode::NativeErrToNdk(res.first);
223     }
224 
225     if (res.second.IsString()) {
226         std::string str = (std::string)(res.second);
227         size_t strLen = str.size();
228         if (strLen >= SIZE_MAX) {
229             LOG_ERROR(" string length overlimit: %{public}zu", strLen);
230             return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
231         }
232         void *ptr = malloc(strLen + 1); // free by caller
233         if (ptr == nullptr) {
234             LOG_ERROR("malloc failed when get string, errno: %{public}d", errno);
235             return OH_Preferences_ErrCode::PREFERENCES_ERROR_MALLOC;
236         }
237         *value = (char *)ptr;
238         int sysErr = memset_s(*value, (strLen + 1), 0, (strLen + 1));
239         if (sysErr != EOK) {
240             LOG_ERROR("memset failed when get string, errCode: %{public}d", sysErr);
241         }
242         if (strLen > 0) {
243             sysErr = memcpy_s(*value, strLen, str.c_str(), strLen);
244             if (sysErr != EOK) {
245                 LOG_ERROR("memcpy failed when get string, errCode: %{public}d", sysErr);
246                 free(ptr);
247                 return OH_Preferences_ErrCode::PREFERENCES_ERROR_MALLOC;
248             }
249         }
250         *valueLen = strLen + 1;
251     } else {
252         LOG_ERROR("Get string failed, value's type is not string, err: %{public}d", res.first);
253         if (res.first == OHOS::NativePreferences::E_OK) {
254             return OH_Preferences_ErrCode::PREFERENCES_ERROR_KEY_NOT_FOUND;
255         }
256     }
257 
258     return ConvertorErrorCode::NativeErrToNdk(res.first);
259 }
260 
OH_Preferences_FreeString(char * string)261 void OH_Preferences_FreeString(char *string)
262 {
263     if (string == nullptr) {
264         LOG_ERROR("free string failed, string is null");
265         return;
266     }
267     free(string);
268 }
269 
OH_Preferences_GetBool(OH_Preferences * preference,const char * key,bool * value)270 int OH_Preferences_GetBool(OH_Preferences *preference, const char *key, bool *value)
271 {
272     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
273     if (innerPreferences== nullptr || key == nullptr || value == nullptr) {
274         LOG_ERROR("get bool failed, preference not open yet: %{public}d, key is null: %{public}d, "
275             "value is null: %{public}d, err: %{public}d", (innerPreferences== nullptr), (key == nullptr),
276             (value == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
277         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
278     }
279 
280     auto res = innerPreferences->GetValue(key, OHOS::NativePreferences::PreferencesValue());
281     if (res.first != OHOS::NativePreferences::E_OK) {
282         LOG_ERROR("Get bool failed, %{public}d", res.first);
283         return ConvertorErrorCode::NativeErrToNdk(res.first);
284     }
285 
286     if (res.second.IsBool()) {
287         *value = (bool)(res.second);
288     } else {
289         LOG_ERROR("Get bool failed, value's type is not bool, err: %{public}d", res.first);
290         if (res.first == OHOS::NativePreferences::E_OK) {
291             return OH_Preferences_ErrCode::PREFERENCES_ERROR_KEY_NOT_FOUND;
292         }
293     }
294 
295     return ConvertorErrorCode::NativeErrToNdk(res.first);
296 }
297 
OH_Preferences_SetInt(OH_Preferences * preference,const char * key,int value)298 int OH_Preferences_SetInt(OH_Preferences *preference, const char *key, int value)
299 {
300     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
301     if (innerPreferences== nullptr || key == nullptr) {
302         LOG_ERROR("set int failed, preference not open yet: %{public}d, key is null: %{public}d, err: %{public}d",
303             (innerPreferences== nullptr), (key == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
304         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
305     }
306 
307     int errCode = innerPreferences->PutInt(key, value);
308     if (errCode != OHOS::NativePreferences::E_OK) {
309         LOG_ERROR("preference put int failed, err: %{public}d", errCode);
310     }
311     return ConvertorErrorCode::NativeErrToNdk(errCode);
312 }
313 
OH_Preferences_SetBool(OH_Preferences * preference,const char * key,bool value)314 int OH_Preferences_SetBool(OH_Preferences *preference, const char *key, bool value)
315 {
316     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
317     if (innerPreferences== nullptr || key == nullptr) {
318         LOG_ERROR("set bool failed, preference not open yet: %{public}d, key is null: %{public}d, err: %{public}d",
319             (innerPreferences== nullptr), (key == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
320         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
321     }
322 
323     int errCode = innerPreferences->PutBool(key, value);
324     if (errCode != OHOS::NativePreferences::E_OK) {
325         LOG_ERROR("preference put bool failed, err: %{public}d", errCode);
326     }
327     return ConvertorErrorCode::NativeErrToNdk(errCode);
328 }
329 
OH_Preferences_SetString(OH_Preferences * preference,const char * key,const char * value)330 int OH_Preferences_SetString(OH_Preferences *preference, const char *key, const char *value)
331 {
332     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
333     if (innerPreferences== nullptr || key == nullptr || value == nullptr) {
334         LOG_ERROR("set str failed, preference not open yet: %{public}d, key is null: %{public}d, "
335             "value is null: %{public}d, err: %{public}d", (innerPreferences== nullptr), (key == nullptr),
336             (value == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
337         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
338     }
339 
340     int errCode = innerPreferences->PutString(key, value);
341     if (errCode != OHOS::NativePreferences::E_OK) {
342         LOG_ERROR("preference put string failed, err: %{public}d", errCode);
343     }
344     return ConvertorErrorCode::NativeErrToNdk(errCode);
345 }
346 
OH_Preferences_Delete(OH_Preferences * preference,const char * key)347 int OH_Preferences_Delete(OH_Preferences *preference, const char *key)
348 {
349     std::shared_ptr<OHOS::NativePreferences::Preferences> innerPreferences= GetNativePreferencesFromOH(preference);
350     if (innerPreferences== nullptr || key == nullptr) {
351         LOG_ERROR("delete failed, preference not open yet: %{public}d, key is null: %{public}d, err: %{public}d",
352             (innerPreferences== nullptr), (key == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
353         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
354     }
355 
356     int errCode = innerPreferences->Delete(key);
357     if (errCode != OHOS::NativePreferences::E_OK) {
358         LOG_ERROR("preference delete value failed, err: %{public}d", errCode);
359     }
360     return ConvertorErrorCode::NativeErrToNdk(errCode);
361 }
362 
OH_Preferences_RegisterDataObserver(OH_Preferences * preference,void * context,OH_PreferencesDataObserver observer,const char * keys[],uint32_t keyCount)363 int OH_Preferences_RegisterDataObserver(OH_Preferences *preference, void *context,
364     OH_PreferencesDataObserver observer, const char *keys[], uint32_t keyCount)
365 {
366     auto preferencesImpl = GetPreferencesImpl(preference);
367     if (preferencesImpl == nullptr || observer == nullptr || keys == nullptr) {
368         LOG_ERROR("register failed, sp is null ? %{public}d, obs is null ? %{public}d, "
369             "keys is null: %{public}d, err: %{public}d", (preferencesImpl == nullptr), (observer == nullptr),
370             (keys == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
371         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
372     }
373 
374     std::vector<std::string> keysVec;
375     for (uint32_t i = 0; i < keyCount; i++) {
376         keysVec.push_back(keys[i]);
377     }
378 
379     return ConvertorErrorCode::NativeErrToNdk(preferencesImpl->RegisterDataObserver(observer, context, keysVec));
380 }
381 
OH_Preferences_UnregisterDataObserver(OH_Preferences * preference,void * context,OH_PreferencesDataObserver observer,const char * keys[],uint32_t keyCount)382 int OH_Preferences_UnregisterDataObserver(OH_Preferences *preference, void *context,
383     OH_PreferencesDataObserver observer, const char *keys[], uint32_t keyCount)
384 {
385     auto preferencesImpl = GetPreferencesImpl(preference);
386     if (preferencesImpl == nullptr || observer == nullptr || keys == nullptr) {
387         LOG_ERROR("unregister failed, sp is null ? %{public}d, obs is null ? %{public}d, "
388             "keys is null: %{public}d, err: %{public}d", (preferencesImpl == nullptr), (observer == nullptr),
389             (keys == nullptr), OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM);
390         return OH_Preferences_ErrCode::PREFERENCES_ERROR_INVALID_PARAM;
391     }
392     std::vector<std::string> keysVec;
393     for (uint32_t i = 0; i < keyCount; i++) {
394         keysVec.push_back(keys[i]);
395     }
396     return ConvertorErrorCode::NativeErrToNdk(preferencesImpl->UnregisterDataObserver(observer, context, keysVec));
397 }
398 
RegisterDataObserver(OH_PreferencesDataObserver observer,void * context,const std::vector<std::string> & keys)399 int OH_PreferencesImpl::RegisterDataObserver(
400     OH_PreferencesDataObserver observer, void *context, const std::vector<std::string> &keys)
401 {
402     std::unique_lock<std::shared_mutex> writeLock(obsMutex_);
403 
404     auto ndkObserver = std::make_shared<NDKPreferencesObserver>(observer, context);
405     int errCode = preferences_->RegisterDataObserver(ndkObserver, keys);
406     if (errCode != OHOS::NativePreferences::E_OK) {
407         LOG_ERROR("register failed, err: %{public}d", errCode);
408     } else {
409         dataObservers_.emplace_back(std::make_pair(std::move(ndkObserver), context));
410     }
411     return ConvertorErrorCode::NativeErrToNdk(errCode);
412 }
413 
NDKPreferencesObserver(OH_PreferencesDataObserver observer,void * context)414 NDKPreferencesObserver::NDKPreferencesObserver(OH_PreferencesDataObserver observer, void *context)
415     : dataObserver_(observer), context_(context) {}
416 
FreePairValue(OH_PreferencesPair * pairs,size_t count)417 inline void FreePairValue(OH_PreferencesPair *pairs, size_t count)
418 {
419     for (size_t i = 0; i < count; i++) {
420         delete pairs[i].value;
421     }
422 }
423 
OnChange(const std::map<std::string,OHOS::NativePreferences::PreferencesValue> & records)424 void NDKPreferencesObserver::OnChange(const std::map<std::string, OHOS::NativePreferences::PreferencesValue> &records)
425 {
426     if (dataObserver_ == nullptr) {
427         LOG_ERROR("failed to trigger change, data observer is null");
428         return;
429     }
430     auto count = records.size();
431     if (count == 0) {
432         return;
433     }
434     OH_PreferencesPair *pairs = new (std::nothrow) OH_PreferencesPair[count];
435     if (pairs == nullptr) {
436         LOG_ERROR("malloc pairs failed when on change, count: %{public}d, errno:%{public}d", static_cast<int>(count),
437             errno);
438         return;
439     }
440     int i = 0;
441     for (const auto &[key, value] : records) {
442         OH_PreferencesValueImpl *valueImpl = new (std::nothrow) OH_PreferencesValueImpl();
443         if (valueImpl == nullptr) {
444             LOG_ERROR("new value object failed");
445             FreePairValue(pairs, i);
446             delete []pairs;
447         }
448         valueImpl->cid = PreferencesNdkStructId::PREFERENCES_OH_VALUE_CID;
449         valueImpl->value_ = value;
450         pairs[i++] = OH_PreferencesPair { PreferencesNdkStructId::PREFERENCES_OH_PAIR_CID, key.c_str(),
451             static_cast<OH_PreferencesValue *>(valueImpl), count};
452     }
453     (dataObserver_)(context_, pairs, count);
454     FreePairValue(pairs, count);
455     delete []pairs;
456 }
457 
OnChange(const std::string & key)458 void NDKPreferencesObserver::OnChange(const std::string &key)
459 {
460 }
461 
UnregisterDataObserver(OH_PreferencesDataObserver observer,void * context,const std::vector<std::string> & keys)462 int OH_PreferencesImpl::UnregisterDataObserver(OH_PreferencesDataObserver observer, void *context,
463     const std::vector<std::string> &keys)
464 {
465     std::unique_lock<std::shared_mutex> writeLock(obsMutex_);
466     for (size_t i = 0; i < dataObservers_.size(); i++) {
467         if (!dataObservers_[i].first->ObserverCompare(observer) || dataObservers_[i].second != context) {
468             continue;
469         }
470 
471         int errCode = preferences_->UnRegisterDataObserver(dataObservers_[i].first, keys);
472         if (errCode != OHOS::NativePreferences::E_OK) {
473             LOG_ERROR("un register observer failed, err: %{public}d", errCode);
474             return ConvertorErrorCode::NativeErrToNdk(errCode);
475         }
476         if (keys.empty()) {
477             dataObservers_[i] = { nullptr, nullptr };
478             dataObservers_.erase(dataObservers_.begin() + i);
479         }
480     }
481     return OH_Preferences_ErrCode::PREFERENCES_OK;
482 }
483 
ObserverCompare(OH_PreferencesDataObserver other)484 bool NDKPreferencesObserver::ObserverCompare(OH_PreferencesDataObserver other)
485 {
486     if (other == nullptr) {
487         return false;
488     }
489     return  other == dataObserver_;
490 }
491