Current state: "Accepted"
Discussion thread: here
Please keep the discussion on the mailing list rather than commenting on the wiki (wiki discussions get unwieldy fast).
Kafka currently supports SASL authentication using SASL/PLAIN mechanism and KIP-84 addresses the addition of SASL/SCRAM. Credential verification in SASL/PLAIN servers is currently based on hard-coded credentials in JAAS configuration similar to Digest-MD5 configuration in Zookeeper. This is useful as a sample, but not suitable for production use since clear passwords are stored on disk. KIP-84 added SCRAM mechanism with Zookeeper as the password store. In production installations where Zookeeper is not secure (e.g. Kafka on the cloud), an alternative password store may be required.
With the current Kafka implementation, the entire
SaslServer implementation needs to be replaced to enable new credential providers. It will be useful to add an extension point for SASL that enables just the credential providers to be replaced with a custom implementation. This should be done in a consistent way for all SASL mechanisms.
This KIP proposes to enable customization of SASL server and clients using configurable callback handlers. Configurable callback handlers for SASL/PLAIN and SASL/SCRAM will enable credential providers to be replaced in a simple and consistent way. In addition to this, configurable callback handlers for both server and clients make it easier to configure new or custom SASL mechanisms that are not implemented in Kafka.
To enable custom SASL mechanisms to be plugged in easily, the login interface for clients and servers will also be made configurable. This will enable custom logic to be added for new SASL mechanisms. This is particularly useful for mechanisms like Kerberos that require periodic token refresh.
Client callback handler class (for clients and brokers using SASL for inter-broker communication)
- Doc: The fully qualified name of a Sasl client callback handler class that implements the
- Default: null (by default, the appropriate internal default callback handlers for the mechanism will be used)
Server callback handler classes (for brokers only)
- Doc: The fully qualified name of a SASL server callback handler class that implements the
AuthenticateCallbackHandlerinterface. The config name must be prefixed by the listener prefix and mechanism name in lower case. For example,
- Default: null (by default, the appropriate internal default callback handlers for each mechanism will be used)
Login class (for clients and brokers)
- Doc: A class that implements the
org.apache.kafka.common.security.auth.Logininterface. For brokers, the config name must be prefixed by the listener prefix and mechanism name in lower case. For example,
listener.name.sasl_ssl.plain.sasl.login.class=com.example.PlainServerLogin for brokers and
sasl.login.class=com.example.KerberosClientLogin for clients.
- Default: null (by default, the internal class
KerberosLoginwill be used if Kerberos is enabled on the listener and
Login callback handler class (for clients and brokers)
- Doc: The fully qualified name of a Sasl login callback handler class that implements the
interface. For servers, the config name must be prefixed by the listener prefix and mechanism name in lower case. For example,
listener.name.sasl_ssl.plain.sasl.login.callback.handler.class=com.example.PlainLoginCallbackHandler for brokers and
sasl.login.callback.handler.class=com.example.PlainLoginCallbackHandler for clients.
- Default: null (by default, the internal class
AbstractLogin.DefaultLoginCallbackHandlerwill be used).
The callback handler interface
AuthenticateCallbackHandler will extend the standard
javax.security.auth.callback.CallbackHandler interface, enabling the handler to be passed directly to
SaslServer/SaslClient implementations. The callback handler configured for a mechanism must include the callbacks as described below:
- If using a
SaslServer/SaslClientimplementation from the JRE, the callbacks required for the mechanism are described in the Java SASL reference.
- When using the
SaslServer/SaslClientimplementation included in Kafka (PLAIN or SCRAM), the callback defined below for the SASL mechanism must be handled.
- Applications using custom implementations of
SaslServer/SaslClientmay define their own callbacks.
Callback handlers which require additional options at runtime (eg. URL of a credential server) may include arguments in the JAAS configuration using the config file or
sasl.jaas.config property (KIP-85). This is similar to the way keytab location is configured for GSSAPI. Client callback handlers can retrieve
Subject.getSubject(AccessController.getContext()) to obtain credentials populated by the login module.
SASL/PLAIN Server Callbacks
SASL/PLAIN servers using the SaslServer implementation included in Kafka must handle
PlainAuthenticateCallback. The username for authentication is provided in
NameCallback similar to other mechanisms in the JRE (eg. Digest-MD5). The password provided by the client during SASL authentication is provided in
PlainAuthenticateCallback. The callback handler sets authenticated flag in the callback after verifying username and password.
SASL/PLAIN server callback handler is requested to handle
PlainAuthenticateCallback in that order. See the sample SASL/PLAIN callback handler for an example.
SASL/SCRAM Server Callbacks
SASL/SCRAM servers using the
SaslServer implementation included in Kafka must handle
ScramCredentialCallback. The username for authentication is provided in
NameCallback similar to other mechanisms in the JRE (eg. Digest-MD5). The callback handler must return SCRAM credential for the user if credentials are available for the username for the configured SCRAM mechanism.
SASL/SCRAM server callback handler is requested to handle
in that order. See the sample SASL/SCRAM callback handler for an example.
SASL/PLAIN callbacks enable authentication of passwords using an external authentication server without requiring Kafka to have knowledge of actual passwords. The callback checks if the (user, password) combination provided during SASL/PLAIN exchange is valid. SASL/SCRAM callbacks do require the salted credentials to perform the SCRAM authentication and hence the credentials are requested in the callback.
LoginManager implementation in Kafka uses implementations of the
Login interface to create a login instance for authentication. This login class instantiates a
javax.security.auth.login.LoginContext and invokes
LoginContext#login() to authenticate a
Subject. Two different login class implementations are used by Kafka,
KerberosLogin if GSSAPI is enabled and
DefaultLogin for all other mechanisms. This KIP proposes to make the Login class configurable so that new mechanisms that require custom logic similar to Kerberos may be added these easily.
ChannelBuilder will create an instance of each configured callback handler using the default constructor. For mechanisms without a callback handler override, the existing default callback handlers (
SaslServerCallbackHandler/SaslClientCallbackHandler) or mechanism-specific server handlers for PLAIN/SCRAM will be created. Callback handler instances will be created once for each enabled mechanism in
ChannelBuilder, instead of per-connection. This enables callback handlers using external authentication servers to cache credentials or reuse connections if required.
SaslClientCallbackHandler will be modified to obtain
Subject.getSubject(AccessController.getContext()) to avoid the current per-connection state.
LoginManager will be updated to use the configured login class if provided and use current classes
KerberosLogin as default. LoginManager currently caches static JAAS configs using the login context name as key and dynamic JAAS configs using the full
sasl.jaas.config value as key. The map will be updated to include the existing string (context name/
sasl.jaas.config) combined with the login class, so that a
Login instance is reused only if both the config as well as the
Login class match.
Use an external authentication server for SASL/PLAIN authentication using the SaslServer implementation for PLAIN included in Kafka
Define a new class that implements
AuthenticateCallbackHandler which handles
PlainAuthenticateCallback and add the class to the broker's
sasl.server.callback.handler.class property. A single instance of this callback handler will be created for the broker. The configured callback handler is responsible for validating the password provided by clients and this may use an external authentication server.
Use custom credential store instead of Zookeeper for storing SCRAM credentials
Set broker callback handler to a class that implements
AuthenticateCallbackHandler which handles
ScramCredentialCallback. SCRAM credentials from a custom store can be returned by the callback handler.
Use a custom SaslServer implementation for SCRAM
If a custom SaslServer implementation is used instead of the one included in Kafka, the custom implementation may require a different set of callbacks. A callback handler for these callbacks may be specified in
Configure a new mechanism not included in Kafka using custom SaslServer/SaslClient
A handler that handles any callbacks required for these server/client implementations may be specified in
for brokers and clients respectively.
Configure a new mechanism using the implementation provided by the JRE
Callbacks defined for the mechanism in the Java implementation must be handled by custom callback handlers if the behaviour differs from the default callbacks in Kafka.
Configure callbacks for different mechanisms on different listeners in the broker
KIP-103 introduced support for multiple listeners in the broker for the same security protocol. This allows brokers to configure different SASL mechanisms for internal and external traffic. The listener name prefix can be applied to
sasl.server.callback.handler.class to define different callback handlers for each of the listeners.
Add a new SASL mechanism that requires periodic re-login (similar to Kerberos token refresh)
Configure a new
Login class by setting the config
sasl.login.class. The custom login class can periodically log out of the current
LoginContext, instantiate a new
LoginContext and login again using
LoginContext#login(). An example of this is
KerberosLogin in the current Kafka implementation.
Sample Callback Handler for SASL/PLAIN
For custom SASL/PLAIN authentication, override
authenticate() with custom implementation that verifies the given
Sample Callback Handler for SASL/SCRAM
For custom credential store for SCRAM, override
credential() with alternative method that obtains credential from the custom store.
Compatibility, Deprecation, and Migration Plan
- What impact (if any) will there be on existing users?
- If we are changing behavior how will we phase out the older behavior?
Existing behaviour will be retained as default
Existing integration and system tests will test the default behaviour. Additional unit and integration tests will be added to test the new configuration:
- Test that PLAIN credential provider can be replaced
- Test that SCRAM password store can be replaced
- Test that new mechanisms not included in Kafka can be run with custom callback handlers and custom Login class.
Define a new credential provider interface instead of using CallbackHandler
The format of credentials required for each mechanism is different. For SASL/PLAIN, the proposed callback handler can be used with external authentication servers which validate passwords without allowing the broker to retrieve password for a user. For SASL/SCRAM, the hashed, salted credentials required for the mechanism are provided to the broker. Different credential providers can be defined to capture these differences, which may make the interface slightly simpler in these two cases compared to callback handlers. But the use of standard Java
CallbackHandler interface is more flexible and future-proof since it is the interface used by
SaslServer/SaslClient implementations to obtain application-specific data.