A fully typed HTTP client with explicit behavior & error handling.
Important
If you only want to support browser environment, you should use package @lylajs/web
instead of lyla
.
Environment | Package | Note |
---|---|---|
web | @lylajs/web |
|
node | @lylajs/node |
|
toutiao miniprogram | @lylajs/tt |
|
weixin miniprogram | @lylajs/wx |
|
qq miniprogram | @lylajs/qq |
|
zhifubao miniprogram | @lylajs/my |
|
uni-app | @lylajs/uni-app |
|
web + nodejs | lyla |
Unless you have explicit cross-platform isomorphic requirements, please don't use this package. |
English · 中文
- Won't share options between different instances, which means your reqeust won't be unexpectedly modified.
- Won't transform response body implicitly (For example transform invalid JSON to string).
- Won't suppress expection silently (JSON parse error, config error, eg.).
- Explicit error handling.
- Supports typescript for response data.
- Supports upload progress (which isn't supported by fetch API).
- Friendly error tracing (with sync trace, you can see where the request is sent on error).
- Access typed custom context object in the whole process.
For difference compared with other libs, see FAQ.
# you can install `lyla` or `@lylajs/xxx`
npm i @lylajs/web # for npm
pnpm i @lylajs/web # for pnpm
yarn add @lylajs/web # for yarn
Important
Lyla use json
field to config request data, not body
! Also, there's no data
field in lyla.
body
field is used to set raw body of the request such as string
or Blob
, which is not a common case.
import { createLyla } from '@lylajs/web'
// For request which need default options, hooks or custom context info,
// it's recommended using `createLyla` to create lyla instance.
// If you only need the simplest usage, you can use
// import { lyla } from '@lylajs/web'
const { lyla } = createLyla({ context: null })
const { json } = await lyla.post('https://example.com', {
json: { foo: 'bar' }
})
// TypeScript
type MyType = {}
// `json`'s type is `MyType`
const { json } = await lyla.post<MyType>('https://example.com', {
json: { foo: 'bar' }
})
function createLyla<C>(
options: LylaRequestOptions<C> & { context: C },
...overrides: LylaRequestOptions<C>[]
): { lyla: Lyla; isLylaError: (e: unknown) => e is LylaError }
Beta compatibility. Currently only work in 2.0.0-beta.1
and later versions.
Create a lyla
instance with retry feature using existing lyla
instance. For detail see [Retry request](#Retry request).
type LylaRequestOptions<C = undefined> = {
url?: string
method?:
| 'get'
| 'GET'
| 'post'
| 'POST'
| 'put'
| 'PUT'
| 'patch'
| 'PATCH'
| 'head'
| 'HEAD'
| 'delete'
| 'DELETE'
| 'options'
| 'OPTIONS'
timeout?: number
/**
* True when credentials are to be included in a cross-origin request.
* False when they are to be excluded in a cross-origin request and when
* cookies are to be ignored in its response.
*/
withCredentials?: boolean
headers?: LylaRequestHeaders
/**
* Type of `response.body`.
*/
responseType?: 'arraybuffer' | 'blob' | 'text'
body?: XMLHttpRequestBodyInit
/**
* JSON value to be written into the request body. It can't be used with
* `body`.
*/
json?: any
/**
* Query object, also known as search params.
* Note, if you want to set `null` or `undefined` as value in query,
* use object like `query: { key: "undefined" }` instead of `query: { key: undefined }`.
* Otherwise, the k-v pair will be ignored.
*/
query?: Record<
string,
| string
| number
| boolean
| Array<string | number | boolean>
| null
| undefined
>
baseUrl?: string
/**
* Abort signal of the request.
*/
signal?: AbortSignal
onUploadProgress?: (progress: LylaProgress<C>) => void
onDownloadProgress?: (progress: LylaProgress<C>) => void
/**
* Whether to allow get request with body. Default is false.
* It's not recommended to use GET request with body since it doesn't conform HTTP
* specification.
*/
allowGetBody?: boolean
hooks?: {
/**
* Callbacks fired when options is passed into the request. In this moment,
* request options haven't be normalized.
*/
onInit?: Array<
(
options: LylaRequestOptions<C>
) => LylaRequestOptions<C> | Promise<LylaRequestOptions<C>>
>
/**
* Callbacks fired before request is sent. In this moment, request options is
* normalized.
*/
onBeforeRequest?: Array<
(
options: LylaRequestOptions<C>
) => LylaRequestOptions<C> | Promise<LylaRequestOptions<C>>
>
/**
* Callbacks fired after headers are received.
*
* only work in @lylajs/web @lylajs/node and lyla.
*/
onHeadersReceived?: Array<
(
payload: {
headers: Record<string, string>
originalRequest: M['originalRequest']
requestOptions: LylaRequestOptionsWithContext<C>
},
reject: (reason: unknown) => void
) => void
>
/**
* Callbacks fired after response is received.
*/
onAfterResponse?: Array<
(
response: LylaResponse<any>,
reject: (reason: unknown) => void
) => LylaResponse<any> | Promise<LylaResponse<any>>
>
/**
* Callbacks fired when there's error while response handling. It's only
* fired by LylaError. Error thrown by user won't triggered the callback,
* for example if user throws an error in `onAfterResponse` hook. The
* callback won't be fired.
*
* Before the callback if finished, the error won't be thrown.
*/
onResponseError?: Array<
(
error: LylaResponseError<C>,
reject: (reason: unknown) => void
) => void | Promise<void>
>
/**
* Callbacks fired when a non-response error occurs (except
* BROKEN_ON_NON_RESPONSE_ERROR)
*/
onNonResponseError?: Array<
(error: LylaNonResponseError<C>) => void | Promise<void>
>
}
/**
* Custom context of the request.
*/
context?: C
/**
* Extra requestion options, these options will be passed to the corresponding
* request method of the platform. Its type depends on the platform.
*/
extraOptions?: {}
}
type LylaResponse<T = any, C = undefined> = {
requestOptions: LylaRequestOptions<C>
status: number
statusText: string
/**
* Headers of the response. All the keys are in lower case.
*/
headers: Record<string, string>
/**
* Response body.
*/
body: PlatformRelevant
/**
* JSON value of the response. If body is not valid JSON text, access the
* field will cause an error.
*/
json: T
/**
* Custom context of the request
*/
context?: C
}
type LylaProgress<C> = {
/**
* Percentage of the progress. From 0 to 100.
*/
percent: number
/**
* Loaded bytes of the progress.
*/
loaded: number
/**
* Total bytes of the progress. If progress is not length-computable it would
* be 0.
*/
total: number
/**
* Whether the total bytes of the progress is computable.
*/
lengthComputable: boolean
/**
* Request options of the request.
*/
requestOptions: LylaRequestOptions<C>
}
type LylaRequestHeaders = Record<string, string | number | undefined>
Request headers can be string
, number
or undefined
. If it's undefined
,
it would override default options' headers. For example:
import { createLyla } from '@lylajs/web'
const { lyla } = createLyla({ headers: { foo: 'bar' }, context: null })
// Request won't have the `foo` header
request.get('http://example.com', { headers: { foo: undefined } })
import { createLyla, LYLA_ERROR } from '@lylajs/web'
const { lyla, isLylaError } = createLyla({ context: null })
try {
const { json } = await lyla.get('https://example.com')
// ...
} catch (e) {
if (isLylaError(e)) {
e.type
// ...
} else {
// ...
}
}
// This is not a percise definition, it platform relavant. For full definition,
// see https://github.com/07akioni/lyla/blob/main/packages/core/src/error.ts
type LylaError<C = undefined> = {
name: string
message: string
type: LYLA_ERROR
// LylaError's corresponding original error. Normally it's useless, only
// invalid JSON will set this field. In most time, you may need `detail` field.
error: Error | undefined
detail: PlatformRelevant // Fail info generated by specific platform
response: PlatformRelevant // Like LylaResponse | undefined
// Custom context of the request
context: C
}
export enum LYLA_ERROR {
/**
* Request encountered an error, fired by XHR `onerror` event. It doesn't mean
* your network has error, for example CORS error also triggers NETWORK_ERROR.
*/
NETWORK = 'NETWORK',
/**
* Request is aborted.
*/
ABORTED = 'ABORTED',
/**
* Response text is not valid JSON.
*/
INVALID_JSON = 'INVALID_JSON',
/**
* Trying resolving `response.json` with `responseType='arraybuffer'` or
* `responseType='blob'`.
*/
INVALID_CONVERSION = 'INVALID_CONVERSION',
/**
* Request timeout.
*/
TIMEOUT = 'TIMEOUT',
/**
* HTTP status error.
*/
HTTP = 'HTTP',
/**
* Request `options` is not valid. It's not a response error.
*/
BAD_REQUEST = 'BAD_REQUEST',
/**
* `onAfterResponse` hook throws error.
*/
BROKEN_ON_AFTER_RESPONSE = 'BROKEN_ON_AFTER_RESPONSE',
/**
* `onBeforeRequest` hook throws error.
*/
BROKEN_ON_BEFORE_REQUEST = 'BROKEN_ON_BEFORE_REQUEST',
/**
* `onInit` hook throws error.
*/
BROKEN_ON_INIT = 'BROKEN_ON_INIT',
/**
* `onResponseError` hook throws error.
*/
BROKEN_ON_RESPONSE_ERROR = 'BROKEN_ON_RESPONSE_ERROR',
/**
* `onNonResponseError` hook throws error.
*/
BROKEN_ON_NON_RESPONSE_ERROR = 'BROKEN_ON_NON_RESPONSE_ERROR',
/**
* `onHeadersReceived` hook throws error.
*/
BROKEN_ON_HEADERS_RECEIVED = 'BROKEN_ON_HEADERS_RECEIVED',
/**
* Lyla instance created with `withRetry` throws an unexpected error. This
* error isn't created by `lyla` instance itself, but thrown by `onRejected`
* or `onResolved` of `withRetry` or the process of creating retry request options
* defined by user.
*
* The error won't be created by `lyla` instance that not created with `withRetry`.
*/
BROKEN_RETRY = 'BROKEN_RETRY',
/**
* A non-lyla error is return by `onRejected` or `onResolved`'s `reject` action.
* Lyla error won't be wrapped in this error.
*
* The error won't be created by `lyla` instance that not created with `withRetry`.
*/
RETRY_REJECTED_BY_NON_LYLA_ERROR = 'RETRY_REJECTED_BY_NON_LYLA_ERROR'
}
import { createLyla } from '@lylajs/web'
const { lyla } = createLyla({
context: null,
hooks: {
onResponseError(error) {
switch error.type {
// ...
}
},
onNonResponseError(error) {
switch error.type {
// ...
}
}
}
})
You can use native AbortController
or LylaAbortController
to abort requests.
Please note that LylaAbortController
doesn't polyfill all APIs of
AbortController
.
import { createLyla, LylaAbortController } from '@lylajs/web'
const controller = new LylaAbortController()
const { lyla } = createLyla({ context: null })
lyla.get('url', {
signal: controller.signal
})
controller.abort()
You can access a context object in hooks, responses & errors.
const { lyla, isLylaError } = createLyla({
context: {
startTime: -1,
endTime: -1,
duration: -1
},
hooks: {
onInit: [
(options) => {
options.context.startTime = Date.now()
return options
}
],
onResponseError: [
(options) => {
options.context.endTime = Date.now()
options.context.duration =
options.context.endTime - options.context.startTime
return options
}
],
onAfterResponse: [
(options) => {
options.context.endTime = Date.now()
options.context.duration =
options.context.endTime - options.context.startTime
return options
}
]
}
})
lyla.get('/foo').then((response) => {
console.log(response.context.duration)
})
Beta compatibility. Currently only work in 2.0.0-beta.1
and later versions.
Lyla provides a withRetry
method to create a lyla instance with retry capability.
lyla.withRetry(options: LylaWithRetryOptions) => Lyla
The type LylaWithRetryOptions
is (simplified version):
type LylaWithRetryOptions<S> = {
onResolved: (params: {
state: S
options: LylaRequestOptions
response: LylaResponse
}) => Promise<
| {
action: 'retry'
value: () => Promise<LylaRequestOptions> | LylaRequestOptions
}
| {
action: 'resolve'
value: LylaResponse
}
| {
action: 'reject'
value: unknown
}
>
onRejected: (params: {
state: S
options: LylaRequestOptionsWithContext
lyla: Lyla
error: LylaError
}) => Promise<
| {
action: 'retry'
value: () => Promise<LylaRequestOptions> | LylaRequestOptions
}
| {
action: 'reject'
value: unknown
}
>
createState: () => S
}
The lyla.withRetry
method returns a new lyla instance with all lyla methods except retry
. This instance will decide whether to retry, continue, or reject based on the return values of onResolved
and onRejected
.
createState
is called at the beginning of each request to create a new state object, which is passed to onResolved
and onRejected
. You can use this object to store some state, such as retry count.
Here is an example of retrying three times:
import { createLyla } from '@lylajs/*' // * is the platform you need
const { lyla: _lyla } = createLyla({ context: null })
const lyla = _lyla.withRetry({
createState: () => ({
count: 0
}),
onResolved: async ({ response }) => {
return {
action: 'resolve',
value: response
}
},
onRejected: async ({ state, error, options }) => {
state.count += 1
if (state.count > 3) {
return {
action: 'reject',
value: error
}
} else {
return {
action: 'retry',
// Retry with the original options
value: () => options
}
}
}
})
An instance created by lyla.withRetry
may encounter three types of errors when sending a request:
- A Lyla Error returned by the reject action, which will be thrown directly without being wrapped.
- A non-Lyla Error (e.g.,
error1
) returned by the reject action, which will be wrapped into aRETRY_REJECTED_BY_NON_LYLA_ERROR
type error and thrown (e.g.,error2
).error1
can be accessed viaerror2.error
. - An exception thrown by
onResolved
oronRejected
, or an exception thrown by the value function of the retry action, which will be wrapped into aBROKEN_RETRY
type error and thrown.
- Why not axios?
axios.defaults
will be applied to all axios instances created byaxios.create
, which means your code may be influenced unexpectedly by others. The behavior can't be changed by any options.axios.defaults
is a global singleton, which means you can't get a clean copy of it. Since your code may run after than others' code that modifies it.- axios will transform invalid JSON to string sliently by default.
- axios can't access an typed context object in request processes.
- Why not ky?
- ky is based on fetch, it can't support upload progress.
- ky is based on fetch, it can't access headers before response is fully resolved.
- ky's Response data type can't be typed.
- ky can't access an typed context object in request processes.