blob: c5c12212fea0fef1e34ac92a497d73eec0a8669d [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.
*/
/**
* @brief Contains Implementation of the ContentAppCommandDelegate
*/
#include "ContentAppCommandDelegate.h"
#include <app-common/zap-generated/cluster-objects.h>
#include <app/CommandHandlerInterface.h>
#include <app/util/config.h>
#include <jni.h>
#include <lib/core/DataModelTypes.h>
#include <lib/support/CHIPJNIError.h>
#include <lib/support/JniReferences.h>
#include <lib/support/JniTypeWrappers.h>
#include <lib/support/jsontlv/TlvJson.h>
#include <platform/PlatformManager.h>
#include <zap-generated/endpoint_config.h>
#include <string>
namespace chip {
namespace AppPlatform {
const std::string FAILURE_KEY = "PlatformError";
const std::string FAILURE_STATUS_KEY = "Status";
const std::string RESPONSE_STATUS_KEY = "Status";
bool isValidJson(const char * response)
{
Json::Reader reader;
Json::CharReaderBuilder readerBuilder;
std::string errors;
Json::Value value;
std::unique_ptr<Json::CharReader> testReader(readerBuilder.newCharReader());
if (!testReader->parse(response, response + std::strlen(response), &value, &errors))
{
ChipLogError(Zcl, "Failed to parse JSON: %s\n", errors.c_str());
return false;
}
// Validate and access JSON data safely
if (!value.isObject())
{
ChipLogError(Zcl, "Invalid JSON structure: not an object");
return false;
}
if (!reader.parse(response, value))
{
return false;
}
return true;
}
void ContentAppCommandDelegate::InvokeCommand(CommandHandlerInterface::HandlerContext & handlerContext)
{
if (handlerContext.mRequestPath.mEndpointId >= FIXED_ENDPOINT_COUNT)
{
DeviceLayer::StackUnlock unlock;
TLV::TLVReader readerForJson;
readerForJson.Init(handlerContext.mPayload);
CHIP_ERROR err = CHIP_NO_ERROR;
Json::Value json;
err = TlvToJson(readerForJson, json);
if (err != CHIP_NO_ERROR)
{
handlerContext.SetCommandHandled();
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath,
chip::Protocols::InteractionModel::Status::InvalidCommand);
return;
}
JNIEnv * env = JniReferences::GetInstance().GetEnvForCurrentThread();
Json::Value value = json["value"];
std::string payload = JsonToString(value);
UtfString jsonString(env, payload.c_str());
if (!mContentAppEndpointManager.HasValidObjectRef())
{
ChipLogProgress(Zcl, "mContentAppEndpointManager is not valid");
return;
}
jstring resp = static_cast<jstring>(env->CallObjectMethod(
mContentAppEndpointManager.ObjectRef(), mSendCommandMethod, static_cast<jint>(handlerContext.mRequestPath.mEndpointId),
static_cast<jlong>(handlerContext.mRequestPath.mClusterId), static_cast<jlong>(handlerContext.mRequestPath.mCommandId),
jsonString.jniValue()));
if (env->ExceptionCheck())
{
ChipLogError(Zcl, "Java exception in ContentAppCommandDelegate::sendCommand");
env->ExceptionDescribe();
env->ExceptionClear();
FormatResponseData(handlerContext, "{\"value\":{}}");
}
else
{
JniUtfString respStr(env, resp);
ChipLogProgress(Zcl, "ContentAppCommandDelegate::InvokeCommand got response %s", respStr.c_str());
if (isValidJson(respStr.c_str()))
{
FormatResponseData(handlerContext, respStr.c_str());
}
else
{
// return dummy value in case JSON is invalid
FormatResponseData(handlerContext, "{\"value\":{}}");
}
}
env->DeleteLocalRef(resp);
}
else
{
handlerContext.SetCommandNotHandled();
}
}
Status ContentAppCommandDelegate::InvokeCommand(EndpointId epId, ClusterId clusterId, CommandId commandId, std::string payload,
bool & commandHandled, Json::Value & value)
{
if (epId >= FIXED_ENDPOINT_COUNT)
{
JNIEnv * env = JniReferences::GetInstance().GetEnvForCurrentThread();
UtfString jsonString(env, payload.c_str());
ChipLogProgress(Zcl, "ContentAppCommandDelegate::InvokeCommand send command being called with payload %s", payload.c_str());
if (!mContentAppEndpointManager.HasValidObjectRef())
{
return chip::Protocols::InteractionModel::Status::Failure;
}
jstring resp =
(jstring) env->CallObjectMethod(mContentAppEndpointManager.ObjectRef(), mSendCommandMethod, static_cast<jint>(epId),
static_cast<jlong>(clusterId), static_cast<jlong>(commandId), jsonString.jniValue());
if (env->ExceptionCheck())
{
ChipLogError(Zcl, "Java exception in ContentAppCommandDelegate::sendCommand");
env->ExceptionDescribe();
env->ExceptionClear();
return chip::Protocols::InteractionModel::Status::Failure;
}
else
{
JniUtfString respStr(env, resp);
ChipLogProgress(Zcl, "ContentAppCommandDelegate::InvokeCommand got response %s", respStr.c_str());
Json::CharReaderBuilder readerBuilder;
std::string errors;
std::unique_ptr<Json::CharReader> testReader(readerBuilder.newCharReader());
if (!testReader->parse(respStr.c_str(), respStr.c_str() + std::strlen(respStr.c_str()), &value, &errors))
{
ChipLogError(Zcl, "Failed to parse JSON: %s\n", errors.c_str());
return chip::Protocols::InteractionModel::Status::Failure;
}
// Validate and access JSON data safely
if (!value.isObject())
{
ChipLogError(Zcl, "Invalid JSON structure: not an object");
return chip::Protocols::InteractionModel::Status::Failure;
}
Json::Reader reader;
if (!reader.parse(respStr.c_str(), value))
{
return chip::Protocols::InteractionModel::Status::Failure;
}
}
env->DeleteLocalRef(resp);
// Parse response here in case there is failure response.
// Return non-success error code to indicate to caller it should not parse response.
if (!value[FAILURE_KEY].empty())
{
value = value[FAILURE_KEY];
if (!value[FAILURE_STATUS_KEY].empty() && value[FAILURE_STATUS_KEY].isUInt())
{
return static_cast<Protocols::InteractionModel::Status>(value[FAILURE_STATUS_KEY].asUInt());
}
return chip::Protocols::InteractionModel::Status::Failure;
}
// Return success to indicate command has been sent, response returned and parsed successfully.
// Caller has to manually parse value input/output parameter to get response status/object.
return chip::Protocols::InteractionModel::Status::Success;
}
else
{
commandHandled = false;
return chip::Protocols::InteractionModel::Status::UnsupportedEndpoint;
}
}
void ContentAppCommandDelegate::FormatResponseData(CommandHandlerInterface::HandlerContext & handlerContext, const char * response)
{
handlerContext.SetCommandHandled();
Json::Reader reader;
Json::Value value;
if (!reader.parse(response, value))
{
return;
}
// handle errors from platform-app
if (!value[FAILURE_KEY].empty())
{
value = value[FAILURE_KEY];
if (!value[FAILURE_STATUS_KEY].empty() && value[FAILURE_STATUS_KEY].isUInt())
{
handlerContext.mCommandHandler.AddStatus(
handlerContext.mRequestPath, static_cast<Protocols::InteractionModel::Status>(value[FAILURE_STATUS_KEY].asUInt()));
return;
}
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, chip::Protocols::InteractionModel::Status::Failure);
return;
}
switch (handlerContext.mRequestPath.mClusterId)
{
case app::Clusters::ContentLauncher::Id: {
Status status;
LaunchResponseType launchResponse = FormatContentLauncherResponse(value, status);
if (status != chip::Protocols::InteractionModel::Status::Success)
{
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, status);
}
else
{
handlerContext.mCommandHandler.AddResponse(handlerContext.mRequestPath, launchResponse);
}
break;
}
case app::Clusters::TargetNavigator::Id: {
Status status;
NavigateTargetResponseType navigateTargetResponse = FormatNavigateTargetResponse(value, status);
if (status != chip::Protocols::InteractionModel::Status::Success)
{
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, status);
}
else
{
handlerContext.mCommandHandler.AddResponse(handlerContext.mRequestPath, navigateTargetResponse);
}
break;
}
case app::Clusters::MediaPlayback::Id: {
Status status;
PlaybackResponseType playbackResponse = FormatMediaPlaybackResponse(value, status);
if (status != chip::Protocols::InteractionModel::Status::Success)
{
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, status);
}
else
{
handlerContext.mCommandHandler.AddResponse(handlerContext.mRequestPath, playbackResponse);
}
break;
}
case app::Clusters::AccountLogin::Id: {
switch (handlerContext.mRequestPath.mCommandId)
{
case app::Clusters::AccountLogin::Commands::GetSetupPIN::Id: {
Status status;
GetSetupPINResponseType getSetupPINresponse = FormatGetSetupPINResponse(value, status);
if (status != chip::Protocols::InteractionModel::Status::Success)
{
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, status);
}
else
{
handlerContext.mCommandHandler.AddResponse(handlerContext.mRequestPath, getSetupPINresponse);
}
break;
}
case app::Clusters::AccountLogin::Commands::Login::Id: {
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, FormatStatusResponse(value));
break;
}
case app::Clusters::AccountLogin::Commands::Logout::Id: {
handlerContext.mCommandHandler.AddStatus(handlerContext.mRequestPath, FormatStatusResponse(value));
break;
}
default:
break;
}
break;
}
default:
handlerContext.SetCommandNotHandled();
}
}
LaunchResponseType ContentAppCommandDelegate::FormatContentLauncherResponse(Json::Value value, Status & status)
{
status = chip::Protocols::InteractionModel::Status::Success;
LaunchResponseType launchResponse;
std::string statusFieldId =
std::to_string(to_underlying(app::Clusters::ContentLauncher::Commands::LauncherResponse::Fields::kStatus));
if (value[statusFieldId].empty())
{
status = chip::Protocols::InteractionModel::Status::Failure;
return launchResponse;
}
else
{
launchResponse.status = static_cast<app::Clusters::ContentLauncher::StatusEnum>(value[statusFieldId].asInt());
std::string dataFieldId =
std::to_string(to_underlying(app::Clusters::ContentLauncher::Commands::LauncherResponse::Fields::kData));
if (!value[dataFieldId].empty())
{
launchResponse.data = chip::MakeOptional(CharSpan::fromCharString(value[dataFieldId].asCString()));
}
}
return launchResponse;
}
NavigateTargetResponseType ContentAppCommandDelegate::FormatNavigateTargetResponse(Json::Value value, Status & status)
{
status = chip::Protocols::InteractionModel::Status::Success;
NavigateTargetResponseType navigateTargetResponse;
std::string statusFieldId =
std::to_string(to_underlying(app::Clusters::TargetNavigator::Commands::NavigateTargetResponse::Fields::kStatus));
if (value[statusFieldId].empty())
{
status = chip::Protocols::InteractionModel::Status::Failure;
return navigateTargetResponse;
}
else
{
navigateTargetResponse.status = static_cast<app::Clusters::TargetNavigator::StatusEnum>(value[statusFieldId].asInt());
std::string dataFieldId =
std::to_string(to_underlying(app::Clusters::TargetNavigator::Commands::NavigateTargetResponse::Fields::kData));
if (!value[dataFieldId].empty())
{
navigateTargetResponse.data = chip::MakeOptional(CharSpan::fromCharString(value[dataFieldId].asCString()));
}
}
return navigateTargetResponse;
}
PlaybackResponseType ContentAppCommandDelegate::FormatMediaPlaybackResponse(Json::Value value, Status & status)
{
status = chip::Protocols::InteractionModel::Status::Success;
PlaybackResponseType playbackResponse;
std::string statusFieldId =
std::to_string(to_underlying(app::Clusters::MediaPlayback::Commands::PlaybackResponse::Fields::kStatus));
if (value[statusFieldId].empty())
{
status = chip::Protocols::InteractionModel::Status::Failure;
return playbackResponse;
}
else
{
playbackResponse.status = static_cast<app::Clusters::MediaPlayback::StatusEnum>(value[statusFieldId].asInt());
std::string dataFieldId =
std::to_string(to_underlying(app::Clusters::MediaPlayback::Commands::PlaybackResponse::Fields::kData));
if (!value[dataFieldId].empty())
{
playbackResponse.data = chip::MakeOptional(CharSpan::fromCharString(value[dataFieldId].asCString()));
}
}
return playbackResponse;
}
GetSetupPINResponseType ContentAppCommandDelegate::FormatGetSetupPINResponse(Json::Value value, Status & status)
{
status = chip::Protocols::InteractionModel::Status::Success;
GetSetupPINResponseType getSetupPINresponse;
std::string setupPINFieldId =
std::to_string(to_underlying(app::Clusters::AccountLogin::Commands::GetSetupPINResponse::Fields::kSetupPIN));
if (!value[setupPINFieldId].empty())
{
getSetupPINresponse.setupPIN = CharSpan::fromCharString(value[setupPINFieldId].asCString());
}
else
{
status = chip::Protocols::InteractionModel::Status::Failure;
}
return getSetupPINresponse;
}
Status ContentAppCommandDelegate::FormatStatusResponse(Json::Value value)
{
// check if JSON has "Status" key
if (!value[RESPONSE_STATUS_KEY].empty() && !value[RESPONSE_STATUS_KEY].isUInt())
{
return static_cast<Protocols::InteractionModel::Status>(value.asUInt());
}
else
{
return chip::Protocols::InteractionModel::Status::Failure;
}
}
} // namespace AppPlatform
} // namespace chip