expo-auth-session
Version:
Expo module for browser-based authentication
62 lines (51 loc) • 1.79 kB
text/typescript
export type Headers = Record<string, string> & {
'Content-Type': string;
Authorization?: string;
Accept?: string;
};
export type FetchRequest = {
headers?: Headers;
body?: Record<string, string>;
dataType?: string;
method?: string;
};
export async function requestAsync<T>(requestUrl: string, fetchRequest: FetchRequest): Promise<T> {
const url = new URL(requestUrl);
const headers: Record<string, string> = {};
const request: RequestInit = {
body: undefined,
method: fetchRequest.method,
mode: 'cors',
headers,
};
const isJsonDataType = fetchRequest.dataType?.toLowerCase() === 'json';
if (fetchRequest.headers) {
for (const i in fetchRequest.headers) {
if (i in fetchRequest.headers) {
headers[i] = fetchRequest.headers[i];
}
}
}
if (fetchRequest.body) {
if (fetchRequest.method?.toUpperCase() === 'POST') {
request.body = new URLSearchParams(fetchRequest.body).toString();
} else {
for (const key of Object.keys(fetchRequest.body)) {
url.searchParams.append(key, fetchRequest.body[key]);
}
}
}
if (isJsonDataType && !headers.Accept && !headers.accept) {
// NOTE: Github authentication will return XML if this includes the standard `*/*`
headers['Accept'] = 'application/json, text/javascript; q=0.01';
}
// Fix a problem with React Native `URL` causing a trailing slash to be added.
const correctedUrl = url.toString().replace(/\/$/, '');
const response = await fetch(correctedUrl, request);
const contentType = response.headers.get('content-type');
if (isJsonDataType || contentType?.includes('application/json')) {
return response.json();
}
// @ts-ignore: Type 'string' is not assignable to type 'T'.
return response.text();
}