From 5da021471720c5c02fe30762a178d6148acb84e7 Mon Sep 17 00:00:00 2001 From: Filipe Silva Date: Sat, 15 Dec 2018 17:41:49 +0000 Subject: [PATCH] Add RxJS The current RxJS documentation site is https://rxjs.dev/. It is very similar to Angular documentation site (https://angular.io/) so I reused most code. Images on the documentation site seem to be broken and so the scrapper cannot download them. You can see an example of a broken image at https://rxjs.dev/api/operators/buffer. Related to https://github.com/freeCodeCamp/devdocs/issues/939 --- assets/stylesheets/application.css.scss | 1 + assets/stylesheets/pages/_rxjs.scss | 24 ++++++ lib/docs/filters/rxjs/clean_html.rb | 101 ++++++++++++++++++++++++ lib/docs/filters/rxjs/entries.rb | 23 ++++++ lib/docs/scrapers/rxjs.rb | 94 ++++++++++++++++++++++ public/icons/docs/rxjs/16.png | Bin 0 -> 5356 bytes public/icons/docs/rxjs/16@2x.png | Bin 0 -> 5084 bytes public/icons/docs/rxjs/SOURCE | 1 + 8 files changed, 244 insertions(+) create mode 100644 assets/stylesheets/pages/_rxjs.scss create mode 100644 lib/docs/filters/rxjs/clean_html.rb create mode 100644 lib/docs/filters/rxjs/entries.rb create mode 100644 lib/docs/scrapers/rxjs.rb create mode 100644 public/icons/docs/rxjs/16.png create mode 100644 public/icons/docs/rxjs/16@2x.png create mode 100644 public/icons/docs/rxjs/SOURCE diff --git a/assets/stylesheets/application.css.scss b/assets/stylesheets/application.css.scss index fd6ffffd..bcd8a7f2 100644 --- a/assets/stylesheets/application.css.scss +++ b/assets/stylesheets/application.css.scss @@ -94,6 +94,7 @@ 'pages/rfc', 'pages/rubydoc', 'pages/rust', + 'pages/rxjs', 'pages/sinon', 'pages/socketio', 'pages/sphinx', diff --git a/assets/stylesheets/pages/_rxjs.scss b/assets/stylesheets/pages/_rxjs.scss new file mode 100644 index 00000000..15e1252b --- /dev/null +++ b/assets/stylesheets/pages/_rxjs.scss @@ -0,0 +1,24 @@ +._rxjs { + @extend %simple; + + .pre-title { @extend %pre-heading; } + + .breadcrumbs { @extend %note; } + .banner { @extend %note-green; } + code.stable { @extend %label-green; } + code.experimental { @extend %label-orange; } + code.deprecated { @extend %label-red; } + .alert.is-important { @extend %note-red; } + .alert.is-helpful, .breadcrumbs { @extend %note-blue; } + + .breadcrumbs { padding-left: 2em; } + + img { margin: 1em 0; } + + .location-badge { + font-style: italic; + text-align: right; + } + + td h3 { margin: 0 !important; } +} diff --git a/lib/docs/filters/rxjs/clean_html.rb b/lib/docs/filters/rxjs/clean_html.rb new file mode 100644 index 00000000..1056b1a6 --- /dev/null +++ b/lib/docs/filters/rxjs/clean_html.rb @@ -0,0 +1,101 @@ +module Docs + class Rxjs + class CleanHtmlFilter < Filter + def call + if root_page? + css('.card-container').remove + at_css('h1').content = 'RxJS Documentation' + end + + css('br', 'hr', '.material-icons', '.header-link', '.breadcrumb').remove + + css('.content', 'article', '.api-header', 'section', '.instance-member').each do |node| + node.before(node.children).remove + end + + css('label', 'h2 > em', 'h3 > em').each do |node| + node.name = 'code' + end + + css('h1 + code').each do |node| + node.before('

') + while node.next_element.name == 'code' + node.previous_element << ' ' + node.previous_element << node.next_element + end + node.previous_element.prepend_child(node) + end + + css('td h3', '.l-sub-section > h3', '.alert h3', '.row-margin > h3', '.api-heading ~ h3', '.api-heading + h2', '.metadata-member h3').each do |node| + node.name = 'h4' + end + + css('.l-sub-section', '.alert', '.banner').each do |node| + node.name = 'blockquote' + end + + css('.file').each do |node| + node.content = node.content.strip + end + + css('.filetree .children').each do |node| + node.css('.file').each do |n| + n.content = " #{n.content}" + end + end + + css('.filetree').each do |node| + node.content = node.css('.file').map(&:inner_html).join("\n") + node.name = 'pre' + node.remove_attribute('class') + end + + css('pre').each do |node| + node.content = node.content.strip + + node['data-language'] = 'typescript' if node['path'].try(:ends_with?, '.ts') + node['data-language'] = 'html' if node['path'].try(:ends_with?, '.html') + node['data-language'] = 'css' if node['path'].try(:ends_with?, '.css') + node['data-language'] = 'js' if node['path'].try(:ends_with?, '.js') + node['data-language'] = 'json' if node['path'].try(:ends_with?, '.json') + node['data-language'] = node['language'].sub(/\Ats/, 'typescript').strip if node['language'] + node['data-language'] ||= 'typescript' if node.content.start_with?('@') + + node.before(%(
#{node['title']}
)) if node['title'] + + if node['class'] && node['class'].include?('api-heading') + node.name = 'h3' + node.inner_html = "#{node.inner_html}" + end + + node.remove_attribute('path') + node.remove_attribute('region') + node.remove_attribute('linenums') + node.remove_attribute('title') + node.remove_attribute('language') + node.remove_attribute('hidecopy') + node.remove_attribute('class') + end + + css('h1[class]').remove_attr('class') + css('table[class]').remove_attr('class') + css('table[width]').remove_attr('width') + css('tr[style]').remove_attr('style') + + if at_css('.api-type-label.module') + at_css('h1').content = subpath.remove('api/') + end + + css('th h3').each do |node| + node.name = 'span' + end + + css('code code').each do |node| + node.before(node.children).remove + end + + doc + end + end + end +end diff --git a/lib/docs/filters/rxjs/entries.rb b/lib/docs/filters/rxjs/entries.rb new file mode 100644 index 00000000..020ce1eb --- /dev/null +++ b/lib/docs/filters/rxjs/entries.rb @@ -0,0 +1,23 @@ +module Docs + class Rxjs + class EntriesFilter < Docs::EntriesFilter + def get_name + name = at_css('h1').content + name.prepend "#{$1}. " if subpath =~ /\-pt(\d+)/ + name + end + + def get_type + if slug.start_with?('guide') + 'Guide' + elsif at_css('.api-type-label.module') + name.split('/').first + elsif slug.start_with?('api/') + slug.split('/').second + else + 'Miscellaneous' + end + end + end + end +end diff --git a/lib/docs/scrapers/rxjs.rb b/lib/docs/scrapers/rxjs.rb new file mode 100644 index 00000000..1825fc80 --- /dev/null +++ b/lib/docs/scrapers/rxjs.rb @@ -0,0 +1,94 @@ +require 'yajl/json_gem' + +module Docs + class Rxjs < UrlScraper + self.name = 'RxJS' + self.type = 'rxjs' + self.links = { + home: 'https://rxjs.dev/', + code: 'https://github.com/ReactiveX/rxjs' + } + + options[:max_image_size] = 256_000 + + options[:attribution] = <<-HTML + © 2015–2018 Google, Inc., Netflix, Inc., Microsoft Corp. and contributors.
+ Code licensed under an Apache-2.0 License. Documentation licensed under CC BY 4.0. + HTML + + module Common + private + + def initial_urls + initial_urls = [] + + Request.run "#{self.class.base_url}generated/navigation.json" do |response| + data = JSON.parse(response.body) + dig = ->(entry) do + initial_urls << url_for("generated/docs/#{entry['url']}.json") if entry['url'] && entry['url'] != 'api' + entry['children'].each(&dig) if entry['children'] + end + data['SideNav'].each(&dig) + end + + Request.run "#{self.class.base_url}generated/docs/api/api-list.json" do |response| + data = JSON.parse(response.body) + dig = ->(entry) do + initial_urls << url_for("generated/docs/#{entry['path']}.json") if entry['path'] + initial_urls << url_for("generated/docs/api/#{entry['name']}.json") if entry['name'] && !entry['path'] + entry['items'].each(&dig) if entry['items'] + end + data.each(&dig) + end + + initial_urls + end + + def handle_response(response) + if response.mime_type.include?('json') + begin + response.options[:response_body] = JSON.parse(response.body)['contents'] + rescue JSON::ParserError + response.options[:response_body] = '' + end + response.headers['Content-Type'] = 'text/html' + response.url.path = response.url.path.sub('/generated/docs/', '/').remove('.json') + response.effective_url.path = response.effective_url.path.sub('/generated/docs/', '/').remove('.json') + end + super + end + end + + version do + self.release = '6.3.3' + self.base_url = 'https://rxjs.dev/' + self.root_path = 'guide/overview' + + html_filters.push 'rxjs/clean_html', 'rxjs/entries' + + options[:follow_links] = false + options[:only_patterns] = [/\Aguide/, /\Aapi/] + options[:fix_urls_before_parse] = ->(url) do + url.sub! %r{\Aguide/}, '/guide/' + url.sub! %r{\Aapi/}, '/api/' + url.sub! %r{\Agenerated/}, '/generated/' + url + end + + include Docs::Rxjs::Common + end + + private + + def parse(response) + response.body.gsub! '', 'live example' + response.body.gsub! ' zaB^>EX>4U6ba`-PAZ2)IW&i+q+O1bvk}M|-{O1%t0w5OYIII!AfsgM5MAb~cd5*Bd zT{Be_%_3z=DcX#`{+#9yd_4JJLd={bC5umxAiki%_I%FM*|6{WVsGPHzj?m!=psDn zyr!Y!3;N?bAqVZR-}_B&x(@3?cV%6W*3KRc9ml#5`P1+8-H6UB;fp)^-^O0TI$iYh zJ!F8^4FltlgczeKCj4_Uq>hc5`-~a~$tVNI)tXb?UmZTNqdYCGBYmSH^!tJMT+y zF#n%z#(p-}U}EsYXFR=co^~Ho_Tt+nC9KO$pGl}2Fyg6! zHSj_HvBm9uW$)J>se^d6}rw@ez1aFegz=RSj??EMu8AF z71Ox!-Iw_iH}(s$1P0@Vnc-lu^(;}Ndt-|>&xCQ&24hcZxZHICK!m*`jBx}8Y(jXu zQ(!ZykP3JDX3^x!;BaArG z$b*NBG7)Q&W|(oNnI}(~EVD6t;bIRIcEbMXlWewdq-zD6VlejnoV?H=A zo)-fMXumkK>_YI1xy6|kkEDnmthu25Iw2E%YZPWN#4GWUgs{w#UR(rOz|BdT_|aXER#!@k|c(v_cc|$MzxAF2KC>Ub_?Z=KV3Q z4V2?zZrfc$t(?J)(L9|nlnz)BCRdL^$er_TYs1qpRk>|M6LtL!00pa8L_oZ=iFCtJ zX`pqyMBs@DsU9F2Lv00HZm&s3v+dZ z?a?>~g?h|I1#dxgT58Lz4M2d99IM22D20ZLH(1)L|H-PCb)C3gP-7VB`T%Q@8c{>I zH+YZ;db0v@0f{UL%C)T8bWjE$U!wa$&rcsq*6lKx9{fC3iyMMt#c4~?w17y$hZCno zL$h#$Q+r^+YP=R+=j?=X0sC)~EoL+)aonksmgcRmDo=*s9ymt?B{pd)G&bo1wDecK zSQUG9vNTSQy+B5XAj)9dS+8iO8blTnK=b2uU-(1>G+0{iPjsO^s1Zy_0RqCpMj;0A zC@dFk&4_os1@S|pKx?`>&yZ&umau)um&W2m!I=U1;-747*%>-Au!GO|knAFx>S9^L2y_TUM(t$ zcv(fpUVNH;ju608E%Nt};I8#Xp#xP<*uO#fjpZAZmT#t5J+0wqZZ**KORy7I^@!HT z(@j!g>(Hnc=mrd*B4w7tIG(73np>oG1Sf@XVSemDX&D-!*Y@JAIpH6xta_trfk@1# z(odq|bS>s@Qutrc8(yNI000UxX+uL$Nkc;*P*P7uNlZlm0C=38mUmQB*%pV-y*Is3 zk`RiN&}(Q?0!R(LNRcioF$oY#z>okUHbhi#L{X8Z2r?+(fTKf^u_B6v0a3B*1Q|rs zac~qHmPur-8Q;8l@6DUvANPK1pS{oBXYYO1x&V;;g9XA&SP6g(p;#2*=f#MPi)Ua5 z0Sxc}18e}`aI>>Q7WhU2nF4&+jBJ?`_!qsp4j}paD$_rV!2tiCl(|_VF#u4QjOX(B z*<2YH$v8b%oF%tU$(Xh@P0lb%&LUZYGFFpw@+@0?_L*f5IrB1vJQ>S#&f;b8cV}o=_hCs$|GJ-ARc>v%@ z$zSl&FIdda6Uz_9&dgda5+tXH875p)hK-XGi{a1DP3Mcn%rFi&jU(bQ*qIqw9N}^R zX3zXt6nSkKvLZX!I5{{lZ7prSDAa#l{F{>Zc9vd*f9@GXANa%eSALld0I;TIwb}ZI zZD|z%UF!i*yZwjFU@riQvc7c=eQ_STd|pz-;w)z?tK8gNO97v2DKF^n`kxMeLtlK) zQoh~qM8wF>;&Ay4=AVc79|!(*9u^V&B)*6*lto0#rc5AA zmbF{R6Nm+wLWV&2pPKj&!~Ue%xt59A_z}>SSOTRX8bE#?04OREAPIY9E70$K3&uwS z`OS;bnV6mX&w~DaSGY|6$QC4jj$=neGPn{^&g`1}S^_j607XCp>OdRl0~5dmw!jg% z01w~;0zoK<1aV+7;DQv80Yo4d6o9p$7?gsoU?->sb)XS6gEnv&bb({wG&lz?fy-b7 z+yPQB4xWH1@CwX85QK%u5EW8~bRa{>9I}O2kQ?L!1w#=~9FzzpLqbRb6+r8tQm7oN zhU%ea=v(M0bQ-z<4MVq}QD_qS6?z9FFbSr?TCfpp1+!pJI0%k}7s1K!GB_VDg15kx za07f0?u1Xnm*5dt3O|9T5r7a8I--j(5f;KmLXmhR2@xTykP@TC z$XgT!MMW`COq2`C9~Fh-qL!gnp*EwcQ3p_+s6NzH)F^5S^$|@*Yog83&gcMiEIJvT zi!Mf2pqtPg=(Fe%^f>wz27{qvj4_TFe@q-E6|(}f8M7PHjyZ)H#*AU6u~@7+)*S1K z4aIV>Vr((C3VRTH5_<(Zj(vk8;&gDfIA2^mPKYbSRp451CvaDA6Sx_?65bH+j1R^0 z@XPUK_(psWeh5E~pCKp{j0vuUNJ1)MEuoUoMmS5jOL##f67`5q#Bid3xQ19sJVZQC z93{RbQAlPaHYtH5A#EY;C!HeQBE2A!$wp)kay(f~-a>9BpCR8TzfqtnSSkc4@Dx@n z)F^Z+Tv2$Yh*vaJ^i*7|n6Fr&ctmkX@u?DC$w-N<#8FzMRHJlM>4ws@GF90|IaE1A zd9!kh@&)Bb6fDJv;zQw4iYWUiXDDM-gsM+vQ@PZ2)JE!A>NpKUGo}U5QfZ~MZ)k(G zDHV!}ol3Myo=T0%aTO^Yp&QWy=;`z_`eFKY`a4xERZmsE>L%4T)hnv6)#j*qsPWZG z)Y{cX)ZVEx)P2;`)VHa3so&E;X_#q*YvgL|(KxH|bPjEf%N*{Uk~xRx+}4CO%`_u4 zS7`3j9MGKB($@0R%F?RRI-~Veo38DlovOV<`-JwS4pqlZN1(Gq=cLYKh6=-zkLZ@rEqJ z6vJJH{f4iNjE!Q9HW+moJu+4^4lvF)ZZ*DZLN;+XS!U8;a?KQD$}&we-EDf=3^ubj zOEIf48#0H@9n1yhyUm9!&=yV>LW>5A8%z?@lbOS8WsX|XErTr!ExRnASs7TxTWz!I zxB6&pZ=G)4Xnn_qViRanXwzf!tF4(W*S5y?+FbHn-?^*jcF%ooXKu&0+hcdro@yUr zzrnuO{)2;~gUF%HVbamSG10Ns@dk^=3S(_%op(Yzc{#0iI_C7&*}+-teAxLH7p6;^ zON+~+dB*ej^BU)kx$3!cTZVb0Xx4mvscU^amdxQG}4}A}wN0Y~dr>SSE=RwbB zUe;bBuMV%*Y-jdL_9<_~+t0hid(emC6XjFwbKh6bH`%w{ z0a^jvfaZXyK*zw9fqg-wpantIK@Wn>fV8I2F~=-fTgudr?_nHF76Ya2X6;&lJCkd=T9WLCY2{WN_I`&o;;c2 zo>GzWRKONg3!bO?r`DyuP76)jpY|y|CcQlamywupR7eq~3Hvg&GxIWsv&^%Kv!u(M zm+f3OB?=NXWkcDEvb)7J+0WE~#6+@QGMeL-QhTd=lZbfxFY`c=@XrK@^Z>#r_aJ-)_o&4IOqwP|aAD6}ptFMPQ!W?fH_ zR?(WGvGsoITZV0)e^+=6ZO?$0o?WWq-yLr2>?D5#sR;N{0TK8_RVDHU(zxvJwqlSuo zn0-0>9yUfd_J7U#y17ZCskG_Ce&K%UfrtZr&5q5@Et)N5t#GTPb@E`s!OP!xf79K@ zY^!glx0fCQha`s{f1CL2^}|7jdylY=w0&pzU2O-oqofn+T;4g=mC_~cj_V#i8hEs~ z$EBy^d&}?lAJaWnb6n+k*$Kjlq7$D^=AWECm38Xr>EzR6y-RxUoQXYituMT9@NCf8 z^XGieo$2@NKY8Bu{ILtp7mi+JUF^E#aH(^^exTzA`yV<69R@px9EZ9uJ6-M>o;Q5r ziu;w*SG}*EyB2Wm(#ZUg;pqt>?FMZqM9Va~FNLGD$ zlbNT*KP&%S`^@CocfWZ2GB6c8HU3=m{L`|I+Sd?{wJo{Z|>UW?q-PQGavb zE$eOnyO?(qGr8}v?<+r;e(3oa^zrVej8C6_1NVgU`<=UGpa1{>24YJ`L;(K){{a7> zy{D4^000SaNLh0L01m?d01m?e$8V@)00007bV*G`2jUD55&|V`TP&pj00RU`L_t(I z%WacOh}C5rhM(W}egFT=xz3q!oSeKwBEqB~X4GN|ClJYGHqj7Wa1lWlLI_#~;xtgW za#LZ8785a8O6jVLK_)qE0w)ob(h(AAnsFSD$8+YK|K_$rn{VNC}B?x`rapI z;wb@eb{&;(*k zXu@+%d*j$v=_h6J+jh3{t)Tvq4BgU(N}K%W-(gdV+30>CJ^K{q@d}DVG{qi{?ZB~D z^S$BUrl-3Ob%ukPCc1bUMtM8EC>7uRN}W#AZNiR4codF za4A{zI>YX7jRJwOsq4irG-cMP20M_9yokCKOsRcGTjf3pdos7mk<^ zM(z8T&T-(lto`^UR?Mw1DTyhlaf<8|>>*ExMAs-ROpB&xj|+ZxhMEL(ra*K)hwF8+ ziEC&Y;ofawcP)@@ULrrAQ|h@+YxkWmlqMu=_m-TRF9hA?37TQrIocI4iw*Q}9i6SC z&(;aUZJgY~zfiFCFXiqVOFs(9C4=wm1@n$VUk5MCXh;xAu+l=LfzI9}yj-$Nr$_Sr z;hq4XZ$h{|B>4Ffs!xO2q1bg9u!O(bAbgXE`8s#CWNZ4O0{jh$WL{&SUMK4S0000< KMNUMnLSTX!`8aC; literal 0 HcmV?d00001 diff --git a/public/icons/docs/rxjs/16@2x.png b/public/icons/docs/rxjs/16@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..a45cd5cb84184a412b136a825e6303f6d4fbcd0b GIT binary patch literal 5084 zcmV<26C>=2P)X+uL$Nkc;*P;zf(X>4Tx07wm;mUmQB*%pV-y*Itk5+Wca^cs2zAksTX z6$DXM^`x7XQc?|s+008spb1j2M!0f022SQPH-!CVp(%f$Br7!UytSOLJ{W@ZFO z_(THK{JlMynW#v{v-a*TfMmPdEWc1DbJqWVks>!kBnAKqMb$PuekK>?0+ds;#ThdH z1j_W4DKdsJG8Ul;qO2n0#IJ1jr{*iW$(WZWsE0n`c;fQ!l&-AnmjxZO1uWyz`0VP>&nP`#i ztsL#`S=Q!g`M=rU9)45(J;-|dRq-b5&z?byo>|{)?5r=n76A4nTALlSzLiw~v~31J z<>9PP?;rs31pu_(obw)rY+jPY;tVGXi|p)da{-@gE-UCa`=5eu%D;v=_nFJ?`&K)q z7e9d`Nfk3?MdhZarb|T3%nS~f&t(1g5dY)AIcd$w!z`Siz!&j_=v7hZlnI21XuE|x zfmo0(WD10T)!}~_HYW!eew}L+XmwuzeT6wtxJd`dZ#@7*BLgIEKY9Xv>st^p3dp{^ zXswa2bB{85{^$B13tWnB;Y>jyQ|9&zk7RNsqAVGs--K+z0uqo1bf5|}fi5rtEMN^B zfHQCd-XH*kfJhJnmIE$G0%<@5vOzxB0181d*a3EfYH$G5fqKvcPJ%XY23!PJzzuK< z41h;K3WmW;Fah3yX$XSw5EY_9s*o0>51B&N5F1(uc|$=^I1~fLLy3?Ol0f;;Ca4%H zgQ}rJP(Ab`bQ-z{U4#0d2hboi2K@njgb|nm(_szR0JebHusa+GN5aeCM0gdP2N%HG z;Yzp`J`T6S7vUT504#-H!jlL<$Or?`Mpy_N@kBz9SR?@vA#0H$qyni$nvf2p8@Y{0 zk#Xb$28W?xm>3qu8RLgpjNxKdVb)?wFx8l2m{v>|<~C*!GlBVnrDD~wrdTJeKXwT= z5u1%I#8zOBU|X=4u>;s)>^mF|$G{ol9B_WP7+f-LHLe7=57&&lfa}8z;U@8Tyei%l z?}87(bMRt(A-)QK9Dg3)j~~XrCy)tR1Z#p1A(kK{Y$Q|=8VKhI{e%(1G*N-5Pjn)N z5P8I0VkxnX*g?EW941ba6iJ387g8iCnY4jaNopcpCOsy-A(P2EWJhusSwLP-t|Xrz zUnLKcKTwn?CKOLf97RIePB}`sKzTrUL#0v;sBY9)s+hW+T2H-1eM)^VN0T#`^Oxhv zt&^*fYnAJldnHel*OzyfUoM{~Um<@={-*r60#U(0!Bc^wuvVc);k3d%g-J!4qLpHZ zVwz%!VuRu}#Ze`^l7W)95>Kf>>9Eozr6C$Z)1`URxU@~QI@)F0FdauXr2Es8>BaOP z=)Lp_WhG@>R;lZ?BJkMlIuMhw8Ap ziF&yDYW2hFJ?fJhni{?u85&g@mo&yT8JcdI$(rSw=QPK(Xj%)k1X|@<=e1rim6`6$ zRAwc!i#egKuI;BS(LSWzt39n_sIypSqfWEV6J3%nTQ@-4ii$R;gsG*9XzhRzXqv2yCs*$VFDx+GXJH|L;wsDH_KI2;^ zu!)^Xl1YupO;gy^-c(?^&$Q1BYvyPsG^;hc$D**@Sy`+`)}T4VJji^bd7Jqw3q6Zi zi=7tT7GEswEK@D(EFW1ZSp`^awCb?>!`j4}Yh7b~$A)U-W3$et-R8BesV(1jzwLcH znq9En7Q0Tn&-M=XBKs!$F$X<|c!#|X_tWYh)GZit(Q)Cp9CDE^WG;+fcyOWARoj*0TI>4EP1lX*cEoMO-Pk?Z z{kZ!p4@(b`M~lalr<3Oz&kJ6Nm#vN_+kA5 z{dW4@^Vjg_`q%qU1ULk&3Fr!>1V#i_2R;ij2@(Z$1jE4r!MlPVFVbHmT+|iPIq0wy5aS{>yK?9ZAjVh%SOwMWgFja zir&;wpi!{CU}&@N=Eg#~LQ&zpEzVmGY{hI9Z0+4-0x zS$$Xe-OToc?Y*V;rTcf_b_jRe-RZjXSeas3UfIyD;9afd%<`i0x4T#DzE)vdabOQ= zk7SRuGN`h>O0Q~1)u-yD>VX=Mn&!Rgd$;YK+Q-}1zu#?t(*cbG#Ronf6db&N$oEid ztwC+YVcg-Y!_VuY>bk#Ye_ww@?MU&F&qswvrN_dLb=5o6*Egs)ls3YRlE$&)amR1{ z;Ppd$6RYV^Go!iq1UMl%@#4q$AMc(FJlT1QeX8jv{h#)>&{~RGq1N2iiMFIRX?sk2 z-|2wUogK~{EkB$8eDsX=nVPf8XG_nK&J~=SIiGia@9y}|z3FhX{g&gc zj=lwb=lWgyFW&aLedUh-of`v-2Kw$UzI*>(+&$@i-u=-BsSjR1%z8NeX#HdC`Hh-Z(6xI-`hmHDqv!v)W&&nrf>M(RhcN6(D;jNN*% z^u_SYjF;2ng}*8Ow)d6MtDk;%`@Lsk$;9w$(d(H%O5UixIr`T2ZRcd@D2{B1TK~z`?omYL3Rb?6fJ?GOA+ z#mCP1P9VF`>GtA)P22A9n{(zzc#!h|NGEL^5Fo9_OTuT1rXK^*G*AUwM6tggcEwi6 z%QryGDU|!uc3J+k4Fd1sLwXqe>Gv17ITwH$w;kr3b+T?7ki7QP_#Pcb z5tUO>NWp{oFbW)Wf!}!We{r*Y@MyS)GKjPx?h$D|13*NebjY*u4BTt#fGYr}y&wLZ zUm_p;jSn6m_1t+B5OCh9hbshtI2McQ0F#V&pKG>{S4r~;^YE?x5t zWEE+kaWD27H-8ZYGoHp~^PWa=j7ItyAoVE*7!E}E6d`&)Rk+AyWMg!jLIA@8KO*q+2FW>1NX%|@;<>>0qsA1GsWc&W2S3?; zP}Yr81{8$uQ@c2KFa(f<7cg#aH)r<2I+yZ)+Q1Qx{p4F!l;b!R4vWDeZ^(e|<~0lB zpU3F0&qC_qTgXV3JUR~9@G{7h8WFqJgY4KXtLwussDg*Dv(T7N{~5l&4Fx-m^97oW zbCELeDhEM<%p=MSkLAPIGyc>p#F3Mh0*xd0%2Ci(>?fHL+LzA5MhI96EcC&a{ZWhX{J@uUx%wc$4fuHd(}1V@Un1A7rHT!^4z?W3=#eJ_yp^IUk__o9M5 zsFE@>N-g3YDA_jl68le^@ly??Bi-;2P&iqk#StyLgVVSj$sCZD0~{P zl*%Z6`!bjf%j~i~PU9{4{*x5&U6jKYBPTJv0kL|t}#V8ExHf)bbror)p{IOW)Q1Z zx-7@A=f)%Lq%RkRnfn#HbYG`*YI0OMYDI-bY!{tWbzfs2e1~&7wsGJM>1s&9@pCZc zPuliDeFv_0FzO15^Ud(lVol2iVDC{Nz5v!_xEblx4AT)%-R=-d9j&(gi5}^Ru)%0# z&c7(A9&|hDFs&ysZ>wFjwvfO*1%1DK=V~DFZUM!mMDz+7a1sMB(d)UAf|3&BMZxJX zK1&We-#`*(>+nXZrwd)4fDv`W8>@kcD@1yYMcua1K${*vfPkW5j%0@(- zF(?{}hiO#Zg{!U4%+I2HUTbNxxs6+e`8f;`qpR6U0 z2bmu!1Go+TA`j(bbu6(o9`<+w5r>*0kd}YQ7FCT#B21%?0zyMpJCbdXt{+~C(Vq{H zFGb)SkCST$l{YtIu-jo?de%PB-w3l!iKiusF=T;f!4frKCtD2NTs7KB$?(V)9`X+uhUA&`!>rp$^Zqamuw_012K4>vcMr^f&Dy6 zh2RlgQATQNR@UAg=D7Ctv)XF714E>=Yq!(q0wO%{F#O1jhv|oa-i&^p#6&9oUm)IpgtH{PfEUU}ow*4n+L>Uaf z4Ij?`(<5Co*%80#A#vMDWCvK57YA^A+3q)&V7@#IfWBye>5)a(Zz;c9dHg8xMTBLK zt<7}^q{6m(>lk1PP`@dQ%;QPB_-Qq%r<_^n2=-x3iuM2F1$YfKG@VA~F_<@2Sl0TG zdz&8a;Jb-&G|fMOzU|BHl6$HEbQg`QmQUGurA6=uLF6)$Nc$2g^N7Z_!EQ(_a2`p* z&Tx_Z!T>Tm*`k*>+lTh(7JA}gLO&lzf<-Y(qj~;{VXtH;i;xVNs7s*H>&4(Z1>~kZ zW*-|k10V!ZrL!a(CbIy&N+x(R<4Kd&+0~c@4hxqDcD((ZA76)LeLo5Tz1_~ux6{Md zc=#NAUDnQ}QWx<34SCqB7|J+3r`94)2IW5P_1OpP!*6MgI2XWKAd0evg~II?ByOtq zN6YgC*emo*>SK_SWKpQNJom*|uK4SSz6M7*f0&4m6HuItUB|FZD8P7ZYTh791C0CU z?zc-{55q>?oev-cv5Uf*MTfdH8CKExC8U)0xGj4-Jy2TNsNgU*8W+QNtgv*53@NlM zlZyt!d!GOqjz8rg_qE!^Z`X{6rbaC9HBE%8)=#5Gzd;0ZStDtPX1O&zW0t^h@bp7u z92BdvDLulE)c*h&&kU)?