1
0
mirror of https://github.com/Tha14/toxic.git synced 2024-07-01 17:07:46 +02:00
toxic/src/misc_tools.c

461 lines
11 KiB
C
Raw Normal View History

2014-02-25 08:28:24 +01:00
/* misc_tools.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-06-22 03:41:38 +02:00
#include <ctype.h>
#include <string.h>
#include <time.h>
#include <limits.h>
#include <dirent.h>
2014-08-28 04:45:11 +02:00
#include <sys/stat.h>
#include "toxic.h"
#include "windows.h"
#include "misc_tools.h"
2014-04-07 10:42:10 +02:00
#include "settings.h"
#include "file_transfers.h"
2013-12-10 01:25:09 +01:00
extern ToxWindow *prompt;
extern struct user_settings *user_settings;
2013-12-10 01:25:09 +01:00
2014-03-14 04:56:46 +01:00
static uint64_t current_unix_time;
void hst_to_net(uint8_t *num, uint16_t numbytes)
2014-07-31 20:53:02 +02:00
{
#ifndef WORDS_BIGENDIAN
uint32_t i;
uint8_t buff[numbytes];
for (i = 0; i < numbytes; ++i) {
buff[i] = num[numbytes - i - 1];
}
memcpy(num, buff, numbytes);
#endif
return;
}
2015-08-28 03:29:34 +02:00
/* Note: The time functions are not thread safe */
2014-03-14 04:56:46 +01:00
void update_unix_time(void)
{
current_unix_time = (uint64_t) time(NULL);
}
uint64_t get_unix_time(void)
{
return current_unix_time;
}
2014-07-08 20:46:50 +02:00
/* Returns 1 if connection has timed out, 0 otherwise */
2015-08-18 07:46:22 +02:00
int timed_out(uint64_t timestamp, uint64_t timeout)
2014-07-08 20:46:50 +02:00
{
2015-08-18 07:46:22 +02:00
return timestamp + timeout <= get_unix_time();
2014-07-08 20:46:50 +02:00
}
2014-03-14 04:56:46 +01:00
/* Get the current local time */
struct tm *get_time(void)
{
struct tm *timeinfo;
uint64_t t = get_unix_time();
timeinfo = localtime((const time_t*) &t);
2014-03-14 04:56:46 +01:00
return timeinfo;
}
/*Puts the current time in buf in the format of [HH:mm:ss] */
void get_time_str(char *buf, int bufsize)
{
if (user_settings->timestamps == TIMESTAMPS_OFF) {
buf[0] = '\0';
return;
}
const char *t = user_settings->timestamp_format;
2014-06-24 00:54:23 +02:00
strftime(buf, bufsize, t, get_time());
}
/* Converts seconds to string in format HH:mm:ss; truncates hours and minutes when necessary */
void get_elapsed_time_str(char *buf, int bufsize, uint64_t secs)
2014-06-24 00:54:23 +02:00
{
if (!secs)
return;
2014-07-08 09:39:42 +02:00
long int seconds = secs % 60;
long int minutes = (secs % 3600) / 60;
long int hours = secs / 3600;
2014-06-24 00:54:23 +02:00
if (!minutes && !hours)
snprintf(buf, bufsize, "%.2ld", seconds);
else if (!hours)
snprintf(buf, bufsize, "%ld:%.2ld", minutes, seconds);
else
snprintf(buf, bufsize, "%ld:%.2ld:%.2ld", hours, minutes, seconds);
}
/*
* Converts a hexidecimal string of length hex_len to binary format and puts the result in output.
* output_size must be exactly half of hex_len.
*
* Returns 0 on success.
* Returns -1 on failure.
*/
int hex_string_to_bin(const char *hex_string, size_t hex_len, char *output, size_t output_size)
{
if (output_size == 0 || hex_len != output_size * 2)
return -1;
for (size_t i = 0; i < output_size; ++i) {
sscanf(hex_string, "%2hhx", &output[i]);
hex_string += 2;
}
return 0;
}
2014-08-11 07:59:01 +02:00
int hex_string_to_bytes(char *buf, int size, const char *keystr)
{
if (size % 2 != 0)
return -1;
int i, res;
const char *pos = keystr;
for (i = 0; i < size; ++i) {
res = sscanf(pos, "%2hhx", &buf[i]);
pos += 2;
if (res == EOF || res < 1)
return -1;
}
return 0;
}
/* Converts a binary representation of a Tox ID into a string.
*
* Returns 0 on success.
* Returns -1 on failure.
*/
int bin_id_to_string(const char *bin_id, size_t bin_id_size, char *output, size_t output_size)
{
if (bin_id_size != TOX_ADDRESS_SIZE || output_size < (TOX_ADDRESS_SIZE * 2 + 1))
return -1;
size_t i;
for (i = 0; i < TOX_ADDRESS_SIZE; ++i)
snprintf(&output[i*2], output_size - (i * 2), "%02X", bin_id[i] & 0xff);
return 0;
}
/* Returns 1 if the string is empty, 0 otherwise */
int string_is_empty(const char *string)
{
if (!string)
return true;
2013-10-22 07:59:06 +02:00
return string[0] == '\0';
}
/* convert a multibyte string to a wide character string and puts in buf. */
int mbs_to_wcs_buf(wchar_t *buf, const char *string, size_t n)
{
size_t len = mbstowcs(NULL, string, 0) + 1;
if (n < len)
return -1;
if ((len = mbstowcs(buf, string, n)) == (size_t) -1)
return -1;
return len;
}
/* converts wide character string into a multibyte string and puts in buf. */
int wcs_to_mbs_buf(char *buf, const wchar_t *string, size_t n)
{
size_t len = wcstombs(NULL, string, 0) + 1;
if (n < len)
return -1;
if ((len = wcstombs(buf, string, n)) == (size_t) -1)
return -1;
return len;
}
2013-12-08 07:18:10 +01:00
/* case-insensitive string compare function for use with qsort */
int qsort_strcasecmp_hlpr(const void *str1, const void *str2)
2013-11-24 08:33:03 +01:00
{
return strcasecmp((const char *) str1, (const char *) str2);
2013-11-24 21:18:05 +01:00
}
2013-11-28 08:53:43 +01:00
/* Returns 1 if nick is valid, 0 if not. A valid toxic nick:
2013-11-28 08:53:43 +01:00
- cannot be empty
- cannot start with a space
- must not contain a forward slash (for logfile naming purposes)
2014-10-07 06:43:03 +02:00
- must not contain contiguous spaces
- must not contain a newline or tab seqeunce */
int valid_nick(const char *nick)
2013-11-28 08:53:43 +01:00
{
if (!nick[0] || nick[0] == ' ')
2014-03-12 01:00:03 +01:00
return 0;
2013-11-28 08:53:43 +01:00
int i;
for (i = 0; nick[i]; ++i) {
2014-10-07 06:43:03 +02:00
if ((nick[i] == ' ' && nick[i + 1] == ' ')
2015-03-26 03:56:45 +01:00
|| nick[i] == '/'
|| nick[i] == '\n'
|| nick[i] == '\t'
|| nick[i] == '\v'
2014-10-07 06:43:03 +02:00
|| nick[i] == '\r')
return 0;
2013-11-28 08:53:43 +01:00
}
2014-03-12 01:00:03 +01:00
return 1;
2013-11-29 00:56:56 +01:00
}
2013-12-11 06:10:09 +01:00
2014-10-07 06:43:03 +02:00
/* Converts all newline/tab chars to spaces (use for strings that should be contained to a single line) */
2015-03-26 03:56:45 +01:00
void filter_str(char *str, size_t len)
{
2015-03-26 03:56:45 +01:00
size_t i;
for (i = 0; i < len; ++i) {
if (str[i] == '\n' || str[i] == '\r' || str[i] == '\t' || str[i] == '\v' || str[i] == '\0')
str[i] = ' ';
}
}
/* gets base file name from path or original file name if no path is supplied.
* Returns the file name length
*/
size_t get_file_name(char *namebuf, size_t bufsize, const char *pathname)
{
int len = strlen(pathname) - 1;
char *path = strdup(pathname);
2014-07-17 09:35:18 +02:00
if (path == NULL)
exit_toxic_err("failed in get_file_name", FATALERR_MEMORY);
2014-07-02 23:30:31 +02:00
while (len >= 0 && pathname[len] == '/')
path[len--] = '\0';
char *finalname = strdup(path);
2014-07-17 09:35:18 +02:00
if (finalname == NULL)
exit_toxic_err("failed in get_file_name", FATALERR_MEMORY);
2014-07-17 09:35:18 +02:00
const char *basenm = strrchr(path, '/');
if (basenm != NULL) {
if (basenm[1])
strcpy(finalname, &basenm[1]);
}
snprintf(namebuf, bufsize, "%s", finalname);
free(finalname);
free(path);
return strlen(namebuf);
}
/* Gets the base directory of path and puts it in dir.
* dir must have at least as much space as path_len.
*
* Returns the length of the base directory.
*/
size_t get_base_dir(const char *path, size_t path_len, char *dir)
{
size_t dir_len = char_rfind(path, '/', path_len);
if (dir_len != 0 && dir_len < path_len)
++dir_len; /* Leave trailing slash */
memcpy(dir, path, dir_len);
dir[dir_len] = '\0';
return dir_len;
}
/* converts str to all lowercase */
void str_to_lower(char *str)
{
int i;
for (i = 0; str[i]; ++i)
str[i] = tolower(str[i]);
}
/* puts friendnum's nick in buf, truncating at TOXIC_MAX_NAME_LENGTH if necessary.
2014-11-16 04:39:24 +01:00
if toxcore API call fails, put UNKNOWN_NAME in buf
Returns nick len */
2015-03-26 03:56:45 +01:00
size_t get_nick_truncate(Tox *m, char *buf, uint32_t friendnum)
{
size_t len = tox_friend_get_name_size(m, friendnum, NULL);
2014-11-16 04:39:24 +01:00
2015-03-26 03:56:45 +01:00
if (len == 0) {
2014-11-16 04:39:24 +01:00
strcpy(buf, UNKNOWN_NAME);
len = strlen(UNKNOWN_NAME);
2015-03-26 03:56:45 +01:00
} else {
tox_friend_get_name(m, friendnum, (uint8_t *) buf, NULL);
2014-11-16 04:39:24 +01:00
}
2014-10-04 01:29:12 +02:00
len = MIN(len, TOXIC_MAX_NAME_LENGTH - 1);
buf[len] = '\0';
filter_str(buf, len);
return len;
}
/* same as get_nick_truncate but for groupchats */
int get_group_nick_truncate(Tox *m, char *buf, int peernum, int groupnum)
{
int len = tox_group_peername(m, groupnum, peernum, (uint8_t *) buf);
2014-11-16 04:39:24 +01:00
if (len == -1) {
strcpy(buf, UNKNOWN_NAME);
len = strlen(UNKNOWN_NAME);
}
len = MIN(len, TOXIC_MAX_NAME_LENGTH - 1);
buf[len] = '\0';
filter_str(buf, len);
return len;
}
2014-10-03 23:53:50 +02:00
/* copies data to msg buffer.
returns length of msg, which will be no larger than size-1 */
2015-03-26 03:56:45 +01:00
size_t copy_tox_str(char *msg, size_t size, const char *data, size_t length)
2014-10-03 23:53:50 +02:00
{
2015-03-26 03:56:45 +01:00
size_t len = MIN(length, size - 1);
2014-10-03 23:53:50 +02:00
memcpy(msg, data, len);
msg[len] = '\0';
return len;
}
/* returns index of the first instance of ch in s starting at idx.
returns length of s if char not found */
int char_find(int idx, const char *s, char ch)
{
2014-07-17 03:32:47 +02:00
int i = idx;
for (i = idx; s[i]; ++i) {
if (s[i] == ch)
break;
}
return i;
}
/* returns index of the last instance of ch in s starting at len.
returns 0 if char not found (skips 0th index). */
int char_rfind(const char *s, char ch, int len)
{
int i = 0;
for (i = len; i > 0; --i) {
if (s[i] == ch)
break;
}
return i;
2014-07-27 23:14:28 +02:00
}
/* Converts bytes to appropriate unit and puts in buf as a string */
void bytes_convert_str(char *buf, int size, uint64_t bytes)
{
double conv = bytes;
const char *unit;
if (conv < KiB) {
unit = "Bytes";
} else if (conv < MiB) {
unit = "KiB";
conv /= (double) KiB;
} else if (conv < GiB) {
unit = "MiB";
conv /= (double) MiB;
} else {
unit = "GiB";
conv /= (double) GiB;
}
snprintf(buf, size, "%.1f %s", conv, unit);
}
2014-08-28 04:45:11 +02:00
/* checks if a file exists. Returns true or false */
2014-09-14 23:46:28 +02:00
bool file_exists(const char *path)
2014-08-28 04:45:11 +02:00
{
struct stat s;
2014-09-14 23:46:28 +02:00
return stat(path, &s) == 0;
2014-08-28 04:45:11 +02:00
}
2014-09-24 06:06:02 +02:00
/* returns file size. If file doesn't exist returns 0. */
2014-09-24 20:23:08 +02:00
off_t file_size(const char *path)
2014-09-24 06:06:02 +02:00
{
struct stat st;
if (stat(path, &st) == -1)
return 0;
2014-09-24 06:06:02 +02:00
2014-09-24 20:23:08 +02:00
return st.st_size;
2014-09-24 06:06:02 +02:00
}
2015-03-26 03:56:45 +01:00
/* compares the first size bytes of fp to signature.
Returns 0 if they are the same, 1 if they differ, and -1 on error.
On success this function will seek back to the beginning of fp */
int check_file_signature(const char *signature, size_t size, FILE *fp)
{
char buf[size];
2014-09-29 19:56:17 +02:00
if (fread(buf, size, 1, fp) != 1)
return -1;
int ret = memcmp(signature, buf, size);
if (fseek(fp, 0L, SEEK_SET) == -1)
return -1;
return ret == 0 ? 0 : 1;
}
2014-11-15 04:15:59 +01:00
/* sets window title in tab bar. */
void set_window_title(ToxWindow *self, const char *title, int len)
{
char cpy[TOXIC_MAX_NAME_LENGTH + 1];
if (self->is_groupchat) /* keep groupnumber in title */
2014-11-15 21:55:45 +01:00
snprintf(cpy, sizeof(cpy), "%d %s", self->num, title);
2014-11-15 04:15:59 +01:00
else
snprintf(cpy, sizeof(cpy), "%s", title);
if (len > MAX_WINDOW_NAME_LENGTH) {
strcpy(&cpy[MAX_WINDOW_NAME_LENGTH - 3], "...");
cpy[MAX_WINDOW_NAME_LENGTH] = '\0';
}
snprintf(self->name, sizeof(self->name), "%s", cpy);
}