]> gitweb.ps.run Git - ps-cgit/blobdiff - parsing.c
Convert subject and message with iconv_msg.
[ps-cgit] / parsing.c
index 30e7648958789fb72930c405d1ba8d8c5a382f27..c7310847561d8d2a7edc11912c2991d12600fef9 100644 (file)
--- a/parsing.c
+++ b/parsing.c
@@ -6,6 +6,8 @@
  *   (see COPYING for full license text)
  */
 
  *   (see COPYING for full license text)
  */
 
+#include <iconv.h>
+
 #include "cgit.h"
 
 int next_char(FILE *f)
 #include "cgit.h"
 
 int next_char(FILE *f)
@@ -176,6 +178,62 @@ void cgit_parse_url(const char *url)
        }
 }
 
        }
 }
 
+static char *iconv_msg(char *msg, const char *encoding)
+{
+       iconv_t msg_conv = iconv_open(PAGE_ENCODING, encoding);
+       size_t inlen = strlen(msg);
+       char *in;
+       char *out;
+       size_t inleft;
+       size_t outleft;
+       char *buf;
+       char *ret;
+       size_t buf_sz;
+       int again, fail;
+
+       if(msg_conv == (iconv_t)-1)
+               return NULL;
+
+       buf_sz = inlen * 2;
+       buf = xmalloc(buf_sz+1);
+       do {
+               in = msg;
+               inleft = inlen;
+
+               out = buf;
+               outleft = buf_sz;
+               iconv(msg_conv, &in, &inleft, &out, &outleft);
+
+               if(inleft == 0) {
+                       fail = 0;
+                       again = 0;
+               } else if(inleft != 0 && errno == E2BIG) {
+                       fail = 0;
+                       again = 1;
+
+                       buf_sz *= 2;
+                       free(buf);
+                       buf = xmalloc(buf_sz+1);
+               } else {
+                       fail = 1;
+                       again = 0;
+               }
+       } while(again && !fail);
+
+       if(fail) {
+               free(buf);
+               ret = NULL;
+       } else {
+               buf = xrealloc(buf, out - buf);
+               *out = 0;
+               ret = buf;
+       }
+
+       iconv_close(msg_conv);
+
+       return ret;
+}
+
 char *substr(const char *head, const char *tail)
 {
        char *buf;
 char *substr(const char *head, const char *tail)
 {
        char *buf;
@@ -199,6 +257,7 @@ struct commitinfo *cgit_parse_commit(struct commit *commit)
        ret->committer_email = NULL;
        ret->subject = NULL;
        ret->msg = NULL;
        ret->committer_email = NULL;
        ret->subject = NULL;
        ret->msg = NULL;
+       ret->msg_encoding = NULL;
 
        if (p == NULL)
                return ret;
 
        if (p == NULL)
                return ret;
@@ -233,6 +292,14 @@ struct commitinfo *cgit_parse_commit(struct commit *commit)
                p = strchr(t, '\n') + 1;
        }
 
                p = strchr(t, '\n') + 1;
        }
 
+       if (!strncmp(p, "encoding ", 9)) {
+               p += 9;
+               t = strchr(p, '\n') + 1;
+               ret->msg_encoding = substr(p, t);
+               p = t;
+       } else
+               ret->msg_encoding = xstrdup(PAGE_ENCODING);
+
        while (*p && (*p != '\n'))
                p = strchr(p, '\n') + 1; // skip unknown header fields
 
        while (*p && (*p != '\n'))
                p = strchr(p, '\n') + 1; // skip unknown header fields
 
@@ -253,6 +320,20 @@ struct commitinfo *cgit_parse_commit(struct commit *commit)
        } else
                ret->subject = substr(p, p+strlen(p));
 
        } else
                ret->subject = substr(p, p+strlen(p));
 
+       if(strcmp(ret->msg_encoding, PAGE_ENCODING)) {
+               t = iconv_msg(ret->subject, ret->msg_encoding);
+               if(t) {
+                       free(ret->subject);
+                       ret->subject = t;
+               }
+
+               t = iconv_msg(ret->msg, ret->msg_encoding);
+               if(t) {
+                       free(ret->msg);
+                       ret->msg = t;
+               }
+       }
+
        return ret;
 }
 
        return ret;
 }