Bluesky app fork with some witchin' additions 💫
witchsky.app
bluesky
fork
client
1import {type Insets, Platform} from 'react-native'
2import {type AppBskyActorDefs, BSKY_LABELER_DID} from '@atproto/api'
3
4import {type ProxyHeaderValue} from '#/state/session/agent'
5import {
6 BLUESKY_PROXY_DID,
7 CHAT_PROXY_DID,
8 ENV_APPVIEW_DID_PROXY,
9 ENV_PUBLIC_BSKY_SERVICE,
10} from '#/env'
11export const LOCAL_DEV_SERVICE =
12 Platform.OS === 'android' ? 'http://10.0.2.2:2583' : 'http://localhost:2583'
13export const STAGING_SERVICE = 'https://staging.bsky.dev'
14export const BSKY_SERVICE = 'https://bsky.social'
15export const BSKY_SERVICE_DID = 'did:web:bsky.social'
16export const PUBLIC_BSKY_SERVICE =
17 ENV_PUBLIC_BSKY_SERVICE || 'https://public.api.bsky.app'
18export const DEFAULT_SERVICE = BSKY_SERVICE
19export const HELP_DESK_URL = `https://tangled.org/jollywhoppers.com/witchsky.app/`
20export const EMBED_SERVICE = 'https://embed.bsky.app'
21export const EMBED_SCRIPT = `${EMBED_SERVICE}/static/embed.js`
22export const BSKY_DOWNLOAD_URL = 'https://bsky.app/download'
23export const APPVIEW_DID_PROXY = ENV_APPVIEW_DID_PROXY
24export const STARTER_PACK_MAX_SIZE = 150
25export const CARD_ASPECT_RATIO = 1200 / 630
26
27// HACK
28// Yes, this is exactly what it looks like. It's a hard-coded constant
29// reflecting the number of new users in the last week. We don't have
30// time to add a route to the servers for this so we're just going to hard
31// code and update this number with each release until we can get the
32// server route done.
33// -prf
34export const JOINED_THIS_WEEK = 560000 // estimate as of 12/18/24
35
36export const DISCOVER_DEBUG_DIDS: Record<string, true> = {
37 'did:plc:oisofpd7lj26yvgiivf3lxsi': true, // hailey.at
38 'did:plc:p2cp5gopk7mgjegy6wadk3ep': true, // samuel.bsky.team
39 'did:plc:ragtjsm2j2vknwkz3zp4oxrd': true, // pfrazee.com
40 'did:plc:vpkhqolt662uhesyj6nxm7ys': true, // why.bsky.team
41 'did:plc:3jpt2mvvsumj2r7eqk4gzzjz': true, // esb.lol
42 'did:plc:vjug55kidv6sye7ykr5faxxn': true, // emilyliu.me
43 'did:plc:tgqseeot47ymot4zro244fj3': true, // iwsmith.bsky.social
44 'did:plc:2dzyut5lxna5ljiaasgeuffz': true, // darrin.bsky.team
45}
46
47const BASE_FEEDBACK_FORM_URL = `${HELP_DESK_URL}`
48export function FEEDBACK_FORM_URL(_params: {
49 email?: string
50 handle?: string
51}): string {
52 return BASE_FEEDBACK_FORM_URL
53}
54
55export const MAX_DISPLAY_NAME = 64
56export const MAX_DESCRIPTION = 256
57
58export const MAX_GRAPHEME_LENGTH = 300
59
60export const MAX_DRAFT_GRAPHEME_LENGTH = 1000
61
62export const MAX_DM_GRAPHEME_LENGTH = 1000
63
64// Recommended is 100 per: https://www.w3.org/WAI/GL/WCAG20/tests/test3.html
65// but increasing limit per user feedback
66export const MAX_ALT_TEXT = 2000
67export const DEFAULT_ALT_TEXT_AI_MODEL = 'google/gemma-4-26b-a4b-it:free'
68export const DEFAULT_ALT_TEXT_AI_PROMPT =
69 'Write alt text for this image. Be concise — 1-2 sentences for simple images. If the image contains readable text, transcribe it rather than describing it. Only describe what you can clearly see; do not guess at names or details.'
70
71export const MAX_REPORT_REASON_GRAPHEME_LENGTH = 2000
72
73export function IS_TEST_USER(handle?: string) {
74 return handle && handle?.endsWith('.test')
75}
76
77export function IS_PROD_SERVICE(url?: string) {
78 return url && url !== STAGING_SERVICE && !url.startsWith(LOCAL_DEV_SERVICE)
79}
80
81export const PROD_DEFAULT_FEED = (rkey: string) =>
82 `at://did:plc:z72i7hdynmk6r22z27h6tvur/app.bsky.feed.generator/${rkey}`
83
84export const STAGING_DEFAULT_FEED = (rkey: string) =>
85 `at://did:plc:yofh3kx63drvfljkibw5zuxo/app.bsky.feed.generator/${rkey}`
86
87export const PROD_FEEDS = [
88 `feedgen|${PROD_DEFAULT_FEED('whats-hot')}`,
89 `feedgen|${PROD_DEFAULT_FEED('thevids')}`,
90]
91
92export const STAGING_FEEDS = [
93 `feedgen|${STAGING_DEFAULT_FEED('whats-hot')}`,
94 `feedgen|${STAGING_DEFAULT_FEED('thevids')}`,
95]
96
97export const POST_IMG_MAX = {
98 width: 2000,
99 height: 2000,
100 size: 1000000,
101}
102
103export const STAGING_LINK_META_PROXY =
104 'https://cardyb.staging.bsky.dev/v1/extract?url='
105
106export const PROD_LINK_META_PROXY = 'https://cardyb.bsky.app/v1/extract?url='
107
108export function LINK_META_PROXY(serviceUrl: string) {
109 if (IS_PROD_SERVICE(serviceUrl)) {
110 return PROD_LINK_META_PROXY
111 }
112
113 return STAGING_LINK_META_PROXY
114}
115
116export const STATUS_PAGE_URL = 'https://status.bsky.app/'
117
118// Hitslop constants
119export const createHitslop = (size: number): Insets => ({
120 top: size,
121 left: size,
122 bottom: size,
123 right: size,
124})
125export const HITSLOP_10 = createHitslop(10)
126export const HITSLOP_20 = createHitslop(20)
127export const HITSLOP_30 = createHitslop(30)
128export const LANG_DROPDOWN_HITSLOP = {top: 10, bottom: 10, left: 4, right: 4}
129export const BACK_HITSLOP = HITSLOP_30
130export const MAX_POST_LINES = 25
131
132export const BSKY_APP_ACCOUNT_DID = 'did:plc:z72i7hdynmk6r22z27h6tvur'
133
134export const BSKY_FEED_OWNER_DIDS = [
135 BSKY_APP_ACCOUNT_DID,
136 'did:plc:vpkhqolt662uhesyj6nxm7ys',
137 'did:plc:q6gjnaw2blty4crticxkmujt',
138]
139
140export const DISCOVER_FEED_URI =
141 'at://did:plc:z72i7hdynmk6r22z27h6tvur/app.bsky.feed.generator/whats-hot'
142export const VIDEO_FEED_URI =
143 'at://did:plc:z72i7hdynmk6r22z27h6tvur/app.bsky.feed.generator/thevids'
144export const STAGING_VIDEO_FEED_URI =
145 'at://did:plc:yofh3kx63drvfljkibw5zuxo/app.bsky.feed.generator/thevids'
146export const VIDEO_FEED_URIS = [VIDEO_FEED_URI, STAGING_VIDEO_FEED_URI]
147export const DISCOVER_SAVED_FEED = {
148 type: 'feed',
149 value: DISCOVER_FEED_URI,
150 pinned: true,
151}
152export const TIMELINE_SAVED_FEED = {
153 type: 'timeline',
154 value: 'following',
155 pinned: true,
156}
157export const VIDEO_SAVED_FEED = {
158 type: 'feed',
159 value: VIDEO_FEED_URI,
160 pinned: true,
161}
162
163export const RECOMMENDED_SAVED_FEEDS: Pick<
164 AppBskyActorDefs.SavedFeed,
165 'type' | 'value' | 'pinned'
166>[] = [DISCOVER_SAVED_FEED, TIMELINE_SAVED_FEED]
167
168export const KNOWN_SHUTDOWN_FEEDS = [
169 'at://did:plc:wqowuobffl66jv3kpsvo7ak4/app.bsky.feed.generator/the-algorithm', // for you by skygaze
170]
171
172export const GIF_SERVICE = 'https://gifs.bsky.app'
173
174export const GIF_SEARCH = (params: string) =>
175 `${GIF_SERVICE}/tenor/v2/search?${params}`
176export const GIF_FEATURED = (params: string) =>
177 `${GIF_SERVICE}/tenor/v2/featured?${params}`
178
179export const GIF_KLIPY_SEARCH = (params: string) =>
180 `${GIF_SERVICE}/klipy/v2/search?${params}`
181export const GIF_KLIPY_FEATURED = (params: string) =>
182 `${GIF_SERVICE}/klipy/v2/featured?${params}`
183
184export const MAX_LABELERS = 20
185
186export const VIDEO_SERVICE = 'https://video.bsky.app'
187export const VIDEO_SERVICE_DID = 'did:web:video.bsky.app'
188
189export const VIDEO_MAX_DURATION_MS = 3 * 60 * 1000 // 3 minutes in milliseconds
190/**
191 * Maximum size of a video in megabytes, _not_ mebibytes. Backend uses
192 * ISO megabytes.
193 */
194export const VIDEO_MAX_SIZE = 1000 * 1000 * 100 // 100mb
195
196export const SUPPORTED_MIME_TYPES = [
197 'video/mp4',
198 'video/mpeg',
199 'video/webm',
200 'video/quicktime',
201 'image/gif',
202] as const
203
204export type SupportedMimeTypes = (typeof SUPPORTED_MIME_TYPES)[number]
205
206export const EMOJI_REACTION_LIMIT = 5
207
208export const urls = {
209 website: {
210 blog: {
211 findFriendsAnnouncement:
212 'https://bsky.social/about/blog/12-16-2025-find-friends',
213 initialVerificationAnnouncement: `https://bsky.social/about/blog/04-21-2025-verification`,
214 searchTipsAndTricks: 'https://bsky.social/about/blog/05-31-2024-search',
215 },
216 support: {
217 findFriendsPrivacyPolicy:
218 'https://bsky.social/about/support/find-friends-privacy-policy',
219 },
220 },
221}
222
223export const PUBLIC_APPVIEW = 'https://api.bsky.app'
224export const PUBLIC_APPVIEW_DID = 'did:web:api.bsky.app'
225export const PUBLIC_STAGING_APPVIEW_DID = 'did:web:api.staging.bsky.dev'
226
227export const DEV_ENV_APPVIEW = `http://localhost:2584` // always the same
228export const DEV_ENV_APPVIEW_DID = `did:plc:dw4kbjf5mn7nhenabiqpkyh3` // always the same
229
230// temp hack for e2e - esb
231export const BLUESKY_PROXY_HEADER = {
232 value: `${BLUESKY_PROXY_DID}#bsky_appview`,
233 get() {
234 return this.value as ProxyHeaderValue
235 },
236 set(value: string) {
237 this.value = value
238 },
239}
240
241export const DM_SERVICE_HEADERS = {
242 'atproto-proxy': `${CHAT_PROXY_DID}#bsky_chat`,
243}
244
245export const BLUESKY_MOD_SERVICE_HEADERS = {
246 'atproto-proxy': `${BSKY_LABELER_DID}#atproto_labeler`,
247}
248
249export const BLUESKY_NOTIF_SERVICE_HEADERS = {
250 'atproto-proxy': `${BLUESKY_PROXY_DID}#bsky_notif`,
251}
252
253export const webLinks = {
254 tos: `https://bsky.social/about/support/tos`,
255 privacy: `https://bsky.social/about/support/privacy-policy`,
256 community: `https://bsky.social/about/support/community-guidelines`,
257 communityDeprecated: `https://bsky.social/about/support/community-guidelines-deprecated`,
258}
259
260export const DOH_ENDPOINT = 'https://cloudflare-dns.com/dns-query'