| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  | import { ProtocolRequest, session } from 'electron/main'; | 
					
						
							|  |  |  | import { createReadStream } from 'fs'; | 
					
						
							|  |  |  | import { Readable } from 'stream'; | 
					
						
							|  |  |  | import { ReadableStream } from 'stream/web'; | 
					
						
							| 
									
										
										
										
											2016-01-11 18:40:23 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-12 19:27:15 +09:00
										 |  |  | // Global protocol APIs.
 | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  | const { registerSchemesAsPrivileged, getStandardSchemes, Protocol } = process._linkedBinding('electron_browser_protocol'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const ERR_FAILED = -2; | 
					
						
							|  |  |  | const ERR_UNEXPECTED = -9; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-12 11:42:49 +02:00
										 |  |  | const isBuiltInScheme = (scheme: string) => ['http', 'https', 'file'].includes(scheme); | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | function makeStreamFromPipe (pipe: any): ReadableStream { | 
					
						
							|  |  |  |   const buf = new Uint8Array(1024 * 1024 /* 1 MB */); | 
					
						
							|  |  |  |   return new ReadableStream({ | 
					
						
							|  |  |  |     async pull (controller) { | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         const rv = await pipe.read(buf); | 
					
						
							|  |  |  |         if (rv > 0) { | 
					
						
							| 
									
										
										
										
											2024-04-23 11:23:19 -04:00
										 |  |  |           controller.enqueue(buf.slice(0, rv)); | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |         } else { | 
					
						
							|  |  |  |           controller.close(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } catch (e) { | 
					
						
							|  |  |  |         controller.error(e); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  | function makeStreamFromFileInfo ({ | 
					
						
							|  |  |  |   filePath, | 
					
						
							|  |  |  |   offset = 0, | 
					
						
							|  |  |  |   length = -1 | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   filePath: string; | 
					
						
							|  |  |  |   offset?: number; | 
					
						
							|  |  |  |   length?: number; | 
					
						
							|  |  |  | }): ReadableStream { | 
					
						
							|  |  |  |   return Readable.toWeb(createReadStream(filePath, { | 
					
						
							|  |  |  |     start: offset, | 
					
						
							|  |  |  |     end: length >= 0 ? offset + length : undefined | 
					
						
							|  |  |  |   })); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  | function convertToRequestBody (uploadData: ProtocolRequest['uploadData']): RequestInit['body'] { | 
					
						
							|  |  |  |   if (!uploadData) return null; | 
					
						
							|  |  |  |   // Optimization: skip creating a stream if the request is just a single buffer.
 | 
					
						
							|  |  |  |   if (uploadData.length === 1 && (uploadData[0] as any).type === 'rawData') return uploadData[0].bytes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const chunks = [...uploadData] as any[]; // TODO: types are wrong
 | 
					
						
							|  |  |  |   let current: ReadableStreamDefaultReader | null = null; | 
					
						
							|  |  |  |   return new ReadableStream({ | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |     async pull (controller) { | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |       if (current) { | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |         const { done, value } = await current.read(); | 
					
						
							|  |  |  |         // (done => value === undefined) as per WHATWG spec
 | 
					
						
							|  |  |  |         if (done) { | 
					
						
							|  |  |  |           current = null; | 
					
						
							|  |  |  |           return this.pull!(controller); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |           controller.enqueue(value); | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |       } else { | 
					
						
							|  |  |  |         if (!chunks.length) { return controller.close(); } | 
					
						
							|  |  |  |         const chunk = chunks.shift()!; | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |         if (chunk.type === 'rawData') { | 
					
						
							|  |  |  |           controller.enqueue(chunk.bytes); | 
					
						
							|  |  |  |         } else if (chunk.type === 'file') { | 
					
						
							|  |  |  |           current = makeStreamFromFileInfo(chunk).getReader(); | 
					
						
							|  |  |  |           return this.pull!(controller); | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |         } else if (chunk.type === 'stream') { | 
					
						
							|  |  |  |           current = makeStreamFromPipe(chunk.body).getReader(); | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |           return this.pull!(controller); | 
					
						
							|  |  |  |         } else if (chunk.type === 'blob') { | 
					
						
							|  |  |  |           // Note that even though `getBlobData()` is a `Session` API, it doesn't
 | 
					
						
							|  |  |  |           // actually use the `Session` context. Its implementation solely relies
 | 
					
						
							|  |  |  |           // on global variables which allows us to implement this feature without
 | 
					
						
							|  |  |  |           // knowledge of the `Session` associated with the current request by
 | 
					
						
							|  |  |  |           // always pulling `Blob` data out of the default `Session`.
 | 
					
						
							|  |  |  |           controller.enqueue(await session.defaultSession.getBlobData(chunk.blobUUID)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           throw new Error(`Unknown upload data chunk type: ${chunk.type}`); | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }) as RequestInit['body']; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-07 09:29:04 +02:00
										 |  |  | function validateResponse (res: Response) { | 
					
						
							|  |  |  |   if (!res || typeof res !== 'object') return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (res.type === 'error') return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-27 22:57:33 +02:00
										 |  |  |   const exists = (key: string) => Object.hasOwn(res, key); | 
					
						
							| 
									
										
										
										
											2023-06-07 09:29:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (exists('status') && typeof res.status !== 'number') return false; | 
					
						
							|  |  |  |   if (exists('statusText') && typeof res.statusText !== 'string') return false; | 
					
						
							|  |  |  |   if (exists('headers') && typeof res.headers !== 'object') return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (exists('body')) { | 
					
						
							|  |  |  |     if (typeof res.body !== 'object') return false; | 
					
						
							|  |  |  |     if (res.body !== null && !(res.body instanceof ReadableStream)) return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  | Protocol.prototype.handle = function (this: Electron.Protocol, scheme: string, handler: (req: Request) => Response | Promise<Response>) { | 
					
						
							|  |  |  |   const register = isBuiltInScheme(scheme) ? this.interceptProtocol : this.registerProtocol; | 
					
						
							|  |  |  |   const success = register.call(this, scheme, async (preq: ProtocolRequest, cb: any) => { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const body = convertToRequestBody(preq.uploadData); | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |       const headers = new Headers(preq.headers); | 
					
						
							|  |  |  |       if (headers.get('origin') === 'null') { | 
					
						
							|  |  |  |         headers.delete('origin'); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |       const req = new Request(preq.url, { | 
					
						
							| 
									
										
										
										
											2024-02-16 20:29:29 +01:00
										 |  |  |         headers, | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |         method: preq.method, | 
					
						
							|  |  |  |         referrer: preq.referrer, | 
					
						
							|  |  |  |         body, | 
					
						
							|  |  |  |         duplex: body instanceof ReadableStream ? 'half' : undefined | 
					
						
							|  |  |  |       } as any); | 
					
						
							|  |  |  |       const res = await handler(req); | 
					
						
							| 
									
										
										
										
											2023-06-07 09:29:04 +02:00
										 |  |  |       if (!validateResponse(res)) { | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |         return cb({ error: ERR_UNEXPECTED }); | 
					
						
							| 
									
										
										
										
											2023-06-07 09:29:04 +02:00
										 |  |  |       } else if (res.type === 'error') { | 
					
						
							|  |  |  |         cb({ error: ERR_FAILED }); | 
					
						
							|  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |         cb({ | 
					
						
							|  |  |  |           data: res.body ? Readable.fromWeb(res.body as ReadableStream<ArrayBufferView>) : null, | 
					
						
							| 
									
										
										
										
											2023-06-07 09:29:04 +02:00
										 |  |  |           headers: res.headers ? Object.fromEntries(res.headers) : {}, | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |           statusCode: res.status, | 
					
						
							|  |  |  |           statusText: res.statusText, | 
					
						
							|  |  |  |           mimeType: (res as any).__original_resp?._responseHead?.mimeType | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } catch (e) { | 
					
						
							|  |  |  |       console.error(e); | 
					
						
							|  |  |  |       cb({ error: ERR_UNEXPECTED }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   if (!success) throw new Error(`Failed to register protocol: ${scheme}`); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Protocol.prototype.unhandle = function (this: Electron.Protocol, scheme: string) { | 
					
						
							|  |  |  |   const unregister = isBuiltInScheme(scheme) ? this.uninterceptProtocol : this.unregisterProtocol; | 
					
						
							|  |  |  |   if (!unregister.call(this, scheme)) { throw new Error(`Failed to unhandle protocol: ${scheme}`); } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Protocol.prototype.isProtocolHandled = function (this: Electron.Protocol, scheme: string) { | 
					
						
							|  |  |  |   const isRegistered = isBuiltInScheme(scheme) ? this.isProtocolIntercepted : this.isProtocolRegistered; | 
					
						
							|  |  |  |   return isRegistered.call(this, scheme); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const protocol = { | 
					
						
							|  |  |  |   registerSchemesAsPrivileged, | 
					
						
							|  |  |  |   getStandardSchemes, | 
					
						
							|  |  |  |   registerStringProtocol: (...args) => session.defaultSession.protocol.registerStringProtocol(...args), | 
					
						
							|  |  |  |   registerBufferProtocol: (...args) => session.defaultSession.protocol.registerBufferProtocol(...args), | 
					
						
							|  |  |  |   registerStreamProtocol: (...args) => session.defaultSession.protocol.registerStreamProtocol(...args), | 
					
						
							|  |  |  |   registerFileProtocol: (...args) => session.defaultSession.protocol.registerFileProtocol(...args), | 
					
						
							|  |  |  |   registerHttpProtocol: (...args) => session.defaultSession.protocol.registerHttpProtocol(...args), | 
					
						
							|  |  |  |   registerProtocol: (...args) => session.defaultSession.protocol.registerProtocol(...args), | 
					
						
							|  |  |  |   unregisterProtocol: (...args) => session.defaultSession.protocol.unregisterProtocol(...args), | 
					
						
							|  |  |  |   isProtocolRegistered: (...args) => session.defaultSession.protocol.isProtocolRegistered(...args), | 
					
						
							|  |  |  |   interceptStringProtocol: (...args) => session.defaultSession.protocol.interceptStringProtocol(...args), | 
					
						
							|  |  |  |   interceptBufferProtocol: (...args) => session.defaultSession.protocol.interceptBufferProtocol(...args), | 
					
						
							|  |  |  |   interceptStreamProtocol: (...args) => session.defaultSession.protocol.interceptStreamProtocol(...args), | 
					
						
							|  |  |  |   interceptFileProtocol: (...args) => session.defaultSession.protocol.interceptFileProtocol(...args), | 
					
						
							|  |  |  |   interceptHttpProtocol: (...args) => session.defaultSession.protocol.interceptHttpProtocol(...args), | 
					
						
							|  |  |  |   interceptProtocol: (...args) => session.defaultSession.protocol.interceptProtocol(...args), | 
					
						
							|  |  |  |   uninterceptProtocol: (...args) => session.defaultSession.protocol.uninterceptProtocol(...args), | 
					
						
							|  |  |  |   isProtocolIntercepted: (...args) => session.defaultSession.protocol.isProtocolIntercepted(...args), | 
					
						
							|  |  |  |   handle: (...args) => session.defaultSession.protocol.handle(...args), | 
					
						
							|  |  |  |   unhandle: (...args) => session.defaultSession.protocol.unhandle(...args), | 
					
						
							|  |  |  |   isProtocolHandled: (...args) => session.defaultSession.protocol.isProtocolHandled(...args) | 
					
						
							|  |  |  | } as typeof Electron.protocol; | 
					
						
							| 
									
										
										
										
											2019-02-12 06:22:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | export default protocol; |