11cb0ef41Sopenharmony_ci// Copyright Joyent, Inc. and other Node contributors.
21cb0ef41Sopenharmony_ci//
31cb0ef41Sopenharmony_ci// Permission is hereby granted, free of charge, to any person obtaining a
41cb0ef41Sopenharmony_ci// copy of this software and associated documentation files (the
51cb0ef41Sopenharmony_ci// "Software"), to deal in the Software without restriction, including
61cb0ef41Sopenharmony_ci// without limitation the rights to use, copy, modify, merge, publish,
71cb0ef41Sopenharmony_ci// distribute, sublicense, and/or sell copies of the Software, and to permit
81cb0ef41Sopenharmony_ci// persons to whom the Software is furnished to do so, subject to the
91cb0ef41Sopenharmony_ci// following conditions:
101cb0ef41Sopenharmony_ci//
111cb0ef41Sopenharmony_ci// The above copyright notice and this permission notice shall be included
121cb0ef41Sopenharmony_ci// in all copies or substantial portions of the Software.
131cb0ef41Sopenharmony_ci//
141cb0ef41Sopenharmony_ci// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
151cb0ef41Sopenharmony_ci// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
161cb0ef41Sopenharmony_ci// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
171cb0ef41Sopenharmony_ci// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
181cb0ef41Sopenharmony_ci// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
191cb0ef41Sopenharmony_ci// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
201cb0ef41Sopenharmony_ci// USE OR OTHER DEALINGS IN THE SOFTWARE.
211cb0ef41Sopenharmony_ci
221cb0ef41Sopenharmony_ci'use strict';
231cb0ef41Sopenharmony_ciconst common = require('../common');
241cb0ef41Sopenharmony_ciif (!common.hasCrypto)
251cb0ef41Sopenharmony_ci  common.skip('missing crypto');
261cb0ef41Sopenharmony_ci
271cb0ef41Sopenharmony_ciconst assert = require('assert');
281cb0ef41Sopenharmony_ciconst tls = require('tls');
291cb0ef41Sopenharmony_ciconst fixtures = require('../common/fixtures');
301cb0ef41Sopenharmony_ci
311cb0ef41Sopenharmony_cifunction loadPEM(n) {
321cb0ef41Sopenharmony_ci  return fixtures.readKey(`${n}.pem`);
331cb0ef41Sopenharmony_ci}
341cb0ef41Sopenharmony_ci
351cb0ef41Sopenharmony_ciconst serverOptions = {
361cb0ef41Sopenharmony_ci  key: loadPEM('agent2-key'),
371cb0ef41Sopenharmony_ci  cert: loadPEM('agent2-cert'),
381cb0ef41Sopenharmony_ci  requestCert: true,
391cb0ef41Sopenharmony_ci  rejectUnauthorized: false,
401cb0ef41Sopenharmony_ci  SNICallback: function(servername, callback) {
411cb0ef41Sopenharmony_ci    const context = SNIContexts[servername];
421cb0ef41Sopenharmony_ci
431cb0ef41Sopenharmony_ci    // Just to test asynchronous callback
441cb0ef41Sopenharmony_ci    setTimeout(function() {
451cb0ef41Sopenharmony_ci      if (context) {
461cb0ef41Sopenharmony_ci        if (context.emptyRegression)
471cb0ef41Sopenharmony_ci          callback(null, {});
481cb0ef41Sopenharmony_ci        else
491cb0ef41Sopenharmony_ci          callback(null, tls.createSecureContext(context));
501cb0ef41Sopenharmony_ci      } else {
511cb0ef41Sopenharmony_ci        callback(null, null);
521cb0ef41Sopenharmony_ci      }
531cb0ef41Sopenharmony_ci    }, 100);
541cb0ef41Sopenharmony_ci  }
551cb0ef41Sopenharmony_ci};
561cb0ef41Sopenharmony_ci
571cb0ef41Sopenharmony_ciconst SNIContexts = {
581cb0ef41Sopenharmony_ci  'a.example.com': {
591cb0ef41Sopenharmony_ci    key: loadPEM('agent1-key'),
601cb0ef41Sopenharmony_ci    cert: loadPEM('agent1-cert'),
611cb0ef41Sopenharmony_ci    ca: [ loadPEM('ca2-cert') ]
621cb0ef41Sopenharmony_ci  },
631cb0ef41Sopenharmony_ci  'b.example.com': {
641cb0ef41Sopenharmony_ci    key: loadPEM('agent3-key'),
651cb0ef41Sopenharmony_ci    cert: loadPEM('agent3-cert')
661cb0ef41Sopenharmony_ci  },
671cb0ef41Sopenharmony_ci  'c.another.com': {
681cb0ef41Sopenharmony_ci    emptyRegression: true
691cb0ef41Sopenharmony_ci  }
701cb0ef41Sopenharmony_ci};
711cb0ef41Sopenharmony_ci
721cb0ef41Sopenharmony_citest({
731cb0ef41Sopenharmony_ci  port: undefined,
741cb0ef41Sopenharmony_ci  key: loadPEM('agent1-key'),
751cb0ef41Sopenharmony_ci  cert: loadPEM('agent1-cert'),
761cb0ef41Sopenharmony_ci  ca: [loadPEM('ca1-cert')],
771cb0ef41Sopenharmony_ci  servername: 'a.example.com',
781cb0ef41Sopenharmony_ci  rejectUnauthorized: false
791cb0ef41Sopenharmony_ci},
801cb0ef41Sopenharmony_ci     true,
811cb0ef41Sopenharmony_ci     { sni: 'a.example.com', authorized: false },
821cb0ef41Sopenharmony_ci     null,
831cb0ef41Sopenharmony_ci     null);
841cb0ef41Sopenharmony_ci
851cb0ef41Sopenharmony_citest({
861cb0ef41Sopenharmony_ci  port: undefined,
871cb0ef41Sopenharmony_ci  key: loadPEM('agent4-key'),
881cb0ef41Sopenharmony_ci  cert: loadPEM('agent4-cert'),
891cb0ef41Sopenharmony_ci  ca: [loadPEM('ca1-cert')],
901cb0ef41Sopenharmony_ci  servername: 'a.example.com',
911cb0ef41Sopenharmony_ci  rejectUnauthorized: false
921cb0ef41Sopenharmony_ci},
931cb0ef41Sopenharmony_ci     true,
941cb0ef41Sopenharmony_ci     { sni: 'a.example.com', authorized: true },
951cb0ef41Sopenharmony_ci     null,
961cb0ef41Sopenharmony_ci     null);
971cb0ef41Sopenharmony_ci
981cb0ef41Sopenharmony_citest({
991cb0ef41Sopenharmony_ci  port: undefined,
1001cb0ef41Sopenharmony_ci  key: loadPEM('agent2-key'),
1011cb0ef41Sopenharmony_ci  cert: loadPEM('agent2-cert'),
1021cb0ef41Sopenharmony_ci  ca: [loadPEM('ca2-cert')],
1031cb0ef41Sopenharmony_ci  servername: 'b.example.com',
1041cb0ef41Sopenharmony_ci  rejectUnauthorized: false
1051cb0ef41Sopenharmony_ci},
1061cb0ef41Sopenharmony_ci     true,
1071cb0ef41Sopenharmony_ci     { sni: 'b.example.com', authorized: false },
1081cb0ef41Sopenharmony_ci     null,
1091cb0ef41Sopenharmony_ci     null);
1101cb0ef41Sopenharmony_ci
1111cb0ef41Sopenharmony_citest({
1121cb0ef41Sopenharmony_ci  port: undefined,
1131cb0ef41Sopenharmony_ci  key: loadPEM('agent3-key'),
1141cb0ef41Sopenharmony_ci  cert: loadPEM('agent3-cert'),
1151cb0ef41Sopenharmony_ci  ca: [loadPEM('ca1-cert')],
1161cb0ef41Sopenharmony_ci  servername: 'c.wrong.com',
1171cb0ef41Sopenharmony_ci  rejectUnauthorized: false
1181cb0ef41Sopenharmony_ci},
1191cb0ef41Sopenharmony_ci     false,
1201cb0ef41Sopenharmony_ci     { sni: 'c.wrong.com', authorized: false },
1211cb0ef41Sopenharmony_ci     null,
1221cb0ef41Sopenharmony_ci     null);
1231cb0ef41Sopenharmony_ci
1241cb0ef41Sopenharmony_citest({
1251cb0ef41Sopenharmony_ci  port: undefined,
1261cb0ef41Sopenharmony_ci  key: loadPEM('agent3-key'),
1271cb0ef41Sopenharmony_ci  cert: loadPEM('agent3-cert'),
1281cb0ef41Sopenharmony_ci  ca: [loadPEM('ca1-cert')],
1291cb0ef41Sopenharmony_ci  servername: 'c.another.com',
1301cb0ef41Sopenharmony_ci  rejectUnauthorized: false
1311cb0ef41Sopenharmony_ci},
1321cb0ef41Sopenharmony_ci     false,
1331cb0ef41Sopenharmony_ci     null,
1341cb0ef41Sopenharmony_ci     'Client network socket disconnected before secure TLS ' +
1351cb0ef41Sopenharmony_ci       'connection was established',
1361cb0ef41Sopenharmony_ci     'Invalid SNI context');
1371cb0ef41Sopenharmony_ci
1381cb0ef41Sopenharmony_cifunction test(options, clientResult, serverResult, clientError, serverError) {
1391cb0ef41Sopenharmony_ci  const server = tls.createServer(serverOptions, (c) => {
1401cb0ef41Sopenharmony_ci    assert.deepStrictEqual(
1411cb0ef41Sopenharmony_ci      serverResult,
1421cb0ef41Sopenharmony_ci      { sni: c.servername, authorized: c.authorized }
1431cb0ef41Sopenharmony_ci    );
1441cb0ef41Sopenharmony_ci  });
1451cb0ef41Sopenharmony_ci
1461cb0ef41Sopenharmony_ci  if (serverResult) {
1471cb0ef41Sopenharmony_ci    assert(!serverError);
1481cb0ef41Sopenharmony_ci    server.on('tlsClientError', common.mustNotCall());
1491cb0ef41Sopenharmony_ci  } else {
1501cb0ef41Sopenharmony_ci    assert(serverError);
1511cb0ef41Sopenharmony_ci    server.on('tlsClientError', common.mustCall((err) => {
1521cb0ef41Sopenharmony_ci      assert.strictEqual(err.message, serverError);
1531cb0ef41Sopenharmony_ci    }));
1541cb0ef41Sopenharmony_ci  }
1551cb0ef41Sopenharmony_ci
1561cb0ef41Sopenharmony_ci  server.listen(0, () => {
1571cb0ef41Sopenharmony_ci    options.port = server.address().port;
1581cb0ef41Sopenharmony_ci    const client = tls.connect(options, () => {
1591cb0ef41Sopenharmony_ci      const result = client.authorizationError &&
1601cb0ef41Sopenharmony_ci        (client.authorizationError === 'ERR_TLS_CERT_ALTNAME_INVALID');
1611cb0ef41Sopenharmony_ci      assert.strictEqual(result, clientResult);
1621cb0ef41Sopenharmony_ci      client.end();
1631cb0ef41Sopenharmony_ci    });
1641cb0ef41Sopenharmony_ci
1651cb0ef41Sopenharmony_ci    client.on('close', common.mustCall(() => server.close()));
1661cb0ef41Sopenharmony_ci
1671cb0ef41Sopenharmony_ci    if (clientError)
1681cb0ef41Sopenharmony_ci      client.on('error', common.mustCall((err) => {
1691cb0ef41Sopenharmony_ci        assert.strictEqual(err.message, clientError);
1701cb0ef41Sopenharmony_ci      }));
1711cb0ef41Sopenharmony_ci    else
1721cb0ef41Sopenharmony_ci      client.on('error', common.mustNotCall());
1731cb0ef41Sopenharmony_ci  });
1741cb0ef41Sopenharmony_ci}
175