labwc/src/common/buf.c

113 lines
2.1 KiB
C
Raw Normal View History

2021-11-13 21:56:53 +00:00
// SPDX-License-Identifier: GPL-2.0-only
#include <ctype.h>
#include <stdbool.h>
2020-07-31 21:31:03 +01:00
#include "common/buf.h"
#include "common/mem.h"
static void
buf_add_one_char(struct buf *s, char ch)
{
if (s->alloc <= s->len + 1) {
s->alloc = s->alloc * 3 / 2 + 16;
s->buf = xrealloc(s->buf, s->alloc);
}
s->buf[s->len++] = ch;
s->buf[s->len] = '\0';
}
void
buf_expand_tilde(struct buf *s)
{
struct buf new;
buf_init(&new);
for (int i = 0 ; i < s->len ; i++) {
if (s->buf[i] == '~') {
buf_add(&new, getenv("HOME"));
} else {
buf_add_one_char(&new, s->buf[i]);
}
}
free(s->buf);
s->buf = new.buf;
s->len = new.len;
s->alloc = new.alloc;
}
static void
strip_curly_braces(char *s)
{
size_t len = strlen(s);
if (s[0] != '{' || s[len - 1] != '}') {
return;
}
len -= 2;
memmove(s, s + 1, len);
s[len] = 0;
}
static bool
isvalid(char p)
{
return isalnum(p) || p == '_' || p == '{' || p == '}';
}
void
buf_expand_shell_variables(struct buf *s)
{
struct buf new;
struct buf environment_variable;
buf_init(&new);
buf_init(&environment_variable);
for (int i = 0 ; i < s->len ; i++) {
if (s->buf[i] == '$' && isvalid(s->buf[i+1])) {
/* expand environment variable */
environment_variable.len = 0;
buf_add(&environment_variable, s->buf + i + 1);
char *p = environment_variable.buf;
while (isvalid(*p)) {
++p;
}
*p = '\0';
i += strlen(environment_variable.buf);
strip_curly_braces(environment_variable.buf);
p = getenv(environment_variable.buf);
if (p) {
buf_add(&new, p);
}
} else {
buf_add_one_char(&new, s->buf[i]);
}
}
free(environment_variable.buf);
free(s->buf);
s->buf = new.buf;
s->len = new.len;
s->alloc = new.alloc;
}
void
buf_init(struct buf *s)
{
s->alloc = 256;
s->buf = xmalloc(s->alloc);
s->buf[0] = '\0';
s->len = 0;
}
void
buf_add(struct buf *s, const char *data)
{
if (!data || data[0] == '\0') {
return;
}
int len = strlen(data);
if (s->alloc <= s->len + len + 1) {
s->alloc = s->alloc + len;
s->buf = xrealloc(s->buf, s->alloc);
}
memcpy(s->buf + s->len, data, len);
s->len += len;
s->buf[s->len] = 0;
}