Go y Golang

Consulta manual con GORM

Otro post que escribo sobre GORM, un ORM de Go. En este caso vamos a ver cómo hacer una consulta manual y un escaneo manual, es decir, algo tipo raw. Algo así como una query personalizada.

Este tipo de consultas son poco comunes pero existen para cuando necesitamos hacer un reporte o traer ciertos datos que no corresponden a un único modelo.

Modelo que va a representar cada fila

En primer lugar comenzamos definiendo el modelo en donde vamos a escanear los valores. Yo voy a poner un ejemplo simple: un modelo que tiene una ganancia (es decir, un valor monetario) y un id del empleado que tiene esa ganancia.

type EmpleadoConGanancia struct {
 IdEmpleado int64   `json:"idEmpleado" gorm:"column:id"`
 Ganancia   float64 `json:"ganancia"`
}

Fíjate en que aunque la columna en el struct se llama IdEmpleado le estoy indicando a GORM que la columna que va a escanear sobre este valor será la que tenga el nombre de id.

Por cierto, cada nombre de propiedad del struct debe coincidir con el nombre de columna de la tabla. Por ejemplo, si en la tabla existe una columna llamada id_usuario la misma será tomada en cuenta si hay una propiedad llamada IdUsuario.

El arreglo en donde se colocan los valores

Luego creamos el arreglo en donde vamos a colocar los valores:

var empleados []EmpleadoConGanancia

Hacer consulta y obtener filas

Ahora hago mi consulta y obtengo las filas. La sintaxis es:

filas, err := bd.Raw("consulta", posibles_parámetros...).Rows()

Es importante comprobar que el error es nil y cerrar las filas con filas.Close (se puede usar defer). En mi caso:

rows, err := bd.Raw(`select sum(operaciones_de_cortes.precio * bultos_de_cortes.piezas) as ganancia,
      empleados.id
      from bihorarios
      inner join bultos_de_empleados
      on bihorarios.id_bulto_de_empleado = bultos_de_empleados.id
      inner join operaciones_de_cortes
      on operaciones_de_cortes.id = bultos_de_empleados.id_operacion_de_corte
      inner join empleados
      on bultos_de_empleados.id_empleado = empleados.id
      inner join bultos_de_cortes
      on bultos_de_cortes.id = bultos_de_empleados.id_bulto_de_corte
      where bultos_de_empleados.fecha_terminacion != ''
      and bultos_de_empleados.fecha_terminacion >= ?
      and bultos_de_empleados.fecha_terminacion <= ?
       group by empleados.id;`, fechaInicio, fechaFin).
 Rows()
if err != nil {
 return empleados, err
}
defer rows.Close()

Que no te confunda la consulta, solo te estoy colocando un ejemplo.

Recorrer y escanear

Finalmente, recorremos las filas, escaneamos y agregamos el valor al arreglo:

for rows.Next() {
  var e EmpleadoConGanancia
  bd.ScanRows(rows, &e)
  empleados = append(empleados, e)
}

Código completo

El código completo queda así; estoy omitiendo la parte en donde obtienes la conexión a la base de datos pues eso puede variar:

func consulta(fechaInicio, fechaFin string) ([]EmpleadoConGanancia, error) {
 var empleados []EmpleadoConGanancia
 bd, err := obtenerBd()
 if err != nil {
  return empleados, err
 }
 defer bd.Close()

 rows, err := bd.Raw(`select sum(operaciones_de_cortes.precio * bultos_de_cortes.piezas) as ganancia,
       empleados.id
       from bihorarios
       inner join bultos_de_empleados
       on bihorarios.id_bulto_de_empleado = bultos_de_empleados.id
       inner join operaciones_de_cortes
       on operaciones_de_cortes.id = bultos_de_empleados.id_operacion_de_corte
       inner join empleados
       on bultos_de_empleados.id_empleado = empleados.id
       inner join bultos_de_cortes
       on bultos_de_cortes.id = bultos_de_empleados.id_bulto_de_corte
       where bultos_de_empleados.fecha_terminacion != ''
       and bultos_de_empleados.fecha_terminacion >= ?
       and bultos_de_empleados.fecha_terminacion <= ?
        group by empleados.id;`, fechaInicio, fechaFin).
  Rows()
 if err != nil {
  return empleados, err
 }
 defer rows.Close()
 for rows.Next() {
  var e EmpleadoConGanancia
  bd.ScanRows(rows, &e)
  empleados = append(empleados, e)
 }
 return empleados, nil
}

Otro ejemplo completo

Aquí tengo otro ejemplo más simple, en lenguaje Go usando Gorm:

type MateriaConConteoDeTarea struct {
 Conteo    int64  `json:"conteo"`
 Materia   string `json:"materia"`
 IdMateria string `json:"idMateria"`
}


func obtenerMateriasConConteoDeTareas(idPeriodo int64) ([]MateriaConConteoDeTarea, error) {
 materias := []MateriaConConteoDeTarea{}
 bd, err := obtenerBd()
 if err != nil {
  return materias, err
 }
 defer bd.Close()
 rows, err := bd.Raw(`select count(*) AS conteo,
materia.nombre AS materia, materia.id AS id_materia from tareas
inner join materia on materia.id = tareas.id_materia
where id_periodo = ? group by materia.id;`, idPeriodo).
  Rows()
 if err != nil {
  return materias, err
 }
 defer rows.Close()
 for rows.Next() {
  var m MateriaConConteoDeTarea
  bd.ScanRows(rows, &m)
  if err != nil {
   return materias, err
  }
  materias = append(materias, m)
 }
 return materias, bd.Error
}

Fíjate que en mi consulta SQL tengo las columnas llamadas conteo, materia e id_materia. En mi struct corresponden, en el mismo orden, a Conteo, Materia e IdMateria.

Estoy aquí para ayudarte 🤝💻


Estoy aquí para ayudarte en todo lo que necesites. Si requieres alguna modificación en lo presentado en este post, deseas asistencia con tu tarea, proyecto o precisas desarrollar un software a medida, no dudes en contactarme. Estoy comprometido a brindarte el apoyo necesario para que logres tus objetivos. Mi correo es parzibyte(arroba)gmail.com, estoy como@parzibyte en Telegram o en mi página de contacto

No te pierdas ninguno de mis posts 🚀🔔

Suscríbete a mi canal de Telegram para recibir una notificación cuando escriba un nuevo tutorial de programación.
parzibyte

Programador freelancer listo para trabajar contigo. Aplicaciones web, móviles y de escritorio. PHP, Java, Go, Python, JavaScript, Kotlin y más :) https://parzibyte.me/blog/software-creado-por-parzibyte/

Entradas recientes

Servidor HTTP en Android con Flutter

El día de hoy te mostraré cómo crear un servidor HTTP (servidor web) en Android…

3 días hace

Imprimir automáticamente todos los PDF de una carpeta

En este post te voy a enseñar a designar una carpeta para imprimir todos los…

4 días hace

Guía para imprimir en plugin versión 1 desde Android

En este artículo te voy a enseñar la guía para imprimir en una impresora térmica…

1 semana hace

Añadir tasa de cambio en sistema de información

Hoy te voy a mostrar un ejemplo de programación para agregar un módulo de tasa…

2 semanas hace

Comprobar validez de licencia de plugin ESC POS

Los usuarios del plugin para impresoras térmicas pueden contratar licencias, y en ocasiones me han…

2 semanas hace

Imprimir euro € en impresora térmica

Hoy voy a enseñarte cómo imprimir el € en una impresora térmica. Vamos a ver…

3 semanas hace

Esta web usa cookies.