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 🙏

© 2026 – Pkg Stats / Ryan Hefner

@ky28059/zulip-js

v2.2.1

Published

Javascript library to access the Zulip API

Readme

zulip-js Node.js CI

Javascript library to access the Zulip API

Usage

Initialization

With API Key

const zulipInit = require('zulip-js');
const config = {
  username: process.env.ZULIP_USERNAME,
  apiKey: process.env.ZULIP_API_KEY,
  realm: process.env.ZULIP_REALM,
};

(async () => {
  const zulip = await zulipInit(config);
  // The zulip object now initialized with config
  console.log(await zulip.streams.subscriptions.retrieve());
})();

With Username & Password

You will need to first retrieve the API key by calling await zulipInit(config).

const zulipInit = require('zulip-js');
const config = {
  username: process.env.ZULIP_USERNAME,
  password: process.env.ZULIP_PASSWORD,
  realm: process.env.ZULIP_REALM,
};

(async () => {
  // Fetch API Key
  const zulip = await zulipInit(config);
  // The zulip object now contains the API Key
  console.log(await zulip.streams.subscriptions.retrieve());
})();

With zuliprc

Create a file called zuliprc (in the same directory as your code) which looks like:

[api]
[email protected]
key=wlueAg7cQXqKpUgIaPP3dmF4vibZXal7
site=http://localhost:9991

Please remember to add this file to your .gitignore! Calling await zulipInit({ zuliprc: 'zuliprc' }) will read this file.

const zulipInit = require('zulip-js');
const path = require('path');
const zuliprc = path.resolve(__dirname, 'zuliprc');
(async () => {
  const zulip = await zulipInit({ zuliprc });
  // The zulip object now contains the config from the zuliprc file
  console.log(await zulip.streams.subscriptions.retrieve());
})();

Examples

Please see some examples in the examples directory.

Also, to easily test an API endpoint while developing, you can run:

$ npm run build
$ npm run call <method> <endpoint> [optional: json_params] [optional: path to zuliprc file]
$ # For example:
$ npm run call GET /users/me
$ npm run call GET /users/me '' ~/path/to/my/zuliprc

Supported endpoints

We support the following endpoints and are striving to have complete coverage of the API. If you want to use some endpoint we do not support presently, you can directly call it as follows:

const params = {
  to: 'bot testing',
  type: 'stream',
  subject: 'Testing zulip-js',
  content: 'Something is horribly wrong....',
};

await zulip.callEndpoint('/messages', 'POST', params);

| Function to call | API Endpoint | Documentation | |------------------------------------------|-------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | zulip.emojis.retrieve() | GET /realm/emoji | retrieves the list of realm specific emojis. | | zulip.emojis.deactivate() | DELETE /realm/emoji/<name> | deactivates the specified emoji. | | zulip.events.retrieve() | GET /events | retrieves events from a queue. You can pass it a params object with the id of the queue you are interested in, the last event id that you have received and wish to acknowledge. You can also specify whether the server should not block on this request until there is a new event (the default is to block). | | zulip.groups.retrieve() | GET /user_groups | retrieves all user groups. | | zulip.groups.create() | POST /user_groups/create | creates a new user group. | | zulip.groups.update() | PATCH /user_groups/<group_id> | updates a user group. | | zulip.groups.deactivate() | DELETE /user_groups/<group_id> | deactivates a user group. | | zulip.groups.members.retrieve() | GET /user_groups/<group_id>/members | retrieves members of a user group. | | zulip.groups.members.update() | POST /user_groups/<group_id>/members | updates members of a user group. | | zulip.groups.members.check() | GET /user_groups/<group_id>/members/<user_id> | checks if a user is a member of a user group. | | zulip.groups.subgroups.retrieve() | GET /user_groups/<group_id>/subgroups | retrieves subgroups of a user group. | | zulip.groups.subgroups.update() | POST /user_groups/<group_id>/subgroups | updates subgroups of a user group. | | zulip.invites.get() | GET /invites | retrieves all invitations. | | zulip.invites.send() | POST /invites | sends an invitation to an email address. | | zulip.invites.resend() | POST /invites/<invite_id>/resend | resends an email invitation. | | zulip.invites.revoke() | DELETE /invites/<invite_id> | revokes an email invitation. | | zulip.invites.multiuse.create() | POST /invites/multiuse | creates a reusable invitation link. | | zulip.invites.multiuse.revoke() | DELETE /invites/multiuse/<invite_id> | revokes a reusable invitation link. | | zulip.linkifiers.retrieve() | GET /realm/linkifiers | return a list of linkifiers in a realm. | | zulip.linkifiers.add() | POST /realm/filters | adds a linkifier to this realm. | | zulip.linkifiers.update() | PATCH /realm/filters/<filter_id> | updates a linkifier. | | zulip.linkifiers.remove() | DELETE /realm/filters/<filter_id> | removes a linkifier from this realm. | | zulip.linkifiers.reorder() | PATCH /realm/linkifiers | reorders linkifiers in this realm. | | zulip.messages.send() | POST /messages | sends a message. | | zulip.messages.retrieve() | GET /messages | retrieves messages from a stream. You need to specify the id of the message to be used as an anchor. Use 1000000000 to retrieve the most recent message, or zulip.users.me.pointer.retrieve() to get the id of the last message the user read. | | zulip.messages.render() | POST /messages/render | gets rendered HTML for a message text. | | zulip.messages.update() | PATCH /messages/<msg_id> | updates the content or topic of the message with the given msg_id. | | zulip.messages.flags.add() | POST /messages/flags | add a flag to a list of messages. Its params are flag which is one of [read, starred, mentioned, wildcard_mentioned, has_alert_word, historical] and messages which is a list of messageIDs. | | zulip.messages.flags.remove() | POST /messages/flags | remove a flag from a list of messages. Its params are flag which is one of [read, starred, mentioned, wildcard_mentioned, has_alert_word, historical] and messages which is a list of messageIDs. | | zulip.messages.getById() | GET /messages/<msg_id> | gets a message by its id. | | zulip.messages.getHistoryById() | GET /messages/<msg_id>/history | return the history of a message | | zulip.messages.deleteReactionById() | DELETE /messages/<msg_id>/reactions | deletes reactions on a message by message id | | zulip.messages.deleteById() | DELETE /messages/<msg_id> | delete the message with the provided message id if the user has permission to do so. | | zulip.queues.register() | POST /register | registers a new queue. You can pass it a params object with the types of events you are interested in and whether you want to receive raw text or html (using markdown). | | zulip.queues.deregister() | DELETE /events | deletes a previously registered queue. | | zulip.reactions.add() | POST /reactions | add a reaction to a message. Accepts a params object with message_id, emoji_name, emoji_code and reaction_type (default is unicode_emoji). | | zulip.reactions.remove() | DELETE /reactions | remove a reaction from a message. Accepts a params object with message_id and emoji_code and reaction_type (default is unicode_emoji). | | zulip.server.settings() | GET /server_settings | returns a dictionary of server settings. | | zulip.streams.retrieve() | GET /streams | retrieves all streams. | | zulip.streams.getStreamId() | GET /get_stream_id | retrieves a stream's id. | | zulip.streams.subscriptions.retrieve() | GET /users/me/subscriptions | retrieves the user's subscriptions. | | zulip.streams.deleteById() | DELETE /streams/<stream_id> | delete the stream with the provided stream id if the user has permission to do so. | | zulip.streams.topics.retrieve() | GET /users/me/<stream_id>/topics | retrieves all the topics in a specific stream. | | zulip.typing.send() | POST /typing | can be used to send a typing notification. The parameters required are to (either a username or a list of usernames) and op (either start or stop). | | zulip.users.retrieve() | GET /users | retrieves all users for this realm. | | zulip.users.create() | POST /users | create a new user. | | zulip.users.me.getProfile() | GET /users/me | retrieves the profile of the user/bot. | | zulip.users.me.alertWords.retrieve() | GET /users/me/alert_words | get array of a user's alert words. | | zulip.users.me.alertWords.add() | POST /users/me/alert_words | add alert words to a user's array. | | zulip.users.me.alertWords.remove() | DELETE /users/me/alert_words | remove alert words from a user's array. | | zulip.users.me.subscriptions.add() | POST /users/me/subscriptions | subscribes a user to a stream/streams. | | zulip.users.me.subscriptions.remove() | DELETE /users/me/subscriptions | remove subscriptions. |

Testing

Use npm test to run the tests.

Writing Tests

Currently, we have a simple testing framework which stubs our network requests and also allows us to test the input passed to it. This is what a sample test for an API endpoint looks like:

import { expect } from 'chai';
import users from '../../lib/resources/users'; // File to test.
import { config, stubNetwork } from '../common'; // Common functions for tests.

describe('Users', () => {
  it('should fetch users', async () => {
    const params = {
      subject: 'test',
      content: 'sample test',
    };
    const output = {
      // The data returned by the API in JSON format.
      already_subscribed: {},
      result: 'success',
    };
    stubNetwork((url, options) => {
      // Function to test the network request parameters.
      expect(url).to.equal(`${config.apiURL}/users`);
      
      const body = bodyToRecord(options.body);
      expect(Object.keys(body).length).to.equal(4);
      expect(body.subject).to.equal(params.subject);
      expect(body.content).to.equal(params.content);
    }, output); // Stub the network modules.

    const data = await users(config).retrieve(params);
    expect(data).to.have.property('result', 'success'); // Function call.
  });
});

Each pull request should contain relevant tests as well as example usage.