Skip to content

Commit

Permalink
🔒 Add secret alias to relevant SASL mechanisms
Browse files Browse the repository at this point in the history
`secret` is similar to `username`: a convenient alias with ambiguous
semantics.  Adding it is reasonable, but I probably wouldn't add it only
for `Net::IMAP`.  However, `Net::SMTP` uses it, and probably others
would find it useful too.
  • Loading branch information
nevans committed Oct 20, 2023
1 parent ca7f3c3 commit 1cb0c17
Show file tree
Hide file tree
Showing 8 changed files with 46 additions and 14 deletions.
4 changes: 2 additions & 2 deletions lib/net/imap/sasl/cram_md5_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -16,15 +16,15 @@
class Net::IMAP::SASL::CramMD5Authenticator
def initialize(user = nil, pass = nil,
authcid: nil, username: nil,
password: nil,
password: nil, secret: nil,
warn_deprecation: true,
**)
if warn_deprecation
warn "WARNING: CRAM-MD5 mechanism is deprecated." # TODO: recommend SCRAM
end
require "digest/md5"
@user = authcid || username || user
@password = password || pass
@password = password || secret || pass
@done = false
end

Expand Down
4 changes: 2 additions & 2 deletions lib/net/imap/sasl/digest_md5_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -69,11 +69,11 @@ class Net::IMAP::SASL::DigestMD5Authenticator
# Any other keyword arguments are silently ignored.
def initialize(user = nil, pass = nil, authz = nil,
username: nil, password: nil, authzid: nil,
authcid: nil,
authcid: nil, secret: nil,
warn_deprecation: true, **)
username = authcid || username || user or
raise ArgumentError, "missing username (authcid)"
password ||= pass or raise ArgumentError, "missing password"
password ||= secret || pass or raise ArgumentError, "missing password"
authzid ||= authz
if warn_deprecation
warn "WARNING: DIGEST-MD5 SASL mechanism was deprecated by RFC6331."
Expand Down
4 changes: 2 additions & 2 deletions lib/net/imap/sasl/login_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -25,14 +25,14 @@ class Net::IMAP::SASL::LoginAuthenticator

def initialize(user = nil, pass = nil,
authcid: nil, username: nil,
password: nil,
password: nil, secret: nil,
warn_deprecation: true,
**)
if warn_deprecation
warn "WARNING: LOGIN SASL mechanism is deprecated. Use PLAIN instead."
end
@user = authcid || username || user
@password = password || pass
@password = password || secret || pass
@state = STATE_USER
end

Expand Down
7 changes: 5 additions & 2 deletions lib/net/imap/sasl/oauthbearer_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,7 @@ class OAuthBearerAuthenticator < OAuthAuthenticator

# An OAuth 2.0 bearer token. See {RFC-6750}[https://www.rfc-editor.org/rfc/rfc6750]
attr_reader :oauth2_token
alias secret oauth2_token

# :call-seq:
# new(oauth2_token, **options) -> authenticator
Expand Down Expand Up @@ -173,10 +174,12 @@ class OAuthBearerAuthenticator < OAuthAuthenticator
# noting that <b><em>application protocols are allowed to
# require</em></b> #authzid (<em>or other parameters, such as</em> #host
# _or_ #port) <b><em>as are specific server implementations</em></b>.
def initialize(arg1 = nil, arg2 = nil, oauth2_token: nil, **args, &blk)
def initialize(arg1 = nil, arg2 = nil,
oauth2_token: nil, secret: nil,
**args, &blk)
username, oauth2_token_arg = arg2.nil? ? [nil, arg1] : [arg1, arg2]
super(username: username, **args, &blk)
@oauth2_token = oauth2_token || oauth2_token_arg or
@oauth2_token = oauth2_token || secret || oauth2_token_arg or
raise ArgumentError, "missing oauth2_token"
end

Expand Down
5 changes: 3 additions & 2 deletions lib/net/imap/sasl/plain_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ class Net::IMAP::SASL::PlainAuthenticator

# A password or passphrase that matches the #username.
attr_reader :password
alias secret password

# 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
Expand Down Expand Up @@ -64,11 +65,11 @@ class Net::IMAP::SASL::PlainAuthenticator
#
# Any other keyword parameters are quietly ignored.
def initialize(user = nil, pass = nil,
authcid: nil,
authcid: nil, secret: nil,
username: nil, password: nil, authzid: nil, **)
username ||= authcid || user or
raise ArgumentError, "missing username (authcid)"
password ||= pass or raise ArgumentError, "missing password"
password ||= secret || pass or raise ArgumentError, "missing password"
raise ArgumentError, "username contains NULL" if username.include?(NULL)
raise ArgumentError, "password contains NULL" if password.include?(NULL)
raise ArgumentError, "authzid contains NULL" if authzid&.include?(NULL)
Expand Down
5 changes: 3 additions & 2 deletions lib/net/imap/sasl/scram_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -80,13 +80,13 @@ class ScramAuthenticator
def initialize(username_arg = nil, password_arg = nil,
authcid: nil, username: nil,
authzid: nil,
password: nil,
password: nil, secret: nil,
min_iterations: 4096, # see both RFC5802 and RFC7677
cnonce: nil, # must only be set in tests
**options)
@username = username || username_arg || authcid or
raise ArgumentError, "missing username (authcid)"
@password = password || password_arg or
@password = password || secret || password_arg or
raise ArgumentError, "missing password"
@authzid = authzid

Expand All @@ -109,6 +109,7 @@ def initialize(username_arg = nil, password_arg = nil,

# A password or passphrase that matches the #username.
attr_reader :password
alias secret password

# Authorization identity: an identity to act as or on behalf of. The
# identity form is application protocol specific. If not provided or
Expand Down
5 changes: 3 additions & 2 deletions lib/net/imap/sasl/xoauth2_authenticator.rb
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@ class Net::IMAP::SASL::XOAuth2Authenticator
# An OAuth2 access token which has been authorized with the appropriate OAuth2
# scopes to use the service for #username.
attr_reader :oauth2_token
alias secret oauth2_token

# :call-seq:
# new(username, oauth2_token, **) -> authenticator
Expand All @@ -68,10 +69,10 @@ class Net::IMAP::SASL::XOAuth2Authenticator
#
# Any other keyword parameters are quietly ignored.
def initialize(user = nil, token = nil, username: nil, oauth2_token: nil,
authzid: nil, **)
authzid: nil, secret: nil, **)
@username = authzid || username || user or
raise ArgumentError, "missing username (authzid)"
@oauth2_token = oauth2_token || token or
@oauth2_token = oauth2_token || secret || token or
raise ArgumentError, "missing oauth2_token"
@done = false
end
Expand Down
26 changes: 26 additions & 0 deletions test/net/imap/test_imap_authenticators.rb
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,10 @@ def test_plain_kw_params
"zid\0cid\0p",
plain(authcid: "cid", password: "p", authzid: "zid").process(nil)
)
assert_equal(
"zid\0cid\0p",
plain(username: "cid", secret: "p", authzid: "zid").process(nil)
)
end

def test_plain_username_kw_sets_both_authcid_and_authzid
Expand Down Expand Up @@ -96,6 +100,15 @@ def test_oauthbearer_response
oauthbearer("mF_9.B5f-4.1JqM", authzid: "[email protected]",
host: "server.example.com", port: 587).process(nil)
)
assert_equal(
"n,[email protected],\1host=server.example.com\1port=587\1" \
"auth=Bearer sssssssss\1\1",
oauthbearer(secret: "sssssssss", username: "[email protected]",
host: "server.example.com", port: 587).process(nil)
)
assert_equal(
"n,a=user,\1auth=Bearer tok\1\1", oauthbearer("user", "tok").process(nil)
)
end

# ----------------------
Expand Down Expand Up @@ -153,6 +166,15 @@ def test_scram_sha1_authenticator
assert authenticator.done?
end

def test_scram_kwargs
authenticator = scram_sha1(authcid: "user", password: "pass")
assert_equal "user", authenticator.authcid
assert_equal "pass", authenticator.password
authenticator = scram_sha1(username: "user", secret: "pass")
assert_equal "user", authenticator.authcid
assert_equal "pass", authenticator.password
end

def test_scram_sha256_authenticator
authenticator = scram_sha256("user", "pencil",
cnonce: "rOprNGfwEbeRWgbNEkqO")
Expand Down Expand Up @@ -210,6 +232,10 @@ def test_xoauth2_kwargs
"user=user\1auth=Bearer kwarg\1\1",
xoauth2(username: "user", oauth2_token: "kwarg").process(nil)
)
assert_equal(
"user=user\1auth=Bearer kwarg\1\1",
xoauth2(authzid: "user", secret: "kwarg").process(nil)
)
end

def test_xoauth2_supports_initial_response
Expand Down

0 comments on commit 1cb0c17

Please sign in to comment.