blob: 02bdba67ad600a2281966a801ba4686a51241e7b [file] [log] [blame]
/*
*
* Copyright (c) 2021 Project CHIP Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <cstdarg>
#include <memory>
#include <type_traits>
#include <app/BufferedReadCallback.h>
#include <app/ChunkedWriteCallback.h>
#include <app/DeviceProxy.h>
#include <app/ReadClient.h>
#include <app/WriteClient.h>
#include <lib/support/CodeUtils.h>
#include <cstdio>
#include <lib/support/logging/CHIPLogging.h>
using namespace chip;
using namespace chip::app;
using PyObject = void;
namespace chip {
namespace python {
struct __attribute__((packed)) AttributePath
{
chip::EndpointId endpointId;
chip::ClusterId clusterId;
chip::AttributeId attributeId;
chip::DataVersion dataVersion;
};
struct __attribute__((packed)) EventPath
{
chip::EndpointId endpointId;
chip::ClusterId clusterId;
chip::EventId eventId;
};
struct __attribute__((packed)) DataVersionFilter
{
chip::EndpointId endpointId;
chip::ClusterId clusterId;
chip::DataVersion dataVersion;
};
using OnReadAttributeDataCallback = void (*)(PyObject * appContext, chip::DataVersion version, chip::EndpointId endpointId,
chip::ClusterId clusterId, chip::AttributeId attributeId,
std::underlying_type_t<Protocols::InteractionModel::Status> imstatus, uint8_t * data,
uint32_t dataLen);
using OnReadEventDataCallback = void (*)(PyObject * appContext, chip::EndpointId endpointId, chip::ClusterId clusterId,
chip::EventId eventId, chip::EventNumber eventNumber, uint8_t priority, uint64_t timestamp,
uint8_t timestampType, uint8_t * data, uint32_t dataLen);
using OnSubscriptionEstablishedCallback = void (*)(PyObject * appContext, uint64_t subscriptionId);
using OnReadErrorCallback = void (*)(PyObject * appContext, uint32_t chiperror);
using OnReadDoneCallback = void (*)(PyObject * appContext);
using OnReportBeginCallback = void (*)(PyObject * appContext);
using OnReportEndCallback = void (*)(PyObject * appContext);
OnReadAttributeDataCallback gOnReadAttributeDataCallback = nullptr;
OnReadEventDataCallback gOnReadEventDataCallback = nullptr;
OnSubscriptionEstablishedCallback gOnSubscriptionEstablishedCallback = nullptr;
OnReadErrorCallback gOnReadErrorCallback = nullptr;
OnReadDoneCallback gOnReadDoneCallback = nullptr;
OnReportBeginCallback gOnReportBeginCallback = nullptr;
OnReportBeginCallback gOnReportEndCallback = nullptr;
class ReadClientCallback : public ReadClient::Callback
{
public:
ReadClientCallback(PyObject * appContext) : mBufferedReadCallback(*this), mAppContext(appContext) {}
app::BufferedReadCallback * GetBufferedReadCallback() { return &mBufferedReadCallback; }
void OnAttributeData(const ConcreteDataAttributePath & aPath, TLV::TLVReader * apData, const StatusIB & aStatus) override
{
//
// We shouldn't be getting list item operations in the provided path since that should be handled by the buffered read
// callback. If we do, that's a bug.
//
VerifyOrDie(!aPath.IsListItemOperation());
size_t bufferLen = (apData == nullptr ? 0 : apData->GetRemainingLength() + apData->GetLengthRead());
std::unique_ptr<uint8_t[]> buffer = std::unique_ptr<uint8_t[]>(apData == nullptr ? nullptr : new uint8_t[bufferLen]);
uint32_t size = 0;
// When the apData is nullptr, means we did not receive a valid attribute data from server, status will be some error
// status.
if (apData != nullptr)
{
// The TLVReader's read head is not pointing to the first element in the container instead of the container itself, use
// a TLVWriter to get a TLV with a normalized TLV buffer (Wrapped with a anonymous tag, no extra "end of container" tag
// at the end.)
TLV::TLVWriter writer;
writer.Init(buffer.get(), bufferLen);
CHIP_ERROR err = writer.CopyElement(TLV::AnonymousTag(), *apData);
if (err != CHIP_NO_ERROR)
{
this->OnError(err);
return;
}
size = writer.GetLengthWritten();
}
gOnReadAttributeDataCallback(mAppContext, aPath.mDataVersion, aPath.mEndpointId, aPath.mClusterId, aPath.mAttributeId,
to_underlying(aStatus.mStatus), buffer.get(), size);
}
void OnSubscriptionEstablished(uint64_t aSubscriptionId) override
{
gOnSubscriptionEstablishedCallback(mAppContext, aSubscriptionId);
}
void OnEventData(const EventHeader & aEventHeader, TLV::TLVReader * apData, const StatusIB * apStatus) override
{
uint8_t buffer[CHIP_CONFIG_DEFAULT_UDP_MTU_SIZE];
uint32_t size = 0;
CHIP_ERROR err = CHIP_NO_ERROR;
// When the apData is nullptr, means we did not receive a valid event data from server, status will be some error
// status.
if (apData != nullptr)
{
// The TLVReader's read head is not pointing to the first element in the container instead of the container itself, use
// a TLVWriter to get a TLV with a normalized TLV buffer (Wrapped with a anonymous tag, no extra "end of container" tag
// at the end.)
TLV::TLVWriter writer;
writer.Init(buffer);
err = writer.CopyElement(TLV::AnonymousTag(), *apData);
if (err != CHIP_NO_ERROR)
{
this->OnError(err);
return;
}
size = writer.GetLengthWritten();
}
else
{
err = CHIP_ERROR_INCORRECT_STATE;
this->OnError(err);
}
gOnReadEventDataCallback(mAppContext, aEventHeader.mPath.mEndpointId, aEventHeader.mPath.mClusterId,
aEventHeader.mPath.mEventId, aEventHeader.mEventNumber, to_underlying(aEventHeader.mPriorityLevel),
aEventHeader.mTimestamp.mValue, to_underlying(aEventHeader.mTimestamp.mType), buffer, size);
}
void OnError(CHIP_ERROR aError) override { gOnReadErrorCallback(mAppContext, aError.AsInteger()); }
void OnReportBegin() override { gOnReportBeginCallback(mAppContext); }
void OnDeallocatePaths(chip::app::ReadPrepareParams && aReadPrepareParams) override
{
if (aReadPrepareParams.mpAttributePathParamsList != nullptr)
{
delete[] aReadPrepareParams.mpAttributePathParamsList;
}
if (aReadPrepareParams.mpEventPathParamsList != nullptr)
{
delete[] aReadPrepareParams.mpEventPathParamsList;
}
if (aReadPrepareParams.mpDataVersionFilterList != nullptr)
{
delete[] aReadPrepareParams.mpDataVersionFilterList;
}
}
void OnReportEnd() override { gOnReportEndCallback(mAppContext); }
void OnDone() override
{
gOnReadDoneCallback(mAppContext);
delete this;
};
void AdoptReadClient(std::unique_ptr<ReadClient> apReadClient) { mReadClient = std::move(apReadClient); }
private:
BufferedReadCallback mBufferedReadCallback;
PyObject * mAppContext;
std::unique_ptr<ReadClient> mReadClient;
};
extern "C" {
struct __attribute__((packed)) PyReadAttributeParams
{
uint32_t minInterval; // MinInterval in subscription request
uint32_t maxInterval; // MaxInterval in subscription request
bool isSubscription;
bool isFabricFiltered;
};
// Encodes n attribute write requests, follows 3 * n arguments, in the (AttributeWritePath*=void *, uint8_t*, size_t) order.
chip::ChipError::StorageType pychip_WriteClient_WriteAttributes(void * appContext, DeviceProxy * device,
uint16_t timedWriteTimeoutMs, size_t n, ...);
chip::ChipError::StorageType pychip_ReadClient_ReadAttributes(void * appContext, ReadClient ** pReadClient,
ReadClientCallback ** pCallback, DeviceProxy * device,
uint8_t * readParamsBuf, size_t n, size_t total, ...);
}
using OnWriteResponseCallback = void (*)(PyObject * appContext, chip::EndpointId endpointId, chip::ClusterId clusterId,
chip::AttributeId attributeId,
std::underlying_type_t<Protocols::InteractionModel::Status> imstatus);
using OnWriteErrorCallback = void (*)(PyObject * appContext, uint32_t chiperror);
using OnWriteDoneCallback = void (*)(PyObject * appContext);
OnWriteResponseCallback gOnWriteResponseCallback = nullptr;
OnWriteErrorCallback gOnWriteErrorCallback = nullptr;
OnWriteDoneCallback gOnWriteDoneCallback = nullptr;
class WriteClientCallback : public WriteClient::Callback
{
public:
WriteClientCallback(PyObject * appContext) : mCallback(this), mAppContext(appContext) {}
WriteClient::Callback * GetChunkedCallback() { return &mCallback; }
void OnResponse(const WriteClient * apWriteClient, const ConcreteDataAttributePath & aPath, app::StatusIB aStatus) override
{
gOnWriteResponseCallback(mAppContext, aPath.mEndpointId, aPath.mClusterId, aPath.mAttributeId,
to_underlying(aStatus.mStatus));
}
void OnError(const WriteClient * apWriteClient, CHIP_ERROR aProtocolError) override
{
gOnWriteErrorCallback(mAppContext, aProtocolError.AsInteger());
}
void OnDone(WriteClient * apWriteClient) override
{
gOnWriteDoneCallback(mAppContext);
delete apWriteClient;
delete this;
};
private:
ChunkedWriteCallback mCallback;
PyObject * mAppContext = nullptr;
};
} // namespace python
} // namespace chip
using namespace chip::python;
extern "C" {
void pychip_WriteClient_InitCallbacks(OnWriteResponseCallback onWriteResponseCallback, OnWriteErrorCallback onWriteErrorCallback,
OnWriteDoneCallback onWriteDoneCallback)
{
gOnWriteResponseCallback = onWriteResponseCallback;
gOnWriteErrorCallback = onWriteErrorCallback;
gOnWriteDoneCallback = onWriteDoneCallback;
}
void pychip_ReadClient_InitCallbacks(OnReadAttributeDataCallback onReadAttributeDataCallback,
OnReadEventDataCallback onReadEventDataCallback,
OnSubscriptionEstablishedCallback onSubscriptionEstablishedCallback,
OnReadErrorCallback onReadErrorCallback, OnReadDoneCallback onReadDoneCallback,
OnReportBeginCallback onReportBeginCallback, OnReportEndCallback onReportEndCallback)
{
gOnReadAttributeDataCallback = onReadAttributeDataCallback;
gOnReadEventDataCallback = onReadEventDataCallback;
gOnSubscriptionEstablishedCallback = onSubscriptionEstablishedCallback;
gOnReadErrorCallback = onReadErrorCallback;
gOnReadDoneCallback = onReadDoneCallback;
gOnReportBeginCallback = onReportBeginCallback;
gOnReportEndCallback = onReportEndCallback;
}
chip::ChipError::StorageType pychip_WriteClient_WriteAttributes(void * appContext, DeviceProxy * device,
uint16_t timedWriteTimeoutMs, size_t n, ...)
{
CHIP_ERROR err = CHIP_NO_ERROR;
std::unique_ptr<WriteClientCallback> callback = std::make_unique<WriteClientCallback>(appContext);
std::unique_ptr<WriteClient> client = std::make_unique<WriteClient>(
app::InteractionModelEngine::GetInstance()->GetExchangeManager(), callback->GetChunkedCallback(),
timedWriteTimeoutMs != 0 ? Optional<uint16_t>(timedWriteTimeoutMs) : Optional<uint16_t>::Missing());
va_list args;
va_start(args, n);
VerifyOrExit(device != nullptr && device->GetSecureSession().HasValue(), err = CHIP_ERROR_INCORRECT_STATE);
{
for (size_t i = 0; i < n; i++)
{
void * path = va_arg(args, void *);
void * tlv = va_arg(args, void *);
int length = va_arg(args, int);
python::AttributePath pathObj;
memcpy(&pathObj, path, sizeof(python::AttributePath));
uint8_t * tlvBuffer = reinterpret_cast<uint8_t *>(tlv);
TLV::TLVReader reader;
reader.Init(tlvBuffer, static_cast<uint32_t>(length));
reader.Next();
SuccessOrExit(
err = client->PutPreencodedAttribute(
chip::app::ConcreteDataAttributePath(pathObj.endpointId, pathObj.clusterId, pathObj.attributeId), reader));
}
}
SuccessOrExit(err = client->SendWriteRequest(device->GetSecureSession().Value()));
client.release();
callback.release();
exit:
va_end(args);
return err.AsInteger();
}
void pychip_ReadClient_Abort(ReadClient * apReadClient, ReadClientCallback * apCallback)
{
VerifyOrDie(apReadClient != nullptr);
VerifyOrDie(apCallback != nullptr);
delete apCallback;
}
chip::ChipError::StorageType pychip_ReadClient_ReadAttributes(void * appContext, ReadClient ** pReadClient,
ReadClientCallback ** pCallback, DeviceProxy * device,
uint8_t * readParamsBuf, size_t n, size_t total, ...)
{
CHIP_ERROR err = CHIP_NO_ERROR;
PyReadAttributeParams pyParams = {};
// The readParamsBuf might be not aligned, using a memcpy to avoid some unexpected behaviors.
memcpy(&pyParams, readParamsBuf, sizeof(pyParams));
std::unique_ptr<ReadClientCallback> callback = std::make_unique<ReadClientCallback>(appContext);
size_t m = total - n;
va_list args;
va_start(args, total);
std::unique_ptr<AttributePathParams[]> readPaths(new AttributePathParams[n]);
std::unique_ptr<chip::app::DataVersionFilter[]> dataVersionFilters(new chip::app::DataVersionFilter[m]);
std::unique_ptr<ReadClient> readClient;
{
for (size_t i = 0; i < n; i++)
{
void * path = va_arg(args, void *);
python::AttributePath pathObj;
memcpy(&pathObj, path, sizeof(python::AttributePath));
readPaths[i] = AttributePathParams(pathObj.endpointId, pathObj.clusterId, pathObj.attributeId);
}
}
for (size_t j = 0; j < m; j++)
{
void * filter = va_arg(args, void *);
python::DataVersionFilter filterObj;
memcpy(&filterObj, filter, sizeof(python::DataVersionFilter));
dataVersionFilters[j] = chip::app::DataVersionFilter(filterObj.endpointId, filterObj.clusterId, filterObj.dataVersion);
}
Optional<SessionHandle> session = device->GetSecureSession();
VerifyOrExit(session.HasValue(), err = CHIP_ERROR_NOT_CONNECTED);
readClient = std::make_unique<ReadClient>(
InteractionModelEngine::GetInstance(), device->GetExchangeManager(), *callback->GetBufferedReadCallback(),
pyParams.isSubscription ? ReadClient::InteractionType::Subscribe : ReadClient::InteractionType::Read);
VerifyOrExit(readClient != nullptr, err = CHIP_ERROR_NO_MEMORY);
{
ReadPrepareParams params(session.Value());
params.mpAttributePathParamsList = readPaths.get();
params.mAttributePathParamsListSize = n;
if (m != 0)
{
params.mpDataVersionFilterList = dataVersionFilters.get();
params.mDataVersionFilterListSize = m;
}
params.mIsFabricFiltered = pyParams.isFabricFiltered;
if (pyParams.isSubscription)
{
params.mMinIntervalFloorSeconds = pyParams.minInterval;
params.mMaxIntervalCeilingSeconds = pyParams.maxInterval;
readPaths.release();
err = readClient->SendAutoResubscribeRequest(std::move(params));
SuccessOrExit(err);
}
else
{
err = readClient->SendRequest(params);
SuccessOrExit(err);
}
}
*pReadClient = readClient.get();
*pCallback = callback.get();
callback->AdoptReadClient(std::move(readClient));
callback.release();
exit:
va_end(args);
return err.AsInteger();
}
chip::ChipError::StorageType pychip_ReadClient_ReadEvents(void * appContext, ReadClient ** pReadClient,
ReadClientCallback ** pCallback, DeviceProxy * device,
uint8_t * readParamsBuf, size_t n, ...)
{
CHIP_ERROR err = CHIP_NO_ERROR;
PyReadAttributeParams pyParams = {};
memcpy(&pyParams, readParamsBuf, sizeof(pyParams));
std::unique_ptr<ReadClientCallback> callback = std::make_unique<ReadClientCallback>(appContext);
va_list args;
va_start(args, n);
std::unique_ptr<EventPathParams[]> readPaths(new EventPathParams[n]);
std::unique_ptr<ReadClient> readClient;
{
for (size_t i = 0; i < n; i++)
{
void * path = va_arg(args, void *);
python::EventPath pathObj;
memcpy(&pathObj, path, sizeof(python::EventPath));
readPaths[i] = EventPathParams(pathObj.endpointId, pathObj.clusterId, pathObj.eventId);
}
}
Optional<SessionHandle> session = device->GetSecureSession();
VerifyOrExit(session.HasValue(), err = CHIP_ERROR_NOT_CONNECTED);
readClient = std::make_unique<ReadClient>(InteractionModelEngine::GetInstance(), device->GetExchangeManager(), *callback.get(),
pyParams.isSubscription ? ReadClient::InteractionType::Subscribe
: ReadClient::InteractionType::Read);
{
ReadPrepareParams params(session.Value());
params.mpEventPathParamsList = readPaths.get();
params.mEventPathParamsListSize = n;
if (pyParams.isSubscription)
{
params.mMinIntervalFloorSeconds = pyParams.minInterval;
params.mMaxIntervalCeilingSeconds = pyParams.maxInterval;
readPaths.release();
err = readClient->SendAutoResubscribeRequest(std::move(params));
SuccessOrExit(err);
}
else
{
err = readClient->SendRequest(params);
SuccessOrExit(err);
}
}
*pReadClient = readClient.get();
*pCallback = callback.get();
callback->AdoptReadClient(std::move(readClient));
callback.release();
exit:
va_end(args);
return err.AsInteger();
}
}