fragment: add new option to control escaping shell arguments

pull/116/head
William Pitcock 2017-02-25 15:32:49 -06:00
parent ae42261c3f
commit 01fe5743ea
3 changed files with 71 additions and 13 deletions

View File

@ -52,6 +52,7 @@ which is composable, mergeable and reorderable.
Calculates the required memory to store a `fragment list` when rendered as a string. Calculates the required memory to store a `fragment list` when rendered as a string.
:param pkgconf_list_t* list: The `fragment list` being rendered. :param pkgconf_list_t* list: The `fragment list` being rendered.
:param bool escape: Whether or not to escape special shell characters.
:return: the amount of bytes required to represent the `fragment list` when rendered :return: the amount of bytes required to represent the `fragment list` when rendered
:rtype: size_t :rtype: size_t
@ -62,6 +63,7 @@ which is composable, mergeable and reorderable.
:param pkgconf_list_t* list: The `fragment list` being rendered. :param pkgconf_list_t* list: The `fragment list` being rendered.
:param char* buf: The buffer to render the fragment list into. :param char* buf: The buffer to render the fragment list into.
:param size_t buflen: The length of the buffer. :param size_t buflen: The length of the buffer.
:param bool escape: Whether or not to escape special shell characters.
:return: nothing :return: nothing
.. c:function:: char *pkgconf_fragment_render(const pkgconf_list_t *list) .. c:function:: char *pkgconf_fragment_render(const pkgconf_list_t *list)
@ -69,6 +71,7 @@ which is composable, mergeable and reorderable.
Allocate memory and render a `fragment list` into it. Allocate memory and render a `fragment list` into it.
:param pkgconf_list_t* list: The `fragment list` being rendered. :param pkgconf_list_t* list: The `fragment list` being rendered.
:param bool escape: Whether or not to escape special shell characters.
:return: An allocated string containing the rendered `fragment list`. :return: An allocated string containing the rendered `fragment list`.
:rtype: char * :rtype: char *

View File

@ -372,8 +372,42 @@ pkgconf_fragment_filter(const pkgconf_client_t *client, pkgconf_list_t *dest, pk
} }
} }
static inline char *
fragment_escape(const char *src)
{
ssize_t outlen = strlen(src) + 10;
char *out = calloc(outlen, 1);
char *dst = out;
while (*src)
{
if ((*src < ' ') ||
(*src > ' ' && *src < '$') ||
(*src > '$' && *src < '(') ||
(*src > ')' && *src < '+') ||
(*src > ':' && *src < '=') ||
(*src > '=' && *src < '@') ||
(*src > 'Z' && *src < '^') ||
(*src == '`') ||
(*src > 'z' && *src < '~') ||
(*src > '~'))
*dst++ = '\\';
*dst++ = *src++;
if ((ptrdiff_t)(dst - out) + 2 > outlen)
{
outlen *= 2;
out = realloc(out, outlen);
}
}
*dst = 0;
return out;
}
static inline size_t static inline size_t
pkgconf_fragment_len(const pkgconf_fragment_t *frag) pkgconf_fragment_len(const pkgconf_fragment_t *frag, bool escape)
{ {
size_t len = 1; size_t len = 1;
@ -381,7 +415,16 @@ pkgconf_fragment_len(const pkgconf_fragment_t *frag)
len += 2; len += 2;
if (frag->data != NULL) if (frag->data != NULL)
{
if (!escape)
len += strlen(frag->data); len += strlen(frag->data);
else
{
char *tmp = fragment_escape(frag->data);
len += strlen(tmp);
free(tmp);
}
}
return len; return len;
} }
@ -394,11 +437,12 @@ pkgconf_fragment_len(const pkgconf_fragment_t *frag)
* Calculates the required memory to store a `fragment list` when rendered as a string. * Calculates the required memory to store a `fragment list` when rendered as a string.
* *
* :param pkgconf_list_t* list: The `fragment list` being rendered. * :param pkgconf_list_t* list: The `fragment list` being rendered.
* :param bool escape: Whether or not to escape special shell characters.
* :return: the amount of bytes required to represent the `fragment list` when rendered * :return: the amount of bytes required to represent the `fragment list` when rendered
* :rtype: size_t * :rtype: size_t
*/ */
size_t size_t
pkgconf_fragment_render_len(const pkgconf_list_t *list) pkgconf_fragment_render_len(const pkgconf_list_t *list, bool escape)
{ {
size_t out = 1; /* trailing nul */ size_t out = 1; /* trailing nul */
pkgconf_node_t *node; pkgconf_node_t *node;
@ -406,7 +450,7 @@ pkgconf_fragment_render_len(const pkgconf_list_t *list)
PKGCONF_FOREACH_LIST_ENTRY(list->head, node) PKGCONF_FOREACH_LIST_ENTRY(list->head, node)
{ {
const pkgconf_fragment_t *frag = node->data; const pkgconf_fragment_t *frag = node->data;
out += pkgconf_fragment_len(frag); out += pkgconf_fragment_len(frag, escape);
} }
return out; return out;
@ -422,10 +466,11 @@ pkgconf_fragment_render_len(const pkgconf_list_t *list)
* :param pkgconf_list_t* list: The `fragment list` being rendered. * :param pkgconf_list_t* list: The `fragment list` being rendered.
* :param char* buf: The buffer to render the fragment list into. * :param char* buf: The buffer to render the fragment list into.
* :param size_t buflen: The length of the buffer. * :param size_t buflen: The length of the buffer.
* :param bool escape: Whether or not to escape special shell characters.
* :return: nothing * :return: nothing
*/ */
void void
pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t buflen) pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t buflen, bool escape)
{ {
pkgconf_node_t *node; pkgconf_node_t *node;
char *bptr = buf; char *bptr = buf;
@ -437,7 +482,7 @@ pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t buflen
const pkgconf_fragment_t *frag = node->data; const pkgconf_fragment_t *frag = node->data;
size_t buf_remaining = buflen - (bptr - buf); size_t buf_remaining = buflen - (bptr - buf);
if (pkgconf_fragment_len(frag) > buf_remaining) if (pkgconf_fragment_len(frag, escape) > buf_remaining)
break; break;
if (frag->type) if (frag->type)
@ -447,7 +492,16 @@ pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t buflen
} }
if (frag->data) if (frag->data)
{
if (!escape)
bptr += pkgconf_strlcpy(bptr, frag->data, buf_remaining); bptr += pkgconf_strlcpy(bptr, frag->data, buf_remaining);
else
{
char *tmp = fragment_escape(frag->data);
bptr += pkgconf_strlcpy(bptr, tmp, buf_remaining);
free(tmp);
}
}
*bptr++ = ' '; *bptr++ = ' ';
} }
@ -463,16 +517,17 @@ pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t buflen
* Allocate memory and render a `fragment list` into it. * Allocate memory and render a `fragment list` into it.
* *
* :param pkgconf_list_t* list: The `fragment list` being rendered. * :param pkgconf_list_t* list: The `fragment list` being rendered.
* :param bool escape: Whether or not to escape special shell characters.
* :return: An allocated string containing the rendered `fragment list`. * :return: An allocated string containing the rendered `fragment list`.
* :rtype: char * * :rtype: char *
*/ */
char * char *
pkgconf_fragment_render(const pkgconf_list_t *list) pkgconf_fragment_render(const pkgconf_list_t *list, bool escape)
{ {
size_t buflen = pkgconf_fragment_render_len(list); size_t buflen = pkgconf_fragment_render_len(list, escape);
char *buf = calloc(1, buflen); char *buf = calloc(1, buflen);
pkgconf_fragment_render_buf(list, buf, buflen); pkgconf_fragment_render_buf(list, buf, buflen, escape);
return buf; return buf;
} }

View File

@ -268,9 +268,9 @@ void pkgconf_fragment_copy(const pkgconf_client_t *client, pkgconf_list_t *list,
void pkgconf_fragment_delete(pkgconf_list_t *list, pkgconf_fragment_t *node); void pkgconf_fragment_delete(pkgconf_list_t *list, pkgconf_fragment_t *node);
void pkgconf_fragment_free(pkgconf_list_t *list); void pkgconf_fragment_free(pkgconf_list_t *list);
void pkgconf_fragment_filter(const pkgconf_client_t *client, pkgconf_list_t *dest, pkgconf_list_t *src, pkgconf_fragment_filter_func_t filter_func, void *data); void pkgconf_fragment_filter(const pkgconf_client_t *client, pkgconf_list_t *dest, pkgconf_list_t *src, pkgconf_fragment_filter_func_t filter_func, void *data);
size_t pkgconf_fragment_render_len(const pkgconf_list_t *list); size_t pkgconf_fragment_render_len(const pkgconf_list_t *list, bool escape);
void pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t len); void pkgconf_fragment_render_buf(const pkgconf_list_t *list, char *buf, size_t len, bool escape);
char *pkgconf_fragment_render(const pkgconf_list_t *list); char *pkgconf_fragment_render(const pkgconf_list_t *list, bool escape);
bool pkgconf_fragment_has_system_dir(const pkgconf_client_t *client, const pkgconf_fragment_t *frag); bool pkgconf_fragment_has_system_dir(const pkgconf_client_t *client, const pkgconf_fragment_t *frag);
/* fileio.c */ /* fileio.c */