Skip to content

Commit

Permalink
fix: splitout example
Browse files Browse the repository at this point in the history
  • Loading branch information
seriouslag committed Apr 12, 2024
1 parent ab5d0cb commit 4603de5
Show file tree
Hide file tree
Showing 21 changed files with 191 additions and 80 deletions.
103 changes: 60 additions & 43 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
</h1>

<p align="center">
Typed wrapper around axios.
Typed wrapper around fetch or axios.
</p>

<p align="center">
Expand All @@ -24,7 +24,8 @@
<p align="center">
This package's API is still developing and will not follow SEMVER until release 1.0.0.

HttpClient helps standardarize making HTTP calls and handling when errors are thrown. HttpClient works both in the browser and node environments. Exposes an interface to abort HTTP calls using <a href="https://developer.mozilla.org/en-US/docs/Web/API/AbortController">AbortController</a>. See below about using [AbortController](#using-abortcontroller) in older environments. Exposes an interface to control how requests and responses are handled. See below about using [HttpClient's Request Strategies](#using-request-strategies). Some strategies are provided in this package, but you can also implement your own strategies. List of strategies are provided below.
HttpClient helps standardizes making HTTP calls regardless of the underlying client used, (fetch is used by default but other clients are available) and handling when errors are thrown. HttpClient works both in the browser and node environments. Exposes an interface to abort HTTP calls using <a href="https://developer.mozilla.org/en-US/docs/Web/API/AbortController">AbortController</a>. See below about using [AbortController](#using-abortcontroller) in older environments. Exposes an interface to control how requests and responses are handled. See below about using [HttpClient's Request Strategies](#using-request-strategies). Some strategies are provided in this package, but you can also implement your own strategies. List of strategies are provided below.

</p>

<h2>Installation</h2>
Expand All @@ -38,6 +39,7 @@ npm install @seriouslag/httpclient
<p>To see additional examples look in the `src/examples/` directory.</p>

Basic example:

```typescript
import { HttpClient } from '@seriouslag/httpclient';

Expand All @@ -48,20 +50,20 @@ interface NamedLink {

interface PokemonPage {
count: number;
next: string|null;
previous: string|null;
next: string | null;
previous: string | null;
results: NamedLink[];
}

const httpClient = new HttpClient();

async function fetchPokemonPage (offset: number = 0, pageSize: number = 20) {
async function fetchPokemonPage(offset: number = 0, pageSize: number = 20) {
const pokemonApiUrl = 'https://pokeapi.co/api/v2';
return await this.httpClient.get<PokemonPage>(`${pokemonApiUrl}/pokemon`, {
params: {
offset: offset,
limit: pageSize,
},
params: {
offset: offset,
limit: pageSize,
},
});
}

Expand All @@ -72,74 +74,88 @@ async function fetchPokemonPage (offset: number = 0, pageSize: number = 20) {
})();
```

<h2>Configuring axios</h2>
<h2>Using axios</h2>

We can use axios as the underlying client by installing the `@seriouslag/httpclient-axios` package.
A custom client adaptor can be provided to the HttpClient constructor, an interface is exposed to allow for custom client adaptors to be created.

```bash
npm install @seriouslag/httpclient @seriouslag/httpclient-axios
```

<p>
Axios can be configured, axios options can be passed into the constructor of HttpClient.
</p>

```typescript
import { HttpClient } from '@seriouslag/httpclient';
import { AxiosClientAdaptor } from '@seriouslag/httpclient-axios';
import { Agent } from 'https';

const httpsAgent = new Agent({
rejectUnauthorized: false,
});

const httpClient = new HttpClient({
axiosOptions: {
httpsAgent,
},
const axiosClientAdaptor = new AxiosClientAdaptor({
httpsAgent,
});

const httpClient = new HttpClient(axiosClientAdaptor);
```

<h2>Using AbortController</h2>
<p>Each of the HTTP methods of the HttpClient accept an instance of a AbortController. This allows HTTP requests to be cancelled if not already resolved.


```typescript
import { HttpClient } from '@seriouslag/httpclient';

interface PokemonPage {
count: number;
next: string|null;
previous: string|null;
next: string | null;
previous: string | null;
results: NamedLink[];
}

const pokemonApiUrl = 'https://pokeapi.co/api/v2';
const httpClient = new HttpClient();
const cancelToken = new AbortController();

const request = httpClient.get<PokemonPage>(`${pokemonApiUrl}/pokemon`, cancelToken);
const request = httpClient.get<PokemonPage>(
`${pokemonApiUrl}/pokemon`,
cancelToken,
);

cancelToken.abort();

try {
const result = await request;
console.log('Expect to not get here because request was aborted.', result)
console.log('Expect to not get here because request was aborted.', result);
} catch (e) {
console.log('Expect to reach here because request was aborted.')
console.log('Expect to reach here because request was aborted.');
}
```

</p>

<h3>AbortController in older environments</h3>
<p>
Abort controller is native to node 15+ and modern browsers. If support is needed for older browsers/node versions then polyfills can be found. This polyfill is used in the Jest test environment for this repository: <a href="https://www.npmjs.com/package/abortcontroller-polyfill">abortcontroller-polyfill</a>

```typescript
import 'abortcontroller-polyfill/dist/abortcontroller-polyfill-only';
import { HttpClient } from '@seriouslag/httpclient';
```typescript
import 'abortcontroller-polyfill/dist/abortcontroller-polyfill-only';
import { HttpClient } from '@seriouslag/httpclient';

const httpClient = new HttpClient();
```

const httpClient = new HttpClient();
```
</p>

<h2>Using Request Strategies</h2>
<p>
A request strategy is middleware to handle how requests are made and how responses are handled. This is exposed to the consumer using the `HttpRequestStrategy` interface. A request strategy can be passed into the HttpClient (it will be defaulted if not) or it can be passed into each request (if not provided then the strategy provided by the HttpClient will be used). A custom strategy can be provided to the HttpClient's constructor.

Provided strategies:

<ul>
<li>DefaultHttpRequestStrategy - Throws when a response's status is not 2XX</li>
<li>ExponentialBackoffRequestStrategy - Retries requests with a backoff. Throws when a response's status is not 2XX</li>
Expand All @@ -157,30 +173,31 @@ import { HttpClient, HttpRequestStrategy } from '@seriouslag/httpclient';

class CreatedHttpRequestStrategy implements HttpRequestStrategy {

/** Passthrough request to axios and check response is created status */
public async request<T = unknown> (client: AxiosInstance, axiosConfig: AxiosRequestConfig) {
const response = await client.request<T>(axiosConfig);
this.checkResponseStatus<T>(response);
return response;
}

/** Validates the HTTP response is successful created status or throws an error */
private checkResponseStatus<T = unknown> (response: HttpResponse<T>): HttpResponse<T> {
const isCreatedResponse = response.status === 201;
if (isCreatedResponse) {
return response;
}
throw response;
}
/\*_ Passthrough request to axios and check response is created status _/
public async request<T = unknown> (client: AxiosInstance, axiosConfig: AxiosRequestConfig) {
const response = await client.request<T>(axiosConfig);
this.checkResponseStatus<T>(response);
return response;
}

/\*_ Validates the HTTP response is successful created status or throws an error _/
private checkResponseStatus<T = unknown> (response: HttpResponse<T>): HttpResponse<T> {
const isCreatedResponse = response.status === 201;
if (isCreatedResponse) {
return response;
}
throw response;
}
}

const httpRequestStrategy = new CreatedHttpRequestStrategy();

// all requests will now throw unless they return an HTTP response with a status of 201
const httpClient = new HttpClient({
httpRequestStrategy,
httpRequestStrategy,
});
```

````

<h3>Using Request Strategy in a request</h3>

Expand All @@ -199,7 +216,7 @@ const httpClient = new HttpClient({
httpRequestStrategy: new DefaultHttpRequestStrategy(),
});
})();
```
````
</p>
Expand Down
7 changes: 7 additions & 0 deletions packages/examples/LICENSE
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
Copyright 2024 Landon Gavin

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
37 changes: 37 additions & 0 deletions packages/examples/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
{
"name": "@seriouslag/examples",
"version": "0.0.0",
"private": true,
"description": "Example code",
"scripts": {
},
"contributors": [
{
"name": "Landon Gavin",
"email": "[email protected]",
"url": "https://landongavin.dev"
}
],
"repository": {
"type": "git",
"url": "git+https://github.com/seriouslag/HttpClient.git"
},
"publishConfig": {
"access": "public"
},
"keywords": [
"httpClient",
"typescript",
"axios",
"fetch"
],
"bugs": {
"url": "https://github.com/seriouslag/HttpClient/issues"
},
"author": "[email protected]",
"license": "MIT",
"files": [
],
"dependencies": {
}
}
32 changes: 32 additions & 0 deletions packages/examples/src/PokemonApi.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import { HttpClient } from '@seriouslag/httpclient';
import { PokemonPage } from './types';

export class PokemonApi {
private pageSize = 20;

constructor(
private baseUrl: string,
private httpClient: HttpClient,
) {}

/**
* Fetches a page of Pokemon from the API.
*/
public async fetchPokemonPage(
cancelToken?: AbortController,
offset: number = 0,
pageSize: number = this.pageSize,
): Promise<PokemonPage> {
const response = await this.httpClient.get<PokemonPage>(
`${this.baseUrl}/pokemon`,
{
params: {
offset: offset,
limit: pageSize,
},
},
cancelToken,
);
return response;
}
}
2 changes: 2 additions & 0 deletions packages/examples/src/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export * from './types';
export * from './PokemonApi';
File renamed without changes.
22 changes: 22 additions & 0 deletions packages/examples/tsconfig.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
{
"extends": "../../tsconfig.json",
"compilerOptions": {
"baseUrl": ".",
"rootDir": "./src",
"paths": {
"@seriouslag/*": ["../*/src/index"]
}
},
"references": [
{
"path": "../httpclient"
}
],
"include": [
"./src/**/*.ts"
],
"exclude": [
"dist/**/*",
"node_modules/**/*"
]
}
5 changes: 3 additions & 2 deletions packages/httpclient-axios/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@seriouslag/httpclient-axios",
"version": "0.0.3",
"version": "0.0.4",
"description": "Typed wrapper HttpClient for axios",
"browser": "dist/index.min.cjs",
"module": "dist/index.esm",
Expand All @@ -14,7 +14,8 @@
"type": "module",
"types": "dist/index.d.ts",
"scripts": {
"build": "rollup -c"
"build": "rollup -c",
"example:axios": "ts-node src/examples/example-axios-options.ts"
},
"contributors": [
{
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { HttpClient } from '@seriouslag/httpclient/src/HttpClient';
import { PokemonApi } from '@seriouslag/httpclient/src/examples/PokemonApi';
import { HttpClient } from '@seriouslag/httpclient';
import { PokemonApi } from '@seriouslag/examples';
import { AxiosClientAdaptor } from '../index';
import { Agent } from 'https';

Expand Down
3 changes: 3 additions & 0 deletions packages/httpclient-axios/tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,9 @@
"references": [
{
"path": "../httpclient"
},
{
"path": "../examples"
}
],
"include": [
Expand Down
6 changes: 2 additions & 4 deletions packages/httpclient/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@seriouslag/httpclient",
"version": "0.0.18",
"version": "0.0.19",
"description": "Typed wrapper HttpClient for axios",
"browser": "dist/index.min.cjs",
"module": "dist/index.esm",
Expand Down Expand Up @@ -53,7 +53,5 @@
"README.md",
"LICENSE"
],
"dependencies": {
"@babel/runtime": "^7.16.5"
}
"dependencies": {}
}
1 change: 0 additions & 1 deletion packages/httpclient/src/HttpClient.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -246,7 +246,6 @@ describe('HttpClient', () => {
await expect(() => promise).rejects.toThrow();
});

/** TODO: This is not working yet; Investigate https://github.com/ctimmerm/axios-mock-adapter/issues/59 */
it('fetch - if token is aborted after axios call, AbortError is throw', async () => {
const url = 'www.google.com';
const method = 'get';
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,6 @@ import { HttpResponse, Request } from '../Adaptors';

/** How HTTP calls will be handled. */
export interface HttpRequestStrategy {
/** Wrapper request around axios to add request and resposne logic */
/** Wrapper request around axios to add request and response logic */
request: <T = unknown>(request: Request<T>) => Promise<HttpResponse<T>>;
}
Loading

0 comments on commit 4603de5

Please sign in to comment.