org.apache.accumulo.server.rpc.UGIAssumingProcessor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of accumulo-server-base Show documentation
Show all versions of accumulo-server-base Show documentation
A common base library for Apache Accumulo servers.
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.accumulo.server.rpc;
import java.io.IOException;
import javax.security.sasl.SaslServer;
import org.apache.accumulo.core.rpc.SaslConnectionParams.SaslMechanism;
import org.apache.hadoop.security.UserGroupInformation;
import org.apache.thrift.TException;
import org.apache.thrift.TProcessor;
import org.apache.thrift.protocol.TProtocol;
import org.apache.thrift.transport.TSaslServerTransport;
import org.apache.thrift.transport.TTransport;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Processor that pulls the SaslServer object out of the transport, and assumes the remote user's
* UGI before calling through to the original processor.
*
* This is used on the server side to set the UGI for each specific call.
*
* Lifted from Apache Hive 0.14
*/
public class UGIAssumingProcessor implements TProcessor {
private static final Logger log = LoggerFactory.getLogger(UGIAssumingProcessor.class);
public static final ThreadLocal rpcPrincipal = new ThreadLocal<>();
public static final ThreadLocal rpcMechanism = new ThreadLocal<>();
private final TProcessor wrapped;
private final UserGroupInformation loginUser;
public UGIAssumingProcessor(TProcessor wrapped) {
this.wrapped = wrapped;
try {
this.loginUser = UserGroupInformation.getLoginUser();
} catch (IOException e) {
log.error("Failed to obtain login user", e);
throw new RuntimeException("Failed to obtain login user", e);
}
}
/**
* The principal of the user who authenticated over SASL.
*/
public static String rpcPrincipal() {
return rpcPrincipal.get();
}
public static ThreadLocal getRpcPrincipalThreadLocal() {
return rpcPrincipal;
}
public static SaslMechanism rpcMechanism() {
return rpcMechanism.get();
}
@Override
public void process(final TProtocol inProt, final TProtocol outProt) throws TException {
TTransport trans = inProt.getTransport();
if (!(trans instanceof TSaslServerTransport)) {
throw new TException("Unexpected non-SASL transport " + trans.getClass() + ": " + trans);
}
TSaslServerTransport saslTrans = (TSaslServerTransport) trans;
SaslServer saslServer = saslTrans.getSaslServer();
String endUser = saslServer.getAuthorizationID();
SaslMechanism mechanism;
try {
mechanism = SaslMechanism.get(saslServer.getMechanismName());
} catch (Exception e) {
log.error("Failed to process RPC with SASL mechanism {}", saslServer.getMechanismName());
throw e;
}
switch (mechanism) {
case GSSAPI:
UserGroupInformation clientUgi = UserGroupInformation.createProxyUser(endUser, loginUser);
final String remoteUser = clientUgi.getUserName();
try {
// Set the principal in the ThreadLocal for access to get authorizations
rpcPrincipal.set(remoteUser);
wrapped.process(inProt, outProt);
} finally {
// Unset the principal after we're done using it just to be sure that it's not incorrectly
// used in the same thread down the line.
rpcPrincipal.set(null);
}
break;
case DIGEST_MD5:
// The CallbackHandler, after deserializing the TokenIdentifier in the name, has already
// updated
// the rpcPrincipal for us. We don't need to do it again here.
try {
rpcMechanism.set(mechanism);
wrapped.process(inProt, outProt);
} finally {
// Unset the mechanism after we're done using it just to be sure that it's not incorrectly
// used in the same thread down the line.
rpcMechanism.set(null);
}
break;
default:
throw new IllegalArgumentException("Cannot process SASL mechanism " + mechanism);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy