MicrosoftDesigner.py 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. from __future__ import annotations
  2. import uuid
  3. import aiohttp
  4. import random
  5. import asyncio
  6. import json
  7. from ...providers.response import ImageResponse
  8. from ...errors import MissingRequirementsError, NoValidHarFileError
  9. from ...typing import AsyncResult, Messages
  10. from ...requests.raise_for_status import raise_for_status
  11. from ...requests.aiohttp import get_connector
  12. from ...requests import get_nodriver
  13. from ..Copilot import get_headers, get_har_files
  14. from ..base_provider import AsyncGeneratorProvider, ProviderModelMixin
  15. from ..helper import get_random_hex, format_image_prompt
  16. from ... import debug
  17. class MicrosoftDesigner(AsyncGeneratorProvider, ProviderModelMixin):
  18. label = "Microsoft Designer"
  19. url = "https://designer.microsoft.com"
  20. working = True
  21. use_nodriver = True
  22. needs_auth = True
  23. default_image_model = "dall-e-3"
  24. image_models = [default_image_model, "1024x1024", "1024x1792", "1792x1024"]
  25. models = image_models
  26. @classmethod
  27. async def create_async_generator(
  28. cls,
  29. model: str,
  30. messages: Messages,
  31. prompt: str = None,
  32. proxy: str = None,
  33. **kwargs
  34. ) -> AsyncResult:
  35. image_size = "1024x1024"
  36. if model != cls.default_image_model and model in cls.image_models:
  37. image_size = model
  38. yield await cls.generate(format_image_prompt(messages, prompt), image_size, proxy)
  39. @classmethod
  40. async def generate(cls, prompt: str, image_size: str, proxy: str = None) -> ImageResponse:
  41. try:
  42. access_token, user_agent = readHAR("https://designerapp.officeapps.live.com")
  43. except NoValidHarFileError as h:
  44. debug.log(f"{cls.__name__}: {h}")
  45. try:
  46. access_token, user_agent = await get_access_token_and_user_agent(cls.url, proxy)
  47. except MissingRequirementsError:
  48. raise h
  49. images = await create_images(prompt, access_token, user_agent, image_size, proxy)
  50. return ImageResponse(images, prompt)
  51. async def create_images(prompt: str, access_token: str, user_agent: str, image_size: str, proxy: str = None, seed: int = None):
  52. url = 'https://designerapp.officeapps.live.com/designerapp/DallE.ashx?action=GetDallEImagesCogSci'
  53. if seed is None:
  54. seed = random.randint(0, 10000)
  55. headers = {
  56. "User-Agent": user_agent,
  57. "Accept": "application/json, text/plain, */*",
  58. "Accept-Language": "en-US",
  59. 'Authorization': f'Bearer {access_token}',
  60. "AudienceGroup": "Production",
  61. "Caller": "DesignerApp",
  62. "ClientId": "b5c2664a-7e9b-4a7a-8c9a-cd2c52dcf621",
  63. "SessionId": str(uuid.uuid4()),
  64. "UserId": get_random_hex(16),
  65. "ContainerId": "1e2843a7-2a98-4a6c-93f2-42002de5c478",
  66. "FileToken": "9f1a4cb7-37e7-4c90-b44d-cb61cfda4bb8",
  67. "x-upload-to-storage-das": "1",
  68. "traceparent": "",
  69. "X-DC-Hint": "FranceCentral",
  70. "Platform": "Web",
  71. "HostApp": "DesignerApp",
  72. "ReleaseChannel": "",
  73. "IsSignedInUser": "true",
  74. "Locale": "de-DE",
  75. "UserType": "MSA",
  76. "x-req-start": "2615401",
  77. "ClientBuild": "1.0.20241120.9",
  78. "ClientName": "DesignerApp",
  79. "Sec-Fetch-Dest": "empty",
  80. "Sec-Fetch-Mode": "cors",
  81. "Sec-Fetch-Site": "cross-site",
  82. "Pragma": "no-cache",
  83. "Cache-Control": "no-cache",
  84. "Referer": "https://designer.microsoft.com/"
  85. }
  86. form_data = aiohttp.FormData()
  87. form_data.add_field('dalle-caption', prompt)
  88. form_data.add_field('dalle-scenario-name', 'TextToImage')
  89. form_data.add_field('dalle-batch-size', '4')
  90. form_data.add_field('dalle-image-response-format', 'UrlWithBase64Thumbnail')
  91. form_data.add_field('dalle-seed', seed)
  92. form_data.add_field('ClientFlights', 'EnableBICForDALLEFlight')
  93. form_data.add_field('dalle-hear-back-in-ms', 1000)
  94. form_data.add_field('dalle-include-b64-thumbnails', 'true')
  95. form_data.add_field('dalle-aspect-ratio-scaling-factor-b64-thumbnails', 0.3)
  96. form_data.add_field('dalle-image-size', image_size)
  97. async with aiohttp.ClientSession(connector=get_connector(proxy=proxy)) as session:
  98. async with session.post(url, headers=headers, data=form_data) as response:
  99. await raise_for_status(response)
  100. response_data = await response.json()
  101. form_data.add_field('dalle-boost-count', response_data.get('dalle-boost-count', 0))
  102. polling_meta_data = response_data.get('polling_response', {}).get('polling_meta_data', {})
  103. form_data.add_field('dalle-poll-url', polling_meta_data.get('poll_url', ''))
  104. while True:
  105. await asyncio.sleep(polling_meta_data.get('poll_interval', 1000) / 1000)
  106. async with session.post(url, headers=headers, data=form_data) as response:
  107. await raise_for_status(response)
  108. response_data = await response.json()
  109. images = [image["ImageUrl"] for image in response_data.get('image_urls_thumbnail', [])]
  110. if images:
  111. return images
  112. def readHAR(url: str) -> tuple[str, str]:
  113. api_key = None
  114. user_agent = None
  115. for path in get_har_files():
  116. with open(path, 'rb') as file:
  117. try:
  118. harFile = json.loads(file.read())
  119. except json.JSONDecodeError:
  120. # Error: not a HAR file!
  121. continue
  122. for v in harFile['log']['entries']:
  123. if v['request']['url'].startswith(url):
  124. v_headers = get_headers(v)
  125. if "authorization" in v_headers:
  126. api_key = v_headers["authorization"].split(maxsplit=1).pop()
  127. if "user-agent" in v_headers:
  128. user_agent = v_headers["user-agent"]
  129. if api_key is None:
  130. raise NoValidHarFileError("No access token found in .har files")
  131. return api_key, user_agent
  132. async def get_access_token_and_user_agent(url: str, proxy: str = None):
  133. browser, stop_browser = await get_nodriver(proxy=proxy, user_data_dir="designer")
  134. try:
  135. page = await browser.get(url)
  136. user_agent = await page.evaluate("navigator.userAgent")
  137. access_token = None
  138. while access_token is None:
  139. access_token = await page.evaluate("""
  140. (() => {
  141. for (var i = 0; i < localStorage.length; i++) {
  142. try {
  143. item = JSON.parse(localStorage.getItem(localStorage.key(i)));
  144. if (item.credentialType == "AccessToken"
  145. && item.expiresOn > Math.floor(Date.now() / 1000)
  146. && item.target.includes("designerappservice")) {
  147. return item.secret;
  148. }
  149. } catch(e) {}
  150. }
  151. })()
  152. """)
  153. if access_token is None:
  154. await asyncio.sleep(1)
  155. await page.close()
  156. return access_token, user_agent
  157. finally:
  158. stop_browser()