* Remove the hackcheck for upgrades * Rename the PostEmbeds folder to match the codebase style * Updates to latest lex refactor * Update to use new bsky agent * Update to use api package's richtext library * Switch to upsertProfile * Add TextEncoder/TextDecoder polyfill * Add Intl.Segmenter polyfill * Update composer to calculate lengths by grapheme * Fix detox * Fix login in e2e * Create account e2e passing * Implement an e2e mocking framework * Don't use private methods on mobx models as mobx can't track them * Add tooling for e2e-specific builds and add e2e media-picker mock * Add some tests and fix some bugs around profile editing * Add shell tests * Add home screen tests * Add thread screen tests * Add tests for other user profile screens * Add search screen tests * Implement profile imagery change tools and tests * Update to new embed behaviors * Add post tests * Fix to profile-screen test * Fix session resumption * Update web composer to new api * 1.11.0 * Fix pagination cursor parameters * Add quote posts to notifications * Fix embed layouts * Remove youtube inline player and improve tap handling on link cards * Reset minimal shell mode on all screen loads and feed swipes (close #299) * Update podfile.lock * Improve post notfound UI (close #366) * Bump atproto packages
83 lines
2.3 KiB
TypeScript
83 lines
2.3 KiB
TypeScript
import {BskyAgent, stringifyLex, jsonToLex} from '@atproto/api'
|
|
import RNFS from 'react-native-fs'
|
|
|
|
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: ArrayBuffer | undefined
|
|
}
|
|
|
|
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 {
|
|
throw new Error('TODO: non-textual response body')
|
|
}
|
|
}
|
|
|
|
clearTimeout(to)
|
|
|
|
return {
|
|
status: resStatus,
|
|
headers: resHeaders,
|
|
body: resBody,
|
|
}
|
|
}
|