Vamos a agregar una clase que se encargue del acceso a datos. Debido a que la aplicación es sencilla y solo utilizaremos como acceso a la base de datos el Entity Framework pasaremos directamente la clase CaducaContext.
Si por ejemplo tu aplicación es más grande y en un futuro puedes utilizar ADO .NET para el acceso a la base de datos o alguno de las alternativas ORM como repoDB, Dapper o Stored Procedures lo correcto sería crear una interfaz. Por lo general se llama IRepository la cual tiene los métodos para agregar, modificar, borrar o consultar registros de la base de datos. Como es una interfaz solo define los métodos y no la clase que se encarga de la conexión a la base de datos. En este ejemplo se conectará mediante Entity Framework y ADO .net
//Interfaz con los métodos comunes como agregar, buscarpublicinterfaceICategoriaRepository{Task<Categoria> ObtenerCategoriaAsync(int id); voidAgregar(Categoria categoria);voidBorrar(Categoria categoria);}//Puedes crear una clase que se conecte a la base de datos// por medio de entity framework corepublicclassCategoriaEFRepository:ICategoriaRepository{privatereadonlyCaducaContext contexto;publicCategoriaRepository (CaducaContext contexto) {this.context= context; }publicasyncTask<Categoria> ObtenerPorIdAsync(int id) {returnawaitcontexto.Categoria.FindAsync(id); } //Los demás métodos}//Puedes tener otra clase que se conecte a la base de datos// con ADO .NETpublicclassCategoriaADORepository:ICategoriaRepository{privateSqlConnection connection;privatestring connectionString;publicCategoriaRepository (SqlConnection connection,string connectionString) {this.connection= connection;this.connectionString= connectionString; }publicasyncTask<Categoria> ObtenerPorIdAsync(int id) {string queryString ="SELECT Id, Clave, Nombre FROM "+"cateogoria WHERE Id> @id ";Categoria categoria =newCategoria();using (SqlConnection connection =newSqlConnection(connectionString)) { // Create the Command and Parameter objects.SqlCommand command =newSqlCommand(queryString, connection);command.Parameters.AddWithValue("@id", id);connection.Open();SqlDataReader reader =awaitcommand.ExecuteReaderAsync();categoria.Id= (int)reader[0];categoria.Clave=reader[1].toString();categoria.Nombre=reader[2].toString();reader.Close(); } return categoria; } //Los demás métodos}publicclassCategoriasController:ControllerBase{privateICategoriaRepository categoriaRepository; //El constructor del controller recibe como parámetro la // interfaz en el cual podemos pasarle un objeto de // cualquiera de nuestras 2 clasespublicCategoriasController(ICategoriaRepository categoriaRepository) {this.categoriaRepository= categoriaRepository; }}
Vamos a crear una clase para conectarnos a la base de datos que reciba como parámetro el objeto CaducaContext el cual podemos cambiarlo para que sea Mysql, SQL Server, SQL Lite o en memoria para pruebas unitarias.
Vamos a crear una nueva carpeta llamada DAO (Data Access Object), en esta carpeta
Agregaremos una clase por cada tabla
Se encargará de:
Agregar un nuevo registro
Modificar
Borrar
Obtener un registro
Obtener todos los registros
Dentro de la carpeta DAO crea una nueva clase llamada CategoriaDAO
Agrega como variable privada el contexto (CaducaContext) y en el constructor de la clase recibe como parámetro el contexto.
CategoriaDAO.cs
publicclassCategoriaDAO{privatereadonlyCaducaContext contexto; /// <summary> /// Clase para acceso a la base de datos /// </summary> /// <paramname="context"></param>publicCategoriaDAO(CaducaContext context) {this.contexto= context; }}
Método para obtener todos los registros
Agrega un método ObtenerTodo a tu clase CategoridaDAO el cual va a obtener todas las categorías. Copia el método de la clase CategoriasController.cs del método Get, recuerda presionar CTRL + . para agregar los import necesarios. Aquí regreso directamente un objeto List para poder facilitar el Debug de tu código.
CategoriaDAO.cs
/// <summary>/// Obtiene todas las categorias/// </summary>/// <returns></returns>publicasyncTask<List<Categoria>> ObtenerTodoAsync(){returnawaitcontexto.Categoria.ToListAsync();}
Cambia tu servicio para inicializar el objeto en el constructor del servicio
CategoriasController.cs
usingSystem.Collections.Generic;publicclassCategoriasController:ControllerBase{privatereadonlyCaducaContext _context; //Agrega el ojeto categoriaDAOprivateCategoriaDAO categoriaDAO;publicCategoriasController(CaducaContext context) { _context = context; //Inicializa categoriaDAO con el contexto recibido // como parámetro categoriaDAO =newCategoriaDAO(context); } [HttpGet]publicasyncTask<List<Categoria>> GetCategoriaAsync() { //Cambia el método get para utilizar el // objeto categoriaDAOreturnawaitcategoriaDAO.ObtenerTodoAsync(); }
Método para buscar un registro por Id
En tu clase CategoriaDAO agrega un método para obtener una categoría por medio de su Id. Puedes copiar el método de tu controller a la función. Con Entitty Framework todas las tablas contienen un método Find y FindAsync por si deseas que sea síncrono o asíncrono en el cual busca un registro por medio del campo definido como llave primaria.
Cambia tu controller para que obtenga la categoría de la nueva clase. Aquí borre el código autogenerado para validar el modelo ya que no hay modelo a validar y la validación del modelo se realizará de forma general más adelante.
Crea una nueva carpeta llamada Core el cual tendrá clases con funciones genéricas. La clase se llama CustomError. Aquí vamos a regresar los mensajes de error de acuerdo al estándar de los servicios rest el cual incluye el StatusCode, el mensaje y adicionalmente el nombre del campo del error.
namespaceCaducaRest.Core{ /// <summary> /// Mensjaes de error /// </summary>publicclassCustomError { /// <summary> /// Código de error /// </summary>publicint StatusCode; /// <summary> /// Mensaje del error /// </summary>publicstring Message; /// <summary> /// Campo con el error /// </summary>publicstring Field; /// <summary> /// Constructor de la clase /// </summary> /// <paramname="statusCode">Código de erorr</param> /// <paramname="message">Mensaje que explica el error</param> /// <paramname="field">Campo que tiene el error</param>publicCustomError(int statusCode,string message,string field) { StatusCode = statusCode; Message = message; Field = field; } }}
Método para agregar una nueva categoría
En la clase CategoriaDAO vamos a crear un nuevo método que se llame AgregarAsync el cual recibe como parámetro un objeto Categoria. En este método vamos a agregar código para verificar que no guarden categorías con claves o nombres duplicados. Agregamos una variable llamada customError a nuestra clase CategoriaDAO.cs
CategoriaDAO.cs
publicclassCategoriaDAO{publicCustomError customError;publicasyncTask<bool> AgregarAsync(Categoria categoria) {Categoria registroRepetido; registroRepetido =contexto.Categoria .FirstOrDefault(c =>c.Nombre==categoria.Nombre);if (registroRepetido !=null) { customError =newCustomError(400,"Ya existe una categoría con este nombre, "+"por favor teclea un nombre diferente","Nombre");returnfalse; } registroRepetido =contexto.Categoria .FirstOrDefault(c =>c.Clave==categoria.Clave);if (registroRepetido !=null) { customError =newCustomError(400,"Ya existe una categoría con esta clave, "+"por favor teclea una clave diferente","Nombre");returnfalse; }contexto.Categoria.Add(categoria);awaitcontexto.SaveChangesAsync();returntrue;}
Cambiamos el controlador para utilizar la función que acabamos de crear categoriaDAO.
La función StatusCode nos permite regresar los mensajes de error de acuerdo al estándar REST, el primer parámetro es el código de error y el segundo es un objeto, puedes regresar un arreglo con todos los errores encontrados o un string con el mensaje de error.
CategoriasController.cs
[HttpPost]publicasyncTask<IActionResult> PostCategoria( [FromBody] Categoria categoria){if (!ModelState.IsValid)return BadRequest(ModelState); //Cambiamos el código para agregar aquí la clase. //Si no fue correcto regresamos el mensaje de error devuelto if (!awaitcategoriaDAO.AgregarAsync(categoria)) { return StatusCode(categoriaDAO.customError.StatusCode,categoriaDAO.customError.Message); }return CreatedAtAction("GetCategoria",new { id =categoria.Id }, categoria); }
Método para modificar una categoría
Vamos a agregar el método para modificar una categoría
CategoriaDAO.cs
/// <summary>/// Modidica una categoria/// </summary>/// <paramname="categoria">Datos de la categoria</param>/// <returns></returns>publicasyncTask<bool> ModificarAsync(Categoria categoria){Categoria registroRepetido;try { //Se busca si existe una categoria con el mismo nombre //pero diferente Id registroRepetido =contexto.Categoria .FirstOrDefault(c =>c.Nombre==categoria.Nombre&&c.Id!=categoria.Id);if (registroRepetido !=null) { customError =newCustomError(400,"Ya existe una categoría con este nombre, "+"por favor teclea un nombre diferente","Nombre");returnfalse; } registroRepetido =contexto.Categoria .FirstOrDefault(c =>c.Clave==categoria.Clave&&c.Id!=categoria.Id);if (registroRepetido !=null) { customError =newCustomError(400,"Ya existe una categoría con esta clave, "+"por favor teclea una clave diferente","Nombre");returnfalse; }contexto.Entry(categoria).State=EntityState.Modified;awaitcontexto.SaveChangesAsync(); }catch (DbUpdateConcurrencyException) {if (!ExisteCategoria(categoria.Id)) { customError =newCustomError(400,"La categoría"+"ya no existe","Categoría");returnfalse; } }returntrue;}privateboolExisteCategoria(int id){returncontexto.Categoria.Any(e =>e.Id== id);}
En tu clase controller cambiamos el código para que utilice nuestra clase
CategoriaController.cs
[HttpPut("{id}")]publicasyncTask<IActionResult> PutCategoria([FromRoute] int id, [FromBody] Categoria categoria){if (!ModelState.IsValid)return BadRequest(ModelState);if (id !=categoria.Id)return BadRequest();if (!awaitcategoriaDAO.ModificarAsync(categoria)) {return StatusCode(categoriaDAO.customError.StatusCode,categoriaDAO.customError.Message); }return NoContent();}
Método para borrar una categoría
Por último vamos a separar el método para borrar una categoría. Utilizamos el método para encontrar la categoría que desea borrar el usuario y si no existe mandamos un mensaje de error.
CategoriaDAO.cs
/// <summary>/// Permite borrar una categoría por Id/// </summary>/// <paramname="id">Id de la categoría</param>/// <returns></returns>publicasyncTask<bool> BorraAsync(int id){var categoria =await ObtenerPorIdAsync(id);if (categoria ==null) { customError =newCustomError(400,"La categoría que deseas borrar ya no existe, "+"probablemente fue borrada por otro usuario","Id");returnfalse; }contexto.Categoria.Remove(categoria);awaitcontexto.SaveChangesAsync();returntrue;}
En el método creado por visual studio se regresa el objeto que fue borrado, en mi caso como no se desea realizar ninguna acción con el registro borrado solo regresare un Ok.
CategoriasController.cs
[HttpDelete("{id}")]publicasyncTask<IActionResult> DeleteCategoria([FromRoute] int id){if (!awaitcategoriaDAO.BorraAsync(id)) {return StatusCode(categoriaDAO.customError.StatusCode,categoriaDAO.customError.Message); }return Ok();}
Al cambiar el acceso a la base de datos a una clase aparte, nuestro código tiene funciones con pocas líneas de código que realizan una única función.