Skip to content

Latest commit

 

History

History
439 lines (333 loc) · 18.8 KB

readme.md

File metadata and controls

439 lines (333 loc) · 18.8 KB

tls.zig

Zig TLS library, characteristics:

  • TLS 1.2 and TLS 1.3 client
  • TLS 1.3 server
  • handles client authentication
  • tested with many domains, handles badssl URL's
  • options to select client cipher suites to use, named groups, ...
  • can configure Wireshark to show decrypted traffic
  • better performance, more modular, more testable, connect to more real world sites than standard library implementation
  • can be used with standard library HTTP client (There is proposal to allow changing TLS implementation of std.http, for now we need to to modify std lib source to switch to alternate TLS library.)

Client

Here is simple example of how to use library.
To upgrade existing tcp connection to the tls connection call tls.client:

    // Establish tcp connection
    var tcp = try std.net.tcpConnectToHost(allocator, host, port);
    defer tcp.close();

    // Load system root certificates
    var root_ca = try tls.config.CertBundle.fromSystem(allocator);
    defer root_ca.deinit(allocator);

    // Upgrade tcp connection to tls
    var conn = try tls.client(tcp, .{
        .host = host,
        .root_ca = root_ca,
    });

After that you can use conn read/write methods as on plain tcp connection.

Options

Second parameter in calling tls.client are tls.config.Client they can be used to force subset of implemented ciphers, set client authentication parameters, allow self insecure signed certificates, collect handshake diagnostics, exchange session keys with Wireshark to view decrypted traffic.

Select cipher suite

To use just ciphers which are graded secure or recommended on https://ciphersuite.info:

    var conn = try tls.client(tcp, .{
        .host = host,
        .root_ca = root_ca,
        .cipher_suites = tls.cipher_suites.secure,
    });

cipher_suites can be used to force tls 1.3 only or tls 1.2 only ciphers. Or to reorder cipher preferences.

Client authentication

If server requires client authentication set auth attribute in options. You need to prepare certificate key pair with client certificate(s) and client private key.

    // Prepare client authentication key pair
    var auth = try tls.config.CertKeyPair.load(allocator, cert_dir, "cert.pem", "key.pem");
    defer auth.deinit(allocator);

    var conn = try tls.client(tcp, .{
        .host = host,
        .root_ca = root_ca,
        .auth = auth,
    });

When client receives certificate request from server during handshake it will respond with client certificates message build from provided certificate bundle and client certificate verify message where verify data is signed with client private key.

If authentication is not provided client will respond with empty certificate message when server requests authentication (as specified in RFC).

Logging tls session keys

Session keys can be written to file so that external programs can decrypt TLS connections. Wireshark can use these log files to decrypt packets. You can tell Wireshark where to find the key file via Edit→Preferences→Protocols→SSL→(Pre)-Master-Secret log filename.

Key logging is enabled by setting the environment variable SSLKEYLOGFILE to point to a file. And enabling key log callback in client options:

    var conn = try tls.client(tcp, .{
        .host = host,
        .root_ca = root_ca,
        .key_log_callback = tls.config.key_log.callback,
    });

Server

Library also has minimal, TLS 1.3 only server implementation. To upgrade tcp to tls connection:

    // Load server certificate key pair
    var auth = try tls.config.CertKeyPair.load(allocator, dir, "localhost_ec/cert.pem", "localhost_ec/key.pem");
    defer auth.deinit(allocator);
    
    // Tcp listener
    const port = 9443;
    const address = std.net.Address.initIp4([4]u8{ 127, 0, 0, 1 }, port);
    var server = try address.listen(.{
        .reuse_address = true,
    });
    
     // Tcp accept
     const tcp = try server.accept();
     defer tcp.stream.close();

     // Upgrade tcp to tls
     var conn = try tls.server(tcp.stream, .{ .auth = auth });
     
     // use conn

Examples

Top sites

Starting from Cloudflare list of 10000 top domains, filtering those which can't be resolved got list of ~6k domains which are used to test establishing tls connection. If the connection fails test runs curl on the same domain, if curl can't connect it is count as error, if curl connect counts as fail. For each domain test reports tls handshake parameters (tls version, cipher suite used, named group and signature scheme).

$ zig-out/bin/top_sites
✔️ facebook.com              tls_1_3 AES_128_GCM_SHA256                       x25519               ecdsa_secp256r1_sha256
✔️ ebay.com                  tls_1_3 AES_128_GCM_SHA256                       x25519               rsa_pss_rsae_sha256
✔️ live.com                  tls_1_3 AES_256_GCM_SHA384                       secp384r1            rsa_pss_rsae_sha256
✔️ drive.google.com          tls_1_3 AES_128_GCM_SHA256                       x25519_kyber768d00   ecdsa_secp256r1_sha256
✔️ github.com                tls_1_3 AES_128_GCM_SHA256                       x25519               ecdsa_secp256r1_sha256
...

stats:
         total: 6280
         success: 6270
                 tls 1.2: 1426
                 tls 1.3: 4844
         fail: 4
         error: 6

Zig's std library tls implementation on the same domains list:

stats:
         total: 6280
         success: 5637
         fail: 581
         error: 62

When I found domain which fails I use http_get example to test whether it is transient error or point to something interesting. Now only transient errors are left in that domains group.

http get

This example will connect to the domain, show response and tls statistic. You can change tls options to force tls version or specific cipher.

$ zig-out/bin/http_get google.com    
HTTP/1.0 301 Moved Permanently

832 bytes read

google.com
         tls version: tls_1_3
         cipher: AES_128_GCM_SHA256
         named group: x25519_kyber768d00
         signature scheme: ecdsa_secp256r1_sha256

badssl

Uses urls from badssl.com to test client implementation.

$ zig-out/bin/badssl 

Certificate Validation (High Risk)
If your browser connects to one of these sites, it could be very easy for an attacker to see and modify everything on web sites that you visit.
        ✅ expired.badssl.com error.CertificateExpired
        ✅ wrong.host.badssl.com error.CertificateHostMismatch
        ✅ self-signed.badssl.com error.CertificateIssuerNotFound
        ✅ untrusted-root.badssl.com error.CertificateIssuerNotFound

Interception Certificates (High Risk)
If your browser connects to one of these sites, it could be very easy for an attacker to see and modify everything on web sites that you visit. This may be due to interception software installed on your device.
        ✅ superfish.badssl.com error.CertificateIssuerNotFound
        ✅ edellroot.badssl.com error.CertificateIssuerNotFound
        ✅ dsdtestprovider.badssl.com error.CertificateIssuerNotFound
        ✅ preact-cli.badssl.com error.CertificateIssuerNotFound
        ✅ webpack-dev-server.badssl.com error.CertificateIssuerNotFound

Broken Cryptography (Medium Risk)
If your browser connects to one of these sites, an attacker with enough resources may be able to see and/or modify everything on web sites that you visit. This is because your browser supports connections settings that are outdated and known to have significant security flaws.
        ✅ rc4.badssl.com error.TlsAlertHandshakeFailure
        ✅ rc4-md5.badssl.com error.TlsAlertHandshakeFailure
        ✅ dh480.badssl.com error.TlsAlertHandshakeFailure
        ✅ dh512.badssl.com error.TlsAlertHandshakeFailure
        ✅ dh1024.badssl.com error.TlsAlertHandshakeFailure
        ✅ null.badssl.com error.TlsAlertHandshakeFailure

Legacy Cryptography (Moderate Risk)
If your browser connects to one of these sites, your web traffic is probably safe from attackers in the near future. However, your connections to some sites might not be using the strongest possible security. Your browser may use these settings in order to connect to some older sites.
        ✅ tls-v1-0.badssl.com error.TlsBadVersion
        ✅ tls-v1-1.badssl.com error.TlsBadVersion
        🆗 cbc.badssl.com
        ✅ 3des.badssl.com error.TlsAlertHandshakeFailure
        ✅ dh2048.badssl.com error.TlsAlertHandshakeFailure

Domain Security Policies
These are special tests for some specific browsers. These tests may be able to tell whether your browser uses advanced domain security policy mechanisms (HSTS, HPKP, SCT) to detect illegitimate certificates.
        🆗 revoked.badssl.com
        🆗 pinning-test.badssl.com
        ✅ no-sct.badssl.com error.CertificateIssuerNotFound

Secure (Uncommon)
These settings are secure. However, they are less common and even if your browser doesn't support them you probably won't have issues with most sites.
        🆗 1000-sans.badssl.com error.TlsUnsupportedFragmentedHandshakeMessage
        🆗 10000-sans.badssl.com error.TlsUnsupportedFragmentedHandshakeMessage
        🆗 sha384.badssl.com error.CertificateExpired
        🆗 sha512.badssl.com error.CertificateExpired
        🆗 rsa8192.badssl.com error.BufferOverflow
        🆗 no-subject.badssl.com error.CertificateExpired
        🆗 no-common-name.badssl.com error.CertificateExpired
        🆗 incomplete-chain.badssl.com error.CertificateIssuerNotFound

Secure (Common)
These settings are secure and commonly used by sites. Your browser will need to support most of these in order to connect to sites securely.
        ✅ tls-v1-2.badssl.com
        ✅ sha256.badssl.com
        ✅ rsa2048.badssl.com
        ✅ ecc256.badssl.com
        ✅ ecc384.badssl.com
        ✅ mozilla-modern.badssl.com

All ciphers

Tries all supported ciphers on some domain.

$ zig-out/bin/all_ciphers cloudflare.com
✔️ AES_128_GCM_SHA256 cloudflare.com
✔️ AES_256_GCM_SHA384 cloudflare.com
✔️ CHACHA20_POLY1305_SHA256 cloudflare.com
✔️ ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 cloudflare.com
✔️ ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 cloudflare.com
✔️ ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 cloudflare.com
✔️ ECDHE_RSA_WITH_AES_128_GCM_SHA256 cloudflare.com
✔️ ECDHE_RSA_WITH_AES_256_GCM_SHA384 cloudflare.com
✔️ ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 cloudflare.com
✔️ ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 cloudflare.com
✔️ ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 cloudflare.com
✔️ ECDHE_ECDSA_WITH_AES_128_CBC_SHA cloudflare.com
✔️ ECDHE_RSA_WITH_AES_128_CBC_SHA256 cloudflare.com
✔️ ECDHE_RSA_WITH_AES_256_CBC_SHA384 cloudflare.com
✔️ ECDHE_RSA_WITH_AES_128_CBC_SHA cloudflare.com
✔️ RSA_WITH_AES_128_CBC_SHA256 cloudflare.com
✔️ RSA_WITH_AES_128_CBC_SHA cloudflare.com

Using cloudflare.com as example because it supports all implemented ciphers.

Server and client example

Create local development certificates and keys:

$ cd example && ./cert.sh && cd -

This uses minica tool. Go compiler and go install dir in the path are required.

Start server and connect to with client to the server.

$ zig build && zig-out/bin/server& ; sleep 1 && zig-out/bin/client ; kill %1

Client authentication

After we have certificates created in previous example, here we will start Go tls server which requires client authentication and connect to that server with various different rsa and ec certificates using both tls 1.2 and 1.3.

$ zig build ; cd example/go_tls_server; go run server.go & ; cd - ; sleep 1 && zig-out/bin/client_auth ; kill %1

Equivalent curl is:

curl https://localhost:8443 --cacert example/cert/minica.pem --cert example/cert/client_rsa/cert.pem --key example/cert/client_rsa/key.pem

Usage with standard library http.Client

This library is only tls protocol implementation. Standard library has great http client. We can replace standard library tls implementation with this one and get http client with both tls 1.2 and 1.3 capability. Here are required changes, assuming that this library is available at lib/std/crypt/tls23 path.

This script will checkout tls.zig library, an fork of the zig repository and link tls.zig to the required path. After that we can point to that standard library copy while building zig project with --zig-lib-dir switch.

git clone https://github.com/ianic/tls.zig        
git clone -b tls23 https://github.com/ianic/zig
ln -s $(pwd)/tls.zig/src zig/lib/std/crypto/tls23

cd tls.zig
zig build --zig-lib-dir ../zig/lib
zig-out/bin/std_top_sites 

Performance comparison with standard library

Starting local server which will stream a text file (src/main.zig in this example) to the connected client:

$ zig build -Doptimize=ReleaseFast && zig-out/bin/server src/main.zig

Running 50 client request to that server by using this library and then by using standard library implementation and comparing them:

$ zig build -Doptimize=ReleaseFast && sudo ~/.local/bin/poop './zig-out/bin/client --cycles 50' 'zig-out/bin/client --cycles 50 --std'
Benchmark 1 (27 runs): ./zig-out/bin/client --cycles 50
  measurement          mean ± σ            min … max           outliers         delta
  wall_time           191ms ± 3.65ms     184ms …  199ms          0 ( 0%)        0%
  peak_rss            803KB ±    0       803KB …  803KB          0 ( 0%)        0%
  cpu_cycles          422M  ± 11.7M      401M  …  447M           0 ( 0%)        0%
  instructions       1.62G  ± 29.4M     1.56G  … 1.65G           0 ( 0%)        0%
  cache_references    478K  ± 83.2K      194K  …  540K           3 (11%)        0%
  cache_misses       12.0K  ± 2.34K     8.67K  … 19.2K           0 ( 0%)        0%
  branch_misses       255K  ± 21.0K      190K  …  277K           2 ( 7%)        0%
Benchmark 2 (23 runs): zig-out/bin/client --cycles 50 --std
  measurement          mean ± σ            min … max           outliers         delta
  wall_time           220ms ± 3.86ms     216ms …  234ms          1 ( 4%)        💩+ 15.5% ±  1.1%
  peak_rss            850KB ±  101KB     803KB … 1.06MB          4 (17%)        💩+  5.9% ±  4.9%
  cpu_cycles          564M  ± 14.7M      509M  …  592M           3 (13%)        💩+ 33.5% ±  1.8%
  instructions       2.21G  ± 54.2M     2.00G  … 2.25G           2 ( 9%)        💩+ 36.3% ±  1.5%
  cache_references    531K  ± 31.8K      424K  …  556K           2 ( 9%)        💩+ 11.1% ±  7.8%
  cache_misses       13.3K  ± 1.99K     9.21K  … 18.6K           1 ( 4%)          + 10.9% ± 10.4%
  branch_misses       250K  ± 9.70K      224K  …  264K           2 ( 9%)          -  2.0% ±  3.8%

Credits