Context
El objeto Context
(Referencia de la API de grammY) 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
. 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
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 Bot 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 Bot) que Telegram envía a tu bot. Este objeto de actualización (ctx
) contiene todos los datos que las fuentes ctx
y similares.
El objeto context siempre contiene información sobre tu bot, accesible a través de ctx
.
Atajos
Hay una serie de accesos directos instalados en el objeto de contexto.
Atajo | Descripción |
---|---|
ctx | Obtiene el objeto mensaje, también los editados |
ctx | Obtiene el objeto chat |
ctx | Obtiene el objeto de chat del remitente de ctx (para mensajes anónimos de canal/grupo) |
ctx | Obtiene el autor del mensaje, la consulta de devolución de llamada, u otras cosas |
ctx | Obtiene 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
y ctx
y ctx
y así sucesivamente, y sólo utilizar siempre ctx
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:
- Debemos tener acceso al objeto
bot
. Esto significa que tenemos que pasar el objetobot
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. - Tenemos que sacar el identificador de chat del contexto, y pasarlo explícitamente a
send
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!Message
En cuanto al punto 1., el objeto contexto simplemente te proporciona acceso al mismo objeto API que encuentras en bot
, se llama ctx
. Ahora puedes escribir ctx
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
), 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 send
con el identificador del chat pre-rellenado para ti.
Función de respuesta de Telegram
Aunque el método se llama ctx
en grammY (y en muchos otros frameworks), no utiliza la función reply de Telegram donde se vincula un mensaje anterior.
Si buscas lo que puede hacer send
en el Referencia Bot API de Telegram, verás un número de opciones, como parse
, disable
, y reply
. 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
y ctx
. Utiliza el autocompletado para ver las opciones disponibles directamente en tu editor de código.
Naturalmente, todos los demás métodos de ctx
tienen un acceso directo con los valores correctos precompletados, como ctx
para responder con una foto, o ctx
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 grammY.
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
, 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
porque siempre enviará mensajes al chat con Alice. En su lugar, llama a ctx
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ás.
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
.
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
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
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
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 My
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 compositores.
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<Mi
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
en el tipo de contexto. De lo contrario,
- no puedes instalar el plugin de sesiones incorporado, y
- no tendrás acceso a
ctx
en tus listeners..session
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 Session
(Referencia API) 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
:
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
>
>
>;