blob: 6aadad83676f7b9cc572b701b720c9d59ce42b67 [file] [log] [blame]
/*
*
* Copyright (c) 2021 Project CHIP Authors
* All rights reserved.
*
* 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 <app/AppBuildConfig.h>
#include <app/InteractionModelEngine.h>
#include <app/MessageDef/EventPath.h>
#include <app/WriteHandler.h>
#include <app/reporting/Engine.h>
namespace chip {
namespace app {
CHIP_ERROR WriteHandler::Init(InteractionModelDelegate * apDelegate)
{
VerifyOrReturnError(apDelegate != nullptr, CHIP_ERROR_INVALID_ARGUMENT);
VerifyOrReturnError(mpExchangeCtx == nullptr, CHIP_ERROR_INCORRECT_STATE);
mpExchangeCtx = nullptr;
mpDelegate = apDelegate;
System::PacketBufferHandle packet = System::PacketBufferHandle::New(chip::app::kMaxSecureSduLengthBytes);
VerifyOrReturnError(!packet.IsNull(), CHIP_ERROR_NO_MEMORY);
mMessageWriter.Init(std::move(packet));
ReturnErrorOnFailure(mWriteResponseBuilder.Init(&mMessageWriter));
AttributeStatusList::Builder attributeStatusListBuilder = mWriteResponseBuilder.CreateAttributeStatusListBuilder();
ReturnErrorOnFailure(attributeStatusListBuilder.GetError());
MoveToState(State::Initialized);
return CHIP_NO_ERROR;
}
void WriteHandler::Shutdown()
{
VerifyOrReturn(mState != State::Uninitialized);
mMessageWriter.Reset();
ClearExistingExchangeContext();
mpDelegate = nullptr;
ClearState();
}
void WriteHandler::ClearExistingExchangeContext()
{
if (mpExchangeCtx != nullptr)
{
mpExchangeCtx->Close();
mpExchangeCtx = nullptr;
}
}
CHIP_ERROR WriteHandler::OnWriteRequest(Messaging::ExchangeContext * apExchangeContext, System::PacketBufferHandle aPayload)
{
CHIP_ERROR err = CHIP_NO_ERROR;
mpExchangeCtx = apExchangeContext;
err = ProcessWriteRequest(std::move(aPayload));
SuccessOrExit(err);
err = SendWriteResponse();
exit:
ChipLogFunctError(err);
Shutdown();
return err;
}
CHIP_ERROR WriteHandler::FinalizeMessage(System::PacketBufferHandle & packet)
{
CHIP_ERROR err = CHIP_NO_ERROR;
AttributeStatusList::Builder attributeStatusList;
VerifyOrExit(mState == State::AddAttributeStatusCode, err = CHIP_ERROR_INCORRECT_STATE);
attributeStatusList = mWriteResponseBuilder.GetAttributeStatusListBuilder().EndOfAttributeStatusList();
err = attributeStatusList.GetError();
SuccessOrExit(err);
mWriteResponseBuilder.EndOfWriteResponse();
err = mWriteResponseBuilder.GetError();
SuccessOrExit(err);
err = mMessageWriter.Finalize(&packet);
SuccessOrExit(err);
exit:
ChipLogFunctError(err);
return err;
}
CHIP_ERROR WriteHandler::SendWriteResponse()
{
CHIP_ERROR err = CHIP_NO_ERROR;
System::PacketBufferHandle packet;
VerifyOrExit(mState == State::AddAttributeStatusCode, err = CHIP_ERROR_INCORRECT_STATE);
err = FinalizeMessage(packet);
SuccessOrExit(err);
VerifyOrExit(mpExchangeCtx != nullptr, err = CHIP_ERROR_INCORRECT_STATE);
err = mpExchangeCtx->SendMessage(Protocols::InteractionModel::MsgType::WriteResponse, std::move(packet));
SuccessOrExit(err);
MoveToState(State::Sending);
exit:
ChipLogFunctError(err);
return err;
}
CHIP_ERROR WriteHandler::ProcessAttributeDataList(TLV::TLVReader & aAttributeDataListReader)
{
CHIP_ERROR err = CHIP_NO_ERROR;
while (CHIP_NO_ERROR == (err = aAttributeDataListReader.Next()))
{
chip::TLV::TLVReader dataReader;
AttributeDataElement::Parser element;
AttributePath::Parser attributePath;
ClusterInfo clusterInfo;
TLV::TLVReader reader = aAttributeDataListReader;
err = element.Init(reader);
SuccessOrExit(err);
err = element.GetAttributePath(&attributePath);
SuccessOrExit(err);
err = attributePath.GetNodeId(&(clusterInfo.mNodeId));
SuccessOrExit(err);
err = attributePath.GetEndpointId(&(clusterInfo.mEndpointId));
SuccessOrExit(err);
err = attributePath.GetClusterId(&(clusterInfo.mClusterId));
SuccessOrExit(err);
err = attributePath.GetFieldId(&(clusterInfo.mFieldId));
if (CHIP_NO_ERROR == err)
{
clusterInfo.mFlags.Set(ClusterInfo::Flags::kFieldIdValid);
}
else if (CHIP_END_OF_TLV == err)
{
err = CHIP_NO_ERROR;
}
SuccessOrExit(err);
err = attributePath.GetListIndex(&(clusterInfo.mListIndex));
if (CHIP_NO_ERROR == err)
{
VerifyOrExit(clusterInfo.mFlags.Has(ClusterInfo::Flags::kFieldIdValid), err = CHIP_ERROR_IM_MALFORMED_ATTRIBUTE_PATH);
clusterInfo.mFlags.Set(ClusterInfo::Flags::kListIndexValid);
}
err = element.GetData(&dataReader);
SuccessOrExit(err);
err = WriteSingleClusterData(clusterInfo, dataReader, this);
SuccessOrExit(err);
}
if (CHIP_END_OF_TLV == err)
{
err = CHIP_NO_ERROR;
}
exit:
ChipLogFunctError(err);
return err;
}
CHIP_ERROR WriteHandler::ProcessWriteRequest(System::PacketBufferHandle && aPayload)
{
CHIP_ERROR err = CHIP_NO_ERROR;
System::PacketBufferTLVReader reader;
WriteRequest::Parser writeRequestParser;
AttributeDataList::Parser attributeDataListParser;
TLV::TLVReader attributeDataListReader;
bool needSuppressResponse = false;
reader.Init(std::move(aPayload));
err = reader.Next();
SuccessOrExit(err);
err = writeRequestParser.Init(reader);
SuccessOrExit(err);
#if CHIP_CONFIG_IM_ENABLE_SCHEMA_CHECK
err = writeRequestParser.CheckSchemaValidity();
SuccessOrExit(err);
#endif
err = writeRequestParser.GetSuppressResponse(&needSuppressResponse);
if (err == CHIP_END_OF_TLV)
{
err = CHIP_NO_ERROR;
}
SuccessOrExit(err);
err = writeRequestParser.GetAttributeDataList(&attributeDataListParser);
SuccessOrExit(err);
attributeDataListParser.GetReader(&attributeDataListReader);
err = ProcessAttributeDataList(attributeDataListReader);
exit:
ChipLogFunctError(err);
return err;
}
CHIP_ERROR WriteHandler::ConstructAttributePath(const AttributePathParams & aAttributePathParams,
AttributeStatusElement::Builder aAttributeStatusElement)
{
AttributePath::Builder attributePath = aAttributeStatusElement.CreateAttributePathBuilder();
if (aAttributePathParams.mFlags.Has(AttributePathParams::Flags::kFieldIdValid))
{
attributePath.FieldId(aAttributePathParams.mFieldId);
}
if (aAttributePathParams.mFlags.Has(AttributePathParams::Flags::kListIndexValid))
{
attributePath.ListIndex(aAttributePathParams.mListIndex);
}
attributePath.NodeId(aAttributePathParams.mNodeId)
.ClusterId(aAttributePathParams.mClusterId)
.EndpointId(aAttributePathParams.mEndpointId)
.EndOfAttributePath();
return attributePath.GetError();
}
CHIP_ERROR WriteHandler::AddAttributeStatusCode(const AttributePathParams & aAttributePathParams,
const Protocols::SecureChannel::GeneralStatusCode aGeneralCode,
const Protocols::Id aProtocolId,
const Protocols::InteractionModel::ProtocolCode aProtocolCode)
{
CHIP_ERROR err = CHIP_NO_ERROR;
StatusElement::Builder statusElementBuilder;
AttributeStatusElement::Builder attributeStatusElement =
mWriteResponseBuilder.GetAttributeStatusListBuilder().CreateAttributeStatusBuilder();
err = attributeStatusElement.GetError();
SuccessOrExit(err);
err = ConstructAttributePath(aAttributePathParams, attributeStatusElement);
SuccessOrExit(err);
statusElementBuilder = attributeStatusElement.CreateStatusElementBuilder();
statusElementBuilder
.EncodeStatusElement(aGeneralCode, aProtocolId.ToFullyQualifiedSpecForm(),
Protocols::InteractionModel::ToUint16(aProtocolCode))
.EndOfStatusElement();
err = statusElementBuilder.GetError();
SuccessOrExit(err);
attributeStatusElement.EndOfAttributeStatusElement();
err = attributeStatusElement.GetError();
SuccessOrExit(err);
MoveToState(State::AddAttributeStatusCode);
exit:
ChipLogFunctError(err);
return err;
}
const char * WriteHandler::GetStateStr() const
{
#if CHIP_DETAIL_LOGGING
switch (mState)
{
case State::Uninitialized:
return "Uninitialized";
case State::Initialized:
return "Initialized";
case State::AddAttributeStatusCode:
return "AddAttributeStatusCode";
case State::Sending:
return "Sending";
}
#endif // CHIP_DETAIL_LOGGING
return "N/A";
}
void WriteHandler::MoveToState(const State aTargetState)
{
mState = aTargetState;
ChipLogDetail(DataManagement, "IM RH moving to [%s]", GetStateStr());
}
void WriteHandler::ClearState()
{
MoveToState(State::Uninitialized);
}
} // namespace app
} // namespace chip