1
0
mirror of https://github.com/Llewellynvdm/conky.git synced 2025-01-28 01:28:30 +00:00

rewrite template engine completely

Two things that pissed me off about the old one:
* only limited support for nesting templates
* totally broken output when using conditionals inside a template

The later one was the hard one to fix. ;)
It requires to already have the full text substituted before the text
objects are being created from it. Generating only the contained objects
broke, because the conditionals got wrong offsets to jump to.

After that was fixed, full nesting support is realised by simply
repeating the replacement until no more template objects are found.
This commit is contained in:
Phil Sutter 2008-12-09 00:07:32 +01:00
parent 5a69344055
commit f96e77c91b

View File

@ -4247,16 +4247,11 @@ static char *backslash_escape(const char *src, char **templates, unsigned int te
{
char *src_dup;
const char *p;
int dup_idx = 0, dup_len;
unsigned int dup_idx = 0, dup_len;
dup_len = strlen(src);
if (templates) {
unsigned int i;
for (i = 0; i < template_count; i++)
dup_len += strlen(templates[i]);
}
src_dup = malloc(dup_len * sizeof(char));
p = src;
while (*p) {
switch (*p) {
@ -4278,6 +4273,8 @@ static char *backslash_escape(const char *src, char **templates, unsigned int te
if ((sscanf(p + 1, "%u%n", &tmpl_num, &digits) <= 0) ||
(tmpl_num > template_count))
break;
dup_len += strlen(templates[tmpl_num - 1]);
src_dup = realloc(src_dup, dup_len * sizeof(char));
sprintf(src_dup + dup_idx, "%s", templates[tmpl_num - 1]);
dup_idx += strlen(templates[tmpl_num - 1]);
p += digits;
@ -4293,7 +4290,6 @@ static char *backslash_escape(const char *src, char **templates, unsigned int te
src_dup = realloc(src_dup, (strlen(src_dup) + 1) * sizeof(char));
return src_dup;
}
static struct text_object_list *extract_variable_text_internal(const char *, char);
/* handle_template_object - core logic of the template object
*
@ -4306,18 +4302,16 @@ static struct text_object_list *extract_variable_text_internal(const char *, cha
* ${template2 root /}
* ${template2 cdrom /mnt/cdrom}
*/
static int handle_template_object(struct text_object_list **list,
char allow_threaded, const char *tmpl, const char *args)
static char *handle_template(const char *tmpl, const char *args)
{
char *args_dup, *template_dup, *p, *p_old;
char *args_dup, *p, *p_old;
char **argsp = NULL;
unsigned int argcnt = 0, template_idx, i;
char *eval_text;
struct text_object_list *eval_list;
if ((sscanf(tmpl, "template%u", &template_idx) != 1) ||
(template_idx > 9))
return 1;
return NULL;
args_dup = strdup(args);
p = args_dup;
@ -4340,33 +4334,85 @@ static int handle_template_object(struct text_object_list **list,
char *tmp;
tmp = backslash_escape(argsp[i], NULL, 0);
DBGP2("%s: substituted arg '%s' to '%s'", tmpl, argsp[i], tmp);
sprintf(argsp[i], "%s", tmp);
free(tmp);
argsp[i] = tmp;
}
p = template_dup = strdup(template[template_idx]);
eval_text = backslash_escape(template[template_idx], argsp, argcnt);
DBGP("substituted %s, output is '%s'", tmpl, eval_text);
eval_list = extract_variable_text_internal(eval_text, allow_threaded);
if (eval_list) {
(*list)->text_objects = realloc((*list)->text_objects,
sizeof(struct text_object) *
((*list)->text_object_count +
eval_list->text_object_count));
memcpy((*list)->text_objects + (*list)->text_object_count,
eval_list->text_objects,
sizeof(struct text_object) *
eval_list->text_object_count);
(*list)->text_object_count += eval_list->text_object_count;
free(eval_list->text_objects);
free(eval_list);
}
free(args_dup);
free(template_dup);
free(eval_text);
for (i = 0; i < argcnt; i++)
free(argsp[i]);
free(argsp);
return eval_text;
}
static char *find_and_replace_templates(const char *inbuf)
{
char *outbuf, *indup, *p, *o, *templ, *args, *tmpl_out;
int stack, outlen;
outlen = strlen(inbuf) + 1;
o = outbuf = calloc(outlen, sizeof(char));
memset(outbuf, 0, outlen * sizeof(char));
p = indup = strdup(inbuf);
while (*p) {
while (*p && *p != '$')
*(o++) = *(p++);
if (!(*p))
break;
if (strncmp(p, "$template", 9) && strncmp(p, "${template", 10)) {
*(o++) = *(p++);
continue;
}
if (*(p + 1) == '{') {
templ = p + 2;
while (*p != ' ')
p++;
*(p++) = '\0';
args = p;
stack = 1;
while (stack > 0) {
p++;
if (*p == '{')
stack++;
else if (*p == '}')
stack--;
}
*(p++) = '\0';
} else {
templ = p + 1;
while (*p != ' ')
p++;
*(p++) = '\0';
args = NULL;
}
tmpl_out = handle_template(templ, args);
if (tmpl_out) {
outlen += strlen(tmpl_out);
outbuf = realloc(outbuf, outlen * sizeof(char));
strcat (outbuf, tmpl_out);
free(tmpl_out);
o = outbuf + strlen(outbuf);
} else {
ERR("failed to handle template '%s' with args '%s'", templ, args);
}
}
*o = '\0';
outbuf = realloc(outbuf, (strlen(outbuf) + 1) * sizeof(char));
free(indup);
return outbuf;
}
static int text_contains_templates(const char *text)
{
if (strcasestr(text, "${template") != NULL)
return 1;
if (strcasestr(text, "$template") != NULL)
return 1;
return 0;
}
@ -4377,9 +4423,21 @@ static struct text_object_list *extract_variable_text_internal(const char *const
char *p, *s, *orig_p;
long line;
p = strndup(const_p, max_user_text);
p = strndup(const_p, max_user_text - 1);
while (text_contains_templates(p)) {
char *tmp;
tmp = find_and_replace_templates(p);
free(p);
p = tmp;
}
s = orig_p = p;
if (strcmp(p, const_p)) {
DBGP("replaced all templates in text: input is\n'%s'\noutput is\n'%s'", const_p, p);
} else {
DBGP("no templates to replace");
}
retval = malloc(sizeof(struct text_object_list));
memset(retval, 0, sizeof(struct text_object_list));
retval->text_object_count = 0;
@ -4475,24 +4533,18 @@ static struct text_object_list *extract_variable_text_internal(const char *const
tmp_p++;
}
if (!strncmp(buf, "template", 8)) {
if (handle_template_object(&retval, allow_threaded, buf, arg))
printf("Error: handling template '%s' failed for args '%s'\n", buf, arg);
} else {
// create new object
obj = construct_text_object(buf, arg,
retval->text_object_count, retval->text_objects, line, allow_threaded);
if (obj != NULL) {
// allocate memory for the object
retval->text_objects = realloc(retval->text_objects,
sizeof(struct text_object) *
(retval->text_object_count + 1));
// assign the new object to the end of the list.
memcpy(
&retval->text_objects[retval->text_object_count++],
obj, sizeof(struct text_object));
free(obj);
}
obj = construct_text_object(buf, arg,
retval->text_object_count, retval->text_objects, line, allow_threaded);
if (obj != NULL) {
// allocate memory for the object
retval->text_objects = realloc(retval->text_objects,
sizeof(struct text_object) *
(retval->text_object_count + 1));
// assign the new object to the end of the list.
memcpy(
&retval->text_objects[retval->text_object_count++],
obj, sizeof(struct text_object));
free(obj);
}
}
continue;