mirror of
https://github.com/Tha14/toxic.git
synced 2024-11-22 15:03:03 +01:00
Refactor peer-specific commands
This takes into account the possibility of multiple peers in the group using the same nick. The /whois command now lists all peers associated with the input name, and all commands that target a peer with a non-unique nick will require the public key to be used.
This commit is contained in:
parent
cd4a5e5fde
commit
f786b7ae6a
@ -88,11 +88,8 @@ void cmd_ignore(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[
|
|||||||
const char *nick = argv[1];
|
const char *nick = argv[1];
|
||||||
uint32_t peer_id;
|
uint32_t peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TOX_ERR_GROUP_TOGGLE_IGNORE err;
|
TOX_ERR_GROUP_TOGGLE_IGNORE err;
|
||||||
@ -128,11 +125,8 @@ void cmd_kick(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MA
|
|||||||
|
|
||||||
uint32_t target_peer_id;
|
uint32_t target_peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &target_peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &target_peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &target_peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TOX_ERR_GROUP_MOD_KICK_PEER err;
|
TOX_ERR_GROUP_MOD_KICK_PEER err;
|
||||||
@ -204,11 +198,8 @@ void cmd_mod(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[MAX
|
|||||||
const char *nick = argv[1];
|
const char *nick = argv[1];
|
||||||
uint32_t target_peer_id;
|
uint32_t target_peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &target_peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &target_peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &target_peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TOX_ERR_GROUP_SELF_QUERY s_err;
|
TOX_ERR_GROUP_SELF_QUERY s_err;
|
||||||
@ -260,11 +251,8 @@ void cmd_unmod(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[M
|
|||||||
const char *nick = argv[1];
|
const char *nick = argv[1];
|
||||||
uint32_t target_peer_id;
|
uint32_t target_peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &target_peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &target_peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &target_peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TOX_ERR_GROUP_SELF_QUERY s_err;
|
TOX_ERR_GROUP_SELF_QUERY s_err;
|
||||||
@ -504,11 +492,8 @@ void cmd_silence(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)
|
|||||||
const char *nick = argv[1];
|
const char *nick = argv[1];
|
||||||
uint32_t target_peer_id;
|
uint32_t target_peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &target_peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &target_peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &target_peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TOX_ERR_GROUP_SELF_QUERY s_err;
|
TOX_ERR_GROUP_SELF_QUERY s_err;
|
||||||
@ -560,11 +545,8 @@ void cmd_unsilence(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*arg
|
|||||||
const char *nick = argv[1];
|
const char *nick = argv[1];
|
||||||
uint32_t target_peer_id;
|
uint32_t target_peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &target_peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &target_peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &target_peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tox_group_peer_get_role(m, self->num, target_peer_id, NULL) != TOX_GROUP_ROLE_OBSERVER) {
|
if (tox_group_peer_get_role(m, self->num, target_peer_id, NULL) != TOX_GROUP_ROLE_OBSERVER) {
|
||||||
@ -700,11 +682,8 @@ void cmd_unignore(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv
|
|||||||
const char *nick = argv[1];
|
const char *nick = argv[1];
|
||||||
uint32_t peer_id;
|
uint32_t peer_id;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, nick, &peer_id) == -1) {
|
if (group_get_peer_id_of_identifier(self, nick, &peer_id) != 0) {
|
||||||
if (group_get_public_key_peer_id(self->num, nick, &peer_id) == -1) {
|
return;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TOX_ERR_GROUP_TOGGLE_IGNORE err;
|
TOX_ERR_GROUP_TOGGLE_IGNORE err;
|
||||||
@ -744,68 +723,80 @@ void cmd_whois(WINDOW *window, ToxWindow *self, Tox *m, int argc, char (*argv)[M
|
|||||||
}
|
}
|
||||||
|
|
||||||
const char *identifier = argv[1];
|
const char *identifier = argv[1];
|
||||||
uint32_t peer_id;
|
bool is_public_key = false;
|
||||||
|
|
||||||
if (group_get_nick_peer_id(self->num, identifier, &peer_id) == -1) {
|
for (size_t i = 0; i < chat->max_idx && !is_public_key; ++i) {
|
||||||
if (group_get_public_key_peer_id(self->num, identifier, &peer_id) == -1) {
|
uint32_t peer_id;
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
|
||||||
return;
|
if (group_get_public_key_peer_id(self->num, identifier, &peer_id) == 0) {
|
||||||
|
is_public_key = true;
|
||||||
|
} else {
|
||||||
|
GroupPeer *peer = &chat->peer_list[i];
|
||||||
|
|
||||||
|
if (!peer->active) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(identifier, peer->name) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
peer_id = peer->peer_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int peer_index = get_peer_index(self->num, peer_id);
|
||||||
|
|
||||||
|
if (peer_index < 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
GroupPeer *peer = &chat->peer_list[peer_index];
|
||||||
|
|
||||||
|
const char *status_str = "Online";
|
||||||
|
|
||||||
|
if (peer->status == TOX_USER_STATUS_BUSY) {
|
||||||
|
status_str = "Busy";
|
||||||
|
} else if (peer->status == TOX_USER_STATUS_AWAY) {
|
||||||
|
status_str = "Away";
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *role_str = "User";
|
||||||
|
|
||||||
|
if (peer->role == TOX_GROUP_ROLE_FOUNDER) {
|
||||||
|
role_str = "Founder";
|
||||||
|
} else if (peer->role == TOX_GROUP_ROLE_MODERATOR) {
|
||||||
|
role_str = "Moderator";
|
||||||
|
} else if (peer->role == TOX_GROUP_ROLE_OBSERVER) {
|
||||||
|
role_str = "Observer";
|
||||||
|
}
|
||||||
|
|
||||||
|
char last_seen_str[128];
|
||||||
|
get_elapsed_time_str_alt(last_seen_str, sizeof(last_seen_str),
|
||||||
|
get_unix_time() - peer->last_active);
|
||||||
|
|
||||||
|
char pk_string[TOX_GROUP_PEER_PUBLIC_KEY_SIZE * 2 + 1] = {0};
|
||||||
|
|
||||||
|
for (size_t i = 0; i < TOX_GROUP_PEER_PUBLIC_KEY_SIZE; ++i) {
|
||||||
|
char d[3];
|
||||||
|
snprintf(d, sizeof(d), "%02X", peer->public_key[i] & 0xff);
|
||||||
|
strcat(pk_string, d);
|
||||||
|
}
|
||||||
|
|
||||||
|
TOX_ERR_GROUP_PEER_QUERY conn_err;
|
||||||
|
Tox_Connection connection_type = tox_group_peer_get_connection_status(m, self->num, peer_id, &conn_err);
|
||||||
|
|
||||||
|
const char *connection_type_str = "Unknown";
|
||||||
|
|
||||||
|
if (conn_err == TOX_ERR_GROUP_PEER_QUERY_OK || connection_type != TOX_CONNECTION_NONE) {
|
||||||
|
connection_type_str = connection_type == TOX_CONNECTION_UDP ? "UDP" : "TCP";
|
||||||
|
}
|
||||||
|
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Public key: %s", pk_string);
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Name: %s", peer->name);
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Role: %s", role_str);
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Status: %s", status_str);
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Connection: %s", connection_type_str);
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Last active: %s", last_seen_str);
|
||||||
}
|
}
|
||||||
|
|
||||||
int peer_index = get_peer_index(self->num, peer_id);
|
|
||||||
|
|
||||||
if (peer_index < 0) {
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Failed to fetch peer index.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
GroupPeer *peer = &chat->peer_list[peer_index];
|
|
||||||
|
|
||||||
const char *status_str = "Online";
|
|
||||||
|
|
||||||
if (peer->status == TOX_USER_STATUS_BUSY) {
|
|
||||||
status_str = "Busy";
|
|
||||||
} else if (peer->status == TOX_USER_STATUS_AWAY) {
|
|
||||||
status_str = "Away";
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *role_str = "User";
|
|
||||||
|
|
||||||
if (peer->role == TOX_GROUP_ROLE_FOUNDER) {
|
|
||||||
role_str = "Founder";
|
|
||||||
} else if (peer->role == TOX_GROUP_ROLE_MODERATOR) {
|
|
||||||
role_str = "Moderator";
|
|
||||||
} else if (peer->role == TOX_GROUP_ROLE_OBSERVER) {
|
|
||||||
role_str = "Observer";
|
|
||||||
}
|
|
||||||
|
|
||||||
char last_seen_str[128];
|
|
||||||
get_elapsed_time_str_alt(last_seen_str, sizeof(last_seen_str),
|
|
||||||
get_unix_time() - peer->last_active);
|
|
||||||
|
|
||||||
char pk_string[TOX_GROUP_PEER_PUBLIC_KEY_SIZE * 2 + 1] = {0};
|
|
||||||
|
|
||||||
for (size_t i = 0; i < TOX_GROUP_PEER_PUBLIC_KEY_SIZE; ++i) {
|
|
||||||
char d[3];
|
|
||||||
snprintf(d, sizeof(d), "%02X", peer->public_key[i] & 0xff);
|
|
||||||
strcat(pk_string, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
TOX_ERR_GROUP_PEER_QUERY conn_err;
|
|
||||||
Tox_Connection connection_type = tox_group_peer_get_connection_status(m, self->num, peer_id, &conn_err);
|
|
||||||
|
|
||||||
const char *connection_type_str = "Unknown";
|
|
||||||
|
|
||||||
if (conn_err == TOX_ERR_GROUP_PEER_QUERY_OK || connection_type != TOX_CONNECTION_NONE) {
|
|
||||||
connection_type_str = connection_type == TOX_CONNECTION_UDP ? "UDP" : "TCP";
|
|
||||||
}
|
|
||||||
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Public key: %s", pk_string);
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Name: %s", peer->name);
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Role: %s", role_str);
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Status: %s", status_str);
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Connection: %s", connection_type_str);
|
|
||||||
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Last active: %s", last_seen_str);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -465,12 +465,13 @@ static void sort_peerlist(uint32_t groupnumber)
|
|||||||
qsort(chat->peer_list, chat->max_idx, sizeof(struct GroupPeer), peer_sort_cmp);
|
qsort(chat->peer_list, chat->max_idx, sizeof(struct GroupPeer), peer_sort_cmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Gets the peer_id associated with nick.
|
/* Puts the peer_id associated with nick in `peer_id`.
|
||||||
*
|
*
|
||||||
* Returns 0 on success.
|
* Returns 0 on success.
|
||||||
* Returns -1 on failure or if nick is not assigned to anyone in the group.
|
* Returns -1 if peer is not found.
|
||||||
|
* Returns -2 if there are more than one peers with nick.
|
||||||
*/
|
*/
|
||||||
int group_get_nick_peer_id(uint32_t groupnumber, const char *nick, uint32_t *peer_id)
|
static int group_get_nick_peer_id(uint32_t groupnumber, const char *nick, uint32_t *peer_id)
|
||||||
{
|
{
|
||||||
GroupChat *chat = get_groupchat(groupnumber);
|
GroupChat *chat = get_groupchat(groupnumber);
|
||||||
|
|
||||||
@ -478,6 +479,8 @@ int group_get_nick_peer_id(uint32_t groupnumber, const char *nick, uint32_t *pee
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t count = 0;
|
||||||
|
|
||||||
for (size_t i = 0; i < chat->max_idx; ++i) {
|
for (size_t i = 0; i < chat->max_idx; ++i) {
|
||||||
GroupPeer *peer = &chat->peer_list[i];
|
GroupPeer *peer = &chat->peer_list[i];
|
||||||
|
|
||||||
@ -486,12 +489,15 @@ int group_get_nick_peer_id(uint32_t groupnumber, const char *nick, uint32_t *pee
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (strcmp(nick, peer->name) == 0) {
|
if (strcmp(nick, peer->name) == 0) {
|
||||||
|
if (++count > 1) {
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
|
||||||
*peer_id = peer->peer_id;
|
*peer_id = peer->peer_id;
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return -1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Gets the peer_id associated with `public_key`.
|
/* Gets the peer_id associated with `public_key`.
|
||||||
@ -529,6 +535,50 @@ int group_get_public_key_peer_id(uint32_t groupnumber, const char *public_key, u
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Puts the peer_id associated with `identifier` in `peer_id`. The string may be
|
||||||
|
* either a nick or a public key.
|
||||||
|
*
|
||||||
|
* On failure, `peer_id` is set to (uint32_t)-1.
|
||||||
|
*
|
||||||
|
* This function is intended to be a helper for groupchat_commands.c and will print
|
||||||
|
* error messages to `self`.
|
||||||
|
* Return 0 on success.
|
||||||
|
* Return -1 if the identifier does not correspond with a peer in the group, or if
|
||||||
|
* the identifier is a nick which is being used by multiple peers.
|
||||||
|
*/
|
||||||
|
int group_get_peer_id_of_identifier(ToxWindow *self, const char *identifier, uint32_t *peer_id)
|
||||||
|
{
|
||||||
|
*peer_id = (uint32_t) -1;
|
||||||
|
|
||||||
|
if (group_get_public_key_peer_id(self->num, identifier, peer_id) == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ret = group_get_nick_peer_id(self->num, identifier, peer_id);
|
||||||
|
|
||||||
|
switch (ret) {
|
||||||
|
case 0: {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
case -1: {
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Invalid peer name or public key.");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
case -2: {
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0,
|
||||||
|
"More than one peer is using this name. Specify the target's public key.");
|
||||||
|
*peer_id = (uint32_t) -1;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
line_info_add(self, false, NULL, NULL, SYS_MSG, 0, 0, "Unspecified error.");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void groupchat_update_last_seen(uint32_t groupnumber, uint32_t peer_id)
|
static void groupchat_update_last_seen(uint32_t groupnumber, uint32_t peer_id)
|
||||||
{
|
{
|
||||||
GroupChat *chat = get_groupchat(groupnumber);
|
GroupChat *chat = get_groupchat(groupnumber);
|
||||||
@ -1076,10 +1126,6 @@ static void groupchat_onGroupRejected(ToxWindow *self, Tox *m, uint32_t groupnum
|
|||||||
const char *msg = NULL;
|
const char *msg = NULL;
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case TOX_GROUP_JOIN_FAIL_NAME_TAKEN:
|
|
||||||
msg = "Nick already in use. Change your nick and use the '/rejoin' command.";
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TOX_GROUP_JOIN_FAIL_PEER_LIMIT:
|
case TOX_GROUP_JOIN_FAIL_PEER_LIMIT:
|
||||||
msg = "Group is full. Try again with the '/rejoin' command.";
|
msg = "Group is full. Try again with the '/rejoin' command.";
|
||||||
break;
|
break;
|
||||||
|
@ -78,12 +78,18 @@ void groupchat_onGroupModeration(ToxWindow *self, Tox *m, uint32_t groupnumber,
|
|||||||
|
|
||||||
void groupchat_rejoin(ToxWindow *self, Tox *m);
|
void groupchat_rejoin(ToxWindow *self, Tox *m);
|
||||||
|
|
||||||
/* Gets the peer_id associated with nick.
|
/* Puts the peer_id associated with `identifier` in `peer_id`. The string may be
|
||||||
|
* either a nick or a public key.
|
||||||
*
|
*
|
||||||
* Returns 0 on success.
|
* On failure, `peer_id` is set to (uint32_t)-1.
|
||||||
* Returns -1 on failure or if nick is not assigned to anyone in the group.
|
*
|
||||||
|
* This function is intended to be a helper for groupchat_commands.c and will print
|
||||||
|
* error messages to `self`.
|
||||||
|
* Return 0 on success.
|
||||||
|
* Return -1 if the identifier does not correspond with a peer in the group.
|
||||||
|
* Return -2 if the identifier is a nick and the nick is in use by multiple peers.
|
||||||
*/
|
*/
|
||||||
int group_get_nick_peer_id(uint32_t groupnumber, const char *nick, uint32_t *peer_id);
|
int group_get_peer_id_of_identifier(ToxWindow *self, const char *identifier, uint32_t *peer_id);
|
||||||
|
|
||||||
/* Gets the peer_id associated with `public_key`.
|
/* Gets the peer_id associated with `public_key`.
|
||||||
*
|
*
|
||||||
|
Loading…
Reference in New Issue
Block a user