# AddressableUrlValidator # # Custom validator for URLs. This is a stricter version of UrlValidator - it also checks # for using the right protocol, but it actually parses the URL checking for any syntax errors. # The regex is also different from `URI` as we use `Addressable::URI` here. # # By default, only URLs for http, https, ssh, and git protocols will be considered valid. # Provide a `:protocols` option to configure accepted protocols. # # Example: # # class User < ActiveRecord::Base # validates :personal_url, addressable_url: true # # validates :ftp_url, addressable_url: { protocols: %w(ftp) } # # validates :git_url, addressable_url: { protocols: %w(http https ssh git) } # end # class AddressableUrlValidator < ActiveModel::EachValidator def validate_each(record, attribute, value) unless valid_url?(value) record.errors.add(attribute, "must be a valid URL") end end private def valid_url?(value) return false unless value value.strip! valid_protocol?(value) && valid_uri?(value) end def default_options @default_options ||= { protocols: %w(http https ssh git) } end def valid_uri?(value) Addressable::URI.parse(value).is_a?(Addressable::URI) rescue Addressable::URI::InvalidURIError false end def valid_protocol?(value) options = default_options.merge(self.options) !!(value =~ /\A#{URI.regexp(options[:protocols])}\z/) end end