1
0
mirror of https://github.com/Tha14/toxic.git synced 2024-11-23 05:23:02 +01:00
toxic/src/message_queue.c

192 lines
5.1 KiB
C
Raw Normal View History

/* message_queue.c
*
*
* Copyright (C) 2014 Toxic All Rights Reserved.
*
* This file is part of Toxic.
*
* Toxic 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 3 of the License, or
* (at your option) any later version.
*
* Toxic 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 Toxic. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include <stdlib.h>
2014-09-07 08:43:53 +02:00
#include "line_info.h"
2014-09-11 07:36:33 +02:00
#include "log.h"
2020-04-18 02:00:00 +02:00
#include "message_queue.h"
#include "misc_tools.h"
#include "toxic.h"
#include "windows.h"
void cqueue_cleanup(struct chat_queue *q)
{
struct cqueue_msg *tmp1 = q->root;
while (tmp1) {
struct cqueue_msg *tmp2 = tmp1->next;
free(tmp1);
tmp1 = tmp2;
}
free(q);
}
void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, int line_id)
{
if (line_id < 0) {
return;
}
2014-09-09 20:06:28 +02:00
struct cqueue_msg *new_m = malloc(sizeof(struct cqueue_msg));
if (new_m == NULL) {
exit_toxic_err("failed in cqueue_message", FATALERR_MEMORY);
}
snprintf(new_m->message, sizeof(new_m->message), "%s", msg);
new_m->len = len;
new_m->type = type;
new_m->line_id = line_id;
2014-09-09 20:06:28 +02:00
new_m->last_send_try = 0;
new_m->receipt = -1;
2014-09-09 20:06:28 +02:00
new_m->next = NULL;
2014-09-09 20:06:28 +02:00
if (q->root == NULL) {
new_m->prev = NULL;
q->root = new_m;
2014-09-09 20:06:28 +02:00
} else {
new_m->prev = q->end;
q->end->next = new_m;
2014-09-09 20:06:28 +02:00
}
q->end = new_m;
}
2014-09-07 08:43:53 +02:00
/* update line to show receipt was received after queue removal */
2014-09-11 07:36:33 +02:00
static void cqueue_mark_read(ToxWindow *self, struct cqueue_msg *msg)
2014-09-07 08:43:53 +02:00
{
struct line_info *line = self->chatwin->hst->line_end;
while (line) {
2014-09-11 07:36:33 +02:00
if (line->id != msg->line_id) {
line = line->prev;
continue;
}
2014-09-08 04:05:17 +02:00
2014-09-11 07:36:33 +02:00
line->type = msg->type == OUT_ACTION ? OUT_ACTION_READ : OUT_MSG_READ;
2014-09-08 04:05:17 +02:00
if (line->noread_flag) {
line->noread_flag = false;
line->read_flag = true;
2014-09-07 08:43:53 +02:00
}
return;
2014-09-07 08:43:53 +02:00
}
}
2014-09-11 07:36:33 +02:00
/* removes message with matching receipt from queue, writes to log and updates line to show the message was received. */
void cqueue_remove(ToxWindow *self, Tox *m, uint32_t receipt)
{
struct chatlog *log = self->chatwin->log;
2014-09-11 07:36:33 +02:00
struct chat_queue *q = self->chatwin->cqueue;
2014-09-09 20:06:28 +02:00
struct cqueue_msg *msg = q->root;
2014-09-07 08:43:53 +02:00
2014-09-09 20:06:28 +02:00
while (msg) {
if (msg->receipt != receipt) {
msg = msg->next;
continue;
}
2014-09-07 08:43:53 +02:00
if (log->log_on) {
char selfname[TOX_MAX_NAME_LENGTH];
tox_self_get_name(m, (uint8_t *) selfname);
2015-03-26 03:56:45 +01:00
size_t len = tox_self_get_name_size(m);
selfname[len] = 0;
write_to_log(msg->message, selfname, log, msg->type == OUT_ACTION);
}
2014-09-11 07:36:33 +02:00
cqueue_mark_read(self, msg);
2014-09-09 20:06:28 +02:00
struct cqueue_msg *next = msg->next;
2014-09-09 20:06:28 +02:00
if (msg->prev == NULL) { /* root */
if (next) {
2014-09-09 20:06:28 +02:00
next->prev = NULL;
}
2014-09-09 20:06:28 +02:00
q->root = next;
} else {
2014-09-09 21:16:27 +02:00
struct cqueue_msg *prev = msg->prev;
prev->next = next;
next->prev = prev;
2014-09-09 20:06:28 +02:00
}
2014-09-07 08:43:53 +02:00
free(msg);
return;
2014-09-09 20:06:28 +02:00
}
}
// We use knowledge of toxcore internals (bad!) to determine that if we haven't received a read receipt for a
// sent packet after this amount of time, the connection has been severed and the packet needs to be re-sent.
#define TRY_SEND_TIMEOUT 32
/*
* Marks all timed out messages in queue as unsent.
*/
static void cqueue_check_timeouts(struct cqueue_msg *msg)
{
while (msg) {
if (timed_out(msg->last_send_try, TRY_SEND_TIMEOUT)) {
msg->receipt = -1;
}
msg = msg->next;
}
}
/*
* Tries to send all messages in the send queue in sequential order.
* If a message fails to send the function will immediately return.
*/
void cqueue_try_send(ToxWindow *self, Tox *m)
2014-09-09 20:06:28 +02:00
{
struct chat_queue *q = self->chatwin->cqueue;
struct cqueue_msg *msg = q->root;
while (msg) {
if (msg->receipt != -1) {
// we can no longer try to send unsent messages until we get receipts for our previous sent
// messages, but we continue to iterate the list, checking timestamps for any further
// successfully sent messages that have not yet gotten a receipt.
cqueue_check_timeouts(msg);
return;
}
TOX_ERR_FRIEND_SEND_MESSAGE err;
Tox_Message_Type type = msg->type == OUT_MSG ? TOX_MESSAGE_TYPE_NORMAL : TOX_MESSAGE_TYPE_ACTION;
uint32_t receipt = tox_friend_send_message(m, self->num, type, (uint8_t *) msg->message, msg->len, &err);
2014-09-07 08:43:53 +02:00
if (err != TOX_ERR_FRIEND_SEND_MESSAGE_OK) {
return;
}
msg->receipt = receipt;
msg->last_send_try = get_unix_time();
msg = msg->next;
}
}