Moved notifications about automatic contact name update to bulk change poll message

This commit is contained in:
Sergei Tsõganov 2022-03-21 09:10:34 +02:00 committed by Sergei Tsõganov
parent e4d56fe576
commit 216048463d
18 changed files with 201 additions and 96 deletions

View file

@ -2,18 +2,37 @@ class Action < ApplicationRecord
has_paper_trail versions: { class_name: 'Version::ActionVersion' } has_paper_trail versions: { class_name: 'Version::ActionVersion' }
belongs_to :user belongs_to :user
belongs_to :contact belongs_to :contact, optional: true
has_many :subactions, class_name: 'Action', foreign_key: 'bulk_action_id', dependent: :destroy
belongs_to :bulk_action, class_name: 'Action', optional: true
validates :operation, inclusion: { in: proc { |action| action.class.valid_operations } } validates :operation, inclusion: { in: proc { |action| action.class.valid_operations } }
class << self class << self
def valid_operations def valid_operations
%w[update] %w[update bulk_update]
end end
end end
def notification_key def notification_key
raise 'Action object is missing' unless contact raise 'Action object is missing' unless bulk_action? || contact
"contact_#{operation}".to_sym "contact_#{operation}".to_sym
end end
def bulk_action?
!!subactions.exists?
end
def to_non_available_contact_codes
return [] unless bulk_action?
subactions.map do |a|
{
code: a.contact&.code,
avail: 0,
reason: 'in use',
}
end
end
end end

View file

@ -0,0 +1,2 @@
class BulkAction < Action
end

View file

@ -47,7 +47,7 @@ class Epp::Contact < Contact
codes = codes.map { |c| c.include?(':') ? c : "#{reg}:#{c}" } codes = codes.map { |c| c.include?(':') ? c : "#{reg}:#{c}" }
res = [] res = []
codes.map { |c| c.include?(':') ? c : "#{reg}:#{c}" }.map { |c| c.strip.upcase }.each do |x| codes.map { |c| c.strip.upcase }.each do |x|
c = find_by_epp_code(x) c = find_by_epp_code(x)
res << (c ? { code: c.code, avail: 0, reason: 'in use' } : { code: x, avail: 1 }) res << (c ? { code: c.code, avail: 0, reason: 'in use' } : { code: x, avail: 1 })
end end

View file

@ -2,7 +2,7 @@ class Notification < ApplicationRecord
include Versions # version/notification_version.rb include Versions # version/notification_version.rb
belongs_to :registrar belongs_to :registrar
belongs_to :action belongs_to :action, optional: true
scope :unread, -> { where(read: false) } scope :unread, -> { where(read: false) }

View file

@ -112,13 +112,17 @@ class RegistrantUser < User
end end
def update_related_contacts def update_related_contacts
contacts = Contact.where(ident: ident, ident_country_code: country.alpha2) grouped_contacts = Contact.where(ident: ident, ident_country_code: country.alpha2)
.where('UPPER(name) != UPPER(?)', username) .where('UPPER(name) != UPPER(?)', username)
.includes(:registrar).group_by { |c| c.registrar }
contacts.each do |contact| grouped_contacts.each do |registrar, contacts|
contact.update(name: username) bulk_action, action = actions.create!(operation: :bulk_update) if contacts.size > 1
action = actions.create!(contact: contact, operation: :update) contacts.each do |c|
contact.registrar.notify(action) if c.update(name: username)
action = actions.create!(contact: c, operation: :update, bulk_action_id: bulk_action&.id)
end
end
registrar.notify(bulk_action || action)
end end
end end

View file

@ -218,9 +218,16 @@ class Registrar < ApplicationRecord
end end
def notify(action) def notify(action)
text = I18n.t("notifications.texts.#{action.notification_key}", contact: action.contact.code) text = I18n.t("notifications.texts.#{action.notification_key}", contact: action.contact&.code,
count: action.subactions&.count)
if action.bulk_action?
notifications.create!(text: text, action_id: action.id,
attached_obj_type: 'BulkAction',
attached_obj_id: action.id)
else
notifications.create!(text: text) notifications.create!(text: text)
end end
end
def e_invoice_iban def e_invoice_iban
iban iban

View file

@ -5,15 +5,7 @@ xml.epp_head do
end end
xml.resData do xml.resData do
xml.tag!('contact:chkData', 'xmlns:contact' => xml << render('epp/contacts/partials/check', builder: xml, results: @results)
Xsd::Schema.filename(for_prefix: 'contact-ee', for_version: '1.1')) do
@results.each do |result|
xml.tag!('contact:cd') do
xml.tag! "contact:id", result[:code], avail: result[:avail]
xml.tag!('contact:reason', result[:reason]) unless result[:avail] == 1
end
end
end
end end
render('epp/shared/trID', builder: xml) render('epp/shared/trID', builder: xml)

View file

@ -0,0 +1,9 @@
builder.tag!('contact:chkData', 'xmlns:contact' =>
Xsd::Schema.filename(for_prefix: 'contact-ee', for_version: '1.1')) do
results.each do |result|
builder.tag!('contact:cd') do
builder.tag! 'contact:id', result[:code], avail: result[:avail]
# builder.tag!('contact:reason', result[:reason]) unless result[:avail] == 1
end
end
end

View file

@ -1,9 +1,12 @@
builder.extension do builder.extension do
builder.tag!('changePoll:changeData', builder.tag!('changePoll:changeData',
'xmlns:changePoll' => Xsd::Schema.filename(for_prefix: 'changePoll')) do 'xmlns:changePoll' => Xsd::Schema.filename(for_prefix: 'changePoll', for_version: '1.0')) do
builder.tag!('changePoll:operation', action.operation) builder.tag!('changePoll:operation', action.operation)
builder.tag!('changePoll:date', action.created_at.utc.xmlschema) builder.tag!('changePoll:date', action.created_at.utc.xmlschema)
builder.tag!('changePoll:svTRID', action.id) builder.tag!('changePoll:svTRID', action.id)
builder.tag!('changePoll:who', action.user) builder.tag!('changePoll:who', action.user)
if action.bulk_action?
builder.tag!('changePoll:reason', 'Auto-update according to official data')
end
end end
end end

View file

@ -9,13 +9,24 @@ xml.epp_head do
xml.msg @notification.text xml.msg @notification.text
end end
if @notification.attached_obj_type == 'DomainTransfer' && @object if @object
case @notification.attached_obj_type
when 'DomainTransfer'
xml.resData do xml.resData do
xml << render('epp/domains/partials/transfer', builder: xml, dt: @object) xml << render('epp/domains/partials/transfer', builder: xml, dt: @object)
end end
when 'BulkAction'
xml.resData do
xml << render(
'epp/contacts/partials/check',
builder: xml,
results: @object.to_non_available_contact_codes
)
end
end
end end
if @notification.action&.contact || @notification.registry_lock? if @notification.action || @notification.registry_lock?
if @notification.registry_lock? if @notification.registry_lock?
state = @notification.text.include?('unlocked') ? 'unlock' : 'lock' state = @notification.text.include?('unlocked') ? 'unlock' : 'lock'
xml.extension do xml.extension do

View file

@ -6,6 +6,7 @@ en:
It was associated with registrant %{old_registrant_code} It was associated with registrant %{old_registrant_code}
and contacts %{old_contacts_codes}. and contacts %{old_contacts_codes}.
contact_update: Contact %{contact} has been updated by registrant contact_update: Contact %{contact} has been updated by registrant
contact_bulk_update: '%{count} contacts have been updated by registrant'
csync: CSYNC DNSSEC %{action} for domain %{domain} csync: CSYNC DNSSEC %{action} for domain %{domain}
registrar_locked: Domain %{domain_name} has been locked by registrant registrar_locked: Domain %{domain_name} has been locked by registrant
registrar_unlocked: Domain %{domain_name} has been unlocked by registrant registrar_unlocked: Domain %{domain_name} has been unlocked by registrant

View file

@ -0,0 +1,5 @@
class AddBulkActions < ActiveRecord::Migration[6.1]
def change
add_column :actions, :bulk_action_id, :integer, default: nil
end
end

View file

@ -304,7 +304,8 @@ CREATE TABLE public.actions (
user_id integer, user_id integer,
operation character varying NOT NULL, operation character varying NOT NULL,
created_at timestamp without time zone, created_at timestamp without time zone,
contact_id integer contact_id integer,
bulk_action_id integer
); );
@ -5400,10 +5401,14 @@ INSERT INTO "schema_migrations" (version) VALUES
('20211126085139'), ('20211126085139'),
('20211231113934'), ('20211231113934'),
('20220106123143'), ('20220106123143'),
<<<<<<< HEAD
('20220113201642'), ('20220113201642'),
('20220113220809'), ('20220113220809'),
('20220124105717'), ('20220124105717'),
('20220216113112'), ('20220216113112'),
('20220228093211'); ('20220228093211');
=======
('20220316140727');
>>>>>>> f98598620 (Moved notifications about automatic contact name update to bulk change poll message)

View file

@ -3,3 +3,23 @@ contact_update:
contact: john contact: john
created_at: <%= Time.zone.parse('2010-07-05').to_s(:db) %> created_at: <%= Time.zone.parse('2010-07-05').to_s(:db) %>
user: registrant user: registrant
contacts_update_bulk_action:
operation: bulk_update
user: registrant
contact_update_subaction_one:
operation: update
contact: william
created_at: <%= Time.zone.parse('2010-07-05').to_s(:db) %>
user: registrant
bulk_action: contacts_update_bulk_action
contact_update_subaction_two:
operation: update
contact: jane
created_at: <%= Time.zone.parse('2010-07-05').to_s(:db) %>
user: registrant
bulk_action: contacts_update_bulk_action

View file

@ -76,7 +76,7 @@ class EppContactCheckBaseTest < EppTestCase
response_xml = Nokogiri::XML(response.body) response_xml = Nokogiri::XML(response.body)
assert_correct_against_schema response_xml assert_correct_against_schema response_xml
assert_equal '0', response_xml.at_xpath('//contact:id', contact: xml_schema)['avail'] assert_equal '0', response_xml.at_xpath('//contact:id', contact: xml_schema)['avail']
assert_equal 'in use', response_xml.at_xpath('//contact:reason', contact: xml_schema).text # assert_equal 'in use', response_xml.at_xpath('//contact:reason', contact: xml_schema).text
end end
def test_multiple_contacts def test_multiple_contacts
@ -127,7 +127,7 @@ class EppContactCheckBaseTest < EppTestCase
assert_correct_against_schema response_xml assert_correct_against_schema response_xml
assert_epp_response :completed_successfully assert_epp_response :completed_successfully
assert_equal "#{@contact.registrar.code}:JOHN-001".upcase, response_xml.at_xpath('//contact:id', contact: xml_schema).text assert_equal "#{@contact.registrar.code}:JOHN-001".upcase, response_xml.at_xpath('//contact:id', contact: xml_schema).text
assert_equal 'in use', response_xml.at_xpath('//contact:reason', contact: xml_schema).text # assert_equal 'in use', response_xml.at_xpath('//contact:reason', contact: xml_schema).text
end end
def test_check_contact_without_prefix def test_check_contact_without_prefix
@ -154,7 +154,7 @@ class EppContactCheckBaseTest < EppTestCase
assert_correct_against_schema response_xml assert_correct_against_schema response_xml
assert_epp_response :completed_successfully assert_epp_response :completed_successfully
assert_equal "#{@contact.registrar.code}:JOHN-001".upcase, response_xml.at_xpath('//contact:id', contact: xml_schema).text assert_equal "#{@contact.registrar.code}:JOHN-001".upcase, response_xml.at_xpath('//contact:id', contact: xml_schema).text
assert_equal 'in use', response_xml.at_xpath('//contact:reason', contact: xml_schema).text # assert_equal 'in use', response_xml.at_xpath('//contact:reason', contact: xml_schema).text
end end
private private

View file

@ -7,15 +7,7 @@ class EppPollTest < EppTestCase
# Deliberately does not conform to RFC5730, which requires the first notification to be returned # Deliberately does not conform to RFC5730, which requires the first notification to be returned
def test_return_latest_notification_when_queue_is_not_empty def test_return_latest_notification_when_queue_is_not_empty
request_xml = <<-XML post epp_poll_path, params: { frame: request_req_xml },
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<epp xmlns="#{Xsd::Schema.filename(for_prefix: 'epp-ee', for_version: '1.0')}">
<command>
<poll op="req"/>
</command>
</epp>
XML
post epp_poll_path, params: { frame: request_xml },
headers: { 'HTTP_COOKIE' => 'session=api_bestnames' } headers: { 'HTTP_COOKIE' => 'session=api_bestnames' }
xml_doc = Nokogiri::XML(response.body) xml_doc = Nokogiri::XML(response.body)
@ -30,16 +22,8 @@ class EppPollTest < EppTestCase
version = Version::DomainVersion.last version = Version::DomainVersion.last
@notification.update(attached_obj_type: 'DomainVersion', attached_obj_id: version.id) @notification.update(attached_obj_type: 'DomainVersion', attached_obj_id: version.id)
request_xml = <<-XML
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<epp xmlns="#{Xsd::Schema.filename(for_prefix: 'epp-ee', for_version: '1.0')}">
<command>
<poll op="req"/>
</command>
</epp>
XML
assert_nothing_raised do assert_nothing_raised do
post epp_poll_path, params: { frame: request_xml }, post epp_poll_path, params: { frame: request_req_xml },
headers: { 'HTTP_COOKIE' => 'session=api_bestnames' } headers: { 'HTTP_COOKIE' => 'session=api_bestnames' }
end end
@ -54,19 +38,11 @@ class EppPollTest < EppTestCase
def test_return_action_data_when_present def test_return_action_data_when_present
@notification.update!(action: actions(:contact_update)) @notification.update!(action: actions(:contact_update))
request_xml = <<-XML post epp_poll_path, params: { frame: request_req_xml },
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<epp xmlns="#{Xsd::Schema.filename(for_prefix: 'epp-ee', for_version: '1.0')}">
<command>
<poll op="req"/>
</command>
</epp>
XML
post epp_poll_path, params: { frame: request_xml },
headers: { 'HTTP_COOKIE' => 'session=api_bestnames' } headers: { 'HTTP_COOKIE' => 'session=api_bestnames' }
xml_doc = Nokogiri::XML(response.body) xml_doc = Nokogiri::XML(response.body)
namespace = Xsd::Schema.filename(for_prefix: 'changePoll') namespace = Xsd::Schema.filename(for_prefix: 'changePoll', for_version: '1.0')
assert_equal 'update', xml_doc.xpath('//changePoll:operation', 'changePoll' => namespace).text assert_equal 'update', xml_doc.xpath('//changePoll:operation', 'changePoll' => namespace).text
assert_equal Time.zone.parse('2010-07-05').utc.xmlschema, assert_equal Time.zone.parse('2010-07-05').utc.xmlschema,
xml_doc.xpath('//changePoll:date', 'changePoll' => namespace).text xml_doc.xpath('//changePoll:date', 'changePoll' => namespace).text
@ -76,18 +52,34 @@ class EppPollTest < EppTestCase
'changePoll' => namespace).text 'changePoll' => namespace).text
end end
def test_return_notifcation_with_bulk_action_data
bulk_action = actions(:contacts_update_bulk_action)
@notification.update!(action: bulk_action,
attached_obj_id: bulk_action.id,
attached_obj_type: 'BulkAction')
post epp_poll_path, params: { frame: request_req_xml },
headers: { 'HTTP_COOKIE' => 'session=api_bestnames' }
xml_doc = Nokogiri::XML(response.body)
namespace = Xsd::Schema.filename(for_prefix: 'changePoll', for_version: '1.0')
assert_equal 2, xml_doc.xpath('//contact:cd', contact: xml_schema).size
assert_epp_response :completed_successfully_ack_to_dequeue
assert_equal 'bulk_update', xml_doc.xpath('//changePoll:operation',
'changePoll' => namespace).text
assert_equal @notification.action.id.to_s, xml_doc.xpath('//changePoll:svTRID',
'changePoll' => namespace).text
assert_equal 'Registrant User', xml_doc.xpath('//changePoll:who',
'changePoll' => namespace).text
assert_equal 'Auto-update according to official data',
xml_doc.xpath('//changePoll:reason', 'changePoll' => namespace).text
end
def test_no_notifications def test_no_notifications
registrars(:bestnames).notifications.delete_all(:delete_all) registrars(:bestnames).notifications.delete_all(:delete_all)
request_xml = <<-XML post epp_poll_path, params: { frame: request_req_xml },
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<epp xmlns="#{Xsd::Schema.filename(for_prefix: 'epp-ee', for_version: '1.0')}">
<command>
<poll op="req"/>
</command>
</epp>
XML
post epp_poll_path, params: { frame: request_xml },
headers: { 'HTTP_COOKIE' => 'session=api_bestnames' } headers: { 'HTTP_COOKIE' => 'session=api_bestnames' }
assert_epp_response :completed_successfully_no_messages assert_epp_response :completed_successfully_no_messages
@ -151,7 +143,16 @@ class EppPollTest < EppTestCase
end end
def test_anonymous_user_cannot_access def test_anonymous_user_cannot_access
request_xml = <<-XML post '/epp/command/poll', params: { frame: request_req_xml },
headers: { 'HTTP_COOKIE' => 'session=non-existent' }
assert_epp_response :authorization_error
end
private
def request_req_xml
<<-XML
<?xml version="1.0" encoding="UTF-8" standalone="no"?> <?xml version="1.0" encoding="UTF-8" standalone="no"?>
<epp xmlns="#{Xsd::Schema.filename(for_prefix: 'epp-ee', for_version: '1.0')}"> <epp xmlns="#{Xsd::Schema.filename(for_prefix: 'epp-ee', for_version: '1.0')}">
<command> <command>
@ -159,10 +160,9 @@ class EppPollTest < EppTestCase
</command> </command>
</epp> </epp>
XML XML
end
post '/epp/command/poll', params: { frame: request_xml }, def xml_schema
headers: { 'HTTP_COOKIE' => 'session=non-existent' } Xsd::Schema.filename(for_prefix: 'contact-ee', for_version: '1.1')
assert_epp_response :authorization_error
end end
end end

View file

@ -7,34 +7,40 @@ class RegistrantUserCreationTest < ActiveSupport::TestCase
first_name: 'JOHN', first_name: 'JOHN',
last_name: 'SMITH' last_name: 'SMITH'
} }
assert_difference 'RegistrantUser.count' do
RegistrantUser.find_or_create_by_api_data(user_data) RegistrantUser.find_or_create_by_api_data(user_data)
end
user = User.find_by(registrant_ident: 'EE-37710100070') user = User.find_by(registrant_ident: 'EE-37710100070')
assert_equal('JOHN SMITH', user.username) assert_equal('JOHN SMITH', user.username)
end end
def test_find_or_create_by_api_data_creates_a_user_with_original_name def test_find_or_create_by_api_data_updates_a_user_with_existing_ident
user_data = { user_data = {
ident: '37710100070', ident: '1234',
country_code: 'US',
first_name: 'John', first_name: 'John',
last_name: 'Smith' last_name: 'Smith',
} }
assert_no_difference 'RegistrantUser.count' do
RegistrantUser.find_or_create_by_api_data(user_data) RegistrantUser.find_or_create_by_api_data(user_data)
end
user = User.find_by(registrant_ident: 'EE-37710100070') user = User.find_by(registrant_ident: 'US-1234')
assert_equal('John Smith', user.username) assert_equal('John Smith', user.username)
end end
def test_updates_related_contacts_name_if_differs_from_e_identity def test_updates_related_contacts_name_if_different_from_e_identity
contact = contacts(:john) registrars = [registrars(:bestnames), registrars(:goodnames)]
contact.update(ident: '39708290276', ident_country_code: 'EE') contacts = [contacts(:john), contacts(:william), contacts(:identical_to_william)]
contacts.each do |c|
c.update(ident: '39708290276', ident_country_code: 'EE')
end
user_data = { user_data = {
ident: '39708290276', ident: '39708290276',
first_name: 'John', first_name: 'John',
last_name: 'Doe' last_name: 'Doe',
} }
RegistrantUser.find_or_create_by_api_data(user_data) RegistrantUser.find_or_create_by_api_data(user_data)
@ -42,7 +48,28 @@ class RegistrantUserCreationTest < ActiveSupport::TestCase
user = User.find_by(registrant_ident: 'EE-39708290276') user = User.find_by(registrant_ident: 'EE-39708290276')
assert_equal('John Doe', user.username) assert_equal('John Doe', user.username)
contact.reload contacts.each do |c|
assert_equal user.username, contact.name c.reload
assert_equal user.username, c.name
assert user.actions.find_by(operation: :update, contact_id: c.id)
end
bulk_action = BulkAction.find_by(user_id: user.id, operation: :bulk_update)
assert_equal 2, bulk_action.subactions.size
registrars.each do |r|
notification = r.notifications.unread.order('created_at DESC').take
if r == registrars(:bestnames)
assert_equal '2 contacts have been updated by registrant', notification.text
assert_equal 'BulkAction', notification.attached_obj_type
assert_equal bulk_action.id, notification.attached_obj_id
assert_equal bulk_action.id, notification.action_id
else
assert_equal 'Contact william-002 has been updated by registrant', notification.text
refute notification.action_id
refute notification.attached_obj_id
refute notification.attached_obj_type
end
end
end end
end end