blob: 10d6c34e7152954248410f8af3be260d42d47c46 [file] [log] [blame]
/*
* 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.sshd.common.cipher;
import java.security.Security;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import org.apache.sshd.client.SshClient;
import org.apache.sshd.client.future.AuthFuture;
import org.apache.sshd.client.session.ClientSession;
import org.apache.sshd.common.keyprovider.FileKeyPairProvider;
import org.apache.sshd.common.util.security.SecurityUtils;
import org.apache.sshd.util.test.BaseTestSupport;
import org.apache.sshd.util.test.CommonTestSupportUtils;
import org.apache.sshd.util.test.ContainerTestCase;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
import org.junit.After;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.experimental.categories.Category;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.testcontainers.containers.GenericContainer;
import org.testcontainers.containers.output.Slf4jLogConsumer;
import org.testcontainers.containers.wait.strategy.Wait;
import org.testcontainers.images.builder.ImageFromDockerfile;
import org.testcontainers.utility.MountableFile;
/**
* Test ciphers against OpenSSH. Force resetting ciphers every time to verify that they are res-initialized correctly.
*
* @author <a href="mailto:dev@mina.apache.org">Apache MINA SSHD Project</a>
*/
@RunWith(Parameterized.class)
@Category(ContainerTestCase.class)
public class OpenSshCipherTest extends BaseTestSupport {
private static final Logger LOG = LoggerFactory.getLogger(OpenSshCipherTest.class);
// Re-use an already defined key
private static final String TEST_RESOURCES = "org/apache/sshd/common/kex/extensions/client";
@Rule
public GenericContainer<?> sshdContainer = new GenericContainer<>(new ImageFromDockerfile()
.withDockerfileFromBuilder(builder -> builder.from("alpine:3.19") //
.run("apk --update add openssh-server") // Installs OpenSSH
// Enable deprecated ciphers
.run("echo 'Ciphers +aes128-cbc,aes192-cbc,aes256-cbc,3des-cbc' >> /etc/ssh/sshd_config")
.run("ssh-keygen -A") // Generate multiple host keys
.run("adduser -D bob") // Add a user
.run("echo 'bob:passwordBob' | chpasswd") // Give it a password to unlock the user
.run("mkdir -p /home/bob/.ssh") // Create the SSH config directory
.entryPoint("/entrypoint.sh") // Sets bob as owner of anything under /home/bob and launches sshd
.build())) //
.withCopyFileToContainer(MountableFile.forClasspathResource(TEST_RESOURCES + "/bob_key.pub"),
"/home/bob/.ssh/authorized_keys")
// entrypoint must be executable. Spotbugs doesn't like 0777, so use hex
.withCopyFileToContainer(
MountableFile.forClasspathResource(TEST_RESOURCES + "/entrypoint.sh", 0x1ff),
"/entrypoint.sh")
.waitingFor(Wait.forLogMessage(".*Server listening on :: port 22.*\\n", 1)).withExposedPorts(22) //
.withLogConsumer(new Slf4jLogConsumer(LOG));
private final String providerName;
private final BuiltinCiphers builtIn;
public OpenSshCipherTest(String providerName, BuiltinCiphers factory, String name) {
this.providerName = providerName;
this.builtIn = factory;
if ("BC".equals(providerName)) {
registerBouncyCastleProviderIfNecessary();
}
}
@Before
public void changeCipher() {
BaseCipher.factory = t -> javax.crypto.Cipher.getInstance(t, providerName);
BaseCipher.alwaysReInit = true;
}
@After
public void resetCipher() {
BaseCipher.factory = SecurityUtils::getCipher;
BaseCipher.alwaysReInit = false;
}
private static void registerBouncyCastleProviderIfNecessary() {
if (Security.getProvider(BouncyCastleProvider.PROVIDER_NAME) == null) {
Security.addProvider(new BouncyCastleProvider());
}
}
private static void addCipher(BuiltinCiphers cipherFactory, List<Object[]> items) {
items.add(new Object[] { "SunJCE", cipherFactory, cipherFactory.getName() });
items.add(new Object[] { "BC", cipherFactory, cipherFactory.getName() });
}
@SuppressWarnings("deprecation")
@Parameters(name = "{2} - {0}")
public static List<Object[]> getParameters() {
List<Object[]> items = new ArrayList<>();
addCipher(BuiltinCiphers.tripledescbc, items);
addCipher(BuiltinCiphers.aes128cbc, items);
addCipher(BuiltinCiphers.aes128ctr, items);
addCipher(BuiltinCiphers.aes128gcm, items);
addCipher(BuiltinCiphers.aes192cbc, items);
addCipher(BuiltinCiphers.aes192ctr, items);
addCipher(BuiltinCiphers.aes256cbc, items);
addCipher(BuiltinCiphers.aes256ctr, items);
addCipher(BuiltinCiphers.aes256gcm, items);
addCipher(BuiltinCiphers.cc20p1305_openssh, items);
return items;
}
@Test
public void testConnection() throws Exception {
FileKeyPairProvider keyPairProvider = CommonTestSupportUtils.createTestKeyPairProvider(TEST_RESOURCES + "/bob_key");
SshClient client = setupTestClient();
client.setKeyIdentityProvider(keyPairProvider);
client.setCipherFactories(Collections.singletonList(builtIn));
client.start();
Integer actualPort = sshdContainer.getMappedPort(22);
String actualHost = sshdContainer.getHost();
try (ClientSession session = client.connect("bob", actualHost, actualPort).verify(CONNECT_TIMEOUT).getSession()) {
AuthFuture authed = session.auth().verify(AUTH_TIMEOUT);
assertTrue(authed.isDone() && authed.isSuccess());
} finally {
client.stop();
}
}
}