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/>.
|
|
|
|
*
|
2013-09-21 02:35:03 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2014-06-22 03:41:38 +02:00
|
|
|
#include <ctype.h>
|
2013-09-21 02:35:03 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <limits.h>
|
2014-07-18 07:29:46 +02:00
|
|
|
#include <dirent.h>
|
2013-09-21 02:35:03 +02:00
|
|
|
|
2014-06-12 00:06:55 +02:00
|
|
|
#include "toxic.h"
|
|
|
|
#include "windows.h"
|
2014-03-24 12:18:58 +01:00
|
|
|
#include "misc_tools.h"
|
2014-04-07 10:42:10 +02:00
|
|
|
#include "settings.h"
|
2013-09-21 02:35:03 +02:00
|
|
|
|
2013-12-10 01:25:09 +01:00
|
|
|
extern ToxWindow *prompt;
|
2014-07-21 01:12:13 +02:00
|
|
|
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 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 */
|
|
|
|
int timed_out(uint64_t timestamp, uint64_t curtime, uint64_t timeout)
|
|
|
|
{
|
|
|
|
return timestamp + timeout <= curtime;
|
|
|
|
}
|
|
|
|
|
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();
|
2014-07-07 04:15:35 +02:00
|
|
|
timeinfo = localtime((const time_t*) &t);
|
2014-03-14 04:56:46 +01:00
|
|
|
return timeinfo;
|
|
|
|
}
|
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
/*Puts the current time in buf in the format of [HH:mm:ss] */
|
2014-07-07 04:15:35 +02:00
|
|
|
void get_time_str(char *buf, int bufsize)
|
2014-03-24 12:18:58 +01:00
|
|
|
{
|
2014-07-21 01:12:13 +02:00
|
|
|
if (user_settings_->timestamps == TIMESTAMPS_OFF) {
|
2014-06-29 02:33:46 +02:00
|
|
|
buf[0] = '\0';
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-21 01:12:13 +02:00
|
|
|
const char *t = user_settings_->time == TIME_12 ? "[%-I:%M:%S] " : "[%H:%M:%S] ";
|
2014-06-24 00:54:23 +02:00
|
|
|
strftime(buf, bufsize, t, get_time());
|
|
|
|
}
|
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
/* Converts seconds to string in format HH:mm:ss; truncates hours and minutes when necessary */
|
2014-07-07 04:15:35 +02:00
|
|
|
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
|
|
|
|
2014-06-24 07:54:08 +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);
|
2014-03-24 12:18:58 +01:00
|
|
|
}
|
|
|
|
|
2014-06-13 07:42:20 +02:00
|
|
|
char *hex_string_to_bin(const char *hex_string)
|
2013-09-21 02:35:03 +02:00
|
|
|
{
|
|
|
|
size_t len = strlen(hex_string);
|
2014-06-13 07:42:20 +02:00
|
|
|
char *val = malloc(len);
|
2013-09-21 02:35:03 +02:00
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
if (val == NULL)
|
|
|
|
exit_toxic_err("failed in hex_string_to_bin", FATALERR_MEMORY);
|
2013-09-21 02:35:03 +02:00
|
|
|
|
|
|
|
size_t i;
|
|
|
|
|
2014-06-13 07:42:20 +02:00
|
|
|
for (i = 0; i < len; ++i, hex_string += 2)
|
|
|
|
sscanf(hex_string, "%2hhx", &val[i]);
|
2013-09-21 02:35:03 +02:00
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2014-03-09 08:42:37 +01:00
|
|
|
/* Returns 1 if the string is empty, 0 otherwise */
|
2014-07-16 18:47:14 +02:00
|
|
|
int string_is_empty(const char *string)
|
2013-09-21 02:35:03 +02:00
|
|
|
{
|
2013-10-22 07:59:06 +02:00
|
|
|
return string[0] == '\0';
|
2013-09-21 02:35:03 +02:00
|
|
|
}
|
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
/* convert a multibyte string to a wide character string and puts in buf. */
|
2014-07-07 04:15:35 +02:00
|
|
|
int mbs_to_wcs_buf(wchar_t *buf, const char *string, size_t n)
|
2013-12-08 04:10:32 +01:00
|
|
|
{
|
|
|
|
size_t len = mbstowcs(NULL, string, 0) + 1;
|
|
|
|
|
|
|
|
if (n < len)
|
|
|
|
return -1;
|
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
if ((len = mbstowcs(buf, string, n)) == (size_t) -1)
|
2013-12-08 04:10:32 +01:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
/* converts wide character string into a multibyte string and puts in buf. */
|
2014-07-07 04:15:35 +02:00
|
|
|
int wcs_to_mbs_buf(char *buf, const wchar_t *string, size_t n)
|
2013-12-08 04:10:32 +01:00
|
|
|
{
|
|
|
|
size_t len = wcstombs(NULL, string, 0) + 1;
|
|
|
|
|
|
|
|
if (n < len)
|
|
|
|
return -1;
|
|
|
|
|
2014-06-24 07:54:08 +02:00
|
|
|
if ((len = wcstombs(buf, string, n)) == (size_t) -1)
|
2013-12-08 04:10:32 +01:00
|
|
|
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 *nick1, const void *nick2)
|
2013-11-24 08:33:03 +01:00
|
|
|
{
|
2013-12-08 07:18:10 +01:00
|
|
|
return strcasecmp((const char *) nick1, (const char *) nick2);
|
2013-11-24 21:18:05 +01:00
|
|
|
}
|
2013-11-28 08:53:43 +01:00
|
|
|
|
2014-03-11 23:57:32 +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
|
2014-04-12 09:39:15 +02:00
|
|
|
- must not contain a forward slash (for logfile naming purposes)
|
2013-11-28 08:53:43 +01:00
|
|
|
- must not contain contiguous spaces */
|
2014-07-07 04:15:35 +02:00
|
|
|
int valid_nick(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-04-19 23:58:13 +02:00
|
|
|
if (nick[i] == ' ' && nick[i + 1] == ' ')
|
2014-03-12 01:00:03 +01:00
|
|
|
return 0;
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2014-04-12 09:39:15 +02:00
|
|
|
if (nick[i] == '/')
|
|
|
|
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-02-12 01:12:26 +01:00
|
|
|
/* gets base file name from path or original file name if no path is supplied */
|
2014-07-09 06:05:13 +02:00
|
|
|
void get_file_name(char *namebuf, int bufsize, const char *pathname)
|
2014-02-12 00:52:04 +01:00
|
|
|
{
|
|
|
|
int idx = strlen(pathname) - 1;
|
2014-07-16 18:47:14 +02:00
|
|
|
char *path = strdup(pathname);
|
2014-02-12 00:52:04 +01:00
|
|
|
|
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
|
|
|
|
2014-02-12 00:52:04 +01:00
|
|
|
while (idx >= 0 && pathname[idx] == '/')
|
2014-07-16 18:47:14 +02:00
|
|
|
path[idx--] = '\0';
|
2014-02-12 00:52:04 +01:00
|
|
|
|
2014-07-16 18:47:14 +02:00
|
|
|
char *finalname = strdup(path);
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2014-07-17 09:35:18 +02:00
|
|
|
if (finalname == NULL)
|
|
|
|
exit_toxic_err("failed in get_file_name", FATALERR_MEMORY);
|
|
|
|
|
2014-07-16 18:47:14 +02:00
|
|
|
const char *basenm = strrchr(path, '/');
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2014-07-16 18:47:14 +02:00
|
|
|
if (basenm != NULL) {
|
|
|
|
if (basenm[1])
|
|
|
|
strcpy(finalname, &basenm[1]);
|
2014-02-12 00:52:04 +01:00
|
|
|
}
|
|
|
|
|
2014-07-16 18:47:14 +02:00
|
|
|
snprintf(namebuf, bufsize, "%s", finalname);
|
|
|
|
free(finalname);
|
|
|
|
free(path);
|
2014-02-12 00:52:04 +01:00
|
|
|
}
|
2014-06-18 19:10:20 +02:00
|
|
|
|
|
|
|
/* converts str to all lowercase */
|
2014-07-07 04:15:35 +02:00
|
|
|
void str_to_lower(char *str)
|
2014-06-18 19:10:20 +02:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; str[i]; ++i)
|
|
|
|
str[i] = tolower(str[i]);
|
|
|
|
}
|
2014-07-08 20:31:59 +02:00
|
|
|
|
|
|
|
/* puts friendnum's nick in buf, truncating at TOXIC_MAX_NAME_LENGTH if necessary.
|
|
|
|
Returns nick len on success, -1 on failure */
|
|
|
|
int get_nick_truncate(Tox *m, char *buf, int friendnum)
|
|
|
|
{
|
|
|
|
int len = tox_get_name(m, friendnum, (uint8_t *) buf);
|
|
|
|
len = MIN(len, TOXIC_MAX_NAME_LENGTH - 1);
|
|
|
|
buf[len] = '\0';
|
|
|
|
return len;
|
|
|
|
}
|
2014-07-16 18:47:14 +02:00
|
|
|
|
|
|
|
/* 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;
|
2014-07-16 18:47:14 +02:00
|
|
|
|
2014-07-18 07:29:46 +02:00
|
|
|
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) {
|
2014-07-16 18:47:14 +02:00
|
|
|
if (s[i] == ch)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
2014-07-21 01:12:13 +02:00
|
|
|
}
|