mirror of
https://github.com/WenPai-org/wp-domain-mapping.git
synced 2025-08-03 22:39:47 +08:00
v2.0 重构
This commit is contained in:
parent
f7fe8d63e0
commit
31bf7145ea
14 changed files with 6815 additions and 1103 deletions
389
includes/admin-ui.php
Normal file
389
includes/admin-ui.php
Normal file
|
@ -0,0 +1,389 @@
|
|||
<?php
|
||||
/**
|
||||
* Admin UI functions for WP Domain Mapping plugin
|
||||
*
|
||||
* @package WP Domain Mapping
|
||||
*/
|
||||
|
||||
// Prevent direct access
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Render domain edit form
|
||||
*
|
||||
* @param object|false $row Domain data if editing, false if adding new
|
||||
*/
|
||||
function dm_edit_domain( $row = false ) {
|
||||
$is_edit = is_object( $row );
|
||||
|
||||
if ( ! $row ) {
|
||||
$row = new stdClass();
|
||||
$row->blog_id = '';
|
||||
$row->domain = '';
|
||||
$row->active = 1;
|
||||
}
|
||||
?>
|
||||
<form id="edit-domain-form" method="POST">
|
||||
<input type="hidden" name="orig_domain" value="<?php echo esc_attr( $is_edit ? $row->domain : '' ); ?>" />
|
||||
<table class="form-table">
|
||||
<tr>
|
||||
<th><label for="blog_id"><?php esc_html_e( 'Site ID', 'wp-domain-mapping' ); ?></label></th>
|
||||
<td>
|
||||
<input type="number" id="blog_id" name="blog_id" value="<?php echo esc_attr( $row->blog_id ); ?>" class="regular-text" required />
|
||||
<?php if ( ! $is_edit ) : ?>
|
||||
<p class="description">
|
||||
<?php
|
||||
$site_list_url = network_admin_url( 'sites.php' );
|
||||
printf(
|
||||
/* translators: %s: URL to sites list */
|
||||
wp_kses(
|
||||
__( 'Not sure about Site ID? <a href="%s" target="_blank">View all sites</a> to find the ID.', 'wp-domain-mapping' ),
|
||||
array( 'a' => array( 'href' => array(), 'target' => array() ) )
|
||||
),
|
||||
esc_url( $site_list_url )
|
||||
);
|
||||
?>
|
||||
</p>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><label for="domain"><?php esc_html_e( 'Domain', 'wp-domain-mapping' ); ?></label></th>
|
||||
<td>
|
||||
<input type="text" id="domain" name="domain" value="<?php echo esc_attr( $row->domain ); ?>" class="regular-text" required
|
||||
placeholder="example.com" pattern="^[a-zA-Z0-9][a-zA-Z0-9-]{0,61}[a-zA-Z0-9]?\.[a-zA-Z]{2,}$" />
|
||||
<p class="description">
|
||||
<?php esc_html_e( 'Enter the domain without http:// or https:// (e.g., example.com)', 'wp-domain-mapping' ); ?>
|
||||
</p>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><label for="active"><?php esc_html_e( 'Primary', 'wp-domain-mapping' ); ?></label></th>
|
||||
<td>
|
||||
<input type="checkbox" id="active" name="active" value="1" <?php checked( $row->active, 1 ); ?> />
|
||||
<span class="description">
|
||||
<?php esc_html_e( 'Set as the primary domain for this site', 'wp-domain-mapping' ); ?>
|
||||
</span>
|
||||
</td>
|
||||
</tr>
|
||||
<?php if ( get_site_option( 'dm_no_primary_domain' ) == 1 ) : ?>
|
||||
<tr>
|
||||
<td colspan="2" class="notice notice-warning">
|
||||
<p><?php esc_html_e( 'Warning! Primary domains are currently disabled in network settings.', 'wp-domain-mapping' ); ?></p>
|
||||
</td>
|
||||
</tr>
|
||||
<?php endif; ?>
|
||||
</table>
|
||||
<p>
|
||||
<input type="submit" class="button button-primary" value="<?php echo $is_edit
|
||||
? esc_attr__( 'Update Domain', 'wp-domain-mapping' )
|
||||
: esc_attr__( 'Add Domain', 'wp-domain-mapping' ); ?>" />
|
||||
|
||||
<?php if ( $is_edit ) : ?>
|
||||
<a href="<?php echo esc_url( admin_url( 'network/sites.php?page=domains' ) ); ?>" class="button button-secondary">
|
||||
<?php esc_html_e( 'Cancel', 'wp-domain-mapping' ); ?>
|
||||
</a>
|
||||
<?php endif; ?>
|
||||
</p>
|
||||
</form>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* Render domain listing table
|
||||
*
|
||||
* @param array $rows Domain data rows
|
||||
*/
|
||||
function dm_domain_listing( $rows ) {
|
||||
global $wpdb;
|
||||
|
||||
if ( ! $rows ) {
|
||||
echo '<div class="notice notice-info"><p>' . esc_html__( 'No domains found.', 'wp-domain-mapping' ) . '</p></div>';
|
||||
return;
|
||||
}
|
||||
|
||||
$edit_url = network_admin_url(
|
||||
file_exists( ABSPATH . 'wp-admin/network/site-info.php' )
|
||||
? 'site-info.php'
|
||||
: ( file_exists( ABSPATH . 'wp-admin/ms-sites.php' ) ? 'ms-sites.php' : 'wpmu-blogs.php' )
|
||||
);
|
||||
?>
|
||||
<div class="tablenav top">
|
||||
<div class="alignleft actions bulkactions">
|
||||
<label for="bulk-action-selector-top" class="screen-reader-text"><?php esc_html_e( 'Select bulk action', 'wp-domain-mapping' ); ?></label>
|
||||
<select id="bulk-action-selector-top" name="action">
|
||||
<option value="-1"><?php esc_html_e( 'Bulk Actions', 'wp-domain-mapping' ); ?></option>
|
||||
<option value="delete"><?php esc_html_e( 'Delete', 'wp-domain-mapping' ); ?></option>
|
||||
</select>
|
||||
<input type="submit" class="button action" value="<?php esc_attr_e( 'Apply', 'wp-domain-mapping' ); ?>" />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<table class="wp-list-table widefat striped domains-table">
|
||||
<thead>
|
||||
<tr>
|
||||
<td class="manage-column column-cb check-column">
|
||||
<input id="select-all" type="checkbox" />
|
||||
</td>
|
||||
<th scope="col" class="column-site-id"><?php esc_html_e( 'Site ID', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col" class="column-site-name"><?php esc_html_e( 'Site Name', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col" class="column-domain"><?php esc_html_e( 'Domain', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col" class="column-primary"><?php esc_html_e( 'Primary', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col" class="column-actions"><?php esc_html_e( 'Actions', 'wp-domain-mapping' ); ?></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<?php foreach ( $rows as $row ) :
|
||||
$site_name = get_blog_option( $row->blog_id, 'blogname', esc_html__( 'Unknown', 'wp-domain-mapping' ) );
|
||||
?>
|
||||
<tr>
|
||||
<th scope="row" class="check-column">
|
||||
<input type="checkbox" class="domain-checkbox" value="<?php echo esc_attr( $row->domain ); ?>" />
|
||||
</th>
|
||||
<td class="column-site-id">
|
||||
<a href="<?php echo esc_url( add_query_arg( array( 'action' => 'editblog', 'id' => $row->blog_id ), $edit_url ) ); ?>">
|
||||
<?php echo esc_html( $row->blog_id ); ?>
|
||||
</a>
|
||||
</td>
|
||||
<td class="column-site-name">
|
||||
<?php echo esc_html( $site_name ); ?>
|
||||
</td>
|
||||
<td class="column-domain">
|
||||
<a href="<?php echo esc_url( dm_ensure_protocol( $row->domain ) ); ?>" target="_blank">
|
||||
<?php echo esc_html( $row->domain ); ?>
|
||||
<span class="dashicons dashicons-external" style="font-size: 14px; line-height: 1.3; opacity: 0.7;"></span>
|
||||
</a>
|
||||
</td>
|
||||
<td class="column-primary">
|
||||
<?php if ( $row->active == 1 ) : ?>
|
||||
<span class="dashicons dashicons-yes-alt" style="color: #46b450;"></span>
|
||||
<span class="screen-reader-text"><?php esc_html_e( 'Yes', 'wp-domain-mapping' ); ?></span>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-no-alt" style="color: #dc3232;"></span>
|
||||
<span class="screen-reader-text"><?php esc_html_e( 'No', 'wp-domain-mapping' ); ?></span>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
<td class="column-actions">
|
||||
<div class="row-actions">
|
||||
<span class="edit">
|
||||
<a href="<?php echo esc_url( add_query_arg( array( 'edit_domain' => $row->domain ), admin_url( 'network/sites.php?page=domains' ) ) ); ?>" class="button button-small">
|
||||
<?php esc_html_e( 'Edit', 'wp-domain-mapping' ); ?>
|
||||
</a>
|
||||
</span>
|
||||
<?php if ( $row->active != 1 ) : ?>
|
||||
<span class="delete">
|
||||
<a href="#" class="button button-small domain-delete-button" data-domain="<?php echo esc_attr( $row->domain ); ?>">
|
||||
<?php esc_html_e( 'Delete', 'wp-domain-mapping' ); ?>
|
||||
</a>
|
||||
</span>
|
||||
<?php endif; ?>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
<?php endforeach; ?>
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<script>
|
||||
jQuery(document).ready(function($) {
|
||||
// Handle select all checkbox
|
||||
$('#select-all').on('change', function() {
|
||||
$('.domain-checkbox').prop('checked', this.checked);
|
||||
});
|
||||
|
||||
// Handle individual checkboxes
|
||||
$('.domain-checkbox').on('change', function() {
|
||||
if (!this.checked) {
|
||||
$('#select-all').prop('checked', false);
|
||||
} else if ($('.domain-checkbox:checked').length === $('.domain-checkbox').length) {
|
||||
$('#select-all').prop('checked', true);
|
||||
}
|
||||
});
|
||||
|
||||
// Handle delete button clicks
|
||||
$('.domain-delete-button').on('click', function(e) {
|
||||
e.preventDefault();
|
||||
|
||||
if (confirm('<?php esc_html_e( 'Are you sure you want to delete this domain?', 'wp-domain-mapping' ); ?>')) {
|
||||
var domain = $(this).data('domain');
|
||||
var data = {
|
||||
action: 'dm_handle_actions',
|
||||
action_type: 'delete',
|
||||
domain: domain,
|
||||
nonce: '<?php echo wp_create_nonce( 'domain_mapping' ); ?>'
|
||||
};
|
||||
|
||||
$.post(ajaxurl, data, function(response) {
|
||||
if (response.success) {
|
||||
location.reload();
|
||||
} else {
|
||||
alert(response.data);
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
});
|
||||
</script>
|
||||
<?php
|
||||
|
||||
if ( get_site_option( 'dm_no_primary_domain' ) == 1 ) {
|
||||
echo '<div class="notice notice-warning inline"><p>' .
|
||||
esc_html__( 'Warning! Primary domains are currently disabled in network settings.', 'wp-domain-mapping' ) .
|
||||
'</p></div>';
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Ensure URL has a protocol
|
||||
*
|
||||
* @param string $domain Domain name
|
||||
* @return string Domain with protocol
|
||||
*/
|
||||
function dm_ensure_protocol( $domain ) {
|
||||
if ( preg_match( '#^https?://#', $domain ) ) {
|
||||
return $domain;
|
||||
}
|
||||
return 'http://' . $domain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Clean domain name (remove protocol and trailing slash)
|
||||
*
|
||||
* @param string $domain Domain name
|
||||
* @return string Cleaned domain
|
||||
*/
|
||||
function dm_clean_domain( $domain ) {
|
||||
// Remove protocol
|
||||
$domain = preg_replace( '#^https?://#', '', $domain );
|
||||
|
||||
// Remove trailing slash
|
||||
$domain = rtrim( $domain, '/' );
|
||||
|
||||
// Convert IDN to ASCII (Punycode)
|
||||
if ( function_exists( 'idn_to_ascii' ) && preg_match( '/[^a-z0-9\-\.]/i', $domain ) ) {
|
||||
if (defined('INTL_IDNA_VARIANT_UTS46')) {
|
||||
// PHP 7.2+
|
||||
$domain = idn_to_ascii( $domain, 0, INTL_IDNA_VARIANT_UTS46 );
|
||||
} else {
|
||||
// PHP < 7.2
|
||||
$domain = idn_to_ascii( $domain );
|
||||
}
|
||||
}
|
||||
|
||||
return $domain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Display IDN warning message
|
||||
*
|
||||
* @return string Warning message
|
||||
*/
|
||||
function dm_idn_warning() {
|
||||
return sprintf(
|
||||
/* translators: %s: URL to punycode converter */
|
||||
wp_kses(
|
||||
__( 'International Domain Names should be in <a href="%s" target="_blank">punycode</a> format.', 'wp-domain-mapping' ),
|
||||
array( 'a' => array( 'href' => array(), 'target' => array() ) )
|
||||
),
|
||||
'https://www.punycoder.com/'
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Render domain logs table
|
||||
*/
|
||||
function dm_domain_logs() {
|
||||
global $wpdb;
|
||||
|
||||
$table_logs = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_LOGS;
|
||||
|
||||
// Make sure the table exists
|
||||
if ($wpdb->get_var("SHOW TABLES LIKE '$table_logs'") != $table_logs) {
|
||||
echo '<div class="notice notice-error"><p>' .
|
||||
esc_html__('Domain mapping logs table is missing. Please deactivate and reactivate the plugin.', 'wp-domain-mapping') .
|
||||
'</p></div>';
|
||||
return;
|
||||
}
|
||||
|
||||
$logs = $wpdb->get_results( "SELECT * FROM {$table_logs} ORDER BY timestamp DESC LIMIT 50" );
|
||||
|
||||
if ( ! $logs ) {
|
||||
echo '<div class="notice notice-info"><p>' . esc_html__( 'No logs available.', 'wp-domain-mapping' ) . '</p></div>';
|
||||
return;
|
||||
}
|
||||
?>
|
||||
<table class="wp-list-table widefat striped logs-table">
|
||||
<thead>
|
||||
<tr>
|
||||
<th scope="col"><?php esc_html_e( 'User', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col"><?php esc_html_e( 'Action', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col"><?php esc_html_e( 'Domain', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col"><?php esc_html_e( 'Site ID', 'wp-domain-mapping' ); ?></th>
|
||||
<th scope="col"><?php esc_html_e( 'Timestamp', 'wp-domain-mapping' ); ?></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<?php foreach ( $logs as $log ) :
|
||||
$user_data = get_userdata( $log->user_id );
|
||||
$username = $user_data ? $user_data->user_login : sprintf( __( 'User #%d', 'wp-domain-mapping' ), $log->user_id );
|
||||
|
||||
// Format action for display
|
||||
switch ( $log->action ) {
|
||||
case 'add':
|
||||
$action_display = '<span class="log-action log-action-add">' . __( 'Added', 'wp-domain-mapping' ) . '</span>';
|
||||
break;
|
||||
case 'edit':
|
||||
$action_display = '<span class="log-action log-action-edit">' . __( 'Updated', 'wp-domain-mapping' ) . '</span>';
|
||||
break;
|
||||
case 'delete':
|
||||
$action_display = '<span class="log-action log-action-delete">' . __( 'Deleted', 'wp-domain-mapping' ) . '</span>';
|
||||
break;
|
||||
default:
|
||||
$action_display = '<span class="log-action">' . esc_html( $log->action ) . '</span>';
|
||||
}
|
||||
|
||||
// Format timestamp for display
|
||||
$timestamp = mysql2date( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ), $log->timestamp );
|
||||
?>
|
||||
<tr>
|
||||
<td><?php echo esc_html( $username ); ?></td>
|
||||
<td><?php echo $action_display; ?></td>
|
||||
<td><?php echo esc_html( $log->domain ); ?></td>
|
||||
<td>
|
||||
<a href="<?php echo esc_url( network_admin_url( 'site-info.php?id=' . $log->blog_id ) ); ?>">
|
||||
<?php echo esc_html( $log->blog_id ); ?>
|
||||
</a>
|
||||
</td>
|
||||
<td title="<?php echo esc_attr( $log->timestamp ); ?>">
|
||||
<?php echo esc_html( $timestamp ); ?>
|
||||
</td>
|
||||
</tr>
|
||||
<?php endforeach; ?>
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<style>
|
||||
.log-action {
|
||||
display: inline-block;
|
||||
padding: 2px 8px;
|
||||
border-radius: 3px;
|
||||
font-size: 12px;
|
||||
font-weight: bold;
|
||||
}
|
||||
.log-action-add {
|
||||
background-color: #dff0d8;
|
||||
color: #3c763d;
|
||||
}
|
||||
.log-action-edit {
|
||||
background-color: #d9edf7;
|
||||
color: #31708f;
|
||||
}
|
||||
.log-action-delete {
|
||||
background-color: #f2dede;
|
||||
color: #a94442;
|
||||
}
|
||||
</style>
|
||||
<?php
|
||||
}
|
897
includes/class-health.php
Normal file
897
includes/class-health.php
Normal file
|
@ -0,0 +1,897 @@
|
|||
<?php
|
||||
/**
|
||||
* 域名健康监控功能
|
||||
*
|
||||
* 处理域名健康状态检查和通知
|
||||
*
|
||||
* @package WP Domain Mapping
|
||||
*/
|
||||
|
||||
// 防止直接访问
|
||||
if (!defined('ABSPATH')) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* 域名健康监控类
|
||||
*/
|
||||
class WP_Domain_Mapping_Health {
|
||||
|
||||
/**
|
||||
* 类实例
|
||||
*
|
||||
* @var WP_Domain_Mapping_Health
|
||||
*/
|
||||
protected static $instance;
|
||||
|
||||
/**
|
||||
* 数据库实例
|
||||
*/
|
||||
private $db;
|
||||
|
||||
/**
|
||||
* 获取类实例
|
||||
*
|
||||
* @return WP_Domain_Mapping_Health
|
||||
*/
|
||||
public static function get_instance() {
|
||||
if (is_null(self::$instance)) {
|
||||
self::$instance = new self();
|
||||
}
|
||||
|
||||
return self::$instance;
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
*/
|
||||
private function __construct() {
|
||||
$this->db = WP_Domain_Mapping_DB::get_instance();
|
||||
|
||||
// 添加菜单页面
|
||||
add_action('network_admin_menu', array($this, 'add_menu_page'), 30);
|
||||
|
||||
// 添加AJAX处理程序
|
||||
add_action('wp_ajax_dm_check_domain_health', array($this, 'ajax_check_domain_health'));
|
||||
|
||||
// 添加计划任务钩子
|
||||
add_action('dm_domain_health_check', array($this, 'scheduled_health_check'));
|
||||
|
||||
// 初始化计划任务
|
||||
$this->initialize_cron();
|
||||
|
||||
// 处理手动健康检查
|
||||
add_action('admin_init', array($this, 'handle_manual_check'));
|
||||
|
||||
// 处理设置保存
|
||||
add_action('admin_init', array($this, 'handle_settings_save'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 初始化定时健康检查
|
||||
*/
|
||||
private function initialize_cron() {
|
||||
// 注册激活时的钩子
|
||||
register_activation_hook(WP_DOMAIN_MAPPING_BASENAME, array($this, 'schedule_health_check'));
|
||||
|
||||
// 注册停用时的钩子
|
||||
register_deactivation_hook(WP_DOMAIN_MAPPING_BASENAME, array($this, 'unschedule_health_check'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加健康监控菜单
|
||||
*/
|
||||
public function add_menu_page() {
|
||||
add_submenu_page(
|
||||
'settings.php',
|
||||
__('Domain Health', 'wp-domain-mapping'),
|
||||
__('Domain Health', 'wp-domain-mapping'),
|
||||
'manage_network',
|
||||
'domain-mapping-health',
|
||||
array($this, 'render_page')
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* 计划健康检查任务
|
||||
*/
|
||||
public function schedule_health_check() {
|
||||
if (!wp_next_scheduled('dm_domain_health_check')) {
|
||||
wp_schedule_event(time(), 'daily', 'dm_domain_health_check');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 取消健康检查任务
|
||||
*/
|
||||
public function unschedule_health_check() {
|
||||
$timestamp = wp_next_scheduled('dm_domain_health_check');
|
||||
if ($timestamp) {
|
||||
wp_unschedule_event($timestamp, 'dm_domain_health_check');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 处理手动健康检查
|
||||
*/
|
||||
public function handle_manual_check() {
|
||||
if (isset($_POST['dm_manual_health_check']) && $_POST['dm_manual_health_check']) {
|
||||
// 验证nonce
|
||||
if (!isset($_POST['dm_manual_health_check_nonce']) || !wp_verify_nonce($_POST['dm_manual_health_check_nonce'], 'dm_manual_health_check')) {
|
||||
wp_die(__('Security check failed.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to perform this action.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 执行健康检查
|
||||
$this->run_health_check_for_all_domains();
|
||||
|
||||
// 重定向回健康页面
|
||||
wp_redirect(add_query_arg(array('page' => 'domain-mapping-health', 'checked' => 1), network_admin_url('settings.php')));
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 处理设置保存
|
||||
*/
|
||||
public function handle_settings_save() {
|
||||
if (isset($_POST['dm_health_settings']) && $_POST['dm_health_settings']) {
|
||||
// 验证nonce
|
||||
if (!isset($_POST['dm_health_settings_nonce']) || !wp_verify_nonce($_POST['dm_health_settings_nonce'], 'dm_health_settings')) {
|
||||
wp_die(__('Security check failed.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to perform this action.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 保存设置
|
||||
$health_check_enabled = isset($_POST['health_check_enabled']) ? (bool) $_POST['health_check_enabled'] : false;
|
||||
$health_notifications_enabled = isset($_POST['health_notifications_enabled']) ? (bool) $_POST['health_notifications_enabled'] : false;
|
||||
$notification_email = isset($_POST['notification_email']) ? sanitize_email($_POST['notification_email']) : '';
|
||||
$ssl_expiry_threshold = isset($_POST['ssl_expiry_threshold']) ? intval($_POST['ssl_expiry_threshold']) : 14;
|
||||
|
||||
update_site_option('dm_health_check_enabled', $health_check_enabled);
|
||||
update_site_option('dm_health_notifications_enabled', $health_notifications_enabled);
|
||||
update_site_option('dm_notification_email', $notification_email);
|
||||
update_site_option('dm_ssl_expiry_threshold', $ssl_expiry_threshold);
|
||||
|
||||
// 如果启用了自动检查,确保计划任务已设置
|
||||
if ($health_check_enabled) {
|
||||
$this->schedule_health_check();
|
||||
} else {
|
||||
$this->unschedule_health_check();
|
||||
}
|
||||
|
||||
// 重定向回健康页面
|
||||
wp_redirect(add_query_arg(array('page' => 'domain-mapping-health', 'settings-updated' => 1), network_admin_url('settings.php')));
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 渲染健康监控页面
|
||||
*/
|
||||
public function render_page() {
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to access this page.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 获取所有域名
|
||||
global $wpdb;
|
||||
$table = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_DOMAINS;
|
||||
$domains = $wpdb->get_results("
|
||||
SELECT d.*, b.domain as original_domain, b.path
|
||||
FROM {$table} d
|
||||
JOIN {$wpdb->blogs} b ON d.blog_id = b.blog_id
|
||||
ORDER BY d.blog_id ASC, d.active DESC
|
||||
");
|
||||
|
||||
// 获取健康检查结果(如果有)
|
||||
$health_results = get_site_option('dm_domain_health_results', array());
|
||||
|
||||
// 渲染页面
|
||||
?>
|
||||
<div class="wrap">
|
||||
<h1><?php echo esc_html(get_admin_page_title()); ?></h1>
|
||||
|
||||
<?php
|
||||
// 显示成功消息(如果有)
|
||||
if (isset($_GET['checked']) && $_GET['checked']) {
|
||||
echo '<div class="notice notice-success is-dismissible"><p>' .
|
||||
__('Domain health check completed.', 'wp-domain-mapping') .
|
||||
'</p></div>';
|
||||
}
|
||||
|
||||
if (isset($_GET['settings-updated']) && $_GET['settings-updated']) {
|
||||
echo '<div class="notice notice-success is-dismissible"><p>' .
|
||||
__('Settings saved.', 'wp-domain-mapping') .
|
||||
'</p></div>';
|
||||
}
|
||||
?>
|
||||
|
||||
<div class="card domain-mapping-card" style="max-width: 100%; margin-top: 20px; padding: 20px; background: #fff; border: 1px solid #ccd0d4; border-radius: 4px; box-shadow: 0 1px 1px rgba(0,0,0,.04);">
|
||||
<h2><?php _e('Domain Health Status', 'wp-domain-mapping'); ?></h2>
|
||||
|
||||
<p>
|
||||
<form method="post" action="">
|
||||
<?php wp_nonce_field('dm_manual_health_check', 'dm_manual_health_check_nonce'); ?>
|
||||
<input type="hidden" name="dm_manual_health_check" value="1">
|
||||
<input type="submit" class="button button-primary" value="<?php esc_attr_e('Check All Domains Now', 'wp-domain-mapping'); ?>">
|
||||
</form>
|
||||
</p>
|
||||
|
||||
<div class="tablenav top">
|
||||
<div class="tablenav-pages">
|
||||
<span class="displaying-num">
|
||||
<?php
|
||||
if (!empty($domains)) {
|
||||
printf(
|
||||
_n('%s domain', '%s domains', count($domains), 'wp-domain-mapping'),
|
||||
number_format_i18n(count($domains))
|
||||
);
|
||||
} else {
|
||||
_e('No domains found', 'wp-domain-mapping');
|
||||
}
|
||||
?>
|
||||
</span>
|
||||
</div>
|
||||
<br class="clear">
|
||||
</div>
|
||||
|
||||
<table class="wp-list-table widefat fixed striped domains-health-table">
|
||||
<thead>
|
||||
<tr>
|
||||
<th class="column-domain"><?php _e('Domain', 'wp-domain-mapping'); ?></th>
|
||||
<th class="column-site"><?php _e('Site', 'wp-domain-mapping'); ?></th>
|
||||
<th class="column-dns"><?php _e('DNS Status', 'wp-domain-mapping'); ?></th>
|
||||
<th class="column-ssl"><?php _e('SSL Status', 'wp-domain-mapping'); ?></th>
|
||||
<th class="column-status"><?php _e('Reachable', 'wp-domain-mapping'); ?></th>
|
||||
<th class="column-last-check"><?php _e('Last Check', 'wp-domain-mapping'); ?></th>
|
||||
<th class="column-actions"><?php _e('Actions', 'wp-domain-mapping'); ?></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<?php if (!empty($domains)) : ?>
|
||||
<?php foreach ($domains as $domain) :
|
||||
$domain_key = md5($domain->domain);
|
||||
$health_data = isset($health_results[$domain_key]) ? $health_results[$domain_key] : null;
|
||||
$site_name = get_blog_option($domain->blog_id, 'blogname', __('Unknown', 'wp-domain-mapping'));
|
||||
?>
|
||||
<tr data-domain="<?php echo esc_attr($domain->domain); ?>" data-blog-id="<?php echo esc_attr($domain->blog_id); ?>">
|
||||
<td class="column-domain">
|
||||
<?php echo esc_html($domain->domain); ?>
|
||||
<?php if ($domain->active) : ?>
|
||||
<span class="dashicons dashicons-star-filled" style="color: #f0b849;" title="<?php esc_attr_e('Primary Domain', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
<td class="column-site">
|
||||
<a href="<?php echo esc_url(network_admin_url('site-info.php?id=' . $domain->blog_id)); ?>">
|
||||
<?php echo esc_html($site_name); ?>
|
||||
<div class="row-actions">
|
||||
<span class="original-domain"><?php echo esc_html($domain->original_domain . $domain->path); ?></span>
|
||||
</div>
|
||||
</a>
|
||||
</td>
|
||||
<td class="column-dns">
|
||||
<?php if ($health_data && isset($health_data['dns_status'])) : ?>
|
||||
<?php if ($health_data['dns_status'] === 'success') : ?>
|
||||
<span class="dashicons dashicons-yes-alt" style="color: #46b450;" title="<?php esc_attr_e('DNS correctly configured', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-warning" style="color: #dc3232;" title="<?php echo esc_attr($health_data['dns_message']); ?>"></span>
|
||||
<?php endif; ?>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-minus" style="color: #999;" title="<?php esc_attr_e('Not checked yet', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
<td class="column-ssl">
|
||||
<?php if ($health_data && isset($health_data['ssl_valid'])) : ?>
|
||||
<?php if ($health_data['ssl_valid']) : ?>
|
||||
<span class="dashicons dashicons-yes-alt" style="color: #46b450;" title="<?php esc_attr_e('SSL certificate valid', 'wp-domain-mapping'); ?>"></span>
|
||||
<div class="row-actions">
|
||||
<span><?php echo esc_html(sprintf(__('Expires: %s', 'wp-domain-mapping'), isset($health_data['ssl_expiry']) ? date('Y-m-d', strtotime($health_data['ssl_expiry'])) : '-')); ?></span>
|
||||
</div>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-warning" style="color: #dc3232;" title="<?php esc_attr_e('SSL certificate issue', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php endif; ?>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-minus" style="color: #999;" title="<?php esc_attr_e('Not checked yet', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
<td class="column-status">
|
||||
<?php if ($health_data && isset($health_data['accessible'])) : ?>
|
||||
<?php if ($health_data['accessible']) : ?>
|
||||
<span class="dashicons dashicons-yes-alt" style="color: #46b450;" title="<?php esc_attr_e('Site is accessible', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-warning" style="color: #dc3232;" title="<?php esc_attr_e('Site is not accessible', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php endif; ?>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-minus" style="color: #999;" title="<?php esc_attr_e('Not checked yet', 'wp-domain-mapping'); ?>"></span>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
<td class="column-last-check">
|
||||
<?php
|
||||
if ($health_data && isset($health_data['last_check'])) {
|
||||
echo esc_html(human_time_diff(strtotime($health_data['last_check']), current_time('timestamp'))) . ' ' . __('ago', 'wp-domain-mapping');
|
||||
} else {
|
||||
_e('Never', 'wp-domain-mapping');
|
||||
}
|
||||
?>
|
||||
</td>
|
||||
<td class="column-actions">
|
||||
<button type="button" class="button button-small check-domain-health" data-domain="<?php echo esc_attr($domain->domain); ?>">
|
||||
<?php _e('Check Now', 'wp-domain-mapping'); ?>
|
||||
</button>
|
||||
|
||||
<?php if ($health_data) : ?>
|
||||
<button type="button" class="button button-small show-health-details" data-domain="<?php echo esc_attr($domain->domain); ?>">
|
||||
<?php _e('Details', 'wp-domain-mapping'); ?>
|
||||
</button>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
</tr>
|
||||
<?php if ($health_data) : ?>
|
||||
<tr class="health-details-row" id="health-details-<?php echo esc_attr($domain_key); ?>" style="display: none;">
|
||||
<td colspan="7">
|
||||
<div class="health-details-content">
|
||||
<h4><?php echo sprintf(__('Health Details for %s', 'wp-domain-mapping'), esc_html($domain->domain)); ?></h4>
|
||||
|
||||
<table class="widefat striped" style="margin-top: 10px;">
|
||||
<tbody>
|
||||
<tr>
|
||||
<th><?php _e('Last Check', 'wp-domain-mapping'); ?></th>
|
||||
<td><?php echo isset($health_data['last_check']) ? esc_html(date('Y-m-d H:i:s', strtotime($health_data['last_check']))) : '-'; ?></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php _e('DNS Status', 'wp-domain-mapping'); ?></th>
|
||||
<td>
|
||||
<?php if (isset($health_data['dns_status'])) : ?>
|
||||
<?php if ($health_data['dns_status'] === 'success') : ?>
|
||||
<span class="dashicons dashicons-yes-alt" style="color: #46b450;"></span>
|
||||
<?php _e('DNS correctly configured', 'wp-domain-mapping'); ?>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-warning" style="color: #dc3232;"></span>
|
||||
<?php echo esc_html($health_data['dns_message']); ?>
|
||||
<?php endif; ?>
|
||||
<?php else : ?>
|
||||
<?php _e('Not checked', 'wp-domain-mapping'); ?>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php _e('Resolved IP', 'wp-domain-mapping'); ?></th>
|
||||
<td><?php echo isset($health_data['resolved_ip']) ? esc_html($health_data['resolved_ip']) : '-'; ?></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php _e('SSL Valid', 'wp-domain-mapping'); ?></th>
|
||||
<td>
|
||||
<?php if (isset($health_data['ssl_valid'])) : ?>
|
||||
<?php if ($health_data['ssl_valid']) : ?>
|
||||
<span class="dashicons dashicons-yes-alt" style="color: #46b450;"></span>
|
||||
<?php _e('Valid', 'wp-domain-mapping'); ?>
|
||||
<?php else : ?>
|
||||
<span class="dashicons dashicons-warning" style="color: #dc3232;"></span>
|
||||
<?php _e('Invalid', 'wp-domain-mapping'); ?>
|
||||
<?php endif; ?>
|
||||
<?php else : ?>
|
||||
<?php _e('Not checked', 'wp-domain-mapping'); ?>
|
||||
<?php endif; ?>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php _e('SSL Expiry', 'wp-domain-mapping'); ?></th>
|
||||
<td><?php echo isset($health_data['ssl_expiry']) ? esc_html(date('Y-m-d', strtotime($health_data['ssl_expiry']))) : '-'; ?></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php _e('Response Code', 'wp-domain-mapping'); ?></th>
|
||||
<td>
|
||||
<?php
|
||||
if (isset($health_data['response_code'])) {
|
||||
echo esc_html($health_data['response_code']);
|
||||
if ($health_data['response_code'] >= 200 && $health_data['response_code'] < 400) {
|
||||
echo ' <span class="dashicons dashicons-yes-alt" style="color: #46b450;"></span>';
|
||||
} else {
|
||||
echo ' <span class="dashicons dashicons-warning" style="color: #dc3232;"></span>';
|
||||
}
|
||||
} else {
|
||||
echo '-';
|
||||
}
|
||||
?>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
<?php endif; ?>
|
||||
<?php endforeach; ?>
|
||||
<?php else : ?>
|
||||
<tr>
|
||||
<td colspan="7"><?php _e('No domains found.', 'wp-domain-mapping'); ?></td>
|
||||
</tr>
|
||||
<?php endif; ?>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<div class="card domain-mapping-card" style="max-width: 100%; margin-top: 20px; padding: 20px; background: #fff; border: 1px solid #ccd0d4; border-radius: 4px; box-shadow: 0 1px 1px rgba(0,0,0,.04);">
|
||||
<h2><?php _e('Health Check Settings', 'wp-domain-mapping'); ?></h2>
|
||||
|
||||
<form method="post" action="">
|
||||
<?php wp_nonce_field('dm_health_settings', 'dm_health_settings_nonce'); ?>
|
||||
<input type="hidden" name="dm_health_settings" value="1">
|
||||
|
||||
<table class="form-table" role="presentation">
|
||||
<tr>
|
||||
<th scope="row"><?php _e('Automatic Health Checks', 'wp-domain-mapping'); ?></th>
|
||||
<td>
|
||||
<fieldset>
|
||||
<legend class="screen-reader-text"><span><?php _e('Automatic Health Checks', 'wp-domain-mapping'); ?></span></legend>
|
||||
<label for="health_check_enabled">
|
||||
<input name="health_check_enabled" type="checkbox" id="health_check_enabled" value="1" <?php checked(get_site_option('dm_health_check_enabled', true)); ?>>
|
||||
<?php _e('Enable automatic daily health checks', 'wp-domain-mapping'); ?>
|
||||
</label>
|
||||
</fieldset>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th scope="row"><?php _e('Email Notifications', 'wp-domain-mapping'); ?></th>
|
||||
<td>
|
||||
<fieldset>
|
||||
<legend class="screen-reader-text"><span><?php _e('Email Notifications', 'wp-domain-mapping'); ?></span></legend>
|
||||
<label for="health_notifications_enabled">
|
||||
<input name="health_notifications_enabled" type="checkbox" id="health_notifications_enabled" value="1" <?php checked(get_site_option('dm_health_notifications_enabled', true)); ?>>
|
||||
<?php _e('Send email notifications when domain health issues are detected', 'wp-domain-mapping'); ?>
|
||||
</label>
|
||||
</fieldset>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th scope="row"><label for="notification_email"><?php _e('Notification Email', 'wp-domain-mapping'); ?></label></th>
|
||||
<td>
|
||||
<input name="notification_email" type="email" id="notification_email" class="regular-text" value="<?php echo esc_attr(get_site_option('dm_notification_email', get_option('admin_email'))); ?>">
|
||||
<p class="description"><?php _e('Email address for domain health notifications.', 'wp-domain-mapping'); ?></p>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th scope="row"><label for="ssl_expiry_threshold"><?php _e('SSL Expiry Warning', 'wp-domain-mapping'); ?></label></th>
|
||||
<td>
|
||||
<input name="ssl_expiry_threshold" type="number" id="ssl_expiry_threshold" min="1" max="90" class="small-text" value="<?php echo esc_attr(get_site_option('dm_ssl_expiry_threshold', 14)); ?>">
|
||||
<span><?php _e('days', 'wp-domain-mapping'); ?></span>
|
||||
<p class="description"><?php _e('Send notifications when SSL certificates are expiring within this many days.', 'wp-domain-mapping'); ?></p>
|
||||
</td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
<p class="submit">
|
||||
<input type="submit" name="submit" id="submit" class="button button-primary" value="<?php esc_attr_e('Save Changes', 'wp-domain-mapping'); ?>">
|
||||
</p>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script type="text/javascript">
|
||||
jQuery(document).ready(function($) {
|
||||
// 单个域名健康检查
|
||||
$('.check-domain-health').on('click', function() {
|
||||
var $button = $(this);
|
||||
var domain = $button.data('domain');
|
||||
var $row = $button.closest('tr');
|
||||
|
||||
$button.prop('disabled', true).text('<?php esc_html_e('Checking...', 'wp-domain-mapping'); ?>');
|
||||
|
||||
$.ajax({
|
||||
url: ajaxurl,
|
||||
type: 'POST',
|
||||
data: {
|
||||
action: 'dm_check_domain_health',
|
||||
domain: domain,
|
||||
nonce: '<?php echo wp_create_nonce('dm_check_domain_health'); ?>'
|
||||
},
|
||||
dataType: 'json',
|
||||
success: function(response) {
|
||||
if (response.success) {
|
||||
// 刷新页面以显示更新的结果
|
||||
location.reload();
|
||||
} else {
|
||||
alert(response.data || '<?php esc_html_e('An error occurred during the health check.', 'wp-domain-mapping'); ?>');
|
||||
$button.prop('disabled', false).text('<?php esc_html_e('Check Now', 'wp-domain-mapping'); ?>');
|
||||
}
|
||||
},
|
||||
error: function() {
|
||||
alert('<?php esc_html_e('An error occurred during the health check.', 'wp-domain-mapping'); ?>');
|
||||
$button.prop('disabled', false).text('<?php esc_html_e('Check Now', 'wp-domain-mapping'); ?>');
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
// 显示/隐藏健康详情
|
||||
$('.show-health-details').on('click', function() {
|
||||
var domain = $(this).data('domain');
|
||||
var domainKey = md5(domain);
|
||||
$('#health-details-' + domainKey).toggle();
|
||||
});
|
||||
|
||||
// 简单MD5函数 (用于生成域名的唯一ID)
|
||||
function md5(string) {
|
||||
function md5cycle(x, k) {
|
||||
var a = x[0], b = x[1], c = x[2], d = x[3];
|
||||
|
||||
a = ff(a, b, c, d, k[0], 7, -680876936);
|
||||
d = ff(d, a, b, c, k[1], 12, -389564586);
|
||||
c = ff(c, d, a, b, k[2], 17, 606105819);
|
||||
b = ff(b, c, d, a, k[3], 22, -1044525330);
|
||||
// ...其余MD5算法...
|
||||
|
||||
x[0] = add32(a, x[0]);
|
||||
x[1] = add32(b, x[1]);
|
||||
x[2] = add32(c, x[2]);
|
||||
x[3] = add32(d, x[3]);
|
||||
}
|
||||
|
||||
function cmn(q, a, b, x, s, t) {
|
||||
a = add32(add32(a, q), add32(x, t));
|
||||
return add32((a << s) | (a >>> (32 - s)), b);
|
||||
}
|
||||
|
||||
function ff(a, b, c, d, x, s, t) {
|
||||
return cmn((b & c) | ((~b) & d), a, b, x, s, t);
|
||||
}
|
||||
|
||||
function add32(a, b) {
|
||||
return (a + b) & 0xFFFFFFFF;
|
||||
}
|
||||
|
||||
// 简化版本,仅用于客户端显示目的
|
||||
var hash = 0;
|
||||
if (string.length === 0) return hash;
|
||||
|
||||
for (var i = 0; i < string.length; i++) {
|
||||
var char = string.charCodeAt(i);
|
||||
hash = ((hash << 5) - hash) + char;
|
||||
hash = hash & hash; // 转换为32位整数
|
||||
}
|
||||
|
||||
return Math.abs(hash).toString(16);
|
||||
}
|
||||
});
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* AJAX处理域名健康检查
|
||||
*/
|
||||
public function ajax_check_domain_health() {
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_send_json_error(__('You do not have sufficient permissions to perform this action.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 验证nonce
|
||||
if (!isset($_POST['nonce']) || !wp_verify_nonce($_POST['nonce'], 'dm_check_domain_health')) {
|
||||
wp_send_json_error(__('Security check failed.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 获取域名
|
||||
$domain = isset($_POST['domain']) ? sanitize_text_field($_POST['domain']) : '';
|
||||
if (empty($domain)) {
|
||||
wp_send_json_error(__('No domain specified.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 执行健康检查
|
||||
$result = $this->check_domain_health($domain);
|
||||
|
||||
// 保存结果
|
||||
$this->save_health_check_result($domain, $result);
|
||||
|
||||
// 返回结果
|
||||
wp_send_json_success($result);
|
||||
}
|
||||
|
||||
/**
|
||||
* 计划任务:执行所有域名健康检查
|
||||
*/
|
||||
public function scheduled_health_check() {
|
||||
// 检查是否启用了自动健康检查
|
||||
if (!get_site_option('dm_health_check_enabled', true)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->run_health_check_for_all_domains();
|
||||
}
|
||||
|
||||
/**
|
||||
* 对所有域名执行健康检查
|
||||
*/
|
||||
private function run_health_check_for_all_domains() {
|
||||
global $wpdb;
|
||||
$table = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_DOMAINS;
|
||||
|
||||
// 获取所有域名
|
||||
$domains = $wpdb->get_col("SELECT domain FROM {$table}");
|
||||
|
||||
// 初始化通知信息
|
||||
$issues = array();
|
||||
|
||||
// 检查每个域名
|
||||
foreach ($domains as $domain) {
|
||||
$result = $this->check_domain_health($domain);
|
||||
$this->save_health_check_result($domain, $result);
|
||||
|
||||
// 检查是否有问题
|
||||
if ($this->has_health_issues($result)) {
|
||||
$issues[$domain] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
// 如果启用了通知并且有问题,发送通知
|
||||
if (!empty($issues) && get_site_option('dm_health_notifications_enabled', true)) {
|
||||
$this->send_health_notification($issues);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查域名健康状态
|
||||
*
|
||||
* @param string $domain 要检查的域名
|
||||
* @return array 健康检查结果
|
||||
*/
|
||||
private function check_domain_health($domain) {
|
||||
$result = array(
|
||||
'domain' => $domain,
|
||||
'last_check' => current_time('mysql'),
|
||||
'dns_status' => 'error',
|
||||
'dns_message' => __('DNS check not performed', 'wp-domain-mapping'),
|
||||
'resolved_ip' => '',
|
||||
'ssl_valid' => false,
|
||||
'ssl_expiry' => '',
|
||||
'accessible' => false,
|
||||
'response_code' => 0
|
||||
);
|
||||
|
||||
// 获取服务器IP或CNAME
|
||||
$server_ip = get_site_option('dm_ipaddress', '');
|
||||
$server_cname = get_site_option('dm_cname', '');
|
||||
|
||||
// 检查DNS设置
|
||||
$domain_ip = gethostbyname($domain);
|
||||
$result['resolved_ip'] = $domain_ip;
|
||||
|
||||
if ($domain_ip && $domain_ip !== $domain) {
|
||||
if ($server_ip && strpos($server_ip, $domain_ip) !== false) {
|
||||
$result['dns_status'] = 'success';
|
||||
$result['dns_message'] = __('Domain A record is correctly pointing to server IP.', 'wp-domain-mapping');
|
||||
} elseif ($server_cname && function_exists('dns_get_record')) {
|
||||
$dns_records = @dns_get_record($domain, DNS_CNAME);
|
||||
$has_valid_cname = false;
|
||||
|
||||
if ($dns_records) {
|
||||
foreach ($dns_records as $record) {
|
||||
if (isset($record['target']) &&
|
||||
(
|
||||
$record['target'] === $server_cname ||
|
||||
strpos($record['target'], $server_cname) !== false
|
||||
)) {
|
||||
$has_valid_cname = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ($has_valid_cname) {
|
||||
$result['dns_status'] = 'success';
|
||||
$result['dns_message'] = __('Domain CNAME record is correctly configured.', 'wp-domain-mapping');
|
||||
} else {
|
||||
$result['dns_message'] = __('Domain is not pointing to the correct server.', 'wp-domain-mapping');
|
||||
}
|
||||
} else {
|
||||
$result['dns_message'] = __('Cannot verify DNS configuration.', 'wp-domain-mapping');
|
||||
}
|
||||
} else {
|
||||
$result['dns_message'] = __('Domain does not resolve to an IP address.', 'wp-domain-mapping');
|
||||
}
|
||||
|
||||
// 检查网站可访问性和SSL
|
||||
$response = $this->test_domain_connection($domain);
|
||||
|
||||
if ($response) {
|
||||
$result['accessible'] = $response['accessible'];
|
||||
$result['response_code'] = $response['response_code'];
|
||||
$result['ssl_valid'] = $response['ssl_valid'];
|
||||
$result['ssl_expiry'] = $response['ssl_expiry'];
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 测试域名连接
|
||||
*
|
||||
* @param string $domain 域名
|
||||
* @return array|false 连接测试结果
|
||||
*/
|
||||
private function test_domain_connection($domain) {
|
||||
if (!function_exists('curl_init')) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$result = array(
|
||||
'accessible' => false,
|
||||
'response_code' => 0,
|
||||
'ssl_valid' => false,
|
||||
'ssl_expiry' => ''
|
||||
);
|
||||
|
||||
// 测试HTTPS连接
|
||||
$ch = curl_init('https://' . $domain);
|
||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($ch, CURLOPT_HEADER, true);
|
||||
curl_setopt($ch, CURLOPT_NOBODY, true);
|
||||
curl_setopt($ch, CURLOPT_TIMEOUT, 10);
|
||||
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
|
||||
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 2);
|
||||
|
||||
$response = curl_exec($ch);
|
||||
$response_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
$result['response_code'] = $response_code;
|
||||
|
||||
if ($response !== false && $response_code > 0) {
|
||||
$result['accessible'] = ($response_code >= 200 && $response_code < 400);
|
||||
$result['ssl_valid'] = ($response !== false);
|
||||
|
||||
// 获取SSL证书信息
|
||||
$ssl_info = curl_getinfo($ch, CURLINFO_CERTINFO);
|
||||
if (!empty($ssl_info) && isset($ssl_info[0]['Expire date'])) {
|
||||
$result['ssl_expiry'] = $ssl_info[0]['Expire date'];
|
||||
}
|
||||
}
|
||||
|
||||
curl_close($ch);
|
||||
|
||||
// 如果HTTPS失败,尝试HTTP
|
||||
if (!$result['accessible']) {
|
||||
$ch = curl_init('http://' . $domain);
|
||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($ch, CURLOPT_HEADER, true);
|
||||
curl_setopt($ch, CURLOPT_NOBODY, true);
|
||||
curl_setopt($ch, CURLOPT_TIMEOUT, 10);
|
||||
|
||||
$response = curl_exec($ch);
|
||||
$response_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
|
||||
if ($response !== false && $response_code > 0) {
|
||||
$result['accessible'] = ($response_code >= 200 && $response_code < 400);
|
||||
$result['response_code'] = $response_code;
|
||||
}
|
||||
|
||||
curl_close($ch);
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
/**
|
||||
* 保存健康检查结果
|
||||
*
|
||||
* @param string $domain 域名
|
||||
* @param array $result 健康检查结果
|
||||
*/
|
||||
private function save_health_check_result($domain, $result) {
|
||||
$health_results = get_site_option('dm_domain_health_results', array());
|
||||
$domain_key = md5($domain);
|
||||
|
||||
$health_results[$domain_key] = $result;
|
||||
update_site_option('dm_domain_health_results', $health_results);
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查结果是否有健康问题
|
||||
*
|
||||
* @param array $result 健康检查结果
|
||||
* @return bool 是否有问题
|
||||
*/
|
||||
private function has_health_issues($result) {
|
||||
// 检查DNS问题
|
||||
if ($result['dns_status'] !== 'success') {
|
||||
return true;
|
||||
}
|
||||
|
||||
// 检查SSL问题
|
||||
if (!$result['ssl_valid']) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// 检查SSL即将到期
|
||||
if (!empty($result['ssl_expiry'])) {
|
||||
$expiry_date = strtotime($result['ssl_expiry']);
|
||||
$threshold = get_site_option('dm_ssl_expiry_threshold', 14);
|
||||
$threshold_date = strtotime('+' . $threshold . ' days');
|
||||
|
||||
if ($expiry_date <= $threshold_date) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
// 检查可访问性问题
|
||||
if (!$result['accessible']) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 发送健康问题通知
|
||||
*
|
||||
* @param array $issues 有问题的域名及其健康检查结果
|
||||
*/
|
||||
private function send_health_notification($issues) {
|
||||
$notification_email = get_site_option('dm_notification_email', get_option('admin_email'));
|
||||
|
||||
if (empty($notification_email)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$site_name = get_bloginfo('name');
|
||||
$subject = sprintf(__('[%s] Domain Mapping Health Alert', 'wp-domain-mapping'), $site_name);
|
||||
|
||||
// 构建邮件内容
|
||||
$message = sprintf(__('Domain health issues were detected on %s.', 'wp-domain-mapping'), $site_name) . "\n\n";
|
||||
$message .= __('The following domains have issues:', 'wp-domain-mapping') . "\n\n";
|
||||
|
||||
foreach ($issues as $domain => $result) {
|
||||
$message .= sprintf(__('Domain: %s', 'wp-domain-mapping'), $domain) . "\n";
|
||||
|
||||
// 添加DNS状态
|
||||
if ($result['dns_status'] !== 'success') {
|
||||
$message .= " - " . sprintf(__('DNS Issue: %s', 'wp-domain-mapping'), $result['dns_message']) . "\n";
|
||||
}
|
||||
|
||||
// 添加SSL状态
|
||||
if (!$result['ssl_valid']) {
|
||||
$message .= " - " . __('SSL Certificate is invalid or missing.', 'wp-domain-mapping') . "\n";
|
||||
} elseif (!empty($result['ssl_expiry'])) {
|
||||
$expiry_date = strtotime($result['ssl_expiry']);
|
||||
$threshold = get_site_option('dm_ssl_expiry_threshold', 14);
|
||||
$threshold_date = strtotime('+' . $threshold . ' days');
|
||||
|
||||
if ($expiry_date <= $threshold_date) {
|
||||
$message .= " - " . sprintf(
|
||||
__('SSL Certificate expires on %s (within %d days).', 'wp-domain-mapping'),
|
||||
date('Y-m-d', $expiry_date),
|
||||
$threshold
|
||||
) . "\n";
|
||||
}
|
||||
}
|
||||
|
||||
// 添加可访问性状态
|
||||
if (!$result['accessible']) {
|
||||
$message .= " - " . __('Site is not accessible.', 'wp-domain-mapping') . "\n";
|
||||
if ($result['response_code'] > 0) {
|
||||
$message .= " " . sprintf(__('HTTP Response Code: %d', 'wp-domain-mapping'), $result['response_code']) . "\n";
|
||||
}
|
||||
}
|
||||
|
||||
$message .= "\n";
|
||||
}
|
||||
|
||||
// 添加解决方案链接
|
||||
$message .= sprintf(
|
||||
__('To view and manage these issues, please visit: %s', 'wp-domain-mapping'),
|
||||
network_admin_url('settings.php?page=domain-mapping-health')
|
||||
) . "\n";
|
||||
|
||||
// 发送邮件
|
||||
wp_mail($notification_email, $subject, $message);
|
||||
}
|
||||
}
|
592
includes/class-importer.php
Normal file
592
includes/class-importer.php
Normal file
|
@ -0,0 +1,592 @@
|
|||
<?php
|
||||
/**
|
||||
* 域名映射导入/导出功能
|
||||
*
|
||||
* 处理域名映射的批量导入和导出
|
||||
*
|
||||
* @package WP Domain Mapping
|
||||
*/
|
||||
|
||||
// 防止直接访问
|
||||
if (!defined('ABSPATH')) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* 域名映射导入/导出类
|
||||
*/
|
||||
class WP_Domain_Mapping_Importer {
|
||||
|
||||
/**
|
||||
* 类实例
|
||||
*
|
||||
* @var WP_Domain_Mapping_Importer
|
||||
*/
|
||||
protected static $instance;
|
||||
|
||||
/**
|
||||
* 数据库实例
|
||||
*/
|
||||
private $db;
|
||||
|
||||
/**
|
||||
* 获取类实例
|
||||
*
|
||||
* @return WP_Domain_Mapping_Importer
|
||||
*/
|
||||
public static function get_instance() {
|
||||
if (is_null(self::$instance)) {
|
||||
self::$instance = new self();
|
||||
}
|
||||
|
||||
return self::$instance;
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
*/
|
||||
private function __construct() {
|
||||
$this->db = WP_Domain_Mapping_DB::get_instance();
|
||||
|
||||
// 添加菜单页面
|
||||
add_action('network_admin_menu', array($this, 'add_menu_page'), 20);
|
||||
|
||||
// 处理表单提交
|
||||
add_action('admin_init', array($this, 'handle_form_submission'));
|
||||
|
||||
// AJAX处理程序
|
||||
add_action('wp_ajax_dm_import_csv', array($this, 'ajax_import_csv'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加导入/导出菜单
|
||||
*/
|
||||
public function add_menu_page() {
|
||||
add_submenu_page(
|
||||
'settings.php',
|
||||
__('Import/Export Domains', 'wp-domain-mapping'),
|
||||
__('Import/Export Domains', 'wp-domain-mapping'),
|
||||
'manage_network',
|
||||
'domain-mapping-import-export',
|
||||
array($this, 'render_page')
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* 渲染导入/导出页面
|
||||
*/
|
||||
public function render_page() {
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to access this page.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 输出成功消息(如果有)
|
||||
if (isset($_GET['imported']) && $_GET['imported']) {
|
||||
$count = intval($_GET['imported']);
|
||||
echo '<div class="notice notice-success is-dismissible"><p>' .
|
||||
sprintf(
|
||||
_n(
|
||||
'%d domain mapping imported successfully.',
|
||||
'%d domain mappings imported successfully.',
|
||||
$count,
|
||||
'wp-domain-mapping'
|
||||
),
|
||||
$count
|
||||
) .
|
||||
'</p></div>';
|
||||
}
|
||||
|
||||
if (isset($_GET['export']) && $_GET['export'] == 'success') {
|
||||
echo '<div class="notice notice-success is-dismissible"><p>' .
|
||||
__('Domain mappings exported successfully.', 'wp-domain-mapping') .
|
||||
'</p></div>';
|
||||
}
|
||||
|
||||
// 显示页面
|
||||
?>
|
||||
<div class="wrap">
|
||||
<h1><?php echo esc_html(get_admin_page_title()); ?></h1>
|
||||
|
||||
<div class="card domain-mapping-card" style="max-width: 100%; margin-top: 20px; padding: 20px; background: #fff; border: 1px solid #ccd0d4; border-radius: 4px; box-shadow: 0 1px 1px rgba(0,0,0,.04);">
|
||||
<h2><?php _e('Export Domain Mappings', 'wp-domain-mapping'); ?></h2>
|
||||
<p><?php _e('Export all domain mappings to a CSV file.', 'wp-domain-mapping'); ?></p>
|
||||
|
||||
<form method="post" action="">
|
||||
<?php wp_nonce_field('domain_mapping_export', 'domain_mapping_export_nonce'); ?>
|
||||
<input type="hidden" name="domain_mapping_export" value="1">
|
||||
|
||||
<div style="margin-bottom: 15px;">
|
||||
<label>
|
||||
<input type="checkbox" name="include_header" value="1" checked>
|
||||
<?php _e('Include column headers', 'wp-domain-mapping'); ?>
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div style="margin-bottom: 15px;">
|
||||
<label for="blog_id_filter"><?php _e('Export for specific site ID (optional):', 'wp-domain-mapping'); ?></label>
|
||||
<input type="number" id="blog_id_filter" name="blog_id_filter" min="1" class="regular-text">
|
||||
<p class="description"><?php _e('Leave empty to export all domains.', 'wp-domain-mapping'); ?></p>
|
||||
</div>
|
||||
|
||||
<p>
|
||||
<input type="submit" class="button button-primary" value="<?php esc_attr_e('Export to CSV', 'wp-domain-mapping'); ?>">
|
||||
</p>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<div class="card domain-mapping-card" style="max-width: 100%; margin-top: 20px; padding: 20px; background: #fff; border: 1px solid #ccd0d4; border-radius: 4px; box-shadow: 0 1px 1px rgba(0,0,0,.04);">
|
||||
<h2><?php _e('Import Domain Mappings', 'wp-domain-mapping'); ?></h2>
|
||||
<p><?php _e('Import domain mappings from a CSV file.', 'wp-domain-mapping'); ?></p>
|
||||
|
||||
<form method="post" enctype="multipart/form-data" id="domain-mapping-import-form">
|
||||
<?php wp_nonce_field('domain_mapping_import', 'domain_mapping_import_nonce'); ?>
|
||||
<input type="hidden" name="domain_mapping_import" value="1">
|
||||
|
||||
<div style="margin-bottom: 15px;">
|
||||
<label for="csv_file"><?php _e('CSV File:', 'wp-domain-mapping'); ?></label><br>
|
||||
<input type="file" id="csv_file" name="csv_file" accept=".csv" required>
|
||||
<p class="description">
|
||||
<?php _e('The CSV file should have the columns: blog_id, domain, active (1 or 0).', 'wp-domain-mapping'); ?><br>
|
||||
<?php _e('Example: 1,example.com,1', 'wp-domain-mapping'); ?>
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div style="margin-bottom: 15px;">
|
||||
<label>
|
||||
<input type="checkbox" name="has_header" value="1" checked>
|
||||
<?php _e('First row contains column headers', 'wp-domain-mapping'); ?>
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div style="margin-bottom: 15px;">
|
||||
<label>
|
||||
<input type="checkbox" name="update_existing" value="1" checked>
|
||||
<?php _e('Update existing mappings', 'wp-domain-mapping'); ?>
|
||||
</label>
|
||||
<p class="description"><?php _e('If unchecked, will skip domains that already exist.', 'wp-domain-mapping'); ?></p>
|
||||
</div>
|
||||
|
||||
<div style="margin-bottom: 15px;">
|
||||
<label>
|
||||
<input type="checkbox" name="validate_sites" value="1" checked>
|
||||
<?php _e('Validate site IDs', 'wp-domain-mapping'); ?>
|
||||
</label>
|
||||
<p class="description"><?php _e('If checked, will only import domains for existing sites.', 'wp-domain-mapping'); ?></p>
|
||||
</div>
|
||||
|
||||
<p>
|
||||
<input type="submit" class="button button-primary" value="<?php esc_attr_e('Import from CSV', 'wp-domain-mapping'); ?>">
|
||||
</p>
|
||||
</form>
|
||||
|
||||
<div id="import-progress" style="display: none; margin-top: 20px;">
|
||||
<p><?php _e('Processing import...', 'wp-domain-mapping'); ?></p>
|
||||
<div class="progress-bar-outer" style="background-color: #f0f0f1; border-radius: 4px; height: 20px; width: 100%; overflow: hidden;">
|
||||
<div class="progress-bar-inner" style="background-color: #2271b1; height: 100%; width: 0%;"></div>
|
||||
</div>
|
||||
<p class="progress-text">0%</p>
|
||||
</div>
|
||||
|
||||
<div id="import-results" style="display: none; margin-top: 20px;">
|
||||
<h3><?php _e('Import Results', 'wp-domain-mapping'); ?></h3>
|
||||
<div class="import-summary"></div>
|
||||
<div class="import-details"></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="card domain-mapping-card" style="max-width: 100%; margin-top: 20px; padding: 20px; background: #fff; border: 1px solid #ccd0d4; border-radius: 4px; box-shadow: 0 1px 1px rgba(0,0,0,.04);">
|
||||
<h2><?php _e('CSV Format', 'wp-domain-mapping'); ?></h2>
|
||||
<p><?php _e('The CSV file should follow this format:', 'wp-domain-mapping'); ?></p>
|
||||
|
||||
<table class="widefat" style="margin-top: 10px;">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>blog_id</th>
|
||||
<th>domain</th>
|
||||
<th>active</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr>
|
||||
<td>1</td>
|
||||
<td>example.com</td>
|
||||
<td>1</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td>2</td>
|
||||
<td>example.org</td>
|
||||
<td>0</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<ul style="margin-top: 15px;">
|
||||
<li><strong>blog_id</strong>: <?php _e('The ID of the WordPress site (required)', 'wp-domain-mapping'); ?></li>
|
||||
<li><strong>domain</strong>: <?php _e('The domain name without http:// or https:// (required)', 'wp-domain-mapping'); ?></li>
|
||||
<li><strong>active</strong>: <?php _e('Set to 1 to make this the primary domain, 0 otherwise (required)', 'wp-domain-mapping'); ?></li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script type="text/javascript">
|
||||
jQuery(document).ready(function($) {
|
||||
$('#domain-mapping-import-form').on('submit', function(e) {
|
||||
e.preventDefault();
|
||||
|
||||
var formData = new FormData(this);
|
||||
formData.append('action', 'dm_import_csv');
|
||||
|
||||
// 显示进度条
|
||||
$('#import-progress').show();
|
||||
|
||||
$.ajax({
|
||||
url: ajaxurl,
|
||||
type: 'POST',
|
||||
data: formData,
|
||||
dataType: 'json',
|
||||
contentType: false,
|
||||
processData: false,
|
||||
success: function(response) {
|
||||
// 隐藏进度条
|
||||
$('#import-progress').hide();
|
||||
|
||||
// 显示结果
|
||||
$('#import-results').show();
|
||||
|
||||
if (response.success) {
|
||||
$('.import-summary').html(
|
||||
'<div class="notice notice-success"><p>' +
|
||||
response.data.message +
|
||||
'</p></div>'
|
||||
);
|
||||
|
||||
var details = '<table class="widefat">' +
|
||||
'<thead><tr>' +
|
||||
'<th>Status</th>' +
|
||||
'<th>Details</th>' +
|
||||
'</tr></thead><tbody>';
|
||||
|
||||
$.each(response.data.details, function(i, item) {
|
||||
var statusClass = 'notice-success';
|
||||
if (item.status === 'error') {
|
||||
statusClass = 'notice-error';
|
||||
} else if (item.status === 'warning') {
|
||||
statusClass = 'notice-warning';
|
||||
}
|
||||
|
||||
details += '<tr class="' + statusClass + '">' +
|
||||
'<td>' + item.status + '</td>' +
|
||||
'<td>' + item.message + '</td>' +
|
||||
'</tr>';
|
||||
});
|
||||
|
||||
details += '</tbody></table>';
|
||||
$('.import-details').html(details);
|
||||
} else {
|
||||
$('.import-summary').html(
|
||||
'<div class="notice notice-error"><p>' +
|
||||
response.data +
|
||||
'</p></div>'
|
||||
);
|
||||
}
|
||||
},
|
||||
error: function() {
|
||||
// 隐藏进度条
|
||||
$('#import-progress').hide();
|
||||
|
||||
// 显示错误
|
||||
$('#import-results').show();
|
||||
$('.import-summary').html(
|
||||
'<div class="notice notice-error"><p>' +
|
||||
'<?php _e('An error occurred during import.', 'wp-domain-mapping'); ?>' +
|
||||
'</p></div>'
|
||||
);
|
||||
},
|
||||
xhr: function() {
|
||||
var xhr = new window.XMLHttpRequest();
|
||||
|
||||
// 上传进度
|
||||
xhr.upload.addEventListener('progress', function(evt) {
|
||||
if (evt.lengthComputable) {
|
||||
var percentComplete = evt.loaded / evt.total * 100;
|
||||
$('.progress-bar-inner').css('width', percentComplete + '%');
|
||||
$('.progress-text').text(Math.round(percentComplete) + '%');
|
||||
}
|
||||
}, false);
|
||||
|
||||
return xhr;
|
||||
}
|
||||
});
|
||||
});
|
||||
});
|
||||
</script>
|
||||
<?php
|
||||
}
|
||||
|
||||
/**
|
||||
* 处理表单提交
|
||||
*/
|
||||
public function handle_form_submission() {
|
||||
// 处理导出
|
||||
if (isset($_POST['domain_mapping_export']) && $_POST['domain_mapping_export']) {
|
||||
$this->handle_export();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 处理CSV导出
|
||||
*/
|
||||
private function handle_export() {
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to export data.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 验证nonce
|
||||
if (!isset($_POST['domain_mapping_export_nonce']) || !wp_verify_nonce($_POST['domain_mapping_export_nonce'], 'domain_mapping_export')) {
|
||||
wp_die(__('Invalid security token. Please try again.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 获取选项
|
||||
$include_header = isset($_POST['include_header']) ? (bool) $_POST['include_header'] : false;
|
||||
$blog_id_filter = isset($_POST['blog_id_filter']) && !empty($_POST['blog_id_filter']) ? intval($_POST['blog_id_filter']) : 0;
|
||||
|
||||
// 获取域名映射数据
|
||||
global $wpdb;
|
||||
$table = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_DOMAINS;
|
||||
$sql = "SELECT blog_id, domain, active FROM {$table}";
|
||||
|
||||
if ($blog_id_filter > 0) {
|
||||
$sql .= $wpdb->prepare(" WHERE blog_id = %d", $blog_id_filter);
|
||||
}
|
||||
|
||||
$domains = $wpdb->get_results($sql, ARRAY_A);
|
||||
|
||||
if (empty($domains)) {
|
||||
// 没有数据
|
||||
wp_redirect(add_query_arg(array('page' => 'domain-mapping-import-export', 'export' => 'empty'), network_admin_url('settings.php')));
|
||||
exit;
|
||||
}
|
||||
|
||||
// 设置CSV输出
|
||||
$filename = 'domain-mappings-' . date('Y-m-d') . '.csv';
|
||||
|
||||
header('Content-Type: text/csv; charset=utf-8');
|
||||
header('Content-Disposition: attachment; filename=' . $filename);
|
||||
|
||||
$output = fopen('php://output', 'w');
|
||||
|
||||
// 添加标题行
|
||||
if ($include_header) {
|
||||
fputcsv($output, array('blog_id', 'domain', 'active'));
|
||||
}
|
||||
|
||||
// 添加数据行
|
||||
foreach ($domains as $domain) {
|
||||
fputcsv($output, $domain);
|
||||
}
|
||||
|
||||
fclose($output);
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* AJAX处理CSV导入
|
||||
*/
|
||||
public function ajax_import_csv() {
|
||||
// 检查权限
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_send_json_error(__('You do not have sufficient permissions to import data.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 验证nonce
|
||||
if (!isset($_POST['domain_mapping_import_nonce']) || !wp_verify_nonce($_POST['domain_mapping_import_nonce'], 'domain_mapping_import')) {
|
||||
wp_send_json_error(__('Invalid security token. Please try again.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 检查文件
|
||||
if (!isset($_FILES['csv_file']) || $_FILES['csv_file']['error'] != UPLOAD_ERR_OK) {
|
||||
wp_send_json_error(__('No file uploaded or upload error.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 获取选项
|
||||
$has_header = isset($_POST['has_header']) ? (bool) $_POST['has_header'] : false;
|
||||
$update_existing = isset($_POST['update_existing']) ? (bool) $_POST['update_existing'] : false;
|
||||
$validate_sites = isset($_POST['validate_sites']) ? (bool) $_POST['validate_sites'] : true;
|
||||
|
||||
// 打开文件
|
||||
$file = fopen($_FILES['csv_file']['tmp_name'], 'r');
|
||||
if (!$file) {
|
||||
wp_send_json_error(__('Could not open the uploaded file.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// 初始化计数器和日志
|
||||
$imported = 0;
|
||||
$skipped = 0;
|
||||
$errors = 0;
|
||||
$log = array();
|
||||
|
||||
// 跳过标题行
|
||||
if ($has_header) {
|
||||
fgetcsv($file);
|
||||
}
|
||||
|
||||
// 处理每一行
|
||||
$core = WP_Domain_Mapping_Core::get_instance();
|
||||
$row_num = $has_header ? 2 : 1; // 考虑标题行
|
||||
|
||||
while (($data = fgetcsv($file)) !== false) {
|
||||
// 检查数据格式
|
||||
if (count($data) < 3) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Invalid format. Expected at least 3 columns.', 'wp-domain-mapping'), $row_num)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// 解析数据
|
||||
$blog_id = intval($data[0]);
|
||||
$domain = $core->clean_domain(trim($data[1]));
|
||||
$active = intval($data[2]);
|
||||
|
||||
// 验证blog_id
|
||||
if ($blog_id <= 0) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Invalid blog ID: %d', 'wp-domain-mapping'), $row_num, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// 验证站点是否存在
|
||||
if ($validate_sites && !get_blog_details($blog_id)) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Site ID %d does not exist.', 'wp-domain-mapping'), $row_num, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// 验证域名格式
|
||||
if (!$core->validate_domain($domain)) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Invalid domain format: %s', 'wp-domain-mapping'), $row_num, $domain)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// 检查域名是否已经存在于其他站点
|
||||
global $wpdb;
|
||||
$table = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_DOMAINS;
|
||||
$existing = $wpdb->get_row($wpdb->prepare(
|
||||
"SELECT * FROM $table WHERE domain = %s",
|
||||
$domain
|
||||
));
|
||||
|
||||
if ($existing) {
|
||||
if ($existing->blog_id != $blog_id) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Domain %s is already mapped to blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $existing->blog_id)
|
||||
);
|
||||
$errors++;
|
||||
} elseif (!$update_existing) {
|
||||
$log[] = array(
|
||||
'status' => 'warning',
|
||||
'message' => sprintf(__('Row %d: Domain %s already exists for blog ID %d. Skipped.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$skipped++;
|
||||
} else {
|
||||
// 更新现有域名
|
||||
$result = $wpdb->update(
|
||||
$table,
|
||||
array('active' => $active),
|
||||
array('domain' => $domain),
|
||||
array('%d'),
|
||||
array('%s')
|
||||
);
|
||||
|
||||
if ($result !== false) {
|
||||
$log[] = array(
|
||||
'status' => 'success',
|
||||
'message' => sprintf(__('Row %d: Updated domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$imported++;
|
||||
} else {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Failed to update domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// 添加新域名
|
||||
$result = $wpdb->insert(
|
||||
$table,
|
||||
array(
|
||||
'blog_id' => $blog_id,
|
||||
'domain' => $domain,
|
||||
'active' => $active
|
||||
),
|
||||
array('%d', '%s', '%d')
|
||||
);
|
||||
|
||||
if ($result) {
|
||||
// 记录操作
|
||||
$this->db->log_action('import', $domain, $blog_id);
|
||||
|
||||
// 清除缓存
|
||||
$this->db->invalidate_domain_cache($blog_id);
|
||||
|
||||
$log[] = array(
|
||||
'status' => 'success',
|
||||
'message' => sprintf(__('Row %d: Added domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$imported++;
|
||||
} else {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Failed to add domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
}
|
||||
}
|
||||
|
||||
$row_num++;
|
||||
}
|
||||
|
||||
fclose($file);
|
||||
|
||||
// 构建响应
|
||||
$message = sprintf(
|
||||
__('Import completed: %d imported, %d skipped, %d errors.', 'wp-domain-mapping'),
|
||||
$imported, $skipped, $errors
|
||||
);
|
||||
|
||||
wp_send_json_success(array(
|
||||
'message' => $message,
|
||||
'imported' => $imported,
|
||||
'skipped' => $skipped,
|
||||
'errors' => $errors,
|
||||
'details' => $log
|
||||
));
|
||||
}
|
||||
}
|
132
includes/class-site-id.php
Normal file
132
includes/class-site-id.php
Normal file
|
@ -0,0 +1,132 @@
|
|||
<?php
|
||||
/**
|
||||
* 站点ID显示功能
|
||||
*
|
||||
* 显示当前WordPress多站点的站点ID
|
||||
*
|
||||
* @package WP Domain Mapping
|
||||
*/
|
||||
|
||||
// 防止直接访问
|
||||
if (!defined('ABSPATH')) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* WordPress站点ID显示类
|
||||
*
|
||||
* 基于WP Show Site ID插件 (by Emanuela Castorina)
|
||||
*/
|
||||
class WP_Domain_Mapping_Site_ID {
|
||||
|
||||
/**
|
||||
* 类实例
|
||||
*
|
||||
* @var WP_Domain_Mapping_Site_ID
|
||||
*/
|
||||
protected static $instance;
|
||||
|
||||
/**
|
||||
* 获取类实例
|
||||
*
|
||||
* @return WP_Domain_Mapping_Site_ID
|
||||
*/
|
||||
public static function get_instance() {
|
||||
if (is_null(self::$instance)) {
|
||||
self::$instance = new self();
|
||||
}
|
||||
|
||||
return self::$instance;
|
||||
}
|
||||
|
||||
/**
|
||||
* 构造函数
|
||||
*/
|
||||
private function __construct() {
|
||||
// 只在多站点环境中工作
|
||||
if (!is_multisite()) {
|
||||
return;
|
||||
}
|
||||
|
||||
// 添加站点ID到WordPress工具栏
|
||||
add_action('admin_bar_menu', array($this, 'add_toolbar_items'), 100);
|
||||
|
||||
// 添加站点ID列到网络管理员的站点列表页面
|
||||
add_filter('manage_sites-network_columns', array($this, 'manage_sites_columns'), 20);
|
||||
add_action('manage_sites_custom_column', array($this, 'show_site_id'), 10, 2);
|
||||
add_action('admin_print_styles', array($this, 'custom_style'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加自定义样式
|
||||
*/
|
||||
public function custom_style() {
|
||||
if ('sites-network' == get_current_screen()->id) {
|
||||
?>
|
||||
<style type="text/css">
|
||||
th#dm_site_id { width: 3.5em; }
|
||||
.dm-site-id-badge {
|
||||
display: inline-block;
|
||||
background: #2271b1;
|
||||
color: #fff;
|
||||
font-weight: 500;
|
||||
padding: 0 5px;
|
||||
border-radius: 3px;
|
||||
font-size: 12px;
|
||||
}
|
||||
</style>
|
||||
<?php
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 向WordPress工具栏添加站点ID
|
||||
*
|
||||
* @param WP_Admin_Bar $admin_bar WordPress工具栏对象
|
||||
*/
|
||||
public function add_toolbar_items($admin_bar) {
|
||||
// 仅向管理员或网络管理员显示
|
||||
if (!current_user_can('manage_options') && !is_super_admin()) {
|
||||
return;
|
||||
}
|
||||
|
||||
$blog_id = get_current_blog_id();
|
||||
|
||||
$admin_bar->add_menu(array(
|
||||
'id' => 'wp-site-ID',
|
||||
'title' => sprintf(__('Site ID: %d', 'wp-domain-mapping'), $blog_id),
|
||||
'href' => is_super_admin() ? esc_url(network_admin_url('site-info.php?id=' . $blog_id)) : '#',
|
||||
'meta' => array(
|
||||
'title' => is_super_admin() ? __('Edit this site', 'wp-domain-mapping') : __('Current site ID', 'wp-domain-mapping'),
|
||||
'class' => 'dm-site-id-menu'
|
||||
),
|
||||
));
|
||||
}
|
||||
|
||||
/**
|
||||
* 向站点列表添加ID列
|
||||
*
|
||||
* @param array $columns 当前列
|
||||
* @return array 修改后的列
|
||||
*/
|
||||
public function manage_sites_columns($columns) {
|
||||
// 在第一列后添加站点ID列
|
||||
$columns = array_slice($columns, 0, 1, true) +
|
||||
array('dm_site_id' => __('Site ID', 'wp-domain-mapping')) +
|
||||
array_slice($columns, 1, count($columns) - 1, true);
|
||||
|
||||
return $columns;
|
||||
}
|
||||
|
||||
/**
|
||||
* 显示站点ID列的内容
|
||||
*
|
||||
* @param string $column 列名
|
||||
* @param int $blog_id 站点ID
|
||||
*/
|
||||
public function show_site_id($column, $blog_id) {
|
||||
if ('dm_site_id' == $column) {
|
||||
echo '<span class="dm-site-id-badge">' . esc_html($blog_id) . '</span>';
|
||||
}
|
||||
}
|
||||
}
|
709
includes/class-tools.php
Normal file
709
includes/class-tools.php
Normal file
|
@ -0,0 +1,709 @@
|
|||
<?php
|
||||
// Prevent direct access
|
||||
if (!defined('ABSPATH')) {
|
||||
exit;
|
||||
}
|
||||
|
||||
class WP_Domain_Mapping_Tools {
|
||||
private static $instance = null;
|
||||
private $db;
|
||||
private $core;
|
||||
|
||||
public static function get_instance() {
|
||||
if (is_null(self::$instance)) {
|
||||
self::$instance = new self();
|
||||
}
|
||||
return self::$instance;
|
||||
}
|
||||
|
||||
private function __construct() {
|
||||
$this->db = WP_Domain_Mapping_DB::get_instance();
|
||||
$this->core = WP_Domain_Mapping_Core::get_instance();
|
||||
|
||||
// Setup hooks
|
||||
$this->setup_hooks();
|
||||
}
|
||||
|
||||
private function setup_hooks() {
|
||||
// Add menu pages for tools
|
||||
add_action('network_admin_menu', array($this, 'add_menu_pages'), 20);
|
||||
|
||||
// Add Site ID to toolbar
|
||||
add_action('admin_bar_menu', array($this, 'add_site_id_to_toolbar'), 100);
|
||||
|
||||
// Add Site ID column to sites list
|
||||
add_filter('manage_sites-network_columns', array($this, 'add_site_id_column'), 20);
|
||||
add_action('manage_sites_custom_column', array($this, 'display_site_id_column'), 10, 2);
|
||||
add_action('admin_print_styles', array($this, 'site_id_column_style'));
|
||||
|
||||
// AJAX handlers
|
||||
add_action('wp_ajax_dm_check_domain_health', array($this, 'ajax_check_domain_health'));
|
||||
add_action('wp_ajax_dm_import_csv', array($this, 'ajax_import_csv'));
|
||||
|
||||
// Scheduled tasks
|
||||
add_action('dm_domain_health_check', array($this, 'scheduled_health_check'));
|
||||
|
||||
// Admin init actions
|
||||
add_action('admin_init', array($this, 'handle_export'));
|
||||
add_action('admin_init', array($this, 'handle_health_manual_check'));
|
||||
add_action('admin_init', array($this, 'handle_health_settings_save'));
|
||||
}
|
||||
|
||||
public function add_menu_pages() {
|
||||
add_submenu_page(
|
||||
'settings.php',
|
||||
__('Domain Health', 'wp-domain-mapping'),
|
||||
__('Domain Health', 'wp-domain-mapping'),
|
||||
'manage_network',
|
||||
'domain-mapping-health',
|
||||
'dm_render_health_page'
|
||||
);
|
||||
|
||||
add_submenu_page(
|
||||
'settings.php',
|
||||
__('Import/Export Domains', 'wp-domain-mapping'),
|
||||
__('Import/Export Domains', 'wp-domain-mapping'),
|
||||
'manage_network',
|
||||
'domain-mapping-import-export',
|
||||
'dm_render_import_export_page'
|
||||
);
|
||||
}
|
||||
|
||||
// Site ID Display Functions
|
||||
public function add_site_id_to_toolbar($admin_bar) {
|
||||
// Only show to admins or network admins
|
||||
if (!current_user_can('manage_options') && !is_super_admin()) {
|
||||
return;
|
||||
}
|
||||
|
||||
$blog_id = get_current_blog_id();
|
||||
|
||||
$admin_bar->add_menu(array(
|
||||
'id' => 'wp-site-ID',
|
||||
'title' => sprintf(__('Site ID: %d', 'wp-domain-mapping'), $blog_id),
|
||||
'href' => is_super_admin() ? esc_url(network_admin_url('site-info.php?id=' . $blog_id)) : '#',
|
||||
'meta' => array(
|
||||
'title' => is_super_admin() ? __('Edit this site', 'wp-domain-mapping') : __('Current site ID', 'wp-domain-mapping'),
|
||||
'class' => 'dm-site-id-menu'
|
||||
),
|
||||
));
|
||||
}
|
||||
|
||||
public function add_site_id_column($columns) {
|
||||
// Add site ID column after first column
|
||||
$columns = array_slice($columns, 0, 1, true) +
|
||||
array('dm_site_id' => __('Site ID', 'wp-domain-mapping')) +
|
||||
array_slice($columns, 1, count($columns) - 1, true);
|
||||
|
||||
return $columns;
|
||||
}
|
||||
|
||||
public function display_site_id_column($column, $blog_id) {
|
||||
if ('dm_site_id' == $column) {
|
||||
echo '<span class="dm-site-id-badge">' . esc_html($blog_id) . '</span>';
|
||||
}
|
||||
}
|
||||
|
||||
public function site_id_column_style() {
|
||||
if ('sites-network' == get_current_screen()->id) {
|
||||
?>
|
||||
<style type="text/css">
|
||||
th#dm_site_id { width: 3.5em; }
|
||||
.dm-site-id-badge {
|
||||
display: inline-block;
|
||||
background: #2271b1;
|
||||
color: #fff;
|
||||
font-weight: 500;
|
||||
padding: 0 5px;
|
||||
border-radius: 3px;
|
||||
font-size: 12px;
|
||||
}
|
||||
</style>
|
||||
<?php
|
||||
}
|
||||
}
|
||||
|
||||
// Health Check Functions
|
||||
public function schedule_health_check() {
|
||||
if (!wp_next_scheduled('dm_domain_health_check')) {
|
||||
wp_schedule_event(time(), 'daily', 'dm_domain_health_check');
|
||||
}
|
||||
}
|
||||
|
||||
public function unschedule_health_check() {
|
||||
$timestamp = wp_next_scheduled('dm_domain_health_check');
|
||||
if ($timestamp) {
|
||||
wp_unschedule_event($timestamp, 'dm_domain_health_check');
|
||||
}
|
||||
}
|
||||
|
||||
public function handle_health_manual_check() {
|
||||
if (isset($_POST['dm_manual_health_check']) && $_POST['dm_manual_health_check']) {
|
||||
// Verify nonce
|
||||
if (!isset($_POST['dm_manual_health_check_nonce']) || !wp_verify_nonce($_POST['dm_manual_health_check_nonce'], 'dm_manual_health_check')) {
|
||||
wp_die(__('Security check failed.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Check permissions
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to perform this action.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Run the health check
|
||||
$this->run_health_check_for_all_domains();
|
||||
|
||||
// Redirect back to health page
|
||||
wp_redirect(add_query_arg(array('page' => 'domain-mapping-health', 'checked' => 1), network_admin_url('settings.php')));
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
public function handle_health_settings_save() {
|
||||
if (isset($_POST['dm_health_settings']) && $_POST['dm_health_settings']) {
|
||||
// Verify nonce
|
||||
if (!isset($_POST['dm_health_settings_nonce']) || !wp_verify_nonce($_POST['dm_health_settings_nonce'], 'dm_health_settings')) {
|
||||
wp_die(__('Security check failed.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Check permissions
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to perform this action.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Save settings
|
||||
$health_check_enabled = isset($_POST['health_check_enabled']) ? (bool) $_POST['health_check_enabled'] : false;
|
||||
$health_notifications_enabled = isset($_POST['health_notifications_enabled']) ? (bool) $_POST['health_notifications_enabled'] : false;
|
||||
$notification_email = isset($_POST['notification_email']) ? sanitize_email($_POST['notification_email']) : '';
|
||||
$ssl_expiry_threshold = isset($_POST['ssl_expiry_threshold']) ? intval($_POST['ssl_expiry_threshold']) : 14;
|
||||
|
||||
update_site_option('dm_health_check_enabled', $health_check_enabled);
|
||||
update_site_option('dm_health_notifications_enabled', $health_notifications_enabled);
|
||||
update_site_option('dm_notification_email', $notification_email);
|
||||
update_site_option('dm_ssl_expiry_threshold', $ssl_expiry_threshold);
|
||||
|
||||
// If auto check is enabled, ensure cron is set
|
||||
if ($health_check_enabled) {
|
||||
$this->schedule_health_check();
|
||||
} else {
|
||||
$this->unschedule_health_check();
|
||||
}
|
||||
|
||||
// Redirect back to health page
|
||||
wp_redirect(add_query_arg(array('page' => 'domain-mapping-health', 'settings-updated' => 1), network_admin_url('settings.php')));
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
public function ajax_check_domain_health() {
|
||||
// Check permissions
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_send_json_error(__('You do not have sufficient permissions to perform this action.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Verify nonce
|
||||
if (!isset($_POST['nonce']) || !wp_verify_nonce($_POST['nonce'], 'dm_check_domain_health')) {
|
||||
wp_send_json_error(__('Security check failed.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Get domain
|
||||
$domain = isset($_POST['domain']) ? sanitize_text_field($_POST['domain']) : '';
|
||||
if (empty($domain)) {
|
||||
wp_send_json_error(__('No domain specified.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Run health check
|
||||
$result = $this->check_domain_health($domain);
|
||||
|
||||
// Save result
|
||||
$this->db->save_health_result($domain, $result);
|
||||
|
||||
// Return result
|
||||
wp_send_json_success($result);
|
||||
}
|
||||
|
||||
public function scheduled_health_check() {
|
||||
// Check if health checks are enabled
|
||||
if (!get_site_option('dm_health_check_enabled', true)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->run_health_check_for_all_domains();
|
||||
}
|
||||
|
||||
private function run_health_check_for_all_domains() {
|
||||
global $wpdb;
|
||||
$table = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_DOMAINS;
|
||||
|
||||
// Get all domains
|
||||
$domains = $wpdb->get_col("SELECT domain FROM {$table}");
|
||||
|
||||
// Initialize issues array
|
||||
$issues = array();
|
||||
|
||||
// Check each domain
|
||||
foreach ($domains as $domain) {
|
||||
$result = $this->check_domain_health($domain);
|
||||
$this->db->save_health_result($domain, $result);
|
||||
|
||||
// Check for issues
|
||||
if ($this->has_health_issues($result)) {
|
||||
$issues[$domain] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
// Send notifications if enabled and issues exist
|
||||
if (!empty($issues) && get_site_option('dm_health_notifications_enabled', true)) {
|
||||
$this->send_health_notification($issues);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private function check_domain_health($domain) {
|
||||
$result = array(
|
||||
'domain' => $domain,
|
||||
'last_check' => current_time('mysql'),
|
||||
'dns_status' => 'error',
|
||||
'dns_message' => __('DNS check not performed', 'wp-domain-mapping'),
|
||||
'resolved_ip' => '',
|
||||
'ssl_valid' => false,
|
||||
'ssl_expiry' => '',
|
||||
'accessible' => false,
|
||||
'response_code' => 0
|
||||
);
|
||||
|
||||
// Get server IP or CNAME
|
||||
$server_ip = get_site_option('dm_ipaddress', '');
|
||||
$server_cname = get_site_option('dm_cname', '');
|
||||
|
||||
// Check DNS setting
|
||||
$domain_ip = gethostbyname($domain);
|
||||
$result['resolved_ip'] = $domain_ip;
|
||||
|
||||
if ($domain_ip && $domain_ip !== $domain) {
|
||||
if ($server_ip && strpos($server_ip, $domain_ip) !== false) {
|
||||
$result['dns_status'] = 'success';
|
||||
$result['dns_message'] = __('Domain A record is correctly pointing to server IP.', 'wp-domain-mapping');
|
||||
} elseif ($server_cname && function_exists('dns_get_record')) {
|
||||
$dns_records = @dns_get_record($domain, DNS_CNAME);
|
||||
$has_valid_cname = false;
|
||||
|
||||
if ($dns_records) {
|
||||
foreach ($dns_records as $record) {
|
||||
if (isset($record['target']) &&
|
||||
(
|
||||
$record['target'] === $server_cname ||
|
||||
strpos($record['target'], $server_cname) !== false
|
||||
)) {
|
||||
$has_valid_cname = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ($has_valid_cname) {
|
||||
$result['dns_status'] = 'success';
|
||||
$result['dns_message'] = __('Domain CNAME record is correctly configured.', 'wp-domain-mapping');
|
||||
} else {
|
||||
$result['dns_message'] = __('Domain is not pointing to the correct server.', 'wp-domain-mapping');
|
||||
}
|
||||
} else {
|
||||
$result['dns_message'] = __('Cannot verify DNS configuration.', 'wp-domain-mapping');
|
||||
}
|
||||
} else {
|
||||
$result['dns_message'] = __('Domain does not resolve to an IP address.', 'wp-domain-mapping');
|
||||
}
|
||||
|
||||
// Check site accessibility and SSL
|
||||
$response = $this->test_domain_connection($domain);
|
||||
|
||||
if ($response) {
|
||||
$result['accessible'] = $response['accessible'];
|
||||
$result['response_code'] = $response['response_code'];
|
||||
$result['ssl_valid'] = $response['ssl_valid'];
|
||||
$result['ssl_expiry'] = $response['ssl_expiry'];
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
private function test_domain_connection($domain) {
|
||||
if (!function_exists('curl_init')) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$result = array(
|
||||
'accessible' => false,
|
||||
'response_code' => 0,
|
||||
'ssl_valid' => false,
|
||||
'ssl_expiry' => ''
|
||||
);
|
||||
|
||||
// Test HTTPS connection
|
||||
$ch = curl_init('https://' . $domain);
|
||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($ch, CURLOPT_HEADER, true);
|
||||
curl_setopt($ch, CURLOPT_NOBODY, true);
|
||||
curl_setopt($ch, CURLOPT_TIMEOUT, 10);
|
||||
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true);
|
||||
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 2);
|
||||
curl_setopt($ch, CURLOPT_CERTINFO, true);
|
||||
|
||||
$response = curl_exec($ch);
|
||||
$response_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
$result['response_code'] = $response_code;
|
||||
|
||||
if ($response !== false && $response_code > 0) {
|
||||
$result['accessible'] = ($response_code >= 200 && $response_code < 400);
|
||||
$result['ssl_valid'] = ($response !== false);
|
||||
|
||||
// Get SSL certificate info
|
||||
$cert_info = curl_getinfo($ch, CURLINFO_CERTINFO);
|
||||
if (!empty($cert_info) && isset($cert_info[0]['Expire date'])) {
|
||||
$result['ssl_expiry'] = $cert_info[0]['Expire date'];
|
||||
}
|
||||
}
|
||||
|
||||
curl_close($ch);
|
||||
|
||||
// If HTTPS failed, try HTTP
|
||||
if (!$result['accessible']) {
|
||||
$ch = curl_init('http://' . $domain);
|
||||
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||
curl_setopt($ch, CURLOPT_HEADER, true);
|
||||
curl_setopt($ch, CURLOPT_NOBODY, true);
|
||||
curl_setopt($ch, CURLOPT_TIMEOUT, 10);
|
||||
|
||||
$response = curl_exec($ch);
|
||||
$response_code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
|
||||
if ($response !== false && $response_code > 0) {
|
||||
$result['accessible'] = ($response_code >= 200 && $response_code < 400);
|
||||
$result['response_code'] = $response_code;
|
||||
}
|
||||
|
||||
curl_close($ch);
|
||||
}
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
private function has_health_issues($result) {
|
||||
// Check DNS issues
|
||||
if ($result['dns_status'] !== 'success') {
|
||||
return true;
|
||||
}
|
||||
|
||||
// Check SSL issues
|
||||
if (!$result['ssl_valid']) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// Check SSL expiry
|
||||
if (!empty($result['ssl_expiry'])) {
|
||||
$expiry_date = strtotime($result['ssl_expiry']);
|
||||
$threshold = get_site_option('dm_ssl_expiry_threshold', 14);
|
||||
$threshold_date = strtotime('+' . $threshold . ' days');
|
||||
|
||||
if ($expiry_date <= $threshold_date) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
// Check accessibility issues
|
||||
if (!$result['accessible']) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
private function send_health_notification($issues) {
|
||||
$notification_email = get_site_option('dm_notification_email', get_option('admin_email'));
|
||||
|
||||
if (empty($notification_email)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$site_name = get_bloginfo('name');
|
||||
$subject = sprintf(__('[%s] Domain Mapping Health Alert', 'wp-domain-mapping'), $site_name);
|
||||
|
||||
// Build email content
|
||||
$message = sprintf(__('Domain health issues were detected on %s.', 'wp-domain-mapping'), $site_name) . "\n\n";
|
||||
$message .= __('The following domains have issues:', 'wp-domain-mapping') . "\n\n";
|
||||
|
||||
foreach ($issues as $domain => $result) {
|
||||
$message .= sprintf(__('Domain: %s', 'wp-domain-mapping'), $domain) . "\n";
|
||||
|
||||
// Add DNS status
|
||||
if ($result['dns_status'] !== 'success') {
|
||||
$message .= " - " . sprintf(__('DNS Issue: %s', 'wp-domain-mapping'), $result['dns_message']) . "\n";
|
||||
}
|
||||
|
||||
// Add SSL status
|
||||
if (!$result['ssl_valid']) {
|
||||
$message .= " - " . __('SSL Certificate is invalid or missing.', 'wp-domain-mapping') . "\n";
|
||||
} elseif (!empty($result['ssl_expiry'])) {
|
||||
$expiry_date = strtotime($result['ssl_expiry']);
|
||||
$threshold = get_site_option('dm_ssl_expiry_threshold', 14);
|
||||
$threshold_date = strtotime('+' . $threshold . ' days');
|
||||
|
||||
if ($expiry_date <= $threshold_date) {
|
||||
$message .= " - " . sprintf(
|
||||
__('SSL Certificate expires on %s (within %d days).', 'wp-domain-mapping'),
|
||||
date('Y-m-d', $expiry_date),
|
||||
$threshold
|
||||
) . "\n";
|
||||
}
|
||||
}
|
||||
|
||||
// Add accessibility status
|
||||
if (!$result['accessible']) {
|
||||
$message .= " - " . __('Site is not accessible.', 'wp-domain-mapping') . "\n";
|
||||
if ($result['response_code'] > 0) {
|
||||
$message .= " " . sprintf(__('HTTP Response Code: %d', 'wp-domain-mapping'), $result['response_code']) . "\n";
|
||||
}
|
||||
}
|
||||
|
||||
$message .= "\n";
|
||||
}
|
||||
|
||||
// Add resolution link
|
||||
$message .= sprintf(
|
||||
__('To view and manage these issues, please visit: %s', 'wp-domain-mapping'),
|
||||
network_admin_url('settings.php?page=domain-mapping-health')
|
||||
) . "\n";
|
||||
|
||||
// Send email
|
||||
wp_mail($notification_email, $subject, $message);
|
||||
}
|
||||
|
||||
// Import/Export Functions
|
||||
public function handle_export() {
|
||||
if (!isset($_POST['domain_mapping_export']) || !$_POST['domain_mapping_export']) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Check permissions
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_die(__('You do not have sufficient permissions to export data.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Verify nonce
|
||||
if (!isset($_POST['domain_mapping_export_nonce']) || !wp_verify_nonce($_POST['domain_mapping_export_nonce'], 'domain_mapping_export')) {
|
||||
wp_die(__('Invalid security token. Please try again.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Get options
|
||||
$include_header = isset($_POST['include_header']) ? (bool) $_POST['include_header'] : false;
|
||||
$blog_id_filter = isset($_POST['blog_id_filter']) && !empty($_POST['blog_id_filter']) ? intval($_POST['blog_id_filter']) : 0;
|
||||
|
||||
// Get domain mapping data
|
||||
global $wpdb;
|
||||
$table = $wpdb->base_prefix . WP_DOMAIN_MAPPING_TABLE_DOMAINS;
|
||||
$sql = "SELECT blog_id, domain, active FROM {$table}";
|
||||
|
||||
if ($blog_id_filter > 0) {
|
||||
$sql .= $wpdb->prepare(" WHERE blog_id = %d", $blog_id_filter);
|
||||
}
|
||||
|
||||
$domains = $wpdb->get_results($sql, ARRAY_A);
|
||||
|
||||
if (empty($domains)) {
|
||||
// No data
|
||||
wp_redirect(add_query_arg(array('page' => 'domain-mapping-import-export', 'export' => 'empty'), network_admin_url('settings.php')));
|
||||
exit;
|
||||
}
|
||||
|
||||
// Set up CSV output
|
||||
$filename = 'domain-mappings-' . date('Y-m-d') . '.csv';
|
||||
|
||||
header('Content-Type: text/csv; charset=utf-8');
|
||||
header('Content-Disposition: attachment; filename=' . $filename);
|
||||
|
||||
$output = fopen('php://output', 'w');
|
||||
|
||||
// Add header row
|
||||
if ($include_header) {
|
||||
fputcsv($output, array('blog_id', 'domain', 'active'));
|
||||
}
|
||||
|
||||
// Add data rows
|
||||
foreach ($domains as $domain) {
|
||||
fputcsv($output, $domain);
|
||||
}
|
||||
|
||||
fclose($output);
|
||||
exit;
|
||||
}
|
||||
|
||||
public function ajax_import_csv() {
|
||||
// Check permissions
|
||||
if (!current_user_can('manage_network')) {
|
||||
wp_send_json_error(__('You do not have sufficient permissions to import data.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Verify nonce
|
||||
if (!isset($_POST['domain_mapping_import_nonce']) || !wp_verify_nonce($_POST['domain_mapping_import_nonce'], 'domain_mapping_import')) {
|
||||
wp_send_json_error(__('Invalid security token. Please try again.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Check file
|
||||
if (!isset($_FILES['csv_file']) || $_FILES['csv_file']['error'] != UPLOAD_ERR_OK) {
|
||||
wp_send_json_error(__('No file uploaded or upload error.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Get options
|
||||
$has_header = isset($_POST['has_header']) ? (bool) $_POST['has_header'] : false;
|
||||
$update_existing = isset($_POST['update_existing']) ? (bool) $_POST['update_existing'] : false;
|
||||
$validate_sites = isset($_POST['validate_sites']) ? (bool) $_POST['validate_sites'] : true;
|
||||
|
||||
// Open file
|
||||
$file = fopen($_FILES['csv_file']['tmp_name'], 'r');
|
||||
if (!$file) {
|
||||
wp_send_json_error(__('Could not open the uploaded file.', 'wp-domain-mapping'));
|
||||
}
|
||||
|
||||
// Initialize counters and log
|
||||
$imported = 0;
|
||||
$skipped = 0;
|
||||
$errors = 0;
|
||||
$log = array();
|
||||
|
||||
// Skip header row
|
||||
if ($has_header) {
|
||||
fgetcsv($file);
|
||||
}
|
||||
|
||||
// Process each row
|
||||
$row_num = $has_header ? 2 : 1; // Account for header row
|
||||
|
||||
while (($data = fgetcsv($file)) !== false) {
|
||||
// Check data format
|
||||
if (count($data) < 3) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Invalid format. Expected at least 3 columns.', 'wp-domain-mapping'), $row_num)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// Parse data
|
||||
$blog_id = intval($data[0]);
|
||||
$domain = $this->core->clean_domain(trim($data[1]));
|
||||
$active = intval($data[2]);
|
||||
|
||||
// Validate blog_id
|
||||
if ($blog_id <= 0) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Invalid blog ID: %d', 'wp-domain-mapping'), $row_num, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// Validate site exists
|
||||
if ($validate_sites && !get_blog_details($blog_id)) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Site ID %d does not exist.', 'wp-domain-mapping'), $row_num, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// Validate domain format
|
||||
if (!$this->core->validate_domain($domain)) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Invalid domain format: %s', 'wp-domain-mapping'), $row_num, $domain)
|
||||
);
|
||||
$errors++;
|
||||
$row_num++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check if domain already exists
|
||||
$existing = $this->db->get_domain_by_name($domain);
|
||||
|
||||
if ($existing) {
|
||||
if ($existing->blog_id != $blog_id) {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Domain %s is already mapped to blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $existing->blog_id)
|
||||
);
|
||||
$errors++;
|
||||
} elseif (!$update_existing) {
|
||||
$log[] = array(
|
||||
'status' => 'warning',
|
||||
'message' => sprintf(__('Row %d: Domain %s already exists for blog ID %d. Skipped.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$skipped++;
|
||||
} else {
|
||||
// Update existing domain
|
||||
$success = $this->db->update_domain($domain, $blog_id, $active);
|
||||
|
||||
if ($success) {
|
||||
$log[] = array(
|
||||
'status' => 'success',
|
||||
'message' => sprintf(__('Row %d: Updated domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$imported++;
|
||||
} else {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Failed to update domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// Add new domain
|
||||
$success = $this->db->add_domain($blog_id, $domain, $active);
|
||||
|
||||
if ($success) {
|
||||
$log[] = array(
|
||||
'status' => 'success',
|
||||
'message' => sprintf(__('Row %d: Added domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$imported++;
|
||||
} else {
|
||||
$log[] = array(
|
||||
'status' => 'error',
|
||||
'message' => sprintf(__('Row %d: Failed to add domain %s for blog ID %d.', 'wp-domain-mapping'),
|
||||
$row_num, $domain, $blog_id)
|
||||
);
|
||||
$errors++;
|
||||
}
|
||||
}
|
||||
|
||||
$row_num++;
|
||||
}
|
||||
|
||||
fclose($file);
|
||||
|
||||
// Build response
|
||||
$message = sprintf(
|
||||
__('Import completed: %d imported, %d skipped, %d errors.', 'wp-domain-mapping'),
|
||||
$imported, $skipped, $errors
|
||||
);
|
||||
|
||||
wp_send_json_success(array(
|
||||
'message' => $message,
|
||||
'imported' => $imported,
|
||||
'skipped' => $skipped,
|
||||
'errors' => $errors,
|
||||
'details' => $log
|
||||
));
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue