summaryrefslogtreecommitdiffstats
path: root/src/blob.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/blob.c')
-rw-r--r--src/blob.c44
1 files changed, 43 insertions, 1 deletions
diff --git a/src/blob.c b/src/blob.c
index 41e8d75..4ddc156 100644
--- a/src/blob.c
+++ b/src/blob.c
@@ -139,6 +139,24 @@ int blob_icmp(blob_t a, blob_t b)
return strncasecmp(a.ptr, b.ptr, a.len);
}
+int blob_find_char(blob_t blob, char c)
+{
+ int i;
+ char *ptr = blob.ptr;
+ for (i = 0; i < blob.len; i++)
+ if (ptr[i] == c)
+ return i;
+ return -1;
+}
+
+unsigned char blob_read_hexbyte(blob_t *b, int i)
+{
+ if (b->len >= i+2)
+ return (dx(b->ptr[i]) << 4) + dx(b->ptr[i+1]);
+ else
+ return 0;
+}
+
void blob_lowercase(blob_t blob)
{
int i;
@@ -147,6 +165,30 @@ void blob_lowercase(blob_t blob)
ptr[i] = tolower(ptr[i]);
}
+void blob_percent_decode(blob_t *blob)
+{
+ int i = blob_find_char(*blob, '%');
+
+ if (i >= 0) {
+ int dest = i;
+ int len = blob->len;
+ char* ptr = blob->ptr;
+
+ for ( ; i < blob->len; i++, dest++) {
+ if (ptr[i] == '%') {
+ ptr[dest] = blob_read_hexbyte(blob, i+1);
+ i += 2;
+ len -= 2;
+ }
+ else {
+ ptr[dest] = ptr[i];
+ }
+ }
+
+ blob->len = len;
+ }
+}
+
int blob_pull_inet_addr(blob_t *b, struct in_addr *saddr)
{
unsigned long ip = 0;
@@ -280,7 +322,7 @@ void blob_push_urldecode(blob_t *to, blob_t url)
blob_expand_head(to, orig, '/');
blob_expand_head_bytes(to, 1); /* back up past the '/' separator */
} else {
- /* copy decoded; FIXME decode percent encoding */
+ blob_percent_decode(&b);
blob_push_byte(to, '/');
blob_push(to, b);
}