simple-fetch-ts
is a TypeScript library designed to simplify HTTP requests using a builder-pattern approach. It provides a fluent API for creating, configuring, and executing various HTTP methods (GET
, POST
, PUT
, PATCH
, DELETE
) with type safety and centralized error handling.
It was written and is maintained by me, Derek. Contribution information is at the bottom of this document.
Install the library via npm:
npm install simple-fetch-ts
- Package Size: 9.9 kB (compressed)
- Unpacked Size: 37.6 kB (after installation)
- Fluent Builder API: Chainable methods for setting headers, query parameters, and body data.
- Type-Safety: Improve type safety by specifying the type of the request body or response directly via generics.
- Simplified Query Parameter Handling: Automatic serialization of query parameters.
- Centralized Error Handling: Consistent error messages for all HTTP methods.
- Query Parameter Case Normalization: Optionally convert query parameter keys to lowercase.
- Lightweight: Built with native
fetch
under the hood.
import { simple } from "simple-fetch-ts";
-
Creating a SimpleBuilder Instance
Use the
simple
factory function to validate the URL and instantiate aSimpleBuilder
.import { QueryParams, simple } from "simple-fetch-ts"; const api = simple("https://api.example.com/resource");
-
Configuring the Request
Chain methods to configure headers, query parameters, and request body:
const myFilters = { page: 1, limit: 10, orderBy: "id" }; const params = myFilters as QueryParams; // QueryParams type validates params i.e. no nested objects const convertToLowerCase = true; const request = api .headers({ Authorization: "Bearer token" }) .params(params, convertToLowerCase) // Optional flag for lowercase query parameter keys .body<BodyType>({ name: "example" });
-
Sending the Request
Choose an HTTP method to execute the request:
const response = await request.post<ExpectedReturnType>(); // generic type defaults to <unknown> if type is not passed by the user const myData = response.data; console.log(myData);
-
All Together Now...
const response = await simple("https://api.example.com/resource") .headers({ Authorization: "Bearer token" }) .params(params, convertToLowerCase) .body<BodyType>({ name: "example" }) .post<ExpectedReturnType>(); console.log(response.data);
If you want an inflexible, but quick way to make a simple fetch request, you can use simpleFetch. simpleFetch skips the response object and returns your parsed
data directly. It uses the same fetch helper as the factory function, handling errors internally. If a null
or undefined
value is returned, simpleFetch will
return null
, even if a non-nullable response type has been defined by the user.
import { simpleFetch } from "simple-fetch-ts";
/**
* Fetches data from the given URL and returns the data part of the response.
* @param url - The URL to fetch data from.
* @param requestHeaders - Optional headers to be sent with the request.
* @returns { T | null} - The data returned from the fetch.
* @throws Will throw an error if the fetch fails or the response is not OK.
*/
const response = await simpleFetch<ExpectedReturnType[]>(
"https://api.example.com/resource",
);
if (data === null) {
console.log("No data found, handling gracefully.");
} else {
console.log("Received data:", data);
}
- Description: Factory function that creates a
SimpleBuilder
instance. - Parameters:
url
(string
): The base URL for the API.
- Throws: Error if the provided URL is invalid.
- Returns: A
SimpleBuilder
instance.
The core class for building and executing HTTP requests.
constructor(url: string, defaultHeaders?: HeadersInit)
- Parameters:
url
(string
): The base URL for the API.defaultHeaders
(HeadersInit
): Optional default headers.
- Adds or updates request headers.
- Serializes and appends query parameters to the URL.
- Sets the request body. Automatically assigns
Content-Type: application/json
if not provided.
- Executes a
GET
request.
- Executes a
POST
request with the configured body and headers.
- Executes a
PUT
request with the configured body and headers.
- Executes a
PATCH
request with the configured body and headers.
- Executes a
DELETE
request with the configured headers.
The library includes helper functions for low-level HTTP requests. These are internally used by SimpleBuilder
.
Performs a GET
request.
Performs a POST
request.
Performs a PUT
request.
Performs a PATCH
request.
Performs a DELETE
request.
interface SimpleResponse<T> {
data: T;
status: number;
headers: Headers;
}
const response = await simple("https://api.example.com/resource")
.body({ name: "example", location: "example" })
.post<ExpectedReturnType>(); // no need to set { "Content-Type": "application/json" } if there is a body, it will automatically be set
console.log("Data: ", response.data);
console.log("Status: ", response.status);
console.log("Headers: ", response.headers);
try {
const response = await fetch("https://api.example.com/resource", {
method: "POST",
body: JSON.stringify({ name: "example", location: "example" }),
headers: { "Content-Type": "application/json" },
});
if (!response.ok) {
return {
error: `Network response status ${response.status} with url ${url}`,
};
}
const parsedResponse: ExpectedReturnType = await response.json();
console.log("Data: ", parsedResponse);
console.log("Status: ", response.status);
console.log("Headers: ", response.headers);
} catch (error: unknown) {
throw new Error(
error instanceof Error ? error.message : "An unknown error occurred",
);
}
const userId = 1;
const requestBody = { name: "Updated Name" };
const response = await simple(`https://api.example.com/users/${userId}`)
.headers({ Authorization: "Bearer token" })
.params({ include: "posts" })
.body(requestBody)
.put<ExpectedReturnType>();
console.log("Updated user:", response.data);
The library ensures consistent error messages for invalid configurations or failed requests. Errors are thrown as Error
objects with detailed messages.
/**
* Custom error class to represent errors that occur during HTTP requests.
*
* Provides detailed information about the failed request, including the HTTP method,
* URL, status code, status text, and optionally the response body.
*
* @extends Error
*/
export class SimpleFetchRequestError extends Error {
/**
* @param method - The HTTP method used for the request (e.g., "GET", "POST").
* @param url - The URL to which the request was made.
* @param status - Optional HTTP status code returned from the server.
* @param statusText - Optional status text accompanying the HTTP status code.
* @param responseBody - Optional response body returned from the server.
*/
constructor(
public method: string,
public url: string,
public status?: number,
public statusText?: string,
public responseBody?: any,
) {
super(
`${method} request to ${url} failed with status ${status ?? "unknown"}: ${
statusText ?? "No status text"
}`,
);
this.name = "SimpleFetchRequestError";
}
}
/**
* Custom error class to handle cases where an invalid URL is provided.
*
* This error is thrown when a URL fails validation, providing a clear message
* and guidance for users to ensure proper URL formatting.
*
* @extends Error
*/
export class InvalidURLError extends Error {
/**
* @param url - The invalid URL that caused the error.
*/
constructor(url: string) {
super(
`A valid URL is required, received: ${url}. Ensure the URL starts with "http://" or "https://".`,
);
this.name = "InvalidURLError";
}
}
try {
const request = simple("thisisnotaurl");
} catch (error) {
if (error instanceof InvalidURLError) {
console.error("Invalid URL provided:", error.message);
} else {
console.error("Unexpected error:", error);
}
}
Both simpleFetch()
and simple.fetch()
use tsFetch()
:
import { SimpleFetchRequestError } from "../../errors/request-error";
import { SimpleResponse } from "../../types";
/**
* Performs a typed fetch request to the specified URL.
*
* @template T - The type of the expected response data.
* @param url - The URL to fetch data from.
* @param requestHeaders - Optional headers to be sent with the request.
* @returns A promise that resolves with the fetched data, status, and headers.
* @throws Will throw an error if the fetch fails or the response is not OK.
*/
export const tsFetch = async <T>(
url: string,
requestHeaders: HeadersInit = {},
): Promise<SimpleResponse<T>> => {
try {
const response = await fetch(url, {
method: "GET",
headers: requestHeaders,
});
if (!response.ok) {
const errorText = await response
.text()
.catch(() => "Unable to parse response text");
throw new SimpleFetchRequestError(
"GET",
url,
response.status,
response.statusText,
errorText,
);
}
const data: T = await response.json();
return {
data,
status: response.status,
headers: response.headers,
};
} catch (error: unknown) {
if (error instanceof SimpleFetchRequestError) {
throw error; // Rethrow for consistent handling upstream
}
throw new Error(
error instanceof Error ? error.message : "An unknown error occurred",
);
}
};
As part of maintaining high-quality standards, simple-fetch-ts
ensures robust testing with extensive test coverage across its features. The library leverages Jest for unit and integration testing, providing confidence in functionality and stability. Here’s a detailed overview of the test coverage:
- Test Suites: 10
- Tests: 51
- Snapshots: 0
- Execution Time: ~9.5 seconds
- Overall Coverage:
- Statements: 98.61%
- Branches: 81.6%
- Functions: 100%
- Lines: 98.49%
File | % Stmts | % Branch | % Funcs | % Lines | Uncovered Line #s |
---|---|---|---|---|---|
All files | 96.41% | 83.16% | 88.37% | 96.19% | |
builder | 100% | 100% | 100% | 100% | |
errors | 100% | 83.33% | 100% | 100% | |
- query-param-error.ts | 100% | 100% | 100% | 100% | |
- request-body-error.ts | 100% | 100% | 100% | 100% | |
- request-error.ts | 100% | 83.33% | 100% | 100% | Line 26 |
- url-validation-error.ts | 100% | 100% | 100% | 100% | |
methods/delete | 92.85% | 80% | 50% | 92.3% | Line 26 |
methods/fetch | 92.85% | 100% | 50% | 92.3% | Line 26 |
methods/patch | 94.11% | 60% | 50% | 93.75% | Line 43 |
methods/post | 94.11% | 60% | 50% | 93.75% | Line 43 |
methods/put | 94.11% | 60% | 50% | 93.75% | Line 40 |
simple-factory | 100% | 100% | 100% | 100% | |
simple-fetch | 100% | 88.88% | 100% | 100% | Line 16 |
utility | 94.59% | 93.75% | 100% | 93.93% | |
- get-content-type.ts | 90.9% | 100% | 100% | 90% | Line 26 |
- url-helpers.ts | 96.15% | 89.47% | 100% | 95.65% | Line 58 |
- High overall coverage: As of version 1.0.7, the library demonstrates near-complete coverage for statements, functions, and lines.
- Branch coverage: Slightly lower branch coverage (81.6%) highlights opportunities for additional tests, particularly in conditional logic within methods like
patch
,post
, andput
. - Uncovered lines:
- Conditional handling in
delete
,patch
,post
, andput
methods. - Specific utility edge cases in
get-content-type.ts
andurl-helpers.ts
.
- Conditional handling in
This high coverage ensures the reliability and accuracy of core features, such as HTTP methods (GET
, POST
, PATCH
, PUT
, and DELETE
), utility functions, and the library's factory functions. By focusing on both typical and edge cases, simple-fetch-ts
is designed to handle a variety of scenarios with minimal risk of failure.
- Increasing branch coverage by adding tests for uncovered conditions.
- Monitoring coverage metrics in CI/CD workflows to maintain high standards with every update.
You can replicate these results by running the following command:
npm run test
This will execute all test suites and generate a coverage report. To view coverage details:
npm run test:coverage
By committing to comprehensive testing, simple-fetch-ts
ensures a reliable and user-friendly library for developers.
Contributions are welcome from the community to help improve simple-fetch-ts
! Whether you have ideas for new features, fixes for bugs, or improvements to documentation, your input is valuable and appreciated.
-
Fork the Repository:
Begin by forking the repository to your GitHub account. -
Clone the Repository:
Clone your forked repository locally:git clone https://github.com/<your-username>/simple-fetch-ts.git cd simple-fetch-ts
-
Create a New Branch:
Create a branch for your feature or fix:git checkout -b feature/my-awesome-feature
-
Make Your Changes:
- Write clear, concise, and maintainable code.
- Include tests for new features or fixes.
- Update documentation if necessary.
-
Run Tests:
Ensure all tests pass before submitting your changes:npm run test
-
Submit a Pull Request:
Push your branch to GitHub and open a pull request against themain
branch of the original repository.
In your pull request:- Clearly describe the changes made.
- Reference any related issues or feature requests.
To maintain a high-quality codebase, we ask contributors to adhere to the following:
- Follow the existing code style and structure.
- Write meaningful commit messages (e.g.,
Fix: Handle edge case in tsPatch
). - Ensure any added functionality is covered by tests.
- Keep pull requests focused on a single change whenever possible.
If you encounter a bug or have a suggestion, please open an issue on the GitHub repository.
Provide as much detail as possible, including steps to reproduce bugs or a clear explanation of the feature request.
Thank you for taking the time to support this project and being a part of the open-source community.
This library is licensed under the ISC License.