Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/chrvadala/aggregate-fn
Aggregate fn is a tiny Javascript utility that groups together multiple async operations.
https://github.com/chrvadala/aggregate-fn
batch debounce javascript nodejs rate-limit throttle utility
Last synced: 21 days ago
JSON representation
Aggregate fn is a tiny Javascript utility that groups together multiple async operations.
- Host: GitHub
- URL: https://github.com/chrvadala/aggregate-fn
- Owner: chrvadala
- License: mit
- Created: 2022-06-29T14:27:10.000Z (over 2 years ago)
- Default Branch: main
- Last Pushed: 2024-03-02T15:25:51.000Z (8 months ago)
- Last Synced: 2024-10-05T12:07:12.812Z (about 1 month ago)
- Topics: batch, debounce, javascript, nodejs, rate-limit, throttle, utility
- Language: JavaScript
- Homepage:
- Size: 1000 KB
- Stars: 3
- Watchers: 2
- Forks: 0
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# aggregate-fn
Aggregate fn is a Javascript utility that groups together multiple async operations. It is really useful when you want to merge many API requests, avoiding rate-limits.
[![chrvadala](https://img.shields.io/badge/website-chrvadala-orange.svg)](https://chrvadala.github.io)
[![Test](https://github.com/chrvadala/aggregate-fn/workflows/Test/badge.svg)](https://github.com/chrvadala/aggregate-fn/actions)
[![Coverage Status](https://coveralls.io/repos/github/chrvadala/aggregate-fn/badge.svg)](https://coveralls.io/github/chrvadala/aggregate-fn)
[![npm](https://img.shields.io/npm/v/aggregate-fn.svg?maxAge=2592000?style=plastic)](https://www.npmjs.com/package/aggregate-fn)
[![Downloads](https://img.shields.io/npm/dm/aggregate-fn.svg)](https://www.npmjs.com/package/aggregate-fn)
[![Donate](https://img.shields.io/badge/donate-GithubSponsor-green.svg)](https://github.com/sponsors/chrvadala)# Features
This utility can:
- aggregate requests up to a configured upper limit
- aggregate requests up to a max configured delay
- provide working metrics```javascript
const { fn, flush, cancel } = aggregateFn(aggregableFn, {
maxWaitTime: 100,
maxItems: 10,
stats: console.log
})
```
![aggregate-fn libray model](https://raw.githubusercontent.com/chrvadala/aggregate-fn/main/aggregate-fn.png)# Documentation
- [APIs](https://github.com/chrvadala/aggregate-fn/blob/main/docs/api.md)# Install
```sh
npm install aggregate-fn
```
# Examples## Simple test
In the following code snippet multiple async operations are aggregated together. Full source code available here [simple.js](https://github.com/chrvadala/aggregate-fn/blob/main/examples/simple.js)````js
import { aggregateFn } from 'aggregate-fn'
import { promisify } from 'util'
const sleep = promisify(setTimeout)// this is a way to gather metrics
const statsPrinter = stats => console.log(`Stats: ${JSON.stringify(stats)}`)// Here you can execute your async task (e.g. call one API)
const myAggregableAsyncFn = async requests => {
const responses = requests.map(request => request * 2)
return responses
}// Init aggregateFn
const { fn, flush, cancel } = aggregateFn(myAggregableAsyncFn, {
maxWaitTime: 200,
maxItems: 2,
stats: statsPrinter
})//Executes multiple parallel async functions
Promise.all([
(async function () {
const res = await fn(1)
console.log(`Request #1: ${res}`)
}()),(async function () {
const res = await fn(2)
console.log(`Request #2: ${res}`)
}()),(async function () {
const res = await fn(3)
console.log(`Request #3: ${res}`)
}()),(async function () {
await sleep(500)
const res = await fn(4)
console.log(`Request #4: ${res}`)
}())
])process.on('SIGINT', () => flush())
process.on('exit', () => cancel())
````## Calling a bulk service
In the following code snippet we reduce the number of requests toward a Spotify service, in order to have a better usage of Spotify's quota. Full source code available here: [spotify.js](https://github.com/chrvadala/aggregate-fn/blob/main/examples/spotify.js)````js
import { aggregateFn } from 'aggregate-fn'// https://developer.spotify.com/documentation/general/guides/authorization/client-credentials/
const SPOTIFY_CLIENTID = process.env.SPOTIFY_CLIENTID
const SPOTIFY_SECRET = process.env.SPOTIFY_SECRET// Init aggregateFn
const { fn: getArtist, flush, cancel } = aggregateFn(_getArtistBulk, {
maxWaitTime: 1000,
maxItems: 10,
stats: stats => console.log(`Stats: ${JSON.stringify(stats)}`)
})// Executes multiple parallels async functions
Promise.all([
(async function () {
const info = await getArtist('1dfeR4HaWDbWqFHLkxsg1d')
console.log(`Request #1: ${info}`)
}()),(async function () {
const info = await getArtist('3fMbdgg4jU18AjLCKBhRSm')
console.log(`Request #2: ${info}`)
}())
])process.on('SIGINT', () => flush())
process.on('exit', () => cancel())// ** Spotify APIs ** //
async function _getArtistBulk (artistIds) {
const token = await _getSpotifyToken(SPOTIFY_CLIENTID, SPOTIFY_SECRET)
const req = await fetch('https://api.spotify.com/v1/artists?ids=' + artistIds.join(','), {
headers: {
Authorization: `Bearer ${token}`,
'Content-Type': 'application/json'
}
})const json = await req.json()
if (!req.ok || !Array.isArray(json.artists)) throw new Error('Unable to retrieve Spotify data. Check your CLIENT_ID and SECRET.')
return json.artists.map(({ name }) => name)
}async function _getSpotifyToken (clientid, secret) {
const req = await fetch('https://accounts.spotify.com/api/token', {
method: 'POST',
headers: {
Authorization: 'Basic ' + (Buffer.from(clientid + ':' + secret).toString('base64')),
'content-type': 'application/x-www-form-urlencoded'
},
body: 'grant_type=client_credentials'
})const json = await req.json()
if (!req.ok || !json.access_token) throw new Error('Unable to retrieve Spotify token. Check your CLIENT_ID and SECRET.')
return json.access_token
}
````# Changelog
- **0.x** - Beta version
- **1.0** - First official version
- **1.1** - Upgrades deps and github actions
- **1.2** - Improves docs, upgrades deps and github actions
- **1.3** - Upgrades deps and github actions# Contributors
- [chrvadala](https://github.com/chrvadala) (author)
In Memory of My Love :tulip: