En este ejercicio se va a retomar la aplicación del 'ahorcado en línea':
Como se observó en el ejercicio anterior, esta aplicación tiene el defecto de guardar datos provisionales (el estado de los 'ahorcados') en un mapa dentro de la memoria del servidor, lo cual crea inconsistencias cuando varias instancias de la misma son montadas bajo un esquema de balanceo de carga.
-
Inicie una de las máquina virtuales Ubuntu trabajada anteriormente, e inicie el servidor con 'redis-server'. Nota: para poder hacer 'copy/paste' en la terminal (la de virtualbox no lo permite), haga una conexión ssh desde la máquina real hacia la virtual.
-
Como la aplicación aún no tiene la interfaz para crear nuevas partidas, se registrarán unas existentes directamente en Redis. Para esto revise en la documentación de REDIS el tipo de dato HASH, y la manera como se agregan tablas hash a una determianda llave. Con esto presente, inicie un cliente redis (redis-cli) en su máquina virtual, y usando los comandos respectivos, cree tres 'hash', uno para cada partida, identificados con la clave "game:1", "game:2", "game:3" (o con los identificadores que usted prefiera). A su vez, cada uno de los 'hash' anteriores debe tener:
- La palabra completa
- La palabra que está siendo descubierta (la que tiene un '_' en lugar de los caracteres no adivinados aún).
- Si el juego fue finalizado o no.
- El nombre del ganador.
-
Agregue las dependencias requeridas para usar Jedis, un cliente Java para REDIS:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency>
-
En la ruta src/main/resources agregue un archivo application.yml con la siguiente configuración (DEBE incluír las tabulaciones):
spring: redis: host: 127.0.0.1 port: 6379 #password:
-
Ahora, va a hacer la implementación del 'GameStateRedisCache' siguiendo el esquema planteado en el diagrama anterior, donde:
- Se creará la clase GameStateRedisCache.
- Se creará la clase HangmanRedisGame, que heredará y SOBREESCRIBIRÁ todos los métodos de la clase HangmanGame.
-
Haga que a la clase GameStateRedisCache se le inyecte el 'StringRedisTempalte':
@Autowired private StringRedisTemplate template;
-
Haga que la clase HangmanRedisGame tenga como atributos adicionales (a los heredados) el identificador de la partida y el 'template' de redis, e incluya un constructor que permita inicializarlos.
-
Redefina todos los métodos heredados de la clase HangmanRedisGame (tryWord, addLetter, etc), para que en lugar de usar los valores almacenados en los atributos, haga uso de los valores almacenados en en caché, teniendo en cuenta el identificador de la partida. Tenga en cuenta el API de Spring para el manejo del 'template' de Redis para Spring:
//consultar el valor en un hash String value=(String)template.opsForHash().get(key, property); //actualizar el valor en un hash template.opsForHash().put(key,property,value)
-
Una vez hecho lo anterior, haga que el método getGame() de GameStateRedisCache retorne una instancia de HangmanRedisGame, a la cual se le pase el identificador y el 'template' de Redis.
-
Ajuste las anotaciones para que la aplicación inyecte el esquema de Caché basado en Redis en lugar del basado sólo en memoria.
-
Verifique el funcionamiento de la aplicación (usando sólo una instancia de la misma).
-
Agruegue el manejo de excepciones donde haga falta, para que si se presentan errores al intentar acceder a una llave de Redis (por errores de conexión o porque la llave no existe), las mismas se escalen hasta el API, y se muestren en el cliente a manera de un mensaje de error legible.
-
Actualice la aplicación en el esquema de balanceo de carga, y rectifique nuevamente el funcionamiento. Para esto debe tener una instancia de Redis en una de las máquinas virtuales, y ambas instancias de la aplicación configuradas para hacer uso de la misma.
En caso de que con la configuración planteada (aplicación y REDIS corriendo en la máquina virtual) haya conflictos con SockJS:
Configurar REDIS para aceptar conexiones desde máquinas externas, editando el archivo /home/ubuntu/redis-stable/redis.conf, cambiando "bind 127.0.0.1" por "bind 0.0.0.0", y reiniciando el servidor con:redis-server /home/ubuntu/redis-stable/redis.conf.
Una vez hecho esto, en la aplicación ajustar el archivo jedis.properties, poner la IP de la máquina virtual (en lugar de 127.0.0.1), y ejecutarla desde el equipo real (en lugar del virtual). ** OJO: Esto sólo se hará como prueba de concepto!, siempre se le debe configurar la seguridad a REDIS antes de permitirle el acceso remoto!. **
Revise: dentro de la implementación hecha (en la clase RedisHangmanGame), se pueden presentar condiciones de carrera. Por qué y cual es la región crítica?.
Para resolver lo anterior, revise el ejemplo de la sección 9 de este artículo cómo implementar un esquema de bloqueo 'optimista' haciendo uso de WATCH y cómo crear transacciones con MULTI. Sin embargo, tenga presente que NO ES POSIBLE ejecutar la lógica hecha en Java (la que analiza qué si hay letras para destapar o no) dentro de la misma.
Para poder poner operaciones más avanzada dentro de una transacción, es posible agregar scripts en el lenguaje de programación LUA, y agregar la ejecución de los mismos dentro de una transacción (un 'MULTI') de Redis, tal como se muestra en la sección 4.11 de la documentación de REDIS.
Recuerde que si los elementos del lenguaje Lua son suficientes para realizar la actualización de la palabra, una alternativa es cambiar la representación de la información. Por ejemplo, guardando las palabras, caracter por caracter, en una lista LSET/LGET.
Para finalizar el ejercicio, se va a sustituír el 'bean' (repositorio) encargado de la persistencia de los usuarios del sistema, por uno que realice la misma en una base de datos NoSQL documental. Para esto, se usará el componente 'spring-data' de Spring de acuerdo con el siguiente diagrama:
Para hacer esto:
-
Regístrese y cree una base de datos MongoDB en mLab.
-
En la base de datos, cree una colección para usuarios, y en la misma registre uno o dos documentos compatibles con la entidad 'User' usada en la aplicación
Por ejemplo:
{ "_id": 112233, "name": "Maria Perez MongoDB", "photoUrl": "http://www.your3dsource.com/images/facepic1.jpeg" }
-
En su proyecto, agregue la dependencia starter-data-mongodb:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-mongodb</artifactId>
</dependency>
-
A la clase usuario agregue la anotación @Document (a nivel de clase) para indicar que la misma podrá hacerse persistente en una base de datos documental, agregando la propiedad 'collection' para indicar en qué colección de la base de datos se registrará su documento JSON correspondiente:
@Document(collection = "users") public class User {
Adicionalmente, agregue la anotación @Id a aquel atributo correspondiente al identificador del documento:
@Id private int id;
-
Para que en esta nueva versión de la aplicación en las dependencias a 'UserRepository' se inyecte una implementación de la persistencia usando MongoDB, haga que la interfaz herede de MongoRepository:
public interface UsersRepository extends MongoRepository<User, Integer>{ ... }
-
Remueva la anotación @Service del 'stub' de UserRepository usado en la versión anterior del ejercicio.
-
Agregue los parámetros de conexión a la base de datos mongodb Tenga en cuenta que el usuario y contraseña NO son los usados para autenticarse en mLabs, sino aquellos creados específicamente para la base de datos:
-
Parámetros (application.yml):
spring: data: mongodb: host: xxxxxx.mlab.com port: 37101 database: xxxxxx username: yyyyyy password: zzzzzz redis: host: 127.0.0.1 port: 6379
- La semántica de los componentes de Spring-Data para MyBatis depende de una convención de nombres tanto de los métodos como de sus parámetros. Teniendo en cuenta esto, cambie el nombre del método usado para buscar un usuario a partir de su identificador (y por ahora elimine los demás métodos, para evitar conflictos):
public User findById(Integer id);
- Ejecute la aplicación y verifique que la consulta de clientes se haga correctamente.
Suponga que los usuarios, en adelante, tendrán un puntaje asociado a sus juegos. Por ahora no se implementará el registro de los puntajes en sí, pero sí serán consultados.
-
Agregue al modelo una clase Score, que tenga: fecha de obtencuón puntaje, y valor de puntaje.
-
Haga la clase User tenga una colección de Score.
-
A través de la interfaz de mLab modifique los documentos existentes (usuarios) para que contengan la respectiva colección de puntajes. En las mismas agregue dos o trés puntajes en fechas diferentes.
-
Al Repositorio de Usuarios (UserRepository) agregue un método que permita consultar aquellos usuarios que tengan puntajes mayores que N. Para esto:
-
revise la sintaxis de $elemMatch, y pruebela en la terminal de consultas de mLab:
-
Revise la sección 6.3.2 de la documentación de Spring, para agregar operaciones a un repositorio a partir de consultas. Sobre la base de lo anterior, agregue un método al repositorio de usuarios.
Haga que en la vista:
- Se muestre el último puntaje obtenido por el usuario, una vez consultado.
- Se muestren los usuarios que han obtenido puntajes mayores a 100.
-
Asegúrese de que la copia del proyecto que entregará quede configurado para funcionar con el middleware de mensajería, el servidor MongoDB en la nube, y el servidor REDIS local.
-
En el directorio raíz de su proyecto, incluya un archivo INSTRUCCIONES.txt, indicando qué llaves se deben tener en el servidor REDIS para el correcto funcionamiento de la aplicación.
-
Comprima y suba el proyecto. No olvide que el ZIP debe incluír copia del repositorio (folder .git).