require "otp/utils" require "otp/base32" require "otp/uri" module OTP class Base include OTP::Utils DEFAULT_DIGITS = 6 DEFAULT_ALGORITHM = "SHA1" attr_accessor :secret attr_accessor :algorithm attr_accessor :digits attr_accessor :issuer, :accountname def initialize(secret=nil, algorithm=nil, digits=nil) self.secret = secret self.algorithm = algorithm || DEFAULT_ALGORITHM self.digits = digits || DEFAULT_DIGITS end def new_secret(num_bytes=10) s = (0...num_bytes).map{ Random.rand(256).chr }.join self.secret = OTP::Base32.encode(s) end def moving_factor raise NotImplementedError end def otp hash = hmac(algorithm, OTP::Base32.decode(secret), pack_int64(moving_factor)) return truncate(hash) end def password pw = (otp % (10 ** digits)).to_s pw = "0" + pw while pw.length < digits return pw end def verify(otp) return false if otp.nil? || otp.empty? return compare(password, otp) end ## URI related methods def to_uri OTP::URI.format(self) end def type_specific_uri_params raise NotImplementedError end def extract_type_specific_uri_param raise NotImplementedError end end end