diff --git a/lib/net/imap/sasl.rb b/lib/net/imap/sasl.rb
index 62096f86..5a361d33 100644
--- a/lib/net/imap/sasl.rb
+++ b/lib/net/imap/sasl.rb
@@ -33,6 +33,16 @@ class IMAP
# +PLAIN+:: See PlainAuthenticator.
# Login using clear-text username and password.
#
+ # +SCRAM-SHA-1+, +SCRAM-SHA-256+::
+ # See ScramAuthenticator.
+ # Login by username and password. The password is not sent
+ # to the server but is used in a salted challenge/response
+ # exchange. One of the benefits over +PLAIN+ is that the
+ # server cannot impersonate the user to other servers.
+ # +SCRAM-SHA-1+ and +SCRAM-SHA-256+ are supported, but any
+ # algorithm supported by OpenSSL::Digest can easily be
+ # added.
+ #
# +OAUTHBEARER+:: See OAuthBearerAuthenticator.
# Login using an OAUTH2 Bearer token. This is the
# standard mechanism for using OAuth2 with \SASL, but it
@@ -77,10 +87,15 @@ module SASL
autoload :Authenticator, "#{sasl_dir}/authenticator"
autoload :Authenticators, "#{sasl_dir}/authenticators"
autoload :GS2Header, "#{sasl_dir}/gs2_header"
+ autoload :ScramAlgorithm, "#{sasl_dir}/scram_algorithm"
+ autoload :ScramAuthenticator, "#{sasl_dir}/scram_authenticator"
+
autoload :AnonymousAuthenticator, "#{sasl_dir}/anonymous_authenticator"
autoload :ExternalAuthenticator, "#{sasl_dir}/external_authenticator"
autoload :OAuthBearerAuthenticator, "#{sasl_dir}/oauthbearer_authenticator"
autoload :PlainAuthenticator, "#{sasl_dir}/plain_authenticator"
+ autoload :ScramSHA1Authenticator, "#{sasl_dir}/scram_sha1_authenticator"
+ autoload :ScramSHA256Authenticator, "#{sasl_dir}/scram_sha256_authenticator"
autoload :XOAuth2Authenticator, "#{sasl_dir}/xoauth2_authenticator"
autoload :CramMD5Authenticator, "#{sasl_dir}/cram_md5_authenticator"
@@ -94,6 +109,8 @@ def self.authenticators
registry.add_authenticator "External"
registry.add_authenticator "OAuthBearer"
registry.add_authenticator "Plain"
+ registry.add_authenticator "Scram-SHA-1"
+ registry.add_authenticator "Scram-SHA-256"
registry.add_authenticator "XOAuth2"
registry.add_authenticator "Login" # deprecated
registry.add_authenticator "Cram-MD5" # deprecated
diff --git a/lib/net/imap/sasl/authenticator.rb b/lib/net/imap/sasl/authenticator.rb
index 170a193e..8683371c 100644
--- a/lib/net/imap/sasl/authenticator.rb
+++ b/lib/net/imap/sasl/authenticator.rb
@@ -59,7 +59,6 @@ module SASL
# * +scram_sha1_salted_passwords+, +scram_sha256_salted_password+ ---
# Salted password(s) (with salt and iteration count) for the +SCRAM-*+
# mechanism family. [salt, iterations, pbkdf2_hmac] tuple.
- # (not implemented yet...)
# * +passcode+ --- passcode for SecurID 2FA (not implemented)
# * +pin+ --- Personal Identification number, e.g. for SecurID 2FA
# (not implemented)
diff --git a/lib/net/imap/sasl/scram_algorithm.rb b/lib/net/imap/sasl/scram_algorithm.rb
new file mode 100644
index 00000000..d74adbdf
--- /dev/null
+++ b/lib/net/imap/sasl/scram_algorithm.rb
@@ -0,0 +1,69 @@
+# frozen_string_literal: true
+
+module Net
+ class IMAP
+ module SASL
+
+ # For method descriptions, see {RFC5802
+ # §2}[https://www.rfc-editor.org/rfc/rfc5802#section-2] and {RFC5802
+ # §3}[https://www.rfc-editor.org/rfc/rfc5802#section-3].
+ #
+ # Expects:
+ # * #Hi, #H, and #HMAC use:
+ # * +#digest+ --- an OpenSSL::Digest.
+ # * #salted_password uses:
+ # * +#salt+ and +#iterations+ --- the server's values for this user
+ # * +#password+
+ # * #auth_message is built from:
+ # * +#client_first_message_bare+ --- contains +#cnonce+
+ # * +#server_first_message+ --- contains +#snonce+
+ # * +#client_final_message_no_proof+ --- contains +#snonce+
+ module ScramAlgorithm
+ def Normalize(str) SASL.saslprep(str) end
+
+ def Hi(str, salt, iterations)
+ length = digest.digest_length
+ OpenSSL::KDF.pbkdf2_hmac(
+ str,
+ salt: salt,
+ iterations: iterations,
+ length: length,
+ hash: digest,
+ )
+ end
+
+ def H(str) digest.digest str end
+
+ def HMAC(key, data) OpenSSL::HMAC.digest(digest, key, data) end
+
+ def XOR(str1, str2)
+ str1.unpack("C*")
+ .zip(str2.unpack("C*"))
+ .map {|a, b| a ^ b }
+ .pack("C*")
+ end
+
+ def auth_message
+ [
+ client_first_message_bare,
+ server_first_message,
+ client_final_message_no_proof,
+ ]
+ .join(",")
+ end
+
+ def salted_password
+ Hi(Normalize(password), salt, iterations)
+ end
+
+ def client_key; HMAC(salted_password, "Client Key") end
+ def server_key; HMAC(salted_password, "Server Key") end
+ def stored_key; H(client_key) end
+ def client_signature; HMAC(stored_key, auth_message) end
+ def server_signature; HMAC(server_key, auth_message) end
+ def client_proof; XOR(client_key, client_signature) end
+ end
+
+ end
+ end
+end
diff --git a/lib/net/imap/sasl/scram_authenticator.rb b/lib/net/imap/sasl/scram_authenticator.rb
new file mode 100644
index 00000000..e80691b6
--- /dev/null
+++ b/lib/net/imap/sasl/scram_authenticator.rb
@@ -0,0 +1,328 @@
+# frozen_string_literal: true
+
+require "openssl"
+require "securerandom"
+
+require_relative "gs2_header"
+require_relative "scram_algorithm"
+
+module Net
+ class IMAP
+ module SASL
+
+ # Abstract base class for the "+SCRAM-*+" family of SASL mechanisms,
+ # defined in RFC5802[https://tools.ietf.org/html/rfc5802]. Use via
+ # Net::IMAP#authenticate.
+ #
+ # Directly supported:
+ # * +SCRAM-SHA-1+ --- ScramAuthenticator::SHA1
+ # * +SCRAM-SHA-256+ --- ScramAuthenticator::SHA256
+ #
+ # New +SCRAM-*+ mechanisms can easily be added for any hash function
+ # supported by OpenSSL::Digest.
+ #
+ # === TLS Channel binding
+ #
+ # The SCRAM-*-PLUS mechanisms and channel binding are not
+ # supported yet.
+ #
+ # === Caching SCRAM secrets
+ #
+ # Caching of salted_password, client_key, stored_key, and server_key
+ # is not supported yet.
+ #
+ # === SCRAM algorithm
+ #
+ # See the documentation and method definitions on ScramAlgorithm for an
+ # overview of the algorithm. The different mechanisms differ only by
+ # which hash function that is used (or by support for channel binding with
+ # +-PLUS+).
+ #
+ # === Saved message parameters
+ #
+ # ==== Client messages
+ #
+ # As client messages are generated and sent, they are validated and saved
+ # as #client_first_message_bare, #client_first_message,
+ # #client_final_message_no_proof, and #client_final_message. Some message
+ # attributes are also saved, such as #cnonce. See also the methods on
+ # GS2Header.
+ #
+ # ==== Server messages
+ #
+ # As server messages are received, they are validated and loaded into
+ # the various attributes, e.g: #snonce, #salt, #iterations, #verifier,
+ # #server_error. The message strings themselves are also saved as
+ # #server_first_message, #server_final_message, and #server_extra_message.
+ #
+ # Unlike many other SASL mechanisms, the +SCRAM-*+ family supports mutual
+ # authentication and can return server error data in the server messages.
+ # If #process raises an AuthenticationFailure for the
+ # server_final_message, then server_error may contain error details.
+ #
+ class ScramAuthenticator < Authenticator
+ include GS2Header
+ include ScramAlgorithm
+
+ def self.digest_algorithm(name)
+ mech = "SCRAM-#{name}"
+ ossl = name.delete("-")
+ singleton_class.class_eval do
+ define_method(:mechanism_name) { mech }
+ define_method(:digest) { OpenSSL::Digest.new ossl }
+ end
+ define_method(:digest) { OpenSSL::Digest.new ossl }
+ end
+
+ ##
+ # :call-seq:
+ # new(username, password, authzid = nil, **) -> auth_ctx
+ # new(authcid:, password:, authzid: nil, **) -> auth_ctx
+ # new(**) {|propname, auth_ctx| propval } -> auth_ctx
+ #
+ # Creates an Authenticator for one of the "+SCRAM-*+" SASL mechanisms.
+ # Each subclass defines #digest to match a specific mechanism.
+ #
+ # Called by Net::IMAP#authenticate and similar methods on other clients.
+ #
+ # === Properties
+ #
+ # * #authcid ― Identity whose #password is used. Aliased as #username.
+ # * #password ― Password or passphrase associated with this #authcid.
+ # * #authzid ― Alternate identity to act as or on behalf of. Optional.
+ # * Not implemented yet: +scram_sha1_salted_passwords+,
+ # +scram_sha256_salted_passwords+, +scram_sha1_salted_password+,
+ # +scram_sha256_salted_password+ --- Cached salted password(s)
+ # tuple(s) (combined with salt and iteration count): [salt,
+ # iterations, pbkdf2_hmac]
+ #
+ # See the documentation on each property method for more details.
+ #
+ # +authcid+, +password+, and +authzid+ may be sent as either positional
+ # or keyword arguments. See Net::IMAP::SASL::Authenticator@Properties
+ # for a detailed description of property assignment, lazy loading, and
+ # callbacks.
+ #
+ def initialize(username_arg = nil, passwd_arg = nil, authzid_arg = nil,
+ authcid: nil, username: nil, password: nil, authzid: nil,
+ min_iterations: 4096, # see both RFC5802 and RFC7677
+ cnonce: nil, # must only be set in tests
+ **options)
+ super
+ propinit :authcid, authcid, username, username_arg, required: true
+ propinit :password, password, passwd_arg, required: true
+ propinit :authzid, authzid, authzid_arg
+
+ @min_iterations = Integer min_iterations
+ @min_iterations.positive? or
+ raise ArgumentError, "min_iterations must be positive"
+ @cnonce = cnonce || SecureRandom.base64(32)
+ end
+
+ ##
+ # :call-seq: authcid -> string or nil
+ #
+ # Authentication identity: the identity that matches the #password.
+ #
+ # RFC-2831[https://tools.ietf.org/html/rfc2831] uses the term +username+.
+ # "Authentication identity" is the generic term used by
+ # RFC-4422[https://tools.ietf.org/html/rfc4422].
+ # RFC-4616[https://tools.ietf.org/html/rfc4616] and many later RFCs
+ # abbreviate to +authcid+. #username is available as an alias for
+ # #authcid, but only :authcid will be sent to callbacks.
+ property :authcid
+ alias username authcid
+
+ ##
+ # :call-seq: password -> string or nil
+ #
+ # A password or passphrase that matches the #authcid.
+ property :password
+
+ ##
+ # :call-seq: authzid -> string or nil
+ #
+ # Authorization identity: an identity to act as or on behalf of. The
+ # identity form is application protocol specific. If not provided or
+ # left blank, the server derives an authorization identity from the
+ # authentication identity. The server is responsible for verifying the
+ # client's credentials and verifying that the identity it associates with
+ # the client's authentication identity is allowed to act as (or on behalf
+ # of) the authorization identity.
+ #
+ # For example, an administrator or superuser might take on another role:
+ #
+ # imap.authenticate "PLAIN", "root", ->{passwd}, authzid: "user"
+ #
+ property :authzid
+
+ # The minimal allowed iteration count. Lower #iterations will raise an
+ # AuthenticationFailure.
+ attr_reader :min_iterations
+
+ # The client nonce, generated by SecureRandom
+ attr_reader :cnonce
+
+ # The server nonce, which must start with #cnonce
+ attr_reader :snonce
+
+ # The salt used by the server for this user
+ attr_reader :salt
+
+ # The iteration count for the selected hash function and user
+ attr_reader :iterations
+
+ # The first message sent to the server
+ attr_reader :client_first_message
+
+ # The final message sent to the server
+ attr_reader :client_final_message
+
+ # The server-sent parameters from its first message
+ attr_reader :server_first_message
+
+ # The server-sent parameters from its final message
+ attr_reader :server_final_message
+
+ # An unexpected server challenge, either sent before the
+ # initial_client_response and ignored, or sent after the
+ # server_final_message and raised an error.
+ attr_reader :server_extra_message
+
+ # The server verifier, which must equal the locally computed server
+ # signature.
+ attr_reader :verifier
+
+ # An error reported by the server during the \SASL exchange.
+ #
+ # Does not include errors reported by the protocol, e.g.
+ # Net::IMAP::NoResponseError.
+ attr_reader :server_error
+
+ # Has the initial_client_response been sent yet?
+ alias sent_first? client_first_message
+
+ # Has the final_client_message been sent yet?
+ alias sent_final? client_final_message
+
+ # Has the first server response been received yet?
+ alias recv_first? server_first_message
+
+ # Has the final server response been received yet?
+ alias recv_final? server_final_message
+
+ # Returns a new OpenSSL::Digest object, set to the appropriate hash
+ # function for the chosen mechanism.
+ #
+ # The class's +Digest+ constant must be set to an OpenSSL::Digest
+ # class.
+ def digest; raise NotImplementedError, "defined in subclasses" end
+
+ # Is the authentication exchange complete?
+ #
+ # If false, another server continuation is required.
+ def done?; sent_final? && recv_final? end
+
+ # responds to the server's challenges
+ def process(challenge)
+ if !sent_first?
+ @server_extra_message = challenge
+ @client_first_message = initial_client_response
+ elsif !recv_first?
+ @server_first_message = challenge
+ parse_server_first_message
+ @client_final_message = final_message_with_proof
+ elsif !recv_final?
+ @server_final_message = challenge
+ parse_server_final_message
+ ""
+ else
+ @server_extra_message = challenge
+ raise AuthenticationFailure, "server sent after complete, %p" % [
+ server_extra_message,
+ ]
+ end
+ end
+
+ # See {RFC5802 §7}[https://www.rfc-editor.org/rfc/rfc5802#section-7]
+ # +client-first-message+.
+ def initial_client_response
+ "#{gs2_header}#{client_first_message_bare}"
+ end
+
+ # See {RFC5802 §7}[https://www.rfc-editor.org/rfc/rfc5802#section-7]
+ # +client-first-message-bare+.
+ def client_first_message_bare
+ @client_first_message_bare ||=
+ format_message(n: gs2_saslname_encode(SASL.saslprep(username)),
+ r: cnonce)
+ end
+
+ # See {RFC5802 §7}[https://www.rfc-editor.org/rfc/rfc5802#section-7]
+ # +client-final-message+.
+ def final_message_with_proof
+ proof = [client_proof].pack("m0")
+ "#{client_final_message_no_proof},p=#{proof}"
+ end
+
+ # See {RFC5802 §7}[https://www.rfc-editor.org/rfc/rfc5802#section-7]
+ # +client-final-message-without-proof+.
+ def client_final_message_no_proof
+ @client_final_message_no_proof ||=
+ format_message(c: [cbind_input].pack("m0"), # channel-binding
+ r: snonce) # nonce
+ end
+
+ # See {RFC5802 §7}[https://www.rfc-editor.org/rfc/rfc5802#section-7]
+ # +cbind-input+.
+ #
+ # >>>
+ # *TODO:* implement channel binding, appending +cbind-data+ here.
+ alias cbind_input gs2_header
+
+ private
+
+ def format_message(hash) hash.map { _1.join("=") }.join(",") end
+
+ def parse_server_first_message
+ sparams = parse_challenge server_first_message
+ @snonce = sparams["r"] or
+ raise AuthenticationFailure, "server did not send nonce"
+ @salt = sparams["s"]&.unpack1("m") or
+ raise AuthenticationFailure, "server did not send salt"
+ @iterations = sparams["i"]&.then {|i| Integer i } or
+ raise AuthenticationFailure, "server did not send iteration count"
+ min_iterations <= iterations or
+ raise AuthenticationFailure, "too few iterations: %d" % [iterations]
+ mext = sparams["m"] and
+ raise AuthenticationFailure, "mandatory extension: %p" % [mext]
+ snonce.start_with? cnonce or
+ raise AuthenticationFailure, "invalid server nonce"
+ end
+
+ def parse_server_final_message
+ sparams = parse_challenge server_final_message
+ @server_error = sparams["e"] and
+ raise AuthenticationFailure, "server error: %s" % [server_error]
+ @verifier = sparams["v"].unpack1("m") or
+ raise AuthenticationFailure, "server did not send verifier"
+ verifier == server_signature or
+ raise AuthenticationFailure, "server verify failed: %p != %p" % [
+ server_signature, verifier
+ ]
+ end
+
+ # RFC5802 specifies "that the order of attributes in client or server
+ # messages is fixed, with the exception of extension attributes", but
+ # this parses it simply as a hash, without respect to order. Note that
+ # repeated keys (violating the spec) will use the last value.
+ def parse_challenge(challenge)
+ challenge.split(/,/).to_h {|pair| pair.split(/=/, 2) }
+ rescue ArgumentError
+ raise AuthenticationFailure, "unparsable challenge: %p" % [challenge]
+ end
+
+ end
+ end
+ end
+end
diff --git a/lib/net/imap/sasl/scram_sha1_authenticator.rb b/lib/net/imap/sasl/scram_sha1_authenticator.rb
new file mode 100644
index 00000000..a6ccb580
--- /dev/null
+++ b/lib/net/imap/sasl/scram_sha1_authenticator.rb
@@ -0,0 +1,19 @@
+# frozen_string_literal: true
+
+module Net
+ class IMAP
+ module SASL
+
+ # Authenticator for the "+SCRAM-SHA-1+" SASL mechanism, defined in
+ # RFC5802[https://tools.ietf.org/html/rfc5802].
+ #
+ # Uses the "SHA-1" digest algorithm from OpenSSL::Digest.
+ #
+ # See ScramAuthenticator.
+ class ScramSHA1Authenticator < ScramAuthenticator
+ digest_algorithm "SHA-1"
+ end
+
+ end
+ end
+end
diff --git a/lib/net/imap/sasl/scram_sha256_authenticator.rb b/lib/net/imap/sasl/scram_sha256_authenticator.rb
new file mode 100644
index 00000000..0e8b57c4
--- /dev/null
+++ b/lib/net/imap/sasl/scram_sha256_authenticator.rb
@@ -0,0 +1,19 @@
+# frozen_string_literal: true
+
+module Net
+ class IMAP
+ module SASL
+
+ # Authenticator for the "+SCRAM-SHA-256+" SASL mechanism, defined in
+ # RFC7677[https://tools.ietf.org/html/rfc7677].
+ #
+ # Uses the "SHA-256" digest algorithm from OpenSSL::Digest.
+ #
+ # See ScramAuthenticator.
+ class ScramSHA256Authenticator < ScramAuthenticator
+ digest_algorithm "SHA-256"
+ end
+
+ end
+ end
+end
diff --git a/test/net/imap/test_imap_authenticators.rb b/test/net/imap/test_imap_authenticators.rb
index 34d7918a..85d315d3 100644
--- a/test/net/imap/test_imap_authenticators.rb
+++ b/test/net/imap/test_imap_authenticators.rb
@@ -40,6 +40,92 @@ def test_plain_no_null_chars
assert_raise(ArgumentError) { plain("u", "p", authzid: "bad\0authz") }
end
+ # ----------------------
+ # SCRAM-SHA-1
+ # SCRAM-SHA-256
+ # SCRAM-SHA-* (etc)
+ # ----------------------
+
+ def test_scram_sha1_authenticator_matches_mechanism
+ authenticator = Net::IMAP::SASL.authenticator("SCRAM-SHA-1", "user", "pass")
+ assert_kind_of(Net::IMAP::SASL::ScramAuthenticator, authenticator)
+ assert_kind_of(Net::IMAP::SASL::ScramSHA1Authenticator, authenticator)
+ end
+
+ def test_scram_sha256_authenticator_matches_mechanism
+ authenticator = Net::IMAP::SASL.authenticator("SCRAM-SHA-256", "user", "pass")
+ assert_kind_of(Net::IMAP::SASL::ScramAuthenticator, authenticator)
+ assert_kind_of(Net::IMAP::SASL::ScramSHA256Authenticator, authenticator)
+ end
+
+ def scram_sha1(*args, **kwargs, &block)
+ Net::IMAP::SASL.authenticator("SCRAM-SHA-1", *args, **kwargs, &block)
+ end
+
+ def scram_sha256(*args, **kwargs, &block)
+ Net::IMAP::SASL.authenticator("SCRAM-SHA-256", *args, **kwargs, &block)
+ end
+
+ def test_scram_sha1_authenticator
+ authenticator = scram_sha1("user", "pencil",
+ cnonce: "fyko+d2lbbFgONRv9qkxdawL")
+ # n = no channel binding
+ # a = authzid
+ # n = authcid
+ # r = random nonce (client)
+ assert_equal("n,,n=user,r=fyko+d2lbbFgONRv9qkxdawL",
+ authenticator.process(nil))
+ refute authenticator.done?
+ assert_equal(
+ # c = b64 of gs2 header and channel binding data
+ # r = random nonce (client + server)
+ # p = b64 client proof
+ # s = salt
+ # i = iteration count
+ "c=biws," \
+ "r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j," \
+ "p=v0X8v3Bz2T0CJGbJQyF0X+HI4Ts=",
+ authenticator.process(
+ "r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j," \
+ "s=QSXCR+Q6sek8bf92," \
+ "i=4096")
+ )
+ refute authenticator.done?
+ assert_empty authenticator.process("v=rmF9pqV8S7suAoZWja4dJRkFsKQ=")
+ assert authenticator.done?
+ end
+
+ def test_scram_sha256_authenticator
+ authenticator = scram_sha256("user", "pencil",
+ cnonce: "rOprNGfwEbeRWgbNEkqO")
+ # n = no channel binding
+ # a = authzid
+ # n = authcid
+ # r = random nonce (client)
+ assert_equal("n,,n=user,r=rOprNGfwEbeRWgbNEkqO",
+ authenticator.process(nil))
+ refute authenticator.done?
+ assert_equal(
+ # c = b64 of gs2 header and channel binding data
+ # r = random nonce (client + server)
+ # p = b64 client proof
+ # s = salt
+ # i = iteration count
+ "c=biws," \
+ "r=rOprNGfwEbeRWgbNEkqO%hvYDpWUa2RaTCAfuxFIlj)hNlF$k0," \
+ "p=dHzbZapWIk4jUhN+Ute9ytag9zjfMHgsqmmiz7AndVQ=",
+ authenticator.process(
+ "r=rOprNGfwEbeRWgbNEkqO%hvYDpWUa2RaTCAfuxFIlj)hNlF$k0," \
+ "s=W22ZaJ0SNY7soEsUEjb6gQ==," \
+ "i=4096")
+ )
+ refute authenticator.done?
+ assert_empty authenticator.process(
+ "v=6rriTRBi23WpRR/wtup+mMhUZUn/dB5nLTJRsjl95G4="
+ )
+ assert authenticator.done?
+ end
+
# ----------------------
# OAUTHBEARER
# ----------------------