bip/src/line.c

227 lines
4.5 KiB
C

/*
* $Id$
*
* This file is part of the bip project
* Copyright (C) 2004,2005 Arnaud Cornet
* Copyright (C) 2004,2005,2022 Loïc Gomez
*
* This program 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 2 of the License, or
* (at your option) any later version.
* See the file "COPYING" for the exact licensing terms.
*/
#include "config.h"
#include "line.h"
#include "util.h"
// TODO resolve assuming signed overflow does not occur when changing X +- C1
// cmp C2 to X cmp C2 -+ C1
#pragma GCC diagnostic ignored "-Wstrict-overflow"
void irc_line_init(struct line *l)
{
memset(l, 0, sizeof(struct line));
array_init(&l->words);
}
void _irc_line_deinit(struct line *l)
{
array_deinit(&l->words);
}
struct line *irc_line_new()
{
struct line *l;
l = bip_malloc(sizeof(struct line));
irc_line_init(l);
return l;
}
void irc_line_write(struct line *l, connection_t *c)
{
char *bytes = irc_line_to_string(l);
write_line(c, bytes);
free(bytes);
}
struct line *irc_line_dup(struct line *line)
{
int i;
struct line *nl = irc_line_new();
char *ptr;
nl->origin = line->origin ? bip_strdup(line->origin) : NULL;
array_each(&line->words, i, ptr)
array_set(&nl->words, i, bip_strdup(ptr));
nl->colon = line->colon;
return nl;
}
char *irc_line_pop(struct line *l)
{
return (char *)array_pop(&l->words);
}
void _irc_line_append(struct line *l, const char *s)
{
array_push(&l->words, bip_strdup(s));
}
void irc_line_append(struct line *l, const char *s)
{
_irc_line_append(l, bip_strdup(s));
}
char *irc_line_to_string(struct line *l)
{
size_t len = 0;
int i;
char *ret;
if (l->origin)
len = strlen(l->origin) + 2;
for (i = 0; i < array_count(&l->words); i++)
len += strlen(array_get(&l->words, i)) + 1;
len += 1; /* remove one trailing space and add \r\n */
len++; /* last args ":" */
ret = bip_malloc(len + 1);
ret[0] = 0;
if (l->origin) {
strcat(ret, ":");
strcat(ret, l->origin);
strcat(ret, " ");
}
for (i = 0; i < array_count(&l->words) - 1; i++) {
strcat(ret, array_get(&l->words, i));
strcat(ret, " ");
}
if (strchr(array_get(&l->words, i), ' ') || l->colon)
strcat(ret, ":");
strcat(ret, array_get(&l->words, i));
strcat(ret, "\r\n");
return ret;
}
char *irc_line_to_string_to(struct line *line, char *nick)
{
char *tmp;
char *l;
const char *prev;
prev = irc_line_elem(line, 1);
tmp = bip_strdup(prev);
array_set(&line->words, 1, nick);
l = irc_line_to_string(line);
array_set(&line->words, 1, tmp);
bip_cfree(prev);
return l;
}
int irc_line_count(struct line *line)
{
return array_count(&line->words);
}
int irc_line_includes(struct line *line, int elem)
{
return array_includes(&line->words, elem);
}
const char *irc_line_elem(struct line *line, int elem)
{
return array_get(&line->words, elem);
}
void irc_line_drop(struct line *line, int elem)
{
bip_cfree(array_drop(&line->words, elem));
}
int irc_line_elem_equals(struct line *line, int elem, const char *cmp)
{
return !strcmp(irc_line_elem(line, elem), cmp);
}
int irc_line_elem_case_equals(struct line *line, int elem, const char *cmp)
{
return !strcasecmp(irc_line_elem(line, elem), cmp);
}
/*
* takes a null terminated string as input w/o \r\n
*/
struct line *irc_line_new_from_string(char *str)
{
struct line *line;
char *space;
size_t len;
line = irc_line_new();
if (str[0] == ':') {
space = str + 1;
while (*space && *space != ' ')
space++;
if (!*space) {
irc_line_free(line);
return NULL;
}
// space is at least str + 1, len >= 0
len = (size_t)(space - str - 1); /* leading ':' */
line->origin = bip_malloc(len + 1);
memcpy(line->origin, str + 1, len);
line->origin[len] = 0;
str = space;
}
while (*str == ' ')
str++;
while (*str) {
char *tmp;
space = str;
if (*space == ':') {
line->colon = 1;
str++;
while (*space)
space++;
} else {
while (*space && *space != ' ')
space++;
}
// str is the start of string
// space is the end of string or end of word
len = (size_t)(space - str);
tmp = bip_malloc(len + 1);
memcpy(tmp, str, len);
tmp[len] = 0;
if (array_count(&line->words) == 0)
strucase(tmp);
array_push(&line->words, tmp);
str = space;
while (*str == ' ')
str++;
}
return line;
}
void irc_line_free(struct line *l)
{
int i;
for (i = 0; i < array_count(&l->words); i++)
bip_cfree(array_get(&l->words, i));
array_deinit(&l->words);
if (l->origin)
free(l->origin);
free(l);
}