Merge remote-tracking branch 'origin/master' into repp-domains

This commit is contained in:
Karl Erik Õunapuu 2021-03-22 14:05:42 +02:00
commit e2f377f1c2
No known key found for this signature in database
GPG key ID: C9DD647298A34764
187 changed files with 1934 additions and 2552 deletions

View file

@ -32,6 +32,9 @@ plugins:
rubocop: rubocop:
enabled: true enabled: true
channel: rubocop-0-74 channel: rubocop-0-74
checks:
Rubocop/Style/ClassAndModuleChildren:
enabled: false
checks: checks:
method-lines: method-lines:
config: config:

View file

@ -1,3 +1,44 @@
18.03.2021
* Added tests for renew and domain status management [#1886](https://github.com/internetee/registry/pull/1886)
12.03.2021
* Removed old classnames from notifications [#1878](https://github.com/internetee/registry/pull/1878)
* improved test coverage [#1860](https://github.com/internetee/registry/pull/1860)
* added test for whois record delete [#1811](https://github.com/internetee/registry/pull/1811)
11.03.2021
* Account activity in registrar and REPP now return balance with each record [#1819](https://github.com/internetee/registry/issues/1819)
* Fixed CookieOverflow error with large authentication service keys [#1879](https://github.com/internetee/registry/pull/1879)
10.03.2021
* Registrant API returns full contact details for admin contacts [#1876](https://github.com/internetee/registry/pull/1876)
09.03.2021
* AWS message id saving over API [#1877](https://github.com/internetee/registry/pull/1877)
* Enabled Zeitwerk autoloader [#1872](https://github.com/internetee/registry/issues/1872)
04.03.2021
* Removed old registrant portal from the project [#1826](https://github.com/internetee/registry/issues/1826)
03.03.2021
* Email notification is sent in case of pendingupdate expiry [#897](https://github.com/internetee/registry/issues/897)
26.02.2021
* Domain delete is not affected by updateProhibited [#1844](https://github.com/internetee/registry/issues/1844)
* Registrant API fix for handling eidas personal identificators [#1864](https://github.com/internetee/registry/pull/1864)
23.02.2021
* UpdateProhibited status affects bulk actions in REPP [#1818](https://github.com/internetee/registry/issues/1818)
* Registrant api domain request now excludes tech only domains by default [#1836](https://github.com/internetee/registry/pull/1836)
22.02.2021
* serverDeleteProhibited prohibts delete action [#1849](https://github.com/internetee/registry/issues/1849)
19.02.2021
* Update prohibited staatus is kept after renew [#1843](https://github.com/internetee/registry/issues/1843)
* Fixed clientHold and serverManualInzone status conflict issue [#1845](https://github.com/internetee/registry/issues/1845)
* Replacing registrant object with another that has the same ident data set does not require registrant verification [#1852](https://github.com/internetee/registry/issues/1852)
11.02.2021 11.02.2021
* Poll messages on locking and unlocking a domain [#1828](https://github.com/internetee/registry/issues/1828) * Poll messages on locking and unlocking a domain [#1828](https://github.com/internetee/registry/issues/1828)
* Registrar's prefix is now checked and added to contact id for info and check requests [#1832](https://github.com/internetee/registry/issues/1832) * Registrar's prefix is now checked and added to contact id for info and check requests [#1832](https://github.com/internetee/registry/issues/1832)

View file

@ -2,12 +2,12 @@ module Admin
class ContactVersionsController < BaseController class ContactVersionsController < BaseController
include ObjectVersionsHelper include ObjectVersionsHelper
load_and_authorize_resource load_and_authorize_resource class: Version::ContactVersion
def index def index
params[:q] ||= {} params[:q] ||= {}
@q = ContactVersion.search(params[:q]) @q = Version::ContactVersion.search(params[:q])
@versions = @q.result.page(params[:page]) @versions = @q.result.page(params[:page])
search_params = params[:q].deep_dup search_params = params[:q].deep_dup
@ -23,7 +23,7 @@ module Admin
end end
end end
versions = ContactVersion.includes(:item).where(whereS).order(created_at: :desc, id: :desc) versions = Version::ContactVersion.includes(:item).where(whereS).order(created_at: :desc, id: :desc)
@q = versions.search(params[:q]) @q = versions.search(params[:q])
@versions = @q.result.page(params[:page]) @versions = @q.result.page(params[:page])
@versions = @versions.per(params[:results_per_page]) if params[:results_per_page].to_i.positive? @versions = @versions.per(params[:results_per_page]) if params[:results_per_page].to_i.positive?
@ -32,8 +32,8 @@ module Admin
def show def show
per_page = 7 per_page = 7
@version = ContactVersion.find(params[:id]) @version = Version::ContactVersion.find(params[:id])
@versions = ContactVersion.where(item_id: @version.item_id).order(created_at: :desc, id: :desc) @versions = Version::ContactVersion.where(item_id: @version.item_id).order(created_at: :desc, id: :desc)
@versions_map = @versions.all.map(&:id) @versions_map = @versions.all.map(&:id)
# what we do is calc amount of results until needed version # what we do is calc amount of results until needed version
@ -49,7 +49,7 @@ module Admin
end end
def search def search
render json: ContactVersion.search_by_query(params[:q]) render json: Version::ContactVersion.search_by_query(params[:q])
end end
def create_where_string(key, value) def create_where_string(key, value)

View file

@ -2,12 +2,12 @@ module Admin
class DomainVersionsController < BaseController class DomainVersionsController < BaseController
include ObjectVersionsHelper include ObjectVersionsHelper
load_and_authorize_resource load_and_authorize_resource class: Version::DomainVersion
def index def index
params[:q] ||= {} params[:q] ||= {}
@q = DomainVersion.includes(:item).search(params[:q]) @q = Version::DomainVersion.includes(:item).search(params[:q])
@versions = @q.result.page(params[:page]) @versions = @q.result.page(params[:page])
search_params = params[:q].deep_dup search_params = params[:q].deep_dup
@ -40,7 +40,7 @@ module Admin
whereS += " AND object->>'registrar_id' IN (#{registrars.map { |r| "'#{r.id.to_s}'" }.join ','})" if registrars.present? whereS += " AND object->>'registrar_id' IN (#{registrars.map { |r| "'#{r.id.to_s}'" }.join ','})" if registrars.present?
whereS += " AND 1=0" if registrars == [] whereS += " AND 1=0" if registrars == []
versions = DomainVersion.includes(:item).where(whereS).order(created_at: :desc, id: :desc) versions = Version::DomainVersion.includes(:item).where(whereS).order(created_at: :desc, id: :desc)
@q = versions.search(params[:q]) @q = versions.search(params[:q])
@versions = @q.result.page(params[:page]) @versions = @q.result.page(params[:page])
@versions = @versions.per(params[:results_per_page]) if params[:results_per_page].to_i.positive? @versions = @versions.per(params[:results_per_page]) if params[:results_per_page].to_i.positive?
@ -50,8 +50,8 @@ module Admin
def show def show
per_page = 7 per_page = 7
@version = DomainVersion.find(params[:id]) @version = Version::DomainVersion.find(params[:id])
@versions = DomainVersion.where(item_id: @version.item_id).order(created_at: :desc, id: :desc) @versions = Version::DomainVersion.where(item_id: @version.item_id).order(created_at: :desc, id: :desc)
@versions_map = @versions.all.map(&:id) @versions_map = @versions.all.map(&:id)
# what we do is calc amount of results until needed version # what we do is calc amount of results until needed version
@ -67,7 +67,7 @@ module Admin
end end
def search def search
render json: DomainVersion.search_by_query(params[:q]) render json: Version::DomainVersion.search_by_query(params[:q])
end end
def create_where_string(key, value) def create_where_string(key, value)

View file

@ -23,14 +23,15 @@ module Api
record = ContactRequest.find_by(id: id) record = ContactRequest.find_by(id: id)
return :not_found unless record return :not_found unless record
record.update_status(contact_request_params) record.update_record(contact_request_params)
render json: record, status: :ok render json: record, status: :ok
rescue StandardError rescue StandardError
head :bad_request head :bad_request
end end
def contact_request_params def contact_request_params
params.require(:contact_request).permit(:email, :whois_record_id, :name, :status, :ip) params.require(:contact_request).permit(:email, :whois_record_id, :name, :status, :ip,
:message_id)
end end
end end
end end

View file

@ -19,6 +19,9 @@ module Api
token = create_token(user) token = create_token(user)
if token if token
msg = "Bearer for #{eid_params[:first_name]} #{eid_params[:last_name]} " \
"(#{eid_params[:ident]}) - '#{token[:access_token]}'"
ToStdout.msg(msg) unless Rails.env.production?
render json: token render json: token
else else
render json: { errors: [{ base: ['Cannot create generate session token'] }] } render json: { errors: [{ base: ['Cannot create generate session token'] }] }
@ -37,7 +40,7 @@ module Api
obj.require(key) obj.require(key)
end end
params.permit(required_params) params.permit(required_params + [:country_code])
end end
def create_token(user) def create_token(user)

View file

@ -4,6 +4,8 @@ module Api
module V1 module V1
module Registrant module Registrant
class DomainsController < ::Api::V1::Registrant::BaseController class DomainsController < ::Api::V1::Registrant::BaseController
before_action :set_tech_flag, only: [:show]
def index def index
limit = params[:limit] || 200 limit = params[:limit] || 200
offset = params[:offset] || 0 offset = params[:offset] || 0
@ -25,7 +27,8 @@ module Api
serializer.to_json serializer.to_json
end end
render json: { count: domains.count, domains: serialized_domains } render json: { total: current_user_domains_total_count, count: domains.count,
domains: serialized_domains }
end end
def show def show
@ -41,10 +44,22 @@ module Api
private private
def current_user_domains def set_tech_flag
current_registrant_user.domains # current_user_domains scope depends on tech flag
# However, if it's not present, tech contact can not see specific domain entry at all.
params.merge!(tech: 'true')
end
def current_user_domains_total_count
current_registrant_user.domains.count
rescue CompanyRegister::NotAvailableError rescue CompanyRegister::NotAvailableError
current_registrant_user.direct_domains current_registrant_user.direct_domains.count
end
def current_user_domains
current_registrant_user.domains(admin: params[:tech] != 'true')
rescue CompanyRegister::NotAvailableError
current_registrant_user.direct_domains(admin: params[:tech] != 'true')
end end
end end
end end

View file

@ -12,9 +12,11 @@ module Epp
@notification = current_user.unread_notifications.order('created_at DESC').take @notification = current_user.unread_notifications.order('created_at DESC').take
render_epp_response 'epp/poll/poll_no_messages' and return unless @notification render_epp_response 'epp/poll/poll_no_messages' and return unless @notification
if @notification.attached_obj_type && @notification.attached_obj_id if @notification.attached_obj_type && @notification.attached_obj_id
begin begin
@object = Object.const_get(@notification.attached_obj_type).find(@notification.attached_obj_id) @object = object_by_type(@notification.attached_obj_type)
.find(@notification.attached_obj_id)
rescue => problem rescue => problem
# the data model might be inconsistent; or ... # the data model might be inconsistent; or ...
# this could happen if the registrar does not dequeue messages, and then the domain was deleted # this could happen if the registrar does not dequeue messages, and then the domain was deleted
@ -31,6 +33,12 @@ module Epp
render_epp_response 'epp/poll/poll_req' render_epp_response 'epp/poll/poll_req'
end end
def object_by_type(object_type)
Object.const_get(object_type)
rescue NameError
Object.const_get("Version::#{object_type}")
end
def ack_poll def ack_poll
@notification = current_user.unread_notifications.find_by(id: params[:parsed_frame].css('poll').first['msgID']) @notification = current_user.unread_notifications.find_by(id: params[:parsed_frame].css('poll').first['msgID'])

View file

@ -1,115 +0,0 @@
class Registrant::ContactsController < RegistrantController
helper_method :domain
helper_method :fax_enabled?
helper_method :domain_filter_params
skip_authorization_check only: %i[edit update]
before_action :set_contact, only: [:show]
def show
@requester_contact = Contact.find_by(ident: current_registrant_user.ident)
authorize! :read, @contact
end
def edit
@contact = current_user_contacts.find(params[:id])
end
def update
@contact = current_user_contacts.find(params[:id])
@contact.attributes = contact_params
response = update_contact_via_api(@contact.uuid)
updated = response.is_a?(Net::HTTPSuccess)
if updated
redirect_to registrant_domain_contact_url(domain, @contact), notice: t('.updated')
else
parsed_response = JSON.parse(response.body, symbolize_names: true)
@errors = parsed_response[:errors]
render :edit
end
end
private
def set_contact
id = params[:id]
contact = domain.contacts.find_by(id: id) || current_user_contacts.find_by(id: id)
contact ||= Contact.find_by(id: id, ident: domain.registrant.ident)
@contact = contact
end
def domain
current_user_domains.find(params[:domain_id])
end
def contact_params
permitted = %i[
name
email
phone
]
permitted << :fax if fax_enabled?
permitted += %i[street zip city state country_code] if Contact.address_processing?
params.require(:contact).permit(*permitted)
end
def access_token
uri = URI.parse("#{ENV['registrant_api_base_url']}/api/v1/registrant/auth/eid")
request = Net::HTTP::Post.new(uri)
request.form_data = access_token_request_params
response = Net::HTTP.start(uri.hostname, uri.port, use_ssl: (uri.scheme == 'https')) do |http|
http.request(request)
end
json_doc = JSON.parse(response.body, symbolize_names: true)
json_doc[:access_token]
end
def access_token_request_params
{ ident: current_registrant_user.ident,
first_name: current_registrant_user.first_name,
last_name: current_registrant_user.last_name }
end
def fax_enabled?
ENV['fax_enabled'] == 'true'
end
def contact_update_api_params
params = contact_params
params = normalize_address_attributes_for_api(params) if Contact.address_processing?
params
end
def normalize_address_attributes_for_api(params)
normalized = params
address_parts = {}
Contact.address_attribute_names.each do |attr|
attr = attr.to_sym
address_parts[attr] = params[attr]
normalized.delete(attr)
end
normalized[:address] = address_parts
normalized
end
def update_contact_via_api(uuid)
uri = URI.parse("#{ENV['registrant_api_base_url']}/api/v1/registrant/contacts/#{uuid}")
request = Net::HTTP::Patch.new(uri)
request['Authorization'] = "Bearer #{access_token}"
request['Content-type'] = 'application/json'
request.body = contact_update_api_params.to_json
Net::HTTP.start(uri.hostname, uri.port, use_ssl: (uri.scheme == 'https')) do |http|
http.request(request)
end
end
def domain_filter_params
params.permit(:domain_filter)
end
end

View file

@ -1,44 +0,0 @@
class Registrant::DomainDeleteConfirmsController < RegistrantController
skip_before_action :authenticate_registrant_user!, only: [:show, :update]
skip_authorization_check only: [:show, :update]
def show
return if params[:confirmed] || params[:rejected]
@domain = Domain.find(params[:id])
@domain = nil unless @domain.registrant_delete_confirmable?(params[:token])
end
def update
@domain = Domain.find(params[:id])
unless @domain.registrant_delete_confirmable?(params[:token])
flash[:alert] = t(:registrant_domain_verification_failed)
return render 'show'
end
@registrant_verification = RegistrantVerification.new(domain_id: @domain.id,
verification_token: params[:token])
initiator = current_registrant_user ? current_registrant_user.username :
t(:user_not_authenticated)
confirmed = params[:confirmed] ? true : false
action = if confirmed
@registrant_verification.domain_registrant_delete_confirm!("email link #{initiator}")
else
@registrant_verification.domain_registrant_delete_reject!("email link #{initiator}")
end
fail_msg = t("registrant_domain_delete_#{confirmed ? 'confirmed' : 'rejected'}_failed".to_sym)
success_msg = t("registrant_domain_verification_#{confirmed ? 'confirmed' : 'rejected'}".to_sym)
flash[:alert] = action ? success_msg : fail_msg
(render 'show' && return) unless action
if confirmed
redirect_to registrant_domain_delete_confirm_path(@domain.id, confirmed: true)
else
redirect_to registrant_domain_delete_confirm_path(@domain.id, rejected: true)
end
end
end

View file

@ -1,44 +0,0 @@
class Registrant::DomainUpdateConfirmsController < RegistrantController
skip_before_action :authenticate_registrant_user!, only: %i[show update]
skip_authorization_check only: %i[show update]
def show
return if params[:confirmed] || params[:rejected]
@domain = Domain.find(params[:id])
@domain = nil unless @domain.registrant_update_confirmable?(params[:token])
end
def update
@domain = Domain.find(params[:id])
unless @domain.registrant_update_confirmable?(params[:token])
flash[:alert] = t(:registrant_domain_verification_failed)
return render 'show'
end
@registrant_verification = RegistrantVerification.new(domain_id: @domain.id,
verification_token: params[:token])
initiator = current_registrant_user ? current_registrant_user.username :
t(:user_not_authenticated)
if params[:rejected]
if @registrant_verification.domain_registrant_change_reject!("email link, #{initiator}")
flash[:notice] = t(:registrant_domain_verification_rejected)
redirect_to registrant_domain_update_confirm_path(@domain.id, rejected: true)
else
flash[:alert] = t(:registrant_domain_verification_rejected_failed)
return render 'show'
end
elsif params[:confirmed]
if @registrant_verification.domain_registrant_change_confirm!("email link, #{initiator}")
Dispute.close_by_domain(@domain.name) if @domain.disputed?
flash[:notice] = t(:registrant_domain_verification_confirmed)
redirect_to registrant_domain_update_confirm_path(@domain.id, confirmed: true)
else
flash[:alert] = t(:registrant_domain_verification_confirmed_failed)
return render 'show'
end
end
end
end

View file

@ -1,79 +0,0 @@
class Registrant::DomainsController < RegistrantController
def index
authorize! :view, :registrant_domains
params[:q] ||= {}
normalize_search_parameters do
@q = current_user_domains.search(search_params)
end
domains = @q.result
respond_to do |format|
format.html do
@domains = domains.page(params[:page])
domains_per_page = params[:results_per_page].to_i
@domains = @domains.per(domains_per_page) if domains_per_page.positive?
end
format.csv do
raw_csv = @q.result.to_csv
send_data raw_csv, filename: 'domains.csv', type: "#{Mime[:csv]}; charset=utf-8"
end
format.pdf do
view = ActionView::Base.new(ActionController::Base.view_paths, domains: domains)
raw_html = view.render(file: 'registrant/domains/list_pdf', layout: false)
raw_pdf = domains.pdf(raw_html)
send_data raw_pdf, filename: 'domains.pdf'
end
end
end
def show
@domain = current_user_domains.find(params[:id])
authorize! :read, @domain
end
def confirmation
authorize! :view, :registrant_domains
domain = current_user_domains.find(params[:id])
if (domain.statuses.include?(DomainStatus::PENDING_UPDATE) ||
domain.statuses.include?(DomainStatus::PENDING_DELETE_CONFIRMATION)) &&
domain.pending_json.present?
@domain = domain
@confirmation_url = confirmation_url(domain)
else
flash[:warning] = I18n.t('available_verification_url_not_found')
redirect_to registrant_domain_path(domain)
end
end
private
def normalize_search_parameters
ca_cache = params[:q][:valid_to_lteq]
begin
end_time = params[:q][:valid_to_lteq].try(:to_date)
params[:q][:valid_to_lteq] = end_time.try(:end_of_day)
rescue
logger.warn('Invalid date')
end
yield
params[:q][:valid_to_lteq] = ca_cache
end
def confirmation_url(domain)
if domain.statuses.include?(DomainStatus::PENDING_UPDATE)
registrant_domain_update_confirm_url(token: domain.registrant_verification_token)
elsif domain.statuses.include?(DomainStatus::PENDING_DELETE_CONFIRMATION)
registrant_domain_delete_confirm_url(token: domain.registrant_verification_token)
end
end
def search_params
params.require(:q).permit(:name_matches, :registrant_ident_eq, :valid_to_gteq, :valid_to_lteq,
:results_per_page)
end
end

View file

@ -1,6 +0,0 @@
class Registrant::RegistrarsController < RegistrantController
def show
@registrar = Registrar.find(params[:id])
authorize! :read, @registrar
end
end

View file

@ -1,17 +0,0 @@
class Registrant::SessionsController < Devise::SessionsController
layout 'registrant/application'
private
def after_sign_in_path_for(_resource_or_scope)
registrant_root_path
end
def after_sign_out_path_for(_resource_or_scope)
new_registrant_user_session_path
end
def user_for_paper_trail
current_registrant_user.present? ? current_registrant_user.id_role_username : 'anonymous'
end
end

View file

@ -17,7 +17,7 @@ class Registrar
private private
def check_ip_restriction def check_ip_restriction
ip_restriction = Authorization::RestrictedIP.new(request.ip) ip_restriction = Authorization::RestrictedIp.new(request.ip)
allowed = ip_restriction.can_access_registrar_area?(current_registrar_user.registrar) allowed = ip_restriction.can_access_registrar_area?(current_registrar_user.registrar)
return if allowed return if allowed

View file

@ -49,7 +49,7 @@ class Registrar
domain_presenters << ::DomainPresenter.new(domain: domain, view: view_context) domain_presenters << ::DomainPresenter.new(domain: domain, view: view_context)
end end
raw_csv = Registrar::DomainListCSVPresenter.new(domains: domain_presenters, raw_csv = Registrar::DomainListCsvPresenter.new(domains: domain_presenters,
view: view_context).to_s view: view_context).to_s
filename = "Domains_#{l(Time.zone.now, format: :filename)}.csv" filename = "Domains_#{l(Time.zone.now, format: :filename)}.csv"
send_data raw_csv, filename: filename, type: "#{Mime[:csv]}; charset=utf-8" send_data raw_csv, filename: filename, type: "#{Mime[:csv]}; charset=utf-8"

View file

@ -72,7 +72,7 @@ class Registrar
end end
def check_ip_restriction def check_ip_restriction
ip_restriction = Authorization::RestrictedIP.new(request.ip) ip_restriction = Authorization::RestrictedIp.new(request.ip)
allowed = ip_restriction.can_access_registrar_area_sign_in_page? allowed = ip_restriction.can_access_registrar_area_sign_in_page?
return if allowed return if allowed

View file

@ -6,8 +6,28 @@ module Repp
def balance def balance
resp = { balance: current_user.registrar.cash_account.balance, resp = { balance: current_user.registrar.cash_account.balance,
currency: current_user.registrar.cash_account.currency } currency: current_user.registrar.cash_account.currency }
resp[:transactions] = activities if params[:detailed] == 'true'
render_success(data: resp) render_success(data: resp)
end end
def activities
arr = []
registrar_activities.each do |a|
arr << { created_at: a.created_at, description: a.description,
type: a.activity_type == 'add_credit' ? 'credit' : 'debit',
sum: a.sum, balance: a.new_balance }
end
arr
end
def registrar_activities
activities = current_user.registrar.cash_account.activities.order(created_at: :desc)
activities = activities.where('created_at >= ?', params[:from]) if params[:from]
activities = activities.where('created_at <= ?', params[:until]) if params[:until]
activities
end
end end
end end
end end

View file

@ -14,7 +14,7 @@ module Sso
# rubocop:disable Style/AndOr # rubocop:disable Style/AndOr
def callback(user, registrar: true) def callback(user, registrar: true)
session[:omniauth_hash] = user_hash session[:omniauth_hash] = user_hash.delete_if { |key, _| key == 'credentials' }
(show_error(registrar: registrar) and return) unless user (show_error(registrar: registrar) and return) unless user
flash[:notice] = t(:signed_in_successfully) flash[:notice] = t(:signed_in_successfully)

View file

@ -19,6 +19,11 @@ module Actions
return return
end end
if contact.delete_prohibited?
contact.errors.add(:statuses, :delete_prohibited)
return
end
commit commit
end end

View file

@ -11,6 +11,7 @@ module Actions
end end
def call def call
domain.is_renewal = true
if !domain.renewable? || domain.invalid? if !domain.renewable? || domain.invalid?
domain.add_renew_epp_errors domain.add_renew_epp_errors
false false

View file

@ -17,7 +17,9 @@ module Domains
def notify_pending_update def notify_pending_update
RegistrantChangeMailer.expired(domain: domain, RegistrantChangeMailer.expired(domain: domain,
registrar: domain.registrar, registrar: domain.registrar,
registrant: domain.registrant).deliver_later registrant: domain.registrant,
send_to: [domain.new_registrant_email,
domain.registrant.email]).deliver_later
end end
def notify_pending_delete def notify_pending_delete

View file

@ -1,20 +0,0 @@
class RegistrantChangeExpiredEmailJob < Que::Job
def run(domain_id)
domain = Domain.find(domain_id)
log(domain)
RegistrantChangeMailer.expired(domain: domain,
registrar: domain.registrar,
registrant: domain.registrant).deliver_now
end
private
def log(domain)
message = "Send RegistrantChangeMailer#expired email for domain #{domain.name} (##{domain.id}) to #{domain.new_registrant_email}"
logger.info(message)
end
def logger
Rails.logger
end
end

View file

@ -6,10 +6,6 @@ class ApplicationMailer < ActionMailer::Base
token = domain.registrant_verification_token token = domain.registrant_verification_token
base_url = ENV['registrant_portal_verifications_base_url'] base_url = ENV['registrant_portal_verifications_base_url']
url = registrant_domain_delete_confirm_url(domain, token: token) if method == 'delete'
url ||= registrant_domain_update_confirm_url(domain, token: token)
return url if base_url.blank?
"#{base_url}/confirmation/#{domain.name_puny}/#{method}/#{token}" "#{base_url}/confirmation/#{domain.name_puny}/#{method}/#{token}"
end end
end end

View file

@ -1,4 +1,4 @@
class PunycodeInterceptor class Interceptors::PunycodeInterceptor
class << self class << self
def delivering_email(message) def delivering_email(message)
message.from = encode_addresses_as_punycode(message.from) message.from = encode_addresses_as_punycode(message.from)
@ -21,4 +21,4 @@ class PunycodeInterceptor
SimpleIDN.to_ascii(domain_part) SimpleIDN.to_ascii(domain_part)
end end
end end
end end

View file

@ -38,13 +38,13 @@ class RegistrantChangeMailer < ApplicationMailer
mail(to: domain.new_registrant_email, subject: subject) mail(to: domain.new_registrant_email, subject: subject)
end end
def expired(domain:, registrar:, registrant:) def expired(domain:, registrar:, registrant:, send_to:)
@domain = DomainPresenter.new(domain: domain, view: view_context) @domain = DomainPresenter.new(domain: domain, view: view_context)
@registrar = RegistrarPresenter.new(registrar: registrar, view: view_context) @registrar = RegistrarPresenter.new(registrar: registrar, view: view_context)
@registrant = RegistrantPresenter.new(registrant: registrant, view: view_context) @registrant = RegistrantPresenter.new(registrant: registrant, view: view_context)
subject = default_i18n_subject(domain_name: domain.name) subject = default_i18n_subject(domain_name: domain.name)
mail(to: domain.new_registrant_email, subject: subject) mail(to: send_to, subject: subject)
end end
private private

View file

@ -89,8 +89,8 @@ class Ability
can :manage, BlockedDomain can :manage, BlockedDomain
can :manage, ReservedDomain can :manage, ReservedDomain
can :manage, DNS::Zone can :manage, DNS::Zone
can :manage, DomainVersion can :manage, Version::DomainVersion
can :manage, ContactVersion can :manage, Version::ContactVersion
can :manage, Billing::Price can :manage, Billing::Price
can :manage, User can :manage, User
can :manage, ApiUser can :manage, ApiUser

View file

@ -13,6 +13,9 @@ class AccountActivity < ApplicationRecord
def update_balance def update_balance
account.balance += sum account.balance += sum
account.save account.save
self.new_balance = account.balance
save
end end
class << self class << self

View file

@ -1,5 +1,5 @@
class Action < ApplicationRecord class Action < ApplicationRecord
has_paper_trail versions: { class_name: 'ActionVersion' } has_paper_trail versions: { class_name: 'Version::ActionVersion' }
belongs_to :user belongs_to :user
belongs_to :contact belongs_to :contact

View file

@ -1,5 +1,5 @@
module Authorization module Authorization
class RestrictedIP class RestrictedIp
def initialize(ip) def initialize(ip)
@ip = ip @ip = ip
end end

View file

@ -1,6 +1,6 @@
module Billing module Billing
class Price < ApplicationRecord class Price < ApplicationRecord
include Concerns::Billing::Price::Expirable include Billing::Price::Expirable
include Versions include Versions
belongs_to :zone, class_name: 'DNS::Zone', required: true belongs_to :zone, class_name: 'DNS::Zone', required: true

View file

@ -1,4 +1,4 @@
module Concerns::Billing::Price::Expirable module Billing::Price::Expirable
extend ActiveSupport::Concern extend ActiveSupport::Concern
class_methods do class_methods do

View file

@ -1,68 +1,64 @@
module Concerns module Contact::Archivable
module Contact extend ActiveSupport::Concern
module Archivable
extend ActiveSupport::Concern
class_methods do class_methods do
def archivable def archivable
unlinked.find_each.select(&:archivable?) unlinked.find_each.select(&:archivable?)
end
end
def archivable?(post: false)
inactive = inactive?
log("Found archivable contact id(#{id}), code (#{code})") if inactive && !post
inactive
end
def archive(verified: false, notify: true, extra_log: false)
unless verified
raise 'Contact cannot be archived' unless archivable?(post: true)
end
notify_registrar_about_archivation if notify
write_to_registrar_log if extra_log
destroy!
end
private
def notify_registrar_about_archivation
registrar.notifications.create!(
text: I18n.t('contact_has_been_archived',
contact_code: code, orphan_months: Setting.orphans_contacts_in_months)
)
end
def inactive?
if DomainVersion.contact_unlinked_more_than?(contact_id: id, period: inactivity_period)
return true
end
DomainVersion.was_contact_linked?(id) ? false : created_at <= inactivity_period.ago
end
def inactivity_period
Setting.orphans_contacts_in_months.months
end
def log(msg)
@log ||= Logger.new(STDOUT)
@log.info(msg)
end
def write_to_registrar_log
registrar_name = registrar.accounting_customer_code
archive_path = ENV['contact_archivation_log_file_dir']
registrar_log_path = "#{archive_path}/#{registrar_name}.txt"
FileUtils.mkdir_p(archive_path) unless Dir.exist?(archive_path)
f = File.new(registrar_log_path, 'a+')
f.write("#{code}\n")
f.close
end
end end
end end
def archivable?(post: false)
inactive = inactive?
log("Found archivable contact id(#{id}), code (#{code})") if inactive && !post
inactive
end
def archive(verified: false, notify: true, extra_log: false)
unless verified
raise 'Contact cannot be archived' unless archivable?(post: true)
end
notify_registrar_about_archivation if notify
write_to_registrar_log if extra_log
destroy!
end
private
def notify_registrar_about_archivation
registrar.notifications.create!(
text: I18n.t('contact_has_been_archived',
contact_code: code, orphan_months: Setting.orphans_contacts_in_months)
)
end
def inactive?
if Version::DomainVersion.contact_unlinked_more_than?(contact_id: id, period: inactivity_period)
return true
end
Version::DomainVersion.was_contact_linked?(id) ? false : created_at <= inactivity_period.ago
end
def inactivity_period
Setting.orphans_contacts_in_months.months
end
def log(msg)
@log ||= Logger.new(STDOUT)
@log.info(msg)
end
def write_to_registrar_log
registrar_name = registrar.accounting_customer_code
archive_path = ENV['contact_archivation_log_file_dir']
registrar_log_path = "#{archive_path}/#{registrar_name}.txt"
FileUtils.mkdir_p(archive_path) unless Dir.exist?(archive_path)
f = File.new(registrar_log_path, 'a+')
f.write("#{code}\n")
f.close
end
end end

View file

@ -1,26 +1,22 @@
module Concerns module Contact::Disclosable
module Contact extend ActiveSupport::Concern
module Disclosable
extend ActiveSupport::Concern
class_methods do class_methods do
attr_accessor :disclosable_attributes attr_accessor :disclosable_attributes
end end
included do included do
self.disclosable_attributes = %w[name email] self.disclosable_attributes = %w[name email]
validate :validate_disclosed_attributes validate :validate_disclosed_attributes
end end
private private
def validate_disclosed_attributes def validate_disclosed_attributes
return if disclosed_attributes.empty? return if disclosed_attributes.empty?
has_undisclosable_attributes = (disclosed_attributes - self.class.disclosable_attributes) has_undisclosable_attributes = (disclosed_attributes - self.class.disclosable_attributes)
.any? .any?
errors.add(:disclosed_attributes, :invalid) if has_undisclosable_attributes errors.add(:disclosed_attributes, :invalid) if has_undisclosable_attributes
end
end
end end
end end

View file

@ -1,4 +1,4 @@
module Concerns::Contact::Identical module Contact::Identical
extend ActiveSupport::Concern extend ActiveSupport::Concern
IDENTIFIABLE_ATTRIBUTES = %w[ IDENTIFIABLE_ATTRIBUTES = %w[

View file

@ -1,4 +1,4 @@
module Concerns::Domain::Activatable module Domain::Activatable
extend ActiveSupport::Concern extend ActiveSupport::Concern
def active? def active?

View file

@ -1,17 +1,13 @@
module Concerns module Domain::BulkUpdatable
module Domain extend ActiveSupport::Concern
module BulkUpdatable
extend ActiveSupport::Concern
def bulk_update_prohibited? def bulk_update_prohibited?
discarded? || statuses_blocks_update? discarded? || statuses_blocks_update?
end end
def statuses_blocks_update? def statuses_blocks_update?
prohibited_array = [DomainStatus::SERVER_UPDATE_PROHIBITED, prohibited_array = [DomainStatus::SERVER_UPDATE_PROHIBITED,
DomainStatus::CLIENT_UPDATE_PROHIBITED] DomainStatus::CLIENT_UPDATE_PROHIBITED]
prohibited_array.any? { |block_status| statuses.include?(block_status) } prohibited_array.any? { |block_status| statuses.include?(block_status) }
end
end
end end
end end

View file

@ -1,6 +1,12 @@
module Concerns::Domain::Deletable module Domain::Deletable
extend ActiveSupport::Concern extend ActiveSupport::Concern
DELETE_STATUSES = [
DomainStatus::PENDING_DELETE_CONFIRMATION,
DomainStatus::PENDING_DELETE,
DomainStatus::FORCE_DELETE,
].freeze
private private
def delete_later def delete_later

View file

@ -1,4 +1,4 @@
module Concerns::Domain::Discardable module Domain::Discardable
extend ActiveSupport::Concern extend ActiveSupport::Concern
def keep def keep

View file

@ -1,44 +1,40 @@
# frozen_string_literal: true # frozen_string_literal: true
module Concerns module Domain::Disputable
module Domain extend ActiveSupport::Concern
module Disputable
extend ActiveSupport::Concern
included do included do
validate :validate_disputed validate :validate_disputed
end end
def mark_as_disputed def mark_as_disputed
statuses.push(DomainStatus::DISPUTED) unless statuses.include?(DomainStatus::DISPUTED) statuses.push(DomainStatus::DISPUTED) unless statuses.include?(DomainStatus::DISPUTED)
save save
end end
def unmark_as_disputed def unmark_as_disputed
statuses.delete_if { |status| status == DomainStatus::DISPUTED } statuses.delete_if { |status| status == DomainStatus::DISPUTED }
save save
end end
def in_disputed_list? def in_disputed_list?
@in_disputed_list ||= Dispute.active.find_by(domain_name: name).present? @in_disputed_list ||= Dispute.active.find_by(domain_name: name).present?
end end
def disputed? def disputed?
Dispute.active.where(domain_name: name).any? Dispute.active.where(domain_name: name).any?
end end
def validate_disputed def validate_disputed
return if persisted? || !in_disputed_list? return if persisted? || !in_disputed_list?
if reserved_pw.blank? if reserved_pw.blank?
errors.add(:base, :required_parameter_missing_disputed) errors.add(:base, :required_parameter_missing_disputed)
return false return false
end
return if Dispute.valid_auth?(name, reserved_pw)
errors.add(:base, :invalid_auth_information_reserved)
end
end end
return if Dispute.valid_auth?(name, reserved_pw)
errors.add(:base, :invalid_auth_information_reserved)
end end
end end

View file

@ -1,4 +1,4 @@
module Concerns::Domain::Expirable module Domain::Expirable
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do

View file

@ -1,4 +1,4 @@
module Concerns::Domain::ForceDelete # rubocop:disable Metrics/ModuleLength module Domain::ForceDelete # rubocop:disable Metrics/ModuleLength
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do
@ -11,6 +11,11 @@ module Concerns::Domain::ForceDelete # rubocop:disable Metrics/ModuleLength
lambda { lambda {
where("(force_delete_data->>'contact_notification_sent_date') is null") where("(force_delete_data->>'contact_notification_sent_date') is null")
} }
HOLD_STATUSES = [
DomainStatus::SERVER_HOLD,
DomainStatus::CLIENT_HOLD,
].freeze
end end
class_methods do class_methods do
@ -19,6 +24,10 @@ module Concerns::Domain::ForceDelete # rubocop:disable Metrics/ModuleLength
end end
end end
def hold_status?
HOLD_STATUSES.any? { |status| statuses.include? status }
end
def notification_template(explicit: nil) def notification_template(explicit: nil)
reason = explicit&.downcase reason = explicit&.downcase
return reason if %w[invalid_email invalid_phone].include?(reason) return reason if %w[invalid_email invalid_phone].include?(reason)

View file

@ -1,63 +1,59 @@
module Concerns module Domain::RegistryLockable
module Domain extend ActiveSupport::Concern
module RegistryLockable
extend ActiveSupport::Concern
def apply_registry_lock def apply_registry_lock
return unless registry_lockable? return unless registry_lockable?
return if locked_by_registrant? return if locked_by_registrant?
transaction do transaction do
statuses << DomainStatus::SERVER_UPDATE_PROHIBITED statuses << DomainStatus::SERVER_UPDATE_PROHIBITED
statuses << DomainStatus::SERVER_DELETE_PROHIBITED statuses << DomainStatus::SERVER_DELETE_PROHIBITED
statuses << DomainStatus::SERVER_TRANSFER_PROHIBITED statuses << DomainStatus::SERVER_TRANSFER_PROHIBITED
self.locked_by_registrant_at = Time.zone.now self.locked_by_registrant_at = Time.zone.now
alert_registrar_lock_changes!(lock: true) alert_registrar_lock_changes!(lock: true)
save! save!
end
end
def registry_lockable?
(statuses & [DomainStatus::PENDING_DELETE_CONFIRMATION,
DomainStatus::PENDING_CREATE, DomainStatus::PENDING_UPDATE,
DomainStatus::PENDING_DELETE, DomainStatus::PENDING_RENEW,
DomainStatus::PENDING_TRANSFER, DomainStatus::FORCE_DELETE]).empty?
end
def locked_by_registrant?
return false unless locked_by_registrant_at
lock_statuses = [DomainStatus::SERVER_UPDATE_PROHIBITED,
DomainStatus::SERVER_DELETE_PROHIBITED,
DomainStatus::SERVER_TRANSFER_PROHIBITED]
(statuses & lock_statuses).count == 3
end
def remove_registry_lock
return unless locked_by_registrant?
transaction do
statuses.delete(DomainStatus::SERVER_UPDATE_PROHIBITED)
statuses.delete(DomainStatus::SERVER_DELETE_PROHIBITED)
statuses.delete(DomainStatus::SERVER_TRANSFER_PROHIBITED)
self.locked_by_registrant_at = nil
alert_registrar_lock_changes!(lock: false)
save!
end
end
def alert_registrar_lock_changes!(lock: true)
translation = lock ? 'locked' : 'unlocked'
registrar.notifications.create!(
text: I18n.t("notifications.texts.registrar_#{translation}",
domain_name: name),
attached_obj_id: name,
attached_obj_type: self.class.name
)
end
end end
end end
def registry_lockable?
(statuses & [DomainStatus::PENDING_DELETE_CONFIRMATION,
DomainStatus::PENDING_CREATE, DomainStatus::PENDING_UPDATE,
DomainStatus::PENDING_DELETE, DomainStatus::PENDING_RENEW,
DomainStatus::PENDING_TRANSFER, DomainStatus::FORCE_DELETE]).empty?
end
def locked_by_registrant?
return false unless locked_by_registrant_at
lock_statuses = [DomainStatus::SERVER_UPDATE_PROHIBITED,
DomainStatus::SERVER_DELETE_PROHIBITED,
DomainStatus::SERVER_TRANSFER_PROHIBITED]
(statuses & lock_statuses).count == 3
end
def remove_registry_lock
return unless locked_by_registrant?
transaction do
statuses.delete(DomainStatus::SERVER_UPDATE_PROHIBITED)
statuses.delete(DomainStatus::SERVER_DELETE_PROHIBITED)
statuses.delete(DomainStatus::SERVER_TRANSFER_PROHIBITED)
self.locked_by_registrant_at = nil
alert_registrar_lock_changes!(lock: false)
save!
end
end
def alert_registrar_lock_changes!(lock: true)
translation = lock ? 'locked' : 'unlocked'
registrar.notifications.create!(
text: I18n.t("notifications.texts.registrar_#{translation}",
domain_name: name),
attached_obj_id: name,
attached_obj_type: self.class.name
)
end
end end

View file

@ -1,57 +1,53 @@
module Concerns module Domain::Releasable
module Domain extend ActiveSupport::Concern
module Releasable
extend ActiveSupport::Concern
class_methods do class_methods do
def release_domains def release_domains
releasable_domains.each do |domain| releasable_domains.each do |domain|
domain.release domain.release
yield domain if block_given? yield domain if block_given?
end
end
private
def releasable_domains
if release_to_auction
where('(delete_date <= ? OR force_delete_date <= ?)' \
' AND ? != ALL(coalesce(statuses, array[]::varchar[]))',
Time.zone.today,
Time.zone.today,
DomainStatus::SERVER_DELETE_PROHIBITED)
else
where('(delete_date <= ? OR force_delete_date <= ?)' \
' AND ? != ALL(coalesce(statuses, array[]::varchar[])) AND' \
' ? != ALL(COALESCE(statuses, array[]::varchar[]))',
Time.zone.today,
Time.zone.today,
DomainStatus::SERVER_DELETE_PROHIBITED,
DomainStatus::DELETE_CANDIDATE)
end
end
end end
end
included do private
class_attribute :release_to_auction
self.release_to_auction = ENV['release_domains_to_auction'] == 'true'
end
def release def releasable_domains
if release_to_auction if release_to_auction
ToStdout.msg 'Destroying domain' where('(delete_date <= ? OR force_delete_date <= ?)' \
destroy! ' AND ? != ALL(coalesce(statuses, array[]::varchar[]))',
ToStdout.msg "Checking if domain_name is auctionable: #{domain_name.auctionable?}" Time.zone.today,
domain_name.sell_at_auction if domain_name.auctionable? Time.zone.today,
DomainStatus::SERVER_DELETE_PROHIBITED)
ToStdout.msg 'Sending registrar notification' else
registrar.notifications.create!(text: "#{I18n.t(:domain_deleted)}: #{name}", where('(delete_date <= ? OR force_delete_date <= ?)' \
attached_obj_id: id, ' AND ? != ALL(coalesce(statuses, array[]::varchar[])) AND' \
attached_obj_type: self.class) ' ? != ALL(COALESCE(statuses, array[]::varchar[]))',
else Time.zone.today,
discard Time.zone.today,
end DomainStatus::SERVER_DELETE_PROHIBITED,
DomainStatus::DELETE_CANDIDATE)
end end
end end
end end
included do
class_attribute :release_to_auction
self.release_to_auction = ENV['release_domains_to_auction'] == 'true'
end
def release
if release_to_auction
ToStdout.msg 'Destroying domain'
destroy!
ToStdout.msg "Checking if domain_name is auctionable: #{domain_name.auctionable?}"
domain_name.sell_at_auction if domain_name.auctionable?
ToStdout.msg 'Sending registrar notification'
registrar.notifications.create!(text: "#{I18n.t(:domain_deleted)}: #{name}",
attached_obj_id: id,
attached_obj_type: self.class)
else
discard
end
end
end end

View file

@ -1,4 +1,4 @@
module Concerns::Domain::Transferable module Domain::Transferable
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do
@ -31,7 +31,9 @@ module Concerns::Domain::Transferable
DomainStatus::PENDING_TRANSFER, DomainStatus::PENDING_TRANSFER,
DomainStatus::FORCE_DELETE, DomainStatus::FORCE_DELETE,
DomainStatus::SERVER_TRANSFER_PROHIBITED, DomainStatus::SERVER_TRANSFER_PROHIBITED,
DomainStatus::CLIENT_TRANSFER_PROHIBITED DomainStatus::CLIENT_TRANSFER_PROHIBITED,
DomainStatus::SERVER_UPDATE_PROHIBITED,
DomainStatus::CLIENT_UPDATE_PROHIBITED,
]).empty? ]).empty?
end end

View file

@ -1,95 +1,93 @@
module Concerns module EmailVerifable
module EmailVerifable extend ActiveSupport::Concern
extend ActiveSupport::Concern
def email_verification def email_verification
@email_verification ||= EmailAddressVerification.find_or_create_by(email: unicode_email, @email_verification ||= EmailAddressVerification.find_or_create_by(email: unicode_email,
domain: domain(email)) domain: domain(email))
end end
def billing_email_verification def billing_email_verification
return unless attribute_names.include?('billing_email') return unless attribute_names.include?('billing_email')
@billing_email_verification ||= EmailAddressVerification @billing_email_verification ||= EmailAddressVerification
.find_or_create_by(email: unicode_billing_email, .find_or_create_by(email: unicode_billing_email,
domain: domain(billing_email)) domain: domain(billing_email))
end end
def email_verification_failed? def email_verification_failed?
email_verification&.failed? email_verification&.failed?
end end
class_methods do
def domain(email)
Mail::Address.new(email).domain&.downcase || 'not_found'
rescue Mail::Field::IncompleteParseError
'not_found'
end
def local(email)
Mail::Address.new(email).local&.downcase || email
rescue Mail::Field::IncompleteParseError
email
end
def punycode_to_unicode(email)
return email if domain(email) == 'not_found'
local = local(email)
domain = SimpleIDN.to_unicode(domain(email))
"#{local}@#{domain}"&.downcase
end
def unicode_to_punycode(email)
return email if domain(email) == 'not_found'
local = local(email)
domain = SimpleIDN.to_ascii(domain(email))
"#{local}@#{domain}"&.downcase
end
end
def unicode_billing_email
self.class.punycode_to_unicode(billing_email)
end
def unicode_email
self.class.punycode_to_unicode(email)
end
class_methods do
def domain(email) def domain(email)
SimpleIDN.to_unicode(self.class.domain(email)) Mail::Address.new(email).domain&.downcase || 'not_found'
rescue Mail::Field::IncompleteParseError
'not_found'
end
def local(email)
Mail::Address.new(email).local&.downcase || email
rescue Mail::Field::IncompleteParseError
email
end end
def punycode_to_unicode(email) def punycode_to_unicode(email)
self.class.punycode_to_unicode(email) return email if domain(email) == 'not_found'
local = local(email)
domain = SimpleIDN.to_unicode(domain(email))
"#{local}@#{domain}"&.downcase
end end
def correct_email_format def unicode_to_punycode(email)
return if email.blank? return email if domain(email) == 'not_found'
result = email_verification.verify local = local(email)
process_result(result: result, field: :email) domain = SimpleIDN.to_ascii(domain(email))
"#{local}@#{domain}"&.downcase
end end
def correct_billing_email_format
return if email.blank?
result = billing_email_verification.verify
process_result(result: result, field: :billing_email)
end
# rubocop:disable Metrics/LineLength
def process_result(result:, field:)
case result[:errors].keys.first
when :smtp
errors.add(field, I18n.t('activerecord.errors.models.contact.attributes.email.email_smtp_check_error'))
when :mx
errors.add(field, I18n.t('activerecord.errors.models.contact.attributes.email.email_mx_check_error'))
when :regex
errors.add(field, I18n.t('activerecord.errors.models.contact.attributes.email.email_regex_check_error'))
end
end
# rubocop:enable Metrics/LineLength
end end
def unicode_billing_email
self.class.punycode_to_unicode(billing_email)
end
def unicode_email
self.class.punycode_to_unicode(email)
end
def domain(email)
SimpleIDN.to_unicode(self.class.domain(email))
end
def punycode_to_unicode(email)
self.class.punycode_to_unicode(email)
end
def correct_email_format
return if email.blank?
result = email_verification.verify
process_result(result: result, field: :email)
end
def correct_billing_email_format
return if email.blank?
result = billing_email_verification.verify
process_result(result: result, field: :billing_email)
end
# rubocop:disable Metrics/LineLength
def process_result(result:, field:)
case result[:errors].keys.first
when :smtp
errors.add(field, I18n.t('activerecord.errors.models.contact.attributes.email.email_smtp_check_error'))
when :mx
errors.add(field, I18n.t('activerecord.errors.models.contact.attributes.email.email_mx_check_error'))
when :regex
errors.add(field, I18n.t('activerecord.errors.models.contact.attributes.email.email_regex_check_error'))
end
end
# rubocop:enable Metrics/LineLength
end end

View file

@ -1,34 +1,30 @@
module Concerns module Invoice::BookKeeping
module Invoice extend ActiveSupport::Concern
module BookKeeping
extend ActiveSupport::Concern
def as_directo_json def as_directo_json
invoice = ActiveSupport::JSON.decode(ActiveSupport::JSON.encode(self)) invoice = ActiveSupport::JSON.decode(ActiveSupport::JSON.encode(self))
invoice['customer'] = compose_directo_customer invoice['customer'] = compose_directo_customer
invoice['issue_date'] = issue_date.strftime('%Y-%m-%d') invoice['issue_date'] = issue_date.strftime('%Y-%m-%d')
invoice['transaction_date'] = account_activity invoice['transaction_date'] = account_activity
.bank_transaction&.paid_at&.strftime('%Y-%m-%d') .bank_transaction&.paid_at&.strftime('%Y-%m-%d')
invoice['language'] = buyer.language == 'en' ? 'ENG' : '' invoice['language'] = buyer.language == 'en' ? 'ENG' : ''
invoice['invoice_lines'] = compose_directo_product invoice['invoice_lines'] = compose_directo_product
invoice invoice
end end
def compose_directo_product def compose_directo_product
[{ 'product_id': Setting.directo_receipt_product_name, 'description': order, [{ 'product_id': Setting.directo_receipt_product_name, 'description': order,
'quantity': 1, 'price': ActionController::Base.helpers.number_with_precision( 'quantity': 1, 'price': ActionController::Base.helpers.number_with_precision(
subtotal, precision: 2, separator: '.' subtotal, precision: 2, separator: '.'
) }].as_json ) }].as_json
end end
def compose_directo_customer def compose_directo_customer
{ {
'code': buyer.accounting_customer_code, 'code': buyer.accounting_customer_code,
'destination': buyer_country_code, 'destination': buyer_country_code,
'vat_reg_no': buyer_vat_no, 'vat_reg_no': buyer_vat_no,
}.as_json }.as_json
end
end
end end
end end

View file

@ -1,28 +1,24 @@
module Concerns module Invoice::Cancellable
module Invoice extend ActiveSupport::Concern
module Cancellable
extend ActiveSupport::Concern
included do included do
scope :non_cancelled, -> { where(cancelled_at: nil) } scope :non_cancelled, -> { where(cancelled_at: nil) }
end end
def cancellable? def cancellable?
unpaid? && not_cancelled? unpaid? && not_cancelled?
end end
def cancel def cancel
raise 'Invoice cannot be cancelled' unless cancellable? raise 'Invoice cannot be cancelled' unless cancellable?
update!(cancelled_at: Time.zone.now) update!(cancelled_at: Time.zone.now)
end end
def cancelled? def cancelled?
cancelled_at cancelled_at
end end
def not_cancelled? def not_cancelled?
!cancelled? !cancelled?
end
end
end end
end end

View file

@ -1,28 +1,24 @@
module Concerns module Invoice::Payable
module Invoice extend ActiveSupport::Concern
module Payable
extend ActiveSupport::Concern
included do included do
scope :unpaid, -> { where('id NOT IN (SELECT invoice_id FROM account_activities WHERE' \ scope :unpaid, -> { where('id NOT IN (SELECT invoice_id FROM account_activities WHERE' \
' invoice_id IS NOT NULL)') } ' invoice_id IS NOT NULL)') }
end end
def payable? def payable?
unpaid? && not_cancelled? unpaid? && not_cancelled?
end end
def paid? def paid?
account_activity account_activity
end end
def receipt_date def receipt_date
account_activity.created_at.to_date account_activity.created_at.to_date
end end
def unpaid? def unpaid?
!paid? !paid?
end
end
end end
end end

View file

@ -1,128 +1,124 @@
module Concerns module Registrar::BookKeeping
module Registrar extend ActiveSupport::Concern
module BookKeeping
extend ActiveSupport::Concern
DOMAIN_TO_PRODUCT = { 'ee': '01EE', 'com.ee': '02COM', 'pri.ee': '03PRI', DOMAIN_TO_PRODUCT = { 'ee': '01EE', 'com.ee': '02COM', 'pri.ee': '03PRI',
'fie.ee': '04FIE', 'med.ee': '05MED' }.freeze 'fie.ee': '04FIE', 'med.ee': '05MED' }.freeze
def monthly_summary(month:) def monthly_summary(month:)
activities = monthly_activites(month) activities = monthly_activites(month)
return unless activities.any? return unless activities.any?
invoice = { invoice = {
'number': 1, 'number': 1,
'customer': compose_directo_customer, 'customer': compose_directo_customer,
'language': language == 'en' ? 'ENG' : '', 'currency': activities.first.currency, 'language': language == 'en' ? 'ENG' : '', 'currency': activities.first.currency,
'date': month.end_of_month.strftime('%Y-%m-%d') 'date': month.end_of_month.strftime('%Y-%m-%d')
}.as_json }.as_json
invoice['invoice_lines'] = prepare_invoice_lines(month: month, activities: activities) invoice['invoice_lines'] = prepare_invoice_lines(month: month, activities: activities)
invoice invoice
end end
def prepare_invoice_lines(month:, activities:) def prepare_invoice_lines(month:, activities:)
lines = [] lines = []
lines << { 'description': title_for_summary(month) } lines << { 'description': title_for_summary(month) }
activities.each do |activity| activities.each do |activity|
fetch_invoice_lines(activity, lines) fetch_invoice_lines(activity, lines)
end end
lines << prepayment_for_all(lines) lines << prepayment_for_all(lines)
lines.as_json lines.as_json
end end
def title_for_summary(date) def title_for_summary(date)
I18n.with_locale(language == 'en' ? 'en' : 'et') do I18n.with_locale(language == 'en' ? 'en' : 'et') do
I18n.t('registrar.monthly_summary_title', date: I18n.l(date, format: '%B %Y')) I18n.t('registrar.monthly_summary_title', date: I18n.l(date, format: '%B %Y'))
end
end
def fetch_invoice_lines(activity, lines)
price = load_price(activity)
if price.duration.include? 'year'
price.duration.to_i.times do |duration|
lines << new_monthly_invoice_line(activity: activity, duration: duration + 1).as_json
end
else
lines << new_monthly_invoice_line(activity: activity).as_json
end
end
def monthly_activites(month)
AccountActivity.where(account_id: account_ids)
.where(created_at: month.beginning_of_month..month.end_of_month)
.where(activity_type: [AccountActivity::CREATE, AccountActivity::RENEW])
end
def new_monthly_invoice_line(activity:, duration: nil)
price = load_price(activity)
line = {
'product_id': DOMAIN_TO_PRODUCT[price.zone_name.to_sym],
'quantity': 1,
'unit': language == 'en' ? 'pc' : 'tk',
}
finalize_invoice_line(line, price: price, duration: duration, activity: activity)
end
def finalize_invoice_line(line, price:, activity:, duration:)
yearly = price.duration.include?('year')
line['price'] = yearly ? (price.price.amount / price.duration.to_i) : price.price.amount
line['description'] = description_in_language(price: price, yearly: yearly)
if duration.present?
add_product_timeframe(line: line, activity: activity, duration: duration) if duration > 1
end
line
end
def add_product_timeframe(line:, activity:, duration:)
create_time = activity.created_at
line['start_date'] = (create_time + (duration - 1).year).end_of_month.strftime('%Y-%m-%d')
line['end_date'] = (create_time + (duration - 1).year + 1).end_of_month.strftime('%Y-%m-%d')
end
def description_in_language(price:, yearly:)
timeframe_string = yearly ? 'yearly' : 'monthly'
locale_string = "registrar.invoice_#{timeframe_string}_product_description"
I18n.with_locale(language == 'en' ? 'en' : 'et') do
I18n.t(locale_string, tld: ".#{price.zone_name}", length: price.duration.to_i)
end
end
def prepayment_for_all(lines)
total = 0
en = language == 'en'
lines.each { |l| total += l['quantity'].to_f * l['price'].to_f }
{
'product_id': Setting.directo_receipt_product_name,
'description': en ? 'Domains prepayment' : 'Domeenide ettemaks',
'quantity': -1,
'price': total,
'unit': en ? 'pc' : 'tk',
}
end
def compose_directo_customer
{
'code': accounting_customer_code,
'destination': address_country_code,
'vat_reg_no': vat_no,
}.as_json
end
def load_price(account_activity)
@pricelists ||= {}
return @pricelists[account_activity.price_id] if @pricelists.key? account_activity.price_id
@pricelists[account_activity.price_id] = account_activity.price
end
end end
end end
def fetch_invoice_lines(activity, lines)
price = load_price(activity)
if price.duration.include? 'year'
price.duration.to_i.times do |duration|
lines << new_monthly_invoice_line(activity: activity, duration: duration + 1).as_json
end
else
lines << new_monthly_invoice_line(activity: activity).as_json
end
end
def monthly_activites(month)
AccountActivity.where(account_id: account_ids)
.where(created_at: month.beginning_of_month..month.end_of_month)
.where(activity_type: [AccountActivity::CREATE, AccountActivity::RENEW])
end
def new_monthly_invoice_line(activity:, duration: nil)
price = load_price(activity)
line = {
'product_id': DOMAIN_TO_PRODUCT[price.zone_name.to_sym],
'quantity': 1,
'unit': language == 'en' ? 'pc' : 'tk',
}
finalize_invoice_line(line, price: price, duration: duration, activity: activity)
end
def finalize_invoice_line(line, price:, activity:, duration:)
yearly = price.duration.include?('year')
line['price'] = yearly ? (price.price.amount / price.duration.to_i) : price.price.amount
line['description'] = description_in_language(price: price, yearly: yearly)
if duration.present?
add_product_timeframe(line: line, activity: activity, duration: duration) if duration > 1
end
line
end
def add_product_timeframe(line:, activity:, duration:)
create_time = activity.created_at
line['start_date'] = (create_time + (duration - 1).year).end_of_month.strftime('%Y-%m-%d')
line['end_date'] = (create_time + (duration - 1).year + 1).end_of_month.strftime('%Y-%m-%d')
end
def description_in_language(price:, yearly:)
timeframe_string = yearly ? 'yearly' : 'monthly'
locale_string = "registrar.invoice_#{timeframe_string}_product_description"
I18n.with_locale(language == 'en' ? 'en' : 'et') do
I18n.t(locale_string, tld: ".#{price.zone_name}", length: price.duration.to_i)
end
end
def prepayment_for_all(lines)
total = 0
en = language == 'en'
lines.each { |l| total += l['quantity'].to_f * l['price'].to_f }
{
'product_id': Setting.directo_receipt_product_name,
'description': en ? 'Domains prepayment' : 'Domeenide ettemaks',
'quantity': -1,
'price': total,
'unit': en ? 'pc' : 'tk',
}
end
def compose_directo_customer
{
'code': accounting_customer_code,
'destination': address_country_code,
'vat_reg_no': vat_no,
}.as_json
end
def load_price(account_activity)
@pricelists ||= {}
return @pricelists[account_activity.price_id] if @pricelists.key? account_activity.price_id
@pricelists[account_activity.price_id] = account_activity.price
end
end end

View file

@ -1,16 +1,12 @@
module Concerns module Registrar::LegalDoc
module Registrar extend ActiveSupport::Concern
module LegalDoc
extend ActiveSupport::Concern
def legaldoc_mandatory? def legaldoc_mandatory?
!legaldoc_not_mandatory? !legaldoc_not_mandatory?
end end
def legaldoc_not_mandatory? def legaldoc_not_mandatory?
setting = Setting.legal_document_is_mandatory setting = Setting.legal_document_is_mandatory
legaldoc_optout || !setting legaldoc_optout || !setting
end
end
end end
end end

View file

@ -7,10 +7,10 @@ module Versions
attr_accessor :version_loader attr_accessor :version_loader
if WITH_CHILDREN.include?(model_name.name) if WITH_CHILDREN.include?(model_name.name)
has_paper_trail versions: { class_name: "#{model_name}Version" }, has_paper_trail versions: { class_name: "Version::#{model_name}Version" },
meta: { children: :children_log } meta: { children: :children_log }
else else
has_paper_trail versions: { class_name: "#{model_name}Version" } has_paper_trail versions: { class_name: "Version::#{model_name}Version" }
end end
# add creator and updator # add creator and updator

View file

@ -1,74 +1,70 @@
module Concerns module Zone::WhoisQueryable
module Zone extend ActiveSupport::Concern
module WhoisQueryable
extend ActiveSupport::Concern
included do included do
after_save :update_whois_record, if: :subzone? after_save :update_whois_record, if: :subzone?
after_destroy :update_whois_record after_destroy :update_whois_record
end end
def subzone? def subzone?
origin.include? '.' origin.include? '.'
end end
def update_whois_record def update_whois_record
UpdateWhoisRecordJob.enqueue origin, 'zone' UpdateWhoisRecordJob.enqueue origin, 'zone'
end end
def generate_data def generate_data
wr = Whois::Record.find_or_initialize_by(name: origin) wr = Whois::Record.find_or_initialize_by(name: origin)
wr.json = generate_json wr.json = generate_json
wr.save wr.save
end end
def generate_json def generate_json
data = {}.with_indifferent_access data = {}.with_indifferent_access
[domain_vars, registrar_vars, registrant_vars].each do |h| [domain_vars, registrar_vars, registrant_vars].each do |h|
data.merge!(h) data.merge!(h)
end
data
end
# Take note - since this concern only used to zone whois queries, dnssec keys are set to
# empty array
def domain_vars
{ disclaimer: Setting.registry_whois_disclaimer, name: origin,
registered: created_at.try(:to_s, :iso8601), status: ['ok (paid and in zone)'],
changed: updated_at.try(:to_s, :iso8601), email: Setting.registry_email,
admin_contacts: [contact_vars], tech_contacts: [contact_vars],
nameservers: nameserver_vars, dnssec_keys: [], dnssec_changed: nil }
end
def registrar_vars
{ registrar: Setting.registry_juridical_name, registrar_website: Setting.registry_url,
registrar_phone: Setting.registry_phone }
end
def registrant_vars
{ registrant: Setting.registry_juridical_name, registrant_reg_no: Setting.registry_reg_no,
registrant_ident_country_code: Setting.registry_country_code, registrant_kind: 'org',
registrant_disclosed_attributes: %w[name email] }
end
def contact_vars
{ name: Setting.registry_invoice_contact, email: Setting.registry_email,
disclosed_attributes: %w[name email] }
end
def nameserver_vars
vars = []
return vars unless ns_records
parsed_ns = ns_records.gsub("\r", '').gsub("\n", '')
parsed_ns.split("#{origin}. IN NS ").each do |ns|
ns.delete_suffix! '.'
vars << ns if ns.match? Nameserver::HOSTNAME_REGEXP
end
vars
end
end end
data
end
# Take note - since this concern only used to zone whois queries, dnssec keys are set to
# empty array
def domain_vars
{ disclaimer: Setting.registry_whois_disclaimer, name: origin,
registered: created_at.try(:to_s, :iso8601), status: ['ok (paid and in zone)'],
changed: updated_at.try(:to_s, :iso8601), email: Setting.registry_email,
admin_contacts: [contact_vars], tech_contacts: [contact_vars],
nameservers: nameserver_vars, dnssec_keys: [], dnssec_changed: nil }
end
def registrar_vars
{ registrar: Setting.registry_juridical_name, registrar_website: Setting.registry_url,
registrar_phone: Setting.registry_phone }
end
def registrant_vars
{ registrant: Setting.registry_juridical_name, registrant_reg_no: Setting.registry_reg_no,
registrant_ident_country_code: Setting.registry_country_code, registrant_kind: 'org',
registrant_disclosed_attributes: %w[name email] }
end
def contact_vars
{ name: Setting.registry_invoice_contact, email: Setting.registry_email,
disclosed_attributes: %w[name email] }
end
def nameserver_vars
vars = []
return vars unless ns_records
parsed_ns = ns_records.gsub("\r", '').gsub("\n", '')
parsed_ns.split("#{origin}. IN NS ").each do |ns|
ns.delete_suffix! '.'
vars << ns if ns.match? Nameserver::HOSTNAME_REGEXP
end
vars
end end
end end

View file

@ -4,11 +4,11 @@ class Contact < ApplicationRecord
include Versions # version/contact_version.rb include Versions # version/contact_version.rb
include EppErrors include EppErrors
include UserEvents include UserEvents
include Concerns::Contact::Transferable include Contact::Transferable
include Concerns::Contact::Identical include Contact::Identical
include Concerns::Contact::Disclosable include Contact::Disclosable
include Concerns::Contact::Archivable include Contact::Archivable
include Concerns::EmailVerifable include EmailVerifable
belongs_to :original, class_name: self.name belongs_to :original, class_name: self.name
belongs_to :registrar, required: true belongs_to :registrar, required: true

View file

@ -1,4 +1,4 @@
module Concerns::Contact::Transferable module Contact::Transferable
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do
@ -9,7 +9,7 @@ module Concerns::Contact::Transferable
def transfer(new_registrar) def transfer(new_registrar)
return identical(new_registrar) if identical(new_registrar) return identical(new_registrar) if identical(new_registrar)
new_contact = self.dup new_contact = dup
new_contact.registrar = new_registrar new_contact.registrar = new_registrar
new_contact.original = self new_contact.original = self
new_contact.code = nil new_contact.code = nil

View file

@ -24,9 +24,10 @@ class ContactRequest < ApplicationRecord
contact_request contact_request
end end
def update_status(params) def update_record(params)
self.status = params['status'] self.status = params['status'] if params['status']
self.ip_address = params['ip'] self.ip_address = params['ip'] if params['ip']
self.message_id = params['ip'] if params['message_id']
save! save!
end end

View file

@ -5,7 +5,7 @@ module DNS
validates :origin, :ttl, :refresh, :retry, :expire, :minimum_ttl, :email, :master_nameserver, presence: true validates :origin, :ttl, :refresh, :retry, :expire, :minimum_ttl, :email, :master_nameserver, presence: true
validates :ttl, :refresh, :retry, :expire, :minimum_ttl, numericality: { only_integer: true } validates :ttl, :refresh, :retry, :expire, :minimum_ttl, numericality: { only_integer: true }
validates :origin, uniqueness: true validates :origin, uniqueness: true
include Concerns::Zone::WhoisQueryable include ::Zone::WhoisQueryable
before_destroy do before_destroy do
throw(:abort) if used? throw(:abort) if used?

View file

@ -1,16 +1,16 @@
class Domain < ApplicationRecord class Domain < ApplicationRecord
include UserEvents include UserEvents
include Versions # version/domain_version.rb include Versions # version/domain_version.rb
include Concerns::Domain::Expirable include Domain::Expirable
include Concerns::Domain::Activatable include Domain::Activatable
include Concerns::Domain::ForceDelete include Domain::ForceDelete
include Concerns::Domain::Discardable include Domain::Discardable
include Concerns::Domain::Deletable include Domain::Deletable
include Concerns::Domain::Transferable include Domain::Transferable
include Concerns::Domain::RegistryLockable include Domain::RegistryLockable
include Concerns::Domain::Releasable include Domain::Releasable
include Concerns::Domain::Disputable include Domain::Disputable
include Concerns::Domain::BulkUpdatable include Domain::BulkUpdatable
attr_accessor :roles attr_accessor :roles
@ -107,13 +107,13 @@ class Domain < ApplicationRecord
validate :status_is_consistant validate :status_is_consistant
def status_is_consistant def status_is_consistant
has_error = (statuses.include?(DomainStatus::SERVER_HOLD) && statuses.include?(DomainStatus::SERVER_MANUAL_INZONE)) has_error = (hold_status? && statuses.include?(DomainStatus::SERVER_MANUAL_INZONE))
unless has_error unless has_error
if (statuses & [DomainStatus::PENDING_DELETE_CONFIRMATION, DomainStatus::PENDING_DELETE, DomainStatus::FORCE_DELETE]).any? if (statuses & DELETE_STATUSES).any?
has_error = statuses.include? DomainStatus::SERVER_DELETE_PROHIBITED has_error = statuses.include? DomainStatus::SERVER_DELETE_PROHIBITED
end
end end
errors.add(:domains, I18n.t(:object_status_prohibits_operation)) if has_error end
errors.add(:domains, I18n.t(:object_status_prohibits_operation)) if has_error
end end
attr_accessor :is_admin attr_accessor :is_admin
@ -198,6 +198,23 @@ class Domain < ApplicationRecord
Setting.nameserver_required Setting.nameserver_required
end end
def registrant_user_admin_registrant_domains(registrant_user)
companies = Contact.registrant_user_company_contacts(registrant_user)
from(
"(#{registrant_user_administered_domains(registrant_user).to_sql} UNION " \
"#{registrant_user_company_registrant(companies).to_sql} UNION " \
"#{registrant_user_domains_company(companies, except_tech: true).to_sql}) AS domains"
)
end
def registrant_user_direct_admin_registrant_domains(registrant_user)
from(
"(#{registrant_user_direct_domains_by_registrant(registrant_user).to_sql} UNION " \
"#{registrant_user_direct_domains_by_contact(registrant_user,
except_tech: true).to_sql}) AS domains"
)
end
def registrant_user_domains(registrant_user) def registrant_user_domains(registrant_user)
from( from(
"(#{registrant_user_domains_by_registrant(registrant_user).to_sql} UNION " \ "(#{registrant_user_domains_by_registrant(registrant_user).to_sql} UNION " \
@ -247,16 +264,20 @@ class Domain < ApplicationRecord
where(registrant: registrant_user.direct_contacts) where(registrant: registrant_user.direct_contacts)
end end
def registrant_user_direct_domains_by_contact(registrant_user) def registrant_user_direct_domains_by_contact(registrant_user, except_tech: false)
joins(:domain_contacts).where(domain_contacts: { contact_id: registrant_user.direct_contacts }) request = { contact_id: registrant_user.direct_contacts }
request[:type] = [AdminDomainContact.name] if except_tech
joins(:domain_contacts).where(domain_contacts: request)
end end
def registrant_user_company_registrant(companies) def registrant_user_company_registrant(companies)
where(registrant: companies) where(registrant: companies)
end end
def registrant_user_domains_company(companies) def registrant_user_domains_company(companies, except_tech: false)
joins(:domain_contacts).where(domain_contacts: { contact: companies }) request = { contact: companies }
request[:type] = [AdminDomainContact.name] if except_tech
joins(:domain_contacts).where(domain_contacts: request)
end end
end end

View file

@ -109,14 +109,12 @@ class DomainStatus < ApplicationRecord
DELETE_PROHIBIT_STATES = [ DELETE_PROHIBIT_STATES = [
DomainStatus::CLIENT_DELETE_PROHIBITED, DomainStatus::CLIENT_DELETE_PROHIBITED,
DomainStatus::SERVER_DELETE_PROHIBITED, DomainStatus::SERVER_DELETE_PROHIBITED,
DomainStatus::CLIENT_UPDATE_PROHIBITED,
DomainStatus::SERVER_UPDATE_PROHIBITED,
DomainStatus::PENDING_CREATE, DomainStatus::PENDING_CREATE,
DomainStatus::PENDING_RENEW, DomainStatus::PENDING_RENEW,
DomainStatus::PENDING_TRANSFER, DomainStatus::PENDING_TRANSFER,
DomainStatus::PENDING_UPDATE, DomainStatus::PENDING_UPDATE,
DomainStatus::PENDING_DELETE DomainStatus::PENDING_DELETE
] ].freeze
def epp_code_map def epp_code_map
{ {

View file

@ -82,8 +82,11 @@ class Epp::Contact < Contact
'2302' => [ # Object exists '2302' => [ # Object exists
[:code, :epp_id_taken] [:code, :epp_id_taken]
], ],
'2304' => [ # Status prohibits operation
[:statuses, :delete_prohibited],
],
'2305' => [ # Association exists '2305' => [ # Association exists
[:domains, :exist] [:domains, :exist],
] ]
} }
end end

View file

@ -201,7 +201,6 @@ class Epp::Domain < Domain
statuses.delete(DomainStatus::SERVER_HOLD) statuses.delete(DomainStatus::SERVER_HOLD)
statuses.delete(DomainStatus::EXPIRED) statuses.delete(DomainStatus::EXPIRED)
statuses.delete(DomainStatus::SERVER_UPDATE_PROHIBITED)
cancel_pending_delete cancel_pending_delete
save save
@ -387,4 +386,13 @@ class Epp::Domain < Domain
result result
end end
end end
private
def verification_needed?(code:)
new_registrant = Registrant.find_by(code: code)
return false if new_registrant.try(:identical_to?, registrant)
registrant.code != code
end
end end

View file

@ -1,8 +1,8 @@
class Invoice < ApplicationRecord class Invoice < ApplicationRecord
include Versions include Versions
include Concerns::Invoice::Cancellable include Invoice::Cancellable
include Concerns::Invoice::Payable include Invoice::Payable
include Concerns::Invoice::BookKeeping include Invoice::BookKeeping
belongs_to :buyer, class_name: 'Registrar' belongs_to :buyer, class_name: 'Registrar'
has_one :account_activity has_one :account_activity
@ -34,7 +34,7 @@ class Invoice < ApplicationRecord
before_create :calculate_total, unless: :total? before_create :calculate_total, unless: :total?
before_create :apply_default_buyer_vat_no, unless: :buyer_vat_no? before_create :apply_default_buyer_vat_no, unless: :buyer_vat_no?
attribute :vat_rate, ::Type::VATRate.new attribute :vat_rate, ::Type::VatRate.new
def set_invoice_number def set_invoice_number
last_no = Invoice.order(number: :desc).limit(1).pluck(:number).first last_no = Invoice.order(number: :desc).limit(1).pluck(:number).first

View file

@ -82,7 +82,7 @@ class LegalDocument < ApplicationRecord
end end
end end
contact_ids = DomainVersion.where(item_id: orig_legal.documentable_id).distinct. contact_ids = Version::DomainVersion.where(item_id: orig_legal.documentable_id).distinct.
pluck("object->>'registrant_id'", "object_changes->>'registrant_id'", pluck("object->>'registrant_id'", "object_changes->>'registrant_id'",
"children->>'tech_contacts'", "children->>'admin_contacts'").flatten.uniq "children->>'tech_contacts'", "children->>'admin_contacts'").flatten.uniq
contact_ids = contact_ids.map{|id| contact_ids = contact_ids.map{|id|

View file

@ -9,7 +9,7 @@ class RegistrantUser < User
delegate :can?, :cannot?, to: :ability delegate :can?, :cannot?, to: :ability
def ident def ident
registrant_ident.to_s.split('-').last registrant_ident.to_s[3..]
end end
def country def country
@ -18,6 +18,8 @@ class RegistrantUser < User
end end
def companies(company_register = CompanyRegister::Client.new) def companies(company_register = CompanyRegister::Client.new)
return [] if ident.include?('-')
company_register.representation_rights(citizen_personal_code: ident, company_register.representation_rights(citizen_personal_code: ident,
citizen_country_code: country.alpha3) citizen_country_code: country.alpha3)
end end
@ -30,11 +32,15 @@ class RegistrantUser < User
Contact.registrant_user_direct_contacts(self) Contact.registrant_user_direct_contacts(self)
end end
def domains def domains(admin: false)
return Domain.registrant_user_admin_registrant_domains(self) if admin
Domain.registrant_user_domains(self) Domain.registrant_user_domains(self)
end end
def direct_domains def direct_domains(admin: false)
return Domain.registrant_user_direct_admin_registrant_domains(self) if admin
Domain.registrant_user_direct_domains(self) Domain.registrant_user_direct_domains(self)
end end
@ -72,7 +78,7 @@ class RegistrantUser < User
return false unless user_data[:last_name] return false unless user_data[:last_name]
user_data[:country_code] ||= 'EE' user_data[:country_code] ||= 'EE'
%i[ident country_code].each { |f| user_data[f].upcase! if user_data[f].is_a?(String) } user_data[:country_code].upcase! if user_data[:country_code].is_a?(String)
find_or_create_by_user_data(user_data) find_or_create_by_user_data(user_data)
end end

View file

@ -1,8 +1,8 @@
class Registrar < ApplicationRecord class Registrar < ApplicationRecord
include Versions # version/registrar_version.rb include Versions # version/registrar_version.rb
include Concerns::Registrar::BookKeeping include Registrar::BookKeeping
include Concerns::EmailVerifable include EmailVerifable
include Concerns::Registrar::LegalDoc include Registrar::LegalDoc
has_many :domains, dependent: :restrict_with_error has_many :domains, dependent: :restrict_with_error
has_many :contacts, dependent: :restrict_with_error has_many :contacts, dependent: :restrict_with_error
@ -30,7 +30,7 @@ class Registrar < ApplicationRecord
validates :vat_rate, numericality: { greater_than_or_equal_to: 0, less_than: 100 }, validates :vat_rate, numericality: { greater_than_or_equal_to: 0, less_than: 100 },
allow_nil: true allow_nil: true
attribute :vat_rate, ::Type::VATRate.new attribute :vat_rate, ::Type::VatRate.new
after_initialize :set_defaults after_initialize :set_defaults
validate :correct_email_format, if: proc { |c| c.will_save_change_to_email? } validate :correct_email_format, if: proc { |c| c.will_save_change_to_email? }
@ -153,7 +153,7 @@ class Registrar < ApplicationRecord
puny = origin.domain.name_puny puny = origin.domain.name_puny
next unless domains.include?(idn) || domains.include?(puny) || domains.empty? next unless domains.include?(idn) || domains.include?(puny) || domains.empty?
if origin.domain.nameservers.where(hostname: new_attributes[:hostname]).any? if domain_not_updatable?(hostname: new_attributes[:hostname], domain: origin.domain)
failed_list << idn failed_list << idn
next next
end end
@ -202,6 +202,10 @@ class Registrar < ApplicationRecord
private private
def domain_not_updatable?(hostname:, domain:)
domain.nameservers.where(hostname: hostname).any? || domain.bulk_update_prohibited?
end
def set_defaults def set_defaults
self.language = Setting.default_language unless language self.language = Setting.default_language unless language
end end

View file

@ -1,5 +1,5 @@
module Type module Type
class VATRate < ActiveRecord::Type::Decimal class VatRate < ActiveRecord::Type::Decimal
def deserialize(value) def deserialize(value)
super * 100 if value super * 100 if value
end end

1
app/models/version.rb Normal file
View file

@ -0,0 +1 @@
class Version; end

View file

@ -1,4 +1,4 @@
class AccountActivityVersion < PaperTrail::Version class Version::AccountActivityVersion < PaperTrail::Version
self.table_name = :log_account_activities self.table_name = :log_account_activities
self.sequence_name = :log_account_activities_id_seq self.sequence_name = :log_account_activities_id_seq
end end

View file

@ -1,4 +1,4 @@
class AccountVersion < PaperTrail::Version class Version::AccountVersion < PaperTrail::Version
self.table_name = :log_accounts self.table_name = :log_accounts
self.sequence_name = :log_accounts_id_seq self.sequence_name = :log_accounts_id_seq
end end

View file

@ -1,4 +1,4 @@
class ActionVersion < PaperTrail::Version class Version::ActionVersion < PaperTrail::Version
self.table_name = :log_actions self.table_name = :log_actions
self.sequence_name = :log_actions_id_seq self.sequence_name = :log_actions_id_seq
end end

View file

@ -1,4 +1,4 @@
class BankStatementVersion < PaperTrail::Version class Version::BankStatementVersion < PaperTrail::Version
self.table_name = :log_bank_statements self.table_name = :log_bank_statements
self.sequence_name = :log_bank_statements_id_seq self.sequence_name = :log_bank_statements_id_seq
end end

View file

@ -1,4 +1,4 @@
class BankTransactionVersion < PaperTrail::Version class Version::BankTransactionVersion < PaperTrail::Version
self.table_name = :log_bank_transactions self.table_name = :log_bank_transactions
self.sequence_name = :log_bank_transactions_id_seq self.sequence_name = :log_bank_transactions_id_seq
end end

View file

@ -1,7 +1,4 @@
module Billing class Version::Billing::PriceVersion < PaperTrail::Version
class PriceVersion < PaperTrail::Version self.table_name = :log_prices
self.table_name = :log_prices self.sequence_name = :log_prices_id_seq
self.sequence_name = :log_prices_id_seq
end
end end

View file

@ -1,4 +1,4 @@
class BlockedDomainVersion < PaperTrail::Version class Version::BlockedDomainVersion < PaperTrail::Version
self.table_name = :log_blocked_domains self.table_name = :log_blocked_domains
self.sequence_name = :log_blocked_domains_id_seq self.sequence_name = :log_blocked_domains_id_seq
end end

View file

@ -1,4 +1,4 @@
class CertificateVersion < PaperTrail::Version class Version::CertificateVersion < PaperTrail::Version
self.table_name = :log_certificates self.table_name = :log_certificates
self.sequence_name = :log_certificates_id_seq self.sequence_name = :log_certificates_id_seq
end end

View file

@ -1,7 +1,7 @@
class ContactVersion < PaperTrail::Version class Version::ContactVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_contacts self.table_name = :log_contacts
self.sequence_name = :log_contacts_id_seq self.sequence_name = :log_contacts_id_seq
# scope :deleted, -> { where(event: 'destroy') } # scope :deleted, -> { where(event: 'destroy') }
end end

View file

@ -1,4 +1,4 @@
class DnskeyVersion < PaperTrail::Version class Version::DnskeyVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_dnskeys self.table_name = :log_dnskeys
self.sequence_name = :log_dnskeys_id_seq self.sequence_name = :log_dnskeys_id_seq

View file

@ -1,4 +1,4 @@
class DomainContactVersion < PaperTrail::Version class Version::DomainContactVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_domain_contacts self.table_name = :log_domain_contacts
self.sequence_name = :log_domain_contacts_id_seq self.sequence_name = :log_domain_contacts_id_seq

View file

@ -1,4 +1,4 @@
class DomainVersion < PaperTrail::Version class Version::DomainVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_domains self.table_name = :log_domains

View file

@ -1,4 +1,4 @@
class InvoiceItemVersion < PaperTrail::Version class Version::InvoiceItemVersion < PaperTrail::Version
self.table_name = :log_invoice_items self.table_name = :log_invoice_items
self.sequence_name = :log_invoice_items_id_seq self.sequence_name = :log_invoice_items_id_seq
end end

View file

@ -1,4 +1,4 @@
class InvoiceVersion < PaperTrail::Version class Version::InvoiceVersion < PaperTrail::Version
self.table_name = :log_invoices self.table_name = :log_invoices
self.sequence_name = :log_invoices_id_seq self.sequence_name = :log_invoices_id_seq
end end

View file

@ -1,4 +1,4 @@
class NameserverVersion < PaperTrail::Version class Version::NameserverVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_nameservers self.table_name = :log_nameservers
self.sequence_name = :log_nameservers_id_seq self.sequence_name = :log_nameservers_id_seq

View file

@ -1,4 +1,4 @@
class NotificationVersion < PaperTrail::Version class Version::NotificationVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_notifications self.table_name = :log_notifications
self.sequence_name = :log_notifications_id_seq self.sequence_name = :log_notifications_id_seq

View file

@ -1,4 +1,4 @@
class PaymentOrderVersion < PaperTrail::Version class Version::PaymentOrderVersion < PaperTrail::Version
self.table_name = :log_payment_orders self.table_name = :log_payment_orders
self.sequence_name = :log_payment_orders_id_seq self.sequence_name = :log_payment_orders_id_seq
end end

View file

@ -1,4 +1,4 @@
class RegistrantVerificationVersion < PaperTrail::Version class Version::RegistrantVerificationVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_registrant_verifications self.table_name = :log_registrant_verifications
self.sequence_name = :log_registrant_verifications_id_seq self.sequence_name = :log_registrant_verifications_id_seq

View file

@ -1,4 +1,4 @@
class RegistrarVersion < PaperTrail::Version class Version::RegistrarVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_registrars self.table_name = :log_registrars
self.sequence_name = :log_registrars_id_seq self.sequence_name = :log_registrars_id_seq

View file

@ -1,4 +1,4 @@
class ReservedDomainVersion < PaperTrail::Version class Version::ReservedDomainVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_reserved_domains self.table_name = :log_reserved_domains
self.sequence_name = :log_reserved_domains_id_seq self.sequence_name = :log_reserved_domains_id_seq

View file

@ -1,4 +1,4 @@
class SettingEntryVersion < PaperTrail::Version class Version::SettingEntryVersion < PaperTrail::Version
self.table_name = :log_setting_entries self.table_name = :log_setting_entries
self.sequence_name = :log_setting_entries self.sequence_name = :log_setting_entries
end end

View file

@ -1,4 +1,4 @@
class SettingVersion < PaperTrail::Version class Version::SettingVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_settings self.table_name = :log_settings
self.sequence_name = :log_settings_id_seq self.sequence_name = :log_settings_id_seq

View file

@ -1,4 +1,4 @@
class UserVersion < PaperTrail::Version class Version::UserVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_users self.table_name = :log_users
self.sequence_name = :log_users_id_seq self.sequence_name = :log_users_id_seq

View file

@ -1,4 +1,4 @@
class WhiteIpVersion < PaperTrail::Version class Version::WhiteIpVersion < PaperTrail::Version
include VersionSession include VersionSession
self.table_name = :log_white_ips self.table_name = :log_white_ips
self.sequence_name = :log_white_ips_id_seq self.sequence_name = :log_white_ips_id_seq

View file

@ -1,4 +1,4 @@
class Registrar::DomainListCSVPresenter class Registrar::DomainListCsvPresenter
def initialize(domains:, view:) def initialize(domains:, view:)
@domains = domains @domains = domains
@view = view @view = view

View file

@ -66,7 +66,7 @@
= domain.registrant.name = domain.registrant.name
- else - else
- contact = Contact.all_versions_for([domain.registrant_id], version.created_at).first - contact = Contact.all_versions_for([domain.registrant_id], version.created_at).first
- if contact.nil? && ver = ContactVersion.where(item_id: domain.registrant_id).last - if contact.nil? && ver = Version::ContactVersion.where(item_id: domain.registrant_id).last
- contact = Contact.new(ver.object.to_h.merge(ver.object_changes.to_h.each_with_object({}) {|(k,v), o| o[k] = v.last })) - contact = Contact.new(ver.object.to_h.merge(ver.object_changes.to_h.each_with_object({}) {|(k,v), o| o[k] = v.last }))
= contact.try(:name) = contact.try(:name)
= "&#160;".html_safe = "&#160;".html_safe

View file

@ -23,7 +23,7 @@
- admin_contacts = domain.admin_contacts - admin_contacts = domain.admin_contacts
- registrant = domain.registrant - registrant = domain.registrant
- unless registrant - unless registrant
- ver = ContactVersion.where(item_id: domain.registrant_id).where(event: :destroy).last - ver = Version::ContactVersion.where(item_id: domain.registrant_id).where(event: :destroy).last
- registrant = ver.reify - registrant = ver.reify
- registrant.version_loader = ver - registrant.version_loader = ver
- registrant = [registrant] - registrant = [registrant]

View file

@ -1,7 +0,0 @@
<div class="alert alert-danger">
<ul>
<% errors.each_value do |errors| %>
<li><%= errors.join('<br>') %></li>
<% end %>
</ul>
</div>

View file

@ -1,64 +0,0 @@
<%= form_for [:registrant, domain, @contact], html: { class: 'form-horizontal' } do |f| %>
<% if @errors.present? %>
<%= render 'api_errors', errors: @errors %>
<% end %>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :name %>
</div>
<div class="col-md-4">
<%= f.text_field :name, required: true, autofocus: true, class: 'form-control' %>
</div>
</div>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :email %>
</div>
<div class="col-md-4">
<%= f.email_field :email, required: true, class: 'form-control' %>
</div>
</div>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :phone %>
</div>
<div class="col-md-4">
<%= f.text_field :phone, required: true, class: 'form-control' %>
</div>
</div>
<% if Contact.address_processing? %>
<div class="panel panel-default">
<div class="panel-heading"><%= t '.address' %></div>
<div class="panel-body">
<%= render 'registrant/contacts/form/address', f: f %>
</div>
</div>
<% end %>
<% if fax_enabled? %>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :fax %>
</div>
<div class="col-md-4">
<%= f.text_field :fax, class: 'form-control' %>
</div>
</div>
<% end %>
<hr/>
<div class="row">
<div class="col-md-6 text-right">
<%= button_tag t('.submit_btn'), class: 'btn btn-success' %>
</div>
</div>
<% end %>

View file

@ -1,12 +0,0 @@
<ol class="breadcrumb">
<li><%= link_to t('registrant.domains.index.header'), registrant_domains_path %></li>
<li><%= link_to domain, registrant_domain_path(domain) %></li>
<li><%= t 'registrant.contacts.contact_index' %></li>
<li><%= link_to @contact, registrant_domain_contact_path(domain, @contact) %></li>
</ol>
<div class="page-header">
<h1><%= t '.header' %></h1>
</div>
<%= render 'form' %>

View file

@ -1,51 +0,0 @@
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :street %>
</div>
<div class="col-md-4">
<%= f.text_field :street, required: true, class: 'form-control' %>
</div>
</div>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :zip %>
</div>
<div class="col-md-4">
<%= f.text_field :zip, required: true, class: 'form-control' %>
</div>
</div>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :city %>
</div>
<div class="col-md-4">
<%= f.text_field :city, required: true, class: 'form-control' %>
</div>
</div>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :state %>
</div>
<div class="col-md-4">
<%= f.text_field :state, class: 'form-control' %>
</div>
</div>
<div class="form-group">
<div class="col-md-2 control-label">
<%= f.label :country_code, 'Country' %>
</div>
<div class="col-md-4">
<%= f.select :country_code, SortedCountry.all_options(f.object.country_code), {},
required: true,
class: 'form-control' %>
</div>
</div>

View file

@ -1,42 +0,0 @@
<ol class="breadcrumb">
<li><%= link_to t('registrant.domains.index.header'), registrant_domains_path %></li>
<li><%= link_to domain, registrant_domain_path(domain) %></li>
<li><%= t 'registrant.contacts.contact_index' %></li>
</ol>
<div class="page-header">
<div class="row">
<div class="col-md-6">
<h1><%= @contact %></h1>
</div>
<% if @contact.managed_by?(current_registrant_user) %>
<div class="col-md-6 text-right">
<%= link_to t('.edit_btn'), edit_registrant_domain_contact_path(domain, @contact),
class: 'btn btn-primary' %>
</div>
<% end %>
</div>
</div>
<div class="row">
<div class="col-md-6">
<%= render 'registrant/contacts/show/general' %>
</div>
<div class="col-md-6">
<%= render 'registrant/contacts/show/address' %>
</div>
</div>
<div class="row">
<div class="col-md-12">
<%= render 'registrant/contacts/show/statuses', contact: @contact %>
</div>
</div>
<div class="row">
<div class="col-md-12">
<%= render 'registrant/contacts/show/domains', contact: @contact %>
</div>
</div>

View file

@ -1,31 +0,0 @@
<div class="panel panel-default">
<div class="panel-heading">
<h3 class="panel-title">
<%= t '.header' %>
</h3>
</div>
<div class="panel-body">
<dl class="dl-horizontal">
<% if @contact.org_name.present? %>
<dt><%= Contact.human_attribute_name :org_name %></dt>
<dd><%= @contact.org_name %></dd>
<% end %>
<dt><%= Contact.human_attribute_name :street %></dt>
<dd><%= @contact.street %></dd>
<dt><%= Contact.human_attribute_name :city %></dt>
<dd><%= @contact.city %></dd>
<dt><%= Contact.human_attribute_name :zip %></dt>
<dd><%= @contact.zip %></dd>
<dt><%= Contact.human_attribute_name :state %></dt>
<dd><%= @contact.state %></dd>
<dt><%= Contact.human_attribute_name :country %></dt>
<dd><%= @contact.country %></dd>
</dl>
</div>
</div>

Some files were not shown because too many files have changed in this diff Show more