# frozen_string_literal: true RSpec.describe UserNotificationsHelper do let(:upload_path) { Discourse.store.upload_path } describe "#email_excerpt" do let(:paragraphs) do [ "<p>This is the first paragraph, but you should read more.</p>", "<p>And here is its friend, the second paragraph.</p>", ] end let(:cooked) { paragraphs.join("\n") } let(:post_quote) { <<~HTML } <aside class="quote no-group" data-post="859" data-topic="30"> <div class="title"> <div class="quote-controls"></div> <img alt width="20" height="20" src="https://example.com/m.png" class="avatar"> modman:</div> <blockquote> <p>This is a post quote</p> </blockquote> </aside> HTML let(:image_paragraph) do '<p><img src="//localhost:3000/uploads/b9.png" width="300" height="300"></p>' end let(:lightbox_image) { <<~HTML } <p><div class="lightbox-wrapper"><a class="lightbox" href="//localhost:3000/uploads/default/original/1X/123456.jpeg" data-download-href="//localhost:3000/uploads/default/123456" title="giant-meteor-2020"><img src="//localhost:3000/uploads/default/original/1X/123456.jpeg" alt="giant-meteor-2020" data-base62-sha1="3jcR88161od6Uthq1ixWKJh2ejp" width="517" height="152" data-small-upload="//localhost:3000/uploads/default/optimized/1X/123456_2_10x10.png"><div class="meta"> <svg class="fa d-icon d-icon-far-image svg-icon" aria-hidden="true"><use href="#far-image"></use></svg><span class="filename">giant-meteor-2020</span><span class="informations">851×251 44 KB</span><svg class="fa d-icon d-icon-discourse-expand svg-icon" aria-hidden="true"><use href="#discourse-expand"></use></svg> </div></a></div></p> HTML let(:expected_lightbox_image) do '<div class="lightbox-wrapper"><a class="lightbox" href="//localhost:3000/uploads/default/original/1X/123456.jpeg" data-download-href="//localhost:3000/uploads/default/123456" title="giant-meteor-2020"><img src="//localhost:3000/uploads/default/original/1X/123456.jpeg" alt="giant-meteor-2020" data-base62-sha1="3jcR88161od6Uthq1ixWKJh2ejp" width="517" height="152" data-small-upload="//localhost:3000/uploads/default/optimized/1X/123456_2_10x10.png"></a></div>' end it "can return the first paragraph" do SiteSetting.digest_min_excerpt_length = 50 expect(helper.email_excerpt(cooked)).to eq(paragraphs[0]) end it "can return another paragraph to satisfy digest_min_excerpt_length" do SiteSetting.digest_min_excerpt_length = 100 expect(helper.email_excerpt(cooked)).to eq(paragraphs.join) end it "doesn't count emoji images" do with_emoji = "<p>Hi <img src=\"/images/emoji/twitter/smile.png?v=#{Emoji::EMOJI_VERSION}\" title=\":smile:\" class=\"emoji\" alt=\":smile:\" loading=\"lazy\" width=\"20\" height=\"20\"></p>" arg = ([with_emoji] + paragraphs).join("\n") SiteSetting.digest_min_excerpt_length = 50 expect(helper.email_excerpt(arg)).to eq([with_emoji, paragraphs[0]].join) end it "only counts link text" do with_link = "<p>Hi <a href=\"https://really-long.essays.com/essay/number/9000/this-one-is-about-friends-and-got-a-C-minus-in-grade-9\">friends</a>!</p>" arg = ([with_link] + paragraphs).join("\n") SiteSetting.digest_min_excerpt_length = 50 expect(helper.email_excerpt(arg)).to eq([with_link, paragraphs[0]].join) end it "uses user quotes but not post quotes" do cooked = <<~HTML <p>BEFORE</p> <blockquote> <p>This is a user quote</p> </blockquote> <aside class="quote" data-post="3" data-topic="87369"> <div class="title">A Title</div> <blockquote> <p>This is a post quote</p> </blockquote> </aside> <p>AFTER</p> HTML expect( helper.email_excerpt(cooked), ).to eq "<p>BEFORE</p><blockquote>\n <p>This is a user quote</p>\n</blockquote><p>AFTER</p>" end it "defaults to content after post quote (image w/ no text)" do cooked = <<~HTML #{post_quote} #{image_paragraph} HTML expect(helper.email_excerpt(cooked)).to eq(image_paragraph) end it "defaults to content after post quote (onebox)" do aside_onebox = '<aside class="onebox wikipedia"><article class="onebox-body"><p>Onebox excerpt here</p></article><div class="onebox-metadata"></div></aside>' cooked = <<~HTML #{post_quote} #{aside_onebox} HTML expect(helper.email_excerpt(cooked)).to eq(aside_onebox) end it "defaults to content after post quote (lightbox image w/ no text)" do cooked = <<~HTML #{post_quote} #{lightbox_image} HTML expect(helper.email_excerpt(cooked)).to eq(expected_lightbox_image) end it "handles when there's only an image" do image_paragraph expect(helper.email_excerpt("#{image_paragraph}")).to eq(image_paragraph) end it "handles when there's only a lightboxed image" do expect(helper.email_excerpt("#{lightbox_image}")).to eq(expected_lightbox_image) end end describe "#logo_url" do describe "local store" do let(:upload) { Fabricate(:upload, sha1: "somesha1") } before { SiteSetting.logo = upload } it "should return the right URL" do expect(helper.logo_url).to eq( "http://test.localhost/#{upload_path}/original/1X/somesha1.png", ) end describe "when cdn path is configured" do before do GlobalSetting.expects(:cdn_url).returns("https://some.localcdn.com").at_least_once end it "should return the right URL" do expect(helper.logo_url).to eq( "https://some.localcdn.com/#{upload_path}/original/1X/somesha1.png", ) end end describe "when logo is an SVG" do let(:upload) { Fabricate(:upload, extension: "svg") } it "should return nil" do expect(helper.logo_url).to eq(nil) end end end describe "s3 store" do let(:upload) { Fabricate(:upload_s3, sha1: "somesha1") } before do setup_s3 SiteSetting.logo = upload end it "should return the right URL" do expect(helper.logo_url).to eq( "http://s3-upload-bucket.s3.dualstack.#{SiteSetting.s3_region}.amazonaws.com/original/1X/somesha1.png", ) end describe "when global cdn path is configured" do it "should return the right url" do GlobalSetting.stubs(:cdn_url).returns("https://some.cdn.com/cluster") expect(helper.logo_url).to eq( "http://s3-upload-bucket.s3.dualstack.#{SiteSetting.s3_region}.amazonaws.com/original/1X/somesha1.png", ) end end describe "when cdn path is configured" do before { SiteSetting.s3_cdn_url = "https://some.cdn.com" } it "should return the right url" do expect(helper.logo_url).to eq("https://some.cdn.com/original/1X/somesha1.png") end end end end end