5.0.0 • Published 3 years ago

airgram v5.0.0

Weekly downloads
194
License
-
Repository
github
Last release
3 years ago

Airgram

Modern Telegram client framework for TypeScript/JavaScript.

Important: this documentation is for version 1.*. If you need MTProto implementation (version 0.1.*), please follow the link.

Features

  • Based on TDLib;
  • Strictly typed;
  • Documentation out of the box;
  • Supports models;
  • Built on middleware;
  • Authorization helper;

All TDLib classes and methods are described and have suitable wrappers in Airgram. There are only two differences:

  • All parameter names are represent in "camelCase".
  • Parameter @type renamed to _.

Requirements

  • TDLib v1.3.0
  • NodeJS

Documentation


Installation

  1. Build TDLib library according the instruction.
  2. Install Airgram:
# npm
npm install airgram
# yarn
yarn add airgram

Getting started

Basic usage:

import { Airgram, Auth, prompt } from 'airgram'

const airgram = new Airgram({
  apiId: Number(process.env.APP_ID!),
  apiHash: process.env.APP_HASH!
})

const auth = new Auth(airgram)

auth.use({
  code: () => prompt(`Please enter the secret code:\n`),
  phoneNumber: () => prompt(`Please enter your phone number:\n`)
})

// Call Telegram method
airgram.api.getMe().then((response) => {
  console.info(response)
})

// Getting all updates
airgram.updates.use(({ update }, next) => {
  if(update) {
    console.log(`"${update._}" ${JSON.stringify(update)}`)
  }
  return next()
})

// Get only new message updates
airgram.updates.on('updateNewMessage', ({ update }, next) => {
  console.info(update)
  return next
})

Configuration

This section describes the options you can pass to Airgram constructor:

import { Airgram } from 'airgram'

const airgram = new Airgram({
  // options
})

TDLib options

KeyTypeNote
commandstringPath to the tdjson (windows) / libtdjson (unix) command.
useTestDcbooleanIf set to true, the Telegram test environment will be used instead of the production environment
databaseDirectorystringThe path to the directory for the persistent database
filesDirectorystringThe path to the directory for storing files
useFileDatabasebooleanIf set to true, information about downloaded and uploaded files will be saved between application restarts
useChatInfoDatabasebooleanIf set to true, the library will maintain a cache of users, basic groups, supergroups, channels and secret chats. Implies useFileDatabase
useMessageDatabasebooleanIf set to true, the library will maintain a cache of chats and messages. Implies useChatInfoDatabase
useSecretChatsbooleanIf set to true, support for secret chats will be enabled
apiIdnumberApplication identifier for Telegram API access, which can be obtained at https://my.telegram.org
apiHashstringApplication identifier hash for Telegram API access, which can be obtained at https://my.telegram.org
tokenstringToken for bot authorization
systemLanguageCodestringIETF language tag of the user's operating system language
deviceModelstringModel of the device the application is being run on
systemVersionstringVersion of the operating system the application is being run on
applicationVersionstringApplication version
enableStorageOptimizerbooleanIf set to true, old files will automatically be deleted
ignoreFileNamesbooleanIf set to true, original file names will be ignored. Otherwise, downloaded files will be saved under names as close as possible to the original name
logFilePathstringPath to a file where the internal TDLib log will be written. Use an empty path to switch back to the default logging behaviour.
logMaxFileSizenumberMaximum size of the file to where the internal TDLib log is written before the file will be auto-rotated. Should be positive.
logVerbosityLevelnumberNew value of the verbosity level for logging. Value 0 corresponds to fatal errors, value 1 corresponds to errors, value 2 corresponds to warnings and debug warnings, value 3 corresponds to informational, value 4 corresponds to debug, value 5 corresponds to verbose debug, value greater than 5 and up to 1024 can be used to enable even more logging.
databaseEncryptionKeystringEncryption key
clientanyInstance of the TDLib JSON client that you can share between threads.

Other options

KeyTypeNote
modelsObjectContains models, which replace plain JSON objects. Details.
createContextFunctionFunction to override middleware context. Details.

API reference

This section describes public API of an Airgram instance:

KeyTypeNote
apiObjectContains wrappers for all TDLib methods.
configObjectAirgram configuration. Readonly.
clientanyInstance of TDLib JSON client that you can share between threads. Readonly.
handleErrorFunctionError handler. Can be overriden by airgram.catch().
catch(handler) => voidOverrides default error handler. Argument handler takes a function: (error: Error, ctx?: Record<string, any>) => void
pause() => voidStop getting responses and updates from TDLib
resume() => voidContinue getting responses and updates from TDLib
destroy() => voidDestroy Airgram and TDLib instances

Authorization

As a bot

Just specify a secret token by Airgram constructor:

import { Airgram, Auth } from 'airgram'

const airgram = new Airgram({
  token: 'xxx'
})

new Auth(airgram)

As a user

Airgram provides component Auth, which implements basic logic for authorization or registration new Telegram accounts. User just needs to specify the phone number, secret code and some other data, if necessary.

import { Airgram, Auth } from 'airgram'

const airgram = new Airgram()
const auth = new Auth(airgram)

auth.use({
  phoneNumber: '+1234567890',
  firstName: 'John',
  lastName: 'Smith'
})

Method auth.use() takes config:

type AuthAnswer = string | (() => string) | (() => Promise<string>)

interface AuthDialog {
  code?: AuthAnswer
  firstName?: AuthAnswer
  lastName?: AuthAnswer,
  phoneNumber?: AuthAnswer
  password?: AuthAnswer
}

You can use helper prompt to communicate with user by the command line:

import { prompt } from 'airgram'

auth.use({
  code: () => prompt(`Please enter the secret code:\n`),
  phoneNumber: () => prompt(`Please enter your phone number:\n`)
})

Authorization callback

If you want to ensure the code will execute only after successful authorization, you can use auth.ready() method:

auth.ready().then(() => {
  console.log('Success!')
})

// or pass callback
auth.ready(() => {
  console.log('Success!')
})

Middleware

Middleware is a chain of callback functions, which are called before a request is send to TDLib. Middleware allows you modify requests and responses to add some additional logic. Middlewares also are using to handle updates.

This is a scaffolding for middleware function:

import { UPDATE } from 'airgram'

airgram.use((ctx, next) => {
  
  // Add some code here
  
  return next()
})

// You can pass an array of predicates: `['updateNewChat', 'updateSupergroup']`
airgram.on(UPDATE.updateNewChat, (ctx, next) => {
  
  // Add some code here
  
  return next()
})

Tip: in the example above we took predicate's value from the UPDATE directory instead of using string value updateNewChat, because this way protects us from typos and we can use IDE autocomplete.

Function takes 2 arguments: ctx and next.

ctx

Argument ctx contains an object with the following structure:

KeyTypeNote
_stringRequest (or update) type.
airgramAirgramInstance of Airgram.
getState() => ObjectReturns current state.
setStateFunctionA function that takes either a new state object, or a function which receives the previous state and returns a new one. It behaves similarly to setState from React.
request{ method: string, params: Object }Object which contains method and parameters of the request. Value will be undefined for updates.
responseObjectObject which contains response data from TDLib. Field will be undefined if the request has not handled.
updateObjectThis field is available only for updates.

You can extend default context by define your own createContext function:

import { ag, Airgram, createContext as createBaseContext, UPDATE, User } from 'airgram'

interface Context extends ag.Context {
  getUser (id: number): User | void
  setUser (id: number, user: User): void
}

const userMap: Map<number, User> = new Map()

function createContext (options: ag.ContextOptions): Context {
  return {
    ...createBaseContext(options),
    getUser (id: number): User | void {
      return userMap.get(id)
    },
    setUser (id: number, user: User): void {
      userMap.set(id, user)
    }
  }
}

const airgram = new Airgram<Context>({
  createContext
})

airgram.updates.on(UPDATE.updateUser, ({ setUser, update }, next) => {
  setUser(update.user.id, update.user)
  return next()
})

airgram.updates.on(UPDATE.updateNewMessage, async ({ getUser, update }) => {
  const user = getUser(update.message.senderUserId)

  if (!user) {
    throw new Error('Unknown user')
  }

  console.log(`${user.username}: ${JSON.stringify(update.message)}`)
})

Since middlewares are meant to be composed, they need an easy way to send metadata about the request down the chain of middleware. To accomplish this, each middleware context has a state object. The state is read by using ctx.getState() and written using ctx.setState(newState) or ctx.setState((prevState) => newState). The API of setState is meant to be similar to React's one. For example:

airgram.use(({ setState }, next) => {
  setState({ foo: 'bar' })
  return next()
})

airgram.use(async ({ getState }) => {
  console.log(`foo = ${getState().foo}`);
  // output: foo = bar
})

Each state can be set by the last argument of the request:

// We can set starting state by the last argument
airgram.api.getChats({limit: 10}, {log: true})

airgram.on('getChats', async ({ _, getState }, next) => {
  if(!getState().log) {
    return next()
  }
  const start = new Date()
  await next()
  const time = new Date() - start;
  console.log(`Request "${_}" took ${time} to complete.`);
})

next

The second argument next is a function which runs the next handler.

Function next() returns a promise Promise<any>, so we can use the next handlers result inside of our middleware:

airgram.use(async (ctx, next) => {
  const start = new Date()
  const { request } = ctx
  const result = await next()
  console.log(`Method: ${request.method}, params: ${JSON.stringify(request.params)}, result: ${JSON.stringify(result)}, ${new Date() - start}ms`)
})

Class as a middleware

Sometimes middleware may be pretty complicated or you want to reuse the code. In this case the most suitable way is to create new class (or use existing) and pass it to use() or on() methods.

Your class must have a middleware() factory method:

class MiddlewareClass {
  handle(ctx) {
    // do some work
  }
  
  middleware() {
    return (ctx, next) => {
      this.handle(ctx)
      return next()
    }
  }
}

airgram.use(new MiddlewareClass())

All methods describes here.

Getting updates

Use methods airgram.updates.use() and airgram.updates.on() to add some handlers for updates. It works almost the same as airgram.use() and airgram.on(), but there are two differences:

  1. By using airgram.updates.use() and airgram.updates.on() methods, callbacks won't be called for requests;
  2. Improved typings for updates.

Models

Airgram provide an excellent feature to create your own models for plain JSON objects which returned by TDLib.

For example, lets add some features to the Chat:

import { Airgram, ApiMethods, CHAT_TYPE, ChatBaseModel, UPDATE } from 'airgram'

class ChatModel extends ChatBaseModel {
  get isBasicGroup (): boolean {
    return this.type._ === CHAT_TYPE.chatTypeBasicGroup
  }

  get isSupergroup (): boolean {
    return this.type._ === CHAT_TYPE.chatTypeSupergroup
  }

  get isPrivateChat (): boolean {
    return this.type._ === CHAT_TYPE.chatTypePrivate
  }

  get isSecretChat (): boolean {
    return this.type._ === CHAT_TYPE.chatTypeSecret
  }

  public async isMeChat (api: ApiMethods): Promise<boolean> {
    if ('userId' in this.type) {
      return (await api.getMe()).id === this.type.userId
    }
    return false
  }
}

const airgram = new Airgram({
  models: {
    chat: ChatModel
  }
})

airgram.updates.on(UPDATE.updateNewChat, async ({ update }) => {
  const { chat } = update
  console.info('isBasicGroup: ', chat.isBasicGroup)
  console.info('isSupergroup: ', chat.isSupergroup)
  console.info('isPrivateChat: ', chat.isPrivateChat)
  console.info('isSecretChat: ', chat.isSecretChat)
  console.info('isMeChat: ', await chat.isMeChat(airgram.api))
})

License

The source code is licensed under GPL v3. License is available here.

5.1.1

3 years ago

5.0.0

3 years ago

4.2.0

3 years ago

4.1.1

4 years ago

4.0.0

4 years ago

3.3.0

4 years ago

3.0.4

4 years ago

3.1.2

4 years ago

3.0.3

4 years ago

3.2.3

4 years ago

3.2.2

4 years ago

3.2.1

4 years ago

3.1.1

4 years ago

3.1.0

4 years ago

2.3.0-next.13

5 years ago

3.0.2

5 years ago

3.0.1

5 years ago

3.0.0

5 years ago

2.3.0-next.12

5 years ago

2.3.0-next.8

5 years ago

2.3.0-next.10

5 years ago

2.3.0

5 years ago

2.3.0-next.6

5 years ago

2.1.1

5 years ago

2.1.0

5 years ago

2.1.0-next.1

5 years ago

2.1.0-next.0

5 years ago

2.0.1

5 years ago

2.0.0

5 years ago

2.0.0-next.2

5 years ago

1.2.1

5 years ago

2.0.0-0

5 years ago

1.2.0

5 years ago

1.1.2

5 years ago

1.1.1

5 years ago

1.1.0

5 years ago

1.1.0-0

5 years ago

1.0.3

5 years ago

1.0.2

5 years ago

1.0.1

6 years ago

1.0.0

6 years ago

1.0.0-1

6 years ago

1.0.0-0

6 years ago

0.1.21

6 years ago

0.1.20

6 years ago

0.1.19

6 years ago

0.1.18

6 years ago

0.1.17

6 years ago

0.1.15

6 years ago

0.1.14

6 years ago

0.1.13

6 years ago

0.1.12

6 years ago

0.1.11

6 years ago

0.1.10

6 years ago

0.1.9

6 years ago

0.1.8

6 years ago

0.1.7

6 years ago

0.1.6

6 years ago

0.1.5

6 years ago

0.1.3

6 years ago

0.1.2

6 years ago

0.1.1

6 years ago

0.1.0-1

6 years ago