1 /*
2  * Copyright (c) 2021 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 #define LOG_TAG "processCommunication"
17 
18 #include "communicator_context.h"
19 #include "device_manager_adapter.h"
20 #include "log_print.h"
21 #include "softbus_adapter.h"
22 #include "process_communicator_impl.h"
23 namespace OHOS {
24 namespace AppDistributedKv {
25 using namespace DistributedDB;
26 using namespace OHOS::DistributedData;
27 using DmAdapter = OHOS::DistributedData::DeviceManagerAdapter;
ProcessCommunicatorImpl()28 ProcessCommunicatorImpl::ProcessCommunicatorImpl()
29 {
30 }
31 
ProcessCommunicatorImpl(RouteHeadHandlerCreator handlerCreator)32 ProcessCommunicatorImpl::ProcessCommunicatorImpl(RouteHeadHandlerCreator handlerCreator)
33     : routeHeadHandlerCreator_(std::move(handlerCreator))
34 {
35 }
36 
~ProcessCommunicatorImpl()37 ProcessCommunicatorImpl::~ProcessCommunicatorImpl()
38 {
39     ZLOGE("destructor.");
40 }
41 
Start(const std::string & processLabel)42 DBStatus ProcessCommunicatorImpl::Start(const std::string &processLabel)
43 {
44     ZLOGI("init commProvider");
45     thisProcessLabel_ = processLabel;
46     PipeInfo pi = {thisProcessLabel_, ""};
47     Status errCode = CommunicationProvider::GetInstance().Start(pi);
48     if (errCode != Status::SUCCESS) {
49         ZLOGE("commProvider_ Start Fail.");
50         return DBStatus::DB_ERROR;
51     }
52     return DBStatus::OK;
53 }
54 
Stop()55 DBStatus ProcessCommunicatorImpl::Stop()
56 {
57     PipeInfo pi = {thisProcessLabel_, ""};
58     Status errCode = CommunicationProvider::GetInstance().Stop(pi);
59     if (errCode != Status::SUCCESS) {
60         ZLOGE("commProvider_ Stop Fail.");
61         return DBStatus::DB_ERROR;
62     }
63     return DBStatus::OK;
64 }
65 
RegOnDeviceChange(const OnDeviceChange & callback)66 DBStatus ProcessCommunicatorImpl::RegOnDeviceChange(const OnDeviceChange &callback)
67 {
68     {
69         std::lock_guard<std::mutex> onDeviceChangeLockGard(onDeviceChangeMutex_);
70         onDeviceChangeHandler_ = callback;
71     }
72 
73     PipeInfo pi = {thisProcessLabel_, ""};
74     if (callback) {
75         Status errCode = DmAdapter::GetInstance().StartWatchDeviceChange(this, pi);
76         if (errCode != Status::SUCCESS) {
77             ZLOGE("commProvider_ StartWatchDeviceChange Fail.");
78             return DBStatus::DB_ERROR;
79         }
80     } else {
81         Status errCode = DmAdapter::GetInstance().StopWatchDeviceChange(this, pi);
82         if (errCode != Status::SUCCESS) {
83             ZLOGE("commProvider_ StopWatchDeviceChange Fail.");
84             return DBStatus::DB_ERROR;
85         }
86     }
87 
88     return DBStatus::OK;
89 }
90 
RegOnDataReceive(const OnDataReceive & callback)91 DBStatus ProcessCommunicatorImpl::RegOnDataReceive(const OnDataReceive &callback)
92 {
93     {
94         std::lock_guard<std::mutex> onDataReceiveLockGard(onDataReceiveMutex_);
95         onDataReceiveHandler_ = callback;
96     }
97 
98     PipeInfo pi = {thisProcessLabel_, ""};
99     if (callback) {
100         Status errCode = CommunicationProvider::GetInstance().StartWatchDataChange(this, pi);
101         if (errCode != Status::SUCCESS) {
102             ZLOGE("commProvider_ StartWatchDataChange Fail.");
103             return DBStatus::DB_ERROR;
104         }
105     } else {
106         Status errCode = CommunicationProvider::GetInstance().StopWatchDataChange(this, pi);
107         if (errCode != Status::SUCCESS) {
108             ZLOGE("commProvider_ StopWatchDataChange Fail.");
109             return DBStatus::DB_ERROR;
110         }
111     }
112     return DBStatus::OK;
113 }
114 
RegOnSendAble(const OnSendAble & sendAbleCallback)115 void ProcessCommunicatorImpl::RegOnSendAble(const OnSendAble &sendAbleCallback)
116 {
117     {
118         std::lock_guard<decltype(sessionMutex_)> lock(sessionMutex_);
119         sessionListener_ = sendAbleCallback;
120     }
121     if (!sendAbleCallback) {
122         ZLOGE("send callback is nullptr.");
123         return;
124     }
125     auto status = CommunicatorContext::GetInstance().RegSessionListener(this);
126     ZLOGD("reg status:%{public}d", status);
127 }
128 
SendData(const DeviceInfos & dstDevInfo,const uint8_t * data,uint32_t length)129 DBStatus ProcessCommunicatorImpl::SendData(const DeviceInfos &dstDevInfo, const uint8_t *data, uint32_t length)
130 {
131     uint32_t totalLength = 0;
132     return SendData(dstDevInfo, data, length, totalLength);
133 }
134 
SendData(const DeviceInfos & dstDevInfo,const uint8_t * data,uint32_t length,uint32_t totalLength)135 DBStatus ProcessCommunicatorImpl::SendData(const DeviceInfos &dstDevInfo, const uint8_t *data, uint32_t length,
136     uint32_t totalLength)
137 {
138     PipeInfo pi = {thisProcessLabel_, ""};
139     const DataInfo dataInfo = { const_cast<uint8_t *>(data), length};
140     DeviceId destination;
141     destination.deviceId = dstDevInfo.identifier;
142     auto [errCode, softBusErrCode] =
143         CommunicationProvider::GetInstance().SendData(pi, destination, dataInfo, totalLength);
144     if (errCode == Status::RATE_LIMIT) {
145         ZLOGD("commProvider_ opening session, status:%{public}d.", static_cast<int>(softBusErrCode));
146         return DBStatus::RATE_LIMIT;
147     }
148     if (errCode != Status::SUCCESS) {
149         ZLOGE("commProvider_ SendData Fail. code:%{public}d", softBusErrCode);
150         if (softBusErrCode == 0) {
151             return DBStatus::DB_ERROR;
152         }
153         return static_cast<DBStatus>(softBusErrCode);
154     }
155     return DBStatus::OK;
156 }
157 
GetMtuSize()158 uint32_t ProcessCommunicatorImpl::GetMtuSize()
159 {
160     return MTU_SIZE;
161 }
162 
GetMtuSize(const DeviceInfos & devInfo)163 uint32_t ProcessCommunicatorImpl::GetMtuSize(const DeviceInfos &devInfo)
164 {
165     return SoftBusAdapter::GetInstance()->GetMtuSize({ devInfo.identifier });
166 }
167 
GetTimeout(const DeviceInfos & devInfo)168 uint32_t ProcessCommunicatorImpl::GetTimeout(const DeviceInfos &devInfo)
169 {
170     return SoftBusAdapter::GetInstance()->GetTimeout({ devInfo.identifier });
171 }
172 
GetLocalDeviceInfos()173 DeviceInfos ProcessCommunicatorImpl::GetLocalDeviceInfos()
174 {
175     DeviceInfos localDevInfos;
176     DeviceInfo devInfo = DmAdapter::GetInstance().GetLocalDevice();
177     localDevInfos.identifier = devInfo.uuid;
178     return localDevInfos;
179 }
180 
GetRemoteOnlineDeviceInfosList()181 std::vector<DeviceInfos> ProcessCommunicatorImpl::GetRemoteOnlineDeviceInfosList()
182 {
183     std::vector<DeviceInfos> remoteDevInfos;
184     std::vector<DeviceInfo> devInfoVec = DmAdapter::GetInstance().GetRemoteDevices();
185     for (auto const &entry : devInfoVec) {
186         DeviceInfos remoteDev;
187         remoteDev.identifier = entry.uuid;
188         remoteDevInfos.push_back(remoteDev);
189     }
190     return remoteDevInfos;
191 }
192 
IsSameProcessLabelStartedOnPeerDevice(const DeviceInfos & peerDevInfo)193 bool ProcessCommunicatorImpl::IsSameProcessLabelStartedOnPeerDevice(const DeviceInfos &peerDevInfo)
194 {
195     PipeInfo pi = {thisProcessLabel_, ""};
196     DeviceId di = {peerDevInfo.identifier};
197     return CommunicationProvider::GetInstance().IsSameStartedOnPeer(pi, di);
198 }
199 
OnMessage(const DeviceInfo & info,const uint8_t * ptr,const int size,const PipeInfo & pipeInfo) const200 void ProcessCommunicatorImpl::OnMessage(const DeviceInfo &info, const uint8_t *ptr, const int size,
201                                         __attribute__((unused)) const PipeInfo &pipeInfo) const
202 {
203     std::lock_guard<std::mutex> onDataReceiveLockGuard(onDataReceiveMutex_);
204     if (onDataReceiveHandler_ == nullptr) {
205         ZLOGE("onDataReceiveHandler_ invalid.");
206         return;
207     }
208     DeviceInfos devInfo;
209     devInfo.identifier = info.uuid;
210     onDataReceiveHandler_(devInfo, ptr, static_cast<uint32_t>(size));
211 }
212 
OnDeviceChanged(const DeviceInfo & info,const DeviceChangeType & type) const213 void ProcessCommunicatorImpl::OnDeviceChanged(const DeviceInfo &info, const DeviceChangeType &type) const
214 {
215     if (type == DeviceChangeType::DEVICE_ONREADY || info.uuid == DmAdapter::CLOUD_DEVICE_UUID) {
216         return;
217     }
218     std::lock_guard<std::mutex> onDeviceChangeLockGuard(onDeviceChangeMutex_);
219     if (onDeviceChangeHandler_ == nullptr) {
220         ZLOGE("onDeviceChangeHandler_ invalid.");
221         return;
222     }
223     DeviceInfos devInfo;
224     devInfo.identifier = info.uuid;
225     onDeviceChangeHandler_(devInfo, (type == DeviceChangeType::DEVICE_ONLINE));
226 }
227 
OnSessionReady(const DeviceInfo & info,int32_t errCode) const228 void ProcessCommunicatorImpl::OnSessionReady(const DeviceInfo &info, int32_t errCode) const
229 {
230     std::lock_guard<decltype(sessionMutex_)> lock(sessionMutex_);
231     if (sessionListener_ == nullptr) {
232         return;
233     }
234     DeviceInfos devInfos;
235     devInfos.identifier = info.uuid;
236     sessionListener_(devInfos, errCode);
237 }
238 
GetExtendHeaderHandle(const ExtendInfo & info)239 std::shared_ptr<ExtendHeaderHandle> ProcessCommunicatorImpl::GetExtendHeaderHandle(const ExtendInfo &info)
240 {
241     if (routeHeadHandlerCreator_ != nullptr) {
242         return routeHeadHandlerCreator_(info);
243     }
244     return {};
245 }
246 
CheckAndGetDataHeadInfo(const uint8_t * data,uint32_t dataLen,uint32_t & headLen,std::vector<std::string> & users)247 DBStatus ProcessCommunicatorImpl::CheckAndGetDataHeadInfo(
248     const uint8_t *data, uint32_t dataLen, uint32_t &headLen, std::vector<std::string> &users)
249 {
250     ZLOGD("begin");
251     if (routeHeadHandlerCreator_ == nullptr) {
252         ZLOGE("header handler creator not registered");
253         return DBStatus::DB_ERROR;
254     }
255     auto handler = routeHeadHandlerCreator_({});
256     if (handler == nullptr) {
257         ZLOGE("failed to get header handler");
258         return DBStatus::DB_ERROR;
259     }
260     auto ret = handler->ParseHeadData(data, dataLen, headLen, users);
261     if (!ret) {
262         ZLOGD("illegal head format");
263         return DBStatus::INVALID_FORMAT;
264     }
265     if (users.empty()) {
266         ZLOGW("no valid user");
267         return DBStatus::NO_PERMISSION;
268     }
269     ZLOGD("ok, result:%{public}zu, user:%{public}s", users.size(), users.front().c_str());
270     return DBStatus::OK;
271 }
272 } // namespace AppDistributedKv
273 } // namespace OHOS