/* obs-websocket Copyright (C) 2016-2019 Stéphane Lepin This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see */ #include #include "OBSRemoteProtocol.h" #include "../WSRequestHandler.h" #include "../rpc/RpcEvent.h" #include "../Utils.h" std::string OBSRemoteProtocol::processMessage(WSRequestHandler& requestHandler, std::string message) { std::string msgContainer(message); const char* msg = msgContainer.c_str(); OBSDataAutoRelease data = obs_data_create_from_json(msg); if (!data) { blog(LOG_ERROR, "invalid JSON payload received for '%s'", msg); return errorResponse(QString::Null(), "invalid JSON payload"); } if (!obs_data_has_user_value(data, "request-type") || !obs_data_has_user_value(data, "message-id")) { return errorResponse(QString::Null(), "missing request parameters"); } QString methodName = obs_data_get_string(data, "request-type"); QString messageId = obs_data_get_string(data, "message-id"); OBSDataAutoRelease params = obs_data_create(); obs_data_apply(params, data); obs_data_unset_user_value(params, "request-type"); obs_data_unset_user_value(params, "message-id"); RpcRequest request(messageId, methodName, params); RpcResponse response = requestHandler.processRequest(request); OBSData additionalFields = response.additionalFields(); switch (response.status()) { case RpcResponse::Status::Ok: return successResponse(messageId, additionalFields); case RpcResponse::Status::Error: return errorResponse(messageId, response.errorMessage(), additionalFields); } return std::string(); } std::string OBSRemoteProtocol::encodeEvent(const RpcEvent& event) { OBSDataAutoRelease eventData = obs_data_create(); OBSData additionalFields = event.additionalFields(); if (additionalFields) { obs_data_apply(eventData, additionalFields); } QString updateType = event.updateType(); obs_data_set_string(eventData, "update-type", updateType.toUtf8().constData()); if (obs_frontend_streaming_active()) { QString streamingTimecode = Utils::nsToTimestamp(event.streamTime()); obs_data_set_string(eventData, "stream-timecode", streamingTimecode.toUtf8().constData()); } if (obs_frontend_recording_active()) { QString recordingTimecode = Utils::nsToTimestamp(event.recordingTime()); obs_data_set_string(eventData, "rec-timecode", recordingTimecode.toUtf8().constData()); } return std::string(obs_data_get_json(eventData)); } std::string OBSRemoteProtocol::buildResponse(QString messageId, QString status, obs_data_t* fields) { OBSDataAutoRelease response = obs_data_create(); if (!messageId.isNull()) { obs_data_set_string(response, "message-id", messageId.toUtf8().constData()); } obs_data_set_string(response, "status", status.toUtf8().constData()); if (fields) { obs_data_apply(response, fields); } std::string responseString = obs_data_get_json(response); return responseString; } std::string OBSRemoteProtocol::successResponse(QString messageId, obs_data_t* fields) { return buildResponse(messageId, "ok", fields); } std::string OBSRemoteProtocol::errorResponse(QString messageId, QString errorMessage, obs_data_t* additionalFields) { OBSDataAutoRelease fields = obs_data_create(); if (additionalFields) { obs_data_apply(fields, additionalFields); } obs_data_set_string(fields, "error", errorMessage.toUtf8().constData()); return buildResponse(messageId, "error", fields); }