'item', //singular name of the listed records 'plural' => 'items', //plural name of the listed records 'ajax' => false //does this table support ajax? )); } /** * Returns created column in datetime format as per user setting time zone. * * @param array $item - data for the columns on the current row * * @return string - the datetime */ public function column_created($item) { return AIOWPSecurity_Utility::convert_timestamp($item['created']); } public function column_default($item, $column_name) { return $item[$column_name]; } /** * Returns id column html to be rendered. * * @param array $item - data for the columns on the current row * * @return string - html string for column rendered */ public function column_id($item) { $ip = $item['ip_or_host']; $is_locked = AIOWPSecurity_Utility::check_locked_ip($ip, '404'); $blacklist_tab = 'blacklist'; $is_blacklist = AIOWPSecurity_Utility::check_blacklist_ip($ip); $actions = array(); $actions['delete'] = '' . __('Delete', 'all-in-one-wp-security-and-firewall') . ''; if ($is_locked) { // Build row actions for locked items $actions['unblock'] = '' . __('Unblock', 'all-in-one-wp-security-and-firewall') . ''; } elseif ($is_blacklist) { $unblock_url_nonce = wp_nonce_url(sprintf('admin.php?page=%s&tab=%s', AIOWPSEC_FIREWALL_MENU_SLUG, $blacklist_tab), "404_log_item_action", "aiowps_nonce"); $actions = array( 'unblock' => ''.__('Unblock', 'all-in-one-wp-security-and-firewall').'', ); } else { // Build row actions for other items $actions['temp_block'] = '' . __('Temporarily block', 'all-in-one-wp-security-and-firewall') . ''; $actions['blacklist_ip'] = '' . __('Blacklist IP', 'all-in-one-wp-security-and-firewall') . ''; } //Return the user_login contents return sprintf('%1$s %2$s', /* $1%s */ $item['id'], /* $2%s */ $this->row_actions($actions) ); } /** * Returns status column html to be rendered. * * @param array $item - data for the columns on the current row * * @return string - html string for column rendered */ public function column_status($item) { global $aio_wp_security; $ip = $item['ip_or_host']; //Check if this IP address is locked $is_locked = AIOWPSecurity_Utility::check_locked_ip($ip, '404'); $blacklisted_string = $aio_wp_security->configs->get_value('aiowps_banned_ip_addresses'); $banned = strpos($blacklisted_string, $ip); if (false !== $banned) { return 'blacklisted'; } elseif ($is_locked) { return 'temporarily blocked'; } else { return ''; } } /** * Returns checkbox column html to be rendered. * * @param array $item - data for the columns on the current row * * @return string - html string for column rendered */ public function column_cb($item) { return sprintf('', /* $1%s */ $this->_args['singular'], //Let's simply repurpose the table's singular label /* $2%s */ $item['id'] //The value of the checkbox should be the record's id ); } public function get_columns() { $columns = array( 'cb' => '', //Render a checkbox 'id' => 'ID', 'event_type' => __('Event type', 'all-in-one-wp-security-and-firewall'), 'ip_or_host' => __('IP address', 'all-in-one-wp-security-and-firewall'), 'url' => __('Attempted URL', 'all-in-one-wp-security-and-firewall'), 'referer_info' => __('Referer', 'all-in-one-wp-security-and-firewall'), 'created' => __('Date and time', 'all-in-one-wp-security-and-firewall'), 'status' => __('Lock status', 'all-in-one-wp-security-and-firewall'), ); $columns = apply_filters('list_404_get_columns', $columns); return $columns; } public function get_sortable_columns() { $sortable_columns = array( 'id' => array('id', false), 'event_type' => array('event_type', false), 'ip_or_host' => array('ip_or_host', false), 'url' => array('url', false), 'referer_info' => array('referer_info', false), 'created' => array('created', false), ); $sortable_columns = apply_filters('list_404_get_sortable_columns', $sortable_columns); return $sortable_columns; } /** * Get bulk actions for the current WordPress screen. * * @return array An associative array of bulk actions where the keys are action names * and the values are the corresponding action labels. */ public function get_bulk_actions() { return array( //'unlock' => 'Unlock', 'bulk_block_ip' => __('Temporarily block IP', 'all-in-one-wp-security-and-firewall'), 'bulk_blacklist_ip' => __('Blacklist IP', 'all-in-one-wp-security-and-firewall'), 'delete' => __('Delete', 'all-in-one-wp-security-and-firewall') ); } /** * Process bulk actions for the current WordPress screen. * * This method checks for the presence of a valid nonce and user capabilities, * then performs the appropriate action based on the selected bulk action. * * @return void */ private function process_bulk_action() { // phpcs:disable WordPress.Security.NonceVerification.Recommended -- PCP warning. This is the nonce. if (empty($_REQUEST['_wpnonce']) || !isset($_REQUEST['_wp_http_referer'])) return; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput -- PCP warning. Ignore. $result = AIOWPSecurity_Utility_Permissions::check_nonce_and_user_cap($_REQUEST['_wpnonce'], 'bulk-items'); if (is_wp_error($result)) return; if ('bulk_block_ip' === $this->current_action()) {//Process delete bulk actions if (!isset($_REQUEST['item'])) { AIOWPSecurity_Admin_Menu::show_msg_error_st(__('Please select some records using the checkboxes', 'all-in-one-wp-security-and-firewall')); } else { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput -- PCP warning, ignore. Sanitized later. $this->block_ip(wp_unslash($_REQUEST['item'])); } } if ('bulk_blacklist_ip' === $this->current_action()) {//Process delete bulk actions if (!isset($_REQUEST['item'])) { AIOWPSecurity_Admin_Menu::show_msg_error_st(__('Please select some records using the checkboxes', 'all-in-one-wp-security-and-firewall')); } else { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput -- PCP warning, ignore. Sanitized later. $this->blacklist_ip_address(wp_unslash($_REQUEST['item'])); } } if ('delete' === $this->current_action()) {//Process delete bulk actions if (!isset($_REQUEST['item'])) { AIOWPSecurity_Admin_Menu::show_msg_error_st(__('Please select some records using the checkboxes', 'all-in-one-wp-security-and-firewall')); } else { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput -- PCP warning, ignore. Sanitized later. $this->delete_404_event_records(wp_unslash($_REQUEST['item'])); } } // phpcs:enable WordPress.Security.NonceVerification.Recommended -- PCP warning. This is the nonce. } /** * Locks an IP address by adding it to the AIOWPSEC_TBL_LOGIN_LOCKOUT table. * * @param array|string $entries - ids that correspond to ip addresses in the AIOWPSEC_TBL_EVENTS table or a single ip address * @param string $username - (optional)username of user being locked * * @return boolean|void */ public function block_ip($entries, $username = '') { global $wpdb; if (is_array($entries)) { //lock multiple records $entries = array_filter($entries, 'is_numeric'); //discard non-numeric ID values $id_list = "(" .implode(",", $entries) .")"; //Create comma separate list for DB operation $events_table = AIOWPSEC_TBL_EVENTS; // phpcs:ignore WordPress.DB.PreparedSQL, WordPress.DB.DirectDatabaseQuery -- PCP error. Ignore. $results = $wpdb->get_col("SELECT ip_or_host FROM $events_table WHERE ID IN " . $id_list); if (empty($results)) { AIOWPSecurity_Admin_Menu::show_msg_error_st(__('Could not process the request because the IP addresses for the selected entries could not be found.', 'all-in-one-wp-security-and-firewall')); return false; } else { foreach ($results as $entry) { if (filter_var($entry, FILTER_VALIDATE_IP)) { AIOWPSecurity_Utility::lock_IP($entry, '404', $username); } } } AIOWPSecurity_Admin_Menu::show_msg_updated_st(__('The selected IP addresses are now temporarily blocked.', 'all-in-one-wp-security-and-firewall')); } } /** * Permanently blocks an IP address by adding it to the blacklist and writing rules to the htaccess file. * * @param array|string $entries - ids that correspond to ip addresses in the AIOWPSEC_TBL_EVENTS table or a single ip address * * @return boolean|void */ public function blacklist_ip_address($entries) { global $wpdb, $aio_wp_security; $aiowps_firewall_config = AIOS_Firewall_Resource::request(AIOS_Firewall_Resource::CONFIG); $bl_ip_addresses = $aio_wp_security->configs->get_value('aiowps_banned_ip_addresses'); //get the currently saved blacklisted IPs $ip_list_array = AIOWPSecurity_Utility_IP::create_ip_list_array_from_string_with_newline($bl_ip_addresses); if (is_array($entries)) { //Get the selected IP addresses $entries = array_filter($entries, 'is_numeric'); //discard non-numeric ID values $id_list = "(" .implode(",", $entries) .")"; //Create comma separate list for DB operation $events_table = AIOWPSEC_TBL_EVENTS; // phpcs:ignore WordPress.DB.PreparedSQL, WordPress.DB.DirectDatabaseQuery -- PCP error. Ignore. $results = $wpdb->get_col("SELECT ip_or_host FROM $events_table WHERE ID IN " . $id_list); if (empty($results)) { AIOWPSecurity_Admin_Menu::show_msg_error_st(__('Could not process the request because the IP addresses for the selected entries could not be found.', 'all-in-one-wp-security-and-firewall')); return false; } else { foreach ($results as $entry) { $ip_list_array[] = $entry; } } } $validated_ip_list_array = AIOWPSecurity_Utility_IP::validate_ip_list($ip_list_array, 'blacklist'); if (is_wp_error($validated_ip_list_array)) { AIOWPSecurity_Admin_Menu::show_msg_error_st(nl2br($validated_ip_list_array->get_error_message())); } else { $banned_ip_data = implode("\n", $validated_ip_list_array); $aio_wp_security->configs->set_value('aiowps_enable_blacklisting', '1'); // Force blacklist feature to be enabled. $aio_wp_security->configs->set_value('aiowps_banned_ip_addresses', $banned_ip_data); $aio_wp_security->configs->save_config(); $aiowps_firewall_config->set_value('aiowps_blacklist_ips', $validated_ip_list_array); AIOWPSecurity_Admin_Menu::show_msg_updated_st(__('The selected IP addresses have been added to the blacklist and will be permanently blocked.', 'all-in-one-wp-security-and-firewall')); } } /** * Deletes one or more records from the AIOWPSEC_TBL_EVENTS table. * * @param array|string|integer $entries - ids or a single id * * @return void|string */ public function delete_404_event_records($entries) { global $wpdb, $aio_wp_security; $events_table = AIOWPSEC_TBL_EVENTS; if (is_array($entries)) { //Delete multiple records $entries = array_map('esc_sql', $entries); //escape every array element $entries = array_filter($entries, 'is_numeric'); //discard non-numeric ID values $id_list = "(" . implode(",", $entries) . ")"; //Create comma separate list for DB operation // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared, WordPress.DB.DirectDatabaseQuery -- PCP error. Ignore. $result = $wpdb->query("DELETE FROM " . $events_table . " WHERE id IN " . $id_list); if ($result) { AIOWPSecurity_Admin_Menu::show_msg_record_deleted_st(); } else { // Error on bulk delete $aio_wp_security->debug_logger->log_debug('Database error occurred when deleting rows from Events table. Database error: '.$wpdb->last_error, 4); AIOWPSecurity_Admin_Menu::show_msg_record_not_deleted_st(); } } } /** * Retrieves all items from AIOWPSEC_TBL_EVENTS according to a search term inside $_REQUEST['s'] and only '404' events if there is no search term. It then assigns to $this->items. * * @param Boolean $ignore_pagination - whether to not paginate * * @return Void */ public function prepare_items($ignore_pagination = false) { /** * First, lets decide how many records per page to show */ $per_page = 100; $columns = $this->get_columns(); $hidden = array(); $sortable = $this->get_sortable_columns(); // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- PCP warning. Nonce checked in previous function. $search_term = isset($_REQUEST['s']) ? sanitize_text_field(wp_unslash($_REQUEST['s'])) : ''; $this->_column_headers = array($columns, $hidden, $sortable); $this->process_bulk_action(); global $wpdb; $events_table_name = AIOWPSEC_TBL_EVENTS; // Ordering parameters // Parameters that are going to be used to order the result // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- PCP warning. No nonce. $orderby = isset($_GET['orderby']) ? sanitize_text_field(wp_unslash($_GET['orderby'])) : ''; // phpcs:ignore WordPress.Security.NonceVerification.Recommended -- PCP warning. No nonce. $order = isset($_GET['order']) ? sanitize_text_field(wp_unslash($_GET['order'])) : ''; $orderby = !empty($orderby) ? esc_sql($orderby) : 'id'; $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')); if (empty($search_term)) { // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery -- PCP warning. Ignore. $data = $wpdb->get_results("SELECT * FROM $events_table_name WHERE `event_type` = '404' ORDER BY $orderby $order", ARRAY_A); } else { // phpcs:ignore WordPress.DB.PreparedSQLPlaceholders.LikeWildcardsInQueryWithPlaceholder, WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery -- PCP error. Ignore. $data = $wpdb->get_results($wpdb->prepare("SELECT * FROM $events_table_name WHERE `ip_or_host` LIKE '%%%s%%' OR `url` LIKE '%%%s%%' OR `referer_info` LIKE '%%%s%%' ORDER BY $orderby $order", $wpdb->esc_like($search_term), $wpdb->esc_like($search_term), $wpdb->esc_like($search_term)), ARRAY_A); } if (!$ignore_pagination) { $current_page = $this->get_pagenum(); $total_items = count($data); $data = array_slice($data, (($current_page - 1) * $per_page), $per_page); $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 )); } foreach ($data as $index => $row) { // Insert an empty status column - we will use later $data[$index]['status'] = ''; } $this->items = $data; } }