- bug fix: comments_forall and category commentable were not checked during
POST and a comment could be inserted - feature 524: anti-spam: - check number of links - check ip address against spamhaus.org block list - action when comment is qualified spam (needs validation or reject) - so far everything is in the config file git-svn-id: http://piwigo.org/svn/trunk@1610 68402e56-0260-453c-a942-63ccdbb3a9ee
This commit is contained in:
parent
221d839769
commit
31e312028c
5 changed files with 176 additions and 84 deletions
|
@ -86,6 +86,17 @@ $conf['top_number'] = 15;
|
||||||
// anti-flood_time : number of seconds between 2 comments : 0 to disable
|
// anti-flood_time : number of seconds between 2 comments : 0 to disable
|
||||||
$conf['anti-flood_time'] = 60;
|
$conf['anti-flood_time'] = 60;
|
||||||
|
|
||||||
|
// qualified spam comments are not registered (false will register them
|
||||||
|
// but they will require admin validation)
|
||||||
|
$conf['comment_spam_reject'] = true;
|
||||||
|
|
||||||
|
// maximum number of links in a comment before it is qualified spam
|
||||||
|
$conf['comment_spam_max_links'] = 3;
|
||||||
|
|
||||||
|
// if the ip address of a comenteer is in spamhaus.org block list, the
|
||||||
|
// comment is qualified spam
|
||||||
|
$conf['comment_spam_check_ip'] = false;
|
||||||
|
|
||||||
// calendar_datefield : date field of table "images" used for calendar
|
// calendar_datefield : date field of table "images" used for calendar
|
||||||
// catgory
|
// catgory
|
||||||
$conf['calendar_datefield'] = 'date_creation';
|
$conf['calendar_datefield'] = 'date_creation';
|
||||||
|
|
|
@ -29,85 +29,46 @@
|
||||||
* This file is included by the picture page to manage user comments
|
* This file is included by the picture page to manage user comments
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
//returns string action to perform on a new comment: validate, moderate, reject
|
||||||
if ( isset( $_POST['content'] ) and !empty($_POST['content']) )
|
function user_comment_check($action, $comment, $picture)
|
||||||
{
|
{
|
||||||
$register_comment = true;
|
global $conf,$user;
|
||||||
$author = !empty($_POST['author'])?$_POST['author']:$lang['guest'];
|
|
||||||
// if a guest try to use the name of an already existing user, he must be
|
if ($action=='reject')
|
||||||
// rejected
|
return $action;
|
||||||
if ( $author != $user['username'] )
|
|
||||||
|
$my_action = $conf['comment_spam_reject'] ? 'reject':'moderate';
|
||||||
|
if ($action==$my_action)
|
||||||
|
return $action;
|
||||||
|
|
||||||
|
// we do here only BASIC spam check (plugins can do more)
|
||||||
|
if ( !$user['is_the_guest'] )
|
||||||
|
return $action;
|
||||||
|
|
||||||
|
$link_count = preg_match_all( '/https?:\/\//',
|
||||||
|
$comment['content'], $matches);
|
||||||
|
|
||||||
|
if ( $link_count>$conf['comment_spam_max_links'] )
|
||||||
|
return $my_action;
|
||||||
|
|
||||||
|
if ( isset($comment['ip']) and $conf['comment_spam_check_ip'] )
|
||||||
{
|
{
|
||||||
$query = 'SELECT COUNT(*) AS user_exists';
|
$rev_ip = implode( '.', array_reverse( explode('.',$comment['ip']) ) );
|
||||||
$query.= ' FROM '.USERS_TABLE;
|
$lookup = $rev_ip . '.sbl-xbl.spamhaus.org.';
|
||||||
$query.= ' WHERE '.$conf['user_fields']['username']." = '".$author."'";
|
$res = gethostbyname( $lookup );
|
||||||
$query.= ';';
|
if ( $lookup != $res )
|
||||||
$row = mysql_fetch_array( pwg_query( $query ) );
|
return $my_action;
|
||||||
if ( $row['user_exists'] == 1 )
|
|
||||||
{
|
|
||||||
$template->assign_block_vars(
|
|
||||||
'information',
|
|
||||||
array('INFORMATION'=>$lang['comment_user_exists']));
|
|
||||||
$register_comment = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( $register_comment )
|
return $action;
|
||||||
{
|
|
||||||
// anti-flood system
|
|
||||||
$reference_date = time() - $conf['anti-flood_time'];
|
|
||||||
$query = 'SELECT id FROM '.COMMENTS_TABLE;
|
|
||||||
$query.= ' WHERE date > FROM_UNIXTIME('.$reference_date.')';
|
|
||||||
$query.= " AND author = '".$author."'";
|
|
||||||
$query.= ';';
|
|
||||||
if ( mysql_num_rows( pwg_query( $query ) ) == 0
|
|
||||||
or $conf['anti-flood_time'] == 0 )
|
|
||||||
{
|
|
||||||
list($dbnow) = mysql_fetch_row(pwg_query('SELECT NOW();'));
|
|
||||||
|
|
||||||
$data = array();
|
|
||||||
$data{'author'} = $author;
|
|
||||||
$data{'date'} = $dbnow;
|
|
||||||
$data{'image_id'} = $page['image_id'];
|
|
||||||
$data{'content'} = htmlspecialchars( $_POST['content'], ENT_QUOTES);
|
|
||||||
|
|
||||||
if (!$conf['comments_validation'] or is_admin())
|
|
||||||
{
|
|
||||||
$data{'validated'} = 'true';
|
|
||||||
$data{'validation_date'} = $dbnow;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$data{'validated'} = 'false';
|
|
||||||
}
|
|
||||||
|
|
||||||
include_once(PHPWG_ROOT_PATH.'admin/include/functions.php');
|
|
||||||
$fields = array('author', 'date', 'image_id', 'content', 'validated',
|
|
||||||
'validation_date');
|
|
||||||
mass_inserts(COMMENTS_TABLE, $fields, array($data));
|
|
||||||
|
|
||||||
// information message
|
|
||||||
$message = $lang['comment_added'];
|
|
||||||
|
|
||||||
if (!$conf['comments_validation'] or is_admin())
|
|
||||||
|
|
||||||
if ( $conf['comments_validation'] and !is_admin() )
|
|
||||||
{
|
|
||||||
$message.= '<br />'.$lang['comment_to_validate'];
|
|
||||||
}
|
|
||||||
$template->assign_block_vars('information',
|
|
||||||
array('INFORMATION'=>$message));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// information message
|
|
||||||
$template->assign_block_vars(
|
|
||||||
'information',
|
|
||||||
array('INFORMATION'=>$lang['comment_anti-flood']));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
add_event_handler('user_comment_check', 'user_comment_check',
|
||||||
|
EVENT_HANDLER_PRIORITY_NEUTRAL, 3);
|
||||||
|
|
||||||
|
|
||||||
// the picture is commentable if it belongs at least to one category which
|
// the picture is commentable if it belongs at least to one category which
|
||||||
// is commentable
|
// is commentable
|
||||||
$page['show_comments'] = false;
|
$page['show_comments'] = false;
|
||||||
|
@ -116,9 +77,137 @@ foreach ($related_categories as $category)
|
||||||
if ($category['commentable'] == 'true')
|
if ($category['commentable'] == 'true')
|
||||||
{
|
{
|
||||||
$page['show_comments'] = true;
|
$page['show_comments'] = true;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ( $page['show_comments'] and isset( $_POST['content'] ) )
|
||||||
|
{
|
||||||
|
if ( $user['is_the_guest'] and !$conf['comments_forall'] )
|
||||||
|
{
|
||||||
|
die ('Session expired');
|
||||||
|
}
|
||||||
|
if (!$conf['comments_validation'] or is_admin())
|
||||||
|
{
|
||||||
|
$comment_action='validate'; //one of validate, moderate, reject
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$comment_action='moderate'; //one of validate, moderate, reject
|
||||||
|
}
|
||||||
|
|
||||||
|
$_POST['content'] = trim( stripslashes($_POST['content']) );
|
||||||
|
|
||||||
|
if ( $user['is_the_guest'] )
|
||||||
|
{
|
||||||
|
$author = empty($_POST['author'])?'guest':$_POST['author'];
|
||||||
|
// if a guest try to use the name of an already existing user, he must be
|
||||||
|
// rejected
|
||||||
|
if ( $author != 'guest' )
|
||||||
|
{
|
||||||
|
$query = 'SELECT COUNT(*) AS user_exists';
|
||||||
|
$query.= ' FROM '.USERS_TABLE;
|
||||||
|
$query.= ' WHERE '.$conf['user_fields']['username']." = '".$author."'";
|
||||||
|
$query.= ';';
|
||||||
|
$row = mysql_fetch_assoc( pwg_query( $query ) );
|
||||||
|
if ( $row['user_exists'] == 1 )
|
||||||
|
{
|
||||||
|
$template->assign_block_vars(
|
||||||
|
'information',
|
||||||
|
array('INFORMATION'=>$lang['comment_user_exists']));
|
||||||
|
$comment_action='reject';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$author = $user['username'];
|
||||||
|
}
|
||||||
|
|
||||||
|
$comm = array(
|
||||||
|
'author' => $author,
|
||||||
|
'content' => $_POST['content'],
|
||||||
|
'image_id' => $page['image_id'],
|
||||||
|
'ip' => $_SERVER['REMOTE_ADDR'],
|
||||||
|
'agent' => $_SERVER['HTTP_USER_AGENT']
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($comment_action!='reject' and empty($comm['content']) )
|
||||||
|
{ // empty comment content
|
||||||
|
$comment_action='reject';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($comment_action!='reject' and $conf['anti-flood_time']>0 )
|
||||||
|
{ // anti-flood system
|
||||||
|
$reference_date = time() - $conf['anti-flood_time'];
|
||||||
|
$query = 'SELECT id FROM '.COMMENTS_TABLE;
|
||||||
|
$query.= ' WHERE date > FROM_UNIXTIME('.$reference_date.')';
|
||||||
|
$query.= " AND author = '".$comm['author']."'";
|
||||||
|
$query.= ';';
|
||||||
|
if ( mysql_num_rows( pwg_query( $query ) ) > 0 )
|
||||||
|
{
|
||||||
|
$template->assign_block_vars(
|
||||||
|
'information',
|
||||||
|
array('INFORMATION'=>$lang['comment_anti-flood']));
|
||||||
|
$comment_action='reject';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// perform more spam check
|
||||||
|
$comment_action = trigger_event('user_comment_check',
|
||||||
|
$comment_action, $comm, $picture['current']
|
||||||
|
);
|
||||||
|
|
||||||
|
if ( $comment_action!='reject' )
|
||||||
|
{
|
||||||
|
list($dbnow) = mysql_fetch_row(pwg_query('SELECT NOW();'));
|
||||||
|
|
||||||
|
$data = $comm;
|
||||||
|
$data['date'] = $dbnow;
|
||||||
|
$data['content'] = addslashes(
|
||||||
|
// this htmlpsecialchars is not good here
|
||||||
|
htmlspecialchars($comm['content'],ENT_QUOTES)
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($comment_action=='validate')
|
||||||
|
{
|
||||||
|
$data['validated'] = 'true';
|
||||||
|
$data['validation_date'] = $dbnow;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$data['validated'] = 'false';
|
||||||
|
}
|
||||||
|
|
||||||
|
include_once(PHPWG_ROOT_PATH.'admin/include/functions.php');
|
||||||
|
$fields = array('author', 'date', 'image_id', 'content', 'validated',
|
||||||
|
'validation_date');
|
||||||
|
mass_inserts(COMMENTS_TABLE, $fields, array($data));
|
||||||
|
$comm['id'] = mysql_insert_id();
|
||||||
|
|
||||||
|
// information message
|
||||||
|
$message = $lang['comment_added'];
|
||||||
|
if ($comment_action!='validate')
|
||||||
|
{
|
||||||
|
$message.= '<br />'.$lang['comment_to_validate'];
|
||||||
|
}
|
||||||
|
$template->assign_block_vars('information',
|
||||||
|
array('INFORMATION'=>$message));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
$template->assign_block_vars('information',
|
||||||
|
array('INFORMATION'=>l10n('comment_not_added') )
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// allow plugins to notify what's going on
|
||||||
|
trigger_action( 'user_comment_insertion',
|
||||||
|
array_merge($comm, array('action'=>$comment_action) )
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if ($page['show_comments'])
|
if ($page['show_comments'])
|
||||||
{
|
{
|
||||||
// number of comment for this picture
|
// number of comment for this picture
|
||||||
|
@ -204,14 +293,7 @@ SELECT id,author,date,image_id,content
|
||||||
{
|
{
|
||||||
$template->assign_block_vars('comments.add_comment', array());
|
$template->assign_block_vars('comments.add_comment', array());
|
||||||
// display author field if the user is not logged in
|
// display author field if the user is not logged in
|
||||||
if (!$user['is_the_guest'])
|
if ($user['is_the_guest'])
|
||||||
{
|
|
||||||
$template->assign_block_vars(
|
|
||||||
'comments.add_comment.author_known',
|
|
||||||
array('KNOWN_AUTHOR'=>$user['username'])
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
$template->assign_block_vars(
|
$template->assign_block_vars(
|
||||||
'comments.add_comment.author_field', array()
|
'comments.add_comment.author_field', array()
|
||||||
|
|
|
@ -413,6 +413,7 @@ $lang['comment date'] = 'comment date';
|
||||||
$lang['comment'] = 'Comment';
|
$lang['comment'] = 'Comment';
|
||||||
$lang['comment_added'] = 'Your comment has been registered';
|
$lang['comment_added'] = 'Your comment has been registered';
|
||||||
$lang['comment_anti-flood'] = 'Anti-flood system : please wait for a moment before trying to post another comment';
|
$lang['comment_anti-flood'] = 'Anti-flood system : please wait for a moment before trying to post another comment';
|
||||||
|
$lang['comment_not_added'] = 'Your comment has NOT been registered because it did not pass the validation rules';
|
||||||
$lang['comment_to_validate'] = 'An administrator must authorize your comment before it is visible.';
|
$lang['comment_to_validate'] = 'An administrator must authorize your comment before it is visible.';
|
||||||
$lang['comment_user_exists'] = 'This login is already used by another user';
|
$lang['comment_user_exists'] = 'This login is already used by another user';
|
||||||
$lang['comments'] = 'Comments';
|
$lang['comments'] = 'Comments';
|
||||||
|
|
|
@ -412,6 +412,7 @@ $lang['comment date'] = 'date du commentaire';
|
||||||
$lang['comment'] = 'Commentaire';
|
$lang['comment'] = 'Commentaire';
|
||||||
$lang['comment_added'] = 'Votre commentaire a été enregistré';
|
$lang['comment_added'] = 'Votre commentaire a été enregistré';
|
||||||
$lang['comment_anti-flood'] = 'Système anti-abus : merci de patienter avant d\'ajouter un nouveau commentaire';
|
$lang['comment_anti-flood'] = 'Système anti-abus : merci de patienter avant d\'ajouter un nouveau commentaire';
|
||||||
|
$lang['comment_not_added'] = 'Votre commentaire n\'a pas été enregistré parce qu\'il ne vérifie pas les règles de validation';
|
||||||
$lang['comment_to_validate'] = 'Un administrateur doit valider votre commentaire afin qu\'il soit visible.';
|
$lang['comment_to_validate'] = 'Un administrateur doit valider votre commentaire afin qu\'il soit visible.';
|
||||||
$lang['comment_user_exists'] = 'Ce nom d\'utilisateur est déjà pris';
|
$lang['comment_user_exists'] = 'Ce nom d\'utilisateur est déjà pris';
|
||||||
$lang['comments'] = 'Commentaires';
|
$lang['comments'] = 'Commentaires';
|
||||||
|
|
|
@ -189,9 +189,6 @@
|
||||||
<!-- BEGIN author_field -->
|
<!-- BEGIN author_field -->
|
||||||
<label>{lang:upload_author}<input type="text" name="author"></label>
|
<label>{lang:upload_author}<input type="text" name="author"></label>
|
||||||
<!-- END author_field -->
|
<!-- END author_field -->
|
||||||
<!-- BEGIN author_known -->
|
|
||||||
<input type="hidden" name="author" value="{comments.add_comment.author_known.KNOWN_AUTHOR}">
|
|
||||||
<!-- END author_known -->
|
|
||||||
<label>{lang:comment}<textarea name="content" rows="10" cols="80"></textarea></label>
|
<label>{lang:comment}<textarea name="content" rows="10" cols="80"></textarea></label>
|
||||||
<input type="submit" value="{lang:submit}">
|
<input type="submit" value="{lang:submit}">
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
|
Loading…
Reference in a new issue