* download CAR file using AtpAgent instead of building URL * add loader icon on download car button * actually save to disk on android * style nits * bottom margin nit * localize toast * remove fallback so back button works correctly * keep throwing an error if mime type isn't used * be more explicit with toasts * send errors to sentry when encountered --------- Co-authored-by: Hailey <me@haileyok.com>
85 lines
2.4 KiB
TypeScript
85 lines
2.4 KiB
TypeScript
import RNFS from 'react-native-fs'
|
|
import {BskyAgent, jsonToLex, stringifyLex} from '@atproto/api'
|
|
|
|
const GET_TIMEOUT = 15e3 // 15s
|
|
const POST_TIMEOUT = 60e3 // 60s
|
|
|
|
export function doPolyfill() {
|
|
BskyAgent.configure({fetch: fetchHandler})
|
|
}
|
|
|
|
interface FetchHandlerResponse {
|
|
status: number
|
|
headers: Record<string, string>
|
|
body: any
|
|
}
|
|
|
|
async function fetchHandler(
|
|
reqUri: string,
|
|
reqMethod: string,
|
|
reqHeaders: Record<string, string>,
|
|
reqBody: any,
|
|
): Promise<FetchHandlerResponse> {
|
|
const reqMimeType = reqHeaders['Content-Type'] || reqHeaders['content-type']
|
|
if (reqMimeType && reqMimeType.startsWith('application/json')) {
|
|
reqBody = stringifyLex(reqBody)
|
|
} else if (
|
|
typeof reqBody === 'string' &&
|
|
(reqBody.startsWith('/') || reqBody.startsWith('file:'))
|
|
) {
|
|
if (reqBody.endsWith('.jpeg') || reqBody.endsWith('.jpg')) {
|
|
// HACK
|
|
// React native has a bug that inflates the size of jpegs on upload
|
|
// we get around that by renaming the file ext to .bin
|
|
// see https://github.com/facebook/react-native/issues/27099
|
|
// -prf
|
|
const newPath = reqBody.replace(/\.jpe?g$/, '.bin')
|
|
await RNFS.moveFile(reqBody, newPath)
|
|
reqBody = newPath
|
|
}
|
|
// NOTE
|
|
// React native treats bodies with {uri: string} as file uploads to pull from cache
|
|
// -prf
|
|
reqBody = {uri: reqBody}
|
|
}
|
|
|
|
const controller = new AbortController()
|
|
const to = setTimeout(
|
|
() => controller.abort(),
|
|
reqMethod === 'post' ? POST_TIMEOUT : GET_TIMEOUT,
|
|
)
|
|
|
|
const res = await fetch(reqUri, {
|
|
method: reqMethod,
|
|
headers: reqHeaders,
|
|
body: reqBody,
|
|
signal: controller.signal,
|
|
})
|
|
|
|
const resStatus = res.status
|
|
const resHeaders: Record<string, string> = {}
|
|
res.headers.forEach((value: string, key: string) => {
|
|
resHeaders[key] = value
|
|
})
|
|
const resMimeType = resHeaders['Content-Type'] || resHeaders['content-type']
|
|
let resBody
|
|
if (resMimeType) {
|
|
if (resMimeType.startsWith('application/json')) {
|
|
resBody = jsonToLex(await res.json())
|
|
} else if (resMimeType.startsWith('text/')) {
|
|
resBody = await res.text()
|
|
} else if (resMimeType === 'application/vnd.ipld.car') {
|
|
resBody = await res.arrayBuffer()
|
|
} else {
|
|
throw new Error('Non-supported mime type')
|
|
}
|
|
}
|
|
|
|
clearTimeout(to)
|
|
|
|
return {
|
|
status: resStatus,
|
|
headers: resHeaders,
|
|
body: resBody,
|
|
}
|
|
}
|