私はワードプレスのプラグインを作っています。アンインストール機能に含めるべき典型的なものは何ですか?
たとえば、インストール機能で作成したテーブルを削除する必要がありますか?
オプションエントリをクリーンアップしますか?
他に何か?
私はワードプレスのプラグインを作っています。アンインストール機能に含めるべき典型的なものは何ですか?
たとえば、インストール機能で作成したテーブルを削除する必要がありますか?
オプションエントリをクリーンアップしますか?
他に何か?
回答:
以下に、上記のアクション中にトリガーされるコールバック関数を安全にフックする正しい方法を示します。
あなたが使用するプラグインでこのコードを使用できるように
検証できる3つの異なるデモプラグインを示し、後で独自のプラグインにコードを実装します。
このトピックは非常に難しく、非常に詳細であり、多数のエッジケースがあるため、この答えは決して完璧ではありません。今後も改善していきますので、定期的にご確認ください。
プラグインのセットアップコールバックはコアによってトリガーされ、コアがこれを行う方法に影響はありません。留意すべきことがいくつかあります。
echo/print
セットアップコールバック時に何も(!)。これによりheaders already sent
メッセージが表示され、コアはプラグインを非アクティブ化および削除することを推奨します...尋ねないでください:わかっています...exit()
実際に何が起こっているかについての洞察を得ることができるように、すべての異なるコールバックにステートメントを追加しました。コメントを外して、機能するものを確認してください。__FILE__ != WP_PLUGIN_INSTALL
プラグインが実際にアンインストールされているかどうかを確認するためにand をチェックすることが非常に重要です(そうでない場合:abort!)。on_deactivation()
開発中に単純にコールバックをトリガーすることをお勧めします。そうすれば、すべてを取り戻すために必要な時間を節約できます。少なくともこれは私がやっていることです。defined( 'ABSPATH' ) OR exit;
wp_die()
適切なアクセス許可を要求する画面で予期しない結果が発生する可能性があります(再試行したい場合はそうです)。これは、コアがユーザーをリダイレクトし、現在$GLOBALS['wp_list_table']->current_action();
をに設定してからerror_scrape
リファラーをチェックするcheck_admin_referer('plugin-activation-error_' . $plugin);
ときに発生し$plugin
ます$_REQUEST['plugin']
。そのため、リダイレクトはページの半分の負荷で発生し、この有線スクロールバーとダイ画面の洞察に黄色の管理通知/メッセージボックスが表示されます。これが発生した場合:落ち着いて、いくつかのexit()
段階的なデバッグでエラーを検索してください。関数定義の前にコールバックをフックすると、これが機能しない可能性があることに注意してください。
<?php
defined( 'ABSPATH' ) OR exit;
/**
* Plugin Name: (WCM) Activate/Deactivate/Uninstall - Functions
* Description: Example Plugin to show activation/deactivation/uninstall callbacks for plain functions.
* Author: Franz Josef Kaiser/wecodemore
* Author URL: http://unserkaiser.com
* Plugin URL: http://wordpress.stackexchange.com/questions/25910/uninstall-activate-deactivate-a-plugin-typical-features-how-to/25979#25979
*/
function WCM_Setup_Demo_on_activation()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
$plugin = isset( $_REQUEST['plugin'] ) ? $_REQUEST['plugin'] : '';
check_admin_referer( "activate-plugin_{$plugin}" );
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
function WCM_Setup_Demo_on_deactivation()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
$plugin = isset( $_REQUEST['plugin'] ) ? $_REQUEST['plugin'] : '';
check_admin_referer( "deactivate-plugin_{$plugin}" );
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
function WCM_Setup_Demo_on_uninstall()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
check_admin_referer( 'bulk-plugins' );
// Important: Check if the file is the one
// that was registered during the uninstall hook.
if ( __FILE__ != WP_UNINSTALL_PLUGIN )
return;
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
register_activation_hook( __FILE__, 'WCM_Setup_Demo_on_activation' );
register_deactivation_hook( __FILE__, 'WCM_Setup_Demo_on_deactivation' );
register_uninstall_hook( __FILE__, 'WCM_Setup_Demo_on_uninstall' );
これは、今日のプラグインで最も一般的な例です。
<?php
defined( 'ABSPATH' ) OR exit;
/**
* Plugin Name: (WCM) Activate/Deactivate/Uninstall - CLASS
* Description: Example Plugin to show activation/deactivation/uninstall callbacks for classes/objects.
* Author: Franz Josef Kaiser/wecodemore
* Author URL: http://unserkaiser.com
* Plugin URL: http://wordpress.stackexchange.com/questions/25910/uninstall-activate-deactivate-a-plugin-typical-features-how-to/25979#25979
*/
register_activation_hook( __FILE__, array( 'WCM_Setup_Demo_Class', 'on_activation' ) );
register_deactivation_hook( __FILE__, array( 'WCM_Setup_Demo_Class', 'on_deactivation' ) );
register_uninstall_hook( __FILE__, array( 'WCM_Setup_Demo_Class', 'on_uninstall' ) );
add_action( 'plugins_loaded', array( 'WCM_Setup_Demo_Class', 'init' ) );
class WCM_Setup_Demo_Class
{
protected static $instance;
public static function init()
{
is_null( self::$instance ) AND self::$instance = new self;
return self::$instance;
}
public static function on_activation()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
$plugin = isset( $_REQUEST['plugin'] ) ? $_REQUEST['plugin'] : '';
check_admin_referer( "activate-plugin_{$plugin}" );
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
public static function on_deactivation()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
$plugin = isset( $_REQUEST['plugin'] ) ? $_REQUEST['plugin'] : '';
check_admin_referer( "deactivate-plugin_{$plugin}" );
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
public static function on_uninstall()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
check_admin_referer( 'bulk-plugins' );
// Important: Check if the file is the one
// that was registered during the uninstall hook.
if ( __FILE__ != WP_UNINSTALL_PLUGIN )
return;
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
public function __construct()
{
# INIT the plugin: Hook your callbacks
}
}
このシナリオでは、メインのプラグインファイルと名前の第二のファイルだということを前提とsetup.php
名付けられたプラグインのサブディレクトリにはinc
:~/wp-content/plugins/your_plugin/inc/setup.php
。これは、プラグインフォルダーがデフォルトのWPフォルダー構造の外にある場合、およびコンテンツディレクトリの名前が変更された場合、またはセットアップファイルの名前が異なる場合にも機能します。inc
フォルダーのみが、プラグインのルートディレクトリからの相対的な同じ名前と場所を持っている必要があります。
注:3つのregister_*_hook()*
関数とクラスを取得して、プラグインにドロップするだけです。
メインプラグインファイル:
<?php
defined( 'ABSPATH' ) OR exit;
/**
* Plugin Name: (WCM) Activate/Deactivate/Uninstall - FILE/CLASS
* Description: Example Plugin
* Author: Franz Josef Kaiser/wecodemore
* Author URL: http://unserkaiser.com
* Plugin URL: http://wordpress.stackexchange.com/questions/25910/uninstall-activate-deactivate-a-plugin-typical-features-how-to/25979#25979
*/
register_activation_hook( __FILE__, array( 'WCM_Setup_Demo_File_Inc', 'on_activation' ) );
register_deactivation_hook( __FILE__, array( 'WCM_Setup_Demo_File_Inc', 'on_deactivation' ) );
register_uninstall_hook( __FILE__, array( 'WCM_Setup_Demo_File_Inc', 'on_uninstall' ) );
add_action( 'plugins_loaded', array( 'WCM_Setup_Demo_File', 'init' ) );
class WCM_Setup_Demo_File
{
protected static $instance;
public static function init()
{
is_null( self::$instance ) AND self::$instance = new self;
return self::$instance;
}
public function __construct()
{
add_action( current_filter(), array( $this, 'load_files' ), 30 );
}
public function load_files()
{
foreach ( glob( plugin_dir_path( __FILE__ ).'inc/*.php' ) as $file )
include_once $file;
}
}
セットアップファイル:
<?php
defined( 'ABSPATH' ) OR exit;
class WCM_Setup_Demo_File_Inc
{
public static function on_activation()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
$plugin = isset( $_REQUEST['plugin'] ) ? $_REQUEST['plugin'] : '';
check_admin_referer( "activate-plugin_{$plugin}" );
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
public static function on_deactivation()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
$plugin = isset( $_REQUEST['plugin'] ) ? $_REQUEST['plugin'] : '';
check_admin_referer( "deactivate-plugin_{$plugin}" );
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
public static function on_uninstall()
{
if ( ! current_user_can( 'activate_plugins' ) )
return;
check_admin_referer( 'bulk-plugins' );
// Important: Check if the file is the one
// that was registered during the uninstall hook.
if ( __FILE__ != WP_UNINSTALL_PLUGIN )
return;
# Uncomment the following line to see the function in action
# exit( var_dump( $_GET ) );
}
}
独自のDBテーブルまたはオプションを持つプラグインを作成する場合、物事を変更またはアップグレードする必要があるシナリオがあります。
悲しいことに今のところ、プラグイン/テーマのインストールまたは更新/アップグレードで何かを実行する可能性はありません。喜んで回避策があります:カスタム関数をカスタムオプションにフックします(そう、それはラメです-しかしそれは動作します)。
function prefix_upgrade_plugin()
{
$v = 'plugin_db_version';
$update_option = null;
// Upgrade to version 2
if ( 2 !== get_option( $v ) )
{
if ( 2 < get_option( $v ) )
{
// Callback function must return true on success
$update_option = custom_upgrade_cb_fn_v3();
// Only update option if it was an success
if ( $update_option )
update_option( $v, 2 );
}
}
// Upgrade to version 3, runs just after upgrade to version 2
if ( 3 !== get_option( $v ) )
{
// re-run from beginning if previous update failed
if ( 2 < get_option( $v ) )
return prefix_upgrade_plugin();
if ( 3 < get_option( $v ) )
{
// Callback function must return true on success
$update_option = custom_upgrade_cb_fn_v3();
// Only update option if it was an success
if ( $update_option )
update_option( $v, 3 );
}
}
// Return the result from the update cb fn, so we can test for success/fail/error
if ( $update_option )
return $update_option;
return false;
}
add_action('admin_init', 'prefix_upgrade_plugin' );
この更新関数は、あまり良くない/よく書かれた例ですが、前述のように:例であり、この手法はうまく機能します。後のアップデートでそれを改善します。
check_admin_referer()
ます。コアはそれ自体を実行せず、とにかく非サニタイズされた$_REQUEST
値と比較するため、サニタイズする必要はありません。しかし、彼らがそのために小さな女の子のように泣き始めたら、ただそれを使用するfilter_var()
かesc_attr()
、それで。
PHPバージョンやインストールされている拡張機能などの必要な機能について現在のシステムをテストするには、次のようなものを使用できます。
<?php # -*- coding: utf-8 -*-
/**
* Plugin Name: T5 Check Plugin Requirements
* Description: Test for PHP version and installed extensions
* Plugin URI:
* Version: 2013.03.31
* Author: Thomas Scholz
* Author URI: http://toscho.de
* Licence: MIT
* License URI: http://opensource.org/licenses/MIT
*/
/*
* Don't start on every page, the plugin page is enough.
*/
if ( ! empty ( $GLOBALS['pagenow'] ) && 'plugins.php' === $GLOBALS['pagenow'] )
add_action( 'admin_notices', 't5_check_admin_notices', 0 );
/**
* Test current system for the features the plugin needs.
*
* @return array Errors or empty array
*/
function t5_check_plugin_requirements()
{
$php_min_version = '5.4';
// see http://www.php.net/manual/en/extensions.alphabetical.php
$extensions = array (
'iconv',
'mbstring',
'id3'
);
$errors = array ();
$php_current_version = phpversion();
if ( version_compare( $php_min_version, $php_current_version, '>' ) )
$errors[] = "Your server is running PHP version $php_current_version but
this plugin requires at least PHP $php_min_version. Please run an upgrade.";
foreach ( $extensions as $extension )
if ( ! extension_loaded( $extension ) )
$errors[] = "Please install the extension $extension to run this plugin.";
return $errors;
}
/**
* Call t5_check_plugin_requirements() and deactivate this plugin if there are error.
*
* @wp-hook admin_notices
* @return void
*/
function t5_check_admin_notices()
{
$errors = t5_check_plugin_requirements();
if ( empty ( $errors ) )
return;
// Suppress "Plugin activated" notice.
unset( $_GET['activate'] );
// this plugin's name
$name = get_file_data( __FILE__, array ( 'Plugin Name' ), 'plugin' );
printf(
'<div class="error"><p>%1$s</p>
<p><i>%2$s</i> has been deactivated.</p></div>',
join( '</p><p>', $errors ),
$name[0]
);
deactivate_plugins( plugin_basename( __FILE__ ) );
}
PHP 5.5のチェックを使用してテストします。
register_activation_hook
ここへの呼び出しはありません-それを使用してみませんか?また、これは前または後register_activation_hook
にregister_activation_hook
発火し、上記に合格しなくても発火しますか?
add_action( 'admin_notices', 't5_check_admin_notices', 0 );
アクティベーションフックに移動しようとしましたが、プラグインはチェックを実行せずにアクティベートします。。。
admin_notices
。そのフックは後に起動するためです。