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