fetch_dependencies.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. #!/usr/bin/env python3
  2. import re
  3. from pathlib import Path
  4. import requests
  5. from bs4 import BeautifulSoup
  6. import nicegui.elements.chart as highcharts
  7. from nicegui import ui
  8. USER_AGENT = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36'
  9. PATH = Path('/tmp/nicegui_dependencies')
  10. PATH.mkdir(exist_ok=True)
  11. def url_to_filename(url: str) -> str:
  12. return re.sub(r'[^a-zA-Z0-9]', '_', url)
  13. def request_buffered_str(url: str) -> str:
  14. filepath = PATH / url_to_filename(url)
  15. if filepath.exists():
  16. return filepath.read_text()
  17. response = requests.get(url, headers={'User-Agent': USER_AGENT})
  18. filepath.write_text(response.text)
  19. return response.text
  20. def request_buffered(url: str) -> bytes:
  21. filepath = PATH / url_to_filename(url)
  22. if filepath.exists():
  23. return filepath.read_bytes()
  24. response = requests.get(url, headers={'User-Agent': USER_AGENT})
  25. filepath.write_bytes(response.content)
  26. return response.content
  27. # Google fonts
  28. url = 'https://fonts.googleapis.com/css2?family=Material+Icons&family=Roboto:wght@100;300;400;500;700;900'
  29. css = request_buffered_str(url)
  30. for font_url in re.findall(r'url\((.*?)\)', css):
  31. font = request_buffered(font_url)
  32. (Path('nicegui/static/fonts') / font_url.split('/')[-1]).write_bytes(font)
  33. css = css.replace('https://fonts.gstatic.com/s/materialicons/v140', 'fonts')
  34. css = css.replace('https://fonts.gstatic.com/s/roboto/v30', 'fonts')
  35. css = css.replace(' U+2122', '\n U+2122')
  36. css = css.replace("'", '"')
  37. Path('nicegui/static/fonts.css').write_text(css)
  38. # quasar.js
  39. url = 'https://cdn.jsdelivr.net/npm/quasar/dist/quasar.umd.prod.js'
  40. js = request_buffered_str(url)
  41. Path('nicegui/static/quasar.umd.prod.js').write_text(js)
  42. version = re.search(r'Quasar Framework v(\d+\.\d+\.\d+)', js).group(1)
  43. url = 'https://cdn.jsdelivr.net/npm/quasar/dist/quasar.prod.css'
  44. css = request_buffered_str(url)
  45. Path('nicegui/static/quasar.prod.css').write_text(css)
  46. print('Quasar:', version)
  47. # vue.js
  48. url = 'https://unpkg.com/vue@3/anything'
  49. info = request_buffered_str(url)
  50. version = re.search(r'Cannot find "/anything" in vue@(\d+\.\d+\.\d+)', info).group(1)
  51. url = 'https://unpkg.com/vue@3/dist/vue.global.prod.js'
  52. js = request_buffered_str(url)
  53. Path('nicegui/static/vue.global.prod.js').write_text(js)
  54. print('Vue:', version)
  55. # socket.io.js
  56. url = 'https://cdn.jsdelivr.net/npm/socket.io-client/dist/socket.io.min.js'
  57. js = request_buffered_str(url)
  58. Path('nicegui/static/socket.io.min.js').write_text(js)
  59. version = re.search(r'Socket.IO v(\d+\.\d+\.\d+)', js).group(1)
  60. print('Socket.io:', version)
  61. # tailwind.js
  62. url = 'https://cdn.tailwindcss.com/'
  63. js = request_buffered_str(url)
  64. Path('nicegui/static/tailwindcss.min.js').write_text(js)
  65. version = re.search(r'{name:"tailwindcss",version:"(\d+\.\d+\.\d+)"', js).group(1)
  66. print('Tailwind CSS:', version)
  67. # tween.js
  68. url = 'https://cdnjs.com/libraries/tween.js'
  69. html = request_buffered_str(url)
  70. soup = BeautifulSoup(html, 'html.parser')
  71. version = soup.find('span', class_='vs__selected').text.strip()
  72. url = f'https://cdnjs.cloudflare.com/ajax/libs/tween.js/{version}/tween.umd.min.js'
  73. js = request_buffered_str(url)
  74. Path('nicegui/elements/lib/tween.umd.min.js').write_text(js)
  75. print('Tween.js:', version)
  76. # plotly.js
  77. url = 'https://plotly.com/javascript/'
  78. html = request_buffered_str(url)
  79. soup = BeautifulSoup(html, 'html.parser')
  80. version = soup.find('a', class_='plotly_js').text.strip().removeprefix('Javascript (v').removesuffix(')')
  81. url = f'https://cdn.plot.ly/plotly-{version}.min.js'
  82. js = request_buffered_str(url)
  83. Path('nicegui/elements/lib/plotly.min.js').write_text(js)
  84. print('Plotly.js:', version)
  85. # ag-grid.js
  86. url = 'https://cdn.jsdelivr.net/npm/ag-grid-community/dist/ag-grid-community.min.js'
  87. js = request_buffered_str(url)
  88. Path('nicegui/elements/lib/ag-grid-community.min.js').write_text(js)
  89. version = re.search(r'@version v(\d+\.\d+\.\d+)', js).group(1)
  90. print('AG Grid:', version)
  91. # nipplejs.js
  92. url = 'https://www.npmjs.com/package/nipplejs'
  93. html = request_buffered_str(url)
  94. soup = BeautifulSoup(html, 'html.parser')
  95. version = soup.find('h3', string='Version').find_next_sibling('div').text.strip()
  96. url = f'https://cdn.jsdelivr.net/npm/nipplejs@{version}/dist/nipplejs.min.js'
  97. js = request_buffered_str(url)
  98. Path('nicegui/elements/lib/nipplejs.min.js').write_text(js)
  99. print('NippleJS:', version)
  100. # mermaid.min.js
  101. url = 'https://cdn.jsdelivr.net/npm/mermaid@9/dist/'
  102. html = request_buffered_str(url)
  103. soup = BeautifulSoup(html, 'html.parser')
  104. # find a with href starting with /npm/mermaid@
  105. version = soup.find('a', href=re.compile(r'^/npm/mermaid@')).text.strip().removeprefix('mermaid@')
  106. url = f'https://cdn.jsdelivr.net/npm/mermaid@{version}/dist/mermaid.min.js'
  107. js = request_buffered_str(url)
  108. Path('nicegui/elements/lib/mermaid.min.js').write_text(js)
  109. print('Mermaid:', version)
  110. # TODO: upgrade to Mermaid 10.0.x? (ESM only and potentially breaking changes)
  111. # highcharts.js
  112. for dependency in highcharts.dependencies:
  113. name = dependency.split('/')[-1]
  114. url = f'https://code.highcharts.com/{name}'
  115. js = request_buffered_str(url)
  116. Path(f'nicegui/elements/lib/{name}').write_text(js)
  117. v = re.search(r'Highcharts JS v(\d+\.\d+\.\d+)', js).group(1)
  118. if name == 'highcharts.js':
  119. version = v
  120. print('Highcharts:', version)
  121. else:
  122. assert version == v
  123. for dependency in highcharts.optional_dependencies:
  124. name = dependency.split('/')[-1]
  125. url = f'https://code.highcharts.com/modules/{name}'
  126. js = request_buffered_str(url)
  127. Path(f'nicegui/elements/lib/highcharts_modules/{name}').write_text(js)
  128. v = re.search(r'JS v(\d+\.\d+\.\d+)', js).group(1)
  129. assert version == v
  130. # three.js
  131. url = 'https://www.npmjs.com/package/three'
  132. html = request_buffered_str(url)
  133. soup = BeautifulSoup(html, 'html.parser')
  134. version = soup.find('h3', string='Version').find_next_sibling('div').text.strip()
  135. url = f'https://cdn.jsdelivr.net/npm/three@{version}/build/three.min.js'
  136. js = request_buffered_str(url)
  137. Path('nicegui/elements/lib/three.min.js').write_text(js)
  138. print('Three.js:', version)
  139. # TODO: using script JS files is not supported after version 0.160.0 --> use ES module instead
  140. # TODO: CSS2DRenderer.js (requires ES modules)
  141. # TODO: CSS3DRenderer.js (requires ES modules)
  142. # TODO: OrbitControls.js (requires ES modules)
  143. # TODO: STLLoader.js (requires ES modules)
  144. ui.run()