Implement main class & fetch function - including robust error handling.

This commit is contained in:
Fred Boniface 2025-03-08 22:29:04 +00:00
parent 292cdbd069
commit 24d8173548
11 changed files with 192 additions and 1 deletions

View File

@ -0,0 +1,9 @@
import { BaseOwlBoardClient } from "./client";
export class LdbClientV2 {
private client: BaseOwlBoardClient;
constructor(client: BaseOwlBoardClient) {
this.client = client;
}
}

View File

@ -0,0 +1,9 @@
import { BaseOwlBoardClient } from "./client";
export class LocationReferenceClientV2 {
private client: BaseOwlBoardClient;
constructor(client: BaseOwlBoardClient) {
this.client = client;
}
}

View File

@ -0,0 +1,9 @@
import { BaseOwlBoardClient } from "./client";
export class MiscClientV2 {
private client: BaseOwlBoardClient;
constructor(client: BaseOwlBoardClient) {
this.client = client;
}
}

View File

@ -0,0 +1,19 @@
import { BaseOwlBoardClient } from "./client";
import { ValidationError } from "../errors";
export class PisClientV2 {
private client: BaseOwlBoardClient;
constructor(client: BaseOwlBoardClient) {
this.client = client;
}
async getStopsByPis(code: string): Promise<any> {
const codeRegex = /^\d{4}$/;
if (!codeRegex.test(code)) {
throw new ValidationError("Invalid input: code must be a four-character string consisting of only numerals")
}
const path = `/api/v2/pis/byCode/${code}`
return this.client.makeRequest("GET", path);
}
}

View File

@ -0,0 +1,9 @@
import { BaseOwlBoardClient } from "./client";
export class TrainClientV2 {
private client: BaseOwlBoardClient;
constructor(client: BaseOwlBoardClient) {
this.client = client;
}
}

View File

@ -0,0 +1,9 @@
import { BaseOwlBoardClient } from "./client";
export class UserClientV2 {
private client: BaseOwlBoardClient;
constructor(client: BaseOwlBoardClient) {
this.client = client;
}
}

96
src/clients/client.ts Normal file
View File

@ -0,0 +1,96 @@
import { version } from "../constants";
import { LdbClientV2 } from "./LdbClientV2";
import { LocationReferenceClientV2 } from "./LocationReferenceClientV2";
import { MiscClientV2 } from "./MiscClientV2";
import { PisClientV2 } from "./PisClientV2";
import { TrainClientV2 } from "./TrainClientV2";
import { UserClientV2 } from "./UserClientV2";
import { ApiError, NetworkError } from "../errors";
export class BaseOwlBoardClient {
protected baseUrl: string;
protected apiKey: string;
protected headers: Record<string, string>;
constructor(baseUrl: string, apiKey: string) {
this.baseUrl = baseUrl;
this.apiKey = apiKey;
this.headers = {
"Content-Type": "application/json",
"Accept": "application/json",
"User-Agent": `OwlBoardTS/v${version}`,
"uuid": this.apiKey,
};
}
async makeRequest<T>(method: string, path: string, body?: any): Promise<T> {
const url = `${this.baseUrl}${path}`;
const options: RequestInit = {
method,
headers: this.headers,
body: body ? JSON.stringify(body) : undefined,
};
let attempt = 0;
const retries = 3;
const delay = 500;
const retryableStatusCodes = [408, 500, 502, 503, 504];
while (attempt < retries) {
try {
const response = await fetch(url, options);
if (!response.ok) {
const responseBody = await response.text();
const apiError = new ApiError("API Request Failed", response.status, responseBody);
if (retryableStatusCodes.includes(response.status)) {
attempt++;
if (attempt >= retries) { // Max attempts reached
throw apiError;
}
// Wait before retrying
await new Promise(res => setTimeout(res, delay));
continue; // Retry
} else { // Non-retryable status codes
throw apiError;
}
}
return (await response.json()) as T;
} catch (error) {
if (error instanceof TypeError) { // Indicates network errors with fetch
if (attempt < retries) {
attempt ++;
await new Promise(res => setTimeout(res, delay));
continue;
}
// Throw network error if max tries reached handling TypeError
throw new NetworkError("Network request failed");
};
throw error; // Rethrow any other error
};
}
// This code should never be reached.
throw new Error("Unexpected error after multiple retries.")
}
}
export class OwlBoardClientV2 extends BaseOwlBoardClient {
pis: PisClientV2;
train: TrainClientV2;
user: UserClientV2;
locationReference: LocationReferenceClientV2;
misc: MiscClientV2;
ldb: LdbClientV2;
constructor(baseUrl: string, apiKey: string) {
super(baseUrl, apiKey);
this.pis = new PisClientV2(this);
this.train = new TrainClientV2(this);
this.user = new UserClientV2(this);
this.locationReference = new LocationReferenceClientV2(this);
this.misc = new MiscClientV2(this);
this.ldb = new LdbClientV2(this);
}
}

1
src/constants.ts Normal file
View File

@ -0,0 +1 @@
export const version: string = "0.0.1"

28
src/errors.ts Normal file
View File

@ -0,0 +1,28 @@
export class ApiError extends Error {
statusCode: number;
responseBody: string;
constructor(message: string, statusCode: number, responseBody: string) {
super(message);
this.name = 'ApiError';
this.statusCode = statusCode;
this.responseBody = responseBody;
Object.setPrototypeOf(this, ApiError.prototype);
}
}
export class ValidationError extends Error {
constructor(message: string) {
super(message);
this.name = 'ValidationError';
Object.setPrototypeOf(this, ValidationError.prototype);
}
}
export class NetworkError extends Error {
constructor(message: string) {
super(message);
this.name = 'NetworkError';
Object.setPrototypeOf(this, NetworkError.prototype);
}
}

View File

@ -15,3 +15,5 @@ OwlBoardTS: An OwlBoard API Client written in TypeScript
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
export { OwlBoardClientV2 } from './clients/client'

View File

@ -107,6 +107,6 @@
// "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */
"skipLibCheck": true /* Skip type checking all .d.ts files. */
},
"include": ["src/**/*"],
"include": ["src/**/*.ts"],
"exclude": ["node_modules"],
}