2014-09-06 19:18:42 +02:00
|
|
|
/* 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>
|
|
|
|
|
|
|
|
#include "toxic.h"
|
|
|
|
#include "windows.h"
|
|
|
|
#include "message_queue.h"
|
2014-09-07 08:43:53 +02:00
|
|
|
#include "misc_tools.h"
|
|
|
|
#include "line_info.h"
|
2014-09-11 07:36:33 +02:00
|
|
|
#include "log.h"
|
2014-09-06 19:18:42 +02:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2015-03-28 07:56:54 +01:00
|
|
|
void cqueue_add(struct chat_queue *q, const char *msg, size_t len, uint8_t type, uint32_t line_id)
|
2014-09-06 19:18:42 +02:00
|
|
|
{
|
2014-09-09 20:06:28 +02:00
|
|
|
struct cqueue_msg *new_m = malloc(sizeof(struct cqueue_msg));
|
2014-09-06 19:18:42 +02:00
|
|
|
|
|
|
|
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 = 0;
|
|
|
|
new_m->next = NULL;
|
2014-09-06 19:18:42 +02:00
|
|
|
|
2014-09-09 20:06:28 +02:00
|
|
|
if (q->root == NULL) {
|
|
|
|
new_m->prev = NULL;
|
2014-09-06 19:18:42 +02:00
|
|
|
q->root = new_m;
|
2014-09-09 20:06:28 +02:00
|
|
|
} else {
|
|
|
|
new_m->prev = q->end;
|
2014-09-06 19:18:42 +02:00
|
|
|
q->end->next = new_m;
|
2014-09-09 20:06:28 +02:00
|
|
|
}
|
2014-09-06 19:18:42 +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) {
|
2014-09-10 22:18:37 +02:00
|
|
|
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
|
|
|
|
2014-09-10 22:18:37 +02:00
|
|
|
if (line->noread_flag == true) {
|
|
|
|
line->len -= 2;
|
|
|
|
line->noread_flag = false;
|
2014-09-07 08:43:53 +02:00
|
|
|
}
|
|
|
|
|
2014-09-10 22:18:37 +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)
|
2014-09-06 19:18:42 +02:00
|
|
|
{
|
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
|
|
|
|
2014-09-11 07:36:33 +02:00
|
|
|
char selfname[TOX_MAX_NAME_LENGTH];
|
2015-03-26 03:56:45 +01:00
|
|
|
tox_self_get_name(m, (uint8_t *) selfname);
|
|
|
|
|
|
|
|
size_t len = tox_self_get_name_size(m);
|
2014-09-11 07:36:33 +02:00
|
|
|
selfname[len] = '\0';
|
|
|
|
|
|
|
|
write_to_log(msg->message, selfname, self->chatwin->log, msg->type == OUT_ACTION);
|
|
|
|
cqueue_mark_read(self, msg);
|
|
|
|
|
2014-09-09 20:06:28 +02:00
|
|
|
struct cqueue_msg *next = msg->next;
|
2014-09-06 19:18:42 +02:00
|
|
|
|
2014-09-09 20:06:28 +02:00
|
|
|
if (msg->prev == NULL) { /* root */
|
|
|
|
if (next)
|
|
|
|
next->prev = NULL;
|
|
|
|
|
|
|
|
free(msg);
|
|
|
|
q->root = next;
|
|
|
|
} else {
|
2014-09-09 21:16:27 +02:00
|
|
|
struct cqueue_msg *prev = msg->prev;
|
2014-09-09 20:06:28 +02:00
|
|
|
free(msg);
|
2014-09-09 21:16:27 +02:00
|
|
|
prev->next = next;
|
2014-09-09 20:06:28 +02:00
|
|
|
}
|
2014-09-07 08:43:53 +02:00
|
|
|
|
2014-09-06 19:18:42 +02:00
|
|
|
return;
|
2014-09-09 20:06:28 +02:00
|
|
|
}
|
|
|
|
}
|
2014-09-06 19:18:42 +02:00
|
|
|
|
2014-10-08 07:50:34 +02:00
|
|
|
#define CQUEUE_TRY_SEND_INTERVAL 60
|
2014-09-10 22:18:37 +02:00
|
|
|
|
2014-09-09 20:06:28 +02:00
|
|
|
/* Tries to send the oldest unsent message in queue. */
|
2014-09-10 22:18:37 +02:00
|
|
|
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;
|
2014-12-04 04:10:21 +01:00
|
|
|
|
|
|
|
if (!msg)
|
|
|
|
return;
|
|
|
|
|
2014-09-07 08:43:53 +02:00
|
|
|
uint64_t curtime = get_unix_time();
|
|
|
|
|
2014-12-04 04:10:21 +01:00
|
|
|
if (msg->receipt != 0 && !timed_out(msg->last_send_try, curtime, CQUEUE_TRY_SEND_INTERVAL))
|
|
|
|
return;
|
2014-09-06 19:18:42 +02:00
|
|
|
|
2014-12-04 04:10:21 +01:00
|
|
|
uint32_t receipt = 0;
|
2014-09-07 08:43:53 +02:00
|
|
|
|
2015-03-28 07:56:54 +01:00
|
|
|
TOX_MESSAGE_TYPE type = msg->type == OUT_MSG ? TOX_MESSAGE_TYPE_NORMAL : TOX_MESSAGE_TYPE_ACTION;
|
|
|
|
receipt = tox_friend_send_message(m, self->num, type, (uint8_t *) msg->message, msg->len, NULL);
|
2014-09-06 19:18:42 +02:00
|
|
|
|
2014-12-04 04:10:21 +01:00
|
|
|
msg->last_send_try = curtime;
|
|
|
|
msg->receipt = receipt;
|
|
|
|
return;
|
2014-09-06 19:18:42 +02:00
|
|
|
}
|