Atualizar plugin da API pessoal

9

Estou desenvolvendo um plugin wordpress no momento que não desejo no repositório de plugins do Wordpress. No entanto, ainda quero poder enviar atualizações para meus clientes a partir de meu próprio repositório de API.

Eu tenho lido bastante sobre isso, e uma coisa que parece ser algo é o pre_set_site_transient_update_pluginsfiltro, no entanto, não consigo encontrar muita informação sobre isso. Eu tentei este tutorial ( http://konstruktors.com/blog/wordpress/2538-automatic-updates-for-plugins-and-themes-hosted-outside-wordpress-extend/ ) que não consegui trabalhar. Posso dizer pelos comentários que outras pessoas podem realmente fazer isso funcionar com o que deve ser quase a versão atual do WP (última resposta em 22 de abril).

Tentei instalar o plug-in a partir do site e colocar a pasta API em um segundo domínio, mas a notificação de atualização que geralmente recebo quando uma atualização está disponível não foi exibida em nenhum lugar.

Não tenho certeza se é realmente possível que plug-ins personalizados executem a atualização automática de outros repositórios, então eu gostaria de saber se alguém aqui tem alguma experiência com essas coisas? A solução no tutorial parecia ser uma solução fácil - será que é possível fazê-lo de uma maneira mais avançada?

Qualquer ajuda para obter essa atualização automática do meu próprio repositório funcionando seria muito apreciada!

(PS: Estou executando o WP versão 3.1.3)

Simon
fonte
Talvez eu esteja atrasado para a festa, mas você pode encontrar um plug-in que eu
criei

Respostas:

2

Sim, isso é possível. Há um capítulo inteiro no desenvolvimento profissional de plugins do WordPress dedicado a isso. Se ainda não o fez, pegue uma cópia. Definitivamente ajudará.

EAMann
fonte
Na verdade, encontrei uma versão em PDF dessa versão online, mas também não parecia funcionar para mim.
Simon
Ele faz o trabalho se você fizer isso direito, eu tenho feito isso, olhar para a API HTTP, codex.wordpress.org/HTTP_API
Wyck
Eu apenas comecei de novo. O que eu tenho até agora é conectar a verificação de atualização do plug-in usando. add_filter("pre_set_site_transient_update_plugins","dne_altapi_check"); Depois disso, tenho a função dne_altapi_check que contém print_r("hi");- no entanto, quando pressiono o botão "Verificar novamente" em atualizações, ele não imprime nada. fazendo algo errado ao conectar-se ao verificador de atualizações?
Simon
Lembro-me de que alguém escreveu classe para a equipe de atualização de plug-ins, mas pode encontrar o link para esse post: /
Mamaduka 3/11
1

Existe um gerenciador comercial de API de atualização de plug-ins e temas para o WooCommerce que funciona especificamente se o plug-in ou o tema não estiver hospedado no wordpress.org. Ele foi desenvolvido para fornecer atualizações para plugins e temas auto-hospedados. O plug-in é para aqueles que não querem escrever por conta própria e precisam de muitos recursos, além de exemplos de trabalho para plugins e temas que estão sendo vendidos.

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

Todd Lahman
fonte
1

Também há um serviço interessante em http://wp-updates.com/ - você obtém um tema ou plugin gratuitamente. FYI - este não é o meu site, mas eu tentei há um tempo e parecia muito bom.

cwd
fonte
Parece um bom serviço, mas eu não notei (quase de graça) HTTPS no painel de controle da Web nem na comunicação: além disso, não encontrei nenhum tipo de verificação de propriedade ao fazer a verificação da atualização (me parece muito simples Solicitação POST), acho que essas coisas podem ser roubadas, sabendo o nome do plug-in e fazendo algumas suposições. Eu adoraria usá-lo, se parecesse um pouco mais profissional no lado da segurança.
reallynice
1

Para uma instalação de site único (não testei em um site múltiplo), existem apenas dois ganchos que você precisa atualizar de um serviço externo, como o github ou o gitlab. No código abaixo, eu uso o gitlab, pois é o que eu uso para hospedar meu código agora. Eu provavelmente deveria abstrair as partes do gitlab ...

O primeiro gancho que você precisará usar é pre_set_site_transient_update_themes. Este é o filtro que o WordPress usa para definir o site_transient para mostrar se há atualizações disponíveis. Use esse gancho para conectar-se à sua versão remota e verificar se há atualizações disponíveis. Se houver, modifique o transitório para que o WordPress saiba que há atualizações e possa mostrar o aviso ao usuário.

O outro gancho que você precisará usar é upgrader_source_selection. Esse filtro é necessário, para o gitlab de qualquer maneira, porque o nome da pasta baixada não é o mesmo que o tema, portanto, usamos esse gancho para renomeá-lo para o nome correto. Se o seu repositório remoto fornece um zip com o nome correto, você nem precisa desse gancho.

O terceiro gancho opcional que você pode usar é auto_update_themese você deseja atualizar automaticamente seu tema. No exemplo abaixo, eu uso esse gancho para atualizar automaticamente apenas esse tema específico.

Este código foi testado apenas com o WordPress 4.9.x. Requer 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 );
  }
}
Nathan Johnson
fonte