Обновление плагина из личного API


9

Я сейчас разрабатываю плагин для WordPress, который мне не нужен в репозитории плагинов для Wordpress. Однако я все еще хочу иметь возможность загружать обновления для своих клиентов из моего собственного API-репозитория.

Я много читал об этом, и одна вещь, которая, кажется, кое-что о чем-то, это pre_set_site_transient_update_pluginsфильтр, однако я не могу найти много информации об этом. Я пробовал этот учебник ( http://konstruktors.com/blog/wordpress/2538-automatic-updates-for-plugins-and-themes-hosted-outside-wordpress-extend/ ), который я не смог получить. Я могу сказать по комментариям, что другие могут действительно заставить это работать с тем, что должно быть почти с текущей версией WP (последний ответ 22 апреля).

Я попытался установить плагин с сайта и поместить папку API во второй домен, но уведомление об обновлении, которое я обычно получаю, когда доступно обновление, нигде не показывалось.

Я не уверен, возможно ли на самом деле, чтобы пользовательские плагины запускали автообновление из других репозиториев, поэтому я хотел бы узнать, есть ли у кого-нибудь здесь какой-либо опыт с этим материалом? Решение в учебном пособии казалось простым решением - интересно, возможно ли как-то сделать это более продвинутым способом?

Буду очень признателен за любую помощь в получении этого автообновления из моего собственного хранилища!

(PS: я использую WP версии 3.1.3)


Я могу опоздать на вечеринку, но вы можете найти плагин, который я создал специально для этого: WP Update Plugin Server
froger.me

Ответы:


7

Для тех, кто находит эту страницу, я предлагаю тем, кто желает предоставить свои собственные обновления вне официального репозитория WP, проверить этот проект на GitHub, который демонстрирует функциональность:

https://github.com/jeremyclark13/automatic-theme-plugin-update


2

Да, это возможно Для этого есть целая глава в Профессиональной разработке плагинов WordPress . Если вы еще этого не сделали, возьмите копию. Это определенно поможет.


На самом деле я нашел PDF-версию этого онлайн, но мне это тоже не показалось.
Саймон

Это работает, если вы делаете это правильно, я сделал это, посмотрите на HTTP API, codex.wordpress.org/HTTP_API
Wyck

Я только начал снова Что я получил, так это подключился к проверке обновлений плагина, используя add_filter("pre_set_site_transient_update_plugins","dne_altapi_check"); После этого у меня есть функция dne_altapi_check, которая содержит print_r("hi");- однако, когда я нажимаю кнопку «Проверить снова» под обновлениями, она вообще ничего не печатает. делать что-то не так при подключении к проверке обновлений?
Симон

Я помню, что кто-то написал класс для сотрудников по обновлению плагинов, но могу найти ссылку на этот пост: /
Mamaduka

1

Это коммерческий менеджер API плагинов и обновлений тем для WooCommerce, который специально работает, если плагин или тема не размещены на wordpress.org. Он предназначен для предоставления обновлений для собственных плагинов и тем. Плагин предназначен для тех, кто не хочет писать его самостоятельно, и нуждается в большом количестве функций, а также в рабочих примерах для плагинов и тем, которые продаются.

http://www.toddlahman.com/shop/wordpress-automatic-update-api-manager/


1

На http://wp-updates.com/ также есть отличный сервис - вы получаете одну тему или плагин бесплатно. К вашему сведению - это не мой сайт, но я пробовал это некоторое время назад, и это выглядело довольно хорошо.


Кажется, хороший сервис, но я не заметил (почти на бесплатном плане) HTTPS ни в веб-панели управления, ни в общении: более того, я не нашел какой-либо проверки владения при выполнении проверки обновлений (мне кажется, очень простой POST запрос), я чувствую, что материал можно украсть, зная имя плагина и делая некоторые предположения. Я бы с удовольствием использовал его, если бы он показался мне более профессиональным в плане безопасности.
действительно хорошая

1

Для установки на одном сайте (я не тестировал ее на нескольких сайтах), есть только две ловушки, которые нужно обновить из внешней службы, такой как github или gitlab. В приведенном ниже коде я использую gitlab, поскольку именно это я и использую для размещения своего кода прямо сейчас. Я, вероятно, должен абстрагировать части gitlab ...

Первый хук, который вам нужно использовать, это pre_set_site_transient_update_themes. Это фильтр, который WordPress использует для установки site_transient, чтобы показать, есть ли доступные обновления. Используйте этот хук, чтобы подключиться к удаленной версии и посмотреть, есть ли доступные обновления. Если есть, то измените переходный процесс так, чтобы WordPress знал, что есть обновления, и мог показать уведомление пользователю.

Другой крючок, который вам нужно использовать, это upgrader_source_selection. Этот фильтр необходим для gitlab в любом случае, потому что имя загруженной папки не совпадает с темой, поэтому мы используем этот хук, чтобы переименовать его в правильное имя. Если ваш удаленный репозиторий предоставляет почтовый индекс с правильным именем, то вам даже не нужен этот хук.

Третий, необязательный, хук, который вы можете использовать, это auto_update_themeесли вы хотите автоматически обновлять вашу тему. В приведенном ниже примере я использую этот хук для автоматического обновления только этой конкретной темы.

Этот код был протестирован только с WordPress 4.9.x. Требуется PHP> 7.0.

functions.php

//* Load the updater.
require PATH_TO . 'updater.php';
$updater = new updater();
\add_action( 'init', [ $updater, 'init' ] );

updater.php

/**
 * @package StackExchange\WordPress
 */
declare( strict_types = 1 );
namespace StackExchange\WordPress;

/**
 * Class for updating the theme.
 */
class updater {

  /**
   * @var Theme slug.
   */
  protected $theme = 'theme';

  /**
   * @var Theme repository name.
   */
  protected $repository = 'project/theme';

  /**
   * @var Repository domain.
   */
  protected $domain = 'https://gitlab.com/';

  /**
   * @var CSS endpoint for repository.
   */
  protected $css_endpoint = '/raw/master/style.css';

  /**
   * @var ZIP endpoint for repository.
   */
  protected $zip_endpoint = '/repository/archive.zip';

  /**
   * @var Remote CSS URI.
   */
  protected $remote_css_uri;

  /**
   * @var Remote ZIP URI.
   */
  protected $remote_zip_uri;

  /**
   * @var Remote version.
   */
  protected $remote_version;

  /**
   * @var Local version.
   */
  protected $local_version;

  /**
   * Method called from the init hook to initiate the updater
   */
  public function init() {
    \add_filter( 'auto_update_theme', [ $this, 'auto_update_theme' ], 20, 2 );
    \add_filter( 'upgrader_source_selection', [ $this, 'upgrader_source_selection' ], 10, 4 );
    \add_filter( 'pre_set_site_transient_update_themes', [ $this, 'pre_set_site_transient_update_themes' ] );
  }

  /**
   * Method called from the auto_update_theme hook.
   * Only auto update this theme.
   * This hook and method are only needed if you want to auto update the theme.
   *
   * @return bool Whether to update the theme.
   */
  public function auto_update_theme( bool $update, \stdClass $item ) : bool {
    return $this->theme === $item->theme;
  }

  /**
   * Rename the unzipped folder to be the same as the existing folder
   *
   * @param string       $source        File source location
   * @param string       $remote_source Remote file source location
   * @param \WP_Upgrader $upgrader      \WP_Upgrader instance
   * @param array        $hook_extra    Extra arguments passed to hooked filters
   *
   * @return string | \WP_Error The updated source location or a \WP_Error object on failure
   */
  public function upgrader_source_selection( string $source, string $remote_source, \WP_Upgrader $upgrader, array $hook_extra ) {
    global $wp_filesystem;

    $update = [ 'update-selected', 'update-selected-themes', 'upgrade-theme' ];

    if( ! isset( $_GET[ 'action' ] ) || ! in_array( $_GET[ 'action' ], $update, true ) ) {
      return $source;
    }

    if( ! isset( $source, $remote_source ) ) {
      return $source;
    }

    if( false === stristr( basename( $source ), $this->theme ) ) {
      return $source;
    }

    $basename = basename( $source );
    $upgrader->skin->feedback( esc_html_e( 'Renaming theme directory.', 'bootstrap' ) );
    $corrected_source = str_replace( $basename, $this->theme, $source );

    if( $wp_filesystem->move( $source, $corrected_source, true ) ) {
      $upgrader->skin->feedback( esc_html_e( 'Rename successful.', 'bootstrap' ) );
      return $corrected_source;
    }

    return new \WP_Error();
  }

  /**
   * Add respoinse to update transient if theme has an update.
   *
   * @param $transient
   *
   * @return
   */
  public function pre_set_site_transient_update_themes( $transient ) {
    require_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php';
    $this->local_version = ( \wp_get_theme( $this->theme ) )->get( 'Version' );

    if( $this->hasUpdate() ) {
      $response = [
        'theme'       => $this->theme,
        'new_version' => $this->remote_version,
        'url'         => $this->construct_repository_uri(),
        'package'     => $this->construct_remote_zip_uri(),
        'branch'      => 'master',
      ];
      $transient->response[ $this->theme ] = $response;
    }

    return $transient;
  }

  /**
   * Construct and return the URI to the remote stylesheet
   *
   * @return string The remote stylesheet URI
   */
  protected function construct_remote_stylesheet_uri() : string {
    return $this->remote_css_uri = $this->domain . $this->repository . $this->css_endpoint;
  }

  /**
   * Construct and return the URI to the remote ZIP file
   *
   * @return string The remote ZIP URI
   */
  protected function construct_remote_zip_uri() : string {
    return $this->remote_zip_uri = $this->domain . $this->repository . $this->zip_endpoint;
  }

  /**
   * Construct and return the URI to remote repository
   *
   * @access protected
   * @since  1.0
   *
   * @return string The remote repository URI
   */
  protected function construct_repository_uri() : string {
    return $this->repository_uri = $this->domain . \trailingslashit( $this->repository );
  }

  /**
   * Get and return the remote version
   *
   * @return string The remote version
   */
  protected function get_remote_version() : string {
    $this->remote_stylesheet_uri = $this->construct_remote_stylesheet_uri();
    $response = $this->remote_get( $this->remote_stylesheet_uri );
    $response = str_replace( "\r", "\n", \wp_remote_retrieve_body( $response ) );
    $headers = [ 'Version' => 'Version' ];

    foreach( $headers as $field => $regex ) {
      if( preg_match( '/^[ \t\/*#@]*' . preg_quote( $regex, '/' ) . ':(.*)$/mi', $response, $match ) && $match[1] ) {
        $headers[ $field ] = _cleanup_header_comment( $match[1] );
      }
      else {
        $headers[ $field ] = '';
      }
    }

    return $this->remote_version = ( '' === $headers[ 'Version' ] ) ? '' : $headers[ 'Version' ];
  }

  /**
   * Return whether the theme has an update
   *
   * @return bool Whether the theme has an update
   */
  protected function hasUpdate() : bool {
    if( ! $this->remote_version ) $this->remote_version = $this->get_remote_version();
    return version_compare( $this->remote_version, $this->local_version, '>' );
  }

  /**
   * Wrapper for \wp_remote_get()
   *
   * @param string $url  The URL to get
   * @param array  $args Array or arguments to pass through to \wp_remote_get()
   *
   * @return array|WP_Error Return the request or an error object
   */
  protected function remote_get( string $url, array $args = [] ) {
    return \wp_remote_get( $url, $args );
  }
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.