mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-13 19:49:51 +01:00
feat: add /warn
command (#5474)
This commit is contained in:
parent
c980162656
commit
c01bfcfffe
|
@ -14,6 +14,7 @@
|
||||||
- Minor: Moderators can now see when users are warned. (#5441)
|
- Minor: Moderators can now see when users are warned. (#5441)
|
||||||
- Minor: Added support for Brave & google-chrome-stable browsers. (#5452)
|
- Minor: Added support for Brave & google-chrome-stable browsers. (#5452)
|
||||||
- Minor: Added drop indicator line while dragging in tables. (#5256)
|
- Minor: Added drop indicator line while dragging in tables. (#5256)
|
||||||
|
- Minor: Added `/warn <username> <reason>` command for mods. This prevents the user from chatting until they acknowledge the warning. (#5474)
|
||||||
- Minor: Introduce HTTP API for plugins. (#5383)
|
- Minor: Introduce HTTP API for plugins. (#5383)
|
||||||
- Bugfix: Fixed tab move animation occasionally failing to start after closing a tab. (#5426)
|
- Bugfix: Fixed tab move animation occasionally failing to start after closing a tab. (#5426)
|
||||||
- Bugfix: If a network request errors with 200 OK, Qt's error code is now reported instead of the HTTP status. (#5378)
|
- Bugfix: If a network request errors with 200 OK, Qt's error code is now reported instead of the HTTP status. (#5378)
|
||||||
|
|
|
@ -326,6 +326,16 @@ public:
|
||||||
(FailureCallback<HelixBanUserError, QString> failureCallback)),
|
(FailureCallback<HelixBanUserError, QString> failureCallback)),
|
||||||
(override)); // /timeout, /ban
|
(override)); // /timeout, /ban
|
||||||
|
|
||||||
|
// /warn
|
||||||
|
// The extra parenthesis around the failure callback is because its type
|
||||||
|
// contains a comma
|
||||||
|
MOCK_METHOD(
|
||||||
|
void, warnUser,
|
||||||
|
(QString broadcasterID, QString moderatorID, QString userID,
|
||||||
|
QString reason, ResultCallback<> successCallback,
|
||||||
|
(FailureCallback<HelixWarnUserError, QString> failureCallback)),
|
||||||
|
(override)); // /warn
|
||||||
|
|
||||||
// /w
|
// /w
|
||||||
// The extra parenthesis around the failure callback is because its type
|
// The extra parenthesis around the failure callback is because its type
|
||||||
// contains a comma
|
// contains a comma
|
||||||
|
|
|
@ -107,6 +107,8 @@ set(SOURCE_FILES
|
||||||
controllers/commands/builtin/twitch/UpdateChannel.hpp
|
controllers/commands/builtin/twitch/UpdateChannel.hpp
|
||||||
controllers/commands/builtin/twitch/UpdateColor.cpp
|
controllers/commands/builtin/twitch/UpdateColor.cpp
|
||||||
controllers/commands/builtin/twitch/UpdateColor.hpp
|
controllers/commands/builtin/twitch/UpdateColor.hpp
|
||||||
|
controllers/commands/builtin/twitch/Warn.cpp
|
||||||
|
controllers/commands/builtin/twitch/Warn.hpp
|
||||||
controllers/commands/common/ChannelAction.cpp
|
controllers/commands/common/ChannelAction.cpp
|
||||||
controllers/commands/common/ChannelAction.hpp
|
controllers/commands/common/ChannelAction.hpp
|
||||||
controllers/commands/CommandContext.hpp
|
controllers/commands/CommandContext.hpp
|
||||||
|
|
|
@ -26,6 +26,7 @@
|
||||||
#include "controllers/commands/builtin/twitch/Unban.hpp"
|
#include "controllers/commands/builtin/twitch/Unban.hpp"
|
||||||
#include "controllers/commands/builtin/twitch/UpdateChannel.hpp"
|
#include "controllers/commands/builtin/twitch/UpdateChannel.hpp"
|
||||||
#include "controllers/commands/builtin/twitch/UpdateColor.hpp"
|
#include "controllers/commands/builtin/twitch/UpdateColor.hpp"
|
||||||
|
#include "controllers/commands/builtin/twitch/Warn.hpp"
|
||||||
#include "controllers/commands/Command.hpp"
|
#include "controllers/commands/Command.hpp"
|
||||||
#include "controllers/commands/CommandContext.hpp"
|
#include "controllers/commands/CommandContext.hpp"
|
||||||
#include "controllers/commands/CommandModel.hpp"
|
#include "controllers/commands/CommandModel.hpp"
|
||||||
|
@ -439,6 +440,8 @@ void CommandController::initialize(Settings &, const Paths &paths)
|
||||||
this->registerCommand("/ban", &commands::sendBan);
|
this->registerCommand("/ban", &commands::sendBan);
|
||||||
this->registerCommand("/banid", &commands::sendBanById);
|
this->registerCommand("/banid", &commands::sendBanById);
|
||||||
|
|
||||||
|
this->registerCommand("/warn", &commands::sendWarn);
|
||||||
|
|
||||||
for (const auto &cmd : TWITCH_WHISPER_COMMANDS)
|
for (const auto &cmd : TWITCH_WHISPER_COMMANDS)
|
||||||
{
|
{
|
||||||
this->registerCommand(cmd, &commands::sendWhisper);
|
this->registerCommand(cmd, &commands::sendWhisper);
|
||||||
|
|
199
src/controllers/commands/builtin/twitch/Warn.cpp
Normal file
199
src/controllers/commands/builtin/twitch/Warn.cpp
Normal file
|
@ -0,0 +1,199 @@
|
||||||
|
#include "controllers/commands/builtin/twitch/Warn.hpp"
|
||||||
|
|
||||||
|
#include "Application.hpp"
|
||||||
|
#include "common/QLogging.hpp"
|
||||||
|
#include "controllers/accounts/AccountController.hpp"
|
||||||
|
#include "controllers/commands/CommandContext.hpp"
|
||||||
|
#include "controllers/commands/common/ChannelAction.hpp"
|
||||||
|
#include "messages/MessageBuilder.hpp"
|
||||||
|
#include "providers/twitch/api/Helix.hpp"
|
||||||
|
#include "providers/twitch/TwitchAccount.hpp"
|
||||||
|
#include "providers/twitch/TwitchChannel.hpp"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
using namespace chatterino;
|
||||||
|
|
||||||
|
void warnUserByID(const ChannelPtr &channel, const QString &channelID,
|
||||||
|
const QString &sourceUserID, const QString &targetUserID,
|
||||||
|
const QString &reason, const QString &displayName)
|
||||||
|
{
|
||||||
|
using Error = HelixWarnUserError;
|
||||||
|
|
||||||
|
getHelix()->warnUser(
|
||||||
|
channelID, sourceUserID, targetUserID, reason,
|
||||||
|
[] {
|
||||||
|
// No response for warns, they're emitted over pubsub instead
|
||||||
|
},
|
||||||
|
[channel, displayName](auto error, auto message) {
|
||||||
|
QString errorMessage = QString("Failed to warn user - ");
|
||||||
|
switch (error)
|
||||||
|
{
|
||||||
|
case Error::ConflictingOperation: {
|
||||||
|
errorMessage += "There was a conflicting warn operation on "
|
||||||
|
"this user. Please try again.";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Error::Forwarded: {
|
||||||
|
errorMessage += message;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Error::Ratelimited: {
|
||||||
|
errorMessage += "You are being ratelimited by Twitch. Try "
|
||||||
|
"again in a few seconds.";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Error::CannotWarnUser: {
|
||||||
|
errorMessage +=
|
||||||
|
QString("You cannot warn %1.").arg(displayName);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Error::UserMissingScope: {
|
||||||
|
// TODO(pajlada): Phrase MISSING_REQUIRED_SCOPE
|
||||||
|
errorMessage += "Missing required scope. "
|
||||||
|
"Re-login with your "
|
||||||
|
"account and try again.";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Error::UserNotAuthorized: {
|
||||||
|
// TODO(pajlada): Phrase MISSING_PERMISSION
|
||||||
|
errorMessage += "You don't have permission to "
|
||||||
|
"perform that action.";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Error::Unknown: {
|
||||||
|
errorMessage += "An unknown error has occurred.";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
channel->addMessage(makeSystemMessage(errorMessage));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
namespace chatterino::commands {
|
||||||
|
|
||||||
|
QString sendWarn(const CommandContext &ctx)
|
||||||
|
{
|
||||||
|
const auto command = QStringLiteral("/warn");
|
||||||
|
const auto usage = QStringLiteral(
|
||||||
|
R"(Usage: "/warn [options...] <username> <reason>" - Warn a user via their username. Reason is required and will be shown to the target user and other moderators. Options: --channel <channel> to override which channel the warn takes place in (can be specified multiple times).)");
|
||||||
|
const auto actions = parseChannelAction(ctx, command, usage, false, true);
|
||||||
|
|
||||||
|
if (!actions.has_value())
|
||||||
|
{
|
||||||
|
if (ctx.channel != nullptr)
|
||||||
|
{
|
||||||
|
ctx.channel->addMessage(makeSystemMessage(actions.error()));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
qCWarning(chatterinoCommands)
|
||||||
|
<< "Error parsing command:" << actions.error();
|
||||||
|
}
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(!actions.value().empty());
|
||||||
|
|
||||||
|
auto currentUser = getIApp()->getAccounts()->twitch.getCurrent();
|
||||||
|
if (currentUser->isAnon())
|
||||||
|
{
|
||||||
|
ctx.channel->addMessage(
|
||||||
|
makeSystemMessage("You must be logged in to warn someone!"));
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const auto &action : actions.value())
|
||||||
|
{
|
||||||
|
const auto &reason = action.reason;
|
||||||
|
if (reason.isEmpty())
|
||||||
|
{
|
||||||
|
ctx.channel->addMessage(
|
||||||
|
makeSystemMessage("Failed to warn, you must specify a reason"));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
QStringList userLoginsToFetch;
|
||||||
|
QStringList userIDs;
|
||||||
|
if (action.target.id.isEmpty())
|
||||||
|
{
|
||||||
|
assert(!action.target.login.isEmpty() &&
|
||||||
|
"Warn Action target username AND user ID may not be "
|
||||||
|
"empty at the same time");
|
||||||
|
userLoginsToFetch.append(action.target.login);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// For hydration
|
||||||
|
userIDs.append(action.target.id);
|
||||||
|
}
|
||||||
|
if (action.channel.id.isEmpty())
|
||||||
|
{
|
||||||
|
assert(!action.channel.login.isEmpty() &&
|
||||||
|
"Warn Action channel username AND user ID may not be "
|
||||||
|
"empty at the same time");
|
||||||
|
userLoginsToFetch.append(action.channel.login);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// For hydration
|
||||||
|
userIDs.append(action.channel.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!userLoginsToFetch.isEmpty())
|
||||||
|
{
|
||||||
|
// At least 1 user ID needs to be resolved before we can take action
|
||||||
|
// userIDs is filled up with the data we already have to hydrate the action channel & action target
|
||||||
|
getHelix()->fetchUsers(
|
||||||
|
userIDs, userLoginsToFetch,
|
||||||
|
[channel{ctx.channel}, actionChannel{action.channel},
|
||||||
|
actionTarget{action.target}, currentUser, reason,
|
||||||
|
userLoginsToFetch](const auto &users) mutable {
|
||||||
|
if (!actionChannel.hydrateFrom(users))
|
||||||
|
{
|
||||||
|
channel->addMessage(makeSystemMessage(
|
||||||
|
QString("Failed to warn, bad channel name: %1")
|
||||||
|
.arg(actionChannel.login)));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!actionTarget.hydrateFrom(users))
|
||||||
|
{
|
||||||
|
channel->addMessage(makeSystemMessage(
|
||||||
|
QString("Failed to warn, bad target name: %1")
|
||||||
|
.arg(actionTarget.login)));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
warnUserByID(channel, actionChannel.id,
|
||||||
|
currentUser->getUserId(), actionTarget.id,
|
||||||
|
reason, actionTarget.displayName);
|
||||||
|
},
|
||||||
|
[channel{ctx.channel}, userLoginsToFetch] {
|
||||||
|
channel->addMessage(makeSystemMessage(
|
||||||
|
QString("Failed to warn, bad username(s): %1")
|
||||||
|
.arg(userLoginsToFetch.join(", "))));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// If both IDs are available, we do no hydration & just use the id as the display name
|
||||||
|
warnUserByID(ctx.channel, action.channel.id,
|
||||||
|
currentUser->getUserId(), action.target.id, reason,
|
||||||
|
action.target.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace chatterino::commands
|
16
src/controllers/commands/builtin/twitch/Warn.hpp
Normal file
16
src/controllers/commands/builtin/twitch/Warn.hpp
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
class QString;
|
||||||
|
|
||||||
|
namespace chatterino {
|
||||||
|
|
||||||
|
struct CommandContext;
|
||||||
|
|
||||||
|
} // namespace chatterino
|
||||||
|
|
||||||
|
namespace chatterino::commands {
|
||||||
|
|
||||||
|
/// /warn
|
||||||
|
QString sendWarn(const CommandContext &ctx);
|
||||||
|
|
||||||
|
} // namespace chatterino::commands
|
|
@ -77,6 +77,7 @@ static const QStringList TWITCH_DEFAULT_COMMANDS{
|
||||||
"delete",
|
"delete",
|
||||||
"announce",
|
"announce",
|
||||||
"requests",
|
"requests",
|
||||||
|
"warn",
|
||||||
};
|
};
|
||||||
|
|
||||||
static const QStringList TWITCH_WHISPER_COMMANDS{"/w", ".w"};
|
static const QStringList TWITCH_WHISPER_COMMANDS{"/w", ".w"};
|
||||||
|
|
|
@ -1463,16 +1463,6 @@ void Helix::removeChannelVIP(
|
||||||
.execute();
|
.execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
void Helix::unbanUser(
|
void Helix::unbanUser(
|
||||||
QString broadcasterID, QString moderatorID, QString userID,
|
QString broadcasterID, QString moderatorID, QString userID,
|
||||||
ResultCallback<> successCallback,
|
ResultCallback<> successCallback,
|
||||||
|
@ -1572,18 +1562,7 @@ void Helix::unbanUser(
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.execute();
|
.execute();
|
||||||
} // These changes are from the helix-command-migration/unban-untimeout branch
|
}
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
// These changes are from the helix-command-migration/unban-untimeout branch
|
|
||||||
|
|
||||||
void Helix::startRaid(
|
void Helix::startRaid(
|
||||||
QString fromBroadcasterID, QString toBroadcasterID,
|
QString fromBroadcasterID, QString toBroadcasterID,
|
||||||
|
@ -2266,6 +2245,107 @@ void Helix::banUser(QString broadcasterID, QString moderatorID, QString userID,
|
||||||
.execute();
|
.execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Warn a user
|
||||||
|
// https://dev.twitch.tv/docs/api/reference#warn-chat-user
|
||||||
|
void Helix::warnUser(
|
||||||
|
QString broadcasterID, QString moderatorID, QString userID, QString reason,
|
||||||
|
ResultCallback<> successCallback,
|
||||||
|
FailureCallback<HelixWarnUserError, QString> failureCallback)
|
||||||
|
{
|
||||||
|
using Error = HelixWarnUserError;
|
||||||
|
|
||||||
|
QUrlQuery urlQuery;
|
||||||
|
|
||||||
|
urlQuery.addQueryItem("broadcaster_id", broadcasterID);
|
||||||
|
urlQuery.addQueryItem("moderator_id", moderatorID);
|
||||||
|
|
||||||
|
QJsonObject payload;
|
||||||
|
{
|
||||||
|
QJsonObject data;
|
||||||
|
data["reason"] = reason;
|
||||||
|
data["user_id"] = userID;
|
||||||
|
|
||||||
|
payload["data"] = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
this->makePost("moderation/warnings", urlQuery)
|
||||||
|
.json(payload)
|
||||||
|
.onSuccess([successCallback](auto result) {
|
||||||
|
if (result.status() != 200)
|
||||||
|
{
|
||||||
|
qCWarning(chatterinoTwitch)
|
||||||
|
<< "Success result for warning a user was"
|
||||||
|
<< result.formatError() << "but we expected it to be 200";
|
||||||
|
}
|
||||||
|
// we don't care about the response
|
||||||
|
successCallback();
|
||||||
|
})
|
||||||
|
.onError([failureCallback](const auto &result) -> void {
|
||||||
|
if (!result.status())
|
||||||
|
{
|
||||||
|
failureCallback(Error::Unknown, result.formatError());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto obj = result.parseJson();
|
||||||
|
auto message = obj.value("message").toString();
|
||||||
|
|
||||||
|
switch (*result.status())
|
||||||
|
{
|
||||||
|
case 400: {
|
||||||
|
if (message.startsWith("The user specified in the user_id "
|
||||||
|
"field may not be warned",
|
||||||
|
Qt::CaseInsensitive))
|
||||||
|
{
|
||||||
|
failureCallback(Error::CannotWarnUser, message);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
failureCallback(Error::Forwarded, message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 401: {
|
||||||
|
if (message.startsWith("Missing scope",
|
||||||
|
Qt::CaseInsensitive))
|
||||||
|
{
|
||||||
|
failureCallback(Error::UserMissingScope, message);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
failureCallback(Error::Forwarded, message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 403: {
|
||||||
|
failureCallback(Error::UserNotAuthorized, message);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 409: {
|
||||||
|
failureCallback(Error::ConflictingOperation, message);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 429: {
|
||||||
|
failureCallback(Error::Ratelimited, message);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: {
|
||||||
|
qCDebug(chatterinoTwitch)
|
||||||
|
<< "Unhandled error warning user:"
|
||||||
|
<< result.formatError() << result.getData() << obj;
|
||||||
|
failureCallback(Error::Unknown, message);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.execute();
|
||||||
|
}
|
||||||
|
|
||||||
// https://dev.twitch.tv/docs/api/reference#send-whisper
|
// https://dev.twitch.tv/docs/api/reference#send-whisper
|
||||||
void Helix::sendWhisper(
|
void Helix::sendWhisper(
|
||||||
QString fromUserID, QString toUserID, QString message,
|
QString fromUserID, QString toUserID, QString message,
|
||||||
|
|
|
@ -621,6 +621,18 @@ enum class HelixBanUserError { // /timeout, /ban
|
||||||
Forwarded,
|
Forwarded,
|
||||||
}; // /timeout, /ban
|
}; // /timeout, /ban
|
||||||
|
|
||||||
|
enum class HelixWarnUserError { // /warn
|
||||||
|
Unknown,
|
||||||
|
UserMissingScope,
|
||||||
|
UserNotAuthorized,
|
||||||
|
Ratelimited,
|
||||||
|
ConflictingOperation,
|
||||||
|
CannotWarnUser,
|
||||||
|
|
||||||
|
// The error message is forwarded directly from the Twitch API
|
||||||
|
Forwarded,
|
||||||
|
}; // /warn
|
||||||
|
|
||||||
enum class HelixWhisperError { // /w
|
enum class HelixWhisperError { // /w
|
||||||
Unknown,
|
Unknown,
|
||||||
UserMissingScope,
|
UserMissingScope,
|
||||||
|
@ -1024,6 +1036,13 @@ public:
|
||||||
ResultCallback<> successCallback,
|
ResultCallback<> successCallback,
|
||||||
FailureCallback<HelixBanUserError, QString> failureCallback) = 0;
|
FailureCallback<HelixBanUserError, QString> failureCallback) = 0;
|
||||||
|
|
||||||
|
// Warn a user
|
||||||
|
// https://dev.twitch.tv/docs/api/reference#warn-chat-user
|
||||||
|
virtual void warnUser(
|
||||||
|
QString broadcasterID, QString moderatorID, QString userID,
|
||||||
|
QString reason, ResultCallback<> successCallback,
|
||||||
|
FailureCallback<HelixWarnUserError, QString> failureCallback) = 0;
|
||||||
|
|
||||||
// Send a whisper
|
// Send a whisper
|
||||||
// https://dev.twitch.tv/docs/api/reference#send-whisper
|
// https://dev.twitch.tv/docs/api/reference#send-whisper
|
||||||
virtual void sendWhisper(
|
virtual void sendWhisper(
|
||||||
|
@ -1346,6 +1365,13 @@ public:
|
||||||
ResultCallback<> successCallback,
|
ResultCallback<> successCallback,
|
||||||
FailureCallback<HelixBanUserError, QString> failureCallback) final;
|
FailureCallback<HelixBanUserError, QString> failureCallback) final;
|
||||||
|
|
||||||
|
// Warn a user
|
||||||
|
// https://dev.twitch.tv/docs/api/reference#warn-chat-user
|
||||||
|
void warnUser(
|
||||||
|
QString broadcasterID, QString moderatorID, QString userID,
|
||||||
|
QString reason, ResultCallback<> successCallback,
|
||||||
|
FailureCallback<HelixWarnUserError, QString> failureCallback) final;
|
||||||
|
|
||||||
// Send a whisper
|
// Send a whisper
|
||||||
// https://dev.twitch.tv/docs/api/reference#send-whisper
|
// https://dev.twitch.tv/docs/api/reference#send-whisper
|
||||||
void sendWhisper(
|
void sendWhisper(
|
||||||
|
|
|
@ -178,13 +178,21 @@ Used in:
|
||||||
|
|
||||||
- `controllers/commands/CommandController.cpp` to send Twitch native shoutout using "/shoutout <username>"
|
- `controllers/commands/CommandController.cpp` to send Twitch native shoutout using "/shoutout <username>"
|
||||||
|
|
||||||
|
### Warn Chat User
|
||||||
|
|
||||||
|
URL: https://dev.twitch.tv/docs/api/reference/#warn-chat-user
|
||||||
|
|
||||||
|
Used in:
|
||||||
|
|
||||||
|
- `controllers/commands/CommandController.cpp` to warn users via "/warn" command
|
||||||
|
|
||||||
## PubSub
|
## PubSub
|
||||||
|
|
||||||
### Whispers
|
### Whispers
|
||||||
|
|
||||||
We listen to the `whispers.<user_id>` PubSub topic to receive information about incoming whispers to the user
|
We listen to the `whispers.<user_id>` PubSub topic to receive information about incoming whispers to the user
|
||||||
|
|
||||||
No EventSub alternative available.
|
The EventSub alternative (`user.whisper.message`) is not yet implemented.
|
||||||
|
|
||||||
### Chat Moderator Actions
|
### Chat Moderator Actions
|
||||||
|
|
||||||
|
@ -192,25 +200,17 @@ We listen to the `chat_moderator_actions.<user_id>.<channel_id>` PubSub topic to
|
||||||
|
|
||||||
We listen to this topic in every channel the user is a moderator.
|
We listen to this topic in every channel the user is a moderator.
|
||||||
|
|
||||||
No complete EventSub alternative available yet. Some functionality can be pieced together but it would not be zero cost, causing the `max_total_cost` of 10 to cause issues.
|
We have not yet migrated to the EventSub equivalent topics:
|
||||||
|
|
||||||
- For showing bans & timeouts: `channel.ban`, but does not work with moderator token???
|
- For showing bans & timeouts => `channel.moderate`
|
||||||
- For showing unbans & untimeouts: `channel.unban`, but does not work with moderator token???
|
- For showing unbans & untimeouts => `channel.moderate`
|
||||||
- Clear/delete message: not in eventsub, and IRC doesn't tell us which mod performed the action
|
- Clear/delete message => `channel.moderate`
|
||||||
- Roomstate (slow(off), followers(off), r9k(off), emoteonly(off), subscribers(off)) => not in eventsub, and IRC doesn't tell us which mod performed the action
|
- Roomstate (slow(off), followers(off), r9k(off), emoteonly(off), subscribers(off)) => `channel.moderate`
|
||||||
- VIP added => not in eventsub, but not critical
|
- VIP/Moderator added/removed => `channel.moderate`
|
||||||
- VIP removed => not in eventsub, but not critical
|
- Raid started/cancelled => `channel.moderate`
|
||||||
- Moderator added => channel.moderator.add eventsub, but doesn't work with moderator token
|
- Add/delete permitted/blocked term => `channel.moderate` (or `automod.terms.update`)
|
||||||
- Moderator removed => channel.moderator.remove eventsub, but doesn't work with moderator token
|
- Modified automod properties => `automod.settings.update`
|
||||||
- Raid started => channel.raid eventsub, but cost=1 for moderator token
|
- Approve/deny unban request => `channel.moderate` (or `channel.unban_request.resolve`)
|
||||||
- Unraid => not in eventsub
|
|
||||||
- Add permitted term => not in eventsub
|
|
||||||
- Delete permitted term => not in eventsub
|
|
||||||
- Add blocked term => not in eventsub
|
|
||||||
- Delete blocked term => not in eventsub
|
|
||||||
- Modified automod properties => not in eventsub
|
|
||||||
- Approve unban request => cannot read moderator message in eventsub
|
|
||||||
- Deny unban request => not in eventsub
|
|
||||||
|
|
||||||
### AutoMod Queue
|
### AutoMod Queue
|
||||||
|
|
||||||
|
@ -218,7 +218,7 @@ We listen to the `automod-queue.<moderator_id>.<channel_id>` PubSub topic to rec
|
||||||
|
|
||||||
We listen to this topic in every channel the user is a moderator.
|
We listen to this topic in every channel the user is a moderator.
|
||||||
|
|
||||||
No EventSub alternative available yet.
|
The EventSub alternative (`automod.message.hold` and `automod.message.update`) is not yet implemented.
|
||||||
|
|
||||||
### Channel Point Rewards
|
### Channel Point Rewards
|
||||||
|
|
||||||
|
@ -230,4 +230,4 @@ The EventSub alternative requires broadcaster auth, which is not a feasible alte
|
||||||
|
|
||||||
We want to listen to the `low-trust-users` PubSub topic to receive information about messages from users who are marked as low-trust.
|
We want to listen to the `low-trust-users` PubSub topic to receive information about messages from users who are marked as low-trust.
|
||||||
|
|
||||||
There is no EventSub alternative available yet.
|
The EventSub alternative (`channel.suspicious_user.message` and `channel.suspicious_user.update`) is not yet implemented.
|
||||||
|
|
Loading…
Reference in a new issue