repos / starfx

a micro-mvc framework for react apps
git clone https://github.com/neurosnap/starfx.git

starfx / mdw
Eric Bower  ·  2024-08-25

fetch.ts

  1import { sleep } from "../deps.ts";
  2import { safe } from "../fx/mod.ts";
  3import type { FetchCtx, FetchJsonCtx } from "../query/mod.ts";
  4import { isObject, noop } from "../query/util.ts";
  5import type { Next } from "../types.ts";
  6
  7/**
  8 * This middleware converts the name provided to {@link createApi}
  9 * into `url` and `method` for the fetch request.
 10 */
 11export function* nameParser<Ctx extends FetchJsonCtx = FetchJsonCtx>(
 12  ctx: Ctx,
 13  next: Next,
 14) {
 15  const httpMethods = [
 16    "get",
 17    "head",
 18    "post",
 19    "put",
 20    "delete",
 21    "connect",
 22    "options",
 23    "trace",
 24    "patch",
 25  ];
 26
 27  const options = ctx.payload || {};
 28  if (!isObject(options)) {
 29    yield* next();
 30    return;
 31  }
 32
 33  let url = Object.keys(options).reduce((acc, key) => {
 34    return acc.replace(`:${key}`, options[key]);
 35  }, ctx.name);
 36
 37  let method = "";
 38  httpMethods.forEach((curMethod) => {
 39    const pattern = new RegExp(`\\s*\\[` + curMethod + `\\]\\s*\\w*`, "i");
 40    const tmpUrl = url.replace(pattern, "");
 41    if (tmpUrl.length !== url.length) {
 42      method = curMethod.toLocaleUpperCase();
 43    }
 44    url = tmpUrl;
 45  }, url);
 46
 47  if (ctx.req().url === "") {
 48    ctx.request = ctx.req({ url });
 49  }
 50
 51  if (method) {
 52    ctx.request = ctx.req({ method });
 53  }
 54
 55  yield* next();
 56}
 57
 58/**
 59 * Automatically sets `content-type` to `application/json` when
 60 * that header is not already present.
 61 */
 62export function* headers<CurCtx extends FetchCtx = FetchCtx>(
 63  ctx: CurCtx,
 64  next: Next,
 65) {
 66  if (!ctx.request) {
 67    yield* next();
 68    return;
 69  }
 70
 71  const cur = ctx.req();
 72  if (!Object.hasOwn(cur.headers, "Content-Type")) {
 73    ctx.request = ctx.req({
 74      headers: { "Content-Type": "application/json" },
 75    });
 76  }
 77
 78  yield* next();
 79}
 80
 81/**
 82 * This middleware takes the `ctx.response` and sets `ctx.json` to the body representation
 83 * requested.  It uses the `ctx.bodyType` property to determine how to represent the body.
 84 * The default is set to `json` which calls `Response.json()`.
 85 *
 86 * @example
 87 * ```ts
 88 * const fetchUsers = api.get('/users', function*(ctx, next) {
 89 *  ctx.bodyType = 'text'; // calls Response.text();
 90 *  yield next();
 91 * })
 92 * ```
 93 */
 94export function* json<CurCtx extends FetchJsonCtx = FetchJsonCtx>(
 95  ctx: CurCtx,
 96  next: Next,
 97) {
 98  if (!ctx.response) {
 99    yield* next();
100    return;
101  }
102
103  if (ctx.response.status === 204) {
104    ctx.json = {
105      ok: true,
106      value: {},
107    };
108    yield* next();
109    return;
110  }
111
112  const data = yield* safe(() => {
113    const resp = ctx.response;
114    if (!resp) throw new Error("response is falsy");
115    return resp[ctx.bodyType]();
116  });
117
118  if (data.ok) {
119    if (ctx.response.ok) {
120      ctx.json = {
121        ok: true,
122        value: data.value,
123      };
124    } else {
125      ctx.json = {
126        ok: false,
127        error: data.value,
128      };
129    }
130  } else {
131    const dta = { message: data.error.message };
132    ctx.json = {
133      ok: false,
134      error: dta,
135    };
136  }
137
138  yield* next();
139}
140
141/*
142 * This middleware takes the `baseUrl` provided to {@link mdw.fetch} and combines it
143 * with the url from `ctx.request.url`.
144 */
145export function composeUrl<CurCtx extends FetchJsonCtx = FetchJsonCtx>(
146  baseUrl = "",
147) {
148  return function* (ctx: CurCtx, next: Next) {
149    const req = ctx.req();
150    ctx.request = ctx.req({ url: `${baseUrl}${req.url}` });
151    yield* next();
152  };
153}
154
155/**
156 * If there's a slug inside the ctx.name (which is the URL segement in this case)
157 * and there is *not* a corresponding truthy value in the payload, then that means
158 * the user has an empty value (e.g. empty string) which means we want to abort the
159 * fetch request.
160 *
161 * e.g. `ctx.name = "/apps/:id"` with `payload = { id: '' }`
162 *
163 * Ideally the action wouldn't have been dispatched at all but that is *not* a
164 * gaurantee we can make here.
165 */
166export function* payload<CurCtx extends FetchJsonCtx = FetchJsonCtx>(
167  ctx: CurCtx,
168  next: Next,
169) {
170  const payload = ctx.payload;
171  if (!payload) {
172    yield* next();
173    return;
174  }
175
176  const keys = Object.keys(payload);
177  for (let i = 0; i < keys.length; i += 1) {
178    const key = keys[i];
179    if (!ctx.name.includes(`:${key}`)) {
180      continue;
181    }
182
183    const val = payload[key];
184    if (!val) {
185      const data =
186        `found :${key} in endpoint name (${ctx.name}) but payload has falsy value (${val})`;
187      ctx.json = {
188        ok: false,
189        error: data,
190      };
191      return;
192    }
193  }
194
195  yield* next();
196}
197
198/*
199 * This middleware simply checks if `ctx.response` already contains a
200 * truthy value, and if it does, bail out of the middleware stack.
201 */
202export function response<CurCtx extends FetchCtx = FetchCtx>(
203  response?: Response,
204) {
205  return function* responseMdw(
206    ctx: CurCtx,
207    next: Next,
208  ) {
209    if (response) {
210      ctx.response = response;
211    }
212    yield* next();
213  };
214}
215
216/*
217 * This middleware makes the `fetch` http request using `ctx.request` and
218 * assigns the response to `ctx.response`.
219 */
220export function* request<CurCtx extends FetchCtx = FetchCtx>(
221  ctx: CurCtx,
222  next: Next,
223) {
224  // if there is already a response then we want to bail so we don't
225  // override it.
226  if (ctx.response) {
227    yield* next();
228    return;
229  }
230
231  const { url, ...req } = ctx.req();
232  const request = new Request(url, req);
233  const result = yield* safe(fetch(request));
234  if (result.ok) {
235    ctx.response = result.value;
236  } else {
237    throw result.error;
238  }
239  yield* next();
240}
241
242function backoffExp(attempt: number): number {
243  if (attempt > 5) return -1;
244  // 1s, 1s, 1s, 2s, 4s
245  return Math.max(2 ** attempt * 125, 1000);
246}
247
248/**
249 * This middleware will retry failed `Fetch` request if `response.ok` is `false`.
250 * It accepts a backoff function to determine how long to continue retrying.
251 * The default is an exponential backoff {@link backoffExp} where the minimum is
252 * 1sec between attempts and it'll reach 4s between attempts at the end with a
253 * max of 5 attempts.
254 *
255 * An example backoff:
256 * @example
257 * ```ts
258 *  // Any value less than 0 will stop the retry middleware.
259 *  // Each attempt will wait 1s
260 *  const backoff = (attempt: number) => {
261 *    if (attempt > 5) return -1;
262 *    return 1000;
263 *  }
264 *
265 * const api = createApi();
266 * api.use(mdw.api());
267 * api.use(api.routes());
268 * api.use(mdw.fetch());
269 *
270 * const fetchUsers = api.get('/users', [
271 *  function*(ctx, next) {
272 *    // ...
273 *    yield next();
274 *  },
275 *  // fetchRetry should be after your endpoint function because
276 *  // the retry middleware will update `ctx.json` before it reaches
277 *  // your middleware
278 *  fetchRetry(backoff),
279 * ])
280 * ```
281 */
282export function fetchRetry<CurCtx extends FetchJsonCtx = FetchJsonCtx>(
283  backoff: (attempt: number) => number = backoffExp,
284) {
285  return function* (ctx: CurCtx, next: Next) {
286    yield* next();
287
288    if (!ctx.response) {
289      return;
290    }
291
292    if (ctx.response.ok) {
293      return;
294    }
295
296    let attempt = 1;
297    let waitFor = backoff(attempt);
298    while (waitFor >= 1) {
299      yield* sleep(waitFor);
300      // reset response so `request` mdw actually runs
301      ctx.response = null;
302      yield* safe(() => request(ctx, noop));
303      yield* safe(() => json(ctx, noop));
304
305      if (ctx.response && (ctx.response as Response).ok) {
306        return;
307      }
308
309      attempt += 1;
310      waitFor = backoff(attempt);
311    }
312  };
313}