quic

package module
v0.0.0-...-6f70bfd Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Apr 26, 2022 License: MIT Imports: 19 Imported by: 0

README

quic-at-home is a toy implementation of a toy transport protocol derived from QUIC. It shouldn't be used, because it is bad. The 3 kLOC of it get you some cool features:

A client has to present a cookie that was previously by the server for it to even consider shaking the client's hand. A cookie is an authentication tag of user's IP address and port tuple produced by ChaCha20-Poly1305, but of course any other AEAD will do just fine. This allows the server to not maintain any per-cookie state.

To avoid the possibility of amplification attacks, cookies are sent only in response to initial packets, which are always 1280 bytes in size.

See internal/cookie for details.

Noise IK handshake

Noise IK allows server to not keep any half-open connections around. When the server receives a packet, it either accepts the connecction or rejects it. Unfortunately this requires client to know the key of the server they are connecting to.

Best-effort message sequence service

Send messages to your peer! They will hopefully arrive. Some of them, maybe. Regardless of outcome, if they arrive, they will in exactly the order they have been sent, and each one will arrive at most once. No (or very large) size limit, although harsh reality imposes one. Like IP fragmentation, but actually good! plus congestion control and authenticated encryption on top. Or like QUIC's DATAGRAM extension, but with fragmentation and extra guarantees about order and no suspectibility to packet replay.

IP migration

Stay connected while switching between Wi-Fi and LTE! Like QUIC's, but passive migration only.

IP migration probes are implemented by sending a single ACK-eliciting packet to both the last successfully probed address and the new one that the endpoint is believed to be migrating to. Once a probe packet for address A is acknowledged in a packet originating from that address, address A is used as the sending address. Otherwise, if ACK of the probe originates from the old address, migration is aborted.

Terrible congestion controller

Congestion controller operates under assumption that transmission rate is always application-bound. This leads to simpler congestion controller, which is always in slow start. To deal with harsh reality of limited link capacities, whenever congestion occurs, congestion window is reset.

And many more!

internal/sec implements Noise_IK_25519_ChaChaPoly_BLAKE2b

internal/udp implements some terrible, cursed garbage a UDP PacketConn with GRO and GSO support (each, respectively, lets you receive and send UDP packets without hogging CPU too hard.)

stream_reassembler.go contains more cursed code a thing that lets you put fragments of a byte sequence back together, without terrible worst-case comp. and memory complexities.

Documentation

Index

Constants

This section is empty.

Variables

View Source
var ErrAgain = errors.New("try again")

Functions

This section is empty.

Types

type Config

type Config struct {
	// StreamReceiveWindow specifies size of the receive window to use for
	// receiving the reliable stream data. StreamReceiveWindow must be
	// non-zero.
	StreamReceiveWindow int

	// MaxStreamBytesInFlight bounds how many bytes carrying reliable stream
	// data can be in flight.
	MaxStreamBytesInFlight int

	// PrivateKey contains the static private key. The public counterpart is
	// presented to the remote party during the handshake. The remote party
	// may discriminate and deny peers based on their public keys.
	PrivateKey PrivateKey

	// Listen for incoming connections.
	Listen bool
}

Config is used to configure a Mux. A config must not be modified once while in use. A Config may be in use by multiple Muxes simultaneously.

type Conn

type Conn struct {
	// contains filtered or unexported fields
}

func (*Conn) Close

func (c *Conn) Close() error

func (*Conn) Read

func (c *Conn) Read(b []byte) (int, error)

If c is closed, Read will read remaining stream contents before reporting an error.

func (*Conn) ReadMsg

func (c *Conn) ReadMsg(b []byte) (int, error)

func (*Conn) ReadMsgFrom

func (c *Conn) ReadMsgFrom(r io.Reader, max int) (int, error)

func (*Conn) SetMsgReceiveWindow

func (c *Conn) SetMsgReceiveWindow(n int)

SetMsgReceiveWindow sets the receive window size of the message ReadWriter. SetMsgReceiveWindow must not be called while the message ReadWriter is being read or written to.

func (*Conn) Write

func (c *Conn) Write(b []byte) (int, error)

func (*Conn) WriteMsg

func (c *Conn) WriteMsg(b []byte) (int, error)

type Mux

type Mux struct {
	// contains filtered or unexported fields
}

func ListenAddrPort

func ListenAddrPort(laddr netip.AddrPort, config *Config) (*Mux, error)

func (*Mux) Accept

func (m *Mux) Accept() (*Conn, error)

func (*Mux) Close

func (m *Mux) Close() error

Close shutdowns the Mux and its connections. Close does not close the connections gracefully -- peers might not observe a CLOSE frame. If graceful shutdown is desired, close each connection before closing the Mux.

func (*Mux) DialContextAddrPort

func (m *Mux) DialContextAddrPort(ctx context.Context, remoteStaticPublicKey PublicKey, raddr netip.AddrPort) (*Conn, error)

DialContextAddrPort dials raddr. Note that when DialContextAddrPort returns, peer might not have completed the handshake.

See github.com/nanokatze/quic-at-home/transportutil.Dial for a more convenient interface.

func (*Mux) LocalAddrPort

func (m *Mux) LocalAddrPort() netip.AddrPort

type PrivateKey

type PrivateKey []byte

func (PrivateKey) Public

func (privKey PrivateKey) Public() PublicKey

type PublicKey

type PublicKey []byte

Directories

Path Synopsis
internal

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL