1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package io.netty5.handler.ssl;
17
18 import io.netty5.util.internal.StringUtil;
19
20 import javax.net.ssl.SSLEngine;
21 import javax.net.ssl.SSLEngineResult;
22 import javax.net.ssl.SSLException;
23 import java.nio.ByteBuffer;
24 import java.util.LinkedHashSet;
25 import java.util.List;
26 import java.util.function.BiConsumer;
27 import java.util.function.BiFunction;
28
29 import static io.netty5.handler.ssl.JdkApplicationProtocolNegotiator.ProtocolSelectionListener;
30 import static io.netty5.handler.ssl.JdkApplicationProtocolNegotiator.ProtocolSelector;
31 import static io.netty5.handler.ssl.SslUtils.toSSLHandshakeException;
32
33 class JdkAlpnSslEngine extends JdkSslEngine {
34 private final ProtocolSelectionListener selectionListener;
35 private final AlpnSelector alpnSelector;
36
37 final class AlpnSelector implements BiFunction<SSLEngine, List<String>, String> {
38 private final ProtocolSelector selector;
39 private boolean called;
40
41 AlpnSelector(ProtocolSelector selector) {
42 this.selector = selector;
43 }
44
45 @Override
46 public String apply(SSLEngine sslEngine, List<String> strings) {
47 assert !called;
48 called = true;
49
50 try {
51 String selected = selector.select(strings);
52 return selected == null ? StringUtil.EMPTY_STRING : selected;
53 } catch (Exception cause) {
54
55
56
57
58 return null;
59 }
60 }
61
62 void checkUnsupported() {
63 if (called) {
64
65
66
67
68 return;
69 }
70 String protocol = getApplicationProtocol();
71 assert protocol != null;
72
73 if (protocol.isEmpty()) {
74
75 selector.unsupported();
76 }
77 }
78 }
79
80 JdkAlpnSslEngine(SSLEngine engine,
81 @SuppressWarnings("deprecation") JdkApplicationProtocolNegotiator applicationNegotiator,
82 boolean isServer, BiConsumer<SSLEngine, AlpnSelector> setHandshakeApplicationProtocolSelector,
83 BiConsumer<SSLEngine, List<String>> setApplicationProtocols) {
84 super(engine);
85 if (isServer) {
86 selectionListener = null;
87 alpnSelector = new AlpnSelector(applicationNegotiator.protocolSelectorFactory().
88 newSelector(this, new LinkedHashSet<>(applicationNegotiator.protocols())));
89 setHandshakeApplicationProtocolSelector.accept(engine, alpnSelector);
90 } else {
91 selectionListener = applicationNegotiator.protocolListenerFactory()
92 .newListener(this, applicationNegotiator.protocols());
93 alpnSelector = null;
94 setApplicationProtocols.accept(engine, applicationNegotiator.protocols());
95 }
96 }
97
98 JdkAlpnSslEngine(SSLEngine engine,
99 @SuppressWarnings("deprecation") JdkApplicationProtocolNegotiator applicationNegotiator,
100 boolean isServer) {
101 this(engine, applicationNegotiator, isServer,
102 JdkAlpnSslUtils::setHandshakeApplicationProtocolSelector,
103 JdkAlpnSslUtils::setApplicationProtocols);
104 }
105
106 private SSLEngineResult verifyProtocolSelection(SSLEngineResult result) throws SSLException {
107 if (result.getHandshakeStatus() == SSLEngineResult.HandshakeStatus.FINISHED) {
108 if (alpnSelector == null) {
109
110 try {
111 String protocol = getApplicationProtocol();
112 assert protocol != null;
113 if (protocol.isEmpty()) {
114
115
116
117
118 selectionListener.unsupported();
119 } else {
120 selectionListener.selected(protocol);
121 }
122 } catch (Throwable e) {
123 throw toSSLHandshakeException(e);
124 }
125 } else {
126 assert selectionListener == null;
127 alpnSelector.checkUnsupported();
128 }
129 }
130 return result;
131 }
132
133 @Override
134 public SSLEngineResult wrap(ByteBuffer src, ByteBuffer dst) throws SSLException {
135 return verifyProtocolSelection(super.wrap(src, dst));
136 }
137
138 @Override
139 public SSLEngineResult wrap(ByteBuffer[] srcs, ByteBuffer dst) throws SSLException {
140 return verifyProtocolSelection(super.wrap(srcs, dst));
141 }
142
143 @Override
144 public SSLEngineResult wrap(ByteBuffer[] srcs, int offset, int len, ByteBuffer dst) throws SSLException {
145 return verifyProtocolSelection(super.wrap(srcs, offset, len, dst));
146 }
147
148 @Override
149 public SSLEngineResult unwrap(ByteBuffer src, ByteBuffer dst) throws SSLException {
150 return verifyProtocolSelection(super.unwrap(src, dst));
151 }
152
153 @Override
154 public SSLEngineResult unwrap(ByteBuffer src, ByteBuffer[] dsts) throws SSLException {
155 return verifyProtocolSelection(super.unwrap(src, dsts));
156 }
157
158 @Override
159 public SSLEngineResult unwrap(ByteBuffer src, ByteBuffer[] dst, int offset, int len) throws SSLException {
160 return verifyProtocolSelection(super.unwrap(src, dst, offset, len));
161 }
162
163 @Override
164 void setNegotiatedApplicationProtocol(String applicationProtocol) {
165
166 }
167
168 @Override
169 public String getNegotiatedApplicationProtocol() {
170 String protocol = getApplicationProtocol();
171 if (protocol != null) {
172 return protocol.isEmpty() ? null : protocol;
173 }
174 return null;
175 }
176
177 @Override
178 public String getApplicationProtocol() {
179 return JdkAlpnSslUtils.getApplicationProtocol(getWrappedEngine());
180 }
181
182 @Override
183 public String getHandshakeApplicationProtocol() {
184 return JdkAlpnSslUtils.getHandshakeApplicationProtocol(getWrappedEngine());
185 }
186
187 @Override
188 public void setHandshakeApplicationProtocolSelector(BiFunction<SSLEngine, List<String>, String> selector) {
189 JdkAlpnSslUtils.setHandshakeApplicationProtocolSelector(getWrappedEngine(), selector);
190 }
191
192 @Override
193 public BiFunction<SSLEngine, List<String>, String> getHandshakeApplicationProtocolSelector() {
194 return JdkAlpnSslUtils.getHandshakeApplicationProtocolSelector(getWrappedEngine());
195 }
196 }