|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "sway/security.h"
|
|
|
|
#include "sway/commands.h"
|
|
|
|
#include "sway/config.h"
|
|
|
|
#include "ipc.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
static struct ipc_policy *current_policy = NULL;
|
|
|
|
|
|
|
|
struct cmd_results *cmd_ipc(int argc, char **argv) {
|
|
|
|
struct cmd_results *error = NULL;
|
|
|
|
if ((error = checkarg(argc, "ipc", EXPECTED_EQUAL_TO, 2))) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *program = argv[0];
|
|
|
|
|
|
|
|
if (config->reading && strcmp("{", argv[1]) != 0) {
|
|
|
|
return cmd_results_new(CMD_INVALID, "ipc",
|
|
|
|
"Expected '{' at start of IPC config definition.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!config->reading) {
|
|
|
|
return cmd_results_new(CMD_FAILURE, "ipc", "Can only be used in config file.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!current_config_path || strcmp(SYSCONFDIR "/sway/security", current_config_path) != 0) {
|
|
|
|
return cmd_results_new(CMD_INVALID, "permit",
|
|
|
|
"This command is only permitted to run from " SYSCONFDIR "/sway/security");
|
|
|
|
}
|
|
|
|
|
|
|
|
current_policy = alloc_ipc_policy(program);
|
|
|
|
list_add(config->ipc_policies, current_policy);
|
|
|
|
|
|
|
|
return cmd_results_new(CMD_BLOCK_IPC, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct cmd_results *cmd_ipc_events(int argc, char **argv) {
|
|
|
|
struct cmd_results *error = NULL;
|
|
|
|
if ((error = checkarg(argc, "events", EXPECTED_EQUAL_TO, 1))) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config->reading && strcmp("{", argv[0]) != 0) {
|
|
|
|
return cmd_results_new(CMD_INVALID, "events",
|
|
|
|
"Expected '{' at start of IPC event config definition.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!config->reading) {
|
|
|
|
return cmd_results_new(CMD_FAILURE, "events", "Can only be used in config file.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return cmd_results_new(CMD_BLOCK_IPC_EVENTS, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct cmd_results *cmd_ipc_cmd(int argc, char **argv) {
|
|
|
|
struct cmd_results *error = NULL;
|
|
|
|
if ((error = checkarg(argc, "ipc", EXPECTED_EQUAL_TO, 1))) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool enabled;
|
|
|
|
if (strcmp(argv[0], "enabled") == 0) {
|
|
|
|
enabled = true;
|
|
|
|
} else if (strcmp(argv[0], "disabled") == 0) {
|
|
|
|
enabled = false;
|
|
|
|
} else {
|
|
|
|
return cmd_results_new(CMD_INVALID, argv[-1],
|
|
|
|
"Argument must be one of 'enabled' or 'disabled'");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct {
|
|
|
|
char *name;
|
|
|
|
enum ipc_feature type;
|
|
|
|
} types[] = {
|
|
|
|
{ "*", IPC_FEATURE_ALL_COMMANDS },
|
|
|
|
{ "command", IPC_FEATURE_COMMAND },
|
|
|
|
{ "workspaces", IPC_FEATURE_GET_WORKSPACES },
|
|
|
|
{ "outputs", IPC_FEATURE_GET_OUTPUTS },
|
|
|
|
{ "tree", IPC_FEATURE_GET_TREE },
|
|
|
|
{ "marks", IPC_FEATURE_GET_MARKS },
|
|
|
|
{ "bar-config", IPC_FEATURE_GET_BAR_CONFIG },
|
|
|
|
{ "inputs", IPC_FEATURE_GET_INPUTS },
|
|
|
|
};
|
|
|
|
|
|
|
|
uint32_t type = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < sizeof(types) / sizeof(types[0]); ++i) {
|
|
|
|
if (strcmp(types[i].name, argv[-1]) == 0) {
|
|
|
|
type = types[i].type;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enabled) {
|
|
|
|
current_policy->features |= type;
|
|
|
|
sway_log(L_DEBUG, "Enabled IPC %s feature", argv[-1]);
|
|
|
|
} else {
|
|
|
|
current_policy->features &= ~type;
|
|
|
|
sway_log(L_DEBUG, "Disabled IPC %s feature", argv[-1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cmd_results_new(CMD_SUCCESS, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct cmd_results *cmd_ipc_event_cmd(int argc, char **argv) {
|
|
|
|
struct cmd_results *error = NULL;
|
|
|
|
if ((error = checkarg(argc, "ipc", EXPECTED_EQUAL_TO, 1))) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool enabled;
|
|
|
|
if (strcmp(argv[0], "enabled") == 0) {
|
|
|
|
enabled = true;
|
|
|
|
} else if (strcmp(argv[0], "disabled") == 0) {
|
|
|
|
enabled = false;
|
|
|
|
} else {
|
|
|
|
return cmd_results_new(CMD_INVALID, argv[-1],
|
|
|
|
"Argument must be one of 'enabled' or 'disabled'");
|
|
|
|
}
|
|
|
|
|
|
|
|
struct {
|
|
|
|
char *name;
|
|
|
|
enum ipc_feature type;
|
|
|
|
} types[] = {
|
|
|
|
{ "*", IPC_FEATURE_ALL_EVENTS },
|
|
|
|
{ "workspace", IPC_FEATURE_EVENT_WORKSPACE },
|
|
|
|
{ "output", IPC_FEATURE_EVENT_OUTPUT },
|
|
|
|
{ "mode", IPC_FEATURE_EVENT_MODE },
|
|
|
|
{ "window", IPC_FEATURE_EVENT_WINDOW },
|
|
|
|
{ "binding", IPC_FEATURE_EVENT_BINDING },
|
|
|
|
{ "input", IPC_FEATURE_EVENT_INPUT },
|
|
|
|
};
|
|
|
|
|
|
|
|
uint32_t type = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < sizeof(types) / sizeof(types[0]); ++i) {
|
|
|
|
if (strcmp(types[i].name, argv[-1]) == 0) {
|
|
|
|
type = types[i].type;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enabled) {
|
|
|
|
current_policy->features |= type;
|
|
|
|
sway_log(L_DEBUG, "Enabled IPC %s event", argv[-1]);
|
|
|
|
} else {
|
|
|
|
current_policy->features &= ~type;
|
|
|
|
sway_log(L_DEBUG, "Disabled IPC %s event", argv[-1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cmd_results_new(CMD_SUCCESS, NULL, NULL);
|
|
|
|
}
|