mirror of
https://github.com/moodle/moodle.git
synced 2025-01-18 22:08:20 +01:00
MDL-18223 merged spam cleaner into head + removed legacy code + fixed report title
This commit is contained in:
parent
ff08798b11
commit
2e3f66d728
367
admin/report/spamcleaner/index.php
Executable file
367
admin/report/spamcleaner/index.php
Executable file
@ -0,0 +1,367 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Spam Cleaner
|
||||||
|
*
|
||||||
|
* Helps an admin to clean up spam in Moodle
|
||||||
|
*
|
||||||
|
* @version $Id$
|
||||||
|
* @authors Dongsheng Cai, Martin Dougiamas, Amr Hourani
|
||||||
|
* @license http://www.gnu.org/copyleft/gpl.html GNU Public License
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Configuration
|
||||||
|
|
||||||
|
// List of known spammy keywords, please add more here
|
||||||
|
|
||||||
|
$autokeywords = array(
|
||||||
|
"<img",
|
||||||
|
"fuck",
|
||||||
|
"casino",
|
||||||
|
"porn",
|
||||||
|
"xxx",
|
||||||
|
"cialis",
|
||||||
|
"viagra",
|
||||||
|
"poker",
|
||||||
|
"warcraft"
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
require_once('../../../config.php');
|
||||||
|
require_once($CFG->libdir.'/adminlib.php');
|
||||||
|
|
||||||
|
require_js(array('yui_dom-event', 'yui_connection', 'yui_json'));
|
||||||
|
|
||||||
|
$keyword = optional_param('keyword', '', PARAM_RAW);
|
||||||
|
$autodetect = optional_param('autodetect', '', PARAM_RAW);
|
||||||
|
$del = optional_param('del', '', PARAM_RAW);
|
||||||
|
$delall = optional_param('delall', '', PARAM_RAW);
|
||||||
|
$ignore = optional_param('ignore', '', PARAM_RAW);
|
||||||
|
$reset = optional_param('reset', '', PARAM_RAW);
|
||||||
|
$id = optional_param('id', '', PARAM_INT);
|
||||||
|
|
||||||
|
require_login();
|
||||||
|
admin_externalpage_setup('reportspamcleaner');
|
||||||
|
|
||||||
|
// Implement some AJAX calls
|
||||||
|
|
||||||
|
// Delete one user
|
||||||
|
if (!empty($del) && confirm_sesskey() && ($id != $USER->id)) {
|
||||||
|
if (isset($SESSION->users_result[$id])) {
|
||||||
|
$user = $SESSION->users_result[$id];
|
||||||
|
if (delete_user($user)) {
|
||||||
|
unset($SESSION->users_result[$id]);
|
||||||
|
echo json_encode(true);
|
||||||
|
} else {
|
||||||
|
echo json_encode(false);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
echo json_encode(false);
|
||||||
|
}
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete lots of users
|
||||||
|
if (!empty($delall) && confirm_sesskey()) {
|
||||||
|
if (!empty($SESSION->users_result)) {
|
||||||
|
foreach ($SESSION->users_result as $userid => $user) {
|
||||||
|
if ($userid != $USER->id) {
|
||||||
|
if (delete_user($user)) {
|
||||||
|
unset($SESSION->users_result[$userid]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
echo json_encode(true);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty($ignore)) {
|
||||||
|
unset($SESSION->users_result[$id]);
|
||||||
|
echo json_encode(true);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
admin_externalpage_print_header();
|
||||||
|
|
||||||
|
// Print headers and things
|
||||||
|
|
||||||
|
print_spamcleaner_javascript();
|
||||||
|
|
||||||
|
print_box(get_string('spamcleanerintro', 'report_spamcleaner'));
|
||||||
|
|
||||||
|
print_box_start(); // The forms section at the top
|
||||||
|
|
||||||
|
?>
|
||||||
|
|
||||||
|
<div class="mdl-align">
|
||||||
|
|
||||||
|
<form method="POST">
|
||||||
|
<input type="text" name="keyword" id="keyword_el" value="<?php p($keyword) ?>" />
|
||||||
|
<input type="hidden" name="sesskey" value="<?php echo sesskey();?>" />
|
||||||
|
<input type="submit" value="<?php echo get_string('spamsearch', 'report_spamcleaner')?>" />
|
||||||
|
</form>
|
||||||
|
<p><?php echo get_string('spameg', 'report_spamcleaner');?></p>
|
||||||
|
|
||||||
|
<hr />
|
||||||
|
|
||||||
|
<form method="POST">
|
||||||
|
<input type="submit" name="autodetect" value="<?php echo get_string('spamauto', 'report_spamcleaner');?>" />
|
||||||
|
</form>
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<?php
|
||||||
|
print_box_end();
|
||||||
|
|
||||||
|
echo '<div id="result" class="mdl-align">';
|
||||||
|
|
||||||
|
// Print list of resulting profiles
|
||||||
|
|
||||||
|
if (!empty($keyword)) { // Use the keyword(s) supplied by the user
|
||||||
|
$keywords = explode(',', $keyword);
|
||||||
|
foreach ($keywords as $key => $keyword) {
|
||||||
|
$keywords[$key] = trim($keyword);
|
||||||
|
}
|
||||||
|
search_spammers($keywords);
|
||||||
|
|
||||||
|
} else if (!empty($autodetect)) { // Use the inbuilt keyword list to detect users
|
||||||
|
search_spammers($autokeywords);
|
||||||
|
}
|
||||||
|
|
||||||
|
echo '</div>';
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
|
/// Functions
|
||||||
|
|
||||||
|
|
||||||
|
function search_spammers($keywords) {
|
||||||
|
|
||||||
|
global $CFG, $USER, $DB;
|
||||||
|
|
||||||
|
if (!is_array($keywords)) {
|
||||||
|
$keywords = array($keywords); // Make it into an array
|
||||||
|
}
|
||||||
|
|
||||||
|
$like = $DB->sql_ilike();
|
||||||
|
|
||||||
|
$keywordfull = array();
|
||||||
|
foreach ($keywords as $keyword) {
|
||||||
|
$keyword = addslashes($keyword); // Just to be safe
|
||||||
|
$keywordfull[] = " description $like '%$keyword%' ";
|
||||||
|
$keywordfull2[] = " p.summary $like '%$keyword%' ";
|
||||||
|
}
|
||||||
|
$conditions = '( '.implode(' OR ', $keywordfull).' )';
|
||||||
|
$conditions2 = '( '.implode(' OR ', $keywordfull2).' )';
|
||||||
|
|
||||||
|
$sql = "SELECT * FROM {user} WHERE deleted = 0 AND id <> {$USER->id} AND $conditions"; // Exclude oneself
|
||||||
|
$sql2= "SELECT u.*, p.summary FROM {user} AS u, {post} AS p WHERE $conditions2 AND u.deleted = 0 AND u.id=p.userid AND u.id <> {$USER->id}";
|
||||||
|
$spamusers_desc = $DB->get_recordset_sql($sql);
|
||||||
|
$spamusers_blog = $DB->get_recordset_sql($sql2);
|
||||||
|
|
||||||
|
$keywordlist = implode(', ', $keywords);
|
||||||
|
print_box(get_string('spamresult', 'report_spamcleaner').s($keywordlist)).' ...';
|
||||||
|
|
||||||
|
print_user_list(array($spamusers_desc, $spamusers_blog), $keywords);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
function print_user_list($users_rs, $keywords) {
|
||||||
|
global $CFG, $SESSION;
|
||||||
|
|
||||||
|
// reset session everytime this function is called
|
||||||
|
$SESSION->users_result = array();
|
||||||
|
$count = 0;
|
||||||
|
|
||||||
|
foreach ($users_rs as $rs){
|
||||||
|
foreach ($rs as $user) {
|
||||||
|
if (!$count) {
|
||||||
|
echo '<table border="1" width="100%" id="data-grid"><tr><th> </th><th>'.get_string('user','admin').'</th><th>'.get_string('spamdesc', 'report_spamcleaner').'</th><th>'.get_string('spamoperation', 'report_spamcleaner').'</th></tr>';
|
||||||
|
}
|
||||||
|
$count++;
|
||||||
|
filter_user($user, $keywords, $count);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$count) {
|
||||||
|
echo get_string('spamcannotfinduser', 'report_spamcleaner');
|
||||||
|
|
||||||
|
} else {
|
||||||
|
echo '</table>';
|
||||||
|
echo '<div class="mld-align">
|
||||||
|
<button id="removeall_btn">'.get_string('spamdeleteall', 'report_spamcleaner').'</button>
|
||||||
|
</div>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function filter_user($user, $keywords, $count) {
|
||||||
|
global $CFG;
|
||||||
|
$image_search = false;
|
||||||
|
if (in_array('<img', $keywords)) {
|
||||||
|
$image_search = true;
|
||||||
|
}
|
||||||
|
if (isset($user->summary)) {
|
||||||
|
$user->description = '<h3>'.get_string('spamfromblog', 'report_spamcleaner').'</h3>'.$user->summary;
|
||||||
|
unset($user->summary);
|
||||||
|
}
|
||||||
|
if (preg_match('#<img.*src=[\"\']('.$CFG->pixpath.')#', $user->description, $matches)
|
||||||
|
&& $image_search) {
|
||||||
|
$result = false;
|
||||||
|
foreach ($keywords as $keyword) {
|
||||||
|
if (preg_match('#'.$keyword.'#', $user->description)
|
||||||
|
&& ($keyword != '<img')) {
|
||||||
|
$result = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($result) {
|
||||||
|
echo print_user_entry($user, $keywords, $count);
|
||||||
|
} else {
|
||||||
|
unset($user);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
echo print_user_entry($user, $keywords, $count);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function print_user_entry($user, $keywords, $count) {
|
||||||
|
|
||||||
|
global $SESSION, $CFG;
|
||||||
|
|
||||||
|
$smalluserobject = new object; // All we need to delete them later
|
||||||
|
$smalluserobject->id = $user->id;
|
||||||
|
$smalluserobject->email = $user->email;
|
||||||
|
$smalluserobject->auth = $user->auth;
|
||||||
|
$smalluserobject->firstname = $user->firstname;
|
||||||
|
$smalluserobject->lastname = $user->lastname;
|
||||||
|
|
||||||
|
if (empty($SESSION->users_result[$user->id])) {
|
||||||
|
$SESSION->users_result[$user->id] = $smalluserobject;
|
||||||
|
$html = '<tr valign="top" id="row-'.$user->id.'" class="result-row">';
|
||||||
|
$html .= '<td width="10">'.$count.'</td>';
|
||||||
|
$html .= '<td width="30%" align="left"><a href="'.$CFG->wwwroot."/user/view.php?course=1&id=".$user->id.'" title="'.$user->username.'">'.fullname($user).'</a>';
|
||||||
|
|
||||||
|
$html .= "<ul>";
|
||||||
|
$profile_set = array('city'=>true, 'country'=>true, 'email'=>true);
|
||||||
|
foreach ($profile_set as $key=>$value) {
|
||||||
|
if (isset($user->$key)){
|
||||||
|
$html .= '<li>'.$user->$key.'</li>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$html .= "</ul>";
|
||||||
|
$html .= '</td>';
|
||||||
|
|
||||||
|
foreach ($keywords as $keyword) {
|
||||||
|
$user->description = format_text(highlight($keyword, $user->description), FORMAT_MOODLE);
|
||||||
|
}
|
||||||
|
|
||||||
|
$html .= '<td align="left">'.$user->description.'</td>';
|
||||||
|
$html .= '<td width="100px" align="center">';
|
||||||
|
$html .= '<button onclick="del_user(this,'.$user->id.')">'.get_string('deleteuser', 'admin').'</button><br />';
|
||||||
|
$html .= '<button onclick="ignore_user(this,'.$user->id.')">'.get_string('ignore', 'admin').'</button>';
|
||||||
|
$html .= '</td>';
|
||||||
|
$html .= '</tr>';
|
||||||
|
return $html;
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
function print_spamcleaner_javascript() {
|
||||||
|
|
||||||
|
$sesskey = sesskey();
|
||||||
|
|
||||||
|
?>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
var row = null;
|
||||||
|
var delall_cb = {
|
||||||
|
success: function(o){
|
||||||
|
try {
|
||||||
|
var resp = YAHOO.lang.JSON.parse(o.responseText);
|
||||||
|
} catch(e) {
|
||||||
|
alert('<?php echo get_string('spaminvalidresult', 'report_spamcleaner');?>');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(resp == true){
|
||||||
|
window.location.href=window.location.href;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function init() {
|
||||||
|
YAHOO.util.Event.addListener("removeall_btn", "click", function(){
|
||||||
|
var yes = confirm('<?php echo get_string('spamdeleteallconfirm', 'report_spamcleaner');?>');
|
||||||
|
if(yes){
|
||||||
|
var cObj = YAHOO.util.Connect.asyncRequest('POST', '<?php echo me();?>?delall=yes&sesskey=<?php echo $sesskey;?>', delall_cb);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
var del_cb = {
|
||||||
|
success: function(o) {
|
||||||
|
try {
|
||||||
|
var resp = YAHOO.lang.JSON.parse(o.responseText);
|
||||||
|
} catch(e) {
|
||||||
|
alert('<?php echo get_string('spaminvalidresult', 'report_spamcleaner');?>');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(row) {
|
||||||
|
if(resp == true){
|
||||||
|
while(row.tagName != 'TR') {
|
||||||
|
row = row.parentNode;
|
||||||
|
}
|
||||||
|
row.parentNode.removeChild(row);
|
||||||
|
row = null;
|
||||||
|
} else {
|
||||||
|
alert('<?php echo get_string('spamcannotdelete', 'report_spamcleaner');?>');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var ignore_cb = {
|
||||||
|
success: function(o){
|
||||||
|
try {
|
||||||
|
var resp = YAHOO.lang.JSON.parse(o.responseText);
|
||||||
|
} catch(e) {
|
||||||
|
alert('<?php echo get_string('spaminvalidresult', 'report_spamcleaner');?>');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(row) {
|
||||||
|
if(resp == true){
|
||||||
|
while(row.tagName != 'TR') {
|
||||||
|
row = row.parentNode;
|
||||||
|
}
|
||||||
|
row.parentNode.removeChild(row);
|
||||||
|
row = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function del_user(obj, id) {
|
||||||
|
var yes = confirm('<?php echo get_string('spamdeleteconfirm', 'report_spamcleaner');?>');
|
||||||
|
if(yes){
|
||||||
|
row = obj;
|
||||||
|
var cObj = YAHOO.util.Connect.asyncRequest('POST', '<?php echo me();?>?del=yes&sesskey=<?php echo $sesskey;?>&id='+id, del_cb);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function ignore_user(obj, id) {
|
||||||
|
row = obj;
|
||||||
|
var cObj = YAHOO.util.Connect.asyncRequest('POST', '<?php echo me();?>?ignore=yes&sesskey=<?php echo $sesskey;?>&id='+id, ignore_cb);
|
||||||
|
}
|
||||||
|
YAHOO.util.Event.onDOMReady(init);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<?php
|
||||||
|
}
|
||||||
|
|
||||||
|
admin_externalpage_print_footer();
|
||||||
|
|
||||||
|
?>
|
5
admin/report/spamcleaner/settings.php
Normal file
5
admin/report/spamcleaner/settings.php
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
// spam cleaner
|
||||||
|
$ADMIN->add('reports', new admin_externalpage('reportspamcleaner', get_string('spamcleaner', 'report_spamcleaner'), "$CFG->wwwroot/$CFG->admin/report/spamcleaner/index.php", 'moodle/site:config'));
|
||||||
|
|
17
lang/en_utf8/report_spamcleaner.php
Normal file
17
lang/en_utf8/report_spamcleaner.php
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
$string['spamauto'] = 'Autodetect common spam patterns';
|
||||||
|
$string['spamcannotdelete'] = 'Cannot delete this user';
|
||||||
|
$string['spamcannotfinduser'] = 'No users matching your search';
|
||||||
|
$string['spamcleaner'] = 'Spam cleaner';
|
||||||
|
$string['spamcleanerintro'] = 'This script allows you to search all user profiles for certain strings and then delete those accounts which are obviously created by spammers. You can search for multiple keywords using commas (eg casino, porn)<br /><br />Moodle docs has more information about <a href=\"http://docs.moodle.org/en/Reducing_spam_in_Moodle\">Reducing spam in Moodle</a>.';
|
||||||
|
$string['spamdesc'] = 'Description';
|
||||||
|
$string['spamdeleteall'] = 'Delete all these user accounts';
|
||||||
|
$string['spamdeleteconfirm'] = 'Are you sure you want to delete this entry? You can not undo this.';
|
||||||
|
$string['spamdeleteallconfirm'] = 'Are you sure you want to delete all these user accounts? You can not undo this.';
|
||||||
|
$string['spameg'] = 'eg: casino, porn, xxx ';
|
||||||
|
$string['spamfromblog'] = 'From blog post:';
|
||||||
|
$string['spaminvalidresult'] = 'Unknown but invalid result';
|
||||||
|
$string['spamoperation'] = 'Operation';
|
||||||
|
$string['spamresult'] = 'Results of searching user profiles containing: ';
|
||||||
|
$string['spamsearch'] = 'Search for these keywords';
|
||||||
|
?>
|
Loading…
x
Reference in New Issue
Block a user