# grfn **Repository Path**: mirrors_trending/grfn ## Basic Information - **Project Name**: grfn - **Description**: 🦅 A tiny (~315B) utility that executes a dependency graph of async functions as concurrently as possible. - **Primary Language**: Unknown - **License**: Apache-2.0 - **Default Branch**: main - **Homepage**: None - **GVP Project**: No ## Statistics - **Stars**: 0 - **Forks**: 0 - **Created**: 2021-01-20 - **Last Updated**: 2026-02-28 ## Categories & Tags **Categories**: Uncategorized **Tags**: None ## README
```js
import { setTimeout } from 'node:timers/promises'
import grfn from 'grfn'
const fn = grfn({
// `e` depends on `a`, `c`, and `d`. Call `e` with the results of the
// functions once their returned promises resolve.
e: [
async (a, c, d) => {
await setTimeout(10)
return a * c * d
},
[`a`, `c`, `d`],
],
// `d` depends on `b`.
d: [
async b => {
await setTimeout(1)
return b * 2
},
[`b`],
],
// `c` depends on `a` and `b`.
c: [
async (a, b) => {
await setTimeout(5)
return a + b
},
[`a`, `b`],
],
// `a` and `b` have no dependencies! But they must still be listed. They take
// the input given to `fn`.
a: async (n1, n2, n3) => {
await setTimeout(15)
return n1 + n2 + n3
},
b: async (n1, n2, n3) => {
await setTimeout(10)
return n1 * n2 * n3
},
})
const output = await fn(4, 2, 3)
// This will be the output of `e` because no function depends on it!
console.log(`final output: ${output}`)
```
Output:
```
final output: 14256
```
### Debugging
The graph will be automatically validated, including cycle detection, via
TypeScript magic!
## API
### `grfn(vertices) => (...args) => Promise`
Returns a function that runs the dependency graph of functions described by
`vertices`:
- Input: passed to the functions that don't have dependencies in the graph.
- Output: a `Promise` that resolves to the value returned from the graph's
_output function_, the function that is not depended on by any function.
#### `vertices`
Type: `{ [key: string]: Function | [Function, string[]?] }`
An object describing a dependency graph of functions.
Each value in `vertices` must be either:
- A pair containing a function and its array of dependencies by key (e.g.
`[fnA, ['keyB', 'keyC']]`)
- Or a function (equivalent to `[fn, []]`)
The following constraints, which are validated via TypeScript magic, must also
be met:
- Each dependency in `vertices` must also appear as a non-dependency:
- Not okay (`b` doesn't appear as a non-dependency):
```js
grfn({
a: [fnA, [`b`]],
})
```
- Okay:
```js
grfn({
a: [fnA, [`b`]],
b: fnB,
})
```
- `vertices` must describe an
[acyclic](https://en.wikipedia.org/wiki/Directed_acyclic_graph) dependency
graph:
- Not okay (cycle: `a -> b -> a`):
```js
grfn({
a: [fnA, [`b`]],
b: [fnB, [`a`]],
})
```
- Okay:
```js
grfn({
a: [fnA, [`b`]],
b: fnB,
})
```
- `vertices` must have exactly one _output function_, a function that is not
depended on by any function:
- Not okay (both `b` and `c` are not depended on by any function):
```js
grfn({
b: [fnB, [`a`]],
c: [fnC, [`a`]],
a: fnA,
})
```
- Okay:
```js
grfn({
d: [fnD, [`b`, `c`]],
b: [fnB, [`a`]],
c: [fnC, [`a`]],
a: fnA,
})
```
## Contributing
Stars are always welcome!
For bugs and feature requests,
[please create an issue](https://github.com/TomerAberbach/grfn/issues/new).
## License
[MIT](https://github.com/TomerAberbach/grfn/blob/main/license-mit) ©
[Tomer Aberbach](https://github.com/TomerAberbach) \
[Apache 2.0](https://github.com/TomerAberbach/grfn/blob/main/license-apache) ©
[Google](https://github.com/TomerAberbach/grfn/blob/main/notice-apache)