mirror of
https://github.com/WordPress/WordPress.git
synced 2025-01-09 18:08:09 +01:00
Privacy: add helper function for anonymizing data in a standardized way.
Props jesperher, allendav, iandunn, birgire, azaozz. Fixes #43545. Built from https://develop.svn.wordpress.org/trunk@42971 git-svn-id: http://core.svn.wordpress.org/trunk@42800 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit is contained in:
parent
9bbb2618e4
commit
5493d8b253
@ -234,7 +234,6 @@ class WP_Community_Events {
|
||||
*/
|
||||
public static function get_unsafe_client_ip() {
|
||||
$client_ip = false;
|
||||
$ip_prefix = '';
|
||||
|
||||
// In order of preference, with the best ones for this purpose first.
|
||||
$address_headers = array(
|
||||
@ -265,57 +264,13 @@ class WP_Community_Events {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Detect what kind of IP address this is.
|
||||
$is_ipv6 = substr_count( $client_ip, ':' ) > 1;
|
||||
$is_ipv4 = ( 3 === substr_count( $client_ip, '.' ) );
|
||||
$anon_ip = wp_privacy_anonymize_ip( $client_ip, true );
|
||||
|
||||
if ( $is_ipv6 && $is_ipv4 ) {
|
||||
// IPv6 compatibility mode, temporarily strip the IPv6 part, and treat it like IPv4.
|
||||
$ip_prefix = '::ffff:';
|
||||
$client_ip = preg_replace( '/^\[?[0-9a-f:]*:/i', '', $client_ip );
|
||||
$client_ip = str_replace( ']', '', $client_ip );
|
||||
$is_ipv6 = false;
|
||||
}
|
||||
|
||||
if ( $is_ipv6 ) {
|
||||
// IPv6 addresses will always be enclosed in [] if there's a port.
|
||||
$left_bracket = strpos( $client_ip, '[' );
|
||||
$right_bracket = strpos( $client_ip, ']' );
|
||||
$percent = strpos( $client_ip, '%' );
|
||||
$netmask = 'ffff:ffff:ffff:ffff:0000:0000:0000:0000';
|
||||
|
||||
// Strip the port (and [] from IPv6 addresses), if they exist.
|
||||
if ( false !== $left_bracket && false !== $right_bracket ) {
|
||||
$client_ip = substr( $client_ip, $left_bracket + 1, $right_bracket - $left_bracket - 1 );
|
||||
} elseif ( false !== $left_bracket || false !== $right_bracket ) {
|
||||
// The IP has one bracket, but not both, so it's malformed.
|
||||
return false;
|
||||
}
|
||||
|
||||
// Strip the reachability scope.
|
||||
if ( false !== $percent ) {
|
||||
$client_ip = substr( $client_ip, 0, $percent );
|
||||
}
|
||||
|
||||
// No invalid characters should be left.
|
||||
if ( preg_match( '/[^0-9a-f:]/i', $client_ip ) ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Partially anonymize the IP by reducing it to the corresponding network ID.
|
||||
if ( function_exists( 'inet_pton' ) && function_exists( 'inet_ntop' ) ) {
|
||||
$client_ip = inet_ntop( inet_pton( $client_ip ) & inet_pton( $netmask ) );
|
||||
}
|
||||
} elseif ( $is_ipv4 ) {
|
||||
// Strip any port and partially anonymize the IP.
|
||||
$last_octet_position = strrpos( $client_ip, '.' );
|
||||
$client_ip = substr( $client_ip, 0, $last_octet_position ) . '.0';
|
||||
} else {
|
||||
if ( '0.0.0.0' === $anon_ip || '::' === $anon_ip ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Restore the IPv6 prefix to compatibility mode addresses.
|
||||
return $ip_prefix . $client_ip;
|
||||
return $anon_ip;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -6127,3 +6127,121 @@ All at ###SITENAME###
|
||||
), $email_change_email['message'], $email_change_email['headers']
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return an anonymized IPv4 or IPv6 address.
|
||||
*
|
||||
* @since 5.0.0 Abstracted from `WP_Community_Events::get_unsafe_client_ip()`.
|
||||
*
|
||||
* @param string $ip_addr The IPv4 or IPv6 address to be anonymized.
|
||||
* @param bool $ipv6_fallback Optional. Whether to return the original IPv6 address if the needed functions
|
||||
* to anonymize it are not present. Default false, return `::` (unspecified address).
|
||||
* @return string The anonymized IP address.
|
||||
*/
|
||||
function wp_privacy_anonymize_ip( $ip_addr, $ipv6_fallback = false ) {
|
||||
// Detect what kind of IP address this is.
|
||||
$ip_prefix = '';
|
||||
$is_ipv6 = substr_count( $ip_addr, ':' ) > 1;
|
||||
$is_ipv4 = ( 3 === substr_count( $ip_addr, '.' ) );
|
||||
|
||||
if ( $is_ipv6 && $is_ipv4 ) {
|
||||
// IPv6 compatibility mode, temporarily strip the IPv6 part, and treat it like IPv4.
|
||||
$ip_prefix = '::ffff:';
|
||||
$ip_addr = preg_replace( '/^\[?[0-9a-f:]*:/i', '', $ip_addr );
|
||||
$ip_addr = str_replace( ']', '', $ip_addr );
|
||||
$is_ipv6 = false;
|
||||
}
|
||||
|
||||
if ( $is_ipv6 ) {
|
||||
// IPv6 addresses will always be enclosed in [] if there's a port.
|
||||
$left_bracket = strpos( $ip_addr, '[' );
|
||||
$right_bracket = strpos( $ip_addr, ']' );
|
||||
$percent = strpos( $ip_addr, '%' );
|
||||
$netmask = 'ffff:ffff:ffff:ffff:0000:0000:0000:0000';
|
||||
|
||||
// Strip the port (and [] from IPv6 addresses), if they exist.
|
||||
if ( false !== $left_bracket && false !== $right_bracket ) {
|
||||
$ip_addr = substr( $ip_addr, $left_bracket + 1, $right_bracket - $left_bracket - 1 );
|
||||
} elseif ( false !== $left_bracket || false !== $right_bracket ) {
|
||||
// The IP has one bracket, but not both, so it's malformed.
|
||||
return '::';
|
||||
}
|
||||
|
||||
// Strip the reachability scope.
|
||||
if ( false !== $percent ) {
|
||||
$ip_addr = substr( $ip_addr, 0, $percent );
|
||||
}
|
||||
|
||||
// No invalid characters should be left.
|
||||
if ( preg_match( '/[^0-9a-f:]/i', $ip_addr ) ) {
|
||||
return '::';
|
||||
}
|
||||
|
||||
// Partially anonymize the IP by reducing it to the corresponding network ID.
|
||||
if ( function_exists( 'inet_pton' ) && function_exists( 'inet_ntop' ) ) {
|
||||
$ip_addr = inet_ntop( inet_pton( $ip_addr ) & inet_pton( $netmask ) );
|
||||
if ( false === $ip_addr) {
|
||||
return '::';
|
||||
}
|
||||
} elseif ( ! $ipv6_fallback ) {
|
||||
return '::';
|
||||
}
|
||||
} elseif ( $is_ipv4 ) {
|
||||
// Strip any port and partially anonymize the IP.
|
||||
$last_octet_position = strrpos( $ip_addr, '.' );
|
||||
$ip_addr = substr( $ip_addr, 0, $last_octet_position ) . '.0';
|
||||
} else {
|
||||
return '0.0.0.0';
|
||||
}
|
||||
|
||||
// Restore the IPv6 prefix to compatibility mode addresses.
|
||||
return $ip_prefix . $ip_addr;
|
||||
}
|
||||
|
||||
/**
|
||||
* Return uniform "anonymous" data by type.
|
||||
*
|
||||
* @since 5.0.0
|
||||
*
|
||||
* @param string $type The type of data to be anonymized.
|
||||
* @param string $data Optional The data to be anonymized.
|
||||
* @return string The anonymous data for the requested type.
|
||||
*/
|
||||
function wp_privacy_anonymize_data( $type, $data = '' ) {
|
||||
|
||||
switch ( $type ) {
|
||||
case 'email':
|
||||
$anonymous = 'deleted@site.invalid';
|
||||
break;
|
||||
case 'url':
|
||||
$anonymous = 'https://site.invalid';
|
||||
break;
|
||||
case 'ip':
|
||||
$anonymous = wp_privacy_anonymize_ip( $data );
|
||||
break;
|
||||
case 'date':
|
||||
$anonymous = '0000-00-00 00:00:00';
|
||||
break;
|
||||
case 'text':
|
||||
/* translators: deleted text */
|
||||
$anonymous = __( '[deleted]' );
|
||||
break;
|
||||
case 'longtext':
|
||||
/* translators: deleted long text */
|
||||
$anonymous = __( 'This content was deleted by the author.' );
|
||||
break;
|
||||
default:
|
||||
$anonymous = '';
|
||||
}
|
||||
|
||||
/**
|
||||
* Filters the anonymous data for each type.
|
||||
*
|
||||
* @since 5.0.0
|
||||
*
|
||||
* @param string $anonymous Anonymized data.
|
||||
* @param string $type Type of the data.
|
||||
* @param string $data Original data.
|
||||
*/
|
||||
return apply_filters( 'wp_privacy_anonymize_data', $anonymous, $type, $data );
|
||||
}
|
||||
|
@ -4,7 +4,7 @@
|
||||
*
|
||||
* @global string $wp_version
|
||||
*/
|
||||
$wp_version = '5.0-alpha-42970';
|
||||
$wp_version = '5.0-alpha-42971';
|
||||
|
||||
/**
|
||||
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.
|
||||
|
Loading…
Reference in New Issue
Block a user