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 #define LOG_TAG "DataProviderConfig"
16
17 #include "data_provider_config.h"
18
19 #include <vector>
20
21 #include "accesstoken_kit.h"
22 #include "account/account_delegate.h"
23 #include "datashare_errno.h"
24 #include "hap_token_info.h"
25 #include "log_print.h"
26 #include "strategies/general/load_config_common_strategy.h"
27 #include "uri_utils.h"
28 #include "utils/anonymous.h"
29
30 namespace OHOS::DataShare {
31 using namespace OHOS::DistributedData;
DataProviderConfig(const std::string & uri,uint32_t callerTokenId)32 DataProviderConfig::DataProviderConfig(const std::string &uri, uint32_t callerTokenId)
33 {
34 providerInfo_.uri = uri;
35 providerInfo_.currentUserId = DistributedKv::AccountDelegate::GetInstance()->GetUserByToken(callerTokenId);
36 if (providerInfo_.currentUserId == 0) {
37 LoadConfigCommonStrategy::GetInfoFromProxyURI(providerInfo_.uri, providerInfo_.currentUserId,
38 callerTokenId, providerInfo_.bundleName);
39 URIUtils::FormatUri(providerInfo_.uri);
40 }
41 uriConfig_ = URIUtils::GetUriConfig(providerInfo_.uri);
42 }
43
GetBundleInfo()44 std::pair<int, BundleConfig> DataProviderConfig::GetBundleInfo()
45 {
46 BundleConfig bundleInfo;
47 providerInfo_.bundleName = uriConfig_.authority;
48 if (providerInfo_.bundleName.empty()) {
49 if (uriConfig_.pathSegments.empty()) {
50 return std::make_pair(E_URI_NOT_EXIST, bundleInfo);
51 }
52 providerInfo_.bundleName = uriConfig_.pathSegments[0];
53 }
54 auto ret = BundleMgrProxy::GetInstance()->GetBundleInfoFromBMS(
55 providerInfo_.bundleName, providerInfo_.currentUserId, bundleInfo);
56 return std::make_pair(ret, bundleInfo);
57 }
58
GetFromProxyData()59 int DataProviderConfig::GetFromProxyData()
60 {
61 auto [errCode, bundleInfo] = GetBundleInfo();
62 if (errCode != E_OK) {
63 ZLOGE("Get bundleInfo failed! bundleName:%{public}s, userId:%{public}d, uri:%{public}s",
64 providerInfo_.bundleName.c_str(), providerInfo_.currentUserId,
65 URIUtils::Anonymous(providerInfo_.uri).c_str());
66 return errCode;
67 }
68 providerInfo_.singleton = bundleInfo.singleton;
69 for (auto &item : bundleInfo.extensionInfos) {
70 if (item.type != AppExecFwk::ExtensionAbilityType::DATASHARE) {
71 continue;
72 }
73 providerInfo_.hasExtension = true;
74 break;
75 }
76 for (auto &hapModuleInfo : bundleInfo.hapModuleInfos) {
77 auto &proxyDatas = hapModuleInfo.proxyDatas;
78 std::sort(proxyDatas.begin(), proxyDatas.end(), [](const ProxyData &curr,
79 const ProxyData &prev) {
80 return curr.uri.length() > prev.uri.length();
81 });
82 for (auto &data : proxyDatas) {
83 if (data.uri.length() > uriConfig_.formatUri.length() ||
84 uriConfig_.formatUri.compare(0, data.uri.length(), data.uri) != 0) {
85 continue;
86 }
87 providerInfo_.readPermission = std::move(data.requiredReadPermission);
88 providerInfo_.writePermission = std::move(data.requiredWritePermission);
89 auto profileInfo = data.profileInfo;
90 if (profileInfo.resultCode == NOT_FOUND) {
91 return E_OK;
92 }
93 if (profileInfo.resultCode == ERROR) {
94 ZLOGE("Profile unmarshall error.uri: %{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
95 return E_ERROR;
96 }
97 return GetFromDataProperties(profileInfo.profile, hapModuleInfo.moduleName);
98 }
99 }
100 return E_URI_NOT_EXIST;
101 }
102
GetFromDataProperties(const ProfileInfo & profileInfo,const std::string & moduleName)103 int DataProviderConfig::GetFromDataProperties(const ProfileInfo &profileInfo,
104 const std::string &moduleName)
105 {
106 if (profileInfo.scope == MODULE_SCOPE) {
107 providerInfo_.moduleName = moduleName;
108 }
109 providerInfo_.storeName = profileInfo.storeName;
110 providerInfo_.tableName = profileInfo.tableName;
111 providerInfo_.type = profileInfo.type;
112 providerInfo_.storeMetaDataFromUri = profileInfo.storeMetaDataFromUri;
113 providerInfo_.backup = profileInfo.backup;
114 providerInfo_.extensionUri = profileInfo.extUri;
115 if (profileInfo.tableConfig.empty()) {
116 return E_OK;
117 }
118 return GetFromExtensionProperties(profileInfo, moduleName);
119 }
120
GetFromExtensionProperties(const ProfileInfo & profileInfo,const std::string & moduleName)121 int DataProviderConfig::GetFromExtensionProperties(const ProfileInfo &profileInfo,
122 const std::string &moduleName)
123 {
124 std::string storeUri = URIUtils::DATA_SHARE_SCHEMA + providerInfo_.bundleName + URIUtils::URI_SEPARATOR +
125 moduleName + URIUtils::URI_SEPARATOR + providerInfo_.storeName;
126 std::string tableUri = storeUri + URIUtils::URI_SEPARATOR + providerInfo_.tableName;
127 providerInfo_.accessCrossMode = DataShareProfileConfig::GetAccessCrossMode(profileInfo, tableUri, storeUri);
128 if (providerInfo_.singleton && providerInfo_.accessCrossMode == AccessCrossMode::USER_UNDEFINED) {
129 ZLOGE("Single app must config user cross mode,bundleName:%{public}s, uri:%{public}s",
130 providerInfo_.bundleName.c_str(), URIUtils::Anonymous(providerInfo_.uri).c_str());
131 return E_ERROR;
132 }
133 if (providerInfo_.singleton && providerInfo_.accessCrossMode == AccessCrossMode::USER_SINGLE) {
134 providerInfo_.tableName.append("_").append(std::to_string(providerInfo_.currentUserId));
135 }
136 return E_OK;
137 }
138
GetFromExtension()139 int DataProviderConfig::GetFromExtension()
140 {
141 if (!GetFromUriPath()) {
142 ZLOGE("Uri path failed! uri:%{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
143 return E_URI_NOT_EXIST;
144 }
145 BundleConfig bundleInfo;
146 auto ret = BundleMgrProxy::GetInstance()->GetBundleInfoFromBMS(
147 providerInfo_.bundleName, providerInfo_.currentUserId, bundleInfo);
148 if (ret != E_OK) {
149 ZLOGE("BundleInfo failed! bundleName: %{public}s", providerInfo_.bundleName.c_str());
150 return ret;
151 }
152 providerInfo_.singleton = bundleInfo.singleton;
153 providerInfo_.allowEmptyPermission = true;
154 for (auto &item : bundleInfo.extensionInfos) {
155 if (item.type != AppExecFwk::ExtensionAbilityType::DATASHARE) {
156 continue;
157 }
158 providerInfo_.hasExtension = true;
159 providerInfo_.readPermission = std::move(item.readPermission);
160 providerInfo_.writePermission = std::move(item.writePermission);
161 auto profileInfo = item.profileInfo;
162 if (profileInfo.resultCode == NOT_FOUND) {
163 return E_OK;
164 }
165 if (profileInfo.resultCode == ERROR) {
166 ZLOGE("Profile Unmarshall failed! uri:%{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
167 return E_ERROR;
168 }
169 return GetFromExtensionProperties(profileInfo.profile, providerInfo_.moduleName);
170 }
171 return E_URI_NOT_EXIST;
172 }
173
GetFromUriPath()174 bool DataProviderConfig::GetFromUriPath()
175 {
176 auto& pathSegments = uriConfig_.pathSegments;
177 if (pathSegments.size() < static_cast<std::size_t>(PATH_PARAM::PARAM_SIZE) ||
178 pathSegments[static_cast<std::size_t>(PATH_PARAM::BUNDLE_NAME)].empty() ||
179 pathSegments[static_cast<std::size_t>(PATH_PARAM::MODULE_NAME)].empty() ||
180 pathSegments[static_cast<std::size_t>(PATH_PARAM::STORE_NAME)].empty() ||
181 pathSegments[static_cast<std::size_t>(PATH_PARAM::TABLE_NAME)].empty()) {
182 ZLOGE("Invalid uri ! uri: %{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
183 return false;
184 }
185 providerInfo_.bundleName = pathSegments[static_cast<std::size_t>(PATH_PARAM::BUNDLE_NAME)];
186 providerInfo_.moduleName = pathSegments[static_cast<std::size_t>(PATH_PARAM::MODULE_NAME)];
187 providerInfo_.storeName = pathSegments[static_cast<std::size_t>(PATH_PARAM::STORE_NAME)];
188 providerInfo_.tableName = pathSegments[static_cast<std::size_t>(PATH_PARAM::TABLE_NAME)];
189 return true;
190 }
191
GetMetaDataFromUri()192 void DataProviderConfig::GetMetaDataFromUri()
193 {
194 if (!providerInfo_.storeMetaDataFromUri) {
195 return;
196 }
197 if (!GetFromUriPath()) {
198 ZLOGE("Uri path failed, not change metaData from uri! uri:%{public}s",
199 URIUtils::Anonymous(providerInfo_.uri).c_str());
200 }
201 }
202
GetProviderInfo()203 std::pair<int, DataProviderConfig::ProviderInfo> DataProviderConfig::GetProviderInfo()
204 {
205 auto ret = GetFromProxyData();
206 if (ret == E_OK) {
207 GetMetaDataFromUri();
208 return std::make_pair(ret, providerInfo_);
209 }
210 ret = GetFromExtension();
211 if (ret != E_OK) {
212 ZLOGE("Get providerInfo failed! ret: %{public}d, uri: %{public}s",
213 ret, URIUtils::Anonymous(providerInfo_.uri).c_str());
214 }
215 return std::make_pair(ret, providerInfo_);
216 }
217 } // namespace OHOS::DataShare
218