Skip to content

Commit d1a1653

Browse files
committedAug 8, 2024·
feat: add quic support
Ref: #1298 Fixes #1924
1 parent 661c30a commit d1a1653

File tree

2 files changed

+68
-1
lines changed

2 files changed

+68
-1
lines changed
 

‎src/lib/connect/index.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -109,7 +109,11 @@ function connect(
109109

110110
if (opts.cert && opts.key) {
111111
if (opts.protocol) {
112-
if (['mqtts', 'wss', 'wxs', 'alis'].indexOf(opts.protocol) === -1) {
112+
if (
113+
['mqtts', 'wss', 'wxs', 'alis', 'quic'].indexOf(
114+
opts.protocol,
115+
) === -1
116+
) {
113117
switch (opts.protocol) {
114118
case 'mqtt':
115119
opts.protocol = 'mqtts'
@@ -147,6 +151,8 @@ function connect(
147151
protocols.ssl = require('./tls').default
148152
protocols.tls = protocols.ssl
149153
protocols.mqtts = require('./tls').default
154+
155+
protocols.quic = require('./quic').default
150156
} else {
151157
protocols.ws = require('./ws').browserStreamBuilder
152158
protocols.wss = require('./ws').browserStreamBuilder

‎src/lib/connect/quic.ts

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import { StreamBuilder } from '../shared'
2+
import _debug from 'debug'
3+
import { createSocket } from 'node:quic'
4+
5+
const debug = _debug('mqttjs:quic')
6+
7+
const buildStream: StreamBuilder = (client, opts) => {
8+
opts.port = opts.port || 8885
9+
opts.host = opts.hostname || opts.host || 'localhost'
10+
opts.servername = opts.host
11+
12+
opts.rejectUnauthorized = opts.rejectUnauthorized !== false
13+
14+
delete opts.path
15+
16+
debug(
17+
'port %d host %s rejectUnauthorized %b',
18+
opts.port,
19+
opts.host,
20+
opts.rejectUnauthorized,
21+
)
22+
23+
const socket = createSocket({
24+
client: {
25+
alpn: opts.alpn || 'mqtt',
26+
},
27+
})
28+
29+
const req = socket.connect({
30+
address: opts.host,
31+
port: opts.port || 8885,
32+
cert: opts.cert,
33+
key: opts.key,
34+
idleTimeout: 0, // disable timeout
35+
})
36+
37+
const stream = req.openStream()
38+
39+
req.on('secure', (servername) => {
40+
// servername is checked for any string for authorisation acceptance
41+
if (opts.rejectUnauthorized && !servername) {
42+
req.emit('error', new Error('TLS not authorized'))
43+
} else {
44+
req.removeListener('error', handleTLSErrors)
45+
}
46+
})
47+
48+
function handleTLSErrors(err) {
49+
if (opts.rejectUnauthorized) {
50+
client.emit('error', err)
51+
}
52+
53+
stream.end()
54+
socket.close()
55+
}
56+
57+
req.on('error', handleTLSErrors)
58+
return stream
59+
}
60+
61+
export default buildStream

0 commit comments

Comments
 (0)
Please sign in to comment.