Si has completado el día 4, ahora deberías estar familiarizado con el patrón MVC y deberías sentir más y más natural esta forma de codificación. Dedica un poco más de tiempo con esto para no tener que volver y mirar hacia atrás. Para practicar un poco el día de ayer, hemos personalizado la páginas Jobeet y en el proceso, también examinamos varios conceptos de Symfony, como el layout, los helpers, y los slots.
Hoy nos sumergiremos en el maravilloso mundo del Framework de Enrutamiento de Symfony .
Las URLs
Si haces clic en un puesto de trabajo en la página principal Jobeet, la URL se parece a esto:
/job/show/id/1
. Si ya has desarrollado sitios web PHP, probablemente estés más acostumbrados a las URL como /job.php?id=1
. ¿Cómo Symfony hace para que funcione? ¿Cómo Symfony determina la acción a llamar basándose en esta URL?
¿Porqué el id
de un job se obtiene con $request->getParameter('id')
?
Hoy, vamos a responder a todas estas preguntas.
Pero primero, vamos a hablar acerca de las URL y exactamente que son ellas. En un contexto web, una URL es el identificador único de un recurso web. Cuando accedes a una URL, estás pidiendo al navegador obtener un recurso identificado por esa URL. Por lo tanto, como la dirección URL es la interfaz entre la página web y el usuario, debe transmitir información significativa sobre algún recurso al que hace referencia. Pero las "tradicionales" URLs realmente no describen al recurso, sino que exponen la estructura interna de la aplicación. Al usuario no le importa que tu sitio web sea desarrollado con el lenguaje PHP o que el puesto de trabajo tiene un cierto identificador en la base de datos. Exponer el funcionamiento interno de tu aplicación es también es bastante malo en lo que medida de seguridad se refiere: ¿Qué pasa si el usuario intenta adivinar la dirección URL de los recursos que no tienen acceso? Así es, el desarrollador debe asegurarlos de la manera adecuada, pero más te vale ocultar la información sensible.
Las URL son tan importantes en Symfony que tiene todo un framework dedicado a su gestión: el framework de enrutamiento. El enrutamiento gestiona el URI interno y la URL externa. Cuando una petición llega, el enrutamiento analiza la URL y la convierte en un URI interno.
Ya has visto el URI interno de la página de puestos de trabajo en la plantilla showSuccess.php
:
'job/show?id='.$job->getId()
El helper url_for()
convierte éste URI interno a una correcta URL:
/job/show/id/1
El URI interno está hecho de varias partes: job
es el módulo, show
es la acción y la cadena de consulta añade los parámetros a pasar a la acción. El modelo genérico para los URIs internos es:
MÓDULO/ACCIÓN?clave=valor&clave_1=valor_1&...
Como el enrutamiento de Symfony es un proceso bidireccional, puedes cambiar las URLs sin cambiar la implementación técnica. Esta es una de las principales ventajas del patrón de diseño sobre controlador frontal.
La Configuración del Enrutamiento
El mapeo entre los URIs internos y las URLs externas esta listo en el archivo de configuración routing.yml
:
# apps/frontend/config/routing.yml homepage: url: / param: { module: default, action: index } default_index: url: /:module param: { action: index } default: url: /:module/:action/*
El archivo routing.yml
describe las rutas. Una ruta tiene un nombre (homepage
), un patrón (/:module/:action/*
), y algunos parámetros (bajo la clave param
).
Cuando una petición llega, el Enrutamiento trata de hacerla coincidir la URL con un patrón dado. La primera ruta que coincida gana, por lo tanto el orden en routing.yml
es importante. Echemos un vistazo a algunos ejemplos para comprender mejor cómo funciona esto.
Cuando solicitas la página de inicio Jobeet, la cual tiene la URL /job
, la primera ruta que coincide es con default_index
. En un patrón, una palabra con un prefijo dos puntos (:
) es una variable, por eso el patrón /:module
significa: Concidir con un /
seguida por cualquier cosa. En nuestro ejemplo, la variable module
será job
como su valor. Este valor puede ser obtenido con $request->getParameter('module')
. Esta ruta también define un valor por defecto para la variable action
. Por eso, para todas las URLs que coincidan con esta ruta, la petición también tendrá un parámetro action
con index
como su valor.
Si solicitas la página /job/show/id/1
, Symfony coincidirá con el último patrón: /:module/:action/*
. En un patrón, un asterisco (*
) coincide con una colección de pares variable/valor separados por una barra (/
):
Parámetro de petición | Valor |
---|---|
módulo | job |
acción | show |
id | 1 |
note
Las variables módulo
y acción
son especiales ya que son utilizados por Symfony
para determinar la acción a ejecutar.
La URL /job/show/id/1
se pueden crear desde una plantilla utilizando la siguiente llamada al helper url_for()
:
url_for('job/show?id='.$job->getId())
También puedes usar el nombre de la ruta gracias al prefijo @
:
url_for('@default?module=job&action=show&id='.$job->getId())
Ambas llamadas son equivalentes, pero esta última es mucho más rápido ya que el enrutamiento no tiene que analizar todas las rutas para encontrar el mejor patrón coincidente, y es menos complicado su implementación (los nombres del módulo y la acción no están presentes en el URI interno).
Personalizaciones del Enrutamiento
Por el momento, cuando la solicitas la URL /
en un navegador, tienes por defecto la página de felicitaciones de Symfony. Esto se debe a que esta URL coincide con la ruta homepage
. Pero tiene sentido cambiarla para que no sea la página de inicio de Jobeet. Para hacer el cambio, modifica la variable module
de la ruta homepage
a job
:
# apps/frontend/config/routing.yml homepage: url: / param: { module: job, action: index }
Puedes ahora cambiar el enlace al logo de Jobeet en el layout para usar la ruta
homepage
:
<!-- apps/frontend/templates/layout.php --> <h1> <a href="<?php echo url_for('@homepage') ?>"> <img src="/legacy/images/jobeet.gif" alt="Jobeet Job Board" /> </a> </h1>
Eso fue fácil!
tip
Cuando se actualiza la configuración del enrutamiento, los cambios son immediatamente tomados en cuenta en el entorno de desarrollo. Pero para hacerlos que tambien funcionen en el entorno de producción, necesitas limpiar el cache.
Para algo un poco más aplicado, vamos a cambiar el la URL de la página job a algo más significativo:
/job/sensio-labs/paris-france/1/web-developer
Sin saber nada acerca de Jobeet, y sin mirar en la página, se puede entender a partir de la URL que Sensio Labs está buscando un Web developer para trabajar en Paris, France.
note
Las URLs amigables son importantes porque ellas transmiten información al usuario. Es también útil cuando copias y pegas la URL en un email o para optimizar tu sitio web para los motores de búsqueda.
El siguiente patrón coincide con esta URL:
/job/:company/:location/:id/:position
Edita el archivo routing.yml
y agrega la ruta job_show_user
al principio del archivo:
job_show_user: url: /job/:company/:location/:id/:position param: { module: job, action: show }
Si actualizas la página de inicio Jobeet, los enlaces a jobs no han cambiado. Esto se debe a que para generar una ruta, necesitas pasar todas las variables requeridas. Por eso, necesitas cambiar la llamada a url_for()
en indexSuccess.php
a:
url_for('job/show?id='.$job->getId().'&company='.$job->getCompany(). '&location='.$job->getLocation().'&position='.$job->getPosition())
Un URI interno también puede ser expresado como un array:
url_for(array( 'module' => 'job', 'action' => 'show', 'id' => $job->getId(), 'company' => $job->getCompany(), 'location' => $job->getLocation(), 'position' => $job->getPosition(), ))
Los Requisitos
Durante el primer día de tutoría, hemos hablado de la validación y manejo de errores por una buena razón. El sistema de enrutamiento tiene incorporada una función de validación. Cada variable patrón pueda ser validada por una expresión regular definida utilizando la linea requirements
en la definición de la ruta:
job_show_user: url: /job/:company/:location/:id/:position param: { module: job, action: show } requirements: id: \d+
La anterior linea requirements
fuerza a que el id
sea un valor numérico. Sino lo es, la ruta no coincide.
La Clase Route
Cada ruta definida en routing.yml
es internamente convertida en un objecto de la clase sfRoute
. Esta clase se puede cambiar mediante una linea class
en la definición de la ruta. Si estás familiarizado con el protocolo HTTP, sabes que éste define varios "métodos",
como GET
, POST
, HEAD
, DELETE
, y PUT
. Los tres primeros cuentan con soporte en todos los navegadores, mientras que los otros dos no.
Para restringir una ruta a sólo la coincidencia con algunos métodos, puedes modificar la clase de enrutamiento a
sfRequestRoute
y añadir un requisito para la variable virtual sf_method
:
job_show_user: url: /job/:company/:location/:id/:position class: sfRequestRoute param: { module: job, action: show } requirements: id: \d+ sf_method: [get]
note
Exigir una ruta para solo algunos métodos HTTP no es totalmente
equivalente a usar sfWebRequest::isMethod()
en tus acciones.
Eso es porque el enrutamiento continuará buscando por una ruta coincidente
si el método no coincide con el esperado
El Objecto de la Clase Route
La nuevo URI interno para un puesto de trabajo es bastante largo y tedioso de escribir, (url_for('job/show?id='.$job->getId().'&company='.$job->getCompany().'&location='.$job->getLocation().'&position='.$job->getPosition())
), pero como acabamos de aprender en la sección anterior, la clase route puede ser modificada. Para la ruta job_show_user
, es mejor utilizar sfPropelRoute
ya que la clase está optimizada para las rutas que representan objetos Propel o colecciones de objetos Propel:
job_show_user: url: /job/:company/:location/:id/:position class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: show } requirements: id: \d+ sf_method: [get]
La linea options
personaliza el comportamiento de la ruta. Aquí, la opción model
define la clase del módelo Propel (JobeetJob
) relacionada a la ruta, y la opción type
define que esta ruta está vinculada a un objeto (también puedes utilizar list
si una ruta representa una colección de objetos).
La ruta job_show_user
es ahora consciente de su relación con JobeetJob
y así podemos simplificar la llamada url_for()
a:
url_for(array('sf_route' => 'job_show_user', 'sf_subject' => $job))
o solo:
url_for('job_show_user', $job)
note
El primer ejemplo es muy útil cuando necesitas pasar más argumentos que sólo un objeto.
Funciona porque todas las variables en la ruta tiene su método correspondiente en la clase JobeetJob
(por ejemplo, la variable company
es reemplazada con el valor de getCompany()
).
Si hechas una mirada a las URL generadas, no son todavía bastantes amigables como queremos que sean:
http://jobeet.localhost/frontend_dev.php/job/Sensio+Labs/Paris%2C+France/1/Web+Developer
Tenemos que "slugear" los valores de columna mediante la sustitución de todos los caracteres no ASCII por un -
. Abre el archivo JobeetJob
y añade los siguientes métodos para la clase:
// lib/model/JobeetJob.php public function getCompanySlug() { return Jobeet::slugify($this->getCompany()); } public function getPositionSlug() { return Jobeet::slugify($this->getPosition()); } public function getLocationSlug() { return Jobeet::slugify($this->getLocation()); }
A continuación, crea el archivo lib/Jobeet.class.php
y añadir el método slugify
en él:
// lib/Jobeet.class.php class Jobeet { static public function slugify($text) { // replace all non letters or digits by - $text = preg_replace('/\W+/', '-', $text); // trim and lowercase $text = strtolower(trim($text, '-')); return $text; } }
note
En este tutorial, nunca mostramos la sentencia de apertura <?php
en los ejemplos de código que solo tienen código PHP puro para optimizar el espacio.
Deberías obviamente recordar agregarlos cuando creas un nuevo archivo PHP.
Tenemos definidos tres nuevos métodos "virtuales" : getCompanySlug()
,
getPositionSlug()
, y getLocationSlug()
. Ellos devuelven sus correspondiente valor de columna después de pasalos por el método slugify()
. Ahora, puedes sustituir los nombres de las columas reales por sus virtuales en la ruta job_show_user
:
job_show_user: url: /job/:company_slug/:location_slug/:id/:position_slug class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: show } requirements: id: \d+ sf_method: [get]
Antes de actualizar la página principal Jobeet, es necesario vaciar la caché, como hemos añadido una nueva clase (Jobeet
):
$ php symfony cc
Tendrás ahora las URLs esperadas:
http://jobeet.localhost/frontend_dev.php/job/sensio-labs/paris-france/1/web-developer
Pero esa es sólo la mitad de la historia. La ruta es capaz de generar una URL sobre la base de un objeto, pero también es capaz de encontrar el objeto en relación con una determinada URL. Los objetos pueden ser recuperados con el método getObject()
de la ruta. Al analizar una petición, el enrutamiento guarda la ruta del objeto coincidentes para que la uses en las acciones. Por lo tanto, cambia el método executeShow()
para utilizar el objeto route para obtener el objeto Jobeet
:
class jobActions extends sfActions { public function executeShow(sfWebRequest $request) { $this->job = $this->getRoute()->getObject(); $this->forward404Unless($this->job); } // ... }
Si intentas obtener un job para un desconocido id
, verás una página de error 404 pero el mensaje de error ha cambiado:
Esto es así porque el error 404 ha sido lanzado de forma automática por el método
getRoute()
. Así, podemos simplificar el método executeShow
aún más:
class jobActions extends sfActions { public function executeShow(sfWebRequest $request) { $this->job = $this->getRoute()->getObject(); } // ... }
tip
Si no deseas que la ruta genere un error 404, puede establecer la opción
allow_empty
a true
.
note
El objeto relacionado de una ruta es cargado ligeramente. Este es obtenido desde
la base de datos solo si llamas al método getRoute()
.
El Enrutamiento en Acciones y Plantillas
En una plantilla, el helper url_for()
convierte una URI interna a una URL external. Algunos otros helpers symfony también toman una URI interna como un argumento, como el helper
link_to()
el cual genera una etiqueta <a>
:
<?php echo link_to($job->getPosition(), 'job_show_user', $job) ?>
Genera el siguiente código HTML:
<a href="/job/sensio-labs/paris-france/1/web-developer">Web Developer</a>
Para ambos url_for()
y link_to()
también pueden generar URL absoluta:
url_for('job_show_user', $job, true); link_to($job->getPosition(), 'job_show_user', $job, true);
Si deseas generar una URL desde una acción, puedes usar el método generateUrl()
:
$this->redirect($this->generateUrl('job_show_user', $job));
La Clase de Colección de Rutas
Para el módulo job
, ya tenemos personalizado la ruta de la acción show
, pero
las URLs para los otros métodos (index
, new
, edit
, create
, update
,
and delete
) están aun gestionadas por la ruta default
:
default: url: /:module/:action/*
La ruta default
es una gran manera de comenzar la codificación sin definir demasiadas rutas. Pero como la ruta actúa como un "catch-all", no puede ser configurada para necesidades específicas.
Como todas las acciones job
están relacionadas con el modelo de la clase JobeetJob
, podemos facilmente definir una ruta particular sfPropelRoute
para cada uno de ellas como ya lo hemos hecho para la acción show
. Sin embargo, como el módulo job
define las clásicas siete posibles acciones para un modelo, también podemos utilizar la clase sfPropelRouteCollection
. Abre el archivo routing.yml
y modificalo para que se lea como sigue:
# apps/frontend/config/routing.yml job: class: sfPropelRouteCollection options: { model: JobeetJob } job_show_user: url: /job/:company_slug/:location_slug/:id/:position_slug class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: show } requirements: id: \d+ sf_method: [get] # default rules homepage: url: / param: { module: job, action: index } default_index: url: /:module param: { action: index } default: url: /:module/:action/*
La ruta job
anterior es en realidad un acceso directo que generará automáticamente las siguientes siete rutas sfPropelRoute
:
job: url: /job.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: list } param: { module: job, action: index, sf_format: html } requirements: { sf_method: get } job_new: url: /job/new.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: new, sf_format: html } requirements: { sf_method: get } job_create: url: /job.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: create, sf_format: html } requirements: { sf_method: post } job_edit: url: /job/:id/edit.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: edit, sf_format: html } requirements: { sf_method: get } job_update: url: /job/:id.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: update, sf_format: html } requirements: { sf_method: put } job_delete: url: /job/:id.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: delete, sf_format: html } requirements: { sf_method: delete } job_show: url: /job/:id.:sf_format class: sfPropelRoute options: { model: JobeetJob, type: object } param: { module: job, action: show, sf_format: html } requirements: { sf_method: get }
note
Algunas rutas generadas por sfPropelRouteCollection
tienen la misma URL. El
enrutamiento está aún disponible para usarlas porque todas ellas tienen diferentes
parámetro en el método HTTP.
Las rutas job_delete
y job_update
require métodos HTTP que no son compatibles con los navegadores (DELETE
y PUT
respectivamente). Esto funciona porque los simula Symfony. Abre la plantilla _form.php
Para ver un ejemplo:
// apps/frontend/modules/job/templates/_form.php <form action="..." ...> <?php if (!$form->getObject()->isNew()): ?> <input type="hidden" name="sf_method" value="PUT" /> <?php endif; ?> <?php echo link_to( 'Delete', 'job/delete?id='.$form->getObject()->getId(), array('method' => 'delete', 'confirm' => 'Are you sure?') ) ?>
Todos los helpers symfony pueden ser invocados para simular cualquier método HTTP que desea pasando el parámetro especial sf_method
.
note
Symfony tiene otros parámetros especiales como sf_method
, todo lo que inicie con el prefijo
sf_
. En las rutas generadas antes, se puede ver otro:
sf_format
, que se explicará en los próximos días.
Debugeando la Ruta
Cuando se utiliza una colección de rutas, a veces es útil listar de rutas generadas. La tarea app:routes
muestra todas las rutas para una aplicación determinada:
$ php symfony app:routes frontend
También puedes tener una gran cantidad de información de depuración para una ruta pasando su nombre como un argumento adicional:
$ php symfony app:routes frontend job_edit
Las Rutas por defecto
Es una buena práctica definir las rutas para todas tus URLs. Ya que la ruta job
define todas las rutas necesarias para describir la aplicación Jobeet, sigue adelante y
quita o comenta las rutas por defecto del archivo de configuración routing.yml
:
# apps/frontend/config/routing.yml #default_index: # url: /:module # param: { action: index } # #default: # url: /:module/:action/*
La aplicación Jobeet debe seguir funcionando como antes.
Nos vemos mañana
Hoy estuvo lleno de una gran cantidad información nueva. Ya has aprendido cómo utilizar el framework de Enrutamiento de Symfony y como desacoplar tus URLs desde la implmentación técnica.
Mañana, no introducieremos ningún concepto nuevo, sino más bien pasar tiempo profundizando lo que hemos cubierto hasta ahora.
note
Si deseas comprobar el código del día de hoy, o de cualquier otro día, el código esta
disponible día a día en el repositorio SVN oficial de Jobeet
(http://svn.jobeet.org/propel/
).
Por ejemplo, puedes obtener el código de hoy de la
etiqueta release_day_05
:
$ svn co http://svn.jobeet.org/propel/tags/release_day_05/ jobeet/
Feedback
tip
Este capítulo ha sido traducido por Roberto Germán Puentes Díaz. Si encuentras algún error que deseas corregir o realizar algún comentario, no dudes en enviarlo por correo a puentesdiaz [arroba] gmail.com
This work is licensed under the Creative Commons Attribution-Share Alike 3.0 Unported License license.