angular.rb 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. require 'yajl/json_gem'
  2. module Docs
  3. class Angular < UrlScraper
  4. self.type = 'angular'
  5. self.links = {
  6. home: 'https://angular.dev/',
  7. code: 'https://github.com/angular/angular'
  8. }
  9. self.base_url = 'https://angular.io/'
  10. self.root_path = 'docs'
  11. options[:max_image_size] = 256_000
  12. options[:attribution] = <<-HTML
  13. Super-powered by Google &copy;2010&ndash;2025.<br />
  14. Code licensed under an MIT-style License. Documentation licensed under CC BY 4.0.
  15. HTML
  16. options[:follow_links] = false
  17. options[:only_patterns] = [/\Aguide/, /\Aapi/, /\Acli/]
  18. options[:fix_urls_before_parse] = ->(url) do
  19. url.sub! %r{\Aguide/}, '/guide/'
  20. url.sub! %r{\Aapi/}, '/api/'
  21. url.sub! %r{\cli/}, '/cli/'
  22. url.sub! %r{\Agenerated/}, '/generated/'
  23. url
  24. end
  25. module JsonNavigation
  26. private
  27. def initial_urls
  28. initial_urls = []
  29. Request.run "#{self.class.base_url}generated/navigation.json" do |response|
  30. data = JSON.parse(response.body)
  31. dig = ->(entry) do
  32. initial_urls << url_for("generated/docs/#{entry['url']}.json") if entry['url'] && entry['url'] != 'api'
  33. entry['children'].each(&dig) if entry['children']
  34. end
  35. data['SideNav'].each(&dig)
  36. end
  37. Request.run "#{self.class.base_url}generated/docs/api/api-list.json" do |response|
  38. data = JSON.parse(response.body)
  39. dig = ->(entry) do
  40. initial_urls << url_for("generated/docs/#{entry['path']}.json") if entry['path']
  41. initial_urls << url_for("generated/docs/api/#{entry['name']}.json") if entry['name'] && !entry['path']
  42. entry['items'].each(&dig) if entry['items']
  43. end
  44. data.each(&dig)
  45. end
  46. initial_urls
  47. end
  48. def handle_response(response)
  49. if response.mime_type.include?('json')
  50. begin
  51. json = JSON.parse(response.body)
  52. response.options[:response_body] = json['contents']
  53. response.url.path = response.url.path.gsub(/generated\/docs\/.*/, json['id'])
  54. response.effective_url.path = response.effective_url.path.gsub(/generated\/docs\/.*/, json['id'])
  55. rescue JSON::ParserError
  56. response.options[:response_body] = ''
  57. end
  58. response.headers['Content-Type'] = 'text/html'
  59. end
  60. super
  61. end
  62. end
  63. module Since12
  64. def url_for(path)
  65. # See encodeToLowercase im aio/tools/transforms/angular-base-package/processors/disambiguateDocPaths.js
  66. path = path.gsub(/[A-Z_]/) {|s| s.downcase + '_'}
  67. super
  68. end
  69. include Docs::Angular::JsonNavigation
  70. end
  71. module Since18
  72. def self.handle_redirects(version)
  73. lambda do |url|
  74. url.sub! '/guide/templates/reference-variables', '/guide/templates/variables#template-reference-variables'
  75. url.sub! '/guide/signals/inputs', '/guide/components/inputs'
  76. url.sub! '/guide/defer', '/guide/templates/defer'
  77. url.sub! '/guide/templates/class-binding', '/guide/templates/binding#css-class-and-style-property-bindings'
  78. url.sub! %r{/guide/components$}, '/guide/components/anatomy-of-components'
  79. url.sub! '/guide/templates/property-binding', '/guide/templates/binding#binding-dynamic-properties-and-attributes'
  80. url.sub! %r{/guide/ngmodules$}, '/guide/ngmodules/overview'
  81. url.sub! '/guide/components/importing', '/guide/components/anatomy-of-components#using-components'
  82. url.sub! '/guide/components/anatomy-of-components', '/guide/components' if version == '20'
  83. url
  84. end
  85. end
  86. end
  87. version do
  88. self.release = '20.3.4'
  89. self.base_url = 'https://angular.dev/'
  90. self.root_path = 'overview'
  91. options[:follow_links] = true
  92. options[:container] = '.docs-app-main-content'
  93. options[:fix_urls] = Since18.handle_redirects(self.version)
  94. html_filters.push 'angular/entries', 'angular/clean_html_v18'
  95. include Docs::Angular::Since18
  96. end
  97. version '19' do
  98. self.release = '19.2.15'
  99. self.base_url = 'https://v19.angular.dev/'
  100. self.root_path = 'overview'
  101. options[:follow_links] = true
  102. options[:container] = '.docs-app-main-content'
  103. options[:fix_urls] = Since18.handle_redirects(self.version)
  104. html_filters.push 'angular/entries', 'angular/clean_html_v18'
  105. include Docs::Angular::Since18
  106. end
  107. version '18' do
  108. self.release = '18.2.14'
  109. self.base_url = 'https://v18.angular.dev/'
  110. self.root_path = 'overview'
  111. options[:follow_links] = true
  112. options[:container] = '.docs-app-main-content'
  113. options[:fix_urls] = Since18.handle_redirects(self.version)
  114. html_filters.push 'angular/entries', 'angular/clean_html_v18'
  115. include Docs::Angular::Since18
  116. end
  117. version '17' do
  118. self.release = '17.0.8'
  119. self.base_url = 'https://v17.angular.io/'
  120. html_filters.push 'angular/clean_html', 'angular/entries'
  121. include Docs::Angular::Since12
  122. end
  123. version '16' do
  124. self.release = '16.2.12'
  125. self.base_url = 'https://v16.angular.io/'
  126. html_filters.push 'angular/clean_html', 'angular/entries'
  127. include Docs::Angular::Since12
  128. end
  129. version '15' do
  130. self.release = '15.2.9'
  131. self.base_url = 'https://v15.angular.io/'
  132. html_filters.push 'angular/clean_html', 'angular/entries'
  133. include Docs::Angular::Since12
  134. end
  135. version '14' do
  136. self.release = '14.2.12'
  137. self.base_url = 'https://v14.angular.io/'
  138. html_filters.push 'angular/clean_html', 'angular/entries'
  139. include Docs::Angular::Since12
  140. end
  141. version '13' do
  142. self.release = '13.3.8'
  143. self.base_url = 'https://v13.angular.io/'
  144. html_filters.push 'angular/clean_html', 'angular/entries'
  145. include Docs::Angular::Since12
  146. end
  147. version '12' do
  148. self.release = '12.2.13'
  149. self.base_url = 'https://v12.angular.io/'
  150. html_filters.push 'angular/clean_html', 'angular/entries'
  151. include Docs::Angular::Since12
  152. end
  153. version '11' do
  154. self.release = '11.2.14'
  155. self.base_url = 'https://v11.angular.io/'
  156. html_filters.push 'angular/clean_html', 'angular/entries'
  157. include Docs::Angular::JsonNavigation
  158. end
  159. version '10' do
  160. self.release = '10.2.3'
  161. self.base_url = 'https://v10.angular.io/'
  162. html_filters.push 'angular/clean_html', 'angular/entries'
  163. include Docs::Angular::JsonNavigation
  164. end
  165. version '9' do
  166. self.release = '9.1.12'
  167. self.base_url = 'https://v9.angular.io/'
  168. html_filters.push 'angular/clean_html', 'angular/entries'
  169. include Docs::Angular::JsonNavigation
  170. end
  171. version '8' do
  172. self.release = '8.2.14'
  173. self.base_url = 'https://v8.angular.io/'
  174. html_filters.push 'angular/clean_html', 'angular/entries'
  175. include Docs::Angular::JsonNavigation
  176. end
  177. version '7' do
  178. self.release = '7.2.15'
  179. self.base_url = 'https://v7.angular.io/'
  180. html_filters.push 'angular/clean_html', 'angular/entries'
  181. include Docs::Angular::JsonNavigation
  182. end
  183. version '6' do
  184. self.release = '6.1.10'
  185. self.base_url = 'https://v6.angular.io/'
  186. html_filters.push 'angular/clean_html', 'angular/entries'
  187. include Docs::Angular::JsonNavigation
  188. end
  189. version '5' do
  190. self.release = '5.2.11'
  191. self.base_url = 'https://v5.angular.io/'
  192. html_filters.push 'angular/clean_html', 'angular/entries'
  193. include Docs::Angular::JsonNavigation
  194. end
  195. version '4' do
  196. self.release = '4.4.6'
  197. self.base_url = 'https://v4.angular.io/'
  198. html_filters.push 'angular/clean_html', 'angular/entries'
  199. include Docs::Angular::JsonNavigation
  200. end
  201. version '2' do
  202. self.release = '2.4.10'
  203. self.base_url = 'https://v2.angular.io/docs/ts/latest/'
  204. self.root_path = 'api/'
  205. html_filters.push 'angular/entries_v2', 'angular/clean_html_v2'
  206. stub 'api/' do
  207. base_url = URL.parse(self.base_url)
  208. capybara = load_capybara_selenium
  209. capybara.app_host = base_url.origin
  210. capybara.visit(base_url.path + 'api/')
  211. capybara.execute_script('return document.body.innerHTML')
  212. end
  213. options[:skip_patterns] = [/deprecated/, /VERSION-let/]
  214. options[:skip] = %w(
  215. index.html
  216. styleguide.html
  217. quickstart.html
  218. cheatsheet.html
  219. guide/cheatsheet.html
  220. guide/style-guide.html)
  221. options[:replace_paths] = {
  222. 'testing/index.html' => 'guide/testing.html',
  223. 'guide/glossary.html' => 'glossary.html',
  224. 'tutorial' => 'tutorial/',
  225. 'api' => 'api/'
  226. }
  227. options[:fix_urls] = -> (url) do
  228. url.sub! %r{\A(https://(?:v2\.)?angular\.io/docs/.+/)index\.html\z}, '\1'
  229. url
  230. end
  231. end
  232. def get_latest_version(opts)
  233. get_npm_version('@angular/core', opts)
  234. end
  235. private
  236. def parse(response)
  237. response.body.gsub! '<code-example', '<pre'
  238. response.body.gsub! '</code-example', '</pre'
  239. response.body.gsub! '<code-pane', '<pre'
  240. response.body.gsub! '</code-pane', '</pre'
  241. response.body.gsub! '<live-example></live-example>', 'live example'
  242. response.body.gsub! '<live-example', '<span'
  243. response.body.gsub! '</live-example', '</span'
  244. super
  245. end
  246. end
  247. end