ASCall is a lightweight, strongly-typed TypeScript library for orchestrating asynchronous operations and returning standardized responses. It provides a flexible framework for managing success/failure outcomes, concurrency, and lifecycle control across complex workflows.
- Installation
- Overview
- Core Concepts
- ASCall Variants
- Usage Examples
- Extending Base Classes
- Testing & Development
- License
Using npm:
npm install ascallUsing yarn:
yarn add ascallASCall standardizes how asynchronous operations are executed and how results are represented. It ensures every call returns a typed, predictable, and consistent response.
The library is built around:
- π§±
Response- standardized result wrapper - βοΈ
ResponseManager- type-safe response builder - π
ASCallfamily - orchestrators for different async calling patterns - π§ Extensible base classes - for building custom call strategies
Encapsulates operation outcomes in a single structure:
const success = new Response(true, { data: 'ok' }, null)
const failure = new Response(false, null, new Error('Something went wrong'))Implements ResponseManager type, it ensures all Response instances are correctly
typed and constructed. Created via createResponseManager() the most basic one
for Response, methods like fail succed don't really need to return the new
instance of Response it's just for typing purpose, you may mutate the response
instance and just cast it as required type
The library provides multiple specialized implementations of the base
ASCallBase:
| Class | Description |
|---|---|
ASCall |
Base implementation for simple async orchestration. |
ASCallDedumped |
Prevents concurrent duplicate calls (deduplication). |
ASCallDedumpedKeyed |
Deduplicates calls by unique keys. |
ASCallDedumpedKeyedTimed |
Deduplicates and throttles calls by key and time window. |
ASCallBase |
|
ASCallDedumpedBase |
All variants share the same core principles and typing system - but apply different call strategies.
import { ASCall, createResponseManager, parseError } from 'ascall'
const ascall = new ASCall(
async () => {
console.log('Executing action...')
return 'Hello from ASCall!'
},
{
parseError,
responseManager: createResponseManager<void, Error>(),
}
)
const response = await ascall.call()
if (response.isSuccess()) {
console.log('β
Success:', response.getPayload())
} else {
console.error('β Failure:', response.getError())
}import { ASCallDedumped, createResponseManager, parseError } from 'ascall'
const dedumpedCall = new ASCallDedumped(
async () => {
console.log('Fetching data...')
return 'Data result'
},
{
parseError,
responseManager: createResponseManager<string, Error>(),
}
)
// Multiple calls executed nearly simultaneously
const [r1, r2, r3] = await Promise.all([
dedumpedCall.call(),
dedumpedCall.call(),
dedumpedCall.call(),
])
console.log(r1 === r2 && r2 === r3) // true -> same Promise reusedUse case: Avoids redundant concurrent API calls (e.g. clicking a button multiple times rapidly).
import { ASCallDedumpedKeyed, createResponseManager, parseError } from 'ascall'
const fetchUser = new ASCallDedumpedKeyed(
async (id: number) => {
console.log('Fetching user', id)
return { id, name: 'User ' + id }
},
{
parseError,
responseManager: createResponseManager<
{ id: number; name: string },
Error
>(),
}
)
// Deduplicates by key (here: user ID)
const [u1, u2] = await Promise.all([fetchUser.call(1, 1), fetchUser.call(1, 1)])
console.log(u1 === u2) // trueUse case: Prevents duplicate network requests for the same key (e.g. same user ID).
import {
ASCallDedumpedKeyedTimed,
createResponseManager,
parseError,
} from 'ascall'
const fetchUser = new ASCallDedumpedKeyedTimed(
async (id: number) => {
console.log('Fetching user', id)
return { id, name: 'User ' + id }
},
{
parseError,
responseManager: createResponseManager<
{ id: number; name: string },
Error
>(),
}
)
// First call triggers an actual request
const first = await fetchUser.call(1, 0, 1)
// Second call within 3s reuses the same response
const second = await fetchUser.call(1, 3000, 1)
setTimeout(async () => {
// After 3 seconds, a new request is made
const third = await fetchUser.call(1, 3000, 1)
console.log(third !== first) // true
}, 4000)Use case: Caches async calls for a short period to avoid unnecessary re-fetching.
All ASCallBase variants are built on a shared base class hierarchy that you
can extend to create custom logic.
For example, you can create your own Timed Retry ASCall:
import { Response, ASCallBase } from 'ascall'
class ASCallRetriable<
TPayload,
TError extends Error,
TCallParams extends unknown[],
TResponse extends Response<undefined, undefined, boolean>,
TResponseSuccess extends Response<TPayload, undefined, true>,
TResponseFailure extends Response<unknown, TError, false>,
> extends ASCallBase<
TPayload,
TError,
TCallParams,
TResponse,
TResponseSuccess,
TResponseFailure
> {
private retries = 3
async makeRequest(...parameters: TCallParams): Promise<TPayload> {
let lastError: unknown
for (let attempt = 1; attempt <= this.retries; attempt++) {
try {
const payload = await this.request(...parameters)
return payload
} catch (error) {
lastError = error
}
}
throw lastError
}
}This design pattern allows you to:
- Override lifecycle methods
- Inject caching, throttling, or queuing behavior
- Maintain type safety throughout
Run all tests:
npm run testGenerate coverage:
npm run test -- --coveragenpm run lint:check
npm run format:checkBuild the package:
npm run buildπ License
ISC License Copyright Β© 2025 - saberls