ITestS3AEncryptionAlgorithmValidation.java

/*
 * 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.hadoop.fs.s3a;

import static org.apache.hadoop.test.LambdaTestUtils.intercept;

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

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.FileSystem;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.fs.contract.s3a.S3AContract;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.Test;

/**
 * Test whether or not encryption settings propagate by choosing an invalid
 * one. We expect the S3AFileSystem to fail to initialize.
 */
@Disabled
public class ITestS3AEncryptionAlgorithmValidation
    extends AbstractS3ATestBase {

  @Test
  public void testEncryptionAlgorithmSetToDES() throws Throwable {
    //skip tests if they aren't enabled
    assumeEnabled();
    intercept(IOException.class, "Unknown encryption algorithm DES", () -> {

      Configuration conf = super.createConfiguration();
      //DES is an invalid encryption algorithm
      conf.set(Constants.S3_ENCRYPTION_ALGORITHM, "DES");
      S3AContract contract = (S3AContract) createContract(conf);
      contract.init();
      //extract the test FS
      FileSystem fileSystem = contract.getTestFileSystem();
      assertNotNull(fileSystem, "null filesystem");
      URI fsURI = fileSystem.getUri();
      LOG.info("Test filesystem = {} implemented by {}", fsURI, fileSystem);
      assertEquals(contract.getScheme(), fsURI.getScheme(),
          "wrong filesystem of " + fsURI);
      fileSystem.initialize(fsURI, conf);
      return fileSystem;
    });
  }

  @Test
  public void testEncryptionAlgorithmSSECWithNoEncryptionKey() throws
    Throwable {
    //skip tests if they aren't enabled
    assumeEnabled();
    intercept(IllegalArgumentException.class, "The value of property " +
        Constants.S3_ENCRYPTION_KEY + " must not be null", () -> {

        Configuration conf = super.createConfiguration();
        //SSE-C must be configured with an encryption key
        conf.set(Constants.S3_ENCRYPTION_ALGORITHM,
            S3AEncryptionMethods.SSE_C.getMethod());
        conf.set(Constants.S3_ENCRYPTION_KEY, null);
        S3AContract contract = (S3AContract) createContract(conf);
        contract.init();
        //extract the test FS
        FileSystem fileSystem = contract.getTestFileSystem();
        assertNotNull(fileSystem, "null filesystem");
        URI fsURI = fileSystem.getUri();
        LOG.info("Test filesystem = {} implemented by {}", fsURI, fileSystem);
        assertEquals(contract.getScheme(), fsURI.getScheme(),
            "wrong filesystem of " + fsURI);
        fileSystem.initialize(fsURI, conf);
        return fileSystem;
    });
  }

  @Test
  public void testEncryptionAlgorithmSSECWithBlankEncryptionKey() throws
    Throwable {
    intercept(IOException.class, S3AUtils.SSE_C_NO_KEY_ERROR, () -> {

      Configuration conf = super.createConfiguration();
      //SSE-C must be configured with an encryption key
      conf.set(Constants.S3_ENCRYPTION_ALGORITHM,
          S3AEncryptionMethods.SSE_C.getMethod());
      conf.set(Constants.S3_ENCRYPTION_KEY, "");
      S3AContract contract = (S3AContract) createContract(conf);
      contract.init();
      //extract the test FS
      FileSystem fileSystem = contract.getTestFileSystem();
      assertNotNull(fileSystem, "null filesystem");
      URI fsURI = fileSystem.getUri();
      LOG.info("Test filesystem = {} implemented by {}", fsURI, fileSystem);
      assertEquals(contract.getScheme(), fsURI.getScheme(),
          "wrong filesystem of " + fsURI);
      fileSystem.initialize(fsURI, conf);
      return fileSystem;
    });
  }

  @Test
  public void testEncryptionAlgorithmSSES3WithEncryptionKey() throws
    Throwable {
    //skip tests if they aren't enabled
    assumeEnabled();
    intercept(IOException.class, S3AUtils.SSE_S3_WITH_KEY_ERROR, () -> {

      Configuration conf = super.createConfiguration();
      //SSE-S3 cannot be configured with an encryption key
      conf.set(Constants.S3_ENCRYPTION_ALGORITHM,
          S3AEncryptionMethods.SSE_S3.getMethod());
      conf.set(Constants.S3_ENCRYPTION_KEY,
          "4niV/jPK5VFRHY+KNb6wtqYd4xXyMgdJ9XQJpcQUVbs=");
      S3AContract contract = (S3AContract) createContract(conf);
      contract.init();
      //extract the test FS
      FileSystem fileSystem = contract.getTestFileSystem();
      assertNotNull(fileSystem, "null filesystem");
      URI fsURI = fileSystem.getUri();
      LOG.info("Test filesystem = {} implemented by {}", fsURI, fileSystem);
      assertEquals(contract.getScheme(), fsURI.getScheme(),
          "wrong filesystem of " + fsURI);
      fileSystem.initialize(fsURI, conf);
      return fileSystem;
    });
  }

  /**
   * Make this a no-op so test setup doesn't fail.
   * @param path path path
   * @throws IOException on any failure
   */
  @Override
  protected void mkdirs(Path path) throws IOException {

  }

}