Skip to content

Commit afea9c9

Browse files
ylangiscvladimirlagunov
authored andcommitted
Try all public key algorithms available for a specific key type in SSH_MSG_USERAUTH_REQUEST. (hierynomus#763)
(cherry picked from commit aabb1be)
1 parent 2dd9365 commit afea9c9

File tree

3 files changed

+32
-7
lines changed

3 files changed

+32
-7
lines changed

src/main/java/net/schmizz/sshj/transport/Transport.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727

2828
import java.io.InputStream;
2929
import java.io.OutputStream;
30+
import java.util.List;
3031
import java.util.concurrent.TimeUnit;
3132

3233
/** Transport layer of the SSH protocol. */
@@ -223,5 +224,5 @@ long write(SSHPacket payload)
223224
void die(Exception e);
224225

225226
KeyAlgorithm getHostKeyAlgorithm();
226-
KeyAlgorithm getClientKeyAlgorithm(KeyType keyType) throws TransportException;
227+
List<KeyAlgorithm> getClientKeyAlgorithms(KeyType keyType) throws TransportException;
227228
}

src/main/java/net/schmizz/sshj/transport/TransportImpl.java

+9-4
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,8 @@
3232
import java.io.IOException;
3333
import java.io.InputStream;
3434
import java.io.OutputStream;
35+
import java.net.InetSocketAddress;
36+
import java.util.ArrayList;
3537
import java.util.List;
3638
import java.util.concurrent.TimeUnit;
3739
import java.util.concurrent.locks.ReentrantLock;
@@ -625,15 +627,18 @@ public KeyAlgorithm getHostKeyAlgorithm() {
625627
}
626628

627629
@Override
628-
public KeyAlgorithm getClientKeyAlgorithm(KeyType keyType) throws TransportException {
630+
public List<KeyAlgorithm> getClientKeyAlgorithms(KeyType keyType) throws TransportException {
629631
List<Factory.Named<KeyAlgorithm>> factories = getConfig().getKeyAlgorithms();
632+
List<KeyAlgorithm> available = new ArrayList<>();
630633
if (factories != null)
631634
for (Factory.Named<KeyAlgorithm> f : factories)
632635
if (
633636
f instanceof KeyAlgorithms.Factory && ((KeyAlgorithms.Factory) f).getKeyType().equals(keyType)
634-
|| !(f instanceof KeyAlgorithms.Factory) && f.getName().equals(keyType.toString())
637+
|| !(f instanceof KeyAlgorithms.Factory) && f.getName().equals(keyType.toString())
635638
)
636-
return f.create();
637-
throw new TransportException("Cannot find an available KeyAlgorithm for type " + keyType);
639+
available.add(f.create());
640+
if (available.isEmpty())
641+
throw new TransportException("Cannot find an available KeyAlgorithm for type " + keyType);
642+
return available;
638643
}
639644
}

src/main/java/net/schmizz/sshj/userauth/method/KeyedAuthMethod.java

+21-2
Original file line numberDiff line numberDiff line change
@@ -27,17 +27,36 @@
2727
import java.io.IOException;
2828
import java.security.PrivateKey;
2929
import java.security.PublicKey;
30+
import java.util.LinkedList;
31+
import java.util.Queue;
3032

3133
public abstract class KeyedAuthMethod
3234
extends AbstractAuthMethod {
3335

3436
protected final KeyProvider kProv;
37+
private Queue<KeyAlgorithm> available;
3538

3639
public KeyedAuthMethod(String name, KeyProvider kProv) {
3740
super(name);
3841
this.kProv = kProv;
3942
}
4043

44+
private KeyAlgorithm getPublicKeyAlgorithm(KeyType keyType) throws TransportException {
45+
if (available == null) {
46+
available = new LinkedList<>(params.getTransport().getClientKeyAlgorithms(keyType));
47+
}
48+
return available.peek();
49+
}
50+
51+
@Override
52+
public boolean shouldRetry() {
53+
if (available != null) {
54+
available.poll();
55+
return !available.isEmpty();
56+
}
57+
return false;
58+
}
59+
4160
protected SSHPacket putPubKey(SSHPacket reqBuf)
4261
throws UserAuthException {
4362
PublicKey key;
@@ -50,7 +69,7 @@ protected SSHPacket putPubKey(SSHPacket reqBuf)
5069
// public key as 2 strings: [ key type | key blob ]
5170
KeyType keyType = KeyType.fromKey(key);
5271
try {
53-
KeyAlgorithm ka = params.getTransport().getClientKeyAlgorithm(keyType);
72+
KeyAlgorithm ka = getPublicKeyAlgorithm(keyType);
5473
if (ka != null) {
5574
reqBuf.putString(ka.getKeyAlgorithm())
5675
.putString(new Buffer.PlainBuffer().putPublicKey(key).getCompactData());
@@ -74,7 +93,7 @@ protected SSHPacket putSig(SSHPacket reqBuf)
7493
final KeyType kt = KeyType.fromKey(key);
7594
Signature signature;
7695
try {
77-
signature = params.getTransport().getClientKeyAlgorithm(kt).newSignature();
96+
signature = getPublicKeyAlgorithm(kt).newSignature();
7897
} catch (TransportException e) {
7998
throw new UserAuthException("No KeyAlgorithm configured for key " + kt);
8099
}

0 commit comments

Comments
 (0)