mirror of
https://github.com/Palakis/obs-websocket.git
synced 2024-08-30 18:12:16 +00:00
418 lines
13 KiB
C++
418 lines
13 KiB
C++
#include "WSRequestHandler.h"
|
|
|
|
#include <QtCore/QByteArray>
|
|
#include <QtGui/QImageWriter>
|
|
|
|
#include "obs-websocket.h"
|
|
#include "Config.h"
|
|
#include "Utils.h"
|
|
#include "WSEvents.h"
|
|
|
|
#define CASE(x) case x: return #x;
|
|
const char *describe_output_format(int format) {
|
|
switch (format) {
|
|
default:
|
|
CASE(VIDEO_FORMAT_NONE)
|
|
CASE(VIDEO_FORMAT_I420)
|
|
CASE(VIDEO_FORMAT_NV12)
|
|
CASE(VIDEO_FORMAT_YVYU)
|
|
CASE(VIDEO_FORMAT_YUY2)
|
|
CASE(VIDEO_FORMAT_UYVY)
|
|
CASE(VIDEO_FORMAT_RGBA)
|
|
CASE(VIDEO_FORMAT_BGRA)
|
|
CASE(VIDEO_FORMAT_BGRX)
|
|
CASE(VIDEO_FORMAT_Y800)
|
|
CASE(VIDEO_FORMAT_I444)
|
|
}
|
|
}
|
|
|
|
const char *describe_color_space(int cs) {
|
|
switch (cs) {
|
|
default:
|
|
CASE(VIDEO_CS_DEFAULT)
|
|
CASE(VIDEO_CS_601)
|
|
CASE(VIDEO_CS_709)
|
|
}
|
|
}
|
|
|
|
const char *describe_color_range(int range) {
|
|
switch (range) {
|
|
default:
|
|
CASE(VIDEO_RANGE_DEFAULT)
|
|
CASE(VIDEO_RANGE_PARTIAL)
|
|
CASE(VIDEO_RANGE_FULL)
|
|
}
|
|
}
|
|
|
|
const char *describe_scale_type(int scale) {
|
|
switch (scale) {
|
|
default:
|
|
CASE(VIDEO_SCALE_DEFAULT)
|
|
CASE(VIDEO_SCALE_POINT)
|
|
CASE(VIDEO_SCALE_FAST_BILINEAR)
|
|
CASE(VIDEO_SCALE_BILINEAR)
|
|
CASE(VIDEO_SCALE_BICUBIC)
|
|
}
|
|
}
|
|
#undef CASE
|
|
|
|
/**
|
|
* Returns the latest version of the plugin and the API.
|
|
*
|
|
* @return {double} `version` OBSRemote compatible API version. Fixed to 1.1 for retrocompatibility.
|
|
* @return {String} `obs-websocket-version` obs-websocket plugin version.
|
|
* @return {String} `obs-studio-version` OBS Studio program version.
|
|
* @return {String} `available-requests` List of available request types, formatted as a comma-separated list string (e.g. : "Method1,Method2,Method3").
|
|
* @return {String} `supported-image-export-formats` List of supported formats for features that use image export (like the TakeSourceScreenshot request type) formatted as a comma-separated list string
|
|
*
|
|
* @api requests
|
|
* @name GetVersion
|
|
* @category general
|
|
* @since 0.3
|
|
*/
|
|
RpcResponse WSRequestHandler::GetVersion(const RpcRequest& request) {
|
|
QString obsVersion = Utils::OBSVersionString();
|
|
|
|
QList<QString> names = messageMap.keys();
|
|
QList<QByteArray> imageWriterFormats = QImageWriter::supportedImageFormats();
|
|
|
|
// (Palakis) OBS' data arrays only support object arrays, so I improvised.
|
|
QString requests;
|
|
names.sort(Qt::CaseInsensitive);
|
|
requests += names.takeFirst();
|
|
for (const QString& reqName : names) {
|
|
requests += ("," + reqName);
|
|
}
|
|
|
|
QString supportedImageExportFormats;
|
|
supportedImageExportFormats += QString::fromUtf8(imageWriterFormats.takeFirst());
|
|
for (const QByteArray& format : imageWriterFormats) {
|
|
supportedImageExportFormats += ("," + QString::fromUtf8(format));
|
|
}
|
|
|
|
OBSDataAutoRelease data = obs_data_create();
|
|
obs_data_set_double(data, "version", 1.1);
|
|
obs_data_set_string(data, "obs-websocket-version", OBS_WEBSOCKET_VERSION);
|
|
obs_data_set_string(data, "obs-studio-version", obsVersion.toUtf8());
|
|
obs_data_set_string(data, "available-requests", requests.toUtf8());
|
|
obs_data_set_string(data, "supported-image-export-formats", supportedImageExportFormats.toUtf8());
|
|
|
|
return request.success(data);
|
|
}
|
|
|
|
/**
|
|
* Tells the client if authentication is required. If so, returns authentication parameters `challenge`
|
|
* and `salt` (see "Authentication" for more information).
|
|
*
|
|
* @return {boolean} `authRequired` Indicates whether authentication is required.
|
|
* @return {String (optional)} `challenge`
|
|
* @return {String (optional)} `salt`
|
|
*
|
|
* @api requests
|
|
* @name GetAuthRequired
|
|
* @category general
|
|
* @since 0.3
|
|
*/
|
|
RpcResponse WSRequestHandler::GetAuthRequired(const RpcRequest& request) {
|
|
bool authRequired = GetConfig()->AuthRequired;
|
|
|
|
OBSDataAutoRelease data = obs_data_create();
|
|
obs_data_set_bool(data, "authRequired", authRequired);
|
|
|
|
if (authRequired) {
|
|
auto config = GetConfig();
|
|
obs_data_set_string(data, "challenge",
|
|
config->SessionChallenge.toUtf8());
|
|
obs_data_set_string(data, "salt",
|
|
config->Salt.toUtf8());
|
|
}
|
|
|
|
return request.success(data);
|
|
}
|
|
|
|
/**
|
|
* Attempt to authenticate the client to the server.
|
|
*
|
|
* @param {String} `auth` Response to the auth challenge (see "Authentication" for more information).
|
|
*
|
|
* @api requests
|
|
* @name Authenticate
|
|
* @category general
|
|
* @since 0.3
|
|
*/
|
|
RpcResponse WSRequestHandler::Authenticate(const RpcRequest& request) {
|
|
if (!request.hasField("auth")) {
|
|
return request.failed("missing request parameters");
|
|
}
|
|
|
|
if (_connProperties.isAuthenticated()) {
|
|
return request.failed("already authenticated");
|
|
}
|
|
|
|
QString auth = obs_data_get_string(request.parameters(), "auth");
|
|
if (auth.isEmpty()) {
|
|
return request.failed("auth not specified!");
|
|
}
|
|
|
|
if (GetConfig()->CheckAuth(auth) == false) {
|
|
return request.failed("Authentication Failed.");
|
|
}
|
|
|
|
_connProperties.setAuthenticated(true);
|
|
return request.success();
|
|
}
|
|
|
|
/**
|
|
* Enable/disable sending of the Heartbeat event
|
|
*
|
|
* @param {boolean} `enable` Starts/Stops emitting heartbeat messages
|
|
*
|
|
* @api requests
|
|
* @name SetHeartbeat
|
|
* @category general
|
|
* @since 4.3.0
|
|
* @deprecated Since 4.9.0. Please poll the appropriate data using requests. Will be removed in v5.0.0.
|
|
*/
|
|
RpcResponse WSRequestHandler::SetHeartbeat(const RpcRequest& request) {
|
|
if (!request.hasField("enable")) {
|
|
return request.failed("Heartbeat <enable> parameter missing");
|
|
}
|
|
|
|
auto events = GetEventsSystem();
|
|
events->HeartbeatIsActive = obs_data_get_bool(request.parameters(), "enable");
|
|
|
|
OBSDataAutoRelease response = obs_data_create();
|
|
obs_data_set_bool(response, "enable", events->HeartbeatIsActive);
|
|
|
|
return request.success(response);
|
|
}
|
|
|
|
/**
|
|
* Set the filename formatting string
|
|
*
|
|
* @param {String} `filename-formatting` Filename formatting string to set.
|
|
*
|
|
* @api requests
|
|
* @name SetFilenameFormatting
|
|
* @category general
|
|
* @since 4.3.0
|
|
*/
|
|
RpcResponse WSRequestHandler::SetFilenameFormatting(const RpcRequest& request) {
|
|
if (!request.hasField("filename-formatting")) {
|
|
return request.failed("<filename-formatting> parameter missing");
|
|
}
|
|
|
|
QString filenameFormatting = obs_data_get_string(request.parameters(), "filename-formatting");
|
|
if (filenameFormatting.isEmpty()) {
|
|
return request.failed("invalid request parameters");
|
|
}
|
|
|
|
Utils::SetFilenameFormatting(filenameFormatting.toUtf8());
|
|
|
|
return request.success();
|
|
}
|
|
|
|
/**
|
|
* Get the filename formatting string
|
|
*
|
|
* @return {String} `filename-formatting` Current filename formatting string.
|
|
*
|
|
* @api requests
|
|
* @name GetFilenameFormatting
|
|
* @category general
|
|
* @since 4.3.0
|
|
*/
|
|
RpcResponse WSRequestHandler::GetFilenameFormatting(const RpcRequest& request) {
|
|
OBSDataAutoRelease response = obs_data_create();
|
|
obs_data_set_string(response, "filename-formatting", Utils::GetFilenameFormatting());
|
|
|
|
return request.success(response);
|
|
}
|
|
|
|
/**
|
|
* Get OBS stats (almost the same info as provided in OBS' stats window)
|
|
*
|
|
* @return {OBSStats} `stats` [OBS stats](#obsstats)
|
|
*
|
|
* @api requests
|
|
* @name GetStats
|
|
* @category general
|
|
* @since 4.6.0
|
|
*/
|
|
RpcResponse WSRequestHandler::GetStats(const RpcRequest& request) {
|
|
OBSDataAutoRelease stats = GetEventsSystem()->GetStats();
|
|
|
|
OBSDataAutoRelease response = obs_data_create();
|
|
obs_data_set_obj(response, "stats", stats);
|
|
|
|
return request.success(response);
|
|
}
|
|
|
|
/**
|
|
* Broadcast custom message to all connected WebSocket clients
|
|
*
|
|
* @param {String} `realm` Identifier to be choosen by the client
|
|
* @param {Object} `data` User-defined data
|
|
*
|
|
* @api requests
|
|
* @name BroadcastCustomMessage
|
|
* @category general
|
|
* @since 4.7.0
|
|
*/
|
|
RpcResponse WSRequestHandler::BroadcastCustomMessage(const RpcRequest& request) {
|
|
if (!request.hasField("realm") || !request.hasField("data")) {
|
|
return request.failed("missing request parameters");
|
|
}
|
|
|
|
QString realm = obs_data_get_string(request.parameters(), "realm");
|
|
OBSDataAutoRelease data = obs_data_get_obj(request.parameters(), "data");
|
|
|
|
if (realm.isEmpty()) {
|
|
return request.failed("realm not specified!");
|
|
}
|
|
|
|
if (!data) {
|
|
return request.failed("data not specified!");
|
|
}
|
|
|
|
auto events = GetEventsSystem();
|
|
events->OnBroadcastCustomMessage(realm, data);
|
|
|
|
return request.success();
|
|
}
|
|
|
|
|
|
/**
|
|
* Get basic OBS video information
|
|
*
|
|
* @return {int} `baseWidth` Base (canvas) width
|
|
* @return {int} `baseHeight` Base (canvas) height
|
|
* @return {int} `outputWidth` Output width
|
|
* @return {int} `outputHeight` Output height
|
|
* @return {String} `scaleType` Scaling method used if output size differs from base size
|
|
* @return {double} `fps` Frames rendered per second
|
|
* @return {String} `videoFormat` Video color format
|
|
* @return {String} `colorSpace` Color space for YUV
|
|
* @return {String} `colorRange` Color range (full or partial)
|
|
*
|
|
* @api requests
|
|
* @name GetVideoInfo
|
|
* @category general
|
|
* @since 4.6.0
|
|
*/
|
|
RpcResponse WSRequestHandler::GetVideoInfo(const RpcRequest& request) {
|
|
obs_video_info ovi;
|
|
obs_get_video_info(&ovi);
|
|
|
|
OBSDataAutoRelease response = obs_data_create();
|
|
obs_data_set_int(response, "baseWidth", ovi.base_width);
|
|
obs_data_set_int(response, "baseHeight", ovi.base_height);
|
|
obs_data_set_int(response, "outputWidth", ovi.output_width);
|
|
obs_data_set_int(response, "outputHeight", ovi.output_height);
|
|
obs_data_set_double(response, "fps", (double)ovi.fps_num / ovi.fps_den);
|
|
obs_data_set_string(response, "videoFormat", describe_output_format(ovi.output_format));
|
|
obs_data_set_string(response, "colorSpace", describe_color_space(ovi.colorspace));
|
|
obs_data_set_string(response, "colorRange", describe_color_range(ovi.range));
|
|
obs_data_set_string(response, "scaleType", describe_scale_type(ovi.scale_type));
|
|
|
|
return request.success(response);
|
|
}
|
|
|
|
/**
|
|
* Open a projector window or create a projector on a monitor. Requires OBS v24.0.4 or newer.
|
|
*
|
|
* @param {String (Optional)} `type` Type of projector: `Preview` (default), `Source`, `Scene`, `StudioProgram`, or `Multiview` (case insensitive).
|
|
* @param {int (Optional)} `monitor` Monitor to open the projector on. If -1 or omitted, opens a window.
|
|
* @param {String (Optional)} `geometry` Size and position of the projector window (only if monitor is -1). Encoded in Base64 using [Qt's geometry encoding](https://doc.qt.io/qt-5/qwidget.html#saveGeometry). Corresponds to OBS's saved projectors.
|
|
* @param {String (Optional)} `name` Name of the source or scene to be displayed (ignored for other projector types).
|
|
*
|
|
* @api requests
|
|
* @name OpenProjector
|
|
* @category general
|
|
* @since 4.8.0
|
|
*/
|
|
RpcResponse WSRequestHandler::OpenProjector(const RpcRequest& request) {
|
|
const char* type = obs_data_get_string(request.parameters(), "type");
|
|
|
|
int monitor = -1;
|
|
if (request.hasField("monitor")) {
|
|
monitor = obs_data_get_int(request.parameters(), "monitor");
|
|
}
|
|
|
|
const char* geometry = obs_data_get_string(request.parameters(), "geometry");
|
|
const char* name = obs_data_get_string(request.parameters(), "name");
|
|
|
|
obs_frontend_open_projector(type, monitor, geometry, name);
|
|
return request.success();
|
|
}
|
|
|
|
/**
|
|
* Executes hotkey routine, identified by hotkey unique name
|
|
*
|
|
* @param {String} `hotkeyName` Unique name of the hotkey, as defined when registering the hotkey (e.g. "ReplayBuffer.Save")
|
|
*
|
|
* @api requests
|
|
* @name TriggerHotkeyByName
|
|
* @category general
|
|
* @since unreleased
|
|
*/
|
|
RpcResponse WSRequestHandler::TriggerHotkeyByName(const RpcRequest& request) {
|
|
const char* name = obs_data_get_string(request.parameters(), "hotkeyName");
|
|
|
|
obs_hotkey_t* hk = Utils::FindHotkeyByName(name);
|
|
if (!hk) {
|
|
return request.failed("hotkey not found");
|
|
}
|
|
obs_hotkey_trigger_routed_callback(obs_hotkey_get_id(hk), true);
|
|
return request.success();
|
|
}
|
|
|
|
/**
|
|
* Executes hotkey routine, identified by bound combination of keys. A single key combination might trigger multiple hotkey routines depending on user settings
|
|
*
|
|
* @param {String} `keyId` Main key identifier (e.g. `OBS_KEY_A` for key "A"). Available identifiers [here](https://github.com/obsproject/obs-studio/blob/master/libobs/obs-hotkeys.h)
|
|
* @param {Object (Optional)} `keyModifiers` Optional key modifiers object. False entries can be ommitted
|
|
* @param {boolean} `keyModifiers.shift` Trigger Shift Key
|
|
* @param {boolean} `keyModifiers.alt` Trigger Alt Key
|
|
* @param {boolean} `keyModifiers.control` Trigger Control (Ctrl) Key
|
|
* @param {boolean} `keyModifiers.command` Trigger Command Key (Mac)
|
|
*
|
|
* @api requests
|
|
* @name TriggerHotkeyBySequence
|
|
* @category general
|
|
* @since unreleased
|
|
*/
|
|
RpcResponse WSRequestHandler::TriggerHotkeyBySequence(const RpcRequest& request) {
|
|
if (!request.hasField("keyId")) {
|
|
return request.failed("missing request keyId parameter");
|
|
}
|
|
|
|
OBSDataAutoRelease data = obs_data_get_obj(request.parameters(), "keyModifiers");
|
|
|
|
obs_key_combination_t combo = {0};
|
|
uint32_t modifiers = 0;
|
|
if (obs_data_get_bool(data, "shift"))
|
|
modifiers |= INTERACT_SHIFT_KEY;
|
|
if (obs_data_get_bool(data, "control"))
|
|
modifiers |= INTERACT_CONTROL_KEY;
|
|
if (obs_data_get_bool(data, "alt"))
|
|
modifiers |= INTERACT_ALT_KEY;
|
|
if (obs_data_get_bool(data, "command"))
|
|
modifiers |= INTERACT_COMMAND_KEY;
|
|
|
|
combo.modifiers = modifiers;
|
|
combo.key = obs_key_from_name(obs_data_get_string(request.parameters(), "keyId"));
|
|
|
|
if (!modifiers
|
|
&& (combo.key == OBS_KEY_NONE || combo.key >= OBS_KEY_LAST_VALUE)) {
|
|
return request.failed("invalid key-modifier combination");
|
|
}
|
|
|
|
// Inject hotkey press-release sequence
|
|
obs_hotkey_inject_event(combo, false);
|
|
obs_hotkey_inject_event(combo, true);
|
|
obs_hotkey_inject_event(combo, false);
|
|
|
|
return request.success();
|
|
}
|