| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105 |
- package main
- import (
- "crypto/rand"
- "crypto/rsa"
- "crypto/tls"
- "crypto/x509"
- "encoding/pem"
- "fmt"
- "io"
- "log"
- "math/big"
- quic "github.com/lucas-clemente/quic-go"
- )
- const addr = "localhost:4242"
- const message = "foobar"
- // We start a server echoing data on the first stream the client opens,
- // then connect with a client, send the message, and wait for its receipt.
- func main() {
- go func() { log.Fatal(echoServer()) }()
- err := clientMain()
- if err != nil {
- panic(err)
- }
- }
- // Start a server that echos all data on the first stream opened by the client
- func echoServer() error {
- listener, err := quic.ListenAddr(addr, generateTLSConfig(), nil)
- if err != nil {
- return err
- }
- sess, err := listener.Accept()
- if err != nil {
- return err
- }
- stream, err := sess.AcceptStream()
- if err != nil {
- panic(err)
- }
- // Echo through the loggingWriter
- _, err = io.Copy(loggingWriter{stream}, stream)
- return err
- }
- func clientMain() error {
- session, err := quic.DialAddr(addr, &tls.Config{InsecureSkipVerify: true}, nil)
- if err != nil {
- return err
- }
- stream, err := session.OpenStreamSync()
- if err != nil {
- return err
- }
- fmt.Printf("Client: Sending '%s'\n", message)
- _, err = stream.Write([]byte(message))
- if err != nil {
- return err
- }
- buf := make([]byte, len(message))
- _, err = io.ReadFull(stream, buf)
- if err != nil {
- return err
- }
- fmt.Printf("Client: Got '%s'\n", buf)
- return nil
- }
- // A wrapper for io.Writer that also logs the message.
- type loggingWriter struct{ io.Writer }
- func (w loggingWriter) Write(b []byte) (int, error) {
- fmt.Printf("Server: Got '%s'\n", string(b))
- return w.Writer.Write(b)
- }
- // Setup a bare-bones TLS config for the server
- func generateTLSConfig() *tls.Config {
- key, err := rsa.GenerateKey(rand.Reader, 1024)
- if err != nil {
- panic(err)
- }
- template := x509.Certificate{SerialNumber: big.NewInt(1)}
- certDER, err := x509.CreateCertificate(rand.Reader, &template, &template, &key.PublicKey, key)
- if err != nil {
- panic(err)
- }
- keyPEM := pem.EncodeToMemory(&pem.Block{Type: "RSA PRIVATE KEY", Bytes: x509.MarshalPKCS1PrivateKey(key)})
- certPEM := pem.EncodeToMemory(&pem.Block{Type: "CERTIFICATE", Bytes: certDER})
- tlsCert, err := tls.X509KeyPair(certPEM, keyPEM)
- if err != nil {
- panic(err)
- }
- return &tls.Config{Certificates: []tls.Certificate{tlsCert}}
- }
|