Cuando se habla del patrón Modelo-Vista-Controlador a veces no queda claro cuál es el papel del Controlador.
Veamos:
El Modelo lleva la lógica de negocio, que básicamente es la representación de las entidades que la aplicación ha de manejar y sus relaciones. Así en una aplicación de gestión de bibliotecas tendremos modelos para los libros, socios, préstamos, etc., así como métodos para prestar libros, devolverlos, y otros muchos.
La Vista lleva la lógica de presentación y cada vista sabe mostrar unos datos que recibe y nada más.
¿Y el controlador? Solemos decir que el controlador pide datos al modelo y se los pasa a la vista, pero eso creo que no deja suficientemente claro su papel.
El controlador se encarga de la lógica de la aplicación. Esta lógica es la que representa lo que ha de hacer el sistema para solicitar los datos adecuados al modelo y qué hacer con ellos. Eso incluiría también saber si hay un usuario autorizado en la sesión, si hay variables del entorno que se deban tener en cuenta (como límites de paginación, variables de sesión y otros), saber qué hacer si no se obtienen datos del modelo, etc.
Lögica de negocio y lógica de aplicación
Publicado por Frankie en 23:19 0 comentarios Enlaces a esta entrada
Etiquetas: MVC
Una idea para no redirigir tanto
Estaba escribiendo el esqueleto de unas acciones para gestionar el registro y confirmación de usuarios cuando me di cuenta de que hay situaciones en las que puede ser buena idea usar Controller->render() para dirigir al usuario a páginas que le informen sobre el resultado de sus acciones, en lugar de Controller->redirect() para hacer lo mismo.
Por ejemplo:
function register() {
if ($this->data) {
if ($this->User->register($this->data)) {
$this->render('registration_ok');
}
}
}
En esta acción, si el registro se hace correctamente, la acción muestra la vista 'registration_ok', que mostraría un mensaje explicando al usuario que su registro ha sido correcto y lo que debe o puede hacer a continuación.
En otro caso, se mostraría la vista por defecto 'register'.
Publicado por Frankie en 01:21 0 comentarios Enlaces a esta entrada
Paginación y find personalizado. Un problema y ¿una solución?
Uno de los temas más habituales en los foros relacionados con CakePHP es el de la paginación de resultados. Surgen bastantes problemas cuando queremos ir un poco más allá de lo básico.
En mi caso, el problema que más fastidiado me tiene es la paginación con métodos find personalizados, ya que, por lo que he podido descubrir, cuando CakePHP calcula el número de registros totales no construye la petición correcta y, con frecuencia, ésta falla y o bien no devuelve la cuenta correcta o incluso el SQL falla porque no se incluyen las columnas adecuadas.
Esto es especialmente molesto cuando usas los métodos personalizados para hacer búsquedas a través de tablas relacionadas, por ejemplo, forzando los join necesarios, o añadiendo claves 'contain'.
Paginación y find personalizados
Es fácil paginar con métodos find personalizados. Basta indicarlo en la variable del controlador paginate, bajo la clave del modelo:
$this->paginate['Modelo'][0] = 'metodo';
Controller->paginate() hace una llamada a find('count') para obtener la cuenta total de registros y luego a find('el_metodo_que_sea') con los parámetros adecuados para obtener una página de datos. El problema es que cuando llama a find('count') éste recibe las condiciones que se hayan establecido en el controlador (a través de la variable paginate, por ejemplo) e ignora las modificaciones de la Query que se hacen en el método find personalizado.
Con frecuencia eso significa que la petición ya fracasa en la base de datos, por errores como "columna desconocida", y en el mejor de los casos nos hace una cuenta incorrecta, como por ejemplo, todos los registros de la tabla, cuando nosotros queremos seleccionar sólo los que tienen el campo "active" igual a 1 o cualquier otra condición que hayamos establecido en el método personalizado.
Por ejemplo, yo suelo utilizar mucho los joins en los métodos personalizados para evitar cargar el controlador con querys muy extensas. Por desgracia esto hace que la paginación "estándar" no sirva y haya que recurrir a otros métodos.
Métodos alternativos de contar
Generalmente se recomienda sobreescribir los métodos del modelo Model->paginate y Model->paginateCount, cosa que no me hace mucha gracia, porque precisamente una de las ventajas de los métodos find personalizados sería disponer de una batería de búsquedas típicas y tener que manejar cada caso en un método del modelo me da la impresión de que favorece un código confuso.
Puesto que personalizar los find es una "buena práctica", pienso que debería estar contemplado en Cake que se puedan usar en paginate con todas las consecuencias.
Solución 1
Mariano Iglesias propuso hace tiempo una solución, que no está mal. Consiste en guardar en un array los diferentes conjuntos de opciones para cada tipo de find y sobreescribir el método genérico find en AppModel. En este método se harían dos cosas:
En primer lugar, cuando se realiza un find personalizado, se mezclan mediante Set::merge las opciones predefinidas y las que se pasan en la llamada. Luego se ejecuta un find('all') con las nuevas opciones.
Cuando la llamada se hace desde Controller->paginate, también se efectúa una llamada a find('count'). La parte que nos interesa es que en esta llamada también se pasa el nombre del método personalizado, de modo, que se consulta el array de opciones y se pasa la información a find('count') para que pueda construir el SQL correcto.
Lo mejor es que eches un vistazo al código de Mariano.
Solución 2
Buceando en el código llegué a la conclusión de que hay otra vía "prometedora". La verdad es que había leído el artículo de Mariano Iglesias por encima y no me había parado a estudiarlo a fondo, lo que a lo mejor me hubiese ahorrado bastante trabajo, ya que en realidad el enfoque es muy parecido: se trata de asegurarse de que find('count') recibe las opciones correctas para construir su query.
El método que pide la cuenta de registros a la base de datos es Model->_findCount(), el cual toma recibe la query solicitada y la convierte en una query de recuento.
Éste método "sabe" si la petición va a ir seguida de una llamada a otro método find y de cuál (en las opciones recibe una clave 'type' que contiene esa información). El caso es que actualmente, _findCount() no hace nada con ese dato, por lo tanto, si nuestro método personalizado modifica la query de un modo u otro, no lo tiene en cuenta.
Por otro lado, uno de los sistemas para crear find personalizados se basa en métodos con dos pasadas. En la primera pasada se modifica la query, y devuelve la modificada, y en la segunda se procesan los resultados.
Así que podemos saber cuál es la query que se ejecutará "preguntándole" al método adecuado por ella. En principio, esto se puede hacer desde el _findCount. Por lo que con unas pocas líneas se consigue que éste tome exactamente la query adecuada.
Y puedes ver el método _findCount() modificado en mi AppModel.
Aparte de eso, envié la idea para ver si se puede hacer algo en el código del core. Puedes ver la discusión en este ticket.
Problemas
Ambas soluciones tienen algunos problemas.
El principal de ellos es que puede ocurrir que el find personalizado haga alguna cosa más que modificar la query (es la objeción que pone Mark Story en la discusión del ticket que puse) y puesto que llamaríamos al método dos veces (en la misma fase) podemos tener resultados indeseados.
Publicado por Frankie en 22:15 0 comentarios Enlaces a esta entrada
Etiquetas: paginación
Join con relaciones muchos a muchos
En la entrada anterior dejaba pendiente el tema de las relaciones muchos a muchos, aunque es probable que ya te hayas dado cuenta de como combinar tablas relacionadas de esta manera.
Este tipo de relaciones requiere una tabla intermedia (o join table) que nos permita asociar las parejas de registros. Las tablas izquierda y derecha se relacionan de uno a muchos con la join table. Por lo tanto tendremos que "unir" la tabla izquierda con la join table y ésta con la tabla derecha.
Etiquetar libros
Vamos a seguir con nuestro ejemplo y vamos a añadir una tabla tags a nuestro sistema para poder etiquetar cada libro con diversas palabras clave descriptoras. Por ejemplo, así:
CREATE TABLE `tags` (
`id` int(11) NOT NULL auto_increment,
`tag` varchar(200) default NULL,
PRIMARY KEY (`id`)
)
Y vamos a introducir algunos valores, para que la tabla quede así:
+----+-----------------------+
| id | tag |
+----+-----------------------+
| 1 | Novela |
| 2 | Lit. Castellana |
| 3 | Lit. Hispanoamericana |
| 4 | Lit. Francesa |
| 5 | Poesia |
+----+-----------------------+
5 rows in set (0,00 sec)
create table books_tags (books_id int(11) not null,tags_id int(11) not null);select * from books join books_tags join tags;select * from books, books_tags, tags;select title, tag from books join books_tags on books.id = books_tags.books_id join tags on books_tags.tags_id = tags.id;select title, author, tag from books join authors on books.author_id = authors.id join books_tags on books.id = books_tags.books_id join tags on books_tags.tags_id = tags.id order by title;select title, author, tag from books left join authors on books.author_id = authors.id join books_tags on books.id = books_tags.books_id join tags on books_tags.tags_id = tags.id order by title;select title, author, tag from books left join authors on books.author_id = authors.id join books_tags on books.id = books_tags.books_id join tags on books_tags.tags_id = tags.id where tags.tag = 'Novela'order by title;select title, author, tag from books left join authors on books.author_id = authors.id join books_tags on books.id = books_tags.books_id join tags on books_tags.tags_id = tags.id where tags.tag = 'Lit. Castellana'order by title;Publicado por Frankie en 10:12 0 comentarios Enlaces a esta entrada
Uniendo tablas con Join
En la entrada anterior dejé caer que explicaría un poco más a fondo los tipos de JOIN que se pueden hacer y qué diferencias hay entre ellos.
Una buena manera de entenderlo es practicando, por lo que es recomendable que crees algunas tablas sencillas y lances las querys como forma de ver en vivo los resultados de cada tipo de JOIN y así entender para qué casos te pueden servir. No hace falta que tengan muchos campos, ni muchos registros.
Un buen ejemplo puede ser una tabla de libros y una de autores, como las que siguen (en este ejemplo estoy usando MySQL):
CREATE TABLE `books` (
`id` int(11) NOT NULL auto_increment,
`title` varchar(200) default NULL,
`author_id` int(11) default NULL,
PRIMARY KEY (`id`)
)
CREATE TABLE `authors` (
`id` int(11) NOT NULL auto_increment,
`author` varchar(200) default NULL,
PRIMARY KEY (`id`)
)
# Dump of table authors# ------------------------------------------------------------
INSERT INTO `authors` (`id`,`author`) VALUES ('1','Cervantes');
INSERT INTO `authors` (`id`,`author`) VALUES ('2','García Márquez');
INSERT INTO `authors` (`id`,`author`) VALUES ('3','Saint_Exupery');
# Dump of table books
# ------------------------------------------------------------
INSERT INTO `books` (`id`,`title`,`author_id`) VALUES ('1','El quijote','1');
INSERT INTO `books` (`id`,`title`,`author_id`) VALUES ('2','100 años de soledad','2');
INSERT INTO `books` (`id`,`title`,`author_id`) VALUES ('3','El Principito','3');SELECT * FROM books JOIN authors+----+---------------------+-----------+----+----------------+
| id | title | author_id | id | author |
+----+---------------------+-----------+----+----------------+
| 1 | El quijote | 1 | 1 | Cervantes |
| 2 | 100 a?os de soledad | 2 | 1 | Cervantes |
| 3 | El Principito | 3 | 1 | Cervantes |
| 1 | El quijote | 1 | 2 | Garc?a M?rquez |
| 2 | 100 a?os de soledad | 2 | 2 | Garc?a M?rquez |
| 3 | El Principito | 3 | 2 | Garc?a M?rquez |
| 1 | El quijote | 1 | 3 | Saint_Exupery |
| 2 | 100 a?os de soledad | 2 | 3 | Saint_Exupery |
| 3 | El Principito | 3 | 3 | Saint_Exupery |
+----+---------------------+-----------+----+----------------+
9 rows in set (0,03 sec)
Por ejemplo, en un campeonato deportivo como una liga de fútbol tendremos una tabla equipos, que recoge el nombre de todos los equipos participantes.
Pues bien, un JOIN de la tabla consigo misma, nos permitirá obtener todos los partidos del campeonato. Eso sí, tendremos que recurrir a los alias para evitar un error de MySQL.
SELECT * FROM equipos AS Local JOIN equipos as Visitante+----+------------+----+------------+
| id | equipo | id | equipo |
+----+------------+----+------------+
| 1 | Barcelona | 1 | Barcelona |
| 2 | Madrid | 1 | Barcelona |
| 3 | Celta | 1 | Barcelona |
| 4 | Villarreal | 1 | Barcelona |
| 1 | Barcelona | 2 | Madrid |
| 2 | Madrid | 2 | Madrid |
| 3 | Celta | 2 | Madrid |
| 4 | Villarreal | 2 | Madrid |
| 1 | Barcelona | 3 | Celta |
| 2 | Madrid | 3 | Celta |
| 3 | Celta | 3 | Celta |
| 4 | Villarreal | 3 | Celta |
| 1 | Barcelona | 4 | Villarreal |
| 2 | Madrid | 4 | Villarreal |
| 3 | Celta | 4 | Villarreal |
| 4 | Villarreal | 4 | Villarreal |
+----+------------+----+------------+
16 rows in set (0,03 sec)
Con todo, esta query necesita alguna restricción para se perfecta, pues nos empareja cada equipo consigo misma, así que podemos añadir condiciones para eliminar esas parejas del resultado.
SELECT * FROM equipos AS Local JOIN equipos AS Visitante WHERE Local.id != Visitante.id+----+------------+----+------------+
| id | equipo | id | equipo |
+----+------------+----+------------+
| 2 | Madrid | 1 | Barcelona |
| 3 | Celta | 1 | Barcelona |
| 4 | Villarreal | 1 | Barcelona |
| 1 | Barcelona | 2 | Madrid |
| 3 | Celta | 2 | Madrid |
| 4 | Villarreal | 2 | Madrid |
| 1 | Barcelona | 3 | Celta |
| 2 | Madrid | 3 | Celta |
| 4 | Villarreal | 3 | Celta |
| 1 | Barcelona | 4 | Villarreal |
| 2 | Madrid | 4 | Villarreal |
| 3 | Celta | 4 | Villarreal |
+----+------------+----+------------+
12 rows in set (0,00 sec)
INNER JOIN
Este tipo de JOINS que nos dan el producto cartesiano son del tipo INNER y los resultados que podemos obtener de ellas estan siempre dentro de ese producto cartesiano.
Como deciamos antes, este tipo de resultados no es muy útil en algunos casos. Volviendo a nuestro ejemplo de ibros y autores, la query nos empareja obras y autores de todas las maneras posibles, lo que no se corresponde con la realidad. Nuestro sistema tiene que tener más conocimiento del mundo y poder utilizarlo al hacer la combinacion de tablas.
Nuestra tabla books cuenta con el campo author_id, la clave foránea que nos indica qué autor corresponde a cada libro. ¿Qué papel puede jugar en la combinacion de tablas?
JOIN admite una cláusula ON para definir qué condiciones deben usarse para que dos registros se combinen. En nuestro ejemplo, el cambo books.author_id debe coincidir con el campo author.id y lo expresamos así:
SELECT * FROM books INNER JOIN authors ON books.author_id = authors.id+----+---------------------+-----------+----+----------------+
| id | title | author_id | id | author |
+----+---------------------+-----------+----+----------------+
| 1 | El quijote | 1 | 1 | Cervantes |
| 2 | 100 a?os de soledad | 2 | 2 | Garc?a M?rquez |
| 3 | El Principito | 3 | 3 | Saint_Exupery |
+----+---------------------+-----------+----+----------------+
3 rows in set (0,05 sec)
De este modo, la query nos devuelve los libros correctamente emparejados con sus autores.
Podemos añadir la cláusula WHERE para especificar condiciones que restrinjan la busqueda de datos y esta puede usar campos de las tablas combinadas. Así, podemos buscar un libro por el nombre de su autor, a pesar de que este dato no está en la tabla books.
SELECT * FROM books INNER JOIN authors ON books.author_id = authors.id WHERE authors.author = 'Cervantes'+----+------------+-----------+----+-----------+
| id | title | author_id | id | author |
+----+------------+-----------+----+-----------+
| 1 | El quijote | 1 | 1 | Cervantes |
+----+------------+-----------+----+-----------+
1 row in set (0,00 sec)
Cuando combinamos ON y WHERE nos puede surgir la duda de si sería mejor poner las condiciones en ON o en WHERE.
La regla práctica sería poner en ON las condiciones para decidir que registros deben emparejarse y en WHERE las condiciones para filtrar o restringir el resultado. La base de datos primero genera la tabla temporal y luego hace el filtrado.
LEFT JOIN
Puede ocurrir que tengamos datos en una tabla que no tengan un registro asociado en la otra. Por ejemplo, añadimos un nuevo libro a nuestra tabla books pero no sabemos su autor (o es anónimo).
Ahora si pedimos una lista de todos los libros registrados con sus autores con la query anterior veremos que no aparecen los libros que no tengan autor. ¡Vaya! En muchos casos este comportamiento no nos interesa, querríamos tener toda la lista de libros aunque no sepamos el autor.
Para eso utilizamos LEFT JOIN.
Este tipo de combinación toma todos los registros válidos de la primera tabla (o tabla izquierda/left) y los combina con los registros de la otra tabla (derecha). Si no hay ningún registro que se pueda combinar lo hace con uno nuevo cuyos campos están todos en NULL.
En nuestros datos actuales tenemos tres libros y conocemos a sus autores correspondientes, así que al pedir la información a la base de datos nos devolverá este resultado:
SELECT * FROM books LEFT JOIN authors ON books.author_id = authors.id+----+---------------------+-----------+------+----------------+
| id | title | author_id | id | author |
+----+---------------------+-----------+------+----------------+
| 1 | El quijote | 1 | 1 | Cervantes |
| 2 | 100 a?os de soledad | 2 | 2 | Garc?a M?rquez |
| 3 | El Principito | 3 | 3 | Saint_Exupery |
+----+---------------------+-----------+------+----------------+
3 rows in set (0,00 sec)
INSERT INTO books (title) values ('Lazarillo de Tormes');SELECT * FROM books INNER JOIN authors ON books.author_id = authors.id;INSERT authors (author) values ('Quevedo')SELECT * FROM books RIGHT JOIN authors ON books.author_id = authors.id;Publicado por Frankie en 20:39 0 comentarios Enlaces a esta entrada
Joins en CakePHP: buscar con registros relacionados
Voy a intentar escribir varias entradas sobre el uso de JOIN con CakePHP, porque me parece un tema interesante, demandado, y relativamente poco conocido en el framework.
Que es eso de JOIN
En SQL la cláusula JOIN nos permite hacer búsquedas en varias tablas de la base de datos que estén relacionadas entre sí.
Así, por ejemplo, si tenemos el típico caso de Post hasAndBelongsToMany Tag, usando JOINS podemos hacer una búsqueda de aquellos Post que hayan sido etiquetados con una Tag determinada.
Para entender el concepto de JOIN puedes imaginarte que tomas cada registro de la tabla "principal" y le añades los campos del registro relacionado en otra tabla, quedándote una especie de "supertabla", y en ella puedes realizar búsquedas sobre cualquier campo de ambas tablas.
Existen varios tipos de JOIN: LEFT, RIGHT e INNER, de los que hablaré en otra entrada, que nos permiten definir cómo queremos que se combinen los datos de las tablas. El resultado puede ser muy diferente en cada caso y el tipo elegido depende de nuestras finalidades.
La situación en CakePHP
Como bien sabes, CakePHP utiliza una sintaxis especial basada en arrays para crear las querys a la base de datos. La ventaja de este sistema es que nos permite cambiar con relativa facilidad del motor de base de datos. Este "lenguaje super-SQL" aparece documentado en el manual, pero no menciona los joins.
Por una parte, algunos tipos de asociaciones de Modelos de CakePHP ejecutan finds creando los JOIN necesarios y es posible hacer búsquedas de un modelo, usando campos de modelos relaciones, pero no es el caso de nuestras queridas hasAndBelongsToMany (creo que las hasMany tampoco).
Otro aspecto importante tiene que ver con la paginación, pues también hay complicaciones cuando se intenta paginar con relaciones habtm.
Hay algunos trucos para hacer estas búsquedas en relaciones habtm (como redefinir las asociaciones según convenga como hasMany y belongsTo de forma temporal, o directamente escribir el query a mano y ejecutarlo con Model->query()), pero este artículo de Nate Abele en Bakery sobre el uso de joins me abrió los ojos a una solución que para mí es mucho mejor: cómo forzar joins en CakePHP.
Joins en CakePHP, la respuesta corta
- Para hacer joins hay que usar el método Model->find($type, $options), donde $type es el tipo de find ('all', 'list', 'first', o uno personalizado) y $options es nuestro array con todos los detalles del query.
- Una de las claves de $options será justamente 'joins'
- Cada 'join' se representa mediante un array que tiene estas claves
- 'table': el nombre de la tabla
- 'alias': normalmente el nombre del modelo asociado a la tabla
- 'type': el tipo de join
- 'conditions': el array de condiciones que nos dice cómo se relacionan las tablas (básicamente las condiciones que pondríamos en la cláusula ON)
- 'foreignKey': es un boolean, aunque no tengo claro cuál es su función y en los ejemplos que he visto se pone a false (echando un vistazo a dbo_source me pregunto si ponerlo en true serviría para hacer algo de automagia con las asociaciones ya existentes, pero no lo tengo nada claro).
- Pones todas las definiciones de joins que necesites.
- Para hacer la búsqueda, añades condiciones en la clave 'conditions' como es habitual.
/* Aquí definimos los join */
$extra['joins'] = array(
array('table' => 'channels',
'alias' => 'Channel',
'type' => 'LEFT',
'foreignKey' => FALSE,
'conditions' => array(
'Channel.id = Item.channel_id',
)
),
array('table' => 'channels_users',
'alias' => 'ChannelsUser',
'type' => 'LEFT',
'foreignKey' => FALSE,
'conditions' => array(
'ChannelsUser.channel_id = Channel.id',
'ChannelsUser.user_id' => $query['user']
)
)
);
/* Aquí están las condiciones, como puedes ver, ya podemos usar campos de los modelos relacionados, en este caso Channel */
$extra['conditions'] = array(
'Channel.private' => 1,
'Item.publish' => 1,
'Item.pubDate <= curdate()',
'or' => array(
'Item.expiration is null',
'Item.expiration > curdate()',
)
);
$extra['order'] = array(
'Item.pubDate' => 'desc',
'Item.created' => 'desc'
);
/* Finalmente mezclo las opciones definidas aquí, con las que se hayan pasado como argumento a find */
$query = Set::merge($query, $extra);
Publicado por Frankie en 21:53 0 comentarios Enlaces a esta entrada
Etiquetas: asociaciones, habtm, join
Una buena guía para sumergirse en HTML5
El libro de Mark Pilgrim Dive into HTML5 tiene una pinta estupenda para empezar a familiarizarse con el nuevo (e inacabado) estándar, y empezar a usarlo en la medida en que algunas de sus novedades ya están soportadas por los navegadores modernos.
Además, el autor hace varias recomendaciones prácticas acerca de cómo utilizar los nuevos elementos y características, dado a la vez soporte a navegadores que aún no las contemplan. Por ejemplo: cómo puedes empezar a utilizar ya los nuevos types del los input en los formularios, o cómo utilizar el elemento video sin dejar de lago el viejo explorer.
Publicado por Frankie en 20:49 0 comentarios Enlaces a esta entrada
Etiquetas: html