2010-01-10 19:10:51 +01:00
< ? php
/**
2010-01-28 17:09:52 +01:00
* These functions are needed to load Multisite .
2010-01-10 19:10:51 +01:00
*
2010-01-28 17:09:52 +01:00
* @ since 3.0 . 0
2010-01-10 19:10:51 +01:00
*
* @ package WordPress
2010-01-28 17:09:52 +01:00
* @ subpackage Multisite
2010-01-10 19:10:51 +01:00
*/
2010-01-28 17:09:52 +01:00
/**
2010-02-01 21:26:08 +01:00
* Whether a subdomain configuration is enabled .
2010-01-28 17:09:52 +01:00
*
2010-03-26 20:13:36 +01:00
* @ since 3.0 . 0
2010-01-28 17:09:52 +01:00
*
* @ return bool True if subdomain configuration is enabled , false otherwise .
*/
function is_subdomain_install () {
2010-05-04 22:48:28 +02:00
if ( defined ( 'SUBDOMAIN_INSTALL' ) )
return SUBDOMAIN_INSTALL ;
2015-05-26 23:51:31 +02:00
return ( defined ( 'VHOST' ) && VHOST == 'yes' );
2010-01-28 17:09:52 +01:00
}
2010-01-10 19:10:51 +01:00
2010-11-24 01:19:38 +01:00
/**
* Returns array of network plugin files to be included in global scope .
*
* The default directory is wp - content / plugins . To change the default directory
2014-11-24 06:53:22 +01:00
* manually , define `WP_PLUGIN_DIR` and `WP_PLUGIN_URL` in `wp-config.php` .
2010-11-24 01:19:38 +01:00
*
* @ access private
* @ since 3.1 . 0
2014-11-24 06:53:22 +01:00
*
* @ return array Files to include .
2010-11-24 01:19:38 +01:00
*/
function wp_get_active_network_plugins () {
$active_plugins = ( array ) get_site_option ( 'active_sitewide_plugins' , array () );
if ( empty ( $active_plugins ) )
return array ();
2010-12-08 02:03:12 +01:00
$plugins = array ();
2010-11-24 01:19:38 +01:00
$active_plugins = array_keys ( $active_plugins );
sort ( $active_plugins );
foreach ( $active_plugins as $plugin ) {
if ( ! validate_file ( $plugin ) // $plugin must validate as file
&& '.php' == substr ( $plugin , - 4 ) // $plugin must end with '.php'
&& file_exists ( WP_PLUGIN_DIR . '/' . $plugin ) // $plugin must exist
)
$plugins [] = WP_PLUGIN_DIR . '/' . $plugin ;
}
return $plugins ;
}
2010-02-01 21:26:08 +01:00
/**
* Checks status of current blog .
*
* Checks if the blog is deleted , inactive , archived , or spammed .
*
* Dies with a default message if the blog does not pass the check .
*
* To change the default message when a blog does not pass the check ,
* use the wp - content / blog - deleted . php , blog - inactive . php and
* blog - suspended . php drop - ins .
*
2013-10-30 15:39:10 +01:00
* @ since 3.0 . 0
*
2015-05-26 23:51:31 +02:00
* @ return true | string Returns true on success , or drop - in file to include .
2010-02-01 21:26:08 +01:00
*/
2010-01-10 19:10:51 +01:00
function ms_site_check () {
2012-10-04 14:40:09 +02:00
$blog = get_blog_details ();
2010-01-20 08:44:47 +01:00
2013-10-08 22:01:09 +02:00
/**
* Filter checking the status of the current blog .
2013-10-25 04:29:52 +02:00
*
2013-10-30 15:39:10 +01:00
* @ since 3.0 . 0
2013-10-25 04:29:52 +02:00
*
2013-10-08 22:01:09 +02:00
* @ param bool null Whether to skip the blog status check . Default null .
*/
$check = apply_filters ( 'ms_site_check' , null );
2010-04-23 18:18:33 +02:00
if ( null !== $check )
2010-04-23 18:50:01 +02:00
return true ;
2010-04-23 18:18:33 +02:00
// Allow super admins to see blocked sites
if ( is_super_admin () )
return true ;
2012-10-04 14:40:09 +02:00
if ( '1' == $blog -> deleted ) {
2010-12-14 09:38:01 +01:00
if ( file_exists ( WP_CONTENT_DIR . '/blog-deleted.php' ) )
2010-02-01 21:26:08 +01:00
return WP_CONTENT_DIR . '/blog-deleted.php' ;
2010-12-14 09:38:01 +01:00
else
2014-05-06 20:46:15 +02:00
wp_die ( __ ( 'This site is no longer available.' ), '' , array ( 'response' => 410 ) );
2010-02-01 21:26:08 +01:00
}
2012-10-04 14:40:09 +02:00
if ( '2' == $blog -> deleted ) {
2010-02-01 21:26:08 +01:00
if ( file_exists ( WP_CONTENT_DIR . '/blog-inactive.php' ) )
return WP_CONTENT_DIR . '/blog-inactive.php' ;
else
2013-11-13 04:23:10 +01:00
wp_die ( sprintf ( __ ( 'This site has not been activated yet. If you are having problems activating your site, please contact <a href="mailto:%1$s">%1$s</a>.' ), str_replace ( '@' , ' AT ' , get_site_option ( 'admin_email' , 'support@' . get_current_site () -> domain ) ) ) );
2010-01-20 08:44:47 +01:00
}
2012-10-04 14:40:09 +02:00
if ( $blog -> archived == '1' || $blog -> spam == '1' ) {
2010-12-14 09:38:01 +01:00
if ( file_exists ( WP_CONTENT_DIR . '/blog-suspended.php' ) )
2010-02-01 21:26:08 +01:00
return WP_CONTENT_DIR . '/blog-suspended.php' ;
2010-12-14 09:38:01 +01:00
else
wp_die ( __ ( 'This site has been archived or suspended.' ), '' , array ( 'response' => 410 ) );
2010-01-20 08:44:47 +01:00
}
2010-01-25 23:09:43 +01:00
2010-01-20 08:44:47 +01:00
return true ;
2010-01-10 19:10:51 +01:00
}
2014-02-14 00:07:14 +01:00
/**
* Retrieve a network object by its domain and path .
*
* @ since 3.9 . 0
*
2015-05-26 23:51:31 +02:00
* @ global wpdb $wpdb
*
2014-04-07 22:59:15 +02:00
* @ param string $domain Domain to check .
* @ param string $path Path to check .
* @ param int | null $segments Path segments to use . Defaults to null , or the full path .
2015-05-26 23:51:31 +02:00
* @ return object | false Network object if successful . False when no network is found .
2014-02-14 00:07:14 +01:00
*/
2014-03-02 23:25:14 +01:00
function get_network_by_path ( $domain , $path , $segments = null ) {
2014-02-14 00:07:14 +01:00
global $wpdb ;
2014-05-06 23:32:16 +02:00
$domains = array ( $domain );
2014-02-14 00:07:14 +01:00
$pieces = explode ( '.' , $domain );
2014-04-07 22:59:15 +02:00
/*
* It 's possible one domain to search is ' com ' , but it might as well
* be 'localhost' or some other locally mapped domain .
*/
2014-02-14 00:07:14 +01:00
while ( array_shift ( $pieces ) ) {
if ( $pieces ) {
$domains [] = implode ( '.' , $pieces );
}
}
2014-03-02 23:25:14 +01:00
/*
* If we ' ve gotten to this function during normal execution , there is
* more than one network installed . At this point , who knows how many
* we have . Attempt to optimize for the situation where networks are
* only domains , thus meaning paths never need to be considered .
*
* This is a very basic optimization ; anything further could have drawbacks
* depending on the setup , so this is best done per - install .
*/
$using_paths = true ;
if ( wp_using_ext_object_cache () ) {
$using_paths = wp_cache_get ( 'networks_have_paths' , 'site-options' );
if ( false === $using_paths ) {
$using_paths = ( bool ) $wpdb -> get_var ( " SELECT id FROM $wpdb->site WHERE path <> '/' LIMIT 1 " );
wp_cache_add ( 'networks_have_paths' , ( int ) $using_paths , 'site-options' );
}
2014-02-14 00:07:14 +01:00
}
2014-03-02 23:25:14 +01:00
$paths = array ();
if ( $using_paths ) {
$path_segments = array_filter ( explode ( '/' , trim ( $path , " / " ) ) );
/**
* Filter the number of path segments to consider when searching for a site .
*
* @ since 3.9 . 0
*
2014-04-07 22:59:15 +02:00
* @ param int | null $segments The number of path segments to consider . WordPress by default looks at
* one path segment . The function default of null only makes sense when you
* know the requested path should match a network .
* @ param string $domain The requested domain .
* @ param string $path The requested path , in full .
2014-03-02 23:25:14 +01:00
*/
$segments = apply_filters ( 'network_by_path_segments_count' , $segments , $domain , $path );
if ( null !== $segments && count ( $path_segments ) > $segments ) {
$path_segments = array_slice ( $path_segments , 0 , $segments );
}
while ( count ( $path_segments ) ) {
$paths [] = '/' . implode ( '/' , $path_segments ) . '/' ;
array_pop ( $path_segments );
}
$paths [] = '/' ;
}
/**
* Determine a network by its domain and path .
*
* This allows one to short - circuit the default logic , perhaps by
* replacing it with a routine that is more optimal for your setup .
*
* Return null to avoid the short - circuit . Return false if no network
* can be found at the requested domain and path . Otherwise , return
* an object from wp_get_network () .
*
* @ since 3.9 . 0
*
2014-04-07 22:59:15 +02:00
* @ param null | bool | object $network Network value to return by path .
* @ param string $domain The requested domain .
* @ param string $path The requested path , in full .
* @ param int | null $segments The suggested number of paths to consult .
* Default null , meaning the entire path was to be consulted .
* @ param array $paths The paths to search for , based on $path and $segments .
2014-03-02 23:25:14 +01:00
*/
$pre = apply_filters ( 'pre_get_network_by_path' , null , $domain , $path , $segments , $paths );
if ( null !== $pre ) {
return $pre ;
}
// @todo Consider additional optimization routes, perhaps as an opt-in for plugins.
// We already have paths covered. What about how far domains should be drilled down (including www)?
2014-02-14 00:07:14 +01:00
$search_domains = " ' " . implode ( " ', ' " , $wpdb -> _escape ( $domains ) ) . " ' " ;
2014-03-02 23:25:14 +01:00
if ( ! $using_paths ) {
$network = $wpdb -> get_row ( " SELECT id, domain, path FROM $wpdb->site
WHERE domain IN ( $search_domains ) ORDER BY CHAR_LENGTH ( domain ) DESC LIMIT 1 " );
if ( $network ) {
return wp_get_network ( $network );
}
return false ;
} else {
$search_paths = " ' " . implode ( " ', ' " , $wpdb -> _escape ( $paths ) ) . " ' " ;
$networks = $wpdb -> get_results ( " SELECT id, domain, path FROM $wpdb->site
WHERE domain IN ( $search_domains ) AND path IN ( $search_paths )
ORDER BY CHAR_LENGTH ( domain ) DESC , CHAR_LENGTH ( path ) DESC " );
}
2014-02-14 00:07:14 +01:00
/*
* Domains are sorted by length of domain , then by length of path .
* The domain must match for the path to be considered . Otherwise ,
* a network with the path of / will suffice .
*/
$found = false ;
foreach ( $networks as $network ) {
if ( $network -> domain === $domain || " www. $network->domain " === $domain ) {
2014-03-02 23:25:14 +01:00
if ( in_array ( $network -> path , $paths , true ) ) {
2014-02-14 00:07:14 +01:00
$found = true ;
break ;
}
}
if ( $network -> path === '/' ) {
$found = true ;
break ;
}
}
if ( $found ) {
2014-03-02 23:25:14 +01:00
return wp_get_network ( $network );
2014-02-14 00:07:14 +01:00
}
return false ;
}
/**
* Retrieve an object containing information about the requested network .
*
* @ since 3.9 . 0
2015-09-14 01:31:26 +02:00
* @ since 4.4 . 0 Converted to leverage WP_Network
2015-05-26 23:51:31 +02:00
*
2014-04-07 22:59:15 +02:00
* @ param object | int $network The network ' s database row or ID .
2015-09-14 01:31:26 +02:00
* @ return WP_Network | false Object containing network information if found , false if not .
2014-02-14 00:07:14 +01:00
*/
function wp_get_network ( $network ) {
if ( ! is_object ( $network ) ) {
2015-09-14 01:31:26 +02:00
$network = WP_Network :: get_instance ( $network );
} else {
$network = new WP_Network ( $network );
2014-02-14 00:07:14 +01:00
}
return $network ;
}
2014-03-02 23:25:14 +01:00
/**
* Retrieve a site object by its domain and path .
*
* @ since 3.9 . 0
*
2015-05-26 23:51:31 +02:00
* @ global wpdb $wpdb
*
2014-04-07 22:59:15 +02:00
* @ param string $domain Domain to check .
* @ param string $path Path to check .
* @ param int | null $segments Path segments to use . Defaults to null , or the full path .
2015-05-26 23:51:31 +02:00
* @ return object | false Site object if successful . False when no site is found .
2014-03-02 23:25:14 +01:00
*/
function get_site_by_path ( $domain , $path , $segments = null ) {
global $wpdb ;
2014-03-03 19:56:14 +01:00
$path_segments = array_filter ( explode ( '/' , trim ( $path , '/' ) ) );
2014-03-02 23:25:14 +01:00
/**
* Filter the number of path segments to consider when searching for a site .
*
* @ since 3.9 . 0
*
2014-04-07 22:59:15 +02:00
* @ param int | null $segments The number of path segments to consider . WordPress by default looks at
* one path segment following the network path . The function default of
* null only makes sense when you know the requested path should match a site .
* @ param string $domain The requested domain .
* @ param string $path The requested path , in full .
2014-03-02 23:25:14 +01:00
*/
$segments = apply_filters ( 'site_by_path_segments_count' , $segments , $domain , $path );
2014-03-03 19:56:14 +01:00
if ( null !== $segments && count ( $path_segments ) > $segments ) {
2014-03-02 23:25:14 +01:00
$path_segments = array_slice ( $path_segments , 0 , $segments );
2010-02-01 21:26:08 +01:00
}
2014-12-20 23:47:22 +01:00
$paths = array ();
2014-03-02 23:25:14 +01:00
while ( count ( $path_segments ) ) {
$paths [] = '/' . implode ( '/' , $path_segments ) . '/' ;
array_pop ( $path_segments );
}
$paths [] = '/' ;
/**
* Determine a site by its domain and path .
*
* This allows one to short - circuit the default logic , perhaps by
* replacing it with a routine that is more optimal for your setup .
*
* Return null to avoid the short - circuit . Return false if no site
* can be found at the requested domain and path . Otherwise , return
* a site object .
*
* @ since 3.9 . 0
*
2014-04-07 22:59:15 +02:00
* @ param null | bool | object $site Site value to return by path .
* @ param string $domain The requested domain .
* @ param string $path The requested path , in full .
* @ param int | null $segments The suggested number of paths to consult .
* Default null , meaning the entire path was to be consulted .
* @ param array $paths The paths to search for , based on $path and $segments .
2014-03-02 23:25:14 +01:00
*/
$pre = apply_filters ( 'pre_get_site_by_path' , null , $domain , $path , $segments , $paths );
if ( null !== $pre ) {
return $pre ;
}
2014-04-07 22:59:15 +02:00
/*
* @ todo
* get_blog_details (), caching , etc . Consider alternative optimization routes ,
* perhaps as an opt - in for plugins , rather than using the pre_ * filter .
* For example : The segments filter can expand or ignore paths .
* If persistent caching is enabled , we could query the DB for a path <> '/'
* then cache whether we can just always ignore paths .
*/
2014-03-02 23:25:14 +01:00
2014-05-06 07:32:36 +02:00
// Either www or non-www is supported, not both. If a www domain is requested,
// query for both to provide the proper redirect.
$domains = array ( $domain );
if ( 'www.' === substr ( $domain , 0 , 4 ) ) {
$domains [] = substr ( $domain , 4 );
$search_domains = " ' " . implode ( " ', ' " , $wpdb -> _escape ( $domains ) ) . " ' " ;
}
2014-03-02 23:25:14 +01:00
if ( count ( $paths ) > 1 ) {
2014-05-06 07:32:36 +02:00
$search_paths = " ' " . implode ( " ', ' " , $wpdb -> _escape ( $paths ) ) . " ' " ;
}
if ( count ( $domains ) > 1 && count ( $paths ) > 1 ) {
$site = $wpdb -> get_row ( " SELECT * FROM $wpdb->blogs WHERE domain IN ( $search_domains ) AND path IN ( $search_paths ) ORDER BY CHAR_LENGTH(domain) DESC, CHAR_LENGTH(path) DESC LIMIT 1 " );
} elseif ( count ( $domains ) > 1 ) {
$sql = $wpdb -> prepare ( " SELECT * FROM $wpdb->blogs WHERE path = %s " , $paths [ 0 ] );
$sql .= " AND domain IN ( $search_domains ) ORDER BY CHAR_LENGTH(domain) DESC LIMIT 1 " ;
$site = $wpdb -> get_row ( $sql );
} elseif ( count ( $paths ) > 1 ) {
$sql = $wpdb -> prepare ( " SELECT * FROM $wpdb->blogs WHERE domain = %s " , $domains [ 0 ] );
$sql .= " AND path IN ( $search_paths ) ORDER BY CHAR_LENGTH(path) DESC LIMIT 1 " ;
2014-03-06 19:18:15 +01:00
$site = $wpdb -> get_row ( $sql );
2014-03-02 23:25:14 +01:00
} else {
2014-05-06 07:32:36 +02:00
$site = $wpdb -> get_row ( $wpdb -> prepare ( " SELECT * FROM $wpdb->blogs WHERE domain = %s AND path = %s " , $domains [ 0 ], $paths [ 0 ] ) );
2014-03-02 23:25:14 +01:00
}
if ( $site ) {
// @todo get_blog_details()
return $site ;
}
return false ;
2010-01-10 19:10:51 +01:00
}
2010-01-28 17:09:52 +01:00
2010-02-01 21:26:08 +01:00
/**
2010-02-20 14:51:32 +01:00
* Displays a failure message .
2010-02-01 21:26:08 +01:00
*
2010-04-30 17:21:10 +02:00
* Used when a blog ' s tables do not exist . Checks for a missing $wpdb -> site table as well .
2010-02-01 21:26:08 +01:00
*
* @ access private
* @ since 3.0 . 0
2015-09-10 18:21:26 +02:00
* @ since 4.4 . 0 The `$domain` and `$path` parameters were added .
2015-05-26 23:51:31 +02:00
*
2015-09-10 05:53:24 +02:00
* @ global wpdb $wpdb
*
* @ param string $domain The requested domain for the error to reference .
* @ param string $path The requested path for the error to reference .
2010-02-01 21:26:08 +01:00
*/
2015-09-10 05:53:24 +02:00
function ms_not_installed ( $domain , $path ) {
global $wpdb ;
2010-02-01 21:26:08 +01:00
2015-03-07 06:35:26 +01:00
if ( ! is_admin () ) {
dead_db ();
}
2012-01-26 21:34:27 +01:00
wp_load_translations_early ();
2012-11-07 23:03:23 +01:00
$title = __ ( 'Error establishing a database connection' );
2015-03-07 06:35:26 +01:00
2010-05-03 11:51:40 +02:00
$msg = '<h1>' . $title . '</h1>' ;
2012-01-26 21:34:27 +01:00
$msg .= '<p>' . __ ( 'If your site does not display, please contact the owner of this network.' ) . '' ;
$msg .= ' ' . __ ( 'If you are the owner of this network please check that MySQL is running properly and all tables are error free.' ) . '</p>' ;
2014-06-10 02:44:15 +02:00
$query = $wpdb -> prepare ( " SHOW TABLES LIKE %s " , $wpdb -> esc_like ( $wpdb -> site ) );
if ( ! $wpdb -> get_var ( $query ) ) {
2015-03-26 16:17:28 +01:00
$msg .= '<p>' . sprintf (
/* translators: %s: table name */
__ ( '<strong>Database tables are missing.</strong> This means that MySQL is not running, WordPress was not installed properly, or someone deleted %s. You really should look at your database now.' ),
'<code>' . $wpdb -> site . '</code>'
) . '</p>' ;
2014-06-10 02:44:15 +02:00
} else {
2015-03-26 16:17:28 +01:00
$msg .= '<p>' . sprintf (
/* translators: 1: site url, 2: table name, 3: database name */
__ ( '<strong>Could not find site %1$s.</strong> Searched for table %2$s in database %3$s. Is that right?' ),
'<code>' . rtrim ( $domain . $path , '/' ) . '</code>' ,
'<code>' . $wpdb -> blogs . '</code>' ,
'<code>' . DB_NAME . '</code>'
) . '</p>' ;
2014-06-10 02:44:15 +02:00
}
2012-01-26 21:34:27 +01:00
$msg .= '<p><strong>' . __ ( 'What do I do now?' ) . '</strong> ' ;
2015-04-12 23:29:32 +02:00
$msg .= __ ( 'Read the <a target="_blank" href="https://codex.wordpress.org/Debugging_a_WordPress_Network">bug report</a> page. Some of the guidelines there may help you figure out what went wrong.' );
2012-01-26 21:34:27 +01:00
$msg .= ' ' . __ ( 'If you’re still stuck with this message, then check that your database contains the following tables:' ) . '</p><ul>' ;
2010-04-30 17:21:10 +02:00
foreach ( $wpdb -> tables ( 'global' ) as $t => $table ) {
if ( 'sitecategories' == $t )
continue ;
$msg .= '<li>' . $table . '</li>' ;
2010-02-01 21:26:08 +01:00
}
$msg .= '</ul>' ;
2015-03-07 06:35:26 +01:00
wp_die ( $msg , $title , array ( 'response' => 500 ) );
2010-03-26 20:13:36 +01:00
}
2014-03-05 07:39:21 +01:00
/**
* This deprecated function formerly set the site_name property of the $current_site object .
*
* This function simply returns the object , as before .
* The bootstrap takes care of setting site_name .
*
* @ access private
* @ since 3.0 . 0
2014-05-25 16:32:16 +02:00
* @ deprecated 3.9 . 0 Use get_current_site () instead .
2014-03-05 07:39:21 +01:00
*
* @ param object $current_site
* @ return object
*/
function get_current_site_name ( $current_site ) {
2014-05-25 16:32:16 +02:00
_deprecated_function ( __FUNCTION__ , '3.9' , 'get_current_site()' );
2014-03-05 07:39:21 +01:00
return $current_site ;
}
2014-04-07 23:28:16 +02:00
/**
* This deprecated function managed much of the site and network loading in multisite .
*
* The current bootstrap code is now responsible for parsing the site and network load as
* well as setting the global $current_site object .
*
* @ access private
* @ since 3.0 . 0
* @ deprecated 3.9 . 0
*
2015-05-26 23:51:31 +02:00
* @ global object $current_site
*
2014-04-07 23:28:16 +02:00
* @ return object
*/
function wpmu_current_site () {
global $current_site ;
_deprecated_function ( __FUNCTION__ , '3.9' );
return $current_site ;
}