Antonio Cheong commited on
Commit
bcd6bf2
·
1 Parent(s): 7a42723

Streaming support

Browse files
Files changed (3) hide show
  1. README.md +19 -10
  2. setup.py +1 -1
  3. src/EdgeGPT.py +63 -6
README.md CHANGED
@@ -22,10 +22,6 @@ Completely optional, but recommended if you have beta access to Bing Chat
22
  - Go to the Application tab → Storage → Cookies
23
  - Find the cookie named "_U"
24
  - Copy the value of the cookie
25
- - Method 1
26
- - `export BING_U="<COOKIE_VALUE>"`
27
- - Method 2
28
- - Use it as command line argument later
29
 
30
  </details>
31
 
@@ -33,11 +29,25 @@ Completely optional, but recommended if you have beta access to Bing Chat
33
  - `python3 -m pip install EdgeGPT`
34
 
35
  ## Demo usage
36
- ### With your own token (Optional)
37
- - If `BING_U` in environment variables: `python3 -m EdgeGPT`
38
- - Else: `python3 -m EdgeGPT "<COOKIE_VALUE>"`
39
- ### Without a cookie
40
- - `python3 -m EdgeGPT`
 
 
 
 
 
 
 
 
 
 
 
 
 
 
41
 
42
  ## Developer
43
  Use Async for the best experience
@@ -119,7 +129,6 @@ if __name__ == "__main__":
119
  ```
120
 
121
  ## Work in progress. Help me
122
- - Response streaming
123
  - Error handling
124
 
125
  # Credits
 
22
  - Go to the Application tab → Storage → Cookies
23
  - Find the cookie named "_U"
24
  - Copy the value of the cookie
 
 
 
 
25
 
26
  </details>
27
 
 
29
  - `python3 -m pip install EdgeGPT`
30
 
31
  ## Demo usage
32
+ ```s
33
+ $ python3 -m EdgeGPT -h
34
+
35
+ EdgeGPT - A demo of reverse engineering the Bing GPT chatbot
36
+ Repo: github.com/acheong08/EdgeGPT
37
+ By: Antonio Cheong
38
+
39
+ !help for help
40
+
41
+ Type !exit to exit
42
+ Enter twice to send message
43
+
44
+ usage: EdgeGPT.py [-h] [--stream] [--bing-cookie BING_COOKIE]
45
+
46
+ options:
47
+ -h, --help show this help message and exit
48
+ --stream
49
+ --bing-cookie BING_COOKIE (Optional)
50
+ ```
51
 
52
  ## Developer
53
  Use Async for the best experience
 
129
  ```
130
 
131
  ## Work in progress. Help me
 
132
  - Error handling
133
 
134
  # Credits
setup.py CHANGED
@@ -3,7 +3,7 @@ from setuptools import setup
3
 
4
  setup(
5
  name="EdgeGPT",
6
- version="0.0.11",
7
  license="GNU General Public License v2.0",
8
  author="Antonio Cheong",
9
  author_email="[email protected]",
 
3
 
4
  setup(
5
  name="EdgeGPT",
6
+ version="0.0.20",
7
  license="GNU General Public License v2.0",
8
  author="Antonio Cheong",
9
  author_email="[email protected]",
src/EdgeGPT.py CHANGED
@@ -6,6 +6,7 @@ import json
6
  import os
7
  import sys
8
  import uuid
 
9
 
10
  import requests
11
  import websockets.client as websockets
@@ -113,12 +114,12 @@ class Conversation:
113
  "x-ms-useragent": "azsdk-js-api-client-factory/1.0.0-beta.1 core-rest-pipeline/1.10.0 OS/Linuxx86_64",
114
  }
115
  # Create cookies
116
- if os.environ.get("BING_U") is None and len(sys.argv) < 2:
117
  url = "https://bing.kpham.workers.dev/"
118
  cookies = {}
119
  else:
120
  cookies = {
121
- "_U": os.environ.get("BING_U") or sys.argv[1],
122
  }
123
  url = "https://www.bing.com/turing/conversation/create"
124
  # Send GET request
@@ -182,6 +183,40 @@ class ChatHub:
182
  if response.get("type") == 2:
183
  return response
184
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
185
  async def __initial_handshake(self):
186
  await self.wss.send(append_identifier({"protocol": "json", "version": 1}))
187
  await self.wss.recv()
@@ -208,6 +243,13 @@ class Chatbot:
208
  Ask a question to the bot
209
  """
210
  return await self.chat_hub.ask(prompt=prompt)
 
 
 
 
 
 
 
211
 
212
  async def close(self):
213
  """
@@ -270,11 +312,20 @@ async def main():
270
  await bot.reset()
271
  continue
272
  print("Bot:")
273
- print(
274
- (await bot.ask(prompt=prompt))["item"]["messages"][1]["adaptiveCards"][0][
275
- "body"
276
- ][0]["text"],
 
277
  )
 
 
 
 
 
 
 
 
278
  await bot.close()
279
 
280
 
@@ -291,4 +342,10 @@ if __name__ == "__main__":
291
  Enter twice to send message
292
  """,
293
  )
 
 
 
 
 
 
294
  asyncio.run(main())
 
6
  import os
7
  import sys
8
  import uuid
9
+ import argparse
10
 
11
  import requests
12
  import websockets.client as websockets
 
114
  "x-ms-useragent": "azsdk-js-api-client-factory/1.0.0-beta.1 core-rest-pipeline/1.10.0 OS/Linuxx86_64",
115
  }
116
  # Create cookies
117
+ if os.environ.get("BING_U") is None:
118
  url = "https://bing.kpham.workers.dev/"
119
  cookies = {}
120
  else:
121
  cookies = {
122
+ "_U": os.environ.get("BING_U"),
123
  }
124
  url = "https://www.bing.com/turing/conversation/create"
125
  # Send GET request
 
183
  if response.get("type") == 2:
184
  return response
185
 
186
+ async def ask_stream(self, prompt: str) -> str:
187
+ """
188
+ Ask a question to the bot
189
+ """
190
+ # Check if websocket is closed
191
+ if self.wss:
192
+ if self.wss.closed:
193
+ self.wss = await websockets.connect(
194
+ "wss://sydney.bing.com/sydney/ChatHub"
195
+ )
196
+ await self.__initial_handshake()
197
+ else:
198
+ self.wss = await websockets.connect("wss://sydney.bing.com/sydney/ChatHub")
199
+ await self.__initial_handshake()
200
+ # Construct a ChatHub request
201
+ self.request.update(prompt=prompt)
202
+ # Send request
203
+ await self.wss.send(append_identifier(self.request.struct))
204
+ stream_cache = ""
205
+ while True:
206
+ objects = str(await self.wss.recv()).split("")
207
+ for obj in objects:
208
+ if obj is None or obj == "":
209
+ continue
210
+ response = json.loads(obj)
211
+ if response.get("type") == 1:
212
+ cache_len = len(stream_cache)
213
+ adaptive_card = response["arguments"][0]["messages"][0]["adaptiveCards"][0]["body"][0]["text"]
214
+ stream_cache = adaptive_card
215
+ adaptive_card = adaptive_card[cache_len:]
216
+ yield adaptive_card
217
+ elif response.get("type") == 2:
218
+ return
219
+
220
  async def __initial_handshake(self):
221
  await self.wss.send(append_identifier({"protocol": "json", "version": 1}))
222
  await self.wss.recv()
 
243
  Ask a question to the bot
244
  """
245
  return await self.chat_hub.ask(prompt=prompt)
246
+
247
+ async def ask_stream(self, prompt: str) -> str:
248
+ """
249
+ Ask a question to the bot
250
+ """
251
+ async for response in self.chat_hub.ask_stream(prompt=prompt):
252
+ yield response
253
 
254
  async def close(self):
255
  """
 
312
  await bot.reset()
313
  continue
314
  print("Bot:")
315
+ if not args.stream:
316
+ print(
317
+ (await bot.ask(prompt=prompt))["item"]["messages"][1]["adaptiveCards"][0][
318
+ "body"
319
+ ][0]["text"],
320
  )
321
+ else:
322
+ async for response in bot.ask_stream(prompt=prompt):
323
+ print(
324
+ response, end=""
325
+ )
326
+ sys.stdout.flush()
327
+ print()
328
+ sys.stdout.flush()
329
  await bot.close()
330
 
331
 
 
342
  Enter twice to send message
343
  """,
344
  )
345
+ parser = argparse.ArgumentParser()
346
+ parser.add_argument("--stream", action="store_true")
347
+ parser.add_argument("--bing-cookie", type=str, default="")
348
+ args = parser.parse_args()
349
+ if args.bing_cookie:
350
+ os.environ["BING_U"] = args.bing_cookie
351
  asyncio.run(main())