mirror of
				https://github.com/Yidadaa/ChatGPT-Next-Web.git
				synced 2025-10-25 18:20:36 +08:00 
			
		
		
		
	using stream_fetch in App
This commit is contained in:
		| @@ -12,9 +12,55 @@ type ResponseEvent = { | |||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | type StreamResponse = { | ||||||
|  |   request_id: number; | ||||||
|  |   status: number; | ||||||
|  |   status_text: string; | ||||||
|  |   headers: Record<string, string>; | ||||||
|  | }; | ||||||
|  |  | ||||||
| export function fetch(url: string, options?: RequestInit): Promise<any> { | export function fetch(url: string, options?: RequestInit): Promise<any> { | ||||||
|   if (window.__TAURI__) { |   if (window.__TAURI__) { | ||||||
|     const { signal, method = "GET", headers = {}, body = [] } = options || {}; |     const { signal, method = "GET", headers = {}, body = [] } = options || {}; | ||||||
|  |     let unlisten: Function | undefined; | ||||||
|  |     let request_id = 0; | ||||||
|  |     const ts = new TransformStream(); | ||||||
|  |     const writer = ts.writable.getWriter(); | ||||||
|  |  | ||||||
|  |     const close = () => { | ||||||
|  |       unlisten && unlisten(); | ||||||
|  |       writer.ready.then(() => { | ||||||
|  |         try { | ||||||
|  |           writer.releaseLock(); | ||||||
|  |         } catch (e) { | ||||||
|  |           console.error(e); | ||||||
|  |         } | ||||||
|  |         ts.writable.close(); | ||||||
|  |       }); | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (signal) { | ||||||
|  |       signal.addEventListener("abort", () => close()); | ||||||
|  |     } | ||||||
|  |     // @ts-ignore 2. listen response multi times, and write to Response.body | ||||||
|  |     window.__TAURI__.event | ||||||
|  |       .listen("stream-response", (e: ResponseEvent) => { | ||||||
|  |         const { request_id: rid, chunk, status } = e?.payload || {}; | ||||||
|  |         if (request_id != rid) { | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |         if (chunk) { | ||||||
|  |           writer && | ||||||
|  |             writer.ready.then(() => { | ||||||
|  |               writer && writer.write(new Uint8Array(chunk)); | ||||||
|  |             }); | ||||||
|  |         } else if (status === 0) { | ||||||
|  |           // end of body | ||||||
|  |           close(); | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       .then((u: Function) => (unlisten = u)); | ||||||
|  |  | ||||||
|     return window.__TAURI__ |     return window.__TAURI__ | ||||||
|       .invoke("stream_fetch", { |       .invoke("stream_fetch", { | ||||||
|         method, |         method, | ||||||
| @@ -26,61 +72,11 @@ export function fetch(url: string, options?: RequestInit): Promise<any> { | |||||||
|             ? Array.from(new TextEncoder().encode(body)) |             ? Array.from(new TextEncoder().encode(body)) | ||||||
|             : [], |             : [], | ||||||
|       }) |       }) | ||||||
|       .then( |       .then((res: StreamResponse) => { | ||||||
|         (res: { |         request_id = res.request_id; | ||||||
|           request_id: number; |         const { status, status_text: statusText, headers } = res; | ||||||
|           status: number; |         return new Response(ts.readable, { status, statusText, headers }); | ||||||
|           status_text: string; |       }) | ||||||
|           headers: Record<string, string>; |  | ||||||
|         }) => { |  | ||||||
|           const { request_id, status, status_text: statusText, headers } = res; |  | ||||||
|           console.log("send request_id", request_id, status, statusText); |  | ||||||
|           let unlisten: Function | undefined; |  | ||||||
|           const ts = new TransformStream(); |  | ||||||
|           const writer = ts.writable.getWriter(); |  | ||||||
|  |  | ||||||
|           const close = () => { |  | ||||||
|             unlisten && unlisten(); |  | ||||||
|             writer.ready.then(() => { |  | ||||||
|               try { |  | ||||||
|                 writer.releaseLock(); |  | ||||||
|               } catch (e) { |  | ||||||
|                 console.error(e); |  | ||||||
|               } |  | ||||||
|               ts.writable.close(); |  | ||||||
|             }); |  | ||||||
|           }; |  | ||||||
|  |  | ||||||
|           const response = new Response(ts.readable, { |  | ||||||
|             status, |  | ||||||
|             statusText, |  | ||||||
|             headers, |  | ||||||
|           }); |  | ||||||
|           if (signal) { |  | ||||||
|             signal.addEventListener("abort", () => close()); |  | ||||||
|           } |  | ||||||
|           // @ts-ignore 2. listen response multi times, and write to Response.body |  | ||||||
|           window.__TAURI__.event |  | ||||||
|             .listen("stream-response", (e: ResponseEvent) => { |  | ||||||
|               const { id, payload } = e; |  | ||||||
|               const { request_id: rid, chunk, status } = payload; |  | ||||||
|               if (request_id != rid) { |  | ||||||
|                 return; |  | ||||||
|               } |  | ||||||
|               if (chunk) { |  | ||||||
|                 writer && |  | ||||||
|                   writer.ready.then(() => { |  | ||||||
|                     writer && writer.write(new Uint8Array(chunk)); |  | ||||||
|                   }); |  | ||||||
|               } else if (status === 0) { |  | ||||||
|                 // end of body |  | ||||||
|                 close(); |  | ||||||
|               } |  | ||||||
|             }) |  | ||||||
|             .then((u: Function) => (unlisten = u)); |  | ||||||
|           return response; |  | ||||||
|         }, |  | ||||||
|       ) |  | ||||||
|       .catch((e) => { |       .catch((e) => { | ||||||
|         console.error("stream error", e); |         console.error("stream error", e); | ||||||
|         throw e; |         throw e; | ||||||
| @@ -88,7 +84,3 @@ export function fetch(url: string, options?: RequestInit): Promise<any> { | |||||||
|   } |   } | ||||||
|   return window.fetch(url, options); |   return window.fetch(url, options); | ||||||
| } | } | ||||||
|  |  | ||||||
| if (undefined !== window) { |  | ||||||
|   window.tauriFetch = fetch; |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -66,9 +66,7 @@ pub async fn stream_fetch( | |||||||
|   let res = response_future.await; |   let res = response_future.await; | ||||||
|   let response = match res { |   let response = match res { | ||||||
|     Ok(res) => { |     Ok(res) => { | ||||||
|       println!("Error: {:?}", res); |  | ||||||
|       // get response and emit to client |       // get response and emit to client | ||||||
|       // .register_uri_scheme_protocol("stream", move |app_handle, request| { |  | ||||||
|       let mut headers = HashMap::new(); |       let mut headers = HashMap::new(); | ||||||
|       for (name, value) in res.headers() { |       for (name, value) in res.headers() { | ||||||
|         headers.insert( |         headers.insert( | ||||||
| @@ -84,7 +82,7 @@ pub async fn stream_fetch( | |||||||
|         while let Some(chunk) = stream.next().await { |         while let Some(chunk) = stream.next().await { | ||||||
|           match chunk { |           match chunk { | ||||||
|             Ok(bytes) => { |             Ok(bytes) => { | ||||||
|               println!("chunk: {:?}", bytes); |               // println!("chunk: {:?}", bytes); | ||||||
|               window.emit(event_name, ChunkPayload{ request_id, chunk: bytes }).unwrap(); |               window.emit(event_name, ChunkPayload{ request_id, chunk: bytes }).unwrap(); | ||||||
|             } |             } | ||||||
|             Err(err) => { |             Err(err) => { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user