api-ape 3.0.0 → 3.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../utils/messageHash.js", "../utils/jss.js", "../client/transports/streaming.js", "../client/connectSocket.js", "../client/index.js"],
4
- "sourcesContent": ["const alphabet = \"0123456789ABCDEFGHJKMNPQRSTVWXYZ\"\n/*\nfunction charValue(char){\n return alphabet.indexOf(char.toUpperCase())\n} // END charValue\n\nfunction fromBase32(b32){\n if (0 === b32.length) {\n return 0\n }\n return charValue(b32.slice(-1)) + fromBase32(b32.slice(0,-1)) * 32\n} // END fromBase32\n*/\nfunction toBase32 (n){\n const remainder = Math.floor(n/32)\n const current = n % 32\n if (0 === remainder) {\n return alphabet[current]\n }\n return toBase32(remainder)+alphabet[current]\n} // END toBase32\n\nfunction jenkinsOneAtATimeHash(keyString){\n \n var hash = 0\n \n for (var charIndex = 0; charIndex < keyString.length; ++charIndex)\n {\n hash += keyString.charCodeAt(charIndex);\n hash += hash << 10;\n hash ^= hash >> 6;\n }\n hash += hash << 3;\n hash ^= hash >> 11;\n //4,294,967,295 is FFFFFFFF, the maximum 32 bit unsigned integer value, used here as a mask.\n return (((hash + (hash << 15)) & 4294967295) >>> 0)\n} // END jenkinsOneAtATimeHash\n\nfunction messageHash(messageSt){\n return toBase32(jenkinsOneAtATimeHash(messageSt))\n} // END messageHash\n\nmodule.exports = messageHash", "//JsonSuperSet\n\n// TODO: add tests\n// check for any repeated ref not just cyclical references\n// support nasted array a<![,,[D]]>:[\"a\",\"b\",[Date]]\n// support array for the same type a<![*D]>:[Date,Date,Date]\n\nfunction encode(obj) {\n const tagLookup = {\n '[object RegExp]': 'R',\n '[object Date]': 'D',\n '[object Error]': 'E',\n \"[object Undefined]\": 'U',\n \"[object Map]\": 'M',\n \"[object Set]\": 'S',\n };\n const visited = new WeakMap();\n\n function encodeValue(value, path = '') {\n const type = typeof value;\n const tag = tagLookup[Object.prototype.toString.call(value)];\n // console.log({tag,value,path})\n if (tag !== undefined) {\n if ('D' === tag) return [tag, value.valueOf()];\n if ('E' === tag) return [tag, [value.name, value.message, value.stack]];\n if ('R' === tag) return [tag, value.toString()];\n if ('U' === tag) return [tag, null];\n if ('S' === tag) return [tag, Array.from(value)];\n if ('M' === tag) return [tag, Object.fromEntries(value)];\n\n return [tag, JSON.stringify(value)];\n } else if (type === 'object' && value !== null) {\n /*if (value.$ID) {\n return ['', value.$ID];\n }*/\n if (visitedEncode.has(value)) {\n return ['P', visitedEncode.get(value)];\n }\n visitedEncode.set(value, path);\n const isArray = Array.isArray(value);\n // keep index with undefined in Array\n const keys = isArray ? Array.from(Array(value.length).keys()) : Object.keys(value);\n const result = isArray ? [] : {};\n const typesFound = [];\n for (let i = 0; i < keys.length; i++) {\n const key = keys[i];\n const [t, v] = encodeValue(value[key], key);\n // console.log([t, v])\n if (isArray) {\n typesFound.push(t);\n result.push(v);\n // remove key with undefined from Objects\n } else if (value[key] !== undefined) {\n result[key + (t ? `<!${t}>` : '')] = v;\n }\n }\n\n visited.delete(value);\n if (isArray && typesFound.find((t) => !!t)) {\n return [`[${typesFound.join()}]`, result];\n }\n return ['', result];\n } else {\n return ['', value];\n }\n } // END encodeValue\n\n let keys = [];\n // console.log(obj)\n if (Array.isArray(obj)) {\n keys = Array.from(Array(obj.length).keys())\n } else {\n keys = Object.keys(obj);\n }\n\n // Track root object to handle self-references\n const visitedEncode = new WeakMap();\n visitedEncode.set(obj, []); // Root path is empty array\n\n function encodeValueWithVisited(value, path = []) {\n const type = typeof value;\n const tag = tagLookup[Object.prototype.toString.call(value)];\n if (tag !== undefined) {\n if ('D' === tag) return [tag, value.valueOf()];\n if ('E' === tag) return [tag, [value.name, value.message, value.stack]];\n if ('R' === tag) return [tag, value.toString()];\n if ('U' === tag) return [tag, null];\n if ('S' === tag) return [tag, Array.from(value)];\n if ('M' === tag) return [tag, Object.fromEntries(value)];\n return [tag, JSON.stringify(value)];\n } else if (type === 'object' && value !== null) {\n if (visitedEncode.has(value)) {\n return ['P', visitedEncode.get(value)]; // Return array path\n }\n visitedEncode.set(value, path);\n const isArray = Array.isArray(value);\n const objKeys = isArray ? Array.from(Array(value.length).keys()) : Object.keys(value);\n const result = isArray ? [] : {};\n const typesFound = [];\n for (let i = 0; i < objKeys.length; i++) {\n const key = objKeys[i];\n const [t, v] = encodeValueWithVisited(value[key], [...path, key]); // Append key to path array\n if (isArray) {\n typesFound.push(t);\n result.push(v);\n } else if (value[key] !== undefined) {\n result[key + (t ? `<!${t}>` : '')] = v;\n }\n }\n if (isArray && typesFound.find((t) => !!t)) {\n return [`[${typesFound.join()}]`, result];\n }\n return ['', result];\n } else {\n return ['', value];\n }\n }\n\n const result = {};\n for (let i = 0; i < keys.length; i++) {\n const key = keys[i];\n // remove key with undefined from Objects\n if (obj[key] !== undefined) {\n const [t, v] = encodeValueWithVisited(obj[key], [key]); // Start path with single key\n result[key + (t ? `<!${t}>` : '')] = v;\n }\n }\n return result;\n} // END encode\n\nfunction stringify(obj) {\n return JSON.stringify(encode(obj))\n}\n\n\nfunction parse(encoded) {\n return decode(JSON.parse(encoded))\n}\n\nfunction decode(data) {\n const result = {};\n const pointers2Res = [];\n const tagLookup = {\n R: (s) => new RegExp(s),\n D: (n) => new Date(n),\n P: function (sourceToPointAt, replaceAtThisPlace) {\n // Both paths are now arrays\n pointers2Res.push([sourceToPointAt, replaceAtThisPlace]);\n return null; // Placeholder, will be replaced by changeAttributeReference\n },\n E: ([name, message, stack]) => {\n let err;\n try {\n err = new global[name](message);\n if (err instanceof Error) err.stack = stack;\n else throw {};\n } catch (e) {\n err = new Error(message);\n err.name = name;\n err.stack = stack;\n }\n return err;\n },\n U: () => undefined,\n S: (a) => new Set(a),\n M: (o) => new Map(Object.entries(o))\n };\n const visited = new Map();\n\n function decodeValue(name, tag, val) {\n // this is now an array path\n const currentPath = Array.isArray(this) ? this : [];\n\n if (tag in tagLookup) {\n return tagLookup[tag](val, currentPath);\n } else if (Array.isArray(val)) {\n if (tag && tag.startsWith('[')) {\n const typeTags = tag.slice(1, -1).split(',');\n const res = [];\n for (let i = 0; i < val.length; i++) {\n // Pass path with array index appended\n const itemPath = [...currentPath, i];\n const decodedValue = decodeValue.call(\n itemPath,\n i.toString(),\n typeTags[i],\n val[i]\n );\n res.push(decodedValue);\n }\n return res;\n } else {\n const res = [];\n for (let i = 0; i < val.length; i++) {\n const decodedValue = decodeValue.call([...currentPath, i], '', '', val[i]);\n res.push(decodedValue);\n }\n return res;\n }\n } else if ('object' === typeof val && val !== null) {\n if (visited.has(val)) {\n return visited.get(val);\n }\n visited.set(val, {});\n const res = {};\n for (const key in val) {\n const [nam, t] = parseKeyWithTags(key);\n const decodedValue = decodeValue.call(\n [...currentPath, nam],\n nam,\n t,\n val[key]\n );\n res[nam] = decodedValue;\n }\n visited.set(val, res);\n return res;\n } else {\n return val;\n }\n } // END decodeValue\n\n function parseKeyWithTags(key) {\n const match = key.match(/(.+)(<!(.)>)/);\n if (match) {\n return [match[1], match[3]];\n }\n // Try multi-character tags like array types [,D,]\n const multiMatch = key.match(/(.+)(<!!(.+)>)/);\n if (multiMatch) {\n return [multiMatch[1], multiMatch[3]];\n }\n // Also handle array type tags that start with [\n const arrayMatch = key.match(/(.+)(<!\\[(.*)>)/);\n if (arrayMatch) {\n return [arrayMatch[1], '[' + arrayMatch[3]];\n }\n return [key, undefined];\n } // END parseKeyWithTags\n\n for (const key in data) {\n const [name, tag] = parseKeyWithTags(key);\n // Start with path containing just the key name\n result[name] = decodeValue.call([name], name, tag, data[key]);\n }\n pointers2Res.forEach(changeAttributeReference.bind(null, result));\n return result;\n} // END decode\n\nfunction changeAttributeReference(obj, [refPath, attrPath]) {\n // refPath and attrPath are now arrays, no splitting needed\n const refKeys = refPath || [];\n const attrKeys = attrPath || [];\n\n // Get the reference target by traversing refPath\n let ref = obj;\n for (let i = 0; i < refKeys.length; i++) {\n ref = ref[refKeys[i]];\n }\n\n // Get the parent of the attribute to set\n let attr = obj;\n for (let i = 0; i < attrKeys.length - 1; i++) {\n attr = attr[attrKeys[i]];\n }\n\n // Set the attribute to point to the reference\n attr[attrKeys[attrKeys.length - 1]] = ref;\n return obj;\n} // END changeAttributeReference\n\n\nmodule.exports = { parse, stringify, encode, decode };\n", "import jss from '../../utils/jss'\n\n/**\n * HTTP Streaming transport - fallback when WebSocket is blocked\n * Uses fetch + ReadableStream for receiving, POST for sending\n */\n\n/**\n * Get base URL for polling endpoints\n */\nfunction getPollUrl() {\n const hostname = window.location.hostname\n const localServers = [\"localhost\", \"127.0.0.1\", \"[::1]\"]\n const isLocal = localServers.includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (isLocal || (port !== 80 && port !== 443)) ? `:${port}` : \"\"\n\n return `${protocol}://${hostname}${portSuffix}/api/ape/poll`\n}\n\n/**\n * Parse JSON objects from a streaming buffer by counting braces\n * Handles strings containing braces correctly\n */\nfunction parseStreamBuffer(buffer) {\n const messages = []\n let start = -1\n let depth = 0\n let inString = false\n let escaped = false\n\n for (let i = 0; i < buffer.length; i++) {\n const char = buffer[i]\n\n if (escaped) {\n escaped = false\n continue\n }\n\n if (char === '\\\\' && inString) {\n escaped = true\n continue\n }\n\n if (char === '\"') {\n inString = !inString\n continue\n }\n\n if (inString) continue\n\n if (char === '{') {\n if (depth === 0) {\n start = i\n }\n depth++\n } else if (char === '}') {\n depth--\n if (depth === 0 && start !== -1) {\n const jsonStr = buffer.slice(start, i + 1)\n try {\n messages.push(jss.parse(jsonStr))\n } catch (e) {\n console.error('\uD83E\uDD8D Failed to parse stream message:', e)\n }\n start = -1\n }\n }\n }\n\n // Return remaining buffer (incomplete message)\n const remaining = start !== -1 ? buffer.slice(start) : ''\n return { messages, remaining }\n}\n\n/**\n * Create streaming transport instance\n */\nfunction createStreamingTransport() {\n let isActive = false\n let abortController = null\n let streamBuffer = ''\n let reconnectTimer = null\n\n // Callbacks\n let onMessage = () => { }\n let onOpen = () => { }\n let onClose = () => { }\n let onError = () => { }\n\n /**\n * Start the streaming connection\n */\n async function connect() {\n if (isActive) return\n\n isActive = true\n abortController = new AbortController()\n\n try {\n const response = await fetch(getPollUrl(), {\n method: 'GET',\n credentials: 'include',\n signal: abortController.signal,\n headers: {\n 'Accept': 'application/json'\n }\n })\n\n if (!response.ok) {\n throw new Error(`Stream connect failed: ${response.status}`)\n }\n\n onOpen()\n\n const reader = response.body.getReader()\n const decoder = new TextDecoder()\n\n async function read() {\n while (isActive) {\n try {\n const { done, value } = await reader.read()\n\n if (done) {\n // Stream ended - reconnect\n scheduleReconnect()\n return\n }\n\n streamBuffer += decoder.decode(value, { stream: true })\n const { messages, remaining } = parseStreamBuffer(streamBuffer)\n streamBuffer = remaining\n\n for (const msg of messages) {\n // Skip heartbeat messages\n if (msg.type === '__heartbeat__') continue\n onMessage(msg)\n }\n } catch (readErr) {\n if (readErr.name === 'AbortError') return\n console.error('\uD83E\uDD8D Stream read error:', readErr)\n scheduleReconnect()\n return\n }\n }\n }\n\n read()\n\n } catch (err) {\n if (err.name === 'AbortError') return\n\n console.error('\uD83E\uDD8D Stream connection error:', err)\n onError(err)\n scheduleReconnect()\n }\n }\n\n /**\n * Schedule reconnection with small delay\n */\n function scheduleReconnect() {\n if (!isActive) return\n\n if (reconnectTimer) {\n clearTimeout(reconnectTimer)\n }\n\n reconnectTimer = setTimeout(() => {\n if (isActive) {\n connect()\n }\n }, 500)\n }\n\n /**\n * Send a message via POST\n */\n async function send(type, data, createdAt) {\n const payload = {\n type,\n data,\n createdAt: new Date(createdAt)\n }\n\n const response = await fetch(getPollUrl(), {\n method: 'POST',\n credentials: 'include',\n headers: {\n 'Content-Type': 'application/json'\n },\n body: jss.stringify(payload)\n })\n\n if (!response.ok) {\n const error = await response.json().catch(() => ({ error: 'Unknown error' }))\n throw new Error(error.error || `Request failed: ${response.status}`)\n }\n\n const result = jss.parse(await response.text())\n return result.data\n }\n\n /**\n * Close the streaming connection\n */\n function close() {\n isActive = false\n\n if (reconnectTimer) {\n clearTimeout(reconnectTimer)\n reconnectTimer = null\n }\n\n if (abortController) {\n abortController.abort()\n abortController = null\n }\n\n streamBuffer = ''\n onClose()\n }\n\n return {\n connect,\n send,\n close,\n isConnected: () => isActive,\n set onMessage(fn) { onMessage = fn },\n set onOpen(fn) { onOpen = fn },\n set onClose(fn) { onClose = fn },\n set onError(fn) { onError = fn }\n }\n}\n\nexport { createStreamingTransport, getPollUrl }\n", "import messageHash from '../utils/messageHash'\nimport jss from '../utils/jss'\nimport { createStreamingTransport } from './transports/streaming'\n\nlet connect;\n\n// Connection state enum\nconst ConnectionState = {\n Offline: 'offline', // navigator.onLine = false\n Walled: 'walled', // Captive portal detected (ping failed)\n Disconnected: 'disconnected',\n Connecting: 'connecting',\n Connected: 'connected',\n Closing: 'closing'\n}\n\n// Connection state tracking - start with offline check\nlet connectionState = (typeof navigator !== 'undefined' && !navigator.onLine)\n ? ConnectionState.Offline\n : ConnectionState.Disconnected\nconst connectionChangeListeners = []\n\nfunction notifyConnectionChange(newState) {\n if (connectionState !== newState) {\n connectionState = newState\n connectionChangeListeners.forEach(fn => fn(newState))\n }\n}\n\n// Configuration\nlet configuredTransport = 'auto' // 'auto' | 'websocket' | 'polling'\n\n// Transport state\nlet currentTransport = null // 'websocket' | 'polling'\nlet streamingTransport = null\nlet wsRetryTimer = null\nlet networkCheckTimer = null\nconst WS_FALLBACK_TIMEOUT = 4000 // Time to wait for WS before fallback\nconst WS_RETRY_INTERVAL = 30000 // Retry WebSocket while in polling mode\nconst PING_TIMEOUT = 3000 // Timeout for ping check\nconst MAX_PING_CLOCK_SKEW = 60000 // Max allowed time difference (60s)\n\n/**\n * Check if running in dev/local mode\n */\nfunction isDevMode() {\n if (typeof window === 'undefined') return false\n return ['localhost', '127.0.0.1', '[::1]'].includes(window.location.hostname)\n}\n\n/**\n * Build ping URL for captive portal detection\n */\nfunction getPingUrl() {\n const hostname = window.location.hostname\n const localServers = ['localhost', '127.0.0.1', '[::1]']\n const isLocal = localServers.includes(hostname)\n const isHttps = window.location.protocol === 'https:'\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n const protocol = isHttps ? 'https' : 'http'\n const portSuffix = (isLocal || (port !== 80 && port !== 443)) ? `:${port}` : ''\n return `${protocol}://${hostname}${portSuffix}/api/ape/ping`\n}\n\n/**\n * Check for captive portal by pinging /api/ape/ping\n * Returns 'ok' if real internet, 'walled' if captive portal detected\n */\nasync function checkCaptivePortal() {\n try {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), PING_TIMEOUT)\n\n const response = await fetch(getPingUrl(), {\n cache: 'no-store',\n signal: controller.signal\n })\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed: HTTP', response.status)\n }\n return 'walled'\n }\n\n const data = await response.json()\n\n // Verify response is genuine (not a captive portal redirect page)\n if (data?.ok !== true) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed: invalid response', data)\n }\n return 'walled'\n }\n\n // Validate timestamp to detect proxy replay attacks\n if (typeof data.ts === 'number') {\n const now = Date.now()\n const skew = Math.abs(now - data.ts)\n if (skew > MAX_PING_CLOCK_SKEW) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed: timestamp too old/stale (skew:', skew, 'ms)')\n }\n return 'walled'\n }\n }\n\n return 'ok'\n } catch (err) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed:', err.message || err)\n }\n return 'walled'\n }\n}\n\n/**\n * Setup navigator.onLine event listeners\n */\nfunction setupOnlineListeners() {\n if (typeof window === 'undefined') return\n\n window.addEventListener('online', () => {\n console.log('\uD83E\uDD8D Browser went online, checking network...')\n // Trigger reconnection attempt\n attemptConnection()\n })\n\n window.addEventListener('offline', () => {\n console.log('\uD83E\uDD8D Browser went offline')\n notifyConnectionChange(ConnectionState.Offline)\n })\n}\n\n// Setup listeners on module load (browser only)\nif (typeof window !== 'undefined') {\n setupOnlineListeners()\n}\n\n\n\n/**\n * Get WebSocket URL - auto-detects from window.location, keeps /api/ape path\n */\nfunction getSocketUrl() {\n const hostname = window.location.hostname\n const localServers = [\"localhost\", \"127.0.0.1\", \"[::1]\"]\n const isLocal = localServers.includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n\n // Default port: 9010 for local dev, otherwise use window.location.port or implicit 443/80\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n\n // Build URL - keep /api/ape path\n const protocol = isHttps ? \"wss\" : \"ws\"\n const portSuffix = (isLocal || port !== 80 && port !== 443) ? `:${port}` : \"\"\n\n return `${protocol}://${hostname}${portSuffix}/api/ape`\n}\n\nlet reconnect = false\nconst connectTimeout = 5000\nconst totalRequestTimeout = 10000\n//const location = window.location\n\nconst joinKey = \"/\"\n// Properties accessed directly on `ape` that should NOT be intercepted\nconst reservedKeys = new Set(['on', 'onConnectionChange', 'transport'])\nconst handler = {\n get(fn, key) {\n // Skip proxy interception for reserved keys - return actual property\n if (reservedKeys.has(key)) {\n return fn[key]\n }\n const wrapperFn = function (a, b) {\n let path = joinKey + key, body;\n if (2 === arguments.length) {\n path += a\n body = b\n } else {\n body = a\n }\n return fn(path, body)\n }\n return new Proxy(wrapperFn, handler)\n } // END get\n}\n\nfunction wrap(api) {\n return new Proxy(api, handler)\n}\n\nlet __socket = false, ready = false, wsSend = false;\nconst waitingOn = {};\n\nlet aWaitingSend = []\nconst receiverArray = [];\nconst ofTypesOb = {};\n\n/**\n * Switch to streaming transport (HTTP long polling fallback)\n */\nfunction switchToStreaming() {\n console.log('\uD83E\uDD8D Switching to HTTP streaming transport')\n currentTransport = 'polling'\n\n if (!streamingTransport) {\n streamingTransport = createStreamingTransport()\n\n // Handle incoming messages from streaming transport\n streamingTransport.onMessage = async (msg) => {\n const { err, type, data } = msg\n\n // Process linked resources and shared files\n let processedData = data\n if (data && !err) {\n try {\n processedData = await fetchLinkedResources(data)\n processedData = await fetchSharedFiles(processedData)\n } catch (fetchErr) {\n console.error(`\uD83E\uDD8D Failed to hydrate streaming data:`, fetchErr)\n }\n }\n\n // Dispatch to type-specific handlers\n if (ofTypesOb[type]) {\n ofTypesOb[type].forEach(worker => worker({ err, type, data: processedData }))\n }\n // Dispatch to general handlers\n receiverArray.forEach(worker => worker({ err, type, data: processedData }))\n }\n\n streamingTransport.onOpen = () => {\n ready = true\n notifyConnectionChange(ConnectionState.Connected)\n console.log('\uD83E\uDD8D HTTP streaming connected')\n\n // Flush waiting messages\n aWaitingSend.forEach(({ type, data, resolve, reject, waiting, createdAt, timer }) => {\n clearTimeout(timer)\n const resultPromise = streamingSend(type, data, createdAt)\n if (waiting) {\n resultPromise.then(resolve).catch(reject)\n }\n })\n aWaitingSend = []\n\n // Start background WebSocket retry\n startWsRetry()\n }\n\n streamingTransport.onClose = () => {\n ready = false\n notifyConnectionChange(ConnectionState.Disconnected)\n }\n\n streamingTransport.onError = (err) => {\n console.error('\uD83E\uDD8D Streaming error:', err)\n }\n }\n\n streamingTransport.connect()\n}\n\n/**\n * Send via streaming transport\n */\nfunction streamingSend(type, data, createdAt) {\n return streamingTransport.send(type, data, createdAt)\n}\n\n/**\n * Start background retry for WebSocket (while in polling mode)\n */\nfunction startWsRetry() {\n if (wsRetryTimer) return\n if (currentTransport !== 'polling') return\n if (configuredTransport === 'polling') return // User explicitly wants polling only\n\n wsRetryTimer = setInterval(() => {\n if (currentTransport !== 'polling') {\n clearInterval(wsRetryTimer)\n wsRetryTimer = null\n return\n }\n\n console.log('\uD83E\uDD8D Attempting WebSocket reconnection...')\n tryWebSocket(true)\n }, WS_RETRY_INTERVAL)\n}\n\n/**\n * Try to establish WebSocket connection\n * @param {boolean} isRetry - If true, this is a background retry attempt\n */\nfunction tryWebSocket(isRetry = false) {\n const ws = new WebSocket(getSocketUrl())\n let fallbackTimer = null\n\n // Set fallback timeout (only for initial connection, not retries)\n if (!isRetry && configuredTransport === 'auto') {\n fallbackTimer = setTimeout(() => {\n if (ws.readyState !== WebSocket.OPEN) {\n console.log('\uD83E\uDD8D WebSocket timeout, falling back to HTTP streaming')\n ws.close()\n switchToStreaming()\n }\n }, WS_FALLBACK_TIMEOUT)\n }\n\n ws.onopen = () => {\n if (fallbackTimer) clearTimeout(fallbackTimer)\n\n // If this is a retry and we're in polling mode, switch back to WebSocket\n if (isRetry && currentTransport === 'polling') {\n console.log('\uD83E\uDD8D WebSocket reconnected, switching from HTTP streaming')\n if (streamingTransport) {\n streamingTransport.close()\n }\n if (wsRetryTimer) {\n clearInterval(wsRetryTimer)\n wsRetryTimer = null\n }\n }\n\n currentTransport = 'websocket'\n __socket = ws\n ready = true\n notifyConnectionChange(ConnectionState.Connected)\n\n aWaitingSend.forEach(({ type, data, resolve, reject, waiting, createdAt, timer }) => {\n clearTimeout(timer)\n const resultPromise = wsSend(type, data, createdAt)\n if (waiting) {\n resultPromise.then(resolve).catch(reject)\n }\n })\n aWaitingSend = []\n }\n\n ws.onmessage = async function (event) {\n const { err, type, queryId, data } = jss.parse(event.data)\n\n // Messages with queryId must fulfill matching promise\n if (queryId) {\n if (waitingOn[queryId]) {\n // Check for linked resources and fetch them before resolving\n if (data && !err) {\n try {\n let hydratedData = await fetchLinkedResources(data)\n hydratedData = await fetchSharedFiles(hydratedData)\n waitingOn[queryId](err, hydratedData)\n } catch (fetchErr) {\n waitingOn[queryId](fetchErr, null)\n }\n } else {\n waitingOn[queryId](err, data)\n }\n delete waitingOn[queryId]\n } else {\n console.error(`\uD83E\uDD8D No matching queryId: ${queryId}`)\n }\n return\n }\n\n // Only messages WITHOUT queryId go to setOnReceiver\n let processedData = data\n if (data && !err) {\n try {\n processedData = await fetchLinkedResources(data)\n processedData = await fetchSharedFiles(processedData)\n } catch (fetchErr) {\n console.error(`\uD83E\uDD8D Failed to hydrate broadcast data:`, fetchErr)\n }\n }\n\n if (ofTypesOb[type]) {\n ofTypesOb[type].forEach(worker => worker({ err, type, data: processedData }))\n }\n receiverArray.forEach(worker => worker({ err, type, data: processedData }))\n }\n\n ws.onerror = function (err) {\n if (fallbackTimer) clearTimeout(fallbackTimer)\n console.error('socket ERROR:', err)\n\n // On initial connection error in auto mode, fallback to streaming\n if (!isRetry && configuredTransport === 'auto' && !ready) {\n switchToStreaming()\n }\n }\n\n ws.onclose = function (event) {\n if (fallbackTimer) clearTimeout(fallbackTimer)\n console.warn('socket disconnect:', event)\n __socket = false\n ready = false\n\n // Only notify disconnected if we're on websocket transport\n if (currentTransport === 'websocket') {\n notifyConnectionChange(ConnectionState.Disconnected)\n setTimeout(() => reconnect && connectSocket(), 500)\n }\n }\n}\n\n/**\n * Find all L-tagged (binary link) properties in data\n * Returns array of { path, hash }\n */\nfunction findLinkedResources(obj, path = '') {\n const resources = []\n\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return resources\n }\n\n if (Array.isArray(obj)) {\n for (let i = 0; i < obj.length; i++) {\n resources.push(...findLinkedResources(obj[i], path ? `${path}.${i}` : String(i)))\n }\n return resources\n }\n\n for (const key of Object.keys(obj)) {\n // Check for L-tag in key (from JJS encoding: key<!L>)\n if (key.endsWith('<!L>')) {\n const cleanKey = key.slice(0, -4)\n const hash = obj[key]\n resources.push({\n path: path ? `${path}.${cleanKey}` : cleanKey,\n hash,\n originalKey: key\n })\n } else {\n resources.push(...findLinkedResources(obj[key], path ? `${path}.${key}` : key))\n }\n }\n\n return resources\n}\n\n/**\n * Find all F-tagged (shared file) properties in data\n * Returns array of { path, hash, originalKey }\n */\nfunction findFileTags(obj, path = '') {\n const files = []\n\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return files\n }\n\n if (Array.isArray(obj)) {\n for (let i = 0; i < obj.length; i++) {\n files.push(...findFileTags(obj[i], path ? `${path}.${i}` : String(i)))\n }\n return files\n }\n\n for (const key of Object.keys(obj)) {\n // Check for F-tag in key (client-to-client shared file marker)\n if (key.endsWith('<!F>')) {\n const cleanKey = key.slice(0, -4)\n const hash = obj[key]\n files.push({\n path: path ? `${path}.${cleanKey}` : cleanKey,\n hash,\n originalKey: key\n })\n } else {\n files.push(...findFileTags(obj[key], path ? `${path}.${key}` : key))\n }\n }\n\n return files\n}\n\n/**\n * Clean up F-tagged keys (rename key<!F> to key)\n */\nfunction cleanFileTags(obj) {\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return obj\n }\n\n if (Array.isArray(obj)) {\n return obj.map(cleanFileTags)\n }\n\n const cleaned = {}\n for (const key of Object.keys(obj)) {\n if (key.endsWith('<!F>')) {\n const cleanKey = key.slice(0, -4)\n cleaned[cleanKey] = obj[key]\n } else {\n cleaned[key] = cleanFileTags(obj[key])\n }\n }\n return cleaned\n}\n\n/**\n * Fetch shared files (client-to-client transfers)\n * Retries if upload is still in progress\n */\nasync function fetchSharedFiles(data, maxRetries = 5) {\n const files = findFileTags(data)\n\n if (files.length === 0) {\n return data\n }\n\n console.log(`\uD83E\uDD8D Fetching ${files.length} shared file(s)`)\n\n const cleanedData = cleanFileTags(data)\n\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (isLocal || (port !== 80 && port !== 443)) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n await Promise.all(files.map(async ({ path, hash }) => {\n let retries = 0\n let backoff = 100 // Start with 100ms\n\n while (retries < maxRetries) {\n try {\n const response = await fetch(`${baseUrl}/api/ape/data/${hash}`, {\n credentials: 'include'\n })\n\n if (!response.ok) {\n // 404 might mean file not uploaded yet, retry\n if (response.status === 404 && retries < maxRetries - 1) {\n retries++\n await new Promise(r => setTimeout(r, backoff))\n backoff *= 2 // Exponential backoff\n continue\n }\n throw new Error(`Failed to fetch shared file: ${response.status}`)\n }\n\n const arrayBuffer = await response.arrayBuffer()\n setValueAtPath(cleanedData, path, arrayBuffer)\n\n // Check if upload is still in progress\n const isComplete = response.headers.get('X-Ape-Complete') === '1'\n if (!isComplete) {\n console.log(`\uD83E\uDD8D Shared file ${hash} still uploading (${response.headers.get('X-Ape-Total-Received') || '?'} bytes)`)\n }\n break\n } catch (err) {\n if (retries >= maxRetries - 1) {\n console.error(`\uD83E\uDD8D Failed to fetch shared file at ${path}:`, err)\n setValueAtPath(cleanedData, path, null)\n }\n retries++\n await new Promise(r => setTimeout(r, backoff))\n backoff *= 2\n }\n }\n }))\n\n return cleanedData\n}\n\n/**\n * Set a value at a nested path in an object\n */\nfunction setValueAtPath(obj, path, value) {\n const parts = path.split('.')\n let current = obj\n\n for (let i = 0; i < parts.length - 1; i++) {\n current = current[parts[i]]\n }\n\n current[parts[parts.length - 1]] = value\n}\n\n/**\n * Clean up L-tagged keys (rename key<!L> to key)\n */\nfunction cleanLinkedKeys(obj) {\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return obj\n }\n\n if (Array.isArray(obj)) {\n return obj.map(cleanLinkedKeys)\n }\n\n const cleaned = {}\n for (const key of Object.keys(obj)) {\n if (key.endsWith('<!L>')) {\n const cleanKey = key.slice(0, -4)\n cleaned[cleanKey] = obj[key]\n } else {\n cleaned[key] = cleanLinkedKeys(obj[key])\n }\n }\n return cleaned\n}\n\n/**\n * Fetch binary resources and hydrate data object\n */\nasync function fetchLinkedResources(data, clientId) {\n const resources = findLinkedResources(data)\n\n if (resources.length === 0) {\n return data\n }\n\n console.log(`\uD83E\uDD8D Fetching ${resources.length} binary resource(s)`)\n\n const cleanedData = cleanLinkedKeys(data)\n\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (isLocal || (port !== 80 && port !== 443)) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n await Promise.all(resources.map(async ({ path, hash }) => {\n try {\n const response = await fetch(`${baseUrl}/api/ape/data/${hash}`, {\n credentials: 'include',\n headers: {\n 'X-Ape-Client-Id': clientId || ''\n }\n })\n\n if (!response.ok) {\n throw new Error(`Failed to fetch binary resource: ${response.status}`)\n }\n\n const arrayBuffer = await response.arrayBuffer()\n setValueAtPath(cleanedData, path, arrayBuffer)\n } catch (err) {\n console.error(`\uD83E\uDD8D Failed to fetch binary resource at ${path}:`, err)\n setValueAtPath(cleanedData, path, null)\n }\n }))\n\n return cleanedData\n}\n\n/**\n * Attempt to establish connection with network pre-checks\n */\nasync function attemptConnection() {\n // Check if browser is online\n if (typeof navigator !== 'undefined' && !navigator.onLine) {\n notifyConnectionChange(ConnectionState.Offline)\n return\n }\n\n // Perform captive portal check\n notifyConnectionChange(ConnectionState.Connecting)\n const pingResult = await checkCaptivePortal()\n\n if (pingResult === 'walled') {\n notifyConnectionChange(ConnectionState.Walled)\n // Retry network check periodically\n scheduleNetworkRetry()\n return\n }\n\n // Network is good, proceed with socket connection\n proceedWithConnection()\n}\n\n/**\n * Schedule a retry of network check (for walled/offline states)\n */\nfunction scheduleNetworkRetry() {\n if (networkCheckTimer) return\n networkCheckTimer = setTimeout(() => {\n networkCheckTimer = null\n attemptConnection()\n }, WS_RETRY_INTERVAL)\n}\n\n/**\n * Proceed with WebSocket/polling connection after network checks pass\n */\nfunction proceedWithConnection() {\n // Determine which transport to use\n if (configuredTransport === 'polling') {\n switchToStreaming()\n } else {\n // 'auto' or 'websocket' - try WebSocket first\n tryWebSocket(false)\n }\n}\n\nfunction connectSocket() {\n // Skip if already connected or connecting\n if (__socket && __socket.readyState !== WebSocket.CLOSED) {\n return buildClientInterface()\n }\n if (currentTransport === 'polling' && streamingTransport?.isConnected()) {\n return buildClientInterface()\n }\n if (connectionState === ConnectionState.Connecting) {\n return buildClientInterface()\n }\n\n // Start connection with network pre-checks\n attemptConnection()\n\n return buildClientInterface()\n}\n\n/**\n * Check if value is binary data (ArrayBuffer, typed array, or Blob)\n */\nfunction isBinaryData(value) {\n if (value === null || value === undefined) return false\n return value instanceof ArrayBuffer ||\n ArrayBuffer.isView(value) ||\n (typeof Blob !== 'undefined' && value instanceof Blob)\n}\n\n/**\n * Get binary type tag (A for ArrayBuffer, B for Blob)\n */\nfunction getBinaryTag(value) {\n if (typeof Blob !== 'undefined' && value instanceof Blob) return 'B'\n return 'A'\n}\n\n/**\n * Generate a simple hash for binary upload\n */\nfunction generateUploadHash(path) {\n let hash = 0\n for (let i = 0; i < path.length; i++) {\n const char = path.charCodeAt(i)\n hash = ((hash << 5) - hash) + char\n hash = hash & hash\n }\n return Math.abs(hash).toString(36)\n}\n\n/**\n * Find and extract binary data from payload\n * Returns { processedData, uploads: [{ path, hash, data, tag }] }\n */\nfunction processBinaryForUpload(data, path = '') {\n if (data === null || data === undefined) {\n return { processedData: data, uploads: [] }\n }\n\n if (isBinaryData(data)) {\n const tag = getBinaryTag(data)\n const hash = generateUploadHash(path || 'root')\n return {\n processedData: { [`__ape_upload__`]: hash },\n uploads: [{ path, hash, data, tag }]\n }\n }\n\n if (Array.isArray(data)) {\n const processedArray = []\n const allUploads = []\n\n for (let i = 0; i < data.length; i++) {\n const itemPath = path ? `${path}.${i}` : String(i)\n const { processedData, uploads } = processBinaryForUpload(data[i], itemPath)\n processedArray.push(processedData)\n allUploads.push(...uploads)\n }\n\n return { processedData: processedArray, uploads: allUploads }\n }\n\n if (typeof data === 'object') {\n const processedObj = {}\n const allUploads = []\n\n for (const key of Object.keys(data)) {\n const itemPath = path ? `${path}.${key}` : key\n const { processedData, uploads } = processBinaryForUpload(data[key], itemPath)\n\n // If this was binary data, mark the key with <!B> or <!A> tag\n if (uploads.length > 0 && processedData?.__ape_upload__) {\n const tag = uploads[uploads.length - 1].tag\n processedObj[`${key}<!${tag}>`] = processedData.__ape_upload__\n } else {\n processedObj[key] = processedData\n }\n allUploads.push(...uploads)\n }\n\n return { processedData: processedObj, uploads: allUploads }\n }\n\n return { processedData: data, uploads: [] }\n}\n\n/**\n * Find and extract binary data for SHARING (client-to-client)\n * Uses <!F> tag instead of <!A>/<!B>\n * Returns { processedData, shares: [{ path, hash, data }] }\n */\nfunction processBinaryForSharing(data, path = '') {\n if (data === null || data === undefined) {\n return { processedData: data, shares: [] }\n }\n\n if (isBinaryData(data)) {\n const hash = generateUploadHash(path || 'share')\n return {\n processedData: { [`__ape_share__`]: hash },\n shares: [{ path, hash, data }]\n }\n }\n\n if (Array.isArray(data)) {\n const processedArray = []\n const allShares = []\n\n for (let i = 0; i < data.length; i++) {\n const itemPath = path ? `${path}.${i}` : String(i)\n const { processedData, shares } = processBinaryForSharing(data[i], itemPath)\n processedArray.push(processedData)\n allShares.push(...shares)\n }\n\n return { processedData: processedArray, shares: allShares }\n }\n\n if (typeof data === 'object') {\n const processedObj = {}\n const allShares = []\n\n for (const key of Object.keys(data)) {\n const itemPath = path ? `${path}.${key}` : key\n const { processedData, shares } = processBinaryForSharing(data[key], itemPath)\n\n // If this was binary data, mark the key with <!F> tag\n if (shares.length > 0 && processedData?.__ape_share__) {\n processedObj[`${key}<!F>`] = processedData.__ape_share__\n } else {\n processedObj[key] = processedData\n }\n allShares.push(...shares)\n }\n\n return { processedData: processedObj, shares: allShares }\n }\n\n return { processedData: data, shares: [] }\n}\n\n/**\n * Upload shared files via HTTP PUT\n * Uses different endpoint pattern for streaming files\n */\nasync function uploadSharedFiles(shares) {\n if (shares.length === 0) return\n\n // Build base URL\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (isLocal || (port !== 80 && port !== 443)) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n console.log(`\uD83E\uDD8D Uploading ${shares.length} shared file(s)`)\n\n await Promise.all(shares.map(async ({ hash, data }) => {\n try {\n // For shared files, use upload pattern with hash as both queryId and pathHash\n const response = await fetch(`${baseUrl}/api/ape/data/_share/${hash}`, {\n method: 'PUT',\n credentials: 'include',\n headers: {\n 'Content-Type': 'application/octet-stream'\n },\n body: data\n })\n\n if (!response.ok) {\n throw new Error(`Shared upload failed: ${response.status}`)\n }\n } catch (err) {\n console.error(`\uD83E\uDD8D Failed to upload shared file ${hash}:`, err)\n throw err\n }\n }))\n}\n\n/**\n * Upload binary data via HTTP PUT\n */\nasync function uploadBinaryData(queryId, uploads) {\n if (uploads.length === 0) return\n\n // Build base URL\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = isLocal ? 9010 : (window.location.port || (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (isLocal || (port !== 80 && port !== 443)) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n console.log(`\uD83E\uDD8D Uploading ${uploads.length} binary file(s)`)\n\n await Promise.all(uploads.map(async ({ hash, data }) => {\n try {\n const response = await fetch(`${baseUrl}/api/ape/data/${queryId}/${hash}`, {\n method: 'PUT',\n credentials: 'include',\n headers: {\n 'Content-Type': 'application/octet-stream'\n },\n body: data\n })\n\n if (!response.ok) {\n throw new Error(`Upload failed: ${response.status}`)\n }\n } catch (err) {\n console.error(`\uD83E\uDD8D Failed to upload binary at ${hash}:`, err)\n throw err\n }\n }))\n}\n\nwsSend = function (type, data, createdAt, dirctCall) {\n let rej, promiseIsLive = false;\n const timeLetForReqToBeMade = (createdAt + totalRequestTimeout) - Date.now()\n\n const timer = setTimeout(() => {\n if (promiseIsLive) {\n rej(new Error(\"Request Timedout for :\" + type))\n }\n }, timeLetForReqToBeMade);\n\n // Process binary data for upload\n const { processedData, uploads } = processBinaryForUpload(data)\n\n const payload = {\n type,\n data: processedData,\n //referer:window.location.href,\n createdAt: new Date(createdAt),\n requestedAt: dirctCall ? undefined\n : new Date()\n }\n const message = jss.stringify(payload)\n const queryId = messageHash(message);\n\n const replyPromise = new Promise((resolve, reject) => {\n rej = reject\n waitingOn[queryId] = (err, result) => {\n clearTimeout(timer)\n replyPromise.then = next.bind(replyPromise)\n if (err) {\n reject(err)\n } else {\n resolve(result)\n }\n }\n __socket.send(message);\n\n // Upload binary data after sending WS message\n if (uploads.length > 0) {\n uploadBinaryData(queryId, uploads).catch(err => {\n console.error('\uD83E\uDD8D Binary upload failed:', err)\n // The server will timeout waiting for the upload\n })\n }\n });\n const next = replyPromise.then;\n replyPromise.then = worker => {\n promiseIsLive = true;\n replyPromise.then = next.bind(replyPromise)\n replyPromise.catch = err.bind(replyPromise)\n return next.call(replyPromise, worker)\n }\n const err = replyPromise.catch;\n replyPromise.catch = worker => {\n promiseIsLive = true;\n replyPromise.catch = err.bind(replyPromise)\n replyPromise.then = next.bind(replyPromise)\n return err.call(replyPromise, worker)\n }\n return replyPromise\n} // END wsSend\n\n\nconst sender = (type, data) => {\n if (\"string\" !== typeof type) {\n throw new Error(\"Missing Path vaule\")\n }\n\n const createdAt = Date.now()\n\n if (ready) {\n return wsSend(type, data, createdAt, true)\n }\n\n const timeLetForReqToBeMade = (createdAt + connectTimeout) - Date.now() // 5sec for reconnect\n\n const timer = setTimeout(() => {\n const errMessage = \"Request not sent for :\" + type\n if (payload.waiting) {\n payload.reject(new Error(errMessage))\n } else {\n throw new Error(errMessage)\n }\n }, timeLetForReqToBeMade);\n\n const payload = { type, data, resolve: undefined, reject: undefined, waiting: false, createdAt, timer };\n const waitingOnOpen = new Promise((res, rej) => { payload.resolve = res; payload.reject = rej; })\n\n const waitingOnOpenThen = waitingOnOpen.then;\n const waitingOnOpenCatch = waitingOnOpen.catch;\n waitingOnOpen.then = worker => {\n payload.waiting = true;\n waitingOnOpen.then = waitingOnOpenThen.bind(waitingOnOpen)\n waitingOnOpen.catch = waitingOnOpenCatch.bind(waitingOnOpen)\n return waitingOnOpenThen.call(waitingOnOpen, worker)\n }\n waitingOnOpen.catch = worker => {\n payload.waiting = true;\n waitingOnOpen.catch = waitingOnOpenCatch.bind(waitingOnOpen)\n waitingOnOpen.then = waitingOnOpenThen.bind(waitingOnOpen)\n return waitingOnOpenCatch.call(waitingOnOpen, worker)\n }\n\n aWaitingSend.push(payload)\n if (!__socket) {\n connectSocket()\n }\n\n return waitingOnOpen\n} // END sender\n\n/**\n * Build the client interface object\n */\nfunction buildClientInterface() {\n return {\n sender: wrap(sender),\n setOnReceiver: (onTypeStFn, handlerFn) => {\n if (\"string\" === typeof onTypeStFn) {\n // Replace handler for this type (prevents duplicates in React StrictMode)\n ofTypesOb[onTypeStFn] = [handlerFn]\n } else {\n // For general receivers, prevent duplicates by checking\n if (!receiverArray.includes(onTypeStFn)) {\n receiverArray.push(onTypeStFn)\n }\n }\n },\n onConnectionChange: (handler) => {\n connectionChangeListeners.push(handler)\n // Immediately call with current state\n handler(connectionState)\n // Return unsubscribe function\n return () => {\n const idx = connectionChangeListeners.indexOf(handler)\n if (idx > -1) connectionChangeListeners.splice(idx, 1)\n }\n },\n // Expose current transport type (read-only)\n get transport() { return currentTransport }\n }\n}\n\nconnectSocket.autoReconnect = () => reconnect = true\nconnectSocket.ConnectionState = ConnectionState\nconnect = connectSocket\n\nexport default connect;\nexport { ConnectionState };\n", "/**\n * Unified api-ape export for browser\n * \n * Auto-detects browser environment, initializes client, and buffers\n * calls until the connection is ready. No more getApeClient().then()!\n * \n * Usage:\n * import api from 'api-ape'\n * \n * // Properties are proxied - calls buffer until connected\n * api.message({ user: 'Bob', text: 'Hello!' })\n * \n * // Subscribe to broadcasts\n * api.on('message', (data) => console.log(data))\n * \n * // Check connection state\n * api.onConnectionChange((state) => console.log(state))\n */\n\n// Only run this in browser environments\nconst isBrowser = typeof window !== 'undefined' && typeof document !== 'undefined'\n\nlet clientPromise = null\nlet resolvedClient = null\nconst bufferedCalls = []\nconst bufferedReceivers = []\nconst connectionChangeHandlers = []\nlet currentConnectionState = 'disconnected'\n\n/**\n * Initialize the client (called once on first use)\n */\nfunction getClient() {\n if (clientPromise) return clientPromise\n\n if (!isBrowser) {\n // Return a dummy object for SSR\n return Promise.resolve(null)\n }\n\n clientPromise = (async () => {\n const connectSocket = (await import('./connectSocket.js')).default\n\n // Connect\n const client = connectSocket()\n connectSocket.autoReconnect()\n\n // Track connection state\n client.onConnectionChange((state) => {\n currentConnectionState = state\n connectionChangeHandlers.forEach(fn => fn(state))\n })\n\n resolvedClient = client\n\n // Flush buffered receivers\n bufferedReceivers.forEach(({ type, handler }) => {\n client.setOnReceiver(type, handler)\n })\n bufferedReceivers.length = 0\n\n // Flush buffered calls\n bufferedCalls.forEach(({ method, args, resolve, reject }) => {\n try {\n const result = client.sender[method](...args)\n if (result && typeof result.then === 'function') {\n result.then(resolve).catch(reject)\n } else {\n resolve(result)\n }\n } catch (err) {\n reject(err)\n }\n })\n bufferedCalls.length = 0\n\n return client\n })()\n\n return clientPromise\n}\n\n/**\n * Create a sender proxy that buffers calls until client is ready\n */\nconst senderProxy = new Proxy({}, {\n get(target, prop) {\n // Reserved properties\n if (prop === 'on') return on\n if (prop === 'onConnectionChange') return onConnectionChange\n if (prop === 'transport') return resolvedClient?.transport || null\n if (prop === 'then' || prop === 'catch') return undefined // Not a Promise\n\n // Return a function that either calls directly or buffers\n return (...args) => {\n // If client is ready, call directly\n if (resolvedClient) {\n return resolvedClient.sender[prop](...args)\n }\n\n // Buffer the call and return a Promise\n return new Promise((resolve, reject) => {\n bufferedCalls.push({ method: prop, args, resolve, reject })\n // Ensure client is initializing\n getClient()\n })\n }\n }\n})\n\n/**\n * Subscribe to broadcasts from the server\n * @param {string} type - Broadcast type to listen for\n * @param {Function} handler - Handler function\n */\nfunction on(type, handler) {\n if (resolvedClient) {\n resolvedClient.setOnReceiver(type, handler)\n } else {\n bufferedReceivers.push({ type, handler })\n getClient()\n }\n}\n\n/**\n * Subscribe to connection state changes\n * @param {Function} handler - Called with state: 'offline' | 'walled' | 'disconnected' | 'connecting' | 'connected'\n * @returns {Function} Unsubscribe function\n */\nfunction onConnectionChange(handler) {\n connectionChangeHandlers.push(handler)\n // Immediately call with current state\n handler(currentConnectionState)\n\n // If client exists, also register with it\n if (resolvedClient) {\n return resolvedClient.onConnectionChange(handler)\n }\n\n // Ensure client is initializing\n getClient()\n\n // Return unsubscribe function\n return () => {\n const idx = connectionChangeHandlers.indexOf(handler)\n if (idx > -1) connectionChangeHandlers.splice(idx, 1)\n }\n}\n\n// Define properties on the proxy to avoid Proxy interception issues\nObject.defineProperty(senderProxy, 'on', {\n value: on,\n writable: false,\n enumerable: false,\n configurable: false\n})\n\nObject.defineProperty(senderProxy, 'onConnectionChange', {\n value: onConnectionChange,\n writable: false,\n enumerable: false,\n configurable: false\n})\n\n// Auto-initialize in browser\nif (isBrowser) {\n getClient()\n}\n\nexport default senderProxy\nexport { on, onConnectionChange, getClient }\n"],
5
- "mappings": "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",
6
- "names": ["require_messageHash", "__commonJSMin", "exports", "module", "alphabet", "toBase32", "n", "remainder", "current", "jenkinsOneAtATimeHash", "keyString", "hash", "charIndex", "messageHash", "messageSt", "require_jss", "__commonJSMin", "exports", "module", "encode", "obj", "tagLookup", "visited", "encodeValue", "value", "path", "type", "tag", "visitedEncode", "isArray", "keys", "result", "typesFound", "i", "key", "t", "v", "encodeValueWithVisited", "objKeys", "stringify", "parse", "encoded", "decode", "data", "pointers2Res", "s", "n", "sourceToPointAt", "replaceAtThisPlace", "name", "message", "stack", "err", "o", "decodeValue", "val", "currentPath", "typeTags", "res", "itemPath", "decodedValue", "nam", "parseKeyWithTags", "match", "multiMatch", "arrayMatch", "changeAttributeReference", "refPath", "attrPath", "refKeys", "attrKeys", "ref", "attr", "getPollUrl", "hostname", "isLocal", "isHttps", "port", "protocol", "portSuffix", "parseStreamBuffer", "buffer", "messages", "start", "depth", "inString", "escaped", "i", "char", "jsonStr", "jss", "e", "remaining", "createStreamingTransport", "isActive", "abortController", "streamBuffer", "reconnectTimer", "onMessage", "onOpen", "onClose", "onError", "connect", "response", "reader", "decoder", "read", "done", "value", "scheduleReconnect", "msg", "readErr", "err", "send", "type", "data", "createdAt", "payload", "error", "close", "fn", "import_jss", "init_streaming", "__esmMin", "connectSocket_exports", "__export", "ConnectionState", "connectSocket_default", "notifyConnectionChange", "newState", "connectionState", "connectionChangeListeners", "fn", "isDevMode", "getPingUrl", "hostname", "isLocal", "isHttps", "port", "protocol", "portSuffix", "checkCaptivePortal", "controller", "timeoutId", "PING_TIMEOUT", "response", "data", "now", "skew", "MAX_PING_CLOCK_SKEW", "err", "setupOnlineListeners", "attemptConnection", "getSocketUrl", "wrap", "api", "handler", "switchToStreaming", "currentTransport", "streamingTransport", "createStreamingTransport", "msg", "type", "processedData", "fetchLinkedResources", "fetchSharedFiles", "fetchErr", "ofTypesOb", "worker", "receiverArray", "ready", "aWaitingSend", "resolve", "reject", "waiting", "createdAt", "timer", "resultPromise", "streamingSend", "startWsRetry", "wsRetryTimer", "configuredTransport", "tryWebSocket", "WS_RETRY_INTERVAL", "isRetry", "ws", "fallbackTimer", "WS_FALLBACK_TIMEOUT", "__socket", "wsSend", "event", "queryId", "jss", "waitingOn", "hydratedData", "reconnect", "connectSocket", "findLinkedResources", "obj", "path", "resources", "i", "key", "cleanKey", "hash", "findFileTags", "files", "cleanFileTags", "cleaned", "maxRetries", "cleanedData", "baseUrl", "retries", "backoff", "r", "arrayBuffer", "setValueAtPath", "value", "parts", "current", "cleanLinkedKeys", "clientId", "scheduleNetworkRetry", "proceedWithConnection", "networkCheckTimer", "buildClientInterface", "isBinaryData", "getBinaryTag", "generateUploadHash", "char", "processBinaryForUpload", "tag", "processedArray", "allUploads", "itemPath", "uploads", "processedObj", "uploadBinaryData", "sender", "onTypeStFn", "handlerFn", "idx", "import_messageHash", "import_jss", "connect", "connectTimeout", "totalRequestTimeout", "joinKey", "reservedKeys", "init_connectSocket", "__esmMin", "init_streaming", "wrapperFn", "a", "b", "body", "dirctCall", "rej", "promiseIsLive", "timeLetForReqToBeMade", "payload", "message", "messageHash", "replyPromise", "result", "next", "errMessage", "waitingOnOpen", "res", "waitingOnOpenThen", "waitingOnOpenCatch", "isBrowser", "clientPromise", "resolvedClient", "bufferedCalls", "bufferedReceivers", "connectionChangeHandlers", "currentConnectionState", "getClient", "connectSocket", "client", "state", "fn", "type", "handler", "method", "args", "resolve", "reject", "result", "err", "senderProxy", "target", "prop", "on", "onConnectionChange", "idx", "index_default"]
4
+ "sourcesContent": ["const alphabet = \"0123456789ABCDEFGHJKMNPQRSTVWXYZ\"\n/*\nfunction charValue(char){\n return alphabet.indexOf(char.toUpperCase())\n} // END charValue\n\nfunction fromBase32(b32){\n if (0 === b32.length) {\n return 0\n }\n return charValue(b32.slice(-1)) + fromBase32(b32.slice(0,-1)) * 32\n} // END fromBase32\n*/\nfunction toBase32 (n){\n const remainder = Math.floor(n/32)\n const current = n % 32\n if (0 === remainder) {\n return alphabet[current]\n }\n return toBase32(remainder)+alphabet[current]\n} // END toBase32\n\nfunction jenkinsOneAtATimeHash(keyString){\n \n var hash = 0\n \n for (var charIndex = 0; charIndex < keyString.length; ++charIndex)\n {\n hash += keyString.charCodeAt(charIndex);\n hash += hash << 10;\n hash ^= hash >> 6;\n }\n hash += hash << 3;\n hash ^= hash >> 11;\n //4,294,967,295 is FFFFFFFF, the maximum 32 bit unsigned integer value, used here as a mask.\n return (((hash + (hash << 15)) & 4294967295) >>> 0)\n} // END jenkinsOneAtATimeHash\n\nfunction messageHash(messageSt){\n return toBase32(jenkinsOneAtATimeHash(messageSt))\n} // END messageHash\n\nmodule.exports = messageHash", "//JsonSuperSet\n\n// TODO: add tests\n// check for any repeated ref not just cyclical references\n// support nasted array a<![,,[D]]>:[\"a\",\"b\",[Date]]\n// support array for the same type a<![*D]>:[Date,Date,Date]\n\nfunction encode(obj) {\n const tagLookup = {\n '[object RegExp]': 'R',\n '[object Date]': 'D',\n '[object Error]': 'E',\n \"[object Undefined]\": 'U',\n \"[object Map]\": 'M',\n \"[object Set]\": 'S',\n };\n const visited = new WeakMap();\n\n function encodeValue(value, path = '') {\n const type = typeof value;\n const tag = tagLookup[Object.prototype.toString.call(value)];\n // console.log({tag,value,path})\n if (tag !== undefined) {\n if ('D' === tag) return [tag, value.valueOf()];\n if ('E' === tag) return [tag, [value.name, value.message, value.stack]];\n if ('R' === tag) return [tag, value.toString()];\n if ('U' === tag) return [tag, null];\n if ('S' === tag) return [tag, Array.from(value)];\n if ('M' === tag) return [tag, Object.fromEntries(value)];\n\n return [tag, JSON.stringify(value)];\n } else if (type === 'object' && value !== null) {\n /*if (value.$ID) {\n return ['', value.$ID];\n }*/\n if (visitedEncode.has(value)) {\n return ['P', visitedEncode.get(value)];\n }\n visitedEncode.set(value, path);\n const isArray = Array.isArray(value);\n // keep index with undefined in Array\n const keys = isArray ? Array.from(Array(value.length).keys()) : Object.keys(value);\n const result = isArray ? [] : {};\n const typesFound = [];\n for (let i = 0; i < keys.length; i++) {\n const key = keys[i];\n const [t, v] = encodeValue(value[key], key);\n // console.log([t, v])\n if (isArray) {\n typesFound.push(t);\n result.push(v);\n // remove key with undefined from Objects\n } else if (value[key] !== undefined) {\n result[key + (t ? `<!${t}>` : '')] = v;\n }\n }\n\n visited.delete(value);\n if (isArray && typesFound.find((t) => !!t)) {\n return [`[${typesFound.join()}]`, result];\n }\n return ['', result];\n } else {\n return ['', value];\n }\n } // END encodeValue\n\n let keys = [];\n // console.log(obj)\n if (Array.isArray(obj)) {\n keys = Array.from(Array(obj.length).keys())\n } else {\n keys = Object.keys(obj);\n }\n\n // Track root object to handle self-references\n const visitedEncode = new WeakMap();\n visitedEncode.set(obj, []); // Root path is empty array\n\n function encodeValueWithVisited(value, path = []) {\n const type = typeof value;\n const tag = tagLookup[Object.prototype.toString.call(value)];\n if (tag !== undefined) {\n if ('D' === tag) return [tag, value.valueOf()];\n if ('E' === tag) return [tag, [value.name, value.message, value.stack]];\n if ('R' === tag) return [tag, value.toString()];\n if ('U' === tag) return [tag, null];\n if ('S' === tag) return [tag, Array.from(value)];\n if ('M' === tag) return [tag, Object.fromEntries(value)];\n return [tag, JSON.stringify(value)];\n } else if (type === 'object' && value !== null) {\n if (visitedEncode.has(value)) {\n return ['P', visitedEncode.get(value)]; // Return array path\n }\n visitedEncode.set(value, path);\n const isArray = Array.isArray(value);\n const objKeys = isArray ? Array.from(Array(value.length).keys()) : Object.keys(value);\n const result = isArray ? [] : {};\n const typesFound = [];\n for (let i = 0; i < objKeys.length; i++) {\n const key = objKeys[i];\n const [t, v] = encodeValueWithVisited(value[key], [...path, key]); // Append key to path array\n if (isArray) {\n typesFound.push(t);\n result.push(v);\n } else if (value[key] !== undefined) {\n result[key + (t ? `<!${t}>` : '')] = v;\n }\n }\n if (isArray && typesFound.find((t) => !!t)) {\n return [`[${typesFound.join()}]`, result];\n }\n return ['', result];\n } else {\n return ['', value];\n }\n }\n\n const result = {};\n for (let i = 0; i < keys.length; i++) {\n const key = keys[i];\n // remove key with undefined from Objects\n if (obj[key] !== undefined) {\n const [t, v] = encodeValueWithVisited(obj[key], [key]); // Start path with single key\n result[key + (t ? `<!${t}>` : '')] = v;\n }\n }\n return result;\n} // END encode\n\nfunction stringify(obj) {\n return JSON.stringify(encode(obj))\n}\n\n\nfunction parse(encoded) {\n return decode(JSON.parse(encoded))\n}\n\nfunction decode(data) {\n const result = {};\n const pointers2Res = [];\n const tagLookup = {\n R: (s) => new RegExp(s),\n D: (n) => new Date(n),\n P: function (sourceToPointAt, replaceAtThisPlace) {\n // Both paths are now arrays\n pointers2Res.push([sourceToPointAt, replaceAtThisPlace]);\n return null; // Placeholder, will be replaced by changeAttributeReference\n },\n E: ([name, message, stack]) => {\n let err;\n try {\n err = new global[name](message);\n if (err instanceof Error) err.stack = stack;\n else throw {};\n } catch (e) {\n err = new Error(message);\n err.name = name;\n err.stack = stack;\n }\n return err;\n },\n U: () => undefined,\n S: (a) => new Set(a),\n M: (o) => new Map(Object.entries(o))\n };\n const visited = new Map();\n\n function decodeValue(name, tag, val) {\n // this is now an array path\n const currentPath = Array.isArray(this) ? this : [];\n\n if (tag in tagLookup) {\n return tagLookup[tag](val, currentPath);\n } else if (Array.isArray(val)) {\n if (tag && tag.startsWith('[')) {\n const typeTags = tag.slice(1, -1).split(',');\n const res = [];\n for (let i = 0; i < val.length; i++) {\n // Pass path with array index appended\n const itemPath = [...currentPath, i];\n const decodedValue = decodeValue.call(\n itemPath,\n i.toString(),\n typeTags[i],\n val[i]\n );\n res.push(decodedValue);\n }\n return res;\n } else {\n const res = [];\n for (let i = 0; i < val.length; i++) {\n const decodedValue = decodeValue.call([...currentPath, i], '', '', val[i]);\n res.push(decodedValue);\n }\n return res;\n }\n } else if ('object' === typeof val && val !== null) {\n if (visited.has(val)) {\n return visited.get(val);\n }\n visited.set(val, {});\n const res = {};\n for (const key in val) {\n const [nam, t] = parseKeyWithTags(key);\n const decodedValue = decodeValue.call(\n [...currentPath, nam],\n nam,\n t,\n val[key]\n );\n res[nam] = decodedValue;\n }\n visited.set(val, res);\n return res;\n } else {\n return val;\n }\n } // END decodeValue\n\n function parseKeyWithTags(key) {\n const match = key.match(/(.+)(<!(.)>)/);\n if (match) {\n return [match[1], match[3]];\n }\n // Try multi-character tags like array types [,D,]\n const multiMatch = key.match(/(.+)(<!!(.+)>)/);\n if (multiMatch) {\n return [multiMatch[1], multiMatch[3]];\n }\n // Also handle array type tags that start with [\n const arrayMatch = key.match(/(.+)(<!\\[(.*)>)/);\n if (arrayMatch) {\n return [arrayMatch[1], '[' + arrayMatch[3]];\n }\n return [key, undefined];\n } // END parseKeyWithTags\n\n for (const key in data) {\n const [name, tag] = parseKeyWithTags(key);\n // Start with path containing just the key name\n result[name] = decodeValue.call([name], name, tag, data[key]);\n }\n pointers2Res.forEach(changeAttributeReference.bind(null, result));\n return result;\n} // END decode\n\nfunction changeAttributeReference(obj, [refPath, attrPath]) {\n // refPath and attrPath are now arrays, no splitting needed\n const refKeys = refPath || [];\n const attrKeys = attrPath || [];\n\n // Get the reference target by traversing refPath\n let ref = obj;\n for (let i = 0; i < refKeys.length; i++) {\n ref = ref[refKeys[i]];\n }\n\n // Get the parent of the attribute to set\n let attr = obj;\n for (let i = 0; i < attrKeys.length - 1; i++) {\n attr = attr[attrKeys[i]];\n }\n\n // Set the attribute to point to the reference\n attr[attrKeys[attrKeys.length - 1]] = ref;\n return obj;\n} // END changeAttributeReference\n\n\nmodule.exports = { parse, stringify, encode, decode };\n", "import jss from '../../utils/jss'\n\n/**\n * HTTP Streaming transport - fallback when WebSocket is blocked\n * Uses fetch + ReadableStream for receiving, POST for sending\n */\n\n/**\n * Get base URL for polling endpoints\n */\nfunction getPollUrl() {\n const hostname = window.location.hostname\n const localServers = [\"localhost\", \"127.0.0.1\", \"[::1]\"]\n const isLocal = localServers.includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n\n // Use window.location.port if available, otherwise fallback (9010 for local dev, 443/80 for prod)\n const port = window.location.port || (isLocal ? 9010 : (isHttps ? 443 : 80))\n\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : \"\"\n\n return `${protocol}://${hostname}${portSuffix}/api/ape/poll`\n}\n\n/**\n * Parse JSON objects from a streaming buffer by counting braces\n * Handles strings containing braces correctly\n */\nfunction parseStreamBuffer(buffer) {\n const messages = []\n let start = -1\n let depth = 0\n let inString = false\n let escaped = false\n\n for (let i = 0; i < buffer.length; i++) {\n const char = buffer[i]\n\n if (escaped) {\n escaped = false\n continue\n }\n\n if (char === '\\\\' && inString) {\n escaped = true\n continue\n }\n\n if (char === '\"') {\n inString = !inString\n continue\n }\n\n if (inString) continue\n\n if (char === '{') {\n if (depth === 0) {\n start = i\n }\n depth++\n } else if (char === '}') {\n depth--\n if (depth === 0 && start !== -1) {\n const jsonStr = buffer.slice(start, i + 1)\n try {\n messages.push(jss.parse(jsonStr))\n } catch (e) {\n console.error('\uD83E\uDD8D Failed to parse stream message:', e)\n }\n start = -1\n }\n }\n }\n\n // Return remaining buffer (incomplete message)\n const remaining = start !== -1 ? buffer.slice(start) : ''\n return { messages, remaining }\n}\n\n/**\n * Create streaming transport instance\n */\nfunction createStreamingTransport() {\n let isActive = false\n let abortController = null\n let streamBuffer = ''\n let reconnectTimer = null\n\n // Callbacks\n let onMessage = () => { }\n let onOpen = () => { }\n let onClose = () => { }\n let onError = () => { }\n\n /**\n * Start the streaming connection\n */\n async function connect() {\n if (isActive) return\n\n isActive = true\n abortController = new AbortController()\n\n try {\n const response = await fetch(getPollUrl(), {\n method: 'GET',\n credentials: 'include',\n signal: abortController.signal,\n headers: {\n 'Accept': 'application/json'\n }\n })\n\n if (!response.ok) {\n throw new Error(`Stream connect failed: ${response.status}`)\n }\n\n onOpen()\n\n const reader = response.body.getReader()\n const decoder = new TextDecoder()\n\n async function read() {\n while (isActive) {\n try {\n const { done, value } = await reader.read()\n\n if (done) {\n // Stream ended - reconnect\n scheduleReconnect()\n return\n }\n\n streamBuffer += decoder.decode(value, { stream: true })\n const { messages, remaining } = parseStreamBuffer(streamBuffer)\n streamBuffer = remaining\n\n for (const msg of messages) {\n // Skip heartbeat messages\n if (msg.type === '__heartbeat__') continue\n onMessage(msg)\n }\n } catch (readErr) {\n if (readErr.name === 'AbortError') return\n console.error('\uD83E\uDD8D Stream read error:', readErr)\n scheduleReconnect()\n return\n }\n }\n }\n\n read()\n\n } catch (err) {\n if (err.name === 'AbortError') return\n\n console.error('\uD83E\uDD8D Stream connection error:', err)\n onError(err)\n scheduleReconnect()\n }\n }\n\n /**\n * Schedule reconnection with small delay\n */\n function scheduleReconnect() {\n if (!isActive) return\n\n if (reconnectTimer) {\n clearTimeout(reconnectTimer)\n }\n\n reconnectTimer = setTimeout(() => {\n if (isActive) {\n connect()\n }\n }, 500)\n }\n\n /**\n * Send a message via POST\n */\n async function send(type, data, createdAt) {\n const payload = {\n type,\n data,\n createdAt: new Date(createdAt)\n }\n\n const response = await fetch(getPollUrl(), {\n method: 'POST',\n credentials: 'include',\n headers: {\n 'Content-Type': 'application/json'\n },\n body: jss.stringify(payload)\n })\n\n if (!response.ok) {\n const error = await response.json().catch(() => ({ error: 'Unknown error' }))\n throw new Error(error.error || `Request failed: ${response.status}`)\n }\n\n const result = jss.parse(await response.text())\n return result.data\n }\n\n /**\n * Close the streaming connection\n */\n function close() {\n isActive = false\n\n if (reconnectTimer) {\n clearTimeout(reconnectTimer)\n reconnectTimer = null\n }\n\n if (abortController) {\n abortController.abort()\n abortController = null\n }\n\n streamBuffer = ''\n onClose()\n }\n\n return {\n connect,\n send,\n close,\n isConnected: () => isActive,\n set onMessage(fn) { onMessage = fn },\n set onOpen(fn) { onOpen = fn },\n set onClose(fn) { onClose = fn },\n set onError(fn) { onError = fn }\n }\n}\n\nexport { createStreamingTransport, getPollUrl }\n", "import messageHash from '../utils/messageHash'\nimport jss from '../utils/jss'\nimport { createStreamingTransport } from './transports/streaming'\n\nlet connect;\n\n// Connection state enum\nconst ConnectionState = {\n Offline: 'offline', // navigator.onLine = false\n Walled: 'walled', // Captive portal detected (ping failed)\n Disconnected: 'disconnected',\n Connecting: 'connecting',\n Connected: 'connected',\n Closing: 'closing'\n}\n\n// Connection state tracking - start with offline check\nlet connectionState = (typeof navigator !== 'undefined' && !navigator.onLine)\n ? ConnectionState.Offline\n : ConnectionState.Disconnected\nconst connectionChangeListeners = []\n\nfunction notifyConnectionChange(newState) {\n if (connectionState !== newState) {\n connectionState = newState\n connectionChangeListeners.forEach(fn => fn(newState))\n }\n}\n\n// Configuration\nlet configuredTransport = 'auto' // 'auto' | 'websocket' | 'polling'\n\n// Transport state\nlet currentTransport = null // 'websocket' | 'polling'\nlet streamingTransport = null\nlet wsRetryTimer = null\nlet networkCheckTimer = null\nconst WS_FALLBACK_TIMEOUT = 4000 // Time to wait for WS before fallback\nconst WS_RETRY_INTERVAL = 30000 // Retry WebSocket while in polling mode\nconst PING_TIMEOUT = 3000 // Timeout for ping check\nconst MAX_PING_CLOCK_SKEW = 60000 // Max allowed time difference (60s)\n\n/**\n * Check if running in dev/local mode\n */\nfunction isDevMode() {\n if (typeof window === 'undefined') return false\n return ['localhost', '127.0.0.1', '[::1]'].includes(window.location.hostname)\n}\n\n/**\n * Build ping URL for captive portal detection\n */\nfunction getPingUrl() {\n const hostname = window.location.hostname\n const isHttps = window.location.protocol === 'https:'\n const port = window.location.port || (isHttps ? 443 : 80)\n const protocol = isHttps ? 'https' : 'http'\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : ''\n return `${protocol}://${hostname}${portSuffix}/api/ape/ping`\n}\n\n/**\n * Check for captive portal by pinging /api/ape/ping\n * Returns 'ok' if real internet, 'walled' if captive portal detected\n */\nasync function checkCaptivePortal() {\n try {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), PING_TIMEOUT)\n\n const response = await fetch(getPingUrl(), {\n cache: 'no-store',\n signal: controller.signal\n })\n clearTimeout(timeoutId)\n\n if (!response.ok) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed: HTTP', response.status)\n }\n return 'walled'\n }\n\n const data = await response.json()\n\n // Verify response is genuine (not a captive portal redirect page)\n if (data?.ok !== true) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed: invalid response', data)\n }\n return 'walled'\n }\n\n // Validate timestamp to detect proxy replay attacks\n if (typeof data.ts === 'number') {\n const now = Date.now()\n const skew = Math.abs(now - data.ts)\n if (skew > MAX_PING_CLOCK_SKEW) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed: timestamp too old/stale (skew:', skew, 'ms)')\n }\n return 'walled'\n }\n }\n\n return 'ok'\n } catch (err) {\n if (isDevMode()) {\n console.error('\uD83E\uDD8D [DEV] Ping failed:', err.message || err)\n }\n return 'walled'\n }\n}\n\n/**\n * Setup navigator.onLine event listeners\n */\nfunction setupOnlineListeners() {\n if (typeof window === 'undefined') return\n\n window.addEventListener('online', () => {\n console.log('\uD83E\uDD8D Browser went online, checking network...')\n // Trigger reconnection attempt\n attemptConnection()\n })\n\n window.addEventListener('offline', () => {\n console.log('\uD83E\uDD8D Browser went offline')\n notifyConnectionChange(ConnectionState.Offline)\n })\n}\n\n// Setup listeners on module load (browser only)\nif (typeof window !== 'undefined') {\n setupOnlineListeners()\n}\n\n\n\n/**\n * Get WebSocket URL - auto-detects from window.location, keeps /api/ape path\n */\nfunction getSocketUrl() {\n const hostname = window.location.hostname\n const localServers = [\"localhost\", \"127.0.0.1\", \"[::1]\"]\n const isLocal = localServers.includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n\n // Use window.location.port if available, otherwise fallback (9010 for local dev, 443/80 for prod)\n const port = window.location.port || (isLocal ? 9010 : (isHttps ? 443 : 80))\n\n // Build URL - keep /api/ape path\n const protocol = isHttps ? \"wss\" : \"ws\"\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : \"\"\n\n return `${protocol}://${hostname}${portSuffix}/api/ape`\n}\n\nlet reconnect = false\nconst connectTimeout = 5000\nconst totalRequestTimeout = 10000\n//const location = window.location\n\nconst joinKey = \"/\"\n// Properties accessed directly on `ape` that should NOT be intercepted\nconst reservedKeys = new Set(['on', 'onConnectionChange', 'transport'])\nconst handler = {\n get(fn, key) {\n // Skip proxy interception for reserved keys - return actual property\n if (reservedKeys.has(key)) {\n return fn[key]\n }\n const wrapperFn = function (a, b) {\n let path = joinKey + key, body;\n if (2 === arguments.length) {\n path += a\n body = b\n } else {\n body = a\n }\n return fn(path, body)\n }\n return new Proxy(wrapperFn, handler)\n } // END get\n}\n\nfunction wrap(api) {\n return new Proxy(api, handler)\n}\n\nlet __socket = false, ready = false, wsSend = false;\nconst waitingOn = {};\n\nlet aWaitingSend = []\nconst receiverArray = [];\nconst ofTypesOb = {};\n\n/**\n * Switch to streaming transport (HTTP long polling fallback)\n */\nfunction switchToStreaming() {\n console.log('\uD83E\uDD8D Switching to HTTP streaming transport')\n currentTransport = 'polling'\n\n if (!streamingTransport) {\n streamingTransport = createStreamingTransport()\n\n // Handle incoming messages from streaming transport\n streamingTransport.onMessage = async (msg) => {\n const { err, type, data } = msg\n\n // Process linked resources and shared files\n let processedData = data\n if (data && !err) {\n try {\n processedData = await fetchLinkedResources(data)\n processedData = await fetchSharedFiles(processedData)\n } catch (fetchErr) {\n console.error(`\uD83E\uDD8D Failed to hydrate streaming data:`, fetchErr)\n }\n }\n\n // Dispatch to type-specific handlers\n if (ofTypesOb[type]) {\n ofTypesOb[type].forEach(worker => worker({ err, type, data: processedData }))\n }\n // Dispatch to general handlers\n receiverArray.forEach(worker => worker({ err, type, data: processedData }))\n }\n\n streamingTransport.onOpen = () => {\n ready = true\n notifyConnectionChange(ConnectionState.Connected)\n console.log('\uD83E\uDD8D HTTP streaming connected')\n\n // Flush waiting messages\n aWaitingSend.forEach(({ type, data, resolve, reject, waiting, createdAt, timer }) => {\n clearTimeout(timer)\n const resultPromise = streamingSend(type, data, createdAt)\n if (waiting) {\n resultPromise.then(resolve).catch(reject)\n }\n })\n aWaitingSend = []\n\n // Start background WebSocket retry\n startWsRetry()\n }\n\n streamingTransport.onClose = () => {\n ready = false\n notifyConnectionChange(ConnectionState.Disconnected)\n }\n\n streamingTransport.onError = (err) => {\n console.error('\uD83E\uDD8D Streaming error:', err)\n }\n }\n\n streamingTransport.connect()\n}\n\n/**\n * Send via streaming transport\n */\nfunction streamingSend(type, data, createdAt) {\n return streamingTransport.send(type, data, createdAt)\n}\n\n/**\n * Start background retry for WebSocket (while in polling mode)\n */\nfunction startWsRetry() {\n if (wsRetryTimer) return\n if (currentTransport !== 'polling') return\n if (configuredTransport === 'polling') return // User explicitly wants polling only\n\n wsRetryTimer = setInterval(() => {\n if (currentTransport !== 'polling') {\n clearInterval(wsRetryTimer)\n wsRetryTimer = null\n return\n }\n\n console.log('\uD83E\uDD8D Attempting WebSocket reconnection...')\n tryWebSocket(true)\n }, WS_RETRY_INTERVAL)\n}\n\n/**\n * Try to establish WebSocket connection\n * @param {boolean} isRetry - If true, this is a background retry attempt\n */\nfunction tryWebSocket(isRetry = false) {\n const ws = new WebSocket(getSocketUrl())\n let fallbackTimer = null\n\n // Set fallback timeout (only for initial connection, not retries)\n if (!isRetry && configuredTransport === 'auto') {\n fallbackTimer = setTimeout(() => {\n if (ws.readyState !== WebSocket.OPEN) {\n console.log('\uD83E\uDD8D WebSocket timeout, falling back to HTTP streaming')\n ws.close()\n switchToStreaming()\n }\n }, WS_FALLBACK_TIMEOUT)\n }\n\n ws.onopen = () => {\n if (fallbackTimer) clearTimeout(fallbackTimer)\n\n // If this is a retry and we're in polling mode, switch back to WebSocket\n if (isRetry && currentTransport === 'polling') {\n console.log('\uD83E\uDD8D WebSocket reconnected, switching from HTTP streaming')\n if (streamingTransport) {\n streamingTransport.close()\n }\n if (wsRetryTimer) {\n clearInterval(wsRetryTimer)\n wsRetryTimer = null\n }\n }\n\n currentTransport = 'websocket'\n __socket = ws\n ready = true\n notifyConnectionChange(ConnectionState.Connected)\n\n aWaitingSend.forEach(({ type, data, resolve, reject, waiting, createdAt, timer }) => {\n clearTimeout(timer)\n const resultPromise = wsSend(type, data, createdAt)\n if (waiting) {\n resultPromise.then(resolve).catch(reject)\n }\n })\n aWaitingSend = []\n }\n\n ws.onmessage = async function (event) {\n const { err, type, queryId, data } = jss.parse(event.data)\n\n // Messages with queryId must fulfill matching promise\n if (queryId) {\n if (waitingOn[queryId]) {\n // Check for linked resources and fetch them before resolving\n if (data && !err) {\n try {\n let hydratedData = await fetchLinkedResources(data)\n hydratedData = await fetchSharedFiles(hydratedData)\n waitingOn[queryId](err, hydratedData)\n } catch (fetchErr) {\n waitingOn[queryId](fetchErr, null)\n }\n } else {\n waitingOn[queryId](err, data)\n }\n delete waitingOn[queryId]\n } else {\n console.error(`\uD83E\uDD8D No matching queryId: ${queryId}`)\n }\n return\n }\n\n // Only messages WITHOUT queryId go to setOnReceiver\n let processedData = data\n if (data && !err) {\n try {\n processedData = await fetchLinkedResources(data)\n processedData = await fetchSharedFiles(processedData)\n } catch (fetchErr) {\n console.error(`\uD83E\uDD8D Failed to hydrate broadcast data:`, fetchErr)\n }\n }\n\n if (ofTypesOb[type]) {\n ofTypesOb[type].forEach(worker => worker({ err, type, data: processedData }))\n }\n receiverArray.forEach(worker => worker({ err, type, data: processedData }))\n }\n\n ws.onerror = function (err) {\n if (fallbackTimer) clearTimeout(fallbackTimer)\n console.error('socket ERROR:', err)\n\n // On initial connection error in auto mode, fallback to streaming\n if (!isRetry && configuredTransport === 'auto' && !ready) {\n switchToStreaming()\n }\n }\n\n ws.onclose = function (event) {\n if (fallbackTimer) clearTimeout(fallbackTimer)\n console.warn('socket disconnect:', event)\n __socket = false\n ready = false\n\n // Only notify disconnected if we're on websocket transport\n if (currentTransport === 'websocket') {\n notifyConnectionChange(ConnectionState.Disconnected)\n setTimeout(() => reconnect && connectSocket(), 500)\n }\n }\n}\n\n/**\n * Find all L-tagged (binary link) properties in data\n * Returns array of { path, hash }\n */\nfunction findLinkedResources(obj, path = '') {\n const resources = []\n\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return resources\n }\n\n if (Array.isArray(obj)) {\n for (let i = 0; i < obj.length; i++) {\n resources.push(...findLinkedResources(obj[i], path ? `${path}.${i}` : String(i)))\n }\n return resources\n }\n\n for (const key of Object.keys(obj)) {\n // Check for L-tag in key (from JJS encoding: key<!L>)\n if (key.endsWith('<!L>')) {\n const cleanKey = key.slice(0, -4)\n const hash = obj[key]\n resources.push({\n path: path ? `${path}.${cleanKey}` : cleanKey,\n hash,\n originalKey: key\n })\n } else {\n resources.push(...findLinkedResources(obj[key], path ? `${path}.${key}` : key))\n }\n }\n\n return resources\n}\n\n/**\n * Find all F-tagged (shared file) properties in data\n * Returns array of { path, hash, originalKey }\n */\nfunction findFileTags(obj, path = '') {\n const files = []\n\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return files\n }\n\n if (Array.isArray(obj)) {\n for (let i = 0; i < obj.length; i++) {\n files.push(...findFileTags(obj[i], path ? `${path}.${i}` : String(i)))\n }\n return files\n }\n\n for (const key of Object.keys(obj)) {\n // Check for F-tag in key (client-to-client shared file marker)\n if (key.endsWith('<!F>')) {\n const cleanKey = key.slice(0, -4)\n const hash = obj[key]\n files.push({\n path: path ? `${path}.${cleanKey}` : cleanKey,\n hash,\n originalKey: key\n })\n } else {\n files.push(...findFileTags(obj[key], path ? `${path}.${key}` : key))\n }\n }\n\n return files\n}\n\n/**\n * Clean up F-tagged keys (rename key<!F> to key)\n */\nfunction cleanFileTags(obj) {\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return obj\n }\n\n if (Array.isArray(obj)) {\n return obj.map(cleanFileTags)\n }\n\n const cleaned = {}\n for (const key of Object.keys(obj)) {\n if (key.endsWith('<!F>')) {\n const cleanKey = key.slice(0, -4)\n cleaned[cleanKey] = obj[key]\n } else {\n cleaned[key] = cleanFileTags(obj[key])\n }\n }\n return cleaned\n}\n\n/**\n * Fetch shared files (client-to-client transfers)\n * Retries if upload is still in progress\n */\nasync function fetchSharedFiles(data, maxRetries = 5) {\n const files = findFileTags(data)\n\n if (files.length === 0) {\n return data\n }\n\n console.log(`\uD83E\uDD8D Fetching ${files.length} shared file(s)`)\n\n const cleanedData = cleanFileTags(data)\n\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = window.location.port || (isLocal ? 9010 : (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n await Promise.all(files.map(async ({ path, hash }) => {\n let retries = 0\n let backoff = 100 // Start with 100ms\n\n while (retries < maxRetries) {\n try {\n const response = await fetch(`${baseUrl}/api/ape/data/${hash}`, {\n credentials: 'include'\n })\n\n if (!response.ok) {\n // 404 might mean file not uploaded yet, retry\n if (response.status === 404 && retries < maxRetries - 1) {\n retries++\n await new Promise(r => setTimeout(r, backoff))\n backoff *= 2 // Exponential backoff\n continue\n }\n throw new Error(`Failed to fetch shared file: ${response.status}`)\n }\n\n const arrayBuffer = await response.arrayBuffer()\n setValueAtPath(cleanedData, path, arrayBuffer)\n\n // Check if upload is still in progress\n const isComplete = response.headers.get('X-Ape-Complete') === '1'\n if (!isComplete) {\n console.log(`\uD83E\uDD8D Shared file ${hash} still uploading (${response.headers.get('X-Ape-Total-Received') || '?'} bytes)`)\n }\n break\n } catch (err) {\n if (retries >= maxRetries - 1) {\n console.error(`\uD83E\uDD8D Failed to fetch shared file at ${path}:`, err)\n setValueAtPath(cleanedData, path, null)\n }\n retries++\n await new Promise(r => setTimeout(r, backoff))\n backoff *= 2\n }\n }\n }))\n\n return cleanedData\n}\n\n/**\n * Set a value at a nested path in an object\n */\nfunction setValueAtPath(obj, path, value) {\n const parts = path.split('.')\n let current = obj\n\n for (let i = 0; i < parts.length - 1; i++) {\n current = current[parts[i]]\n }\n\n current[parts[parts.length - 1]] = value\n}\n\n/**\n * Clean up L-tagged keys (rename key<!L> to key)\n */\nfunction cleanLinkedKeys(obj) {\n if (obj === null || obj === undefined || typeof obj !== 'object') {\n return obj\n }\n\n if (Array.isArray(obj)) {\n return obj.map(cleanLinkedKeys)\n }\n\n const cleaned = {}\n for (const key of Object.keys(obj)) {\n if (key.endsWith('<!L>')) {\n const cleanKey = key.slice(0, -4)\n cleaned[cleanKey] = obj[key]\n } else {\n cleaned[key] = cleanLinkedKeys(obj[key])\n }\n }\n return cleaned\n}\n\n/**\n * Fetch binary resources and hydrate data object\n */\nasync function fetchLinkedResources(data, clientId) {\n const resources = findLinkedResources(data)\n\n if (resources.length === 0) {\n return data\n }\n\n console.log(`\uD83E\uDD8D Fetching ${resources.length} binary resource(s)`)\n\n const cleanedData = cleanLinkedKeys(data)\n\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = window.location.port || (isLocal ? 9010 : (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n await Promise.all(resources.map(async ({ path, hash }) => {\n try {\n const response = await fetch(`${baseUrl}/api/ape/data/${hash}`, {\n credentials: 'include',\n headers: {\n 'X-Ape-Client-Id': clientId || ''\n }\n })\n\n if (!response.ok) {\n throw new Error(`Failed to fetch binary resource: ${response.status}`)\n }\n\n const arrayBuffer = await response.arrayBuffer()\n setValueAtPath(cleanedData, path, arrayBuffer)\n } catch (err) {\n console.error(`\uD83E\uDD8D Failed to fetch binary resource at ${path}:`, err)\n setValueAtPath(cleanedData, path, null)\n }\n }))\n\n return cleanedData\n}\n\n/**\n * Attempt to establish connection with network pre-checks\n */\nasync function attemptConnection() {\n // Check if browser is online\n if (typeof navigator !== 'undefined' && !navigator.onLine) {\n notifyConnectionChange(ConnectionState.Offline)\n return\n }\n\n // Perform captive portal check\n notifyConnectionChange(ConnectionState.Connecting)\n const pingResult = await checkCaptivePortal()\n\n if (pingResult === 'walled') {\n notifyConnectionChange(ConnectionState.Walled)\n // Retry network check periodically\n scheduleNetworkRetry()\n return\n }\n\n // Network is good, proceed with socket connection\n proceedWithConnection()\n}\n\n/**\n * Schedule a retry of network check (for walled/offline states)\n */\nfunction scheduleNetworkRetry() {\n if (networkCheckTimer) return\n networkCheckTimer = setTimeout(() => {\n networkCheckTimer = null\n attemptConnection()\n }, WS_RETRY_INTERVAL)\n}\n\n/**\n * Proceed with WebSocket/polling connection after network checks pass\n */\nfunction proceedWithConnection() {\n // Determine which transport to use\n if (configuredTransport === 'polling') {\n switchToStreaming()\n } else {\n // 'auto' or 'websocket' - try WebSocket first\n tryWebSocket(false)\n }\n}\n\nfunction connectSocket() {\n // Skip if already connected or connecting\n if (__socket && __socket.readyState !== WebSocket.CLOSED) {\n return buildClientInterface()\n }\n if (currentTransport === 'polling' && streamingTransport?.isConnected()) {\n return buildClientInterface()\n }\n if (connectionState === ConnectionState.Connecting) {\n return buildClientInterface()\n }\n\n // Start connection with network pre-checks\n attemptConnection()\n\n return buildClientInterface()\n}\n\n/**\n * Check if value is binary data (ArrayBuffer, typed array, or Blob)\n */\nfunction isBinaryData(value) {\n if (value === null || value === undefined) return false\n return value instanceof ArrayBuffer ||\n ArrayBuffer.isView(value) ||\n (typeof Blob !== 'undefined' && value instanceof Blob)\n}\n\n/**\n * Get binary type tag (A for ArrayBuffer, B for Blob)\n */\nfunction getBinaryTag(value) {\n if (typeof Blob !== 'undefined' && value instanceof Blob) return 'B'\n return 'A'\n}\n\n/**\n * Generate a simple hash for binary upload\n */\nfunction generateUploadHash(path) {\n let hash = 0\n for (let i = 0; i < path.length; i++) {\n const char = path.charCodeAt(i)\n hash = ((hash << 5) - hash) + char\n hash = hash & hash\n }\n return Math.abs(hash).toString(36)\n}\n\n/**\n * Find and extract binary data from payload\n * Returns { processedData, uploads: [{ path, hash, data, tag }] }\n */\nfunction processBinaryForUpload(data, path = '') {\n if (data === null || data === undefined) {\n return { processedData: data, uploads: [] }\n }\n\n if (isBinaryData(data)) {\n const tag = getBinaryTag(data)\n const hash = generateUploadHash(path || 'root')\n return {\n processedData: { [`__ape_upload__`]: hash },\n uploads: [{ path, hash, data, tag }]\n }\n }\n\n if (Array.isArray(data)) {\n const processedArray = []\n const allUploads = []\n\n for (let i = 0; i < data.length; i++) {\n const itemPath = path ? `${path}.${i}` : String(i)\n const { processedData, uploads } = processBinaryForUpload(data[i], itemPath)\n processedArray.push(processedData)\n allUploads.push(...uploads)\n }\n\n return { processedData: processedArray, uploads: allUploads }\n }\n\n if (typeof data === 'object') {\n const processedObj = {}\n const allUploads = []\n\n for (const key of Object.keys(data)) {\n const itemPath = path ? `${path}.${key}` : key\n const { processedData, uploads } = processBinaryForUpload(data[key], itemPath)\n\n // If this was binary data, mark the key with <!B> or <!A> tag\n if (uploads.length > 0 && processedData?.__ape_upload__) {\n const tag = uploads[uploads.length - 1].tag\n processedObj[`${key}<!${tag}>`] = processedData.__ape_upload__\n } else {\n processedObj[key] = processedData\n }\n allUploads.push(...uploads)\n }\n\n return { processedData: processedObj, uploads: allUploads }\n }\n\n return { processedData: data, uploads: [] }\n}\n\n/**\n * Find and extract binary data for SHARING (client-to-client)\n * Uses <!F> tag instead of <!A>/<!B>\n * Returns { processedData, shares: [{ path, hash, data }] }\n */\nfunction processBinaryForSharing(data, path = '') {\n if (data === null || data === undefined) {\n return { processedData: data, shares: [] }\n }\n\n if (isBinaryData(data)) {\n const hash = generateUploadHash(path || 'share')\n return {\n processedData: { [`__ape_share__`]: hash },\n shares: [{ path, hash, data }]\n }\n }\n\n if (Array.isArray(data)) {\n const processedArray = []\n const allShares = []\n\n for (let i = 0; i < data.length; i++) {\n const itemPath = path ? `${path}.${i}` : String(i)\n const { processedData, shares } = processBinaryForSharing(data[i], itemPath)\n processedArray.push(processedData)\n allShares.push(...shares)\n }\n\n return { processedData: processedArray, shares: allShares }\n }\n\n if (typeof data === 'object') {\n const processedObj = {}\n const allShares = []\n\n for (const key of Object.keys(data)) {\n const itemPath = path ? `${path}.${key}` : key\n const { processedData, shares } = processBinaryForSharing(data[key], itemPath)\n\n // If this was binary data, mark the key with <!F> tag\n if (shares.length > 0 && processedData?.__ape_share__) {\n processedObj[`${key}<!F>`] = processedData.__ape_share__\n } else {\n processedObj[key] = processedData\n }\n allShares.push(...shares)\n }\n\n return { processedData: processedObj, shares: allShares }\n }\n\n return { processedData: data, shares: [] }\n}\n\n/**\n * Upload shared files via HTTP PUT\n * Uses different endpoint pattern for streaming files\n */\nasync function uploadSharedFiles(shares) {\n if (shares.length === 0) return\n\n // Build base URL\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = window.location.port || (isLocal ? 9010 : (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n console.log(`\uD83E\uDD8D Uploading ${shares.length} shared file(s)`)\n\n await Promise.all(shares.map(async ({ hash, data }) => {\n try {\n // For shared files, use upload pattern with hash as both queryId and pathHash\n const response = await fetch(`${baseUrl}/api/ape/data/_share/${hash}`, {\n method: 'PUT',\n credentials: 'include',\n headers: {\n 'Content-Type': 'application/octet-stream'\n },\n body: data\n })\n\n if (!response.ok) {\n throw new Error(`Shared upload failed: ${response.status}`)\n }\n } catch (err) {\n console.error(`\uD83E\uDD8D Failed to upload shared file ${hash}:`, err)\n throw err\n }\n }))\n}\n\n/**\n * Upload binary data via HTTP PUT\n */\nasync function uploadBinaryData(queryId, uploads) {\n if (uploads.length === 0) return\n\n // Build base URL\n const hostname = window.location.hostname\n const isLocal = [\"localhost\", \"127.0.0.1\", \"[::1]\"].includes(hostname)\n const isHttps = window.location.protocol === \"https:\"\n const port = window.location.port || (isLocal ? 9010 : (isHttps ? 443 : 80))\n const protocol = isHttps ? \"https\" : \"http\"\n const portSuffix = (port !== 80 && port !== 443) ? `:${port}` : \"\"\n const baseUrl = `${protocol}://${hostname}${portSuffix}`\n\n console.log(`\uD83E\uDD8D Uploading ${uploads.length} binary file(s)`)\n\n await Promise.all(uploads.map(async ({ hash, data }) => {\n try {\n const response = await fetch(`${baseUrl}/api/ape/data/${queryId}/${hash}`, {\n method: 'PUT',\n credentials: 'include',\n headers: {\n 'Content-Type': 'application/octet-stream'\n },\n body: data\n })\n\n if (!response.ok) {\n throw new Error(`Upload failed: ${response.status}`)\n }\n } catch (err) {\n console.error(`\uD83E\uDD8D Failed to upload binary at ${hash}:`, err)\n throw err\n }\n }))\n}\n\nwsSend = function (type, data, createdAt, dirctCall) {\n let rej, promiseIsLive = false;\n const timeLetForReqToBeMade = (createdAt + totalRequestTimeout) - Date.now()\n\n const timer = setTimeout(() => {\n if (promiseIsLive) {\n rej(new Error(\"Request Timedout for :\" + type))\n }\n }, timeLetForReqToBeMade);\n\n // Process binary data for upload\n const { processedData, uploads } = processBinaryForUpload(data)\n\n const payload = {\n type,\n data: processedData,\n //referer:window.location.href,\n createdAt: new Date(createdAt),\n requestedAt: dirctCall ? undefined\n : new Date()\n }\n const message = jss.stringify(payload)\n const queryId = messageHash(message);\n\n const replyPromise = new Promise((resolve, reject) => {\n rej = reject\n waitingOn[queryId] = (err, result) => {\n clearTimeout(timer)\n replyPromise.then = next.bind(replyPromise)\n if (err) {\n reject(err)\n } else {\n resolve(result)\n }\n }\n __socket.send(message);\n\n // Upload binary data after sending WS message\n if (uploads.length > 0) {\n uploadBinaryData(queryId, uploads).catch(err => {\n console.error('\uD83E\uDD8D Binary upload failed:', err)\n // The server will timeout waiting for the upload\n })\n }\n });\n const next = replyPromise.then;\n replyPromise.then = worker => {\n promiseIsLive = true;\n replyPromise.then = next.bind(replyPromise)\n replyPromise.catch = err.bind(replyPromise)\n return next.call(replyPromise, worker)\n }\n const err = replyPromise.catch;\n replyPromise.catch = worker => {\n promiseIsLive = true;\n replyPromise.catch = err.bind(replyPromise)\n replyPromise.then = next.bind(replyPromise)\n return err.call(replyPromise, worker)\n }\n return replyPromise\n} // END wsSend\n\n\nconst sender = (type, data) => {\n if (\"string\" !== typeof type) {\n throw new Error(\"Missing Path vaule\")\n }\n\n const createdAt = Date.now()\n\n if (ready) {\n return wsSend(type, data, createdAt, true)\n }\n\n const timeLetForReqToBeMade = (createdAt + connectTimeout) - Date.now() // 5sec for reconnect\n\n const timer = setTimeout(() => {\n const errMessage = \"Request not sent for :\" + type\n if (payload.waiting) {\n payload.reject(new Error(errMessage))\n } else {\n throw new Error(errMessage)\n }\n }, timeLetForReqToBeMade);\n\n const payload = { type, data, resolve: undefined, reject: undefined, waiting: false, createdAt, timer };\n const waitingOnOpen = new Promise((res, rej) => { payload.resolve = res; payload.reject = rej; })\n\n const waitingOnOpenThen = waitingOnOpen.then;\n const waitingOnOpenCatch = waitingOnOpen.catch;\n waitingOnOpen.then = worker => {\n payload.waiting = true;\n waitingOnOpen.then = waitingOnOpenThen.bind(waitingOnOpen)\n waitingOnOpen.catch = waitingOnOpenCatch.bind(waitingOnOpen)\n return waitingOnOpenThen.call(waitingOnOpen, worker)\n }\n waitingOnOpen.catch = worker => {\n payload.waiting = true;\n waitingOnOpen.catch = waitingOnOpenCatch.bind(waitingOnOpen)\n waitingOnOpen.then = waitingOnOpenThen.bind(waitingOnOpen)\n return waitingOnOpenCatch.call(waitingOnOpen, worker)\n }\n\n aWaitingSend.push(payload)\n if (!__socket) {\n connectSocket()\n }\n\n return waitingOnOpen\n} // END sender\n\n/**\n * Build the client interface object\n */\nfunction buildClientInterface() {\n return {\n sender: wrap(sender),\n setOnReceiver: (onTypeStFn, handlerFn) => {\n if (\"string\" === typeof onTypeStFn) {\n // Replace handler for this type (prevents duplicates in React StrictMode)\n ofTypesOb[onTypeStFn] = [handlerFn]\n } else {\n // For general receivers, prevent duplicates by checking\n if (!receiverArray.includes(onTypeStFn)) {\n receiverArray.push(onTypeStFn)\n }\n }\n },\n onConnectionChange: (handler) => {\n connectionChangeListeners.push(handler)\n // Immediately call with current state\n handler(connectionState)\n // Return unsubscribe function\n return () => {\n const idx = connectionChangeListeners.indexOf(handler)\n if (idx > -1) connectionChangeListeners.splice(idx, 1)\n }\n },\n // Expose current transport type (read-only)\n get transport() { return currentTransport }\n }\n}\n\nconnectSocket.autoReconnect = () => reconnect = true\nconnectSocket.ConnectionState = ConnectionState\nconnect = connectSocket\n\nexport default connect;\nexport { ConnectionState };\n", "/**\n * Unified api-ape export for browser\n * \n * Auto-detects browser environment, initializes client, and buffers\n * calls until the connection is ready. No more getApeClient().then()!\n * \n * Usage:\n * import api from 'api-ape'\n * \n * // Properties are proxied - calls buffer until connected\n * api.message({ user: 'Bob', text: 'Hello!' })\n * \n * // Subscribe to broadcasts\n * api.on('message', (data) => console.log(data))\n * \n * // Check connection state\n * api.onConnectionChange((state) => console.log(state))\n */\n\n// Only run this in browser environments\nconst isBrowser = typeof window !== 'undefined' && typeof document !== 'undefined'\n\nlet clientPromise = null\nlet resolvedClient = null\nconst bufferedCalls = []\nconst bufferedReceivers = []\nconst connectionChangeHandlers = []\nlet currentConnectionState = 'disconnected'\n\n/**\n * Initialize the client (called once on first use)\n */\nfunction getClient() {\n if (clientPromise) return clientPromise\n\n if (!isBrowser) {\n // Return a dummy object for SSR\n return Promise.resolve(null)\n }\n\n clientPromise = (async () => {\n const connectSocket = (await import('./connectSocket.js')).default\n\n // Connect\n const client = connectSocket()\n connectSocket.autoReconnect()\n\n // Track connection state\n client.onConnectionChange((state) => {\n currentConnectionState = state\n connectionChangeHandlers.forEach(fn => fn(state))\n })\n\n resolvedClient = client\n\n // Flush buffered receivers\n bufferedReceivers.forEach(({ type, handler }) => {\n client.setOnReceiver(type, handler)\n })\n bufferedReceivers.length = 0\n\n // Flush buffered calls\n bufferedCalls.forEach(({ method, args, resolve, reject }) => {\n try {\n const result = client.sender[method](...args)\n if (result && typeof result.then === 'function') {\n result.then(resolve).catch(reject)\n } else {\n resolve(result)\n }\n } catch (err) {\n reject(err)\n }\n })\n bufferedCalls.length = 0\n\n return client\n })()\n\n return clientPromise\n}\n\n/**\n * Create a sender proxy that buffers calls until client is ready\n */\nconst senderProxy = new Proxy({}, {\n get(target, prop) {\n // Reserved properties\n if (prop === 'on') return on\n if (prop === 'onConnectionChange') return onConnectionChange\n if (prop === 'transport') return resolvedClient?.transport || null\n if (prop === 'then' || prop === 'catch') return undefined // Not a Promise\n\n // Return a function that either calls directly or buffers\n return (...args) => {\n // If client is ready, call directly\n if (resolvedClient) {\n return resolvedClient.sender[prop](...args)\n }\n\n // Buffer the call and return a Promise\n return new Promise((resolve, reject) => {\n bufferedCalls.push({ method: prop, args, resolve, reject })\n // Ensure client is initializing\n getClient()\n })\n }\n }\n})\n\n/**\n * Subscribe to broadcasts from the server\n * @param {string} type - Broadcast type to listen for\n * @param {Function} handler - Handler function\n */\nfunction on(type, handler) {\n if (resolvedClient) {\n resolvedClient.setOnReceiver(type, handler)\n } else {\n bufferedReceivers.push({ type, handler })\n getClient()\n }\n}\n\n/**\n * Subscribe to connection state changes\n * @param {Function} handler - Called with state: 'offline' | 'walled' | 'disconnected' | 'connecting' | 'connected'\n * @returns {Function} Unsubscribe function\n */\nfunction onConnectionChange(handler) {\n connectionChangeHandlers.push(handler)\n // Immediately call with current state\n handler(currentConnectionState)\n\n // If client exists, also register with it\n if (resolvedClient) {\n return resolvedClient.onConnectionChange(handler)\n }\n\n // Ensure client is initializing\n getClient()\n\n // Return unsubscribe function\n return () => {\n const idx = connectionChangeHandlers.indexOf(handler)\n if (idx > -1) connectionChangeHandlers.splice(idx, 1)\n }\n}\n\n// Define properties on the proxy to avoid Proxy interception issues\nObject.defineProperty(senderProxy, 'on', {\n value: on,\n writable: false,\n enumerable: false,\n configurable: false\n})\n\nObject.defineProperty(senderProxy, 'onConnectionChange', {\n value: onConnectionChange,\n writable: false,\n enumerable: false,\n configurable: false\n})\n\n// Auto-initialize in browser\nif (isBrowser) {\n getClient()\n}\n\nexport default senderProxy\nexport { on, onConnectionChange, getClient }\n"],
5
+ "mappings": "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",
6
+ "names": ["require_messageHash", "__commonJSMin", "exports", "module", "alphabet", "toBase32", "n", "remainder", "current", "jenkinsOneAtATimeHash", "keyString", "hash", "charIndex", "messageHash", "messageSt", "require_jss", "__commonJSMin", "exports", "module", "encode", "obj", "tagLookup", "visited", "encodeValue", "value", "path", "type", "tag", "visitedEncode", "isArray", "keys", "result", "typesFound", "i", "key", "t", "v", "encodeValueWithVisited", "objKeys", "stringify", "parse", "encoded", "decode", "data", "pointers2Res", "s", "n", "sourceToPointAt", "replaceAtThisPlace", "name", "message", "stack", "err", "a", "o", "decodeValue", "val", "currentPath", "typeTags", "res", "itemPath", "decodedValue", "nam", "parseKeyWithTags", "match", "multiMatch", "arrayMatch", "changeAttributeReference", "refPath", "attrPath", "refKeys", "attrKeys", "ref", "attr", "getPollUrl", "hostname", "isLocal", "isHttps", "port", "protocol", "portSuffix", "parseStreamBuffer", "buffer", "messages", "start", "depth", "inString", "escaped", "i", "char", "jsonStr", "jss", "e", "remaining", "createStreamingTransport", "isActive", "abortController", "streamBuffer", "reconnectTimer", "onMessage", "onOpen", "onClose", "onError", "connect", "response", "reader", "decoder", "read", "done", "value", "scheduleReconnect", "msg", "readErr", "err", "send", "type", "data", "createdAt", "payload", "error", "close", "fn", "import_jss", "init_streaming", "__esmMin", "connectSocket_exports", "__export", "ConnectionState", "connectSocket_default", "notifyConnectionChange", "newState", "connectionState", "connectionChangeListeners", "fn", "isDevMode", "getPingUrl", "hostname", "isHttps", "port", "protocol", "portSuffix", "checkCaptivePortal", "controller", "timeoutId", "PING_TIMEOUT", "response", "data", "now", "skew", "MAX_PING_CLOCK_SKEW", "err", "setupOnlineListeners", "attemptConnection", "getSocketUrl", "isLocal", "wrap", "api", "handler", "switchToStreaming", "currentTransport", "streamingTransport", "createStreamingTransport", "msg", "type", "processedData", "fetchLinkedResources", "fetchSharedFiles", "fetchErr", "ofTypesOb", "worker", "receiverArray", "ready", "aWaitingSend", "resolve", "reject", "waiting", "createdAt", "timer", "resultPromise", "streamingSend", "startWsRetry", "wsRetryTimer", "configuredTransport", "tryWebSocket", "WS_RETRY_INTERVAL", "isRetry", "ws", "fallbackTimer", "WS_FALLBACK_TIMEOUT", "__socket", "wsSend", "event", "queryId", "jss", "waitingOn", "hydratedData", "reconnect", "connectSocket", "findLinkedResources", "obj", "path", "resources", "i", "key", "cleanKey", "hash", "findFileTags", "files", "cleanFileTags", "cleaned", "maxRetries", "cleanedData", "baseUrl", "retries", "backoff", "r", "arrayBuffer", "setValueAtPath", "value", "parts", "current", "cleanLinkedKeys", "clientId", "scheduleNetworkRetry", "proceedWithConnection", "networkCheckTimer", "buildClientInterface", "isBinaryData", "getBinaryTag", "generateUploadHash", "char", "processBinaryForUpload", "tag", "processedArray", "allUploads", "itemPath", "uploads", "processedObj", "uploadBinaryData", "sender", "onTypeStFn", "handlerFn", "idx", "import_messageHash", "import_jss", "connect", "connectTimeout", "totalRequestTimeout", "joinKey", "reservedKeys", "init_connectSocket", "__esmMin", "init_streaming", "wrapperFn", "a", "b", "body", "dirctCall", "rej", "promiseIsLive", "timeLetForReqToBeMade", "payload", "message", "messageHash", "replyPromise", "result", "next", "errMessage", "waitingOnOpen", "res", "waitingOnOpenThen", "waitingOnOpenCatch", "isBrowser", "clientPromise", "resolvedClient", "bufferedCalls", "bufferedReceivers", "connectionChangeHandlers", "currentConnectionState", "getClient", "connectSocket", "client", "state", "fn", "type", "handler", "method", "args", "resolve", "reject", "result", "err", "senderProxy", "target", "prop", "on", "onConnectionChange", "idx", "index_default"]
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "api-ape",
3
- "version": "3.0.0",
3
+ "version": "3.0.2",
4
4
  "description": "Remote Procedure Events (RPE) - A lightweight WebSocket framework for building real-time APIs. Call server functions from the browser like local methods with automatic reconnection, HTTP streaming fallback, and extended JSON encoding.",
5
5
  "main": "index.js",
6
6
  "browser": "./client/index.js",
@@ -69,4 +69,4 @@
69
69
  "esbuild": "^0.27.2",
70
70
  "jest": "^29.3.1"
71
71
  }
72
- }
72
+ }
package/server/README.md CHANGED
@@ -47,6 +47,19 @@ server/
47
47
  npm i api-ape
48
48
  ```
49
49
 
50
+ ### V3 Import Patterns
51
+
52
+ ```js
53
+ // CommonJS
54
+ const api = require('api-ape') // Client proxy (default)
55
+ const { ape } = require('api-ape') // Server initializer
56
+
57
+ // ESM
58
+ import api, { ape } from 'api-ape'
59
+ ```
60
+
61
+ ### Basic Server Setup
62
+
50
63
  ```js
51
64
  const { createServer } = require('http')
52
65
  const { ape } = require('api-ape')
@@ -64,12 +77,28 @@ ape(server, {
64
77
  server.listen(3000)
65
78
  ```
66
79
 
80
+ ### Dual-Purpose `ape` Function
81
+
82
+ The `ape` function intelligently detects how it's being used:
83
+
84
+ ```js
85
+ // Called with HTTP server → Server setup
86
+ ape(server, { where: 'api' })
87
+
88
+ // Called with data → API call to /ape endpoint
89
+ ape({ someData: 'value' }) // Calls /ape on connected server
90
+
91
+ // Or via the proxy
92
+ api.ape({ someData: 'value' }) // Same as above
93
+ ```
94
+
67
95
  ## Server-to-Server Connection
68
96
 
69
97
  Your server can connect to **another** api-ape server as a client. The API is 100% identical to browser usage:
70
98
 
71
99
  ```js
72
- import api, { ape } from 'api-ape'
100
+ const api = require('api-ape')
101
+ const { ape } = require('api-ape')
73
102
 
74
103
  // Start your own server
75
104
  ape(server, { where: 'api' })
@@ -5,15 +5,15 @@ Connect multiple api-ape server instances via a shared database for horizontal s
5
5
  ## Quick Start
6
6
 
7
7
  ```js
8
- import ape from 'api-ape/server';
9
- import { createClient } from 'redis';
8
+ const { ape } = require('api-ape')
9
+ const { createClient } = require('redis')
10
10
 
11
11
  // Connect to your database
12
- const redis = createClient();
13
- await redis.connect();
12
+ const redis = createClient()
13
+ await redis.connect()
14
14
 
15
15
  // Join the cluster — APE creates its own namespace
16
- ape.joinVia(redis);
16
+ ape.joinVia(redis)
17
17
  ```
18
18
 
19
19
  That's it. APE will:
package/server/client.js CHANGED
@@ -217,6 +217,12 @@ function onConnectionChange(handler) {
217
217
  */
218
218
  const handler = {
219
219
  get(target, prop) {
220
+ // First check if property exists on target (set via defineProperty)
221
+ // This allows named exports like 'ape' to be accessed directly
222
+ if (Reflect.has(target, prop)) {
223
+ return Reflect.get(target, prop)
224
+ }
225
+
220
226
  // Reserved properties - same as browser
221
227
  if (prop === 'on') return on
222
228
  if (prop === 'onConnectionChange') return onConnectionChange
@@ -228,10 +234,13 @@ const handler = {
228
234
  // Return a function that either calls directly or buffers
229
235
  const wrapperFn = function (a, b) {
230
236
  let path = joinKey + prop, body
231
- if (arguments.length === 2) {
237
+ // Two args: first is path segment (string), second is body
238
+ // One arg: it's always the body (matches browser client behavior)
239
+ if (arguments.length === 2 && typeof a === 'string') {
232
240
  path += a
233
241
  body = b
234
242
  } else {
243
+ // Single arg or non-string first arg: treat first arg as body
235
244
  body = a
236
245
  }
237
246
  return queueOrSend(path, body)
@@ -284,10 +293,10 @@ Object.defineProperty(api, 'close', {
284
293
  configurable: false
285
294
  })
286
295
 
287
- // Auto-connect if APE_SERVER is set
288
- if (serverUrl) {
289
- connect()
290
- }
296
+ // NOTE: We do NOT auto-connect on module load, even if APE_SERVER is set.
297
+ // Connection only happens when:
298
+ // 1. api.connect(url) is called explicitly
299
+ // 2. A method is called and APE_SERVER env is set (lazy connection)
291
300
 
292
301
  // Export the same interface as browser
293
302
  module.exports = api
@@ -297,3 +306,6 @@ module.exports.onConnectionChange = onConnectionChange
297
306
  module.exports.connect = connect
298
307
  module.exports.close = close
299
308
  module.exports.ConnectionState = ConnectionState
309
+ // Internal: expose queueOrSend for ape dual-purpose function
310
+ module.exports._queueOrSend = queueOrSend
311
+