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

org.apache.tinkerpop.gremlin.server.KdcFixture Maven / Gradle / Ivy

The newest version!
/*
 * 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
 *
 * http://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.tinkerpop.gremlin.server;

import org.apache.kerby.kerberos.kerb.KrbException;
import org.apache.kerby.kerberos.kerb.client.KrbClient;
import org.apache.kerby.kerberos.kerb.client.KrbConfig;
import org.apache.kerby.kerberos.kerb.client.KrbConfigKey;
import org.apache.kerby.kerberos.kerb.server.SimpleKdcServer;
import org.apache.kerby.kerberos.kerb.type.ticket.TgtTicket;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.File;
import java.net.Inet4Address;

/**
 * This class is derived from the following classes from https://github.com/apache/directory-kerby/blob/kerby-all-1.0.0-RC2:
 *  - org.apache.kerby.kerberos.kerb.server.TestKdcServer
 *  - org.apache.kerby.kerberos.kerb.server.KdcTestBase
 *  - org.apache.kerby.kerberos.kerb.server.LoginTestBase
 *
 * See also: gremlin-server/src/main/static/NOTICE
 *
 * @author Marc de Lignie
 */
public class KdcFixture {

    private static final Logger logger = LoggerFactory.getLogger(KdcFixture.class);

    final String clientPassword = "123456";
    final String clientPassword2 = "1234562";
    final String userPassword = "password";
    final String clientPrincipalName = "drankye";
    final String clientPrincipalName2 = "drankye2";
    final String userPrincipalName = "stephen";
    final String serverPrincipalName = "test-service";
    final String ticketCacheFileName = "test-tkt.cc";
    final String ticketCacheFileName2 = "test-tkt2.cc";
    final String serviceKeytabFileName = "test-service.keytab";

    final String clientPrincipal;
    final String clientPrincipal2;
    final String serverPrincipal;
    final String userPrincipal;
    final File testDir;
    final File ticketCacheFile;
    final File ticketCacheFile2;
    final File serviceKeytabFile;

    final String gremlinHostname;
    final String kdcHostname;
    private SimpleKdcServer kdcServer;

    public KdcFixture(final String moduleBaseDir) {
        this(moduleBaseDir, "localhost");
    }

    public KdcFixture(final String moduleBaseDir, final String kdcHostName) {
        this.kdcHostname = kdcHostName;
        gremlinHostname = findHostname();
        serverPrincipal = serverPrincipalName + "/" + gremlinHostname + "@" + KdcFixture.TestKdcServer.KDC_REALM;
        clientPrincipal = clientPrincipalName + "@" + KdcFixture.TestKdcServer.KDC_REALM;
        clientPrincipal2 = clientPrincipalName2 + "@" + KdcFixture.TestKdcServer.KDC_REALM;
        userPrincipal = userPrincipalName  + "@" + KdcFixture.TestKdcServer.KDC_REALM;

        final File targetDir = new File(moduleBaseDir, "target");
        testDir = new File(targetDir, "kdc");
        testDir.mkdirs();
        ticketCacheFile = new File(testDir, ticketCacheFileName);
        ticketCacheFile2 = new File(testDir, ticketCacheFileName2);
        serviceKeytabFile = new File(testDir, serviceKeytabFileName);
    }

    private String findHostname() {
        // Hostname setting must be consistent with the way gremlin-console sets gremlin-server's hostname
        // and derives gremlin-server's principal name. Also, the hostname needs to be lowercase for use
        // in principal names.
        String hostname = "";
        try {
            hostname = Inet4Address.getLocalHost().getCanonicalHostName().toLowerCase();
        } catch (Exception e) {
            logger.error("Hostname not found: " + e.getMessage());
        }
        return hostname;
    }

    private class TestKdcServer extends SimpleKdcServer {
        public static final String KDC_REALM = "TEST.COM";
        public final String HOSTNAME = kdcHostname;
        public static final int KDC_PORT = 4588;

        TestKdcServer() throws KrbException {
            setKdcRealm(KDC_REALM);
            setKdcHost(HOSTNAME);
            setAllowTcp(true);
            setAllowUdp(true);
            setKdcTcpPort(KDC_PORT);
            setKdcUdpPort(KDC_PORT);

            final KrbClient krbClnt = getKrbClient();
            final KrbConfig krbConfig = krbClnt.getKrbConfig();
            krbConfig.setString(KrbConfigKey.PERMITTED_ENCTYPES,
                    "aes128-cts-hmac-sha1-96 des-cbc-crc des-cbc-md5 des3-cbc-sha1");
            krbConfig.setString(KrbConfigKey.DEFAULT_REALM, KDC_REALM);
            krbClnt.setTimeout(10 * 1000);
        }
    }

    public void setUp() throws Exception {
        setUpKdcServer();
        setUpPrincipals();
    }

    private void setUpKdcServer() throws Exception {
        kdcServer = new TestKdcServer();
        kdcServer.setWorkDir(testDir);
        kdcServer.init();

        // we start/stop the Kdc Server with some rapidity in tests. in low resource environments (travis or docker),
        // it's possible that the socket for the server is left in TIME_WAIT when it comes time to restart. with a
        // bit of retry and pausing perhaps it will make tests less flaky in those environments
        for (int ix = 0; ix < 10; ix++) {
            try {
                kdcServer.start();
                break;
            } catch (Exception ex) {
                // on the last try ending in failure just toss the exception
                if (ix == 9) throw ex;
                final int pause = (ix + 1) * 1500;
                logger.warn(String.format("Failed to start Kerberos Server - pausing for %s milliseconds and trying again - try #%s", pause, ix + 1), ex);
                Thread.sleep(pause);
            }
        }
    }

    private void setUpPrincipals() throws KrbException {
        kdcServer.createPrincipals(serverPrincipal);
        kdcServer.exportPrincipal(serverPrincipal, serviceKeytabFile);

        kdcServer.createPrincipal(clientPrincipal, clientPassword);
        final TgtTicket tgt = kdcServer.getKrbClient().requestTgt(clientPrincipal, clientPassword);
        kdcServer.getKrbClient().storeTicket(tgt, ticketCacheFile);

        kdcServer.createPrincipal(clientPrincipal2, clientPassword2);
        final TgtTicket tgt2 = kdcServer.getKrbClient().requestTgt(clientPrincipal2, clientPassword2);
        kdcServer.getKrbClient().storeTicket(tgt2, ticketCacheFile2);

        kdcServer.createPrincipal(userPrincipal, userPassword);
    }

    public void close() {
        deletePrincipals();

        try {
            kdcServer.stop();
        } catch (KrbException krbex) {
            logger.warn("Tried to stop KdcServer but encountered an exception", krbex);
        }

        ticketCacheFile.delete();
        ticketCacheFile2.delete();
        serviceKeytabFile.delete();
        testDir.delete();
    }

    void deletePrincipals() {
        try {
            kdcServer.getKadmin().deleteBuiltinPrincipals();
        } catch (KrbException krbex) {
            logger.warn("Tried to delete builtin Principals on teardown but failed", krbex);
        }

        deletePrincipal(serverPrincipal);
        deletePrincipal(clientPrincipal);
        deletePrincipal(clientPrincipal2);
    }

    private void deletePrincipal(final String principalToDelete) {
        try {
            kdcServer.deletePrincipal(principalToDelete);
        } catch (KrbException krbex) {
            logger.warn(String.format("Tried to delete %s Principals on teardown but failed", principalToDelete), krbex);
        }
    }

    public void createPrincipal(final String principal) throws KrbException {
        kdcServer.createPrincipal(principal);
    }

    public static void main(final String[] args) throws Exception{
        final String projectBaseDir = args[0];
        // The KDC in docker/gremlin-server.sh needs to be exposed to both the container and the host
        final KdcFixture kdcFixture = new KdcFixture(projectBaseDir, "0.0.0.0");
        kdcFixture.setUp();
        logger.info("KDC started with configuration {}/target/kdc/krb5.conf", projectBaseDir);
        while (true) {
            Thread.sleep(1000);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy