1
0

fetch_dependencies.py 6.6 KB

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