def initialize(account, attributes)
@account = account
@attributes = attributes
- @name = attributes['name'].strip[0, 255]
- @value = attributes['value'].strip[0, 255]
+ @name = attributes['name'].strip[0, string_limit]
+ @value = attributes['value'].strip[0, string_limit]
@verified_at = attributes['verified_at']&.to_datetime
@errors = {}
end
verified_at.present?
end
+ def value_for_verification
+ @value_for_verification ||= begin
+ if account.local?
+ value
+ else
+ ActionController::Base.helpers.strip_tags(value)
+ end
+ end
+ end
+
def verifiable?
- value.present? && value.start_with?('http://', 'https://')
+ value_for_verification.present? && value_for_verification.start_with?('http://', 'https://')
end
def mark_verified!
def to_h
{ name: @name, value: @value, verified_at: @verified_at }
end
+
+ private
+
+ def string_limit
+ if account.local?
+ 255
+ else
+ 2047
+ end
+ end
end
class << self
RSpec.describe VerifyLinkService, type: :service do
subject { described_class.new }
- let(:account) { Fabricate(:account, username: 'alice') }
- let(:field) { Account::Field.new(account, 'name' => 'Website', 'value' => 'http://example.com') }
+ context 'given a local account' do
+ let(:account) { Fabricate(:account, username: 'alice') }
+ let(:field) { Account::Field.new(account, 'name' => 'Website', 'value' => 'http://example.com') }
- before do
- stub_request(:head, 'https://redirect.me/abc').to_return(status: 301, headers: { 'Location' => ActivityPub::TagManager.instance.url_for(account) })
- stub_request(:get, 'http://example.com').to_return(status: 200, body: html)
- subject.call(field)
- end
-
- context 'when a link contains an <a> back' do
- let(:html) do
- <<-HTML
- <!doctype html>
- <body>
- <a href="#{ActivityPub::TagManager.instance.url_for(account)}" rel="me">Follow me on Mastodon</a>
- </body>
- HTML
+ before do
+ stub_request(:head, 'https://redirect.me/abc').to_return(status: 301, headers: { 'Location' => ActivityPub::TagManager.instance.url_for(account) })
+ stub_request(:get, 'http://example.com').to_return(status: 200, body: html)
+ subject.call(field)
end
- it 'marks the field as verified' do
- expect(field.verified?).to be true
+ context 'when a link contains an <a> back' do
+ let(:html) do
+ <<-HTML
+ <!doctype html>
+ <body>
+ <a href="#{ActivityPub::TagManager.instance.url_for(account)}" rel="me">Follow me on Mastodon</a>
+ </body>
+ HTML
+ end
+
+ it 'marks the field as verified' do
+ expect(field.verified?).to be true
+ end
end
- end
- context 'when a link contains an <a rel="noopener"> back' do
- let(:html) do
- <<-HTML
- <!doctype html>
- <body>
- <a href="#{ActivityPub::TagManager.instance.url_for(account)}" rel="noopener me" target="_blank">Follow me on Mastodon</a>
- </body>
- HTML
+ context 'when a link contains an <a rel="noopener"> back' do
+ let(:html) do
+ <<-HTML
+ <!doctype html>
+ <body>
+ <a href="#{ActivityPub::TagManager.instance.url_for(account)}" rel="noopener me" target="_blank">Follow me on Mastodon</a>
+ </body>
+ HTML
+ end
+
+ it 'marks the field as verified' do
+ expect(field.verified?).to be true
+ end
end
- it 'marks the field as verified' do
- expect(field.verified?).to be true
+ context 'when a link contains a <link> back' do
+ let(:html) do
+ <<-HTML
+ <!doctype html>
+ <head>
+ <link type="text/html" href="#{ActivityPub::TagManager.instance.url_for(account)}" rel="me" />
+ </head>
+ HTML
+ end
+
+ it 'marks the field as verified' do
+ expect(field.verified?).to be true
+ end
end
- end
- context 'when a link contains a <link> back' do
- let(:html) do
- <<-HTML
- <!doctype html>
- <head>
- <link type="text/html" href="#{ActivityPub::TagManager.instance.url_for(account)}" rel="me" />
- </head>
- HTML
+ context 'when a link goes through a redirect back' do
+ let(:html) do
+ <<-HTML
+ <!doctype html>
+ <head>
+ <link type="text/html" href="https://redirect.me/abc" rel="me" />
+ </head>
+ HTML
+ end
+
+ it 'marks the field as verified' do
+ expect(field.verified?).to be true
+ end
end
- it 'marks the field as verified' do
- expect(field.verified?).to be true
+ context 'when a link does not contain a link back' do
+ let(:html) { '' }
+
+ it 'marks the field as verified' do
+ expect(field.verified?).to be false
+ end
end
end
- context 'when a link goes through a redirect back' do
- let(:html) do
- <<-HTML
- <!doctype html>
- <head>
- <link type="text/html" href="https://redirect.me/abc" rel="me" />
- </head>
- HTML
- end
+ context 'given a remote account' do
+ let(:account) { Fabricate(:account, username: 'alice', domain: 'example.com', url: 'https://profile.example.com/alice') }
+ let(:field) { Account::Field.new(account, 'name' => 'Website', 'value' => '<a href="http://example.com" rel="me"><span class="invisible">http://</span><span class="">example.com</span><span class="invisible"></span></a>') }
- it 'marks the field as verified' do
- expect(field.verified?).to be true
+ before do
+ stub_request(:get, 'http://example.com').to_return(status: 200, body: html)
+ subject.call(field)
end
- end
- context 'when a link does not contain a link back' do
- let(:html) { '' }
+ context 'when a link contains an <a> back' do
+ let(:html) do
+ <<-HTML
+ <!doctype html>
+ <body>
+ <a href="https://profile.example.com/alice" rel="me">Follow me on Mastodon</a>
+ </body>
+ HTML
+ end
- it 'marks the field as verified' do
- expect(field.verified?).to be false
+ it 'marks the field as verified' do
+ expect(field.verified?).to be true
+ end
end
end
end