Consultas de filtro y bot.on()

El primer argumento de bot.on() es una cadena llamada filter query.

Introducción

La mayoría (¿todos?) de los otros frameworks de bots permiten realizar una forma primitiva de filtrado para las actualizaciones, por ejemplo, sólo on("message") y similares. El resto del filtrado de los mensajes se deja al desarrollador, lo que a menudo conduce a interminables declaraciones if en su código.

Por el contrario, grammY incluye su propio lenguaje de consulta que puedes utilizar para filtrar exactamente los mensajes que quieras.

Esto permite utilizar más de 500 filtros diferentes, y es posible que añadamos más con el tiempo. Todos los filtros válidos se pueden autocompletar en el editor de código. Por lo tanto, puedes simplemente escribir bot.on(""), abrir el autocompletado, y buscar entre todas las consultas escribiendo algo

Filtro de búsqueda de consultas

La inferencia de tipos de bot.on() comprenderá la consulta de filtro que hayas elegido. Por lo tanto, ajusta algunos tipos en el contexto que se sabe que existen.

bot.on("message", (ctx) => {
  // Podría ser undefined si el mensaje recibido no tiene texto.
  const text: string | undefined = ctx.msg.text;
});
bot.on("message:text", (ctx) => {
  // El texto siempre está definido porque este manejador es llamado cuando se recibe un mensaje de texto.
  const text: string = ctx.msg.text;
});

En cierto sentido, grammY implementa las consultas de filtro tanto en tiempo de ejecución, como en el nivel de tipo.

Ejemplos de consulta

Aquí hay algunos ejemplos de consultas:

Consultas regulares

Filtros simples para actualizaciones, y sub-filtros:

bot.on("mensaje"); // se llama cuando se recibe cualquier mensaje
bot.on("mensaje:texto"); // sólo mensajes de texto
bot.on("message:photo"); // sólo mensajes de foto

Filtro para entidades

Subfiltros que van un nivel más allá:

bot.on("message:entities:url"); // mensajes que contienen una URL
bot.on("message:entities:code"); // mensajes que contienen un fragmento de código
bot.on("edited_message:entities"); // mensajes editados con cualquier tipo de entidades

Omitir valores

Puedes omitir algunos valores en las consultas del filtro. Entonces grammY buscará entre diferentes valores para que coincidan con tu consulta.

bot.on(":text"); // cualquier mensaje de texto y cualquier mensaje de texto de los canales
bot.on("::url"); // mensajes con URL en el texto o en el pie de foto (fotos, etc)
bot.on("::email"); // mensajes o publicaciones de canales con email en texto o pie de foto

Si se omite el valor first, se pueden obtener tanto los mensajes como los mensajes del canal. Recuerde que ctx.msg le da acceso tanto a los mensajes como a las publicaciones del canal, lo que coincida con la consulta.

Si se omite el valor segundo, se obtiene acceso tanto a las entidades como a los subtítulos. Puede omitir tanto la primera como la segunda parte al mismo tiempo.

Accesos directos

El motor de consulta de grammY permite definir accesos directos que agrupan consultas relacionadas.

msg

El atajo msg agrupa los nuevos mensajes y los nuevos mensajes del canal. En otras palabras, usar msg es equivalente a escuchar los eventos "message" y "channel_post".

bot.on("msg"); // cualquier mensaje o mensaje de canal
bot.on("msg:text"); // exactamente lo mismo que `:text`.

edit

Este atajo de teclado edit agrupa los mensajes editados y los mensajes editados del canal.

bot.on("edit"); // cualquier edición de mensajes o publicaciones del canal
bot.on("edit:text"); // ediciones de mensajes de texto
bot.on("edit::url"); // ediciones de mensajes o entradas de canal con URL
bot.on("edit:location"); // actualización de la ubicación en vivo

:media

El atajo :media agrupa los mensajes de foto y vídeo.

bot.on("message:media"); // mensajes de foto y video
bot.on("edited_channel_post:media"); // mensajes editados del canal con medios
bot.on(":media"); // mensajes multimedia o publicaciones del canal

:archivo

El atajo :file agrupa todos los mensajes que contienen un archivo. Por lo tanto, puedes estar seguro de que await ctx.getFile() te dará un objeto archivo.

bot.on(":file"); // archivos en mensajes o mensajes del canal
bot.on("edit:file"); // ediciones de mensajes de archivos o mensajes de canales de archivos

Consejos útiles

Puedes detectar bots en las consultas con la parte de consulta :is_bot. El azúcar sintáctico :me se puede utilizar para referirse a tu bot desde una consulta, que comparará los identificadores de usuario por ti.

bot.on("message:new_chat_members:is_bot"); // un bot se ha unido al chat
bot.on("message:left_chat_member:me"); // tu bot abandonó el chat (fue eliminado)

Filtrar por propiedades del usuario

Si quieres filtrar por otras propiedades de un usuario, necesitas realizar una petición adicional, por ejemplo await ctx.getAuthor() para el autor del mensaje. Las consultas de filtrado no realizarán secretamente otras peticiones a la API por ti. Sigue siendo sencillo realizar este tipo de filtrado:

bot.on("message").filter(
  async (ctx) => {
    const user = await ctx.getAuthor();
    return user.status === "creator" || user.status === "administrator";
  },
  (ctx) => {
    // Maneja mensajes de creadores y administradores.
  },
);

Combinación de varias consultas

Puede combinar cualquier número de consultas de filtro con las operaciones AND y OR.

Combinar con OR

Si quiere instalar alguna pieza de middleware detrás de la concatenación OR de dos consultas, puede pasar ambas a bot.on() en un array.

// Se ejecuta si la actualización es sobre un mensaje O una edición de un mensaje
bot.on(["message", "edited_message"] /* , ... */);
// Se ejecuta si se encuentra un hashtag O un correo electrónico O una entidad de mención en el texto o en el pie de foto
bot.on(["::hashtag", "::email", "::mention"] /* , ... */);

El middleware se ejecutará si cualquiera de las consultas proporcionadas coincide. El orden de las consultas no importa.

Combinar con AND

Si quieres instalar alguna pieza de middleware detrás de la concatenación AND de dos consultas, puedes encadenar las llamadas a bot.on().

// Coincide con las URLs reenviadas
bot.on("::url").on(":forward_date" /* , ... */);
// Coincide con las fotos que contienen un hashtag en el pie de foto
bot.on(":photo").on("::hashtag" /* , ... */);

El middleware se ejecutará si todas las consultas proporcionadas coinciden. El orden de las consultas no importa.

Construcción de consultas complejas

Es técnicamente posible combinar consultas de filtro a fórmulas más complicadas si están en CNFopen in new window, aunque es poco probable que esto sea útil.

bot
  // Coincide con todas las publicaciones del canal o los mensajes reenviados ...
  .on(["channel_post", ":forward_date"])
  // ... que contengan texto ...
  .on(":text")
  // ... con al menos una URL, un hashtag o un cashtag.
  .on(["::url", "::hashtag", "::cashtag"] /* , ... */);

La inferencia de tipo de ctx recorrerá toda la cadena de llamadas e inspeccionará cada elemento de las tres llamadas a .on. Como ejemplo, puede detectar que ctx.msg.text es una propiedad necesaria para el fragmento de código anterior.

Consejos útiles

Aquí hay algunas características menos conocidas de las consultas de filtro que pueden ser útiles. Algunas de ellas son un poco avanzadas, así que no dudes en pasar a la siguiente sección.

Actualizaciones de los miembros del chat

Puedes utilizar la siguiente consulta de filtro para recibir actualizaciones de estado sobre tu bot.

bot.on("mi_miembro_del_chat"); // iniciar, detener, unirse o abandonar

En los chats privados, esto se activa cuando el bot se inicia o se detiene. En los grupos, esto se dispara cuando el bot es añadido o eliminado. Ahora puedes inspeccionar ctx.myChatMember para saber qué ha pasado exactamente.

Esto no debe confundirse con

bot.on("chat_member");

que se puede utilizar para detectar los cambios de estado de otros miembros del chat, como cuando la gente se une, es promovida, etc.

Ten en cuenta que las actualizaciones de chat_member deben ser habilitadas explícitamente especificando allowed_updates al iniciar tu bot.

Combinación de consultas con otros métodos

Puedes combinar consultas de filtro con otros métodos de la clase Composer (Referencia de la APIopen in new window), como command o filter. Esto permite crear potentes patrones de manejo de mensajes.

bot.on(":forward_date").command("help"); // comandos /help reenviados

// Solo maneja comandos en chats privados.
const pm = bot.filter((ctx) => ctx.chat?.type === "private");
pm.command("start");
pm.command("help");

Filtrado por tipo de remitente del mensaje

Hay cinco tipos diferentes de autores de mensajes en Telegram:

  1. Autores de mensajes del canal
  2. Reenvíos automáticos desde canales vinculados en grupos de discusión
  3. Cuentas de usuarios normales, esto incluye a los bots (es decir, mensajes “normales”)
  4. Administradores que envían en nombre del grupo (administradores anónimosopen in new window)
  5. Usuarios que envían mensajes como uno de sus canales

Puedes combinar las consultas de filtro con otros mecanismos de gestión de actualizaciones para averiguar el tipo de autor del mensaje.

// Mensajes del canal enviados por `ctx.senderChat`
bot.on("channel_post");

// Reenvío automático desde el canal `ctx.senderChat`:
bot.on("message:is_automatic_forward");
// Mensajes regulares enviados por `ctx.from`
bot.on("mensaje").filter((ctx) => ctx.senderChat === undefined);
// Administrador anónimo en `ctx.chat`
bot.on("message").filter((ctx) => ctx.senderChat?.id === ctx.chat.id);
// Usuarios que envían mensajes en nombre de su canal `ctx.senderChat`
bot.on("message").filter((ctx) =>
  ctx.senderChat !== undefined && ctx.senderChat.id !== ctx.chat.id
);

Filtrado por propiedades del usuario

Si quieres filtrar por otras propiedades de un usuario, tienes que realizar una petición adicional, por ejemplo await ctx.getAuthor() para el autor del mensaje. Las consultas de filtrado no realizarán secretamente otras peticiones a la API por ti. Sigue siendo sencillo realizar este tipo de filtrado:

bot.on("mensaje").filter(
  async (ctx) => {
    const user = await ctx.getAuthor();
    return user.status === "creador" | user.status === "administrador";
  },
  (ctx) => {
    // Maneja los mensajes de los creadores y administradores.
  },
);

Reutilización de la lógica de consulta del filtro

Internamente, bot.on se basa en una función llamada matchFilter. Toma una consulta de filtro y la compila en una función de predicado. El predicado se pasa simplemente a bot.filter para filtrar las actualizaciones.

Puedes importar matchFilter directamente si quieres usarlo en tu propia lógica. Por ejemplo, puedes decidir eliminar todas las actualizaciones que coincidan con una determinada consulta:

// Deja caer todos los mensajes de texto o las publicaciones del canal de texto.
bot.drop(matchFilter(":text"));

Análogamente, puedes hacer uso de los tipos de consulta de filtro que grammY utiliza internamente:

Reutilización de los tipos de consulta de filtro

Internamente, matchFilter utiliza los [type predicates] de TypeScript (https://www.typescriptlang.org/docs/handbook/2/narrowing.html#using-type-predicates) para acotar el tipo de ctx. Toma un tipo C extends Context y un Q extends FilterQuery y produce ctx is Filter<C, Q>. En otras palabras, el tipo Filter es lo que realmente recibes para tu ctx en el middleware.

Puedes importar Filter directamente si quieres utilizarlo en tu propia lógica. Por ejemplo, puedes decidir definir una función manejadora que maneje objetos de contexto específicos que fueron filtrados por una consulta de filtro:

function handler(ctx: Filter<Context, ":text">) {
  // maneja el objeto de contexto filtrado
}

bot.on(":text", handler);

Consulta las referencias de la API para matchFilteropen in new window, Filteropen in new window, y FilterQueryopen in new window para seguir leyendo.

El lenguaje de consulta

Esta sección está pensada para los usuarios que quieran tener un conocimiento más profundo de las consultas de filtrado en grammY, pero no contiene ningún conocimiento necesario para crear un bot.

Estructura de la consulta

Cada consulta consta de un máximo de tres partes de consulta. Dependiendo del número de partes de consulta que tenga una consulta, diferenciamos entre consultas L1, L2 y L3, como "message", "message:entities" y "message:entities:url", respectivamente.

Las partes de la consulta están separadas por dos puntos (:). Nos referimos a la parte hasta los primeros dos puntos o el final de la cadena de consulta como la L1 parte de una consulta. La parte desde los primeros dos puntos hasta los segundos o hasta el final de la cadena de consulta se denomina parte L2 de la consulta. La parte que va desde los segundos dos puntos hasta el final de la cadena de consulta se denomina parte L3 de la consulta.

Ejemplo:

Filter QueryL1 partL2 partL3 part
"message""message"undefinedundefined
"message:entities""message""entities"undefined
"message:entities:mention""message""entities""mention"

Validación de consultas

Aunque el sistema de tipos debería detectar todas las consultas de filtro inválidas en tiempo de compilación, grammY también comprueba todas las consultas de filtro pasadas en tiempo de ejecución durante la configuración. Cada consulta de filtro pasada se compara con una estructura de validación que comprueba si es válida. No sólo es bueno fallar inmediatamente durante la configuración en lugar de en tiempo de ejecución, también ha sucedido antes que los errores en TypeScript causen serios problemas con el sofisticado sistema de inferencia de tipos que potencia las consultas de filtro. Si esto vuelve a ocurrir en el futuro, esto evitará problemas que podrían ocurrir de otra manera. En este caso, se le proporcionarán mensajes de error útiles.

Rendimiento

grammY puede comprobar cada consulta de filtro en tiempo constante (amortizado) por actualización, independientemente de la estructura de la consulta o de la actualización entrante.

La validación de las consultas de filtrado ocurre sólo una vez, cuando se inicializa el bot y se llama a bot.on().

Al iniciarse, grammY deriva una función de predicado de la consulta de filtro dividiéndola en sus partes de consulta. Cada parte se asigna a una función que realiza una única comprobación in, o dos comprobaciones si la parte se omite y hay que comprobar dos valores. Estas funciones se combinan entonces para formar un predicado que sólo tiene que comprobar tantos valores como sean relevantes para la consulta, sin iterar sobre las claves del objeto Update.

Este sistema utiliza menos operaciones que algunas bibliotecas de la competencia, que necesitan realizar comprobaciones de contención en arrays cuando se enrutan las actualizaciones. El sistema de consulta de filtros de grammY es mucho más potente.