Chirag's Blog
January 26, 2025
Serverless isn't just trendy—it's rewriting how software scales.
Netflix streams billions of hours without servers. Coca-Cola automates workflows without infrastructure. Figma and T-Mobile ditch downtime. What do they know that you don't?
The secret? Event-driven serverless code. It's the backbone of apps that scale instantly, cut costs, and survive traffic spikes. No servers. No guesswork. Just code that reacts.
This isn't hype—it's a blueprint. Ready to build smarter? Let's break down how event-driven serverless turns scalability from a challenge into a reflex.
Spoiler alert: servers are still there.
There are multiple definitions for this term online, often filled with complex jargon. The best way I like to define it is:
A "fashion" of deploying your code where YOU don't have to think about the servers running your code.
Let's take an example:
Take Bob. He built mytrashcode.com but panicked at "server setup." He's a developer, not a sysadmin. Instead, he uploaded his code to a cloud provider. They handled security, scaling, and traffic—his site went live. No late-night server meltdowns. No panic during traffic surges. Done.
Managing your own servers usually takes one of two paths. You either run physical hardware—like turning an old laptop into a DIY server—or rent a Virtual Private Server (VPS) from providers like DigitalOcean Droplets, Azure VMs, or AWS Lightsail. These fall under IaaS (Infrastructure as a Service), where the cloud company provides the bare-metal infrastructure, but the rest—updates, scaling, security—is entirely up to you.
Does this mean self-managing servers is impossible?
Not at all. Plenty of teams still do it. But managing your own servers comes with a lot of... challenges, including:
Serverless code doesn't need to be monolithic, i.e., all code doesn't need to be in the same place. It can be a collection of bite-sized, event-triggered functions.
A Function is nothing but a set of code that performs a specific task. When writing your entire code serverless, you'll find that you can divide your code into various functions, each handling a specific part of your application. Let's understand this more deeply with an example:
When Bob first logs into mytrashcode.com as a new user, the system triggers a "send welcome email" function before redirecting him. Subsequent logins bypass this function entirely, routing him straight to the dashboard. This separation serves a critical purpose—while 99% of users interact solely with the dashboard, isolating secondary functions (like email triggers) enables independent scaling.
Though trivial in this example, the cost implications compound dramatically at scale. Each decoupled function operates on its own resource allocation curve—high-frequency features like dashboard access demand consistent infrastructure, while one-time actions (welcome emails) can scale down during inactive periods. This modular approach prevents overprovisioning for rarely triggered events, even before considering complex systems with hundreds of interdependent functions.
Okay, so just a quick recap—we now know:
So, where do you deploy this architecture? Leading platforms like AWS Lambda, Azure Functions, and Google Cloud Functions support it, but we'll focus on Appwrite Functions.
Appwrite, an open-source Backend-as-a-Service (BaaS), bundles authentication, databases, storage, and serverless functions into a single toolkit. This tight integration streamlines deployment—instead of managing fragmented cloud services, Appwrite centralizes backend logic, letting you deploy event-driven functions with minimal overhead. For developers prioritizing simplicity without sacrificing scalability, this unified approach reduces operational friction significantly.
So, let's deploy our function!
Before writing code, set up your backend on Appwrite:
Now, let's simulate a server-side project using the node-appwrite
package:
1mkdir my-project2cd my-project
1npm init -y2appwrite init
1npm install dotenv node-appwrite
1appwrite init function
For the runtime, I selected Node 20, but you can choose any runtime.
src/main.js
:1import dotenv from "dotenv";2import { Account, Client, Users } from "node-appwrite";34dotenv.config();56const client = new Client();7client.setEndpoint("https://cloud.appwrite.io/v1");8client.setProject(process.env.PROJECT_ID);910const users = new Users(client);11const account = new Account(client);1213const main = async () => {14 await account.create("test-user", "test@test.com", "test@123", "test");15 const session = await account.createEmailPasswordSession(16 "test@test.com",17 "test@123",18 );19 console.log(session);20};2122main();
Add a start
script in package.json
to run node src/main.js
.
Create a .env
file with the required environment variables.
This function simulates a new user creation and login, logging the session details.
Note: Replace the email IDs with actual emails to receive the email.
Now, let's set up the function logic. Navigate to functions/your-function
where your function resides.
For this demo, we'll use Resend to send emails:
resend
package:1npm install resend
src/main.js
with this code:1import { Resend } from "resend";23// https://appwrite.io/docs/advanced/platform/events4export default async ({ res, req, log }) => {5 const resend = new Resend(process.env.RESEND_API_KEY);67 await resend.emails.send({8 from: "hello@yourdomain.com",9 to: req.body.email,10 subject: "Hello!",11 text: "Hi, its nice to meet you!",12 });1314 log("Email sent successfully");1516 return res.json({17 success: true,18 message: "Email sent successfully",19 });20};
You need to set up an account on Resend to get the API Key. Resend also requires you to connect to your own domain to send emails. You can read more about it on the Resend docs.
Now, let's push the created function to the console using:
1appwrite push functions
The final step is to set up the event that connects the two pieces of code together using the users.*.create
event:
users.*.create
.And... voila! Your program is done. If everything is set up correctly, running your main script should send the newly created user an invitation email. Try it using:
1npm run start
In conclusion, serverless architecture is more than just a passing trend—it's a transformative approach to building and scaling modern applications.
Platforms like Appwrite further simplify the process, offering a unified backend solution that integrates seamlessly with serverless functions. Whether you're a solo developer like Bob or part of a larger team, adopting serverless can turn scalability from a daunting challenge into an effortless reflex.
Thanks for reading!
You can also connect with me here: https://www.chiragaggarwal.tech/