View Single Post
  #5  
Old 04-30-2010, 09:51 PM
juanune2 juanune2 is offline
 
Join Date: Apr 2010
Posts: 1
Благодарил(а): 0 раз(а)
Поблагодарили: 0 раз(а) в 0 сообщениях
Default

I hate replying to a month-old thread, but I wasn't even a vBulletin user a month ago. : )

I ran into the same problem, and... really can't believe that it exists. This is a glaring hole and you really couldn't use it in a production environment without fixing (at least not as more than a toy). I chose vBulletin over other solutions because of some of the power that it offers wrt communities, so I won't go into a rant... as I still believe I made the right choice.

I'm honestly not sure why this exists. I've found the issue not just on smart quotes, but also on things like ….

I've done some investigation through the code and have determined that there isn't a good way of doing this without touching half of the code in the system, as there is no general-purpose text-cleaning function. Note that since they're high ASCII values, modyifing these is something that should be done immediately upon ingestion, but there doesn't appear to be a good way of doing that. There are even too many ways to enter text from the client-side, and no general text parser in the script (not that you should trust client parsing). It will also be language dependent.

That being said, there is still a bad way of doing it, and I've implemented a brute-force method. Note... I have only done cursory testing with this. I have found no issues as of yet, but please make sure that you do some sanity checks.

STEP 1: [REQUIRED!]
#1 on the list of things to do before using my hack is to move ALL attachments, images, profile images, and all other binary data out of the database and into the filesystem. I have no issues with this, since I'm one of those guys that doesn't believe that you should ever have this kind of data in there in the first place.

Here's info on how, why/why not to do that:
vbulletin docs for moving attachments
and
vbulletin docs for moving user pictures

STEP 2
Create a patch file. My file is called functions_custom.php, and lives in a 'custom' directory off of my root. It is a bit verbose (I chose a format that could be easily read), but there the full text:
PHP Code:
<?php
function convert_extraspecial_chars($string)
{
// uncomment the following line to turn the functionality off.
//return $string;  

     
$search = array();
     
$replace = array();

     
$search[] = chr(130);
    
$replace[] = '\'';
     
$search[] = chr(131);
    
$replace[] = '';
     
$search[] = chr(132);
    
$replace[] = '';
     
$search[] = chr(133);
    
$replace[] = '...';
     
$search[] = chr(134);
    
$replace[] = '';
     
$search[] = chr(135);
    
$replace[] = '';
     
$search[] = chr(136);
    
$replace[] = '';
     
$search[] = chr(137);
    
$replace[] = '';
     
$search[] = chr(138);
    
$replace[] = '';
     
$search[] = chr(139);
    
$replace[] = '';
     
$search[] = chr(140);
    
$replace[] = '';
     
$search[] = chr(174);
    
$replace[] = '(r)';
     
$search[] = chr(175);
    
$replace[] = '(c)';
    
    
    
     
$search[] = chr(145);
    
$replace[] = '\'';
     
$search[] = chr(146);
    
$replace[] = '\'';
     
$search[] = chr(147);
    
$replace[] = '"';
     
$search[] = chr(148);
    
$replace[] = '"';
     
$search[] = chr(149);
    
$replace[] = '"';
     
$search[] = chr(150);
    
$replace[] = '*';
     
$search[] = chr(151);
    
$replace[] = '-';
     
$search[] = chr(152);
    
$replace[] = '-';
     
$search[] = chr(153);
    
$replace[] = '';
     
$search[] = chr(154);
    
$replace[] = 'tm';
     
$search[] = chr(155);
    
$replace[] = '';
     
$search[] = chr(156);
    
$replace[] = '\'';
 
 
return 
str_replace($search$replace$string);
}  
?>
STEP 3
Hook your patch into the only place that seems to be a culmination point for input parsing, which is escape_string() inside of class_core.php (around line 717):
PHP Code:
    function escape_string($string)
    {
        require_once(
'../custom/functions_custom.php');
        
$string convert_extraspecial_chars($string);
        if (
$this->functions['escape_string'] == $this->functions['real_escape_string'])
        {
            return 
$this->functions['escape_string']($string$this->connection_master);
        }
        else
        {
            return 
$this->functions['escape_string']($string);
        }
    } 
What this does is force an extra cleaning pass for all info that passes into the DB, stripping certain high-ascii values. If you tried to skip step 1, you'll wind up corrupting 99% of binary files uploaded to the system... so don't do that.

I can't comment on general usage, since I've only worked through this on one pre-production installation, but it is working nicely here. As such, put this through a test pass before using it. If it works, great... put some comments in here. If it doesn't... maybe I can offer some assistance.

-- j

--------------- Added [DATE]1272684214[/DATE] at [TIME]1272684214[/TIME] ---------------

Ok, after dealing with this a bit more...

I changed the map file so that the mappings strip out high ascii characters instead of putting them through an HTML entities setup. Although... the real problem lies in how the database calls are structured. Again, the fix would mean changing every call in the system, and I'm sure that nobody is keen to do that.
Reply With Quote
 
X vBulletin 3.8.12 by vBS Debug Information
  • Page Generation 0.01285 seconds
  • Memory Usage 1,837KB
  • Queries Executed 11 (?)
More Information
Template Usage:
  • (1)SHOWTHREAD_SHOWPOST
  • (1)ad_footer_end
  • (1)ad_footer_start
  • (1)ad_header_end
  • (1)ad_header_logo
  • (1)ad_navbar_below
  • (2)bbcode_php
  • (1)footer
  • (1)gobutton
  • (1)header
  • (1)headinclude
  • (6)option
  • (1)post_thanks_box
  • (1)post_thanks_button
  • (1)post_thanks_javascript
  • (1)post_thanks_navbar_search
  • (1)post_thanks_postbit_info
  • (1)postbit
  • (1)postbit_onlinestatus
  • (1)postbit_wrapper
  • (1)spacer_close
  • (1)spacer_open 

Phrase Groups Available:
  • global
  • postbit
  • reputationlevel
  • showthread
Included Files:
  • ./showpost.php
  • ./global.php
  • ./includes/init.php
  • ./includes/class_core.php
  • ./includes/config.php
  • ./includes/functions.php
  • ./includes/class_hook.php
  • ./includes/modsystem_functions.php
  • ./includes/functions_bigthree.php
  • ./includes/class_postbit.php
  • ./includes/class_bbcode.php
  • ./includes/functions_reputation.php
  • ./includes/functions_post_thanks.php 

Hooks Called:
  • init_startup
  • init_startup_session_setup_start
  • init_startup_session_setup_complete
  • cache_permissions
  • fetch_postinfo_query
  • fetch_postinfo
  • fetch_threadinfo_query
  • fetch_threadinfo
  • fetch_foruminfo
  • style_fetch
  • cache_templates
  • global_start
  • parse_templates
  • global_setup_complete
  • showpost_start
  • bbcode_fetch_tags
  • bbcode_create
  • postbit_factory
  • showpost_post
  • postbit_display_start
  • post_thanks_function_post_thanks_off_start
  • post_thanks_function_post_thanks_off_end
  • post_thanks_function_fetch_thanks_start
  • post_thanks_function_fetch_thanks_end
  • post_thanks_function_thanked_already_start
  • post_thanks_function_thanked_already_end
  • fetch_musername
  • postbit_imicons
  • bbcode_parse_start
  • bbcode_parse_complete_precache
  • bbcode_parse_complete
  • postbit_display_complete
  • post_thanks_function_can_thank_this_post_start
  • showpost_complete