com.bigstep.datalake.TokenAspect.java Source code

Java tutorial

Introduction

Here is the source code for com.bigstep.datalake.TokenAspect.java

Source

/**
 * 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
 * <p>
 * http://www.apache.org/licenses/LICENSE-2.0
 * <p>
 * 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 com.bigstep.datalake;

import com.google.common.annotations.VisibleForTesting;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.DelegationTokenRenewer;
import org.apache.hadoop.fs.DelegationTokenRenewer.Renewable;
import org.apache.hadoop.fs.FileSystem;
import org.apache.hadoop.hdfs.HAUtil;
import org.apache.hadoop.hdfs.security.token.delegation.DelegationTokenIdentifier;
import org.apache.hadoop.hdfs.web.HftpFileSystem;
import org.apache.hadoop.hdfs.web.HsftpFileSystem;
import org.apache.hadoop.hdfs.web.SWebHdfsFileSystem;
import org.apache.hadoop.hdfs.web.WebHdfsFileSystem;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.net.NetUtils;
import org.apache.hadoop.security.SecurityUtil;
import org.apache.hadoop.security.UserGroupInformation;
import org.apache.hadoop.security.token.Token;
import org.apache.hadoop.security.token.TokenRenewer;
import org.apache.hadoop.security.token.delegation.AbstractDelegationTokenSelector;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.URI;

/**
 * This class implements the aspects that relate to delegation tokens for all
 * HTTP-based file system.
 */
final class TokenAspect<T extends FileSystem & Renewable> {
    private final DTSelecorByKind dtSelector;
    private final T fs;
    private final Log LOG;
    private final Text serviceName;
    private DelegationTokenRenewer.RenewAction<?> action;
    private DelegationTokenRenewer dtRenewer = null;
    private boolean hasInitedToken;

    TokenAspect(T fs, final Text serviceName, final Text kind) {
        this.LOG = LogFactory.getLog(fs.getClass());
        this.fs = fs;
        this.dtSelector = new DTSelecorByKind(kind);
        this.serviceName = serviceName;
    }

    synchronized void ensureTokenInitialized() throws IOException {
        // we haven't inited yet, or we used to have a token but it expired
        if (!hasInitedToken || (action != null && !action.isValid())) {
            //since we don't already have a token, go get one
            Token<?> token = fs.getDelegationToken(null);
            // security might be disabled
            if (token != null) {
                fs.setDelegationToken(token);
                addRenewAction(fs);
                LOG.debug("Created new DT for " + token.getService());
            }
            hasInitedToken = true;
        }
    }

    public synchronized void reset() {
        hasInitedToken = false;
    }

    synchronized void initDelegationToken(UserGroupInformation ugi) {
        Token<?> token = selectDelegationToken(ugi);
        if (token != null) {
            LOG.debug("Found existing DT for " + token.getService());
            fs.setDelegationToken(token);
            hasInitedToken = true;
        }
    }

    synchronized void removeRenewAction() throws IOException {
        if (dtRenewer != null) {
            dtRenewer.removeRenewAction(fs);
        }
    }

    @VisibleForTesting
    Token<DelegationTokenIdentifier> selectDelegationToken(UserGroupInformation ugi) {
        return dtSelector.selectToken(serviceName, ugi.getTokens());
    }

    private synchronized void addRenewAction(final T webhdfs) {
        if (dtRenewer == null) {
            dtRenewer = DelegationTokenRenewer.getInstance();
        }

        action = dtRenewer.addRenewAction(webhdfs);
    }

    /**
     * Callbacks for token management
     */
    interface TokenManagementDelegator {
        void cancelDelegationToken(final Token<?> token) throws IOException;

        long renewDelegationToken(final Token<?> token) throws IOException;
    }

    public static class TokenManager extends TokenRenewer {

        private static String getSchemeByKind(Text kind) {
            if (kind.equals(HftpFileSystem.TOKEN_KIND)) {
                return HftpFileSystem.SCHEME;
            } else if (kind.equals(HsftpFileSystem.TOKEN_KIND)) {
                return HsftpFileSystem.SCHEME;
            } else if (kind.equals(WebHdfsFileSystem.TOKEN_KIND)) {
                return WebHdfsFileSystem.SCHEME;
            } else if (kind.equals(SWebHdfsFileSystem.TOKEN_KIND)) {
                return SWebHdfsFileSystem.SCHEME;
            } else {
                throw new IllegalArgumentException("Unsupported scheme");
            }
        }

        @Override
        public void cancel(Token<?> token, Configuration conf) throws IOException {
            getInstance(token, conf).cancelDelegationToken(token);
        }

        @Override
        public boolean handleKind(Text kind) {
            return kind.equals(HftpFileSystem.TOKEN_KIND) || kind.equals(HsftpFileSystem.TOKEN_KIND)
                    || kind.equals(WebHdfsFileSystem.TOKEN_KIND) || kind.equals(SWebHdfsFileSystem.TOKEN_KIND);
        }

        @Override
        public boolean isManaged(Token<?> token) throws IOException {
            return true;
        }

        @Override
        public long renew(Token<?> token, Configuration conf) throws IOException {
            return getInstance(token, conf).renewDelegationToken(token);
        }

        private TokenManagementDelegator getInstance(Token<?> token, Configuration conf) throws IOException {
            final URI uri;
            final String scheme = getSchemeByKind(token.getKind());
            if (HAUtil.isTokenForLogicalUri(token)) {
                uri = HAUtil.getServiceUriFromToken(scheme, token);
            } else {
                final InetSocketAddress address = SecurityUtil.getTokenServiceAddr(token);
                uri = URI.create(scheme + "://" + NetUtils.getHostPortString(address));
            }
            return (TokenManagementDelegator) FileSystem.get(uri, conf);
        }
    }

    private static class DTSelecorByKind extends AbstractDelegationTokenSelector<DelegationTokenIdentifier> {
        public DTSelecorByKind(final Text kind) {
            super(kind);
        }
    }
}