Context

El objeto Context (Referencia de la API de grammYopen in new window) es una parte importante de grammY.

Siempre que registres un oyente en tu objeto bot, este oyente recibirá un objeto Context.

bot.on("message", (ctx) => {
  // `ctx` es el objeto `Context`.
});

Puedes utilizar el objeto de contexto para:

Tenga en cuenta que los objetos Context se llaman comúnmente ctx.

Información disponible

Cuando un usuario envía un mensaje a tu bot, puedes acceder a él a través de ctx.message. Como ejemplo, para obtener el texto del mensaje, puedes hacer esto:

bot.on("message", (ctx) => {
  // `ctx` será un `string` cuando se procesen mensajes de texto.
  // Será `undefined` si el mensaje recibido no tiene ningún mensaje de texto,
  // por ejemplo, fotos, pegatinas y otros mensajes.
  const txt = ctx.message.text;
});

Del mismo modo, puedes acceder a otras propiedades del objeto mensaje, por ejemplo ctx.message.chat para obtener información sobre el chat donde se envió el mensaje. Revisa la parte sobre Mensajes en la Referencia de la API de Telegram Botopen in new window para ver qué datos están disponibles. Alternativamente, puedes usar simplemente el autocompletado en tu editor de código para ver las posibles opciones.

Si registras tu listener para otros tipos, ctx también te dará información sobre ellos. Ejemplo:

bot.on("edited_message", (ctx) => {
  // Obtenga el nuevo texto editado del mensaje.
  const editedText = ctx.editedMessage.text;
});

Además, puedes obtener acceso al objeto crudo Update (Referencia de la API de Telegram Botopen in new window) que Telegram envía a tu bot. Este objeto de actualización (ctx.update) contiene todos los datos que las fuentes ctx.message y similares.

El objeto context siempre contiene información sobre tu bot, accesible a través de ctx.me.

Atajos

Hay una serie de accesos directos instalados en el objeto de contexto.

AtajoDescripción
ctx.msgObtiene el objeto mensaje, también los editados
ctx.chatObtiene el objeto chat
ctx.senderChatObtiene el objeto de chat del remitente de ctx.msg (para mensajes anónimos de canal/grupo)
ctx.fromObtiene el autor del mensaje, la consulta de devolución de llamada, u otras cosas
ctx.inlineMessageIdObtiene el identificador del mensaje en línea para las consultas de devolución de llamada o los resultados elegidos en línea

En otras palabras, también puedes hacer esto:

bot.on("message", (ctx) => {
  // Obtenga el texto del mensaje.
  const text = ctx.msg.text;
});

bot.on("edited_message", (ctx) => {
  // Obtener el nuevo texto editado del mensaje.
  const editedText = ctx.msg.text;
});

Por lo tanto, si lo desea, puede olvidarse de ctx.message y ctx.channelPost y ctx.editedMessage y así sucesivamente, y sólo utilizar siempre ctx.msg en su lugar.

Acciones disponibles

Si quieres responder a un mensaje de un usuario, puedes escribir esto:

bot.on("message", async (ctx) => {
  // Obtener el identificador del chat.
  const chatId = ctx.msg.chat.id;
  // El texto a responder con
  const text = "¡Recibí tu mensaje!";
  // Enviar la respuesta.
  await bot.api.sendMessage(chatId, text);
});

Puedes notar dos cosas que no son óptimas en esto:

  1. Debemos tener acceso al objeto bot. Esto significa que tenemos que pasar el objeto bot por toda nuestra base de código para responder, lo cual es molesto cuando tienes más de un archivo fuente y defines tu listener en otro lugar.
  2. Tenemos que sacar el identificador de chat del contexto, y pasarlo explícitamente a sendMessage de nuevo. Esto también es molesto, porque lo más probable es que siempre quieras responder al mismo usuario que envió un mensaje. ¡Imagina cuántas veces escribirías lo mismo una y otra vez!

En cuanto al punto 1., el objeto contexto simplemente te proporciona acceso al mismo objeto API que encuentras en bot.api, se llama ctx.api. Ahora puedes escribir ctx.api.sendMessage en su lugar y ya no tienes que pasar tu objeto bot. Fácil.

Sin embargo, la verdadera fuerza es arreglar el punto 2. El objeto context te permite simplemente enviar una respuesta así:

Traducción realizada con la versión gratuita del traductor www.DeepL.com/Translator

bot.on("message", async (ctx) => {
  await ctx.reply("¡Recibí tu mensaje!");
});

// O, incluso más corto:
bot.on("message", (ctx) => ctx.reply("¡Te tengo!"));

¡Genial! 🎉

Bajo el capó, el contexto ya conoce su identificador de chat (es decir, ctx.msg.chat.id), por lo que te da el método reply para simplemente enviar un mensaje de vuelta al mismo chat. Internamente, reply vuelve a llamar a sendMessage con el identificador del chat pre-rellenado para ti.

Función de respuesta de Telegram

Aunque el método se llama ctx.reply en grammY (y en muchos otros frameworks), no utiliza la función reply de Telegramopen in new window donde se vincula un mensaje anterior.

Si buscas lo que puede hacer sendMessage en el Referencia Bot API de Telegramopen in new window, verás un número de opciones, como parse_mode, disable_web_page_preview, y reply_to_message_id. Esta última puede utilizarse para convertir un mensaje en una respuesta:

await ctx.reply("^ ¡Esto es un mensaje!", {
  reply_to_message_id: ctx.msg.message_id,
});

El mismo objeto de opciones se puede pasar a bot.api.sendMessage y ctx.api.sendMessage. Utiliza el autocompletado para ver las opciones disponibles directamente en tu editor de código.

Naturalmente, todos los demás métodos de ctx.api tienen un acceso directo con los valores correctos precompletados, como ctx.replyWithPhoto para responder con una foto, o ctx.exportChatInviteLink para obtener un enlace de invitación para el chat correspondiente. Si quieres tener una visión general de los accesos directos que existen, el autocompletado es tu amigo, junto con la Referencia de la API de grammYopen in new window.

Ten en cuenta que puede que no quieras reaccionar siempre en el mismo chat. En este caso, puedes volver a utilizar los métodos ctx.api, y especificar todas las opciones al llamarlos. Por ejemplo, si recibes un mensaje de Alice y quieres reaccionar enviando un mensaje a Bob, entonces no puedes usar ctx.reply porque siempre enviará mensajes al chat con Alice. En su lugar, llama a ctx.api.sendMessage y especifica el identificador del chat de Bob.

Cómo se crean los objetos de contexto

Cada vez que tu bot recibe un nuevo mensaje de Telegram, se envuelve en un objeto de actualización. De hecho, los objetos de actualización no sólo pueden contener nuevos mensajes, sino también todo tipo de cosas, como ediciones de mensajes, respuestas a encuestas, y mucho másopen in new window.

Un objeto de contexto nuevo se crea exactamente una vez para cada actualización entrante. Los contextos para las diferentes actualizaciones son objetos completamente no relacionados, sólo hacen referencia a la misma información del bot a través de ctx.me.

El mismo objeto de contexto para una actualización será compartido por todo el middleware instalado (documentación) en el bot.

Personalización del objeto de contexto

Si eres nuevo en los objetos de contexto, no necesitas preocuparte por el resto de esta página.

Si lo desea, puede instalar sus propias propiedades en el objeto de contexto.

Vía Middleware (Recomendado)

Las personalizaciones pueden hacerse fácilmente en middleware.

¿Middle qué?

Esta sección requiere una comprensión del middleware, así que en caso de que aún no hayas saltado a esta sección, aquí hay un resumen muy breve.

Todo lo que necesitas saber es que varios manejadores pueden procesar el mismo objeto de contexto. Hay manejadores especiales que pueden modificar ctx antes de que se ejecuten otros manejadores, y las modificaciones del primer manejador serán visibles para todos los manejadores posteriores.

La idea es instalar el middleware antes de registrar otros listeners. Entonces puedes establecer las propiedades que quieras dentro de estos manejadores.

A modo de ejemplo, digamos que quieres establecer una propiedad llamada ctx.config en el objeto contexto. En este ejemplo, la usaremos para almacenar alguna configuración sobre el proyecto para que todos los manejadores tengan acceso a ella. La configuración hará que sea más fácil detectar si el bot es utilizado por su desarrollador o por los usuarios regulares.

Justo después de crear tu bot, haz esto:

const BOT_DEVELOPER = 123456; // identificador del chat del desarrollador del bot
bot.use(async (ctx, next) => {
  // Modifica el objeto de contexto aquí estableciendo la configuración.
  ctx.config = {
    botDeveloper: BOT_DEVELOPER,
    isDeveloper: ctx.from?.id === BOT_DEVELOPER,
  };
  // Ejecutar los manejadores restantes.
  await next();
});

Después de esto, puedes usar ctx.config en los manejadores restantes.

bot.command("start", async (ctx) => {
  // ¡Trabaja con el contexto modificado aquí!
  if (ctx.config.isDeveloper) await ctx.reply("¡Hola mamá! <3");
  else await ctx.reply("¡Bienvenido, humano!");
});

Sin embargo, te darás cuenta de que TypeScript no sabe que ctx.config está disponible, a pesar de que estamos asignando la propiedad correctamente. Así que, aunque el código funcionará en tiempo de ejecución, no compila. Para solucionar esto, tenemos que ajustar el tipo del contexto y añadir la propiedad.

interface BotConfig {
  botDeveloper: number;
  isDeveloper: boolean;
}

type MyContext = Context & {
  config: BotConfig;
};

El nuevo tipo MyContext ahora describe con precisión los objetos de contexto que nuestro bot está manejando realmente.

Tendrás que asegurarte de mantener los tipos sincronizados con las propiedades que inicialices. Podemos utilizar el nuevo tipo pasándolo al constructor del Bot.

const bot = new Bot<MyContext>("");

En resumen, la configuración se verá así:

const BOT_DEVELOPER = 123456; // identificador del chat del desarrollador del bot

// Definir el tipo de contexto personalizado.
interface BotConfig {
  botDeveloper: number;
  isDeveloper: boolean;
}
type MyContext = Context & {
  config: BotConfig;
};

const bot = new Bot<MyContext>("");

// Establecer propiedades personalizadas en los objetos de contexto.
bot.use(async (ctx, next) => {
  ctx.config = {
    botDeveloper: BOT_DEVELOPER,
    isDeveloper: ctx.from?.id === BOT_DEVELOPER,
  };
  await next();
});

// Definir manejadores para objetos de contexto personalizados.
const bot = new Bot<MyContext>("<token>");
bot.command("start", async (ctx) => {
  if (ctx.config.isDeveloper) await ctx.reply("¡Hola mamá!");
  else await ctx.reply("Bienvenido");
});
const BOT_DEVELOPER = 123456; // identificador del chat del desarrollador del bot

const bot = new Bot("");

// Establecer propiedades personalizadas en los objetos de contexto.
bot.use(async (ctx, next) => {
  ctx.config = {
    botDeveloper: BOT_DEVELOPER,
    isDeveloper: ctx.from?.id === BOT_DEVELOPER,
  };
  await next();
});

// Definir manejadores para objetos de contexto personalizados.
bot.command("start", async (ctx) => {
  if (ctx.config.isDeveloper) await ctx.reply("¡Hola mamá!");
  else await ctx.reply("Bienvenido");
});

Naturalmente, el tipo de contexto personalizado también se puede pasar a otras cosas que manejan middleware, como compositoresopen in new window.

const composer = new Composer<MyContext>();

Algunos plugins también requieren que pases un tipo de contexto personalizado, como el plugin router o el plugin menu. Consulta su documentación para ver cómo pueden utilizar un tipo de contexto personalizado. Estos tipos se llaman context flavors, como se describe aquí abajo.

Vía Herencia

Además de establecer propiedades personalizadas en el objeto de contexto, puede subclasificar la clase Context.

class MiContexto extends Context {
  // etc
}

Sin embargo, te recomendamos que personalices el objeto contexto vía middleware porque es mucho más flexible y funciona mucho mejor si quieres instalar plugins.

Ahora veremos cómo utilizar clases personalizadas para los objetos de contexto.

Cuando construyas tu bot, puedes pasar un constructor de contexto personalizado que se utilizará para instanciar los objetos de contexto. Ten en cuenta que tu clase debe extender Context.

import { Bot, Context } from "grammy";
import type { Update, UserFromGetMe } from "@grammyjs/types";

// Definir una clase de contexto personalizada.
class MyContext extends Context {
  // Establecer algunas propiedades personalizadas.
  public readonly customProp: number;

  constructor(update: Update, api: Api, me: UserFromGetMe) {
    super(update, api, me);
    this.customProp = me.username.length * 42;
  }
}

// Pasar el constructor de la clase de contexto personalizado como una opción.
const bot = new Bot("<token>", {
  ContextConstructor: MyContext,
});

bot.on("message", (ctx) => {
  // `ctx` es ahora de tipo `MyContext`.
  const prop = ctx.customProp;
});

bot.start();
const { Bot, Context } = require("grammy");

// Definir una clase de contexto personalizada.
class MyContext extends Context {
  // Establecer algunas propiedades personalizadas.
  public readonly customProp;

  constructor(update, api, me) {
    super(update, api, me);
    this.customProp = me.username.length * 42;
  }
}

// Pasar el constructor de la clase de contexto personalizado como una opción.
const bot = new Bot("<token>", {
  ContextConstructor: MyContext,
});

bot.on("message", (ctx) => {
  // `ctx` es ahora de tipo `MyContext`.
  const prop = ctx.customProp;
});

bot.start();
import { Bot, Context } from "https://deno.land/x/grammy@v1.11.2/mod.ts";
import type { Update, UserFromGetMe } from "https://esm.sh/@grammyjs/types";

// Definir una clase de contexto personalizada.
class MyContext extends Context {
  // Establecer algunas propiedades personalizadas.
  public readonly customProp: number;

  constructor(update: Update, api: Api, me: UserFromGetMe) {
    super(update, api, me);
    this.customProp = me.username.length * 42;
  }
}

// Pasar el constructor de la clase de contexto personalizado como una opción.
const bot = new Bot("<token>", {
  ContextConstructor: MyContext,
});

bot.on("message", (ctx) => {
  // `ctx` es ahora de tipo `MyContext`.
  const prop = ctx.customProp;
});

bot.start();

Fíjate en que el tipo de contexto personalizado será inferido automáticamente cuando utilices una subclase. Ya no necesitas escribir Bot<MiContexto> porque ya has especificado el constructor de tu subclase en el objeto de opciones de new Bot().

Sin embargo, esto hace que sea muy difícil (si no imposible) instalar plugins, ya que a menudo necesitan que instales context flavors.

Context Flavors

Context flavors son una forma de informar a TypeScript sobre las nuevas propiedades de su objeto de contexto. Estas nuevas propiedades pueden ser enviadas en plugins u otros módulos y luego instaladas en tu bot.

Los context flavors también son capaces de transformar los tipos de propiedades existentes utilizando procedimientos automáticos que son definidos por los plugins.

Additive Context Flavors

Como ya se ha dicho, hay dos tipos diferentes de context flavors. El básico se llama additive context flavor, y siempre que hablamos de sabor de contexto, nos referimos a esta forma básica. Veamos cómo funciona.

Por ejemplo, cuando tienes session data, debes registrar ctx.session en el tipo de contexto. De lo contrario,

  1. no puedes instalar el plugin de sesiones incorporado, y
  2. no tendrás acceso a ctx.session en tus listeners.

Aunque usaremos las sesiones como ejemplo aquí, cosas similares se aplican para muchas otras cosas. De hecho, la mayoría de los plugins te darán un context flavor que debes usar.

Un context flavor es simplemente un pequeño tipo nuevo que define las propiedades que deben añadirse al tipo de contexto. Veamos un ejemplo de un flavor.

interface SessionFlavor<S> {
  session: S;
}

El tipo SessionFlavor (Referencia APIopen in new window) es sencillo: sólo define la propiedad session. Toma un parámetro de tipo que definirá la estructura real de los datos de la sesión.

¿Qué utilidad tiene esto? Así es como puedes dar flavor a tu context con session data:

import { Context, SessionFlavor } from "grammy";

// Declarar que `ctx.session` es de tipo `string`.
type MyContext = Context & SessionFlavor<string>;

Ahora puedes usar el plugin de sesión, y tienes acceso a ctx.session:

bot.on("message", (ctx) => {
  // Ahora `str` es de tipo `string`.
  const str = ctx.session;
});

Transformative Context Flavors

El otro tipo de context flavor es más potente. En lugar de instalarse con el operador &, deben instalarse así:

import { Context } from "grammy";
import { SomeFlavorA } from "my-plugin";

type MyContext = SomeFlavorA<Context>;

Todo lo demás funciona de la misma manera.

Cada plugin (oficial) indica en su documentación si debe usarse a través de un additive or via transformative context flavor.

Combinando Diferentes Context Flavors

Si tienes diferentes additive context flavors, puedes simplemente instalarlos así:

type MyContext = Context & FlavorA & FlavorB & FlavorC;

El orden de los flavors del contexto no importa, puedes combinarlos en el orden que quieras.

Múltiple transformative context flavors también pueden combinarse:

type MyContext = FlavorX<FlavorY<FlavorZ<Context>>>;

Aquí, el orden podría importar, ya que FlavorZ transforma primero a Context, luego a FlavorY, y el resultado de esto será transformado de nuevo por FlavorX. (En la práctica, no hay que preocuparse por esto porque los plugins no suelen chocar entre sí).

Incluso se pueden mezclar additive and transformative flavors:

type MyContext = FlavorX<
  FlavorY<
    FlavorZ<
      Context & FlavorA & FlavorB & FlavorC
    >
  >
>;