• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 = 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             providerInfo_.allowLists = std::move(data.profileInfo.profile.allowLists);
100             auto profileInfo = data.profileInfo;
101             if (profileInfo.resultCode == NOT_FOUND) {
102                 return E_OK;
103             }
104             if (profileInfo.resultCode == ERROR) {
105                 ZLOGE("Profile unmarshall error.uri: %{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
106                 return E_ERROR;
107             }
108             return GetFromDataProperties(profileInfo.profile, hapModuleInfo.moduleName);
109         }
110     }
111     return E_URI_NOT_EXIST;
112 }
113 
GetFromDataProperties(const ProfileInfo & profileInfo,const std::string & moduleName)114 int DataProviderConfig::GetFromDataProperties(const ProfileInfo &profileInfo,
115     const std::string &moduleName)
116 {
117     if (profileInfo.scope == MODULE_SCOPE) {
118         providerInfo_.moduleName = moduleName;
119     }
120     providerInfo_.storeName = profileInfo.storeName;
121     providerInfo_.tableName = profileInfo.tableName;
122     providerInfo_.type = profileInfo.type;
123     providerInfo_.storeMetaDataFromUri = profileInfo.storeMetaDataFromUri;
124     providerInfo_.backup = profileInfo.backup;
125     providerInfo_.extensionUri = profileInfo.extUri;
126     if (profileInfo.tableConfig.empty()) {
127         return E_OK;
128     }
129     return GetFromExtensionProperties(profileInfo, moduleName);
130 }
131 
GetFromExtensionProperties(const ProfileInfo & profileInfo,const std::string & moduleName)132 int DataProviderConfig::GetFromExtensionProperties(const ProfileInfo &profileInfo,
133     const std::string &moduleName)
134 {
135     std::string storeUri = URIUtils::DATA_SHARE_SCHEMA + providerInfo_.bundleName + URIUtils::URI_SEPARATOR +
136             moduleName + URIUtils::URI_SEPARATOR + providerInfo_.storeName;
137     std::string tableUri = storeUri + URIUtils::URI_SEPARATOR + providerInfo_.tableName;
138     providerInfo_.accessCrossMode = DataShareProfileConfig::GetAccessCrossMode(profileInfo, tableUri, storeUri);
139     if (providerInfo_.singleton && providerInfo_.accessCrossMode == AccessCrossMode::USER_UNDEFINED) {
140         ZLOGE("Single app must config user cross mode,bundleName:%{public}s, uri:%{public}s",
141             providerInfo_.bundleName.c_str(), URIUtils::Anonymous(providerInfo_.uri).c_str());
142         return E_ERROR;
143     }
144     if (providerInfo_.singleton && providerInfo_.accessCrossMode == AccessCrossMode::USER_SINGLE) {
145         providerInfo_.tableName.append("_").append(std::to_string(providerInfo_.visitedUserId));
146     }
147     return E_OK;
148 }
149 
GetFromExtension()150 int DataProviderConfig::GetFromExtension()
151 {
152     providerInfo_.isFromExtension = true;
153     if (!GetFromUriPath()) {
154         ZLOGE("Uri path failed! uri:%{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
155         return E_URI_NOT_EXIST;
156     }
157     BundleConfig bundleInfo;
158     auto ret = BundleMgrProxy::GetInstance()->GetBundleInfoFromBMS(
159         providerInfo_.bundleName, providerInfo_.visitedUserId, bundleInfo, providerInfo_.appIndex);
160     if (ret != E_OK) {
161         ZLOGE("BundleInfo failed! bundleName: %{public}s", providerInfo_.bundleName.c_str());
162         return ret;
163     }
164     providerInfo_.singleton = bundleInfo.singleton;
165     providerInfo_.allowEmptyPermission = true;
166     for (auto &item : bundleInfo.extensionInfos) {
167         if (item.type != AppExecFwk::ExtensionAbilityType::DATASHARE) {
168             continue;
169         }
170         providerInfo_.hasExtension = true;
171         providerInfo_.readPermission = std::move(item.readPermission);
172         providerInfo_.writePermission = std::move(item.writePermission);
173         auto profileInfo = item.profileInfo;
174         if (profileInfo.resultCode == NOT_FOUND) {
175             return E_OK;
176         }
177         if (profileInfo.resultCode == ERROR) {
178             ZLOGE("Profile Unmarshall failed! uri:%{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
179             return E_ERROR;
180         }
181         return GetFromExtensionProperties(profileInfo.profile, providerInfo_.moduleName);
182     }
183     return E_URI_NOT_EXIST;
184 }
185 
GetFromUriPath()186 bool DataProviderConfig::GetFromUriPath()
187 {
188     auto& pathSegments = uriConfig_.pathSegments;
189     if (pathSegments.size() < static_cast<std::size_t>(PATH_PARAM::PARAM_SIZE) ||
190         pathSegments[static_cast<std::size_t>(PATH_PARAM::BUNDLE_NAME)].empty() ||
191         pathSegments[static_cast<std::size_t>(PATH_PARAM::MODULE_NAME)].empty() ||
192         pathSegments[static_cast<std::size_t>(PATH_PARAM::STORE_NAME)].empty() ||
193         pathSegments[static_cast<std::size_t>(PATH_PARAM::TABLE_NAME)].empty()) {
194         ZLOGE("Invalid uri ! uri: %{public}s", URIUtils::Anonymous(providerInfo_.uri).c_str());
195         return false;
196     }
197     providerInfo_.bundleName = pathSegments[static_cast<std::size_t>(PATH_PARAM::BUNDLE_NAME)];
198     providerInfo_.moduleName = pathSegments[static_cast<std::size_t>(PATH_PARAM::MODULE_NAME)];
199     providerInfo_.storeName = pathSegments[static_cast<std::size_t>(PATH_PARAM::STORE_NAME)];
200     providerInfo_.tableName = pathSegments[static_cast<std::size_t>(PATH_PARAM::TABLE_NAME)];
201     return true;
202 }
203 
GetMetaDataFromUri()204 void DataProviderConfig::GetMetaDataFromUri()
205 {
206     if (!providerInfo_.storeMetaDataFromUri) {
207         return;
208     }
209     if (!GetFromUriPath()) {
210         ZLOGE("Uri path failed, not change metaData from uri! uri:%{public}s",
211             URIUtils::Anonymous(providerInfo_.uri).c_str());
212     }
213 }
214 
GetProviderInfo()215 std::pair<int, DataProviderConfig::ProviderInfo> DataProviderConfig::GetProviderInfo()
216 {
217     if (providerInfo_.appIndex == -1) {
218         return std::make_pair(E_APPINDEX_INVALID, providerInfo_);
219     }
220     if (providerInfo_.visitedUserId == -1) {
221         return std::make_pair(E_INVALID_USER_ID, providerInfo_);
222     }
223     auto ret = GetFromProxyData();
224     if (ret == E_OK) {
225         GetMetaDataFromUri();
226         return std::make_pair(ret, providerInfo_);
227     }
228     ret = GetFromExtension();
229     if (ret != E_OK) {
230         ZLOGE("Get providerInfo failed! ret: %{public}d, uri: %{public}s",
231             ret, URIUtils::Anonymous(providerInfo_.uri).c_str());
232     }
233     return std::make_pair(ret, providerInfo_);
234 }
235 } // namespace OHOS::DataShare
236