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_ciconst testCases = [
321cb0ef41Sopenharmony_ci  { ca: ['ca1-cert'],
331cb0ef41Sopenharmony_ci    key: 'agent2-key',
341cb0ef41Sopenharmony_ci    cert: 'agent2-cert',
351cb0ef41Sopenharmony_ci    servers: [
361cb0ef41Sopenharmony_ci      { ok: true, key: 'agent1-key', cert: 'agent1-cert' },
371cb0ef41Sopenharmony_ci      { ok: false, key: 'agent2-key', cert: 'agent2-cert' },
381cb0ef41Sopenharmony_ci      { ok: false, key: 'agent3-key', cert: 'agent3-cert' },
391cb0ef41Sopenharmony_ci    ] },
401cb0ef41Sopenharmony_ci
411cb0ef41Sopenharmony_ci  { ca: [],
421cb0ef41Sopenharmony_ci    key: 'agent2-key',
431cb0ef41Sopenharmony_ci    cert: 'agent2-cert',
441cb0ef41Sopenharmony_ci    servers: [
451cb0ef41Sopenharmony_ci      { ok: false, key: 'agent1-key', cert: 'agent1-cert' },
461cb0ef41Sopenharmony_ci      { ok: false, key: 'agent2-key', cert: 'agent2-cert' },
471cb0ef41Sopenharmony_ci      { ok: false, key: 'agent3-key', cert: 'agent3-cert' },
481cb0ef41Sopenharmony_ci    ] },
491cb0ef41Sopenharmony_ci
501cb0ef41Sopenharmony_ci  { ca: ['ca1-cert', 'ca2-cert'],
511cb0ef41Sopenharmony_ci    key: 'agent2-key',
521cb0ef41Sopenharmony_ci    cert: 'agent2-cert',
531cb0ef41Sopenharmony_ci    servers: [
541cb0ef41Sopenharmony_ci      { ok: true, key: 'agent1-key', cert: 'agent1-cert' },
551cb0ef41Sopenharmony_ci      { ok: false, key: 'agent2-key', cert: 'agent2-cert' },
561cb0ef41Sopenharmony_ci      { ok: true, key: 'agent3-key', cert: 'agent3-cert' },
571cb0ef41Sopenharmony_ci    ] },
581cb0ef41Sopenharmony_ci];
591cb0ef41Sopenharmony_ci
601cb0ef41Sopenharmony_ci
611cb0ef41Sopenharmony_cifunction loadPEM(n) {
621cb0ef41Sopenharmony_ci  return fixtures.readKey(`${n}.pem`);
631cb0ef41Sopenharmony_ci}
641cb0ef41Sopenharmony_ci
651cb0ef41Sopenharmony_cilet successfulTests = 0;
661cb0ef41Sopenharmony_ci
671cb0ef41Sopenharmony_cifunction testServers(index, servers, clientOptions, cb) {
681cb0ef41Sopenharmony_ci  const serverOptions = servers[index];
691cb0ef41Sopenharmony_ci  if (!serverOptions) {
701cb0ef41Sopenharmony_ci    cb();
711cb0ef41Sopenharmony_ci    return;
721cb0ef41Sopenharmony_ci  }
731cb0ef41Sopenharmony_ci
741cb0ef41Sopenharmony_ci  const ok = serverOptions.ok;
751cb0ef41Sopenharmony_ci
761cb0ef41Sopenharmony_ci  if (serverOptions.key) {
771cb0ef41Sopenharmony_ci    serverOptions.key = loadPEM(serverOptions.key);
781cb0ef41Sopenharmony_ci  }
791cb0ef41Sopenharmony_ci
801cb0ef41Sopenharmony_ci  if (serverOptions.cert) {
811cb0ef41Sopenharmony_ci    serverOptions.cert = loadPEM(serverOptions.cert);
821cb0ef41Sopenharmony_ci  }
831cb0ef41Sopenharmony_ci
841cb0ef41Sopenharmony_ci  const server = tls.createServer(serverOptions, common.mustCall(function(s) {
851cb0ef41Sopenharmony_ci    s.end('hello world\n');
861cb0ef41Sopenharmony_ci  }));
871cb0ef41Sopenharmony_ci
881cb0ef41Sopenharmony_ci  server.listen(0, common.mustCall(function() {
891cb0ef41Sopenharmony_ci    let b = '';
901cb0ef41Sopenharmony_ci
911cb0ef41Sopenharmony_ci    console.error('connecting...');
921cb0ef41Sopenharmony_ci    clientOptions.port = this.address().port;
931cb0ef41Sopenharmony_ci    const client = tls.connect(clientOptions, common.mustCall(function() {
941cb0ef41Sopenharmony_ci      const authorized = client.authorized ||
951cb0ef41Sopenharmony_ci          (client.authorizationError === 'ERR_TLS_CERT_ALTNAME_INVALID');
961cb0ef41Sopenharmony_ci
971cb0ef41Sopenharmony_ci      console.error(`expected: ${ok} authed: ${authorized}`);
981cb0ef41Sopenharmony_ci
991cb0ef41Sopenharmony_ci      assert.strictEqual(authorized, ok);
1001cb0ef41Sopenharmony_ci      server.close();
1011cb0ef41Sopenharmony_ci    }));
1021cb0ef41Sopenharmony_ci
1031cb0ef41Sopenharmony_ci    client.on('data', function(d) {
1041cb0ef41Sopenharmony_ci      b += d.toString();
1051cb0ef41Sopenharmony_ci    });
1061cb0ef41Sopenharmony_ci
1071cb0ef41Sopenharmony_ci    client.on('end', common.mustCall(function() {
1081cb0ef41Sopenharmony_ci      assert.strictEqual(b, 'hello world\n');
1091cb0ef41Sopenharmony_ci    }));
1101cb0ef41Sopenharmony_ci
1111cb0ef41Sopenharmony_ci    client.on('close', common.mustCall(function() {
1121cb0ef41Sopenharmony_ci      testServers(index + 1, servers, clientOptions, cb);
1131cb0ef41Sopenharmony_ci    }));
1141cb0ef41Sopenharmony_ci  }));
1151cb0ef41Sopenharmony_ci}
1161cb0ef41Sopenharmony_ci
1171cb0ef41Sopenharmony_ci
1181cb0ef41Sopenharmony_cifunction runTest(testIndex) {
1191cb0ef41Sopenharmony_ci  const tcase = testCases[testIndex];
1201cb0ef41Sopenharmony_ci  if (!tcase) return;
1211cb0ef41Sopenharmony_ci
1221cb0ef41Sopenharmony_ci  const clientOptions = {
1231cb0ef41Sopenharmony_ci    port: undefined,
1241cb0ef41Sopenharmony_ci    ca: tcase.ca.map(loadPEM),
1251cb0ef41Sopenharmony_ci    key: loadPEM(tcase.key),
1261cb0ef41Sopenharmony_ci    cert: loadPEM(tcase.cert),
1271cb0ef41Sopenharmony_ci    rejectUnauthorized: false
1281cb0ef41Sopenharmony_ci  };
1291cb0ef41Sopenharmony_ci
1301cb0ef41Sopenharmony_ci
1311cb0ef41Sopenharmony_ci  testServers(0, tcase.servers, clientOptions, common.mustCall(function() {
1321cb0ef41Sopenharmony_ci    successfulTests++;
1331cb0ef41Sopenharmony_ci    runTest(testIndex + 1);
1341cb0ef41Sopenharmony_ci  }));
1351cb0ef41Sopenharmony_ci}
1361cb0ef41Sopenharmony_ci
1371cb0ef41Sopenharmony_ci
1381cb0ef41Sopenharmony_cirunTest(0);
1391cb0ef41Sopenharmony_ci
1401cb0ef41Sopenharmony_ci
1411cb0ef41Sopenharmony_ciprocess.on('exit', function() {
1421cb0ef41Sopenharmony_ci  console.log(`successful tests: ${successfulTests}`);
1431cb0ef41Sopenharmony_ci  assert.strictEqual(successfulTests, testCases.length);
1441cb0ef41Sopenharmony_ci});
145