All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.firebirdsql.gds.ng.wire.auth.ClientAuthBlock Maven / Gradle / Ivy

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * Firebird Open Source JavaEE Connector - JDBC Driver
 *
 * Distributable under LGPL license.
 * You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * LGPL License for more details.
 *
 * This file was created by members of the firebird development team.
 * All individual contributions remain the Copyright (C) of those
 * individuals.  Contributors to this file are either listed here or
 * can be obtained from a source control history command.
 *
 * All rights reserved.
 */
package org.firebirdsql.gds.ng.wire.auth;

import org.firebirdsql.gds.*;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.gds.ng.IAttachProperties;
import org.firebirdsql.logging.Logger;
import org.firebirdsql.logging.LoggerFactory;

import java.io.IOException;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.sql.SQLException;
import java.util.*;
import java.util.regex.Pattern;

import static org.firebirdsql.gds.impl.wire.WireProtocolConstants.*;

/**
 * Manages client authentication with multiple pluginProviders.
 * 

* This is similar to the {@code ClntAuthBlock} in Firebird. For ease of understanding, and correlating with the * Firebird implementation, this uses the same (or very similar) terms and structure (if it makes sense). *

* * @author Mark Rotteveel * @since 3.0 */ public final class ClientAuthBlock { private static final Logger log = LoggerFactory.getLogger(ClientAuthBlock.class); private static final Pattern AUTH_PLUGIN_LIST_SPLIT = Pattern.compile("[ \t,;]+"); private static final String DEFAULT_AUTH_PLUGINS = "Srp256,Srp"; private static final Map PLUGIN_MAPPING = getAvailableAuthenticationPlugins(); private final IAttachProperties attachProperties; private List pluginProviders; private final Set serverPlugins = new LinkedHashSet<>(); private AuthenticationPlugin currentPlugin; private boolean authComplete; private boolean firstTime = true; public ClientAuthBlock(IAttachProperties attachProperties) throws SQLException { this.attachProperties = attachProperties; resetClient(null); } public String getLogin() { return attachProperties.getUser(); } public String getNormalizedLogin() { return normalizeLogin(getLogin()); } public String getPassword() { return attachProperties.getPassword(); } public boolean isAuthComplete() { return authComplete; } public void setAuthComplete(boolean authComplete) { this.authComplete = authComplete; } public String getCurrentPluginName() { return currentPlugin != null ? currentPlugin.getName() : null; } /** * @return Comma separated list of available plugins. */ public String getPluginNames() { return getPluginNames(pluginProviders); } public byte[] getClientData() { return currentPlugin != null ? currentPlugin.getClientData() : null; } public void setFirstTime(boolean firstTime) { this.firstTime = firstTime; } public boolean isFirstTime() { return firstTime; } public void authenticateStep0() throws SQLException { Iterator providerIterator = pluginProviders.iterator(); while (providerIterator.hasNext()) { AuthenticationPluginSpi provider = providerIterator.next(); AuthenticationPlugin plugin = provider.createPlugin(); log.debug("Trying authentication plugin " + plugin); try { switch (plugin.authenticate(this)) { case AUTH_SUCCESS: case AUTH_MORE_DATA: currentPlugin = plugin; return; case AUTH_CONTINUE: providerIterator.remove(); break; } } catch (SQLException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_login).cause(ex).toFlatSQLException(); } } } public void resetClient(byte[] serverInfo) throws SQLException { if (serverInfo != null) { if (currentPlugin != null && currentPlugin.hasServerData()) { // We should not change plugins iterator now return; } ClumpletReader serverList = new ClumpletReader(ClumpletReader.Kind.UnTagged, serverInfo); if (serverList.find(TAG_KNOWN_PLUGINS)) { String serverPluginNames = serverList.getString(StandardCharsets.US_ASCII); serverPlugins.clear(); serverPlugins.addAll(splitPluginList(serverPluginNames)); } } firstTime = true; currentPlugin = null; pluginProviders = getSupportedPluginProviders(); if (!serverPlugins.isEmpty()) { LinkedList mergedProviderList = new LinkedList<>(); for (AuthenticationPluginSpi clientProvider : pluginProviders) { if (serverPlugins.contains(clientProvider.getPluginName())) { mergedProviderList.add(clientProvider); } } if (mergedProviderList.isEmpty()) { throw new FbExceptionBuilder().exception(ISCConstants.isc_login) .exception(ISCConstants.isc_random).messageParameter("No matching plugins on server") .toFlatSQLException(); } pluginProviders = mergedProviderList; } } public void setServerData(byte[] serverData) { if (currentPlugin == null) { // TODO Check if this is valid to ignore log.debug("Received server data without current plugin"); } else { currentPlugin.setServerData(serverData); } } private static String getPluginNames(List pluginProviders) { if (pluginProviders.size() == 0) { return null; } StringBuilder names = new StringBuilder(); for (int idx = 0; idx < pluginProviders.size(); idx++) { if (idx > 0) { names.append(','); } names.append(pluginProviders.get(idx).getPluginName()); } return names.toString(); } public void writePluginDataTo(OutputStream userId) throws IOException { final String user = getLogin(); if (user != null) { final byte[] loginBytes = user.getBytes(StandardCharsets.UTF_8); userId.write(CNCT_login); int loginLength = Math.min(loginBytes.length, 255); userId.write(loginLength); userId.write(loginBytes, 0, loginLength); } final String pluginName = getCurrentPluginName(); if (pluginName != null) { userId.write(CNCT_plugin_name); final byte[] pluginNameBytes = pluginName.getBytes(StandardCharsets.UTF_8); userId.write(pluginNameBytes.length); userId.write(pluginNameBytes, 0, pluginNameBytes.length); } final String pluginList = getPluginNames(); if (pluginList != null) { userId.write(CNCT_plugin_list); final byte[] pluginListBytes = pluginList.getBytes(StandardCharsets.UTF_8); userId.write(pluginListBytes.length); userId.write(pluginListBytes, 0, pluginListBytes.length); } if (currentPlugin != null) { byte[] specificDataBytes = currentPlugin.getClientData(); if (specificDataBytes != null) { addMultiPartConnectParameter(userId, CNCT_specific_data, specificDataBytes); } } } private void addMultiPartConnectParameter(OutputStream userId, int paramType, byte[] specificDataBytes) throws IOException { // write specific data int remaining = specificDataBytes.length; int position = 0; int step = 0; while (remaining > 0) { userId.write(paramType); int toWrite = Math.min(remaining, 254); userId.write(toWrite + 1); userId.write(step++); userId.write(specificDataBytes, position, toWrite); remaining -= toWrite; position += toWrite; } } public boolean switchPlugin(String pluginName) { if (hasPlugin() && Objects.equals(getCurrentPluginName(), pluginName)) { return false; } for (Iterator iterator = pluginProviders.iterator(); iterator.hasNext(); ) { AuthenticationPluginSpi pluginProvider = iterator.next(); if (pluginProvider.getPluginName().equals(pluginName)) { currentPlugin = pluginProvider.createPlugin(); return true; } iterator.remove(); } return false; } public boolean hasPlugin() { return currentPlugin != null; } public AuthenticationPlugin.AuthStatus authenticate() throws SQLException { return currentPlugin.authenticate(this); } public void authFillParametersBlock(ConnectionParameterBuffer pb) throws SQLException { Iterator providerIterator = pluginProviders.iterator(); while (providerIterator.hasNext()) { AuthenticationPluginSpi provider = providerIterator.next(); AuthenticationPlugin plugin; if (hasPlugin() && provider.getPluginName().equals(getCurrentPluginName())) { plugin = currentPlugin; } else { plugin = provider.createPlugin(); } log.debug("Trying authentication plugin " + plugin); try { switch (plugin.authenticate(this)) { case AUTH_SUCCESS: case AUTH_MORE_DATA: log.debug("Trying authentication plugin " + plugin + " is OK"); currentPlugin = plugin; cleanParameterBuffer(pb); extractDataToParameterBuffer(pb); return; case AUTH_CONTINUE: providerIterator.remove(); break; } } catch (SQLException ex) { throw new FbExceptionBuilder().exception(ISCConstants.isc_login).cause(ex).toFlatSQLException(); } log.debug(String.format("try next plugin, %s skipped", plugin)); } } /** * TODO Need to handle this differently * @return {@code true} if the encryption is supported * @throws SQLException * If it is impossible to determine if encryption is supported (eg there is no current auth plugin) */ public boolean supportsEncryption() throws SQLException { if (currentPlugin == null) { throw new SQLException("No authentication plugin available"); } return currentPlugin.generatesSessionKey(); } /** * @return Session key * @throws SQLException * If a session key cannot be provided */ public byte[] getSessionKey() throws SQLException { if (currentPlugin == null) { throw new SQLException("No authentication plugin available"); } return currentPlugin.getSessionKey(); } /** * Normalizes a login by uppercasing unquoted user names, or stripping and unescaping (double) quoted user names. * * @param login Login to process * @return Normalized login */ static String normalizeLogin(String login) { if (login == null || login.isEmpty()) { return login; } // Contrary to Firebird, check if login is enclosed in double quotes, not just starting with a double quote if (login.length() > 2 && login.charAt(0) == '"' && login.charAt(login.length() - 1) == '"') { return normalizeQuotedLogin(login); } return login.toUpperCase(Locale.ROOT); } private static String normalizeQuotedLogin(String login) { final StringBuilder sb = new StringBuilder(login.length() - 2); sb.append(login, 1, login.length() - 1); for (int idx = 0; idx < sb.length(); idx++) { // Double double quotes ("") escape a double quote in a quoted string if (sb.charAt(idx) == '"') { // Strip double quote escape sb.deleteCharAt(idx); if (idx < sb.length() && sb.charAt(idx) == '"') { // Retain escaped double quote idx += 1; } else { // The character after escape is not a double quote, we terminate the conversion and truncate. // Firebird does this as well (see common/utils.cpp#dpbItemUpper) sb.setLength(idx); return sb.toString(); } } } return sb.toString(); } private void extractDataToParameterBuffer(ConnectionParameterBuffer pb) { byte[] clientData = getClientData(); if (clientData == null || clientData.length == 0) { return; } String pluginName = getCurrentPluginName(); ParameterTagMapping tagMapping = pb.getTagMapping(); if (firstTime) { if (pluginName != null) { pb.addArgument(tagMapping.getAuthPluginNameTag(), pluginName); } pb.addArgument(tagMapping.getAuthPluginListTag(), getPluginNames()); firstTime = false; log.debug("first time - added plugName & pluginList"); } pb.addArgument(tagMapping.getSpecificAuthDataTag(), clientData); log.debug(String.format("Added %d bytes of spec data with tag isc_dpb_specific_auth_data", clientData.length)); } private void cleanParameterBuffer(ConnectionParameterBuffer pb) { ParameterTagMapping tagMapping = pb.getTagMapping(); pb.removeArgument(tagMapping.getPasswordTag()); pb.removeArgument(tagMapping.getEncryptedPasswordTag()); pb.removeArgument(tagMapping.getTrustedAuthTag()); } private List getSupportedPluginProviders() throws SQLException { List requestedPluginNames = getRequestedPluginNames(); List pluginProviders = new ArrayList<>(requestedPluginNames.size()); for (String pluginName : requestedPluginNames) { AuthenticationPluginSpi pluginSpi = PLUGIN_MAPPING.get(pluginName); if (pluginSpi != null) { pluginProviders.add(pluginSpi); } else { log.warn("No authentication plugin available with name " + pluginName); } } if (pluginProviders.isEmpty()) { throw new FbExceptionBuilder().exception(JaybirdErrorCodes.jb_noKnownAuthPlugins) .messageParameter(requestedPluginNames.toString()) .toFlatSQLException(); } return pluginProviders; } private List getRequestedPluginNames() { String pluginListString = attachProperties.getAuthPlugins(); if (pluginListString == null || pluginListString.isEmpty()) { pluginListString = DEFAULT_AUTH_PLUGINS; } return splitPluginList(pluginListString); } private static List splitPluginList(String pluginList) { return Arrays.asList(AUTH_PLUGIN_LIST_SPLIT.split(pluginList)); } // TODO Move plugin loading to separate class? private static Map getAvailableAuthenticationPlugins() { Map pluginMapping = new HashMap<>(); for (AuthenticationPluginSpi pluginSpi : getAvailableAuthenticationPluginSpis()) { String pluginName = pluginSpi.getPluginName(); if (pluginMapping.containsKey(pluginName)) { log.warn("Authentication plugin provider for " + pluginName + " already registered. Skipping " + pluginSpi.getClass().getName()); continue; } pluginMapping.put(pluginName, pluginSpi); } return Collections.unmodifiableMap(pluginMapping); } private static List getAvailableAuthenticationPluginSpis() { try { ServiceLoader pluginLoader = ServiceLoader.load(AuthenticationPluginSpi.class, ClientAuthBlock.class.getClassLoader()); List pluginList = new ArrayList<>(); // We can't use foreach here, because the plugins are lazily loaded, which might trigger a ServiceConfigurationError Iterator pluginIterator = pluginLoader.iterator(); int retry = 0; while (retry < 2) { try { while (pluginIterator.hasNext()) { try { AuthenticationPluginSpi plugin = pluginIterator.next(); pluginList.add(plugin); } catch (Exception | ServiceConfigurationError e) { log.warn("Can't register plugin, see debug level for more information (skipping): " + e); log.debug("Failed to load plugin with exception", e); } } break; } catch (ServiceConfigurationError e) { log.error("Error finding next AuthenticationPluginSpi", e); retry++; } } if (!pluginList.isEmpty()) { return pluginList; } else { log.warn("No authentication plugins loaded through service loader, falling back to default list"); } } catch (Exception e) { String message = "Unable to load authentication plugins through ServiceLoader, using fallback list"; log.warn(message + ": " + e + "; see debug level for stacktrace"); log.debug(message, e); } return loadFallbackPluginProviders(ClientAuthBlock.class.getClassLoader()); } private static List loadFallbackPluginProviders(ClassLoader classLoader) { List fallbackPluginProviders = new ArrayList<>(6); for (String providerName : new String[] { "org.firebirdsql.gds.ng.wire.auth.legacy.LegacyAuthenticationPluginSpi", "org.firebirdsql.gds.ng.wire.auth.srp.SrpAuthenticationPluginSpi", "org.firebirdsql.gds.ng.wire.auth.srp.Srp224AuthenticationPluginSpi", "org.firebirdsql.gds.ng.wire.auth.srp.Srp256AuthenticationPluginSpi", "org.firebirdsql.gds.ng.wire.auth.srp.Srp384AuthenticationPluginSpi", "org.firebirdsql.gds.ng.wire.auth.srp.Srp512AuthenticationPluginSpi", }) { try { Class clazz = Class.forName(providerName, true, classLoader); AuthenticationPluginSpi provider = (AuthenticationPluginSpi) clazz.getDeclaredConstructor().newInstance(); fallbackPluginProviders.add(provider); } catch (ReflectiveOperationException e) { log.warn("Could not load plugin provider (see debug level for details) " + providerName + ", reason: " + e); log.debug("Failed to load plugin provider " + providerName + " with exception", e); } } return fallbackPluginProviders; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy