npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

grammy-guard

v0.5.0

Published

Guard middlewares for grammY

Downloads

5,921

Readme

A library makes it easy to create guard middlewares, and also contains a set of filters that are often used.

Installation

Node

npm install grammy-guard

Deno

import { guard } from "https://deno.land/x/grammy_guard/mod.ts";

Usage

import { Bot } from "grammy";
import { guard, isPrivateChat, reply } from "grammy-guard";

const bot = new Bot(process.env.BOT_TOKEN as string);

bot.command(
  "start",
  guard(
    isPrivateChat,
    reply("/start is only available in private chat!"),
  ),
  (ctx) => ctx.reply("Hello!"),
);

bot.start();

Guard

flowchart LR
  U("Incoming update") --> G("Guard check")
  G -- filter is passed --> UH("Update handler")
  G -- filter is not passed --> EH("Error handler")

To create a guard, pass a filter as the first argument and an error handler as the second argument to guard constructor.

If no error handler is passed and an update does not pass the guard check, it will be silently ignored.

import { guard, isAdmin } from "grammy-guard";

bot.on("message", guard(isAdmin), (ctx) => ctx.reply("Hello!"));

Reply with Error

To create a guard that sends an error message to a user if an update does not satisfy the filter conditions, pass an error handler as the second argument.

import { guard, isAdmin } from "grammy-guard";

bot.on(
  "message",
  guard(isAdmin, (ctx) => ctx.reply("You are not an admin")),
  (ctx) => ctx.reply("Hello!"),
);

Universal Reply

The library also provides reply helper that allows you to use guards to handle both message and callback_query updates:

import { guard, isAdmin, reply } from "grammy-guard";

const isAdminGuard = guard(isAdmin, reply("You are not an admin"));

bot.on(
  "message",
  // Error message will be sent using sendMessage method
  isAdminGuard,
  (ctx) => ctx.reply("Hello!"),
);

bot.on(
  "callback_query",
  // Error message will be sent using answerCallbackQuery method
  isAdminGuard,
  (ctx) => ctx.reply("Hello!"),
);

You can use the context properties for the response as well.

import { guard, isAdmin, reply } from "grammy-guard";

const isAdminGuard = guard(
  isAdmin,
  reply((ctx) => `${ctx.from?.first_name}, you are not an admin`),
);

bot.on(
  "message",
  // Error message will be sent using sendMessage method
  isAdminGuard,
  (ctx) => ctx.reply("Hello!"),
);

bot.on(
  "callback_query",
  // Error message will be sent using answerCallbackQuery method
  isAdminGuard,
  (ctx) => ctx.reply("Hello!"),
);

Combining Filters

The library provides helpers and and or for combining filters and not for negation.

import { and, guard, not, or, reply } from "grammy-guard";

bot.command(
  "specific",
  guard(
    or(
      isUserHasId(1),
      isUserHasUsername("username1", "username2"),
    ),
    reply(
      "/specific only available to user with ID 1 or usernames @username1 or @username2!",
    ),
  ),
  (ctx) => ctx.reply("Hello!"),
);

bot.command(
  "admin",
  guard(
    and(isAdmin, isGroupChat),
    // same as
    // [isAdmin, isGroupChat],
    reply("/admin only available to admins in group chats!"),
  ),
  (ctx) => ctx.reply("Hello!"),
);

bot.command(
  "start",
  guard(
    and(not(isAdmin), isGroupChat),
    reply("/start only available for non-admins in group chats!"),
  ),
  (ctx) => ctx.reply("Hello!"),
);

Filters

You can filter updates that satisfy the filter conditions using the filter method.:

import { isUser } from "grammy-guard";

// only process updates where the message sender is a user
bot.on("message").filter(isUser, (ctx) => {
  // handle update
});

You can check if the context satisfies the filter conditions:

import { isUser } from "grammy-guard";

bot.on("message", (ctx) => {
  if (isUser(ctx)) {
    // the message sender is a user
  } else {
    // the message sender is not a user
  }
});

In both cases, the type of context will be narrowed.

Filters for Message Senders

  • User
    • isUser - checks if the message sender is a user.
    • isUserHasId(...id: number[]) - checks if the user has a given ID.
    • isUserHasUsername(...username: string[]) - checks if the user has a given username.
      Note: Passed values are case insensitive, usernames will be converted to lower case before comparison.
  • Bot
    • isBot - checks if the message sender is a bot.
    • isBotHasId(...id: number[]) - checks if the bot has a given ID.
    • isBotHasUsername(...username: string[]) - checks if the bot has a given username.
      Note: Passed values are case insensitive, usernames will be converted to lower case before comparison.
  • Sender Chat
    • isSenderChat - checks if the message sender is a sender chat.
    • isSenderChatHasId(...id: number[]) - checks if the sender chat has a given ID.
    • isSenderChatHasUsername(...username: string[]) - checks if the sender chat has a given username.
      Note: Passed values are case insensitive, usernames will be converted to lower case before comparison.
  • isAdmin - checks if the message sender is an admin
  • isUserFromReply - checks if the user is the same user who sent the message to which the bot replied. It is useful that only the user who called up the menu can use the callback buttons.

Filters for Chat Members

  • isChatMemberStatus(status: string) - checks if the chat member's new status is a given status.
  • isMyChatMemberStatus(status: string) - checks if the bot's new chat member status is a given status.

Filters for Chats and Channels

  • isChat - checks if ctx.chat is not undefined.
  • isChatHasId(...id: number[]) - checks if the chat has a given ID.
  • isChatHasUsername(...username: string[]) - checks if the chat has a givenusername.
    Note: Passed values are case insensitive, usernames will be converted to lower case before comparison.
  • isPrivateChat - checks if it is a private chat.
  • isBasicGroup - checks if it is a basic group chat.
  • isSupergroup - checks if it is a supergroup chat.
  • isGroupChat - checks if it is a basic group or a supergroup chat.
  • isChannel - checks if it is a channel.

Reusing Filters

import { isUserHasId } from "grammy-guard";

const isBotOwner = isUserHasId(1337);

// only process updates from the bot owner
bot.on("message").filter(isBotOwner, (ctx) => {
  // handle update
});

Filtering by User Requests

import { createUserRequestRegistry } from "grammy-guard";

const userRequests = createUserRequestRegistry()
  // .add(reason for request, identifier of request)
  .add("ban", 1)
  .add("unban", 2)

// only process updates where `user_shared.request_id` is 1
bot.filter(userRequests.filter("ban"), ctx => {
  // ctx.msg.user_shared.user_id is number
})

// only process updates where `user_shared.request_id` is 2
bot.filter(userRequests.filter("unban"), ctx => {
  // ctx.msg.user_shared.user_id is number
})

// get request id for keyboard
ctx.reply("Hello!", {
  reply_markup: new Keyboard()
    .requestUser("Ban User", userRequests.getId("ban")),
    .requestUser("Unban User", userRequests.getId("unban"))
})

Filtering by Chat Requests

import { createChatRequestRegistry } from "grammy-guard";

const chatRequests = createChatRequestRegistry()
  // .add(reason for request, identifier of request)
  .add("link-chat", 1)
  .add("link-channel", 2)

// only process updates where `chat_shared.request_id` is 1
bot.filter(chatRequests.filter("link-chat"), ctx => {
  // ctx.msg.chat_shared.chat_id is number
})

// only process updates where `chat_shared.request_id` is 2
bot.filter(chatRequests.filter("link-channel"), ctx => {
  // ctx.msg.chat_shared.chat_id is number
})

// get request id for keyboard
ctx.reply("Hello!", {
  reply_markup: new Keyboard()
    .requestChat("Link Chat", chatRequests.getId("link-chat"), {
      chat_is_channel: false,
    }),
    .requestChat("Link Channel", chatRequests.getId("link-channel"), {
      chat_is_channel: true,
    })
})