2015-12-20 14:54:54 +03:00
|
|
|
#include "ruby.h"
|
|
|
|
#include "ruby/encoding.h"
|
|
|
|
|
|
|
|
static VALUE rb_cCGI, rb_mUtil, rb_mEscape;
|
|
|
|
|
|
|
|
static void
|
|
|
|
html_escaped_cat(VALUE str, char c)
|
|
|
|
{
|
|
|
|
switch (c) {
|
|
|
|
case '\'':
|
|
|
|
rb_str_cat_cstr(str, "'");
|
|
|
|
break;
|
|
|
|
case '&':
|
|
|
|
rb_str_cat_cstr(str, "&");
|
|
|
|
break;
|
|
|
|
case '"':
|
|
|
|
rb_str_cat_cstr(str, """);
|
|
|
|
break;
|
|
|
|
case '<':
|
|
|
|
rb_str_cat_cstr(str, "<");
|
|
|
|
break;
|
|
|
|
case '>':
|
|
|
|
rb_str_cat_cstr(str, ">");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-21 23:40:02 +03:00
|
|
|
static inline void
|
|
|
|
preserve_original_state(VALUE orig, VALUE dest)
|
|
|
|
{
|
|
|
|
rb_enc_associate(dest, rb_enc_get(orig));
|
|
|
|
|
2015-12-22 08:31:31 +03:00
|
|
|
RB_OBJ_INFECT_RAW(dest, orig);
|
2015-12-21 23:40:02 +03:00
|
|
|
}
|
|
|
|
|
2015-12-20 14:54:54 +03:00
|
|
|
static VALUE
|
|
|
|
optimized_escape_html(VALUE str)
|
|
|
|
{
|
|
|
|
long i, len, modified = 0, beg = 0;
|
|
|
|
VALUE dest;
|
|
|
|
const char *cstr;
|
|
|
|
|
|
|
|
len = RSTRING_LEN(str);
|
|
|
|
cstr = RSTRING_PTR(str);
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
switch (cstr[i]) {
|
|
|
|
case '\'':
|
|
|
|
case '&':
|
|
|
|
case '"':
|
|
|
|
case '<':
|
|
|
|
case '>':
|
|
|
|
if (!modified) {
|
|
|
|
modified = 1;
|
|
|
|
dest = rb_str_buf_new(len);
|
|
|
|
}
|
|
|
|
|
|
|
|
rb_str_cat(dest, cstr + beg, i - beg);
|
|
|
|
beg = i + 1;
|
|
|
|
|
|
|
|
html_escaped_cat(dest, cstr[i]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (modified) {
|
|
|
|
rb_str_cat(dest, cstr + beg, len - beg);
|
2015-12-21 23:40:02 +03:00
|
|
|
preserve_original_state(str, dest);
|
2015-12-20 14:54:54 +03:00
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
else {
|
2015-12-22 08:31:31 +03:00
|
|
|
return rb_str_dup(str);
|
2015-12-20 14:54:54 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* call-seq:
|
|
|
|
* CGI.escapeHTML(string) -> string
|
|
|
|
*
|
|
|
|
* Returns HTML-escaped string.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static VALUE
|
|
|
|
cgiesc_escape_html(VALUE self, VALUE str)
|
|
|
|
{
|
|
|
|
StringValue(str);
|
|
|
|
|
|
|
|
if (rb_enc_str_asciicompat_p(str)) {
|
|
|
|
return optimized_escape_html(str);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return rb_call_super(1, &str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Init_escape(void)
|
|
|
|
{
|
|
|
|
rb_cCGI = rb_define_class("CGI", rb_cObject);
|
|
|
|
rb_mEscape = rb_define_module_under(rb_cCGI, "Escape");
|
|
|
|
rb_mUtil = rb_define_module_under(rb_cCGI, "Util");
|
|
|
|
rb_define_method(rb_mEscape, "escapeHTML", cgiesc_escape_html, 1);
|
|
|
|
rb_prepend_module(rb_mUtil, rb_mEscape);
|
|
|
|
rb_extend_object(rb_cCGI, rb_mEscape);
|
|
|
|
}
|