Back

Monitoring latency: Cloudflare Workers vs Fly vs Koyeb vs Railway vs Render

Monitoring latency: Cloudflare Workers vs Fly vs Koyeb vs Railway vs Render
TD
Thibault Le Ouay Ducasse
13 min read
education

⚠️ We are using the default settings for each provider and conducting datacenter to datacenter requests. A real-world application's results are going to be different. ⚠️

You want to know which cloud providers offer the lowest latency?

In this post, I compare the latency of Cloudflare Workers, Fly, Koyeb, Railway and Render using OpenStatus.

I deployed the application on the cheapest or free tier offered by each provider.

For this test, I used a basic Hono server that returns a simple text response.

You can find the code here, it’s open source 😉.

OpenStatus monitored our endpoint every 10 minutes from 6 locations located in Amsterdam, Ashburn, Hong Kong, Johannesburg, Sao Paulo and Sydney.

It's a good way to test our own product and improve it.

Let's analyze the data from the past two weeks.

Cloudflare workers

Cloudflare Workers is a serverless platform by Cloudflare. It lets you build new applications using JavaScript/Typescript. You can deploy up to 100 worker scripts for free, running on more than 275 network locations.

Latency metrics

uptime

100%

fails

0#

total pings

10,956#

avg

182ms

p75

138ms

p90

695ms

p95

778ms

p99

991ms

Timing metrics

| Region | DNS (ms) | Connection (ms) | TLS Handshake (ms) | TTFB (ms) | Transfert (ms) | | ------ | -------- | --------------- | ------------------ | --------- | -------------- | | AMS | 17 | 2 | 17 | 27 | 0 | | GRU | 38 | 2 | 13 | 28 | 0 | | HKG | 19 | 2 | 13 | 29 | 0 | | IAD | 24 | 1 | 14 | 30 | 0 | | JNB | 123 | 168 | 182 | 185 | 0 | | SYD | 51 | 1 | 11 | 25 | 0 |

I can notice that Johannesburg's latency is about ten times higher than that of the other monitors.

Headers

From the Cloudflare request I can get the location of the workers that handle the request, with Cf-ray in the headers response.

| Checker region | Workers region | number of request | | -------------- | -------------- | ----------------- | | HKG | HKG | 1831 | | SYD | SYD | 1831 | | AMS | AMS | 1831 | | IAD | IAD | 1831 | | GRU | GRU | 1791 | | GRU | GIG | 40 | | JNB | AMS | 741 | | JNB | MUC | 4 | | JNB | HKG | 5 | | JNB | SIN | 6 | | JNB | NRT | 8 | | JNB | EWR | 10 | | JNB | CDG | 82 | | JNB | FRA | 276 | | JNB | LHR | 699 | | JNB | AMS | 741 |

I can see all the request from JNB is never routed to a nearby data-center.

Apart from the strange routing error in Johannesburg, Cloudflare workers are fast worldwide.

I have not experienced any cold start issues.

Fly.io

Fly.io simplifies deploying and running server-side applications globally. Developers can deploy their applications near users worldwide for low latency and high performance. It uses a lightweight Firecracker VM to easily deploy Docker images.

Latency metrics

uptime

100%

fails

0#

total pings

10,952#

avg

1,471ms

p75

1,514ms

p90

1,555ms

p95

1,626ms

p99

2,547ms

Timing metrics

| Region | DNS (ms) | Connection (ms) | TLS Handshake (ms) | TTFB (ms) | Transfert (ms) | | ------ | -------- | --------------- | ------------------ | --------- | -------------- | | AMS | 6 | 1 | 8 | 1469 | 0 | | GRU | 5 | 0 | 4 | 1431 | 0 | | HKG | 4 | 0 | 5 | 1473 | 0 | | IAD | 3 | 0 | 5 | 1470 | 0 | | JNB | 24 | 0 | 5 | 1423 | 0 | | SYD | 3 | 0 | 3 | 1489 | 0 |

The DNS is fast, our checker is attempting to connect to a region in the same data center, but our machine's cold start is slowing us down, leading to the high TTFB.

Here’s our config for Fly.io:

The primary region of our server is Amsterdam, and the fly instances is getting paused after a period of inactivity.

The machine starts slowly, as indicated by the logs showing a start time of 1.513643778s.

OpenStatus Prod metrics

If you update your fly.toml file to include the following, you can get the zero cold start and achieve a better latency.

This is our data for our production server deploy on Fly.io.

uptime

100%

fails

0#

total pings

12,076#

avg

61ms

p75

67ms

p90

164ms

p95

198ms

p99

327ms

We use Fly.io in production, and the machine never sleeps, yielding much better results.

Koyeb

Koyeb is a developer-friendly serverless platform that allows for global app deployment without the need for operations, servers, or infrastructure management. Koyeb offers a free Starter plan that includes one Web Service, one Database service. The platform focuses on ease of deployment and scalability for developers

Latency metrics

uptime

100%

fails

0#

total pings

10,955#

avg

539ms

p75

738ms

p90

881ms

p95

1,013ms

p99

1,525ms

Timing metrics

| Region | DNS (ms) | Connection (ms) | TLS Handshake (ms) | TTFB (ms) | Transfert (ms) | | ------ | -------- | --------------- | ------------------ | --------- | -------------- | | AMS | 50 | 2 | 17 | 107 | 0 | | GRU | 139 | 65 | 75 | 407 | 0 | | HKG | 48 | 2 | 13 | 321 | 0 | | IAD | 35 | 1 | 12 | 129 | 0 | | JNB | 298 | 1 | 11 | 720 | 0 | | SYD | 97 | 1 | 10 | 711 | 0 |

Headers

The request headers show that none of our requests are cached. They contain cf-cache-status: dynamic. Cloudflare handles the Koyeb edge layer. https://www.koyeb.com/blog/building-a-multi-region-service-mesh-with-kuma-envoy-anycast-bgp-and-mtls

Our requests follow this route:

Let’s see where did we hit the cf workers

| Checker region | Workers region | number of request | | -------------- | -------------- | ----------------- | | AMS | AMS | 1866 | | GRU | GRU | 504 | | GRU | IAD | 38 | | GRU | MIA | 688 | | GRU | EWR | 337 | | GRU | CIG | 299 | | HKG | HKG | 1866 | | IAD | IAD | 1866 | | JNB | JNB | 1861 | | JNB | AMS | 1 | | SYD | SYD | 1866 |

Koyeb Global Load Balancer region we hit:

| Checker region | Koyeb Global Load Balancer | number of request | | -------------- | -------------------------- | ----------------- | | AMS | FRA1 | 1866 | | GRU | WAS1 | 1866 | | HKG | SIN1 | 1866 | | IAD | WAS1 | 1866 | | JNB | PAR1 | 4 | | JNB | SIN1 | 1864 | | JNB | FRA1 | 1 | | JNB | SIN1 | 1866 |

I have deployed our app in the Frankfurt data-center.

Railway

Railway is a cloud platform designed for building, shipping, and monitoring applications without the need for Platform Engineers. It simplifies the application development process by offering seamless deployment and monitoring capabilities.

Latency metrics

uptime

99.99%

fails

1#

total pings

10,955#

avg

381ms

p75

469ms

p90

653ms

p95

661ms

p99

850ms

Timing metrics

| Region | DNS (ms) | Connection (ms) | TLS Handshake (ms) | TTFB (ms) | Transfert (ms) | | ------ | -------- | --------------- | ------------------ | --------- | -------------- | | AMS | 9 | 21 | 18 | 158 | 0 | | GRU | 14 | 115 | 127 | 178 | 0 | | HKG | 8 | 45 | 54 | 225 | 0 | | IAD | 7 | 2 | 14 | 65 | 0 | | JNB | 18 | 193 | 178 | 319 | 0 | | SYD | 21 | 108 | 105 | 280 | 0 |

Headers

The headers don't provide any information.

Railway is using Google Cloud Platform. It’s the only service that does not allow us to pick a specific region on the free plan. Our test app will be located to us-west1 Portland, Oregon. We can see that the latency is the lowest in IAD.

By default our app did not scale down to 0. It was always running. We don't have any cold start.

Render

Render is a platform that simplifies deploying and scaling web applications and services. It offers features like automated SSL, automatic scaling, native support for popular frameworks, and one-click deployments from Git. The platform focuses on simplicity and developer productivity.

Latency metrics

uptime

99.89%

fails

12#

total pings

10,946#

avg

451ms

p75

447ms

p90

591ms

p95

707ms

p99

902ms

Timing metrics

| Region | DNS (ms) | Connection (ms) | TLS Handshake (ms) | TTFB (ms) | Transfert (ms) | | ------ | -------- | --------------- | ------------------ | --------- | -------------- | | AMS | 20 | 2 | 7 | 107 | 0 | | GRU | 61 | 2 | 6 | 407 | 0 | | HKG | 76 | 2 | 6 | 321 | 0 | | IAD | 15 | 1 | 5 | 129 | 0 | | JNB | 36 | 161 | 167 | 720 | 0 | | SYD | 103 | 1 | 4 | 711 | 0 |

Headers

The headers don't provide any information.

I have deployed our app in the Frankfurt data-center.

According to the Render docs, the free tier will shut down the service after 15 minutes of inactivity. However, our app is being accessed by a monitor every 10 minutes. We should never scale down to 0.

I think the failures are due to the cold start of our app. We have a default timeout of 30s and the render app takes up to 50s to start.We might have hit an inflection point between cold and warm.

Conclusion

Here are the results of our test:

| Provider | Uptime | Fails Ping | Total Pings | AVG latency (ms) | P75 (ms) | P90 (ms) | P95 (ms) | P99 (ms) | | ---------- | ------ | ---------- | ----------- | ---------------- | -------- | -------- | -------- | -------- | | CF Workers | 100 | 0 | 10,956 | 182 | 138 | 690 | 778 | 991 | | Fly.io | 100 | 0 | 10,952 | 1,471 | 1,514 | 1,555 | 1,626 | 2,547 | | Koyeb | 100 | 0 | 10,955 | 536 | 738 | 881 | 1,013 | 1,525 | | Railway | 99.991 | 1 | 10,955 | 381 | 469 | 653 | 661 | 850 | | Render | 99.89 | 12 | 10,946 | 451 | 447 | 591 | 707 | 902 |

If you value low latency, Cloudflare Workers are the best option for fast global performance without cold start issues. They deploy your app worldwide efficiently.

For multi-region deployment, check out Koyeb and Fly.io.

For specific region deployment, Railway and Render are good choices.

Choosing a cloud provider involves considering not just latency but also user experience and pricing.

We use Fly.io in production and are satisfied with it.

Vercel

I haven't included Vercel in this test. But we have a blog post comparing Vercel Serverless vs Edge vs Serverless. You can find it here.

If you want to monitor your API or website, create an account on OpenStatus.