webhook.rb 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. # frozen_string_literal: true
  2. # == Schema Information
  3. #
  4. # Table name: webhooks
  5. #
  6. # id :bigint(8) not null, primary key
  7. # url :string not null
  8. # events :string default([]), not null, is an Array
  9. # secret :string default(""), not null
  10. # enabled :boolean default(TRUE), not null
  11. # created_at :datetime not null
  12. # updated_at :datetime not null
  13. #
  14. class Webhook < ApplicationRecord
  15. EVENTS = %w(
  16. account.created
  17. report.created
  18. ).freeze
  19. attr_writer :current_account
  20. scope :enabled, -> { where(enabled: true) }
  21. validates :url, presence: true, url: true
  22. validates :secret, presence: true, length: { minimum: 12 }
  23. validates :events, presence: true
  24. validate :validate_events
  25. validate :validate_permissions
  26. before_validation :strip_events
  27. before_validation :generate_secret
  28. def rotate_secret!
  29. update!(secret: SecureRandom.hex(20))
  30. end
  31. def enable!
  32. update!(enabled: true)
  33. end
  34. def disable!
  35. update!(enabled: false)
  36. end
  37. def required_permissions
  38. events.map { |event| Webhook.permission_for_event(event) }
  39. end
  40. def self.permission_for_event(event)
  41. case event
  42. when 'account.approved', 'account.created', 'account.updated'
  43. :manage_users
  44. when 'report.created'
  45. :manage_reports
  46. end
  47. end
  48. private
  49. def validate_events
  50. errors.add(:events, :invalid) if events.any? { |e| !EVENTS.include?(e) }
  51. end
  52. def validate_permissions
  53. errors.add(:events, :invalid_permissions) if defined?(@current_account) && required_permissions.any? { |permission| !@current_account.user_role.can?(permission) }
  54. end
  55. def strip_events
  56. self.events = events.map { |str| str.strip.presence }.compact if events.present?
  57. end
  58. def generate_secret
  59. self.secret = SecureRandom.hex(20) if secret.blank?
  60. end
  61. end