about summary refs log tree commit diff
path: root/app/validators/ed25519_signature_validator.rb
blob: 77a21b8373ca12c24abc802be0d5abba0de184d6 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
# frozen_string_literal: true

class Ed25519SignatureValidator < ActiveModel::EachValidator
  def validate_each(record, attribute, value)
    return if value.blank?

    verify_key = Ed25519::VerifyKey.new(Base64.decode64(option_to_value(record, :verify_key)))
    signature  = Base64.decode64(value)
    message    = option_to_value(record, :message)

    record.errors[attribute] << I18n.t('crypto.errors.invalid_signature') unless verified?(verify_key, signature, message)
  end

  private

  def verified?(verify_key, signature, message)
    verify_key.verify(signature, message)
  rescue Ed25519::VerifyError, ArgumentError
    false
  end

  def option_to_value(record, key)
    if options[key].is_a?(Proc)
      options[key].call(record)
    else
      record.public_send(options[key])
    end
  end
end