PHP 8.2.31
Preview: connectors.php Size: 23.52 KB
/home/nshryvcy/radiantskinclinics.org/wp-includes/connectors.php

<?php
/**
 * Connectors API.
 *
 * @package WordPress
 * @subpackage Connectors
 * @since 7.0.0
 */

use WordPress\AiClient\AiClient;
use WordPress\AiClient\Providers\Http\DTO\ApiKeyRequestAuthentication;

/**
 * Checks if a connector is registered.
 *
 * @since 7.0.0
 *
 * @see WP_Connector_Registry::is_registered()
 *
 * @param string $id The connector identifier.
 * @return bool True if the connector is registered, false otherwise.
 */
function wp_is_connector_registered( string $id ): bool {
	$registry = WP_Connector_Registry::get_instance();
	if ( null === $registry ) {
		return false;
	}

	return $registry->is_registered( $id );
}

/**
 * Retrieves a registered connector.
 *
 * @since 7.0.0
 *
 * @see WP_Connector_Registry::get_registered()
 *
 * @param string $id The connector identifier.
 * @return array|null {
 *     Connector data, or null if not registered.
 *
 *     @type string $name           The connector's display name.
 *     @type string $description    The connector's description.
 *     @type string $logo_url       Optional. URL to the connector's logo image.
 *     @type string $type           The connector type, e.g. 'ai_provider'.
 *     @type array  $authentication {
 *         Authentication configuration. When method is 'api_key', includes
 *         credentials_url, setting_name, and optionally constant_name and
 *         env_var_name. When 'none', only method is present.
 *
 *         @type string $method          The authentication method: 'api_key' or 'none'.
 *         @type string $credentials_url Optional. URL where users can obtain API credentials.
 *         @type string $setting_name    Optional. The setting name for the API key.
 *         @type string $constant_name   Optional. PHP constant name for the API key.
 *         @type string $env_var_name    Optional. Environment variable name for the API key.
 *     }
 *     @type array  $plugin         {
 *         Optional. Plugin data for install/activate UI.
 *
 *         @type string   $file      The plugin's main file path relative to the plugins
 *                                   directory (e.g. 'my-plugin/my-plugin.php' or 'hello.php').
 *         @type callable $is_active Callback to determine whether the plugin is active. Receives no arguments and must return bool.
 *                                   Defaults to `__return_true`.
 *     }
 * }
 * @phpstan-return ?array{
 *     name: non-empty-string,
 *     description: string,
 *     logo_url?: non-empty-string,
 *     type: non-empty-string,
 *     authentication: array{
 *         method: 'api_key'|'none',
 *         credentials_url?: non-empty-string,
 *         setting_name?: non-empty-string,
 *         constant_name?: non-empty-string,
 *         env_var_name?: non-empty-string
 *     },
 *     plugin: array{
 *         file?: non-empty-string,
 *         is_active: callable(): bool,
 *     }
 * }
 */
function wp_get_connector( string $id ): ?array {
	$registry = WP_Connector_Registry::get_instance();
	if ( null === $registry ) {
		return null;
	}

	return $registry->get_registered( $id );
}

/**
 * Retrieves all registered connectors.
 *
 * @since 7.0.0
 *
 * @see WP_Connector_Registry::get_all_registered()
 *
 * @return array {
 *     Connector settings keyed by connector ID.
 *
 *     @type array ...$0 {
 *         Data for a single connector.
 *
 *         @type string      $name           The connector's display name.
 *         @type string      $description    The connector's description.
 *         @type string      $logo_url       Optional. URL to the connector's logo image.
 *         @type string      $type           The connector type, e.g. 'ai_provider'.
 *         @type array       $authentication {
 *             Authentication configuration. When method is 'api_key', includes
 *             credentials_url, setting_name, and optionally constant_name and
 *             env_var_name. When 'none', only method is present.
 *
 *             @type string $method          The authentication method: 'api_key' or 'none'.
 *             @type string $credentials_url Optional. URL where users can obtain API credentials.
 *             @type string $setting_name    Optional. The setting name for the API key.
 *             @type string $constant_name   Optional. PHP constant name for the API key.
 *             @type string $env_var_name    Optional. Environment variable name for the API key.
 *         }
 *         @type array       $plugin         {
 *             Optional. Plugin data for install/activate UI.
 *
 *             @type string   $file      The plugin's main file path relative to the plugins
 *                                       directory (e.g. 'my-plugin/my-plugin.php' or 'hello.php').
 *             @type callable $is_active Callback to determine whether the plugin is active. Receives no arguments and must return bool.
 *                                       Defaults to `__return_true`.
 *         }
 *     }
 * }
 * @phpstan-return array<string, array{
 *     name: non-empty-string,
 *     description: string,
 *     logo_url?: non-empty-string,
 *     type: non-empty-string,
 *     authentication: array{
 *         method: 'api_key'|'none',
 *         credentials_url?: non-empty-string,
 *         setting_name?: non-empty-string,
 *         constant_name?: non-empty-string,
 *         env_var_name?: non-empty-string
 *     },
 *     plugin: array{
 *         file?: non-empty-string,
 *         is_active: callable(): bool,
 *     }
 * }>
 */
function wp_get_connectors(): array {
	$registry = WP_Connector_Registry::get_instance();
	if ( null === $registry ) {
		return array();
	}

	return $registry->get_all_registered();
}

/**
 * Resolves an AI provider logo file path to a URL.
 *
 * Converts an absolute file path to a plugin URL. The path must reside within
 * the plugins or must-use plugins directory.
 *
 * @since 7.0.0
 * @access private
 *
 * @param string $path Absolute path to the logo file.
 * @return non-empty-string|null The URL to the logo file, or null if the path is invalid.
 */
function _wp_connectors_resolve_ai_provider_logo_url( string $path ): ?string {
	if ( ! $path ) {
		return null;
	}

	$path = wp_normalize_path( $path );

	if ( ! file_exists( $path ) ) {
		return null;
	}

	$mu_plugin_dir = wp_normalize_path( WPMU_PLUGIN_DIR );
	if ( str_starts_with( $path, $mu_plugin_dir . '/' ) ) {
		$logo_url = plugins_url( substr( $path, strlen( $mu_plugin_dir ) ), WPMU_PLUGIN_DIR . '/.' );
		return $logo_url ? $logo_url : null;
	}

	$plugin_dir = wp_normalize_path( WP_PLUGIN_DIR );
	if ( str_starts_with( $path, $plugin_dir . '/' ) ) {
		$logo_url = plugins_url( substr( $path, strlen( $plugin_dir ) ) );
		return $logo_url ? $logo_url : null;
	}

	_doing_it_wrong(
		__FUNCTION__,
		__( 'Provider logo path must be located within the plugins or must-use plugins directory.' ),
		'7.0.0'
	);

	return null;
}

/**
 * Initializes the connector registry with default connectors and fires the registration action.
 *
 * Creates the registry instance, registers built-in connectors (which cannot be unhooked),
 * and then fires the `wp_connectors_init` action for plugins to register their own connectors.
 *
 * @since 7.0.0
 * @access private
 */
function _wp_connectors_init(): void {
	$registry = new WP_Connector_Registry();
	WP_Connector_Registry::set_instance( $registry );

	// Only register default AI providers if AI support is enabled.
	if ( wp_supports_ai() ) {
		_wp_connectors_register_default_ai_providers( $registry );
	}

	// Non-AI default connectors.
	$registry->register(
		'akismet',
		array(
			'name'           => __( 'Akismet Anti-spam' ),
			'description'    => __( 'Protect your site from spam.' ),
			'type'           => 'spam_filtering',
			'plugin'         => array(
				'file'      => 'akismet/akismet.php',
				'is_active' => static function () {
					return defined( 'AKISMET_VERSION' );
				},
			),
			'authentication' => array(
				'method'          => 'api_key',
				'credentials_url' => 'https://akismet.com/get/',
				'setting_name'    => 'wordpress_api_key',
				'constant_name'   => 'WPCOM_API_KEY',
			),
		)
	);

	/**
	 * Fires when the connector registry is ready for plugins to register connectors.
	 *
	 * Built-in connectors and any AI providers auto-discovered from the WP AI Client
	 * registry have already been registered at this point and cannot be unhooked.
	 *
	 * AI provider plugins that register with the WP AI Client do not need to use
	 * this action — their connectors are created automatically. This action is
	 * primarily for registering non-AI-provider connectors or overriding metadata
	 * on existing connectors.
	 *
	 * Use `$registry->register()` within this action to add new connectors.
	 * To override an existing connector, unregister it first, then re-register
	 * with updated data.
	 *
	 * Example — overriding metadata on an auto-discovered connector:
	 *
	 *     add_action( 'wp_connectors_init', function ( WP_Connector_Registry $registry ) {
	 *         if ( $registry->is_registered( 'anthropic' ) ) {
	 *             $connector = $registry->unregister( 'anthropic' );
	 *             $connector['description'] = __( 'Custom description for Anthropic.', 'my-plugin' );
	 *             $registry->register( 'anthropic', $connector );
	 *         }
	 *     } );
	 *
	 * @since 7.0.0
	 *
	 * @param WP_Connector_Registry $registry Connector registry instance.
	 */
	do_action( 'wp_connectors_init', $registry );
}

/**
 * Registers connectors for the built-in AI providers.
 *
 * @since 7.0.0
 * @access private
 *
 * @param WP_Connector_Registry $registry The connector registry instance.
 */
function _wp_connectors_register_default_ai_providers( WP_Connector_Registry $registry ): void {
	// Built-in connectors.
	$defaults = array(
		'anthropic' => array(
			'name'           => 'Anthropic',
			'description'    => __( 'Text generation with Claude.' ),
			'type'           => 'ai_provider',
			'plugin'         => array(
				'file' => 'ai-provider-for-anthropic/plugin.php',
			),
			'authentication' => array(
				'method'          => 'api_key',
				'credentials_url' => 'https://platform.claude.com/settings/keys',
			),
		),
		'google'    => array(
			'name'           => 'Google',
			'description'    => __( 'Text and image generation with Gemini and Imagen.' ),
			'type'           => 'ai_provider',
			'plugin'         => array(
				'file' => 'ai-provider-for-google/plugin.php',
			),
			'authentication' => array(
				'method'          => 'api_key',
				'credentials_url' => 'https://aistudio.google.com/api-keys',
			),
		),
		'openai'    => array(
			'name'           => 'OpenAI',
			'description'    => __( 'Text and image generation with GPT and Dall-E.' ),
			'type'           => 'ai_provider',
			'plugin'         => array(
				'file' => 'ai-provider-for-openai/plugin.php',
			),
			'authentication' => array(
				'method'          => 'api_key',
				'credentials_url' => 'https://platform.openai.com/api-keys',
			),
		),
	);

	// Merge AI Client registry data on top of defaults.
	// Registry values (from provider plugins) take precedence over hardcoded fallbacks.
	$ai_registry = AiClient::defaultRegistry();

	foreach ( array_filter( $ai_registry->getRegisteredProviderIds() ) as $connector_id ) {
		$provider_class_name = $ai_registry->getProviderClassName( $connector_id );
		$provider_metadata   = $provider_class_name::metadata();

		$auth_method = $provider_metadata->getAuthenticationMethod();
		$is_api_key  = null !== $auth_method && $auth_method->isApiKey();

		if ( $is_api_key ) {
			$credentials_url = $provider_metadata->getCredentialsUrl();
			$authentication  = array(
				'method' => 'api_key',
			);
			if ( $credentials_url ) {
				$authentication['credentials_url'] = $credentials_url;
			}
		} else {
			$authentication = array( 'method' => 'none' );
		}

		$name        = $provider_metadata->getName();
		$description = $provider_metadata->getDescription();
		$logo_url    = $provider_metadata->getLogoPath()
			? _wp_connectors_resolve_ai_provider_logo_url( $provider_metadata->getLogoPath() )
			: null;

		if ( isset( $defaults[ $connector_id ] ) ) {
			// Override fields with non-empty registry values.
			if ( $name ) {
				$defaults[ $connector_id ]['name'] = $name;
			}
			if ( $description ) {
				$defaults[ $connector_id ]['description'] = $description;
			}
			if ( $logo_url ) {
				$defaults[ $connector_id ]['logo_url'] = $logo_url;
			}
			// Always update auth method; keep existing credentials_url as fallback.
			$defaults[ $connector_id ]['authentication']['method'] = $authentication['method'];
			if ( ! empty( $authentication['credentials_url'] ) ) {
				$defaults[ $connector_id ]['authentication']['credentials_url'] = $authentication['credentials_url'];
			}
		} else {
			$defaults[ $connector_id ] = array(
				'name'           => $name ? $name : ucwords( $connector_id ),
				'description'    => $description ? $description : '',
				'type'           => 'ai_provider',
				'authentication' => $authentication,
			);
			if ( $logo_url ) {
				$defaults[ $connector_id ]['logo_url'] = $logo_url;
			}
		}
	}

	// Register all default connectors directly on the registry.
	foreach ( $defaults as $id => $args ) {
		if ( 'api_key' === $args['authentication']['method'] ) {
			$sanitized_id = str_replace( '-', '_', $id );

			$args['authentication']['setting_name'] = "connectors_ai_{$sanitized_id}_api_key";

			// All AI providers use the {CONSTANT_CASE_ID}_API_KEY naming convention.
			$constant_case_key = strtoupper( (string) preg_replace( '/([a-z])([A-Z])/', '$1_$2', $sanitized_id ) ) . '_API_KEY';

			$args['authentication']['constant_name'] = $constant_case_key;
			$args['authentication']['env_var_name']  = $constant_case_key;
		}

		$args['plugin']['is_active'] = static function () use ( $ai_registry, $id ): bool {
			try {
				return $ai_registry->hasProvider( $id );
			} catch ( Exception $e ) {
				return false;
			}
		};

		$registry->register( $id, $args );
	}
}

/**
 * Masks an API key, showing only the last 4 characters.
 *
 * @since 7.0.0
 * @access private
 *
 * @param string $key The API key to mask.
 * @return string The masked key, e.g. "************fj39".
 */
function _wp_connectors_mask_api_key( string $key ): string {
	if ( strlen( $key ) <= 4 ) {
		return $key;
	}

	return str_repeat( "\u{2022}", min( strlen( $key ) - 4, 16 ) ) . substr( $key, -4 );
}

/**
 * Determines the source of an API key for a given connector.
 *
 * Checks in order: environment variable, PHP constant, database.
 * Environment variable and constant are only checked when their
 * respective names are provided.
 *
 * @since 7.0.0
 * @access private
 *
 * @param string $setting_name  The option name for the API key (e.g., 'connectors_spam_filtering_my_plugin_api_key').
 * @param string $env_var_name  Optional. Environment variable name to check (e.g., 'MY_PLUGIN_API_KEY').
 * @param string $constant_name Optional. PHP constant name to check (e.g., 'MY_PLUGIN_API_KEY').
 * @return string The key source: 'env', 'constant', 'database', or 'none'.
 */
function _wp_connectors_get_api_key_source( string $setting_name, string $env_var_name = '', string $constant_name = '' ): string {
	// Check environment variable first.
	if ( '' !== $env_var_name ) {
		$env_value = getenv( $env_var_name );
		if ( false !== $env_value && '' !== $env_value ) {
			return 'env';
		}
	}

	// Check PHP constant.
	if ( '' !== $constant_name && defined( $constant_name ) ) {
		$const_value = constant( $constant_name );
		if ( is_string( $const_value ) && '' !== $const_value ) {
			return 'constant';
		}
	}

	// Check database.
	$db_value = get_option( $setting_name, '' );
	if ( '' !== $db_value ) {
		return 'database';
	}

	return 'none';
}

/**
 * Checks whether an API key is valid for a given provider.
 *
 * @since 7.0.0
 * @access private
 *
 * @param string $key         The API key to check.
 * @param string $provider_id The WP AI client provider ID.
 * @return bool|null True if valid, false if invalid, null if unable to determine.
 */
function _wp_connectors_is_ai_api_key_valid( string $key, string $provider_id ): ?bool {
	try {
		$registry = AiClient::defaultRegistry();

		if ( ! $registry->hasProvider( $provider_id ) ) {
			_doing_it_wrong(
				__FUNCTION__,
				sprintf(
					/* translators: %s: AI provider ID. */
					__( 'The provider "%s" is not registered in the AI client registry.' ),
					$provider_id
				),
				'7.0.0'
			);
			return null;
		}

		$registry->setProviderRequestAuthentication(
			$provider_id,
			new ApiKeyRequestAuthentication( $key )
		);

		return $registry->isProviderConfigured( $provider_id );
	} catch ( Exception $e ) {
		wp_trigger_error( __FUNCTION__, $e->getMessage() );
		return null;
	}
}

/**
 * Masks and validates connector API keys in REST responses.
 *
 * On every `/wp/v2/settings` response, masks connector API key values so raw
 * keys are never exposed via the REST API.
 *
 * On POST or PUT requests, validates each updated key against the provider
 * before masking. If validation fails, the key is reverted to an empty string.
 *
 * @since 7.0.0
 * @access private
 *
 * @param WP_REST_Response $response The response object.
 * @param WP_REST_Server   $server   The server instance.
 * @param WP_REST_Request  $request  The request object.
 * @return WP_REST_Response The modified response with masked/validated keys.
 */
function _wp_connectors_rest_settings_dispatch( WP_REST_Response $response, WP_REST_Server $server, WP_REST_Request $request ): WP_REST_Response {
	if ( '/wp/v2/settings' !== $request->get_route() ) {
		return $response;
	}

	$data = $response->get_data();
	if ( ! is_array( $data ) ) {
		return $response;
	}

	$is_update = 'POST' === $request->get_method() || 'PUT' === $request->get_method();

	foreach ( wp_get_connectors() as $connector_id => $connector_data ) {
		$auth = $connector_data['authentication'];
		if ( 'api_key' !== $auth['method'] || empty( $auth['setting_name'] ) ) {
			continue;
		}

		$setting_name = $auth['setting_name'];
		if ( ! array_key_exists( $setting_name, $data ) ) {
			continue;
		}

		$value = $data[ $setting_name ];

		// On update, validate AI provider keys before masking.
		// Non-AI connectors accept keys as-is; the service plugin handles its own validation.
		if ( $is_update && is_string( $value ) && '' !== $value && 'ai_provider' === $connector_data['type'] ) {
			if ( true !== _wp_connectors_is_ai_api_key_valid( $value, $connector_id ) ) {
				update_option( $setting_name, '' );
				$data[ $setting_name ] = '';
				continue;
			}
		}

		// Mask the key in the response.
		if ( is_string( $value ) && '' !== $value ) {
			$data[ $setting_name ] = _wp_connectors_mask_api_key( $value );
		}
	}

	$response->set_data( $data );
	return $response;
}
add_filter( 'rest_post_dispatch', '_wp_connectors_rest_settings_dispatch', 10, 3 );

/**
 * Registers default connector settings.
 *
 * @since 7.0.0
 * @access private
 */
function _wp_register_default_connector_settings(): void {
	$registered_settings = get_registered_settings();

	foreach ( wp_get_connectors() as $connector_data ) {
		$auth = $connector_data['authentication'];
		if ( 'api_key' !== $auth['method'] || empty( $auth['setting_name'] ) ) {
			continue;
		}

		// Skip if the setting is already registered (e.g. by an owning plugin).
		if ( isset( $registered_settings[ $auth['setting_name'] ] ) ) {
			continue;
		}

		if ( ! isset( $connector_data['plugin']['is_active'] ) || ! is_callable( $connector_data['plugin']['is_active'] ) ) {
			continue;
		}

		if ( ! call_user_func( $connector_data['plugin']['is_active'] ) ) {
			continue;
		}

		register_setting(
			'connectors',
			$auth['setting_name'],
			array(
				'type'              => 'string',
				'label'             => sprintf(
					/* translators: %s: Connector name. */
					__( '%s API Key' ),
					$connector_data['name']
				),
				'description'       => sprintf(
					/* translators: %s: Connector name. */
					__( 'API key for the %s connector.' ),
					$connector_data['name']
				),
				'default'           => '',
				'show_in_rest'      => true,
				'sanitize_callback' => 'sanitize_text_field',
			)
		);
	}
}
add_action( 'init', '_wp_register_default_connector_settings', 20 );

/**
 * Passes stored connector API keys to the WP AI client.
 *
 * @since 7.0.0
 * @access private
 */
function _wp_connectors_pass_default_keys_to_ai_client(): void {
	try {
		$ai_registry = AiClient::defaultRegistry();
		foreach ( wp_get_connectors() as $connector_id => $connector_data ) {
			if ( 'ai_provider' !== $connector_data['type'] ) {
				continue;
			}

			$auth = $connector_data['authentication'];
			if ( 'api_key' !== $auth['method'] || empty( $auth['setting_name'] ) ) {
				continue;
			}

			if ( ! $ai_registry->hasProvider( $connector_id ) ) {
				continue;
			}

			// Skip if the key is already provided via env var or constant.
			$key_source = _wp_connectors_get_api_key_source( $auth['setting_name'], $auth['env_var_name'] ?? '', $auth['constant_name'] ?? '' );
			if ( 'env' === $key_source || 'constant' === $key_source ) {
				continue;
			}

			$api_key = get_option( $auth['setting_name'], '' );
			if ( ! is_string( $api_key ) || '' === $api_key ) {
				continue;
			}

			$ai_registry->setProviderRequestAuthentication(
				$connector_id,
				new ApiKeyRequestAuthentication( $api_key )
			);
		}
	} catch ( Exception $e ) {
		wp_trigger_error( __FUNCTION__, $e->getMessage() );
	}
}
add_action( 'init', '_wp_connectors_pass_default_keys_to_ai_client', 20 );

/**
 * Exposes connector settings to the connectors-wp-admin script module.
 *
 * @since 7.0.0
 * @access private
 *
 * @param array<string, mixed> $data Existing script module data.
 * @return array<string, mixed> Script module data with connectors added.
 */
function _wp_connectors_get_connector_script_module_data( array $data ): array {
	$registry = AiClient::defaultRegistry();

	if ( ! function_exists( 'validate_plugin' ) ) {
		require_once ABSPATH . 'wp-admin/includes/plugin.php';
	}

	$connectors = array();
	foreach ( wp_get_connectors() as $connector_id => $connector_data ) {
		$auth     = $connector_data['authentication'];
		$auth_out = array( 'method' => $auth['method'] );

		if ( 'api_key' === $auth['method'] ) {
			$auth_out['settingName']    = $auth['setting_name'] ?? '';
			$auth_out['credentialsUrl'] = $auth['credentials_url'] ?? null;
			$key_source                 = _wp_connectors_get_api_key_source( $auth['setting_name'] ?? '', $auth['env_var_name'] ?? '', $auth['constant_name'] ?? '' );
			$auth_out['keySource']      = $key_source;

			if ( 'ai_provider' === $connector_data['type'] ) {
				try {
					$auth_out['isConnected'] = $registry->hasProvider( $connector_id ) && $registry->isProviderConfigured( $connector_id );
				} catch ( Exception $e ) {
					$auth_out['isConnected'] = false;
				}
			} else {
				$auth_out['isConnected'] = 'none' !== $key_source;
			}
		}

		$connector_out = array(
			'name'           => $connector_data['name'],
			'description'    => $connector_data['description'],
			'logoUrl'        => ! empty( $connector_data['logo_url'] ) ? $connector_data['logo_url'] : null,
			'type'           => $connector_data['type'],
			'authentication' => $auth_out,
		);

		if ( ! empty( $connector_data['plugin']['file'] ) ) {
			$file         = $connector_data['plugin']['file'];
			$is_activated = (bool) call_user_func( $connector_data['plugin']['is_active'] );
			$is_installed = $is_activated || 0 === validate_plugin( $file );

			$connector_out['plugin'] = array(
				'file'        => $file,
				'isInstalled' => $is_installed,
				'isActivated' => $is_activated,
			);
		}

		$connectors[ $connector_id ] = $connector_out;
	}
	ksort( $connectors );
	$data['connectors']        = $connectors;
	$data['isFileModDisabled'] = ! wp_is_file_mod_allowed( 'install_plugins' );
	return $data;
}
add_filter( 'script_module_data_options-connectors-wp-admin', '_wp_connectors_get_connector_script_module_data' );

Directory Contents

Dirs: 33 × Files: 257

Name Size Perms Modified Actions
- drwxr-xr-x 2026-04-30 05:26:20
Edit Download
ai-client DIR
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
assets DIR
- drwxr-xr-x 2026-05-21 12:19:04
Edit Download
- drwxr-xr-x 2026-04-30 05:26:36
Edit Download
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
blocks DIR
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
build DIR
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
- drwxr-xr-x 2026-02-23 02:03:57
Edit Download
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
css DIR
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
customize DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
fonts DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
html-api DIR
- drwxr-xr-x 2026-02-26 12:18:32
Edit Download
ID3 DIR
- drwxr-xr-x 2026-02-23 02:01:28
Edit Download
images DIR
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
IXR DIR
- drwxr-xr-x 2026-02-23 02:01:28
Edit Download
js DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
l10n DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
- drwxr-xr-x 2026-05-21 12:19:03
Edit Download
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
PHPMailer DIR
- drwxr-xr-x 2026-02-23 02:01:28
Edit Download
pomo DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
Requests DIR
- drwxr-xr-x 2026-02-23 02:01:28
Edit Download
rest-api DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
SimplePie DIR
- drwxr-xr-x 2026-02-23 02:01:28
Edit Download
sitemaps DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
Text DIR
- drwxr-xr-x 2026-02-23 02:01:28
Edit Download
- drwxr-xr-x 2026-02-23 02:03:57
Edit Download
widgets DIR
- drwxr-xr-x 2026-02-23 02:01:29
Edit Download
23.80 KB lrw-r--r-- 2026-03-23 23:41:52
Edit Download
7.82 KB lrw-r--r-- 2026-02-19 15:57:02
Edit Download
38.39 KB lrw-r--r-- 2026-05-13 07:29:46
Edit Download
2.49 KB lrw-r--r-- 2026-04-27 04:00:40
Edit Download
11.90 KB lrw-r--r-- 2025-09-03 16:18:32
Edit Download
19.38 KB lrw-r--r-- 2026-03-23 12:58:48
Edit Download
7.35 KB lrw-r--r-- 2025-10-20 12:52:24
Edit Download
28.05 KB lrw-r--r-- 2026-03-13 03:24:40
Edit Download
316 B lrw-r--r-- 2021-08-11 13:08:02
Edit Download
15.24 KB lrw-r--r-- 2026-02-10 21:37:40
Edit Download
61.33 KB lrw-r--r-- 2026-02-25 20:27:46
Edit Download
17.83 KB lrw-r--r-- 2026-01-11 11:36:50
Edit Download
116.64 KB lrw-r--r-- 2026-05-19 13:06:44
Edit Download
12.47 KB lrw-r--r-- 2025-03-20 03:15:36
Edit Download
15.07 KB lrw-r--r-- 2024-03-23 18:20:12
Edit Download
10.76 KB lrw-r--r-- 2026-03-14 23:23:50
Edit Download
13.17 KB lrw-r--r-- 2025-04-30 02:44:38
Edit Download
33.83 KB lrw-r--r-- 2025-11-04 23:31:30
Edit Download
42.61 KB lrw-r--r-- 2026-01-06 10:57:56
Edit Download
55.65 KB lrw-r--r-- 2026-03-19 17:41:50
Edit Download
12.53 KB lrw-r--r-- 2026-02-23 10:34:46
Edit Download
29.30 KB lrw-r--r-- 2026-03-15 03:55:50
Edit Download
539 B lrw-r--r-- 2024-10-01 02:50:20
Edit Download
367 B lrw-r--r-- 2022-06-17 15:20:14
Edit Download
2.55 KB lrw-r--r-- 2025-01-23 00:48:26
Edit Download
42.65 KB lrw-r--r-- 2025-08-25 17:10:30
Edit Download
401 B lrw-r--r-- 2022-06-17 15:20:14
Edit Download
6.61 KB lrw-r--r-- 2024-09-18 01:08:16
Edit Download
664 B lrw-r--r-- 2020-07-21 16:58:02
Edit Download
20.63 KB lrw-r--r-- 2024-10-26 00:26:20
Edit Download
2.18 KB lrw-r--r-- 2023-04-05 17:12:26
Edit Download
453 B lrw-r--r-- 2024-10-01 02:50:20
Edit Download
457 B lrw-r--r-- 2021-01-26 18:45:58
Edit Download
36.83 KB lrw-r--r-- 2023-02-03 18:35:20
Edit Download
2.41 KB lrw-r--r-- 2023-09-14 16:46:20
Edit Download
8.28 KB lrw-r--r-- 2023-09-08 13:32:24
Edit Download
13.89 KB lrw-r--r-- 2024-03-18 19:46:14
Edit Download
11.76 KB lrw-r--r-- 2025-01-22 02:26:24
Edit Download
2.65 KB lrw-r--r-- 2023-09-14 16:46:20
Edit Download
7.43 KB lrw-r--r-- 2023-09-14 16:46:20
Edit Download
17.58 KB lrw-r--r-- 2026-03-13 03:26:42
Edit Download
5.14 KB lrw-r--r-- 2022-09-12 19:47:14
Edit Download
16.70 KB lrw-r--r-- 2025-04-03 17:53:28
Edit Download
8.07 KB lrw-r--r-- 2026-02-14 05:37:42
Edit Download
2.92 KB lrw-r--r-- 2025-09-29 01:56:28
Edit Download
1.32 KB lrw-r--r-- 2022-09-12 19:47:14
Edit Download
4.60 KB lrw-r--r-- 2025-08-07 18:47:34
Edit Download
11.57 KB lrw-r--r-- 2026-01-05 10:24:34
Edit Download
2.50 KB lrw-r--r-- 2025-10-21 11:14:02
Edit Download
1.95 KB lrw-r--r-- 2026-03-10 15:29:34
Edit Download
11.25 KB lrw-r--r-- 2026-03-13 03:25:20
Edit Download
4.28 KB lrw-r--r-- 2025-12-04 00:47:34
Edit Download
10.07 KB lrw-r--r-- 2026-03-11 01:11:48
Edit Download
68.32 KB lrw-r--r-- 2026-01-22 06:12:36
Edit Download
6.27 KB lrw-r--r-- 2026-01-11 11:50:42
Edit Download
6.40 KB lrw-r--r-- 2026-03-19 18:40:54
Edit Download
1.99 KB lrw-r--r-- 2024-09-20 06:07:12
Edit Download
6.91 KB lrw-r--r-- 2026-01-05 10:24:34
Edit Download
4.91 KB lrw-r--r-- 2025-09-29 20:29:36
Edit Download
16.83 KB lrw-r--r-- 2026-01-05 10:24:34
Edit Download
24.14 KB lrw-r--r-- 2026-01-05 10:24:34
Edit Download
3.93 KB lrw-r--r-- 2026-01-05 10:24:34
Edit Download
47.49 KB lrw-r--r-- 2025-12-23 04:14:38
Edit Download
9.15 KB lrw-r--r-- 2026-03-03 10:54:46
Edit Download
14.07 KB lrw-r--r-- 2026-05-08 18:24:44
Edit Download
25.51 KB lrw-r--r-- 2025-09-07 06:47:36
Edit Download
198.13 KB lrw-r--r-- 2026-05-13 11:20:44
Edit Download
56.61 KB lrw-r--r-- 2026-01-05 10:41:32
Edit Download
10.46 KB lrw-r--r-- 2025-01-23 00:48:26
Edit Download
10.95 KB lrw-r--r-- 2024-10-13 23:09:12
Edit Download
29.26 KB lrw-r--r-- 2026-02-28 04:18:40
Edit Download
70.89 KB lrw-r--r-- 2026-03-12 01:01:46
Edit Download
35.13 KB lrw-r--r-- 2026-02-24 11:33:36
Edit Download
16.69 KB lrw-r--r-- 2026-01-28 06:07:42
Edit Download
2.59 KB lrw-r--r-- 2025-12-07 09:16:34
Edit Download
39.95 KB lrw-r--r-- 2026-03-02 14:11:52
Edit Download
70.54 KB lrw-r--r-- 2026-05-13 11:20:44
Edit Download
15.54 KB lrw-r--r-- 2026-01-05 21:04:58
Edit Download
7.33 KB lrw-r--r-- 2023-02-21 21:39:20
Edit Download
253 B lrw-r--r-- 2024-09-27 23:28:14
Edit Download
7.96 KB lrw-r--r-- 2024-10-22 14:16:16
Edit Download
3.23 KB lrw-r--r-- 2025-07-31 03:03:30
Edit Download
969 B lrw-r--r-- 2024-10-01 02:50:20
Edit Download
16.25 KB lrw-r--r-- 2026-01-09 09:28:52
Edit Download
7.10 KB lrw-r--r-- 2026-01-05 10:58:26
Edit Download
12.95 KB lrw-r--r-- 2025-09-03 16:18:32
Edit Download
6.53 KB lrw-r--r-- 2023-06-22 18:57:24
Edit Download
3.43 KB lrw-r--r-- 2026-03-10 16:24:48
Edit Download
5.84 KB lrw-r--r-- 2023-06-22 18:36:26
Edit Download
1.97 KB lrw-r--r-- 2022-12-16 02:32:18
Edit Download
4.14 KB lrw-r--r-- 2026-01-30 17:52:40
Edit Download
2.91 KB lrw-r--r-- 2022-09-12 19:47:14
Edit Download
16.37 KB lrw-r--r-- 2026-01-10 10:03:50
Edit Download
40.67 KB lrw-r--r-- 2026-02-20 17:53:38
Edit Download
7.67 KB lrw-r--r-- 2026-03-20 19:41:02
Edit Download
20.22 KB lrw-r--r-- 2025-09-03 16:18:32
Edit Download
36.11 KB lrw-r--r-- 2025-08-27 01:05:30
Edit Download
17.01 KB lrw-r--r-- 2025-12-17 01:21:38
Edit Download
7.27 KB lrw-r--r-- 2024-02-28 03:38:16
Edit Download
6.62 KB lrw-r--r-- 2025-05-11 21:16:30
Edit Download
16.45 KB lrw-r--r-- 2026-02-13 22:52:44
Edit Download
1.79 KB lrw-r--r-- 2024-02-06 06:25:14
Edit Download
29.79 KB lrw-r--r-- 2026-01-06 11:08:04
Edit Download
6.67 KB lrw-r--r-- 2025-10-21 19:59:34
Edit Download
8.98 KB lrw-r--r-- 2025-06-19 00:39:52
Edit Download
19.25 KB lrw-r--r-- 2025-12-23 04:14:38
Edit Download
12.01 KB lrw-r--r-- 2024-09-14 02:12:16
Edit Download
17.11 KB lrw-r--r-- 2025-04-05 02:00:28
Edit Download
6.74 KB lrw-r--r-- 2024-03-06 10:05:12
Edit Download
30.86 KB lrw-r--r-- 2026-03-03 22:12:44
Edit Download
4.95 KB lrw-r--r-- 2026-01-06 10:36:58
Edit Download
4.25 KB lrw-r--r-- 2025-10-01 17:23:28
Edit Download
24.59 KB lrw-r--r-- 2026-01-11 11:50:42
Edit Download
29.95 KB lrw-r--r-- 2026-01-29 01:55:42
Edit Download
6.33 KB lrw-r--r-- 2026-03-03 10:54:46
Edit Download
159.50 KB lrw-r--r-- 2026-03-19 01:03:50
Edit Download
6.72 KB lrw-r--r-- 2022-10-04 07:59:14
Edit Download
10.90 KB lrw-r--r-- 2026-01-07 21:48:04
Edit Download
4.80 KB lrw-r--r-- 2026-03-02 04:04:46
Edit Download
3.44 KB lrw-r--r-- 2026-03-02 04:04:46
Edit Download
11.18 KB lrw-r--r-- 2025-02-23 16:11:22
Edit Download
62.20 KB lrw-r--r-- 2025-11-25 07:00:36
Edit Download
2.46 KB lrw-r--r-- 2023-09-08 13:32:24
Edit Download
9.10 KB lrw-r--r-- 2025-12-24 02:20:32
Edit Download
39.65 KB lrw-r--r-- 2026-04-30 08:59:38
Edit Download
35.93 KB lrw-r--r-- 2026-05-19 16:24:44
Edit Download
7.15 KB lrw-r--r-- 2025-02-11 16:14:22
Edit Download
3.47 KB lrw-r--r-- 2025-09-17 02:47:32
Edit Download
1.87 KB lrw-r--r-- 2025-01-23 00:48:26
Edit Download
30.74 KB lrw-r--r-- 2025-12-23 04:14:38
Edit Download
7.28 KB lrw-r--r-- 2026-03-03 10:54:46
Edit Download
7.38 KB lrw-r--r-- 2025-11-23 10:08:30
Edit Download
13.04 KB lrw-r--r-- 2026-03-11 05:32:44
Edit Download
19.12 KB lrw-r--r-- 2025-06-16 21:08:32
Edit Download
18.12 KB lrw-r--r-- 2025-03-27 01:42:28
Edit Download
39.80 KB lrw-r--r-- 2025-12-23 04:14:38
Edit Download
5.14 KB lrw-r--r-- 2026-03-03 10:54:46
Edit Download
979 B lrw-r--r-- 2024-02-15 00:27:10
Edit Download
18.49 KB lrw-r--r-- 2026-03-10 15:43:18
Edit Download
10.24 KB lrw-r--r-- 2024-11-20 07:50:24
Edit Download
1.77 KB lrw-r--r-- 2024-06-04 15:55:14
Edit Download
34.86 KB lrw-r--r-- 2026-02-13 22:52:44
Edit Download
7.19 KB lrw-r--r-- 2024-06-06 12:02:16
Edit Download
169.57 KB lrw-r--r-- 2026-02-20 07:25:46
Edit Download
64.22 KB lrw-r--r-- 2026-03-19 17:15:38
Edit Download
27.95 KB lrw-r--r-- 2024-07-20 03:44:16
Edit Download
4.69 KB lrw-r--r-- 2025-02-19 03:32:22
Edit Download
2.88 KB lrw-r--r-- 2026-01-11 11:50:42
Edit Download
43.07 KB lrw-r--r-- 2026-01-11 11:50:42
Edit Download
2.25 KB lrw-r--r-- 2025-02-17 16:24:22
Edit Download
22.48 KB lrw-r--r-- 2026-03-03 10:54:46
Edit Download
13.01 KB lrw-r--r-- 2024-07-26 11:56:14
Edit Download
3.27 KB lrw-r--r-- 2022-09-12 19:47:14
Edit Download
17.99 KB lrw-r--r-- 2026-03-13 03:25:20
Edit Download
209.98 KB lrw-r--r-- 2026-01-10 10:29:48
Edit Download
25.75 KB lrw-r--r-- 2026-01-06 11:08:04
Edit Download
115.86 KB lrw-r--r-- 2025-12-04 01:54:34
Edit Download
373 B lrw-r--r-- 2022-09-20 18:17:12
Edit Download
343 B lrw-r--r-- 2022-09-20 18:17:12
Edit Download
338 B lrw-r--r-- 2022-09-20 18:17:12
Edit Download
2.11 KB lrw-r--r-- 2026-03-24 16:19:52
Edit Download
100.79 KB lrw-r--r-- 2026-05-08 06:21:42
Edit Download
130.94 KB lrw-r--r-- 2026-03-10 15:43:18
Edit Download
19.10 KB lrw-r--r-- 2025-10-21 18:03:28
Edit Download
15.69 KB lrw-r--r-- 2026-01-09 16:47:48
Edit Download
23.52 KB lrw-r--r-- 2026-05-11 17:42:42
Edit Download
43.94 KB lrw-r--r-- 2026-02-14 01:26:40
Edit Download
400 B lrw-r--r-- 2022-06-17 15:20:14
Edit Download
11.10 KB lrw-r--r-- 2026-03-24 16:19:52
Edit Download
36.54 KB lrw-r--r-- 2026-05-13 07:29:46
Edit Download
2.24 KB lrw-r--r-- 2025-01-23 00:48:26
Edit Download
189.43 KB lrw-r--r-- 2026-03-13 03:24:40
Edit Download
338 B lrw-r--r-- 2022-06-17 15:20:14
Edit Download
37.99 KB lrw-r--r-- 2026-02-16 04:18:44
Edit Download
4.00 KB lrw-r--r-- 2026-01-06 10:36:58
Edit Download
9.47 KB lrw-r--r-- 2026-06-01 00:27:42
Edit Download
5.38 KB lrw-r--r-- 2024-03-04 17:41:10
Edit Download
3.05 KB lrw-r--r-- 2025-01-23 00:48:26
Edit Download
2.61 KB lrw-r--r-- 2020-01-29 05:45:18
Edit Download
1.16 KB lrw-r--r-- 2020-01-29 05:45:18
Edit Download
4.04 KB lrw-r--r-- 2024-03-04 17:41:10
Edit Download
3.71 KB lrw-r--r-- 2020-01-29 05:45:18
Edit Download
24.60 KB lrw-r--r-- 2026-01-29 03:43:42
Edit Download
9.56 KB lrw-r--r-- 2026-02-12 15:49:42
Edit Download
346.38 KB lrw-r--r-- 2026-03-06 23:02:46
Edit Download
283.52 KB lrw-r--r-- 2026-04-15 10:45:38
Edit Download
20.01 KB lrw-r--r-- 2026-05-19 16:24:44
Edit Download
8.45 KB lrw-r--r-- 2025-12-23 01:45:32
Edit Download
170.83 KB lrw-r--r-- 2026-05-19 16:24:44
Edit Download
20.29 KB lrw-r--r-- 2026-02-10 20:20:52
Edit Download
26.62 KB lrw-r--r-- 2026-05-08 01:23:44
Edit Download
5.72 KB lrw-r--r-- 2025-02-24 18:43:24
Edit Download
4.63 KB lrw-r--r-- 2023-07-11 02:38:26
Edit Download
0 B lrw-r--r-- 2025-05-19 02:20:16
Edit Download
80.64 KB lrw-r--r-- 2026-03-10 16:28:40
Edit Download
69.74 KB lrw-r--r-- 2026-05-08 01:23:44
Edit Download
156.39 KB lrw-r--r-- 2026-03-18 20:56:50
Edit Download
55.15 KB lrw-r--r-- 2026-03-01 00:59:46
Edit Download
162 B lrw-r--r-- 2019-10-08 21:19:04
Edit Download
61.79 KB lrw-r--r-- 2026-03-20 21:11:54
Edit Download
218.55 KB lrw-r--r-- 2026-03-20 21:11:54
Edit Download
65.17 KB lrw-r--r-- 2026-02-04 00:31:46
Edit Download
25.71 KB lrw-r--r-- 2026-03-13 03:25:20
Edit Download
4.81 KB lrw-r--r-- 2024-06-14 00:50:14
Edit Download
6.48 KB lrw-r--r-- 2023-02-24 06:23:20
Edit Download
21.24 KB lrw-r--r-- 2026-03-13 03:25:20
Edit Download
2.79 KB lrw-r--r-- 2025-10-17 21:14:32
Edit Download
89.69 KB lrw-r--r-- 2025-10-27 20:35:36
Edit Download
19.57 KB lrw-r--r-- 2026-03-18 15:00:42
Edit Download
3.69 KB lrw-r--r-- 2023-05-02 15:26:24
Edit Download
4.11 KB lrw-r--r-- 2025-08-27 17:42:30
Edit Download
40.75 KB lrw-r--r-- 2026-02-12 03:11:46
Edit Download
25.38 KB lrw-r--r-- 2025-01-23 00:48:26
Edit Download
43.23 KB lrw-r--r-- 2026-03-10 16:34:44
Edit Download
102.62 KB lrw-r--r-- 2026-05-08 19:59:44
Edit Download
6.18 KB lrw-r--r-- 2025-02-04 00:52:24
Edit Download
124.57 KB lrw-r--r-- 2026-03-16 02:51:42
Edit Download
35.65 KB lrw-r--r-- 2025-11-04 04:47:34
Edit Download
6.90 KB lrw-r--r-- 2026-01-10 10:29:48
Edit Download
67.01 KB lrw-r--r-- 2026-02-16 04:18:44
Edit Download
10.62 KB lrw-r--r-- 2024-12-21 04:35:24
Edit Download
289.58 KB lrw-r--r-- 2026-05-08 19:59:44
Edit Download
36.23 KB lrw-r--r-- 2025-09-01 01:43:30
Edit Download
200 B lrw-r--r-- 2020-11-12 16:17:08
Edit Download
200 B lrw-r--r-- 2020-11-12 16:17:08
Edit Download
98.52 KB lrw-r--r-- 2026-05-08 19:59:44
Edit Download
29.99 KB lrw-r--r-- 2026-01-06 11:08:04
Edit Download
19.00 KB lrw-r--r-- 2026-01-06 10:57:56
Edit Download
5.06 KB lrw-r--r-- 2022-04-06 19:33:04
Edit Download
255 B lrw-r--r-- 2020-11-17 03:52:06
Edit Download
22.66 KB lrw-r--r-- 2025-09-03 16:18:32
Edit Download
159.30 KB lrw-r--r-- 2026-05-19 16:24:44
Edit Download
11.66 KB lrw-r--r-- 2026-05-19 16:24:44
Edit Download
258 B lrw-r--r-- 2020-02-06 11:33:12
Edit Download
23.47 KB lrw-r--r-- 2026-01-05 11:04:34
Edit Download
3.16 KB lrw-r--r-- 2021-05-15 21:38:06
Edit Download
8.40 KB lrw-r--r-- 2025-08-27 14:34:28
Edit Download
441 B lrw-r--r-- 2020-11-12 16:17:08
Edit Download
7.39 KB lrw-r--r-- 2024-05-03 08:47:12
Edit Download
172.99 KB lrw-r--r-- 2026-03-03 10:54:46
Edit Download
544 B lrw-r--r-- 2023-10-01 04:22:28
Edit Download
4.17 KB lrw-r--r-- 2026-03-11 01:11:48
Edit Download
35.96 KB lrw-r--r-- 2026-02-20 07:25:46
Edit Download
1.85 KB lrw-r--r-- 2026-01-05 17:16:10
Edit Download
2.82 KB lrw-r--r-- 2026-01-05 21:04:58
Edit Download
3.96 KB lrw-r--r-- 2026-01-11 11:36:50
Edit Download
8.83 KB lrw-r--r-- 2026-03-09 22:43:38
Edit Download
131.48 KB lrw-r--r-- 2026-03-03 17:26:46
Edit Download
37.38 KB lrw-r--r-- 2026-01-09 07:48:52
Edit Download
174.63 KB lrw-r--r-- 2026-02-25 21:29:46
Edit Download
7.09 KB lrw-r--r-- 2025-10-21 06:35:32
Edit Download
6.45 KB lrw-r--r-- 2026-02-07 11:06:44
Edit Download
1.08 KB lrw-r--r-- 2026-05-20 21:39:46
Edit Download
602 B lrw-r--r-- 2026-01-19 21:58:46
Edit Download
69.17 KB lrw-r--r-- 2026-03-12 01:01:46
Edit Download
445 B lrw-r--r-- 2022-07-22 02:45:12
Edit Download
799 B lrw-r--r-- 2025-01-23 00:48:26
Edit Download

If ZipArchive is unavailable, a .tar will be created (no compression).