3.4 Guide de Migration

CakePHP 3.4 est une mise à jour de CakePHP 3.3 dont la compatibilité API est complète. Cette page souligne les changements et améliorations faits dans 3.4.

PHP 5.6 devient le minimum requis

CakePHP 3.4 a maintenant besoin d’au minimum PHP 5.6.0 puisque PHP 5.5 n’est plus supporté et ne recevra plus de correctifs de sécurité.

Dépréciations

La liste qui suit regroupe les méthodes, les propriétés et les comportements dépréciés. Ces différents éléments continueront de fonctionner jusqu’à la version 4.0.0, à partir de laquelle ils seront supprimés.

Dépréciations sur Request & Response

La majorité des dépréciations de la version 3.4 se trouve sur les objets Request et Response. Les méthodes existantes qui modifient directement les objets sont maintenant dépréciées et remplacées par les méthodes qui suivent le pattern des « objets immutables » décrit dans le standard PSR-7.

Plusieurs propriétés de Cake\Network\Request ont été dépréciées :

  • Request::$params est dépréciée. Utilisez Request::getAttribute('params') à la place.

  • Request::$data est dépréciée. Utilisez Request::getData() à la place.

  • Request::$query est dépréciée. Utilisez Request::getQueryParams() à la place.

  • Request::$cookies est dépréciée. Utilisez Request::getCookie() à la place.

  • Request::$base est dépréciée. Utilisez Request::getAttribute('base') à la place.

  • Request::$webroot est dépréciée. Utilisez Request::getAttribute('webroot') à la place.

  • Request::$here est dépréciée. Utilisez Request::getRequestTarget() à la place.

  • Request::$_session a été renommée Request::$session.

Certaines méthodes de Cake\Network\Request ont été dépréciées :

  • Les méthodes __get() & __isset() sont dépréciées. Utilisez getParam() à la place.

  • method() est dépréciée. Utilisez plutôt getMethod().

  • setInput() est dépréciée. Utilisez plutôt withBody().

  • Les méthodes ArrayAccess ont toutes été dépréciées.

  • Request::param() est dépréciée. Utilisez plutôt Request::getParam() .

  • Request::data() est dépréciée. Utilisez plutôt Request::getData().

  • Request::query() est dépréciée. Utilisez plutôt Request::getQuery().

  • Request::cookie() est dépréciée. Utilisez plutôt Request::getCookie().

Plusieurs méthodes de Cake\Network\Response ont été dépréciées soit parce qu’elles faisaient doublon avec les méthodes PSR-7, soit parce qu’elles ont été rendues obsolètes du fait de l’introduction du stack PSR-7 :

  • Response::header() est dépréciée. Utilisez plutôt getHeaderLine(), hasHeader() ou Response::getHeader().

  • Response::body() est dépréciée. Utilisez plutôt Response::withBody().

  • Response::statusCode() est dépréciée. Utilisez plutôt Response::getStatusCode().

  • Response::httpCodes() Cette méthode ne devrait plus être utilisée. CakePHP supporte maintenant tous les statuts standards recommandés.

  • Response::protocol() est dépréciée. Utilisez plutôt Response::getProtocolVersion().

  • send(), sendHeaders(), _sendHeader(), _sendContent(), _setCookies(), _setContentType(), et stop() sont dépréciées et rendues obsolètes par le stack HTTP PSR-7.

Du fait que les responses sont maintenant des objets immutables du fait des recommandations du standard PSR-7, de nombreuses méthodes « helper » de l’objet Response ont été dépréciées. Leurs variantes immutables sont maintenant recommandées :

  • Response::location() devient Response::withLocation()

  • Response::disableCache() devient Response::withDisabledCache()

  • Response::type() devient Response::withType()

  • Response::charset() devient Response::withCharset()

  • Response::cache() devient Response::withCache()

  • Response::modified() devient Response::withModified()

  • Response::expires() devient Response::withExpires()

  • Response::sharable() devient Response::withSharable()

  • Response::maxAge() devient Response::withMaxAge()

  • Response::vary() devient Response::withVary()

  • Response::etag() devient Response::withEtag()

  • Response::compress() devient Response::withCompression()

  • Response::length() devient Response::withLength()

  • Response::mustRevalidate() devient Response::withMustRevalidate()

  • Response::notModified() devient Response::withNotModified()

  • Response::cookie() devient Response::withCookie()

  • Response::file() devient Response::withFile()

  • Response::download() devient Response::withDownload()

Veuillez vous référez à la section Adopter les Responses Immutable pour plus d’informations avant de mettre à jour votre code car utiliser les méthodes immutables va demander plus de changements que le simple remplacement des méthodes.

Autres dépréciations

  • Les propriétés _public_ de Cake\Event\Event sont dépréciées, de nouvelles méthodes ont été ajoutées pour lire et écrire ces propriétés.

  • Event::name() est dépréciée. Utilisez Event::getName() à la place.

  • Event::subject() est dépréciée. Utilisez Event::getSubject() à la place.

  • Event::result() est dépréciée. Utilisez Event::getResult() à la place.

  • Event::data() est dépréciée. Utilisez Event::getData() à la place.

  • La valeur de Auth.redirect stockée en session n’est plus utilisée. Un paramètre d’URL est maintenant utilisé pour stocker l’URL de redirection. Ceci retire cependant la possibilité de définir une URL de redirection en session en dehors des scénarios de login.

  • AuthComponent ne stocke plus les URLs de redirection quand l’URL non autorisée n’est pas une action GET.

  • L’option ajaxLogin du AuthComponent est dépréciée. Vous devez maintenant utiliser le code de statut HTTP 403 pour déclencher le bon comportement côté client.

  • La méthode beforeRedirect du RequestHandlerComponent est dépréciée.

  • Le code de statut HTTP 306 de Cake\Network\Response est dépréciée. Sa phrase de statut est maintenant “Unused” car ce code de statut n’est pas standard.

  • Cake\Database\Schema\Table a été renommée en Cake\Database\Schema\TableSchema. Le nom précédent portait à confusion pour de nombreux utilisateurs.

  • L’option fieldList pour Cake\ORM\Table::newEntity() et patchEntity() a été renommée en fields pour être plus cohérent avec les autres parties de l’ORM.

  • Router::parse() est dépréciée. Router::parseRequest() est maintenant la méthode recommandée car elle accepte une request en argument et donne plus de contrôle et de flexibilité dans la manipulation des requêtes entrantes.

  • Route::parse() est dépréciée. Route::parseRequest() est maintenant la méthode recommandée car elle accepte une request en argument et donne plus de contrôle et de flexibilité dans la manipulation des requêtes entrantes.

  • FormHelper::input() est dépréciée. Utilisez FormHelper::control() à la place.

  • FormHelper::inputs() est dépréciée. Utilisez FormHelper::controls() à la place.

  • FormHelper::allInputs() est dépréciée. Utilisez FormHelper::allControls() à la place.

  • Mailer::layout() est dépréciée. Utilisez Mailer::setLayout() exposée par Mailer::__call() à la place.

Dépréciation des getters / setters combinés

Par le passé, CakePHP a exposé des méthodes combinées qui opéraient à la fois comme getter et comme setter. Ces méthodes compliquaient l’auto-complétion de certains IDE et auraient compliqué la mise en place de typage strictes sur les retours des méthodes dans le futur. Pour ces raisons, les getters / setters combinés sont maintenant séparés dans différentes méthodes.

La liste qui suit regroupe les méthodes qui sont dépréciées et remplacées par des méthodes getX() et setX() :

Cake\Core\InstanceConfigTrait
  • config()

Cake\Core\StaticConfigTrait
  • config()

  • dsnClassMap()

CakeConsoleConsoleOptionParse
  • command()

  • description()

  • epliog()

CakeDatabaseConnection
  • driver()

  • schemaCollection()

  • useSavePoints() (devenue enableSavePoints() / isSavePointsEnabled())

CakeDatabaseDriver
  • autoQuoting (devenue enableAutoQuoting() / isAutoQuotingEnabled())

CakeDatabaseExpressionFunctionExpression
  • name()

CakeDatabaseExpressionQueryExpression
  • tieWith() (devenue setConjunction() / getConjunction())

CakeDatabaseExpressionValuesExpression
  • columns()

  • values()

  • query()

CakeDatabaseQuery
  • connection()

  • selectTypeMap()

  • bufferResults() (devenue enableBufferedResults() / isBufferedResultsEnabled())

CakeDatabaseSchemaCachedCollection
  • cacheMetadata()

CakeDatabaseSchemaTableSchema
  • options()

  • temporary() (devenue setTemporary() / isTemporary())

CakeDatabaseTypeMap
  • defaults()

  • types()

CakeDatabaseTypeMapTrait
  • typeMap()

  • defaultTypes()

Cake\ORM\Association
  • name()

  • cascadeCallbacks()

  • source()

  • target()

  • conditions()

  • bindingKey()

  • foreignKey()

  • dependent()

  • joinType()

  • property()

  • strategy()

  • finder()

Cake\ORM\Association\BelongsToMany
  • targetForeignKey()

  • saveStrategy()

  • conditions()

Cake\ORM\Association\HasMany
  • saveStrategy()

  • foreignKey()

  • sort()

Cake\ORM\Association\HasOne
  • foreignKey()

CakeORMEagerLoadable
  • config()

  • setter part of canBeJoined() (devenue setCanBeJoined())

CakeORMEagerLoader
  • matching() (getMatching() devra être appelée après setMatching() pour conserver l’ancien comportement)

  • autoFields() (devenue enableAutoFields() / isAutoFieldsEnabled())

CakeORMLocatorTableLocator
  • config()

CakeORMQuery
  • eagerLoader()

  • hydrate() (now enableHydration()/isHydrationEnabled())

  • autoFields() (now enableAutoFields()/isAutoFieldsEnabled())

CakeORMTable
  • table()

  • alias()

  • registryAlias()

  • connection()

  • schema()

  • primaryKey()

  • displayField()

  • entityClass()

CakeMailerEmail
  • from()

  • sender()

  • replyTo()

  • readReceipt()

  • returnPath()

  • to()

  • cc()

  • bcc()

  • charset()

  • headerCharset()

  • emailPattern()

  • subject()

  • template() (devenue setTemplate() / getTemplate() et setLayout() / getLayout())

  • viewRender() (devenue setViewRenderer() / getViewRenderer())

  • viewVars()

  • theme()

  • helpers()

  • emailFormat()

  • transport()

  • messageId()

  • domain()

  • attachments()

  • configTransport()

  • profile()

CakeValidationValidator
  • provider()

CakeViewStringTemplateTrait
  • templates()

CakeViewViewBuilder
  • templatePath()

  • layoutPath()

  • plugin()

  • helpers()

  • theme()

  • template()

  • layout()

  • options()

  • name()

  • className()

  • autoLayout() (devenue enableAutoLayout() / isAutoLayoutEnabled())

Adopter les Responses Immutable

Avant de migrer votre code pour qu’il utilise les nouvelles méthodes de l’objet Response, sachez que les nouvelles méthodes sont bâties sur un concept différent. Les objets immutables sont généralement indiquées par le préfixe with (par exemple : withLocation()). Du fait que ces méthodes évoluent dans un contexte immutable, elles retournent de nouvelles instances que vous devez assigner à des variables ou des propriétés. Partons du principe que vous aviez du code de Controller similaire à celui-ci:

$response = $this->response;
$response->location('/login')
$response->header('X-something', 'a value');

Si vous faites un simple « rechercher / remplacer » pour changer le nom des méthodes, cela ne fonctionnera pas. Vous devriez plutôt remplacer votre code pour qu’il ressemble à ceci:

$this->response = $this->response
    ->withLocation('/login')
    ->withHeader('X-something', 'a value');

Voici les points clés à comprendre :

  1. Le résultat de vos changements doit être ré-assigné à $this->response. C’est le point le plus important pour conserver le fonctionnement souhaité.

  2. Les méthodes « setter » peuvent être chaînées. Cela vous permet d’éviter de stocker tous les états intermédiaires.

Astuces pour Migrer vos Components

Dans les versions précédentes de CakePHP, les Components possédaient souvent des références aux objets Request et Response pour pouvoir les modifier. Avant d’utiliser les méthodes immutable, vous devriez utiliser les Response attachées au Controller:

// Dans une méthode de Component (or callback)
$this->response->header('X-Rate-Limit', $this->remaining);

// Deviendrait
$controller = $this->getController();
$controller->response = $controller->response->withHeader('X-Rate-Limit', $this->remaining);

Dans les callbacks des Components, vous pouvez utiliser l’objet Event pour accéder à la Response / au Controller:

public function beforeRender($event)
{
    $controller = $event->getSubject();
    $controller->response = $controller->response->withHeader('X-Teapot', 1);
}

Astuce

Plutôt que conserver une référence aux Responses, récupérez plutôt la Response actuelle stockée dans le Controller et modifiez la propriété response quand vous avez terminé vos modifications.

Changement de comportements

Bien que ces changements garde la compatibilité API, ce sont tout de même des variations mineures qui pourraient avoir un impact sur votre application :

  • Les résultats de ORM\Query ne feront plus de typecast sur les alias de colonnes basé sur le type de colonne original. Par exemple, si vous faites un alias de created en created_time, vous obtiendrez maintenant une instance de Time plutôt qu’une chaîne de caractères.

  • Le AuthComponent utilise maintenant un paramètre URL pour stocker l’adresse de redirection quand un utilisateur non identifié est redirigé sur la page de connexion. Auparavant, l’URL de redirection était stockée en session. Utiliser un paramètre d’URL permet une meilleure compatibilité avec les différents navigateurs.

  • Le système de reflection de base de données traite maintenant les types de colonnes inconnus comme string et non plus comme text. L’impact de ce changement est notamment visible sur le FormHelper qui va générer des inputs à la place de textarea pour les types de colonnes inconnus.

  • AuthComponent ne va plus stocker ses messages Flash via la clé “auth”. Ils seront maintenant rendu avec le template “error” et sous la clé flash “default”. Ceci a été fait dans le but de simplifier AuthComponent.

  • Mailer\Email va maintenant automatiquement détecter les types de contenus des pièces jointes en utilisant mime_content_type si le « content-type » n’est pas spécifié. Auparavant, les pièces jointes étaient considérées comme “application/octet-stream” par défaut.

  • CakePHP utilise maintenant l’opérateur ... à la place de call_user_func_array(). Si vous passez des tableaux associatifs, vous devez mettre à jour votre code pour passer des tableaux à index numériques en utilisant la méthode array_values() pour les méthodes suivantes :

    • Cake\Mailer\Mailer::send()

    • Cake\Controller\Controller::setAction()

    • Cake\Http\ServerRequest::is()

Visibility Changes

  • MailerAwareTrait::getMailer() est maintenant protected.

  • CellTrait::cell() est maintenant protected.

Si les traits ci-dessus sont utilisés dans vos controllers, leurs méthodes publiques pouvaient être appelées par les règles de routing par défaut en tant qu’actions. Ces changements permettent d’apporter une sécurité à vos controllers. Si vous avez besoin que ces méthodes conservent une visibilité public, vous aurez besoin de mettre à jour les instructions use comme ceci:

use CellTrait {
    cell as public;
}
use MailerAwareTrait {
    getMailer as public;
}

Collection

  • CollectionInterface::chunkWithKeys() a été ajoutée. Les implémentations de CollectionInterface des utilisateurs devront maintenant implémenter cette méthode.

  • Collection::chunkWithKeys() a été ajoutée.

Erreur

  • Debugger::setOutputMask() et Debugger::outputMask() ont été ajoutées. Ces méthodes vous permettent de configurer des propriétés / clés de tableau qui devraient être masquées lors d’affichages générés par le Debugger (lors d’un appel à debug() par exemple).

Event

  • Event::getName() a été ajoutée.

  • Event::getSubject() a été ajoutée.

  • Event::getData() a été ajoutée.

  • Event::setData() a été ajoutée.

  • Event::getResult() a été ajoutée.

  • Event::setResult() a été ajoutée.

I18n

Routing

  • RouteBuilder::prefix() accepte maintenant un tableau de paramètres par défaut à ajouter à chaque route « connectée ».

  • Les routes peuvent maintenant être « matché » sur des hosts spécifiques à l’aide de l’option _host.

Email

  • Email::setPriority()/Email::getPriority() ont été ajoutées.

HtmlHelper

  • HtmlHelper::scriptBlock() n’englobe plus le Javascript dans un tag <![CDATA[ ]] par défaut. L’option safe qui contrôle ce comportement a maintenant sa valeur par défaut à false. Utiliser le tag <![CDATA[ ]] était seulement requis pour le XHTML qui n’est plus le doctype prédominant pour les pages web actuellement.

PaginatorHelper

  • PaginatorHelper::numbers() utilise maintenant une ellipse HTML au lieu de “…” dans les templates par défaut.

  • PaginatorHelper::total() a été ajoutée et permet de lire le nombre total de pages pour le résultat de requête actuellement paginé.

  • PaginatorHelper::generateUrlParams() a été ajoutée et est utilisée comme méthode de construction d’URL « bas niveau ».

  • PaginatorHelper::meta() peut maintenant créer des liens pour “first” et “last”.

FormHelper

  • Vous pouvez maintenant configurer les sources à partir desquelles FormHelper lit. Ceci simplifie la création des formulaires GET. Consultez Récupérer les valeurs du formulaire depuis la query string pour plus d’informations.

  • FormHelper::control() a été ajoutée.

  • FormHelper::controls() a été ajoutée.

  • FormHelper::allControls() a été ajoutée.

Validation

  • Validation::falsey() et Validation::truthy() ont été ajoutées.

TranslateBehavior

  • TranslateBehavior::translationField() a été ajoutée.

PluginShell

  • cake plugin load et cake plugin unload supportent maintenant une option --cli qui permet de mettre à jour bootstrap_cli.php à la place de bootstrap.php.

TestSuite

  • Le support de PHPUnit 6 a été ajouté. Puisque cette version du framework a au minimum besoin de PHP 5.6.0, les versions supportées de PHPUnit sont maintenant ^5.7|^6.0