Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
A convenience wrapper around fetch for the browser (and anything that has fetch
).
This package’s files are distributed in the ES module format and have not been transpiled.
Links:
Features (see Features for more detailed descriptions):
ResponseError
for error responsesWhy is it called retrieve
? I wanted to call it makeRequest
(I like clean and explicit names), but that already exists on npm. So I went with retrieve
because that's similar to fetch
.
Install the retrieve
package.
npm install retrieve
Import the retrieve
function and use it.
import { retrieve } from 'retrieve'
const { data, response } = await retrieve({ url: 'http://example.org' })
console.dir(data, response)
Download the retrieve
module.
curl -O 'https://cdn.jsdelivr.net/npm/retrieve@latest/dist/retrieve.js'
Import the retrieve
function and use it.
<script type="module">
import { retrieve } from './retrieve.js'
const { data, response } = await retrieve({ url: 'http://example.org' })
console.dir(data, response)
</script>
Basic usage of retrieve
looks like this:
const { data } = await retrieve({
url: 'https://pokeapi.co/api/v2/pokemon',
})
config
A RetrieveConfig
object.
url
The request URL.
URL
: Will be used as-is.string
:
config.baseUrl
).baseUrl
(optional)Default: window.location.origin
in browser environments; otherwise, undefined
Base for request URL. Ignored if config.url
is a URL
object or an absolute URL string
.
params
(optional)Request query parameters. Will be appended to the request URL. Parameters already existing on the request URL will be overridden. New parameters will be added.
FormData is intentionally not supported because it cannot be easily and reliably turned into an URLSearchParams
object. If you can guarantee that your FormData
object doesn't hold files, you can provide config.params
using new URLSearchParams(formData)
.
init
(optional)Init object passed to fetch
.
The following changes are made to the init
object before it is passed to fetch
(but without changing config.init
):
Headers: If no “content-type” header is set, it is determined automatically where appropriate:
config.data
is an ArrayBuffer of Blob objectconfig.data
is a stringconfig.data
is set and the request method isn't GET or HEADNote that if config.data
is set to a FormData
object, an existing content type will be removed. Read the warning on MDN: Using FormData Objects: Sending files using a FormData object for an explanation.
Body: If config.data
is set, it will be used for fetch's init.body
. See config.data
description for more information. Otherwise, if config.init.body
is set, it will be used for fetch's init.body
.
Signal: If config.timeout
is set to a positive number, it will be used to create fetch's init.signal
using AbortSignal.timeout(config.timeout)
.
data
(optional)Request body data.
If config.data
is set:
init.body
is set to the result of JSON.stringify(config.data)
init.body
is set to config.data
. It's your responsibility to make sure config.data
can be used on init.body
(see fetch() global function: parameters).requestErrorMessage
(optional)Default: 'Unknown request error'
Message for request errors.
If set, it overrides the underlying error's own message which will then be set on the request error's cause
property.
responseErrorMessage
(optional)Default: $statusCode $statusText
(e.g. '404 Not Found'
)
Message for response errors.
timeout
(optional)Default: 0
(no timeout)
Request timeout in milliseconds.
beforeRequestHandlers
(optional)Processed right before a request is sent (i.e. before calling fetch
). Allows making changes to the parameters passed to fetch
after they've been processed by retrieve
.
Example:
const config = {
url: 'https://api.example.org',
beforeRequestHandlers: [
(url, init) => {
const url = import.meta.env.MODE === 'development'
? new URL('http://localhost:1234/api')
: url
return [url, init]
},
],
}
requestErrorHandlers
(optional)Processed if sending the request failed (i.e. the promise returned by fetch
was rejected). Allows implementing corrective measures.
Exceptions during the processing of a request error handler are not caught.
A request error handler can have one of two results:
{ status: 'maintained', value: error }
){ status: 'corrected', value: response }
)Returning a result object with the corrected status and a Response
object allows retrieve
to continue processing the request as if no error occurred in the first place. Then, no further error request handlers will be processed.
Example:
const config = {
url: 'https://api.example.org',
requestErrorHandlers: [
async (requestError, url, init) => {
// Do something to fix the error cause
const response = await fetch(url, init)
return { status: 'corrected', value: response }
},
],
}
Returning a result object with the maintained status and an Error
object makes retrieve
continue treating the request as having errored. Note also that all request error handlers will be processed as long as the previous handlers maintain the error state.
Example:
const config = {
url: 'https://api.example.org',
requestErrorHandlers: [
(requestError, url, init) => {
// Do something with requestError
requestError.message = 'ERR: ' + requestError.message
return { status: 'maintained', value: requestError }
},
],
}
responseSuccessHandlers
(optional)Processed if sending the request succeeded and a response with a status code 200–299 was returned (i.e. the promise returned by fetch
is fulfilled and yields a Response
object whose ok
property is set to true
).
Exceptions during the processing of a response success handler are not caught.
Example:
const config = {
url: 'https://api.example.org',
responseErrorHandlers: [
async (retrieveResponse, url, init) => {
// Do something with retrieveResponse
return retrieveResponse
},
],
}
responseErrorHandlers
(optional)Processed if sending the request succeeded and a response with a status code >=300 was returned (i.e. the promise returned by fetch
is fulfilled and yields a Response
object whose ok
property is set to false
).
Exceptions during the processing of a response error handler are not caught.
A response error handler can have one of two results:
{ status: 'maintained', value: error }
){ status: 'corrected', value: response }
)Returning a result object with the corrected status and a Response
object allows retrieve
to continue processing the response as if no error occurred in the first place. Then, no further error response handlers will be processed.
Example:
const config = {
url: 'https://api.example.org',
responseErrorHandlers: [
async (error, retrieveResponse, url, init) => {
if (retrieveResponse.response.status === 401) {
// Do something to fix the error cause (e.g. refresh the user's session)
const response = await fetch(url, init)
return { status: 'corrected', value: response }
}
return { status: 'maintained', value: error }
},
],
}
Returning a result object with the maintained status and an ResponseError
object makes retrieve
continue treating the response as having errored. Note also that all response error handlers will be processed as long as the previous handlers maintain the error state.
Example:
const config = {
url: 'https://api.example.org',
responseErrorHandlers: [
async (error, retrieveResponse, url, init) => {
// Do something with error
error.message = 'ERR: ' + error.message
return { status: 'maintained', value: error }
},
],
}
A Promise
that resolves to a RetrieveResponse
object.
TypeError
A TypeError
is thrown when fetch
does (see fetch() global function: Exceptions).
ResponseError
A ResponseError
is thrown for fetch
responses with a status code >=300.
By default, this error will be an instance of ResponseError
which will have access to the original Response
object returned by fetch
and the deserialized data
also found on RetrieveResponse
objects:
try {
await retrieve({
url: 'https://pokeapi.co/api/v2/pokemon/grogu/',
})
} catch (error) {
if (error instanceof ResponseError) {
console.log(error.response, error.data)
}
}
Note that when using response error handlers that the final error is determined by you and may or may not be a ResponseError
.
async function example() {
const { data, response } = await retrieve({
url: 'https://pokeapi.co/api/v2/pokemon/pikachu/',
})
console.dir(data, response)
}
example()
async function example() {
try {
await retrieve({
url: 'https://pokeapi.co/api/v2/pokemon/grogu/',
})
} catch (error) {
console.dir(error)
if (error instanceof ResponseError) {
console.log(error.data)
}
}
}
example()
async function example() {
await retrieve({
url: 'http://api.example.org/status',
responseErrorHandlers: [
async (error, retrieveResponse, url, init) => {
if (retrieveResponse.response.status === 401) {
// Do something to fix the error cause (e.g. refresh the user's session)
const response = await fetch(url, init)
return { status: 'corrected', value: response }
}
return { status: 'maintained', value: error }
},
],
})
}
example()
Warning: This is an educational example only. As it stands, a plain HTML form
element without any JavaScript will handle such a use case just fine and do a better job of it. No need for retrieve
.
<form method="POST" enctype="multipart/form-data">
<label>
Name
<input type="text" name="name" value="value">
</label>
<label>
Age
<input type="number" name="age" value="0">
</label>
<label>
File
<input type="file" name="file">
</label>
<button>Submit</button>
</form>
const form = document.querySelector('form')
form.addEventListener('submit', function (event) {
event.preventDefault()
const form = event.target
retrieve({
url: form.action,
data: new FormData(form),
init: {
method: form.method,
},
})
})
Warning: This is an educational example only. As it stands, a plain HTML form
element without any JavaScript will handle such a use case just fine and do a better job of it. No need for retrieve
.
<form>
<label>
Name
<input type="text" name="name" value="value">
</label>
<label>
Age
<input type="number" name="age" value="0">
</label>
<button>Submit</button>
</form>
const form = document.querySelector('form')
form.addEventListener('submit', function (event) {
event.preventDefault()
const form = event.target
retrieve({
url: form.action,
params: new URLSearchParams(new FormData(form)),
init: {
method: form.method,
},
})
})
Use a response error handler to transform a well-defined error format on your API responses into a custom API error class.
class ApiError extends Error {
code = null
constructor(message, code = null) {
super(message)
this.code = code
}
toJSON() {
return {
code: this.code,
message: this.message,
}
}
}
async function example() {
try {
await retrieve({
url: 'http://api.example.org/status',
responseErrorHandlers: [
async (error, { data }) => {
let message = error.message
let code = null
if (data && typeof data === 'object') {
if ('message' in data && typeof data.message === 'string') {
message = data.message
}
if ('code' in data && typeof data.code === 'string') {
code = data.code
}
}
return {
status: 'maintained',
value: new ApiError(message, code),
}
},
],
})
} catch (error) {
if (error instanceof ApiError) {
console.error(`${error.code}: ${error.message}`)
} else {
console.error(error)
}
}
}
example()
The content type for the request is guessed based on the request body format (if one isn't set already).
application/octet-stream
if config.data
is an ArrayBuffer
of Blob
objectplain/text
if config.data
is a stringapplication/json
if config.data
is set and the request method isn't GET or HEADThe request body is automatically serialized for JSON request bodies.
The response body is automatically deserialized for JSON, FormData
, or text response bodies based on the response's content-type header.
In case of receiving a response with a status code >=300 from the underlying fetch
call, retrieve
will return a rejecting promise (with a ResponseError
). The behavior of fetch
is to return a resolving promise (with a Response
) instead.
Four types of interceptors are supported:
Both error interceptors support error correcting logic triggered by returning a new Response
object (e.g. the result of a new fetch
call).
See Example: retrying requests
This package uses semantic versioning.
2.0.0 (2024-10-20)
ResponseError
to expect a RetrieveResponse
object instead of a Response
object. How to update: Replace new ResponseError(response, ...)
with new ResponseError({ response, data: null }, ...)
.add data to ResponseError (bdddbdc)
Add a new data
property to ResponseError
s holding the same deserialized value as the data
property on RetrieveResponse
objects. If an exception occurs during deserialization, the value will be null
.
BREAKING CHANGE: Change the first parameter of ResponseError
to expect a RetrieveResponse
object instead of a Response
object. How to update: Replace new ResponseError(response, ...)
with new ResponseError({ response, data: null }, ...)
.
FAQs
A convenience wrapper around fetch
The npm package retrieve receives a total of 1 weekly downloads. As such, retrieve popularity was classified as not popular.
We found that retrieve demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 0 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.