'item', // singular name of the listed records 'plural' => 'items', // plural name of the listed records 'ajax' => false // does this table support ajax? )); } public function column_default($item, $column_name) { return $item[$column_name]; } public function column_comment_author_IP($item) { //Build row actions if (!is_main_site() || 'blocked' === $item['status']) { //Suppress the block link if site is a multi site AND not the main site or the status is blocked $actions = array(); //blank array } else { //Add IP to block URL $ip = $item['comment_author_IP']; $actions = array( 'block' => ''.__('Block', 'all-in-one-wp-security-and-firewall').'', ); } //Return the user_login contents return sprintf('%1$s %2$s', /*$1%s*/ $item['comment_author_IP'], /*$2%s*/ $this->row_actions($actions) ); } public function column_cb($item) { return sprintf( '', /*$1%s*/ $this->_args['singular'], //Let's simply repurpose the table's singular label /*$2%s*/ esc_attr($item['comment_author_IP']) //The value of the checkbox should be the record's id ); } public function get_columns() { $columns = array( 'cb' => '', //Render a checkbox 'comment_author_IP' => __('Spammer IP', 'all-in-one-wp-security-and-firewall'), 'amount' => __('Number of spam comments from this IP', 'all-in-one-wp-security-and-firewall'), 'status' => __('Status', 'all-in-one-wp-security-and-firewall'), ); return $columns; } public function get_sortable_columns() { $sortable_columns = array( 'comment_author_IP' => array('comment_author_IP',false), 'amount' => array('amount',false), 'status' => array('status',false), ); return $sortable_columns; } public function get_bulk_actions() { if (!is_main_site()) { //Suppress the block link if site is a multi site AND not the main site $actions = array(); //blank array } else { $actions = array( 'block' => __('Block', 'all-in-one-wp-security-and-firewall') ); } return $actions; } /** * This function handles bulk actions on the table * * @return void */ private function process_bulk_action() { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- This IS the nonce check. if (empty($_REQUEST['_wpnonce']) || !isset($_REQUEST['_wp_http_referer'])) return; // phpcs:ignore WordPress.Security.NonceVerification.Recommended, WordPress.Security.ValidatedSanitizedInput -- This IS the nonce check. $result = AIOWPSecurity_Utility_Permissions::check_nonce_and_user_cap($_REQUEST['_wpnonce'], 'bulk-items'); if (is_wp_error($result)) return; if ('block' === $this->current_action()) { //Process block bulk actions if (!isset($_REQUEST['item'])) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- Nonce already checked above. $error_msg = '

'; $error_msg .= esc_html__('Please select some records using the checkboxes', 'all-in-one-wp-security-and-firewall'); $error_msg .= '

'; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- PCP error. Output already escaped. echo $error_msg; } else { // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- Nonce already checked above. $this->block_spammer_ip_records((filter_var(wp_unslash($_REQUEST['item']), FILTER_VALIDATE_IP))); } } } /** * This function will add the selected IP addresses to the blacklist. * * @param int|array $entries - either an array of IDs or a single ID of ip to be blocked * * @return void */ public function block_spammer_ip_records($entries) { if (is_array($entries)) { $entries = array_map('esc_sql', $entries); // Escape every array element //Bulk selection using checkboxes were used foreach ($entries as $ip_add) { AIOWPSecurity_Blocking::add_ip_to_block_list($ip_add, 'spam'); } } AIOWPSecurity_Admin_Menu::show_msg_updated_st(__('The selected IP addresses are now permanently blocked.', 'all-in-one-wp-security-and-firewall')); } /** * This function prepare the items rendered on the table * * @return void */ public function prepare_items() { //First, lets decide how many records per page to show $per_page = 100; $columns = $this->get_columns(); $hidden = array(); $sortable = $this->get_sortable_columns(); $this->_column_headers = array($columns, $hidden, $sortable); $this->process_bulk_action(); global $wpdb; global $aio_wp_security; $minimum_comments_per_ip = $aio_wp_security->configs->get_value('aiowps_spam_ip_min_comments'); if (empty($minimum_comments_per_ip)) { $minimum_comments_per_ip = 5; } // Ordering parameters //Parameters that are going to be used to order the result isset($_GET["orderby"]) ? $orderby = wp_strip_all_tags(wp_unslash($_GET["orderby"])) : $orderby = ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- No nonce to check. isset($_GET["order"]) ? $order = wp_strip_all_tags(wp_unslash($_GET["order"])) : $order = ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- No nonce to check. $orderby = !empty($orderby) ? esc_sql($orderby) : 'amount'; $order = !empty($order) ? esc_sql($order) : 'DESC'; $orderby = AIOWPSecurity_Utility::sanitize_value_by_array($orderby, $sortable); $order = AIOWPSecurity_Utility::sanitize_value_by_array($order, array('DESC' => '1', 'ASC' => '1')); // status is not a key in the database so we don't want to sort the database results, but sort the array later if ('status' == $orderby) { $sql = $wpdb->prepare("SELECT comment_author_IP, COUNT(*) AS amount FROM $wpdb->comments WHERE comment_approved = 'spam' GROUP BY comment_author_IP HAVING amount >= %d ", $minimum_comments_per_ip); } else { // phpcs:disable WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- $orderby cannot be prepared. $sql = $wpdb->prepare("SELECT comment_author_IP, COUNT(*) AS amount FROM $wpdb->comments WHERE comment_approved = 'spam' GROUP BY comment_author_IP HAVING amount >= %d ORDER BY $orderby $order ", $minimum_comments_per_ip); // phpcs:enable WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- $orderby cannot be prepared. } // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared, WordPress.DB.DirectDatabaseQuery -- Preparing done in conditional above. $data = $wpdb->get_results($sql, ARRAY_A); // Get all permanently blocked IP addresses $block_list = AIOWPSecurity_Blocking::get_list_blocked_ips(); foreach ($data as $key => $value) { if (in_array($value['comment_author_IP'], $block_list)) { $data[$key]['status'] = 'blocked'; } else { $data[$key]['status'] = 'not blocked'; } } if ('status' == $orderby) { $keys = array_column($data, 'status'); if ('asc' == $order) { array_multisort($keys, SORT_ASC, SORT_STRING, $data); } else { array_multisort($keys, SORT_DESC, SORT_STRING, $data); } } $current_page = $this->get_pagenum(); $total_items = count($data); $data = array_slice($data, (($current_page - 1) * $per_page), $per_page); $this->items = $data; $this->set_pagination_args(array( 'total_items' => $total_items, //WE have to calculate the total number of items 'per_page' => $per_page, //WE have to determine how many items to show on a page 'total_pages' => ceil($total_items / $per_page) //WE have to calculate the total number of pages )); } }