Menu

Best Network Technologies

By Noam Yadgar
#design-patterns #network #tcp #grpc #rest #websocket #message broker #system-design #software-engineering

Network communication in general is a very wide topic. It can probably fit in a large section of the New York Public Library (if not already). Fitting even a brief overview of this subject from bottom to top in a single article, might be a bit pretentious. So I won’t do it. Instead, I’ll be focusing mainly on software design patterns and technologies for communicating over a network.

Let’s unravel the mystery

We will start our journey with TCP/IP . Unless your program directly communicates using electromagnetic waves, there’s a good chance that your program will communicate with the world, using TCP/IP. I know you’ve heard this term before, but what is it?

TCP/IP, or more accurately the Internet Protocol Suite, is pretty much the internet as we know it. This suite consists a set of communication protocols organized into four layers:

  • Datalink: The Datalink layer handles the physical infrastructure for communicating over a network. Things like hardware drivers, NICs (Network Interface Card), wireless networks, Ethernet cables, modems, and more.
  • Internet: The Internet layer is controlling data delivery, as well as routing. It takes files, chopped into packets of data, and delivers these packets through different routes. When they reach their destination, it’s also responsible for reassembling the packets. The internet layer is essentially hoping through networks to find optimal routes for packets to be delivered. It hops through a network of networks => an inter-network => internet :)
  • Transport: The Transport layer forms a connection between two devices. It’s also responsible for chopping the data into packets and attaching appropriate headers for the internet layer. Two of the most popular transport protocols are:
    • TCP (Transport Control Protocol): Guarantees no data loss, receiver needs to send acknowledgments. (more common)
    • UDP (User Datagram Protocol): Loss-tolerant and faster connection, receiver is not sending acknowledgments.
  • Application: The Application layer is the one we’re going to focus on in this article. Since this is the layer we (developers/users) interact with the most. This layer uses Ports to communicate with the transport layer. Some of the most popular application protocols are: HTTP, POP3, FTP, SMTP, SSH, IMAP, and many more.

Let’s make our simple TCP server

To unravel the mystery, let’s create a simple application on top of a TCP server. We’ll be using Go because setting up a TCP server with Go is incredibly easy.

 1package main
 2
 3import (
 4	"fmt"
 5	"net"
 6)
 7
 8func main() {
 9	srv, err := net.Listen("tcp", "localhost:7776")
10	if err != nil {
11		panic(err)
12	}
13	defer srv.Close()
14
15	fmt.Println("starting server")
16	for {
17		c, err := srv.Accept()
18		if err != nil {
19			panic(err)
20		}
21		go handleReq(c)
22	}
23}
24
25func handleReq(c net.Conn) {
26	defer c.Close()
27	buf := make([]byte, 512)
28	c.Write([]byte("Enter your name:\n"))
29	if _, err := c.Read(buf); err != nil {
30		c.Write([]byte(err.Error()))
31		return
32	}
33	c.Write([]byte(fmt.Sprintf("Hello %s", string(buf))))
34}

If we’ll go ahead and run this main.go file:

1go run main.go
2starting server

In another terminal, We can connect directly to localhost:7776 using a tool like netcat:

1nc localhost 7776
2Enter your name:
3Jane
4Hello Jane

Cool! We just launched our very own TCP server. Theoretically, we can write our application protocol and implement this protocol in our different programs, so they would be able to communicate with each other, using this protocol.

Of course, writing your own protocol is not an easy task. Lucky for us, there are some brilliant applications already written and ready to use whenever we need to plug our program into a network. All of them cover a variety of use cases. Here are some of the most useful software-network design patterns and technologies:


REST API

REST API (Representational State Transfer), is probably the most popular and successful design pattern. It was proposed by Roy Fielding in 2000. The REST API leverages the HTTP (Hypertext Transfer Protocol) that has been around since 1989.

Use Case

To create a server-client relationship, mostly for websites, APIs and occasionally microservices.

General Architecture:
REST APIs are based on Resources and a set of four operations (known as CRUD) that clients can communicate with a server:

  • Create: Using the HTTP POST method, lets clients create resources.
  • Read: Using the HTTP GET method, lets clients read resources.
  • Update: Using the HTTP PUT/PATCH method, lets clients update resources.
  • Delete: Using the HTTP DELETE method, lets clients delete resources.
PROS CONS
Reliable and stateless Fixed endpoints
Easy to scale Less suitable for triggering live events
Widely supported Clients must adapt to the API
Automatic caching Uniform interface

Example

Searching for a book in an imaginary bookstore might look like this:

GET https://api.bookstore.net/books/a-great-book

The server response will be typically in a JSON format:

 1[
 2  {
 3    "id": "17a1ee01-9a6b-4954-8a53-8fb2291e8fff",
 4    "title": "a great book",
 5    "author": "jacob jacobson",
 6    "genre": "education",
 7    "price": 14.99,
 8    "currency": "USD",
 9    "description": "a very good book",
10    "stock": 48
11  }
12]

The REST API doesn’t need that much of an advertisement, since it’s still probably the most popular design choice for software communication, especially websites and web communication in general.

GraphQL

The GraphQL is not a design pattern/convention, but a technology. It was originally written by Meta in 2012 and was open-sourced in 2015. Similarly to a REST API, the GraphQL API is using HTTP. But, unlike REST API, GraphQL has a layer of software written on top of HTTP. All requests are made with an HTTP GET request, using a single endpoint.

Use Case

GraphQL was made as a more flexible and bandwidth-light alternative to REST APIs. As smartphones gained popularity, the number of API consumers and the need for different client apps rose significantly. This caused two main issues:

  • All clients had to adapt the API’s interface. It mostly means that it’s harder to deploy schema updates for the served resources.
  • REST APIs have fixed responses, which means that when a client needs to query for custom data or joined resources, it needs to make multiple API calls. On large scales, this translates to an explosion of expensive API calls.

The Graph Query Language solves this by letting the client query data that fits the client’s needs exactly. Even if it means interacting with relational resources in a single call.

PROS CONS
Flexible and Predictable Requires extra software from both ends
Easily scalable Doesn’t use HTTP native status codes
Stateless No automatic caching
Economic More complicated relative to a REST API

Example

Let’s query only for book IDs, titles, and stock amounts

1{
2  books(filter: { title: { eq: "a great book" } }) {
3    items {
4      id
5      title
6      stock
7    }
8  }
9}

The server response might be:

 1{
 2  "data": {
 3    "items": [
 4      {
 5        "id": "17a1ee01-9a6b-4954-8a53-8fb2291e8fff",
 6        "name": "a great book",
 7        "stock": 48
 8      }
 9    ]
10  }
11}

Since GraphQL is considered an alternative to REST API, the two are commonly compared. And since REST API is simpler to implement, I think that the use of GraphQL should be considered heavily before choosing to implement it.

WebSocket

Unlike the last two, WebSocket is an application-level protocol. While HTTP requests are prefixed with http:// or https:// for secured requests, WebSocket will use ws:// or wss:// for secured connection.

A WebSocket is essentially a Full-Duplex (Bi-Directional) TCP session between two sides. A client is making an HTTP GET request to a WebSocket server. The server is responding with a 101 “switching protocol” status, and a single, opened TCP connection is being made by the two sides, allowing them to exchange data in real-time.

Use Case

Let’s imagine you’re writing a beautiful application that lets remote users edit documents collaboratively. To make for the best user experience, you want your app to reflect changes in realtime. So if you and I are working on the same document, we will see each other inputs in real-time.

We can make this effect using a REST API. By making the client constantly request updates from the server (maybe every n units of time). This pattern is known as “short polling” and it’s very expensive and inefficient. We can do a lot better, by implementing a “long polling” mechanism. The client asks for updates, the server keeps the connection open until an update occurs (or it passed the timeout). Once the response is ready, the server answers the call, closes the connection, and, the client can immediately ask for updates again.

But using a REST API for this can be a bit slow, and besides, we don’t want to make requests while the users are not active (when the server constantly reaches timeouts). A WebSocket will keep a Full-Duplex TCP “highway” between each client and the server, allowing for event-driven data exchange, instead of time-based data exchange.

PROS CONS
Realtime data exchange Stateful
Fast and efficient Challenging to scale out
Widely supported Hardware requirements are higher (relative to others)
Easy to implement Required to keep connections alive for a long time

Example

Here’s a simple NodeJs chat. This is the server:

 1const ws = require("ws");
 2
 3const wss = new ws.WebSocketServer({ port: 9999 });
 4
 5console.log("starting chat");
 6wss.on("connection", (ws) => {
 7  ws.on("message", (data) => {
 8    wss.clients.forEach((client) => {
 9      client.send(data);
10    });
11  });
12});

And the client:

 1const w = require("ws");
 2const readline = require("readline");
 3const user = process.argv[2];
 4
 5const rl = readline.createInterface({
 6  input: process.stdin,
 7  output: process.stdout,
 8});
 9const ws = new w.WebSocket("ws://localhost:9999");
10
11ws.on("open", () => {
12  rl.on("line", (msg) => {
13    ws.send(user + " > " + msg);
14    process.stdout.moveCursor(0, -1);
15    process.stdout.clearLine(1);
16  });
17});
18
19ws.on("message", (data) => console.log("%s", data));

If we run the server and two other separated terminals of clients, we can chat:

1cat > hi, I am a cat  cat > hi, I am a cat
2dog > hi cat!         dog > hi cat!
3I am a dog...       

gRPC

Moving outside of the browser, gRPC (Remote Procedure Call), is an open-sourced, modern implementation of RPC created by Google in 2015. Similarly to REST API, gRPC uses the HTTP protocol. To be more accurate, it’s built on top of HTTP/2.0, which is faster and supports streams. But this is where the technical similarities end, as the gRPC framework takes a very different approach.

To quote the official documentation:

In gRPC, a client application can directly call a method on a server application on a different machine as if it were a local object, making it easier for you to create distributed applications and services.

gRPC uses something called Protocol Buffers (.proto files). A protobuf file describes structured data (such as interfaces and datatypes) in a language-neutral syntax. With a Protocol Buffer compiler (protoc), you can generate data access classes in all popular programming languages. Those classes will have methods to send, receive and serialize protobuf data, which RPC is sending as pure binaries. For each language, you can use the gRPC package to spin up a server and hook it to the compiled protobufs services, creating a bidirectional communication between clients and a server, using native code and fast data serialization (around 5 times faster than JSON)

Use Case

The gRPC framework is great for microservices communication. Using the .proto files, It’s easy to create API contracts that have well-defined data types and methods. The protobuf compiler (protoc) automatically creates data access classes in different programming languages of choice, for both a gRPC Server and a gRPC Stub (client).

PROS CONS
Supports streams in bidirectional communication Requires extra software
High performance and well-defined APIs Less compatible with functional languages
Great for scale Doesn’t have full browser support
Auto-generated boilerplates No consistent error and caching infrastructure

Example

 1syntax = "proto3";
 2
 3service Greeter {
 4  rpc SayHello (HelloRequest) returns (HelloReply) {}
 5}
 6
 7message HelloRequest {
 8  string name = 1;
 9}
10
11message HelloReply {
12  string greet = 1;
13}

This simple .proto file defines a service with a single method and two datatypes. Let’s compile this file to Golang (of course we can choose a different language)

1protoc --proto_path=./proto \
2       --go_out=. --go_opt=Mgreeter.proto=/greeter \
3       --go-grpc_out=. --go-grpc_opt=Mgreeter.proto=/greeter \
4       proto/greeter.proto

This will automatically generate a Go package, consisting of two files:

1greeter.pb.go
2greeter_grpc.pb.go

We can use this package and implement a gRPC server:

 1package main
 2
 3import (
 4	"context"
 5	"log"
 6	"net"
 7
 8	"google.golang.org/grpc"
 9	pb "my_grpc_service/greeter"
10)
11
12type server struct {
13	pb.UnimplementedGreeterServer
14}
15
16func (s *server) SayHello(ctx context.Context, in *pb.HelloRequest) (*pb.HelloReply, error) {
17	return &pb.HelloReply{Greet: "Hello " + in.GetName()}, nil
18}
19
20func main() {
21	srv, err := net.Listen("tcp", ":5555")
22	if err != nil {
23		panic(err)
24	}
25	s := grpc.NewServer()
26	pb.RegisterGreeterServer(s, &server{})
27	log.Printf("server srvtening at %v", srv.Addr())
28	if err := s.Serve(srv); err != nil {
29		log.Fatalf("failed to serve: %v", err)
30	}
31}

And a client:

 1package main
 2
 3import (
 4	"context"
 5	"flag"
 6	"fmt"
 7	"time"
 8
 9	"google.golang.org/grpc"
10	"google.golang.org/grpc/credentials/insecure"
11	pb "my_grpc_service/greeter"
12)
13
14var name = flag.String("name", "all", "Name to greet")
15
16func main() {
17	flag.Parse()
18	conn, err := grpc.Dial("localhost:5555", grpc.WithTransportCredentials(insecure.NewCredentials()))
19	if err != nil {
20    panic(err)
21	}
22	defer conn.Close()
23	c := pb.NewGreeterClient(conn)
24
25	ctx, cancel := context.WithTimeout(context.Background(), time.Second)
26	defer cancel()
27
28	if res, err := c.SayHello(ctx, &pb.HelloRequest{Name: *name}); err != nil {
29		fmt.Println(err)
30	} else {
31	  fmt.Println(res.GetGreet())
32  }
33}

We can run our server:

1go run greeter_server/main.go
22023/01/13 14:21:58 server listening at [::]:5555

And in another terminal, run our client project with a name flag:

1go run greeter_client/main.go --name Jane
2Hello Jane

Notice how our client directly calls SayHello as if it was a local procedure.

Message Broker

In essence, a message broker sits in the middle of endpoints, manages their communication, and, decouples them. Instead of directly connecting microservices, we spin up a dedicated service that acts as a mediator between the microservices.

This form of architecture makes a lot of sense in the topological aspect of microservices networking. Two of the most popular design patterns that are based on a message broker are:

  • Message Queue : A service can send a message to another service, but instead of directly sending the message, this message will be stored by the broker in a FIFO (1st in 1st out) message queue. The broker can store this queue in some form of persistent memory, so whenever the target service is ready, it can pull messages from the queue. This pattern decouples the sender and the receiver and creates an asynchronous behavior. The target service doesn’t necessarily need to be alive for the message to be sent. We can set up multiple consumers for a single queue, and they will “compete” for the messages, essentially pulling out messages from the same queue. This is great when you need to scale out your service.
  • Publish/Subscription This pattern is a bit more flexible than a “simple” message queue. Instead of setting an end to end queues by the broker, we will use topics/channels. Publishers can produce messages to specific topics and, consumers can subscribe to any topic, they want to consume messages. Consumers don’t necessarily have to “compete” for messages from the same topics, but receive the same messages. Some “pub/sub” brokers like Apache Kafka go as far as using streams instead of queues. So consumers can read \(n\) messages, and commit to a stream offset.

Use Case

Here is some graph theory for you: Imagine a cluster of microservices, all working together to serve a product. Let’s suppose we want to connect all microservices to each other. Using any form of duplex communication can potentially yield \(\sum\_{i=0}^{n-1}i\) connections (duplex) to manage as \(n=services\).

So, a cluster of just 10 services, directly communicating can form 45 distinct connections. Besides the large number of connections we have to manage (which can also be managed quite elegantly using a service discovary ), there are a few other big issues, a message broker will overcome:

  • Clients don’t have to be alive, waiting for responses.
  • Services don’t have to be alive to receive “requests”.
  • Services can be completely oblivious of each other.

Our 10 services all connected together will form 10 connection to a message broker (Known as a Star network topology). It’s important to point out the message brokers are typically not optimized for large data transfer (as opposed to technologies like gRPC).

PROS CONS
Services are decoupled Requires a running service
Asynchronous behavior Requires a form of persistent memory
Great for scale If the brokers are down, we lose all communication
Easy to connect Might be overkill for small projects

Example

Let’s spin up the popular message broker RabbitMQ , using Docker

1docker run -d --hostname codepilot --name rabbitmq -p 5672:5672 rabbitmq:3

Now we can write a program that connects and sends a message to a quque:

This code is taken from the official RabbitMQ tutorial

 1package main
 2
 3import (
 4	"context"
 5	"log"
 6	"time"
 7
 8	amqp "github.com/rabbitmq/amqp091-go"
 9)
10
11func failOnError(err error, msg string) {
12	if err != nil {
13		log.Panicf("%s: %s", msg, err)
14	}
15}
16
17func main() {
18	conn, err := amqp.Dial("amqp://guest:guest@localhost:5672/")
19	failOnError(err, "Failed to connect to RabbitMQ")
20	defer conn.Close()
21
22	ch, err := conn.Channel()
23	failOnError(err, "Failed to open a channel")
24	defer ch.Close()
25
26	q, err := ch.QueueDeclare(
27		"greeter", // name
28		false,     // durable
29		false,     // delete when unused
30		false,     // exclusive
31		false,     // no-wait
32		nil,       // arguments
33	)
34	failOnError(err, "Failed to declare a queue")
35	ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
36	defer cancel()
37
38	body := "Hello World!"
39	err = ch.PublishWithContext(ctx,
40		"",     // exchange
41		q.Name, // routing key
42		false,  // mandatory
43		false,  // immediate
44		amqp.Publishing{
45			ContentType: "text/plain",
46			Body:        []byte(body),
47		})
48	failOnError(err, "Failed to publish a message")
49	log.Printf(" [x] Sent %s\n", body)
50}

And let’s write a program that receives messages from this queue:

 1package main
 2
 3import (
 4	"log"
 5
 6	amqp "github.com/rabbitmq/amqp091-go"
 7)
 8
 9func failOnError(err error, msg string) {
10	if err != nil {
11		log.Panicf("%s: %s", msg, err)
12	}
13}
14
15func main() {
16	conn, err := amqp.Dial("amqp://guest:guest@localhost:5672/")
17	failOnError(err, "Failed to connect to RabbitMQ")
18	defer conn.Close()
19
20	ch, err := conn.Channel()
21	failOnError(err, "Failed to open a channel")
22	defer ch.Close()
23
24	q, err := ch.QueueDeclare(
25		"greeter", // name
26		false,     // durable
27		false,     // delete when unused
28		false,     // exclusive
29		false,     // no-wait
30		nil,       // arguments
31	)
32	failOnError(err, "Failed to declare a queue")
33
34	msgs, err := ch.Consume(
35		q.Name, // queue
36		"",     // consumer
37		true,   // auto-ack
38		false,  // exclusive
39		false,  // no-local
40		false,  // no-wait
41		nil,    // args
42	)
43	failOnError(err, "Failed to register a consumer")
44
45	var forever chan struct{}
46
47	go func() {
48		for d := range msgs {
49			log.Printf("Received a message: %s", d.Body)
50		}
51	}()
52
53	log.Printf(" [*] Waiting for messages. To exit press CTRL+C")
54	<-forever
55}

If we run the sender.go

1go run sender.go
22023/02/02 16:00:11  [x] Sent Hello World!

Let’s stop the sender and run the receiver.go

1go run receiver.go
22023/02/02 16:01:38  [*] Waiting for messages. To exit press CTRL+C
32023/02/02 16:01:38 Received a message: Hello World!

Notice how we didn’t have to keep the two endpoints alive at the same time. The sender is simply sending messages to the greeter queue and, the receiver is simply consuming messages from this queue.

Wrap up

This is where I’m going to end, as this article is already getting quite long. Of course, there are many more protocols and technologies to talk about, such as SMTP for email transfer, FTP for file transfer, SSH for shell connection and many more. Maybe I’ll cover some of them in the future. I hope you’ve found this article interesting and enriching. If you have any suggestions, please don’t hesitate to contact me.

Thank you for reading.