From a97c018d9f4f82fa3e59172ab80db98fb62cea8a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Necati=20=C3=96zmen?= Date: Thu, 26 Sep 2024 11:20:53 +0300 Subject: [PATCH] Update grp post (#6367) --- ...-vs-rest.md => 2024-09-24-grpc-vs-rest.md} | 405 +++++++++++++++++- 1 file changed, 402 insertions(+), 3 deletions(-) rename documentation/blog/{2024-01-22-grpc-vs-rest.md => 2024-09-24-grpc-vs-rest.md} (66%) diff --git a/documentation/blog/2024-01-22-grpc-vs-rest.md b/documentation/blog/2024-09-24-grpc-vs-rest.md similarity index 66% rename from documentation/blog/2024-01-22-grpc-vs-rest.md rename to documentation/blog/2024-09-24-grpc-vs-rest.md index eb418c267b77..57a7ad1ed9e3 100644 --- a/documentation/blog/2024-01-22-grpc-vs-rest.md +++ b/documentation/blog/2024-09-24-grpc-vs-rest.md @@ -8,7 +8,7 @@ image: https://refine.ams3.cdn.digitaloceanspaces.com/blog/2023-06-23-grpc-vs-re hide_table_of_contents: false --- -**_This article was last updated on January 22, 2024 to add more comparison and usecase information for gRPC vs REST to provide a more clear insight_** +**_This article was last updated on September 24, 2024 to include migration strategies, security considerations, and real-world examples to provide a more comprehensive comparison between gRPC and REST._** ## Introduction @@ -26,11 +26,13 @@ Steps we'll cover: - [What is gRPC?](#what-is-grpc) - [How gRPC APIs work](#how-grpc-apis-work) - [How developers can use gRPC and REST](#how-developers-can-use-grpc-and-rest) +- [Security Considerations](#security-considerations) + - [Authentication and Authorization](#authentication-and-authorization) + - [Data Encryption](#data-encryption) - [Similarities and differences between gRPC and REST](#similarities-and-differences-between-grpc-and-rest) - - [Similarities between gRPC and REST](#similarities-between-grpc-and-rest) - - [Differences between gRPC and REST](#differences-between-grpc-and-rest) - [Comparison summary](#comparison-summary) - [When to use REST and when to use gRPC](#when-to-use-rest-and-when-to-use-grpc) +- [Bonus: Migration Strategies](#bonus-migration-strategies) ## Understanding APIs @@ -276,6 +278,161 @@ Below illustration will help you understand how both these technologies work. - **Simple CRUD operations:** REST is well-suited for standard web APIs that perform CRUD (Create, Read, Update, Delete) operations. For example, a web service for a book inventory system can use RESTful APIs for adding new books, retrieving book details, updating prices, or deleting old records. - **Building public APIs for broader reach:** When developing public APIs intended for a wide range of clients, including third-party developers, REST is often preferred due to its simplicity and widespread familiarity. An example is a social media platform providing a RESTful API for fetching user profiles or posting messages, allowing easy integration for a variety of external services and applications. +## Security Considerations + +When choosing between gRPC and REST, it's important to understand how each handles security. Both offer ways to manage authentication, authorization, and data encryption, but they do it differently. + +### Authentication and Authorization + +#### gRPC + +- **SSL/TLS Support:** gRPC uses SSL/TLS by default to secure communication between the client and server. This means that data is encrypted and protected from eavesdropping. + +- **Mutual Authentication:** gRPC supports mutual SSL/TLS authentication, where both the client and server present certificates to verify each other. + +- **Token-Based Authentication with Metadata:** You can pass tokens or credentials using metadata in gRPC calls. + + _Example in Python:_ + + ```python + # Client-side code + import grpc + from your_proto_pb2_grpc import YourServiceStub + from your_proto_pb2 import YourRequest + + def run(): + with grpc.secure_channel('localhost:50051', grpc.ssl_channel_credentials()) as channel: + stub = YourServiceStub(channel) + metadata = [('authorization', 'Bearer your_token')] + response = stub.YourMethod(YourRequest(), metadata=metadata) + print(response) + ``` + +- **Custom Authentication:** You can implement custom authentication mechanisms using interceptors or middleware. + +#### REST + +- **OAuth 2.0 Integration:** REST APIs often use OAuth 2.0 for authentication and authorization. This allows secure, token-based access. + + _Example in Node.js with Express:_ + + ```javascript + // Server-side code + const express = require("express"); + const app = express(); + const jwt = require("jsonwebtoken"); + + app.use(express.json()); + + function authenticateToken(req, res, next) { + const authHeader = req.headers["authorization"]; + const token = authHeader && authHeader.split(" ")[1]; + + if (token == null) return res.sendStatus(401); + + jwt.verify(token, "your_secret_key", (err, user) => { + if (err) return res.sendStatus(403); + req.user = user; + next(); + }); + } + + app.get("/protected", authenticateToken, (req, res) => { + res.json({ message: "This is protected data." }); + }); + + app.listen(3000, () => console.log("Server running on port 3000")); + ``` + +- **API Keys and Tokens:** REST can use API keys, JSON Web Tokens (JWT), and other methods to control access. + +- **HTTP Basic and Digest Authentication:** These methods are simpler but should be used over HTTPS to be secure. + +### Data Encryption + +#### gRPC + +- **Encrypted Communication:** gRPC ensures all communication is encrypted using SSL/TLS. + +- **Protocol Buffers Serialization:** Data is serialized using Protocol Buffers, which is efficient and not human-readable. However, this is not encryption, so sensitive data should still be sent over secure channels. + + _Example of a `.proto` file:_ + + ```protobuf + syntax = "proto3"; + + service YourService { + rpc YourMethod (YourRequest) returns (YourResponse); + } + + message YourRequest { + string data = 1; + } + + message YourResponse { + string result = 1; + } + ``` + +- **Server Implementation in Go:** + + ```go + // Server-side code in Go + import ( + "google.golang.org/grpc" + "google.golang.org/grpc/credentials" + "log" + "net" + ) + + func main() { + creds, err := credentials.NewServerTLSFromFile("server.crt", "server.key") + if err != nil { + log.Fatalf("Failed to load certificates: %v", err) + } + + server := grpc.NewServer(grpc.Creds(creds)) + // Register your services here + + lis, err := net.Listen("tcp", ":50051") + if err != nil { + log.Fatalf("Failed to listen: %v", err) + } + + if err := server.Serve(lis); err != nil { + log.Fatalf("Failed to serve: %v", err) + } + } + ``` + +#### REST + +- **HTTPS Usage:** REST APIs typically use HTTPS to encrypt data between the client and server. + + ```javascript + // Server-side code + const express = require("express"); + const https = require("https"); + const fs = require("fs"); + + const app = express(); + + // Define your routes here + + const options = { + key: fs.readFileSync("server.key"), + cert: fs.readFileSync("server.cert"), + }; + + https.createServer(options, app).listen(3000, () => { + console.log("HTTPS server running on port 3000"); + }); + ``` + +- **Data Formats:** REST often uses JSON or XML, which are human-readable. While HTTPS secures the data in transit, sensitive data should be handled carefully. + +- **Additional Encryption:** For extra security, you can encrypt sensitive data within the payload before sending it. + ## Similarities and differences between gRPC and REST ### Similarities between gRPC and REST @@ -311,6 +468,248 @@ The architectural style of REST API differs from that of gRPC API due to their d Use REST when you are developing web application and a more standardized approach with wide compatibility. REST is ideal for CRUD operations and when simplicity and caching are priorities. Go for gRPC if you are building application with high-performance requirements, efficient communication in microservices, and for systems where language interoperability and network efficiency are critical, such as in real-time communication. +## Bonus: Migration Strategies + +Transitioning from REST to gRPC can offer performance improvements and enhanced functionality. Here's a guide to help developers migrate their existing REST APIs to gRPC smoothly. + +### Step-by-Step Guide to Migrating from REST to gRPC + +1. **Assess Your Current API Structure** + + - **Inventory Endpoints**: List all your REST endpoints and their functionalities. + - **Identify Core Services**: Determine which services will benefit most from gRPC's performance improvements. + +2. **Define Protobuf Messages and Services** + + - **Create `.proto` Files**: Define your data structures and services using Protocol Buffers. + + ```protobuf + syntax = "proto3"; + + package example; + + service UserService { + rpc GetUser (GetUserRequest) returns (GetUserResponse); + } + + message GetUserRequest { + int32 user_id = 1; + } + + message GetUserResponse { + int32 user_id = 1; + string name = 2; + string email = 3; + } + ``` + +3. **Generate gRPC Code from Protobuf Definitions** + + - Use the `protoc` compiler to generate server and client code in your chosen language. + + ```bash + protoc --go_out=plugins=grpc:. *.proto + ``` + +4. **Implement the gRPC Services** + + - **Server-Side Implementation**: Write the server logic for your gRPC services. + + _Example in Go:_ + + ```go + // Server-side code + import ( + "context" + "net" + + "google.golang.org/grpc" + pb "path/to/your/protobuf/package" + ) + + type server struct { + pb.UnimplementedUserServiceServer + } + + func (s *server) GetUser(ctx context.Context, in *pb.GetUserRequest) (*pb.GetUserResponse, error) { + // Business logic to retrieve user + return &pb.GetUserResponse{ + UserId: in.UserId, + Name: "John Doe", + Email: "john.doe@example.com", + }, nil + } + + func main() { + lis, _ := net.Listen("tcp", ":50051") + s := grpc.NewServer() + pb.RegisterUserServiceServer(s, &server{}) + s.Serve(lis) + } + ``` + +5. **Update Clients to Use gRPC** + + - **Client-Side Implementation**: Rewrite client applications to communicate with the gRPC server. + + _Example in Python:_ + + ```python + # Client-side code + import grpc + from example_pb2 import GetUserRequest + from example_pb2_grpc import UserServiceStub + + def run(): + with grpc.insecure_channel('localhost:50051') as channel: + stub = UserServiceStub(channel) + response = stub.GetUser(GetUserRequest(user_id=123)) + print("User Details:", response) + + if __name__ == '__main__': + run() + ``` + +### Maintain REST Endpoints During Transition + +In a smooth transition from REST to gRPC, one would maintain the existing endpoint for REST and introduce the gRPC services in parallel. An API Gateway would route clients to either the REST service or the gRPC service, depending on the needs of the client. For backward compatibility, onboarding the clients one at a time with minimum disruption would be done, and API versioning handles the different stages of that process. + +Unit tests and integration tests should be provided for any new gRPC services that are written. Compare their performance to REST endpoints for assurance of functionality. Deployment: With Docker, containerize your services or use similar tooling to gain a consistent approach. As changes are integrated into the CI/CD pipeline, automation will test and deploy. + +### Potential Pitfalls and How to Avoid Them + +- **Incomplete Protobuf Definitions**: Ensure all data models are accurately defined in your `.proto` files to prevent serialization issues. +- **Network Configuration**: gRPC uses HTTP/2, so make sure your infrastructure supports it. +- **Security Concerns**: Implement SSL/TLS for encrypted communication, similar to how you secure REST APIs. + +### Case Study: Migrating a RESTful User Service to gRPC + +Let's consider a simple user service that retrieves user information. + +**Original REST Endpoint (Node.js with Express):** + +```javascript +// Server-side code +const express = require("express"); +const app = express(); + +app.get("/user/:id", (req, res) => { + const userId = req.params.id; + // Logic to get user data + res.json({ + user_id: userId, + name: "John Doe", + email: "john.doe@example.com", + }); +}); + +app.listen(3000, () => console.log("REST API running on port 3000")); +``` + +**Migrated gRPC Service (Node.js):** + +1. **Define the `.proto` File** + + ```protobuf + syntax = "proto3"; + + package example; + + service UserService { + rpc GetUser (GetUserRequest) returns (GetUserResponse); + } + + message GetUserRequest { + int32 user_id = 1; + } + + message GetUserResponse { + int32 user_id = 1; + string name = 2; + string email = 3; + } + ``` + +2. **Implement the Server** + + ```javascript + // Server-side code + const grpc = require("@grpc/grpc-js"); + const protoLoader = require("@grpc/proto-loader"); + + const packageDefinition = protoLoader.loadSync("example.proto"); + const proto = grpc.loadPackageDefinition(packageDefinition).example; + + function getUser(call, callback) { + const userId = call.request.user_id; + // Logic to get user data + callback(null, { + user_id: userId, + name: "John Doe", + email: "john.doe@example.com", + }); + } + + const server = new grpc.Server(); + server.addService(proto.UserService.service, { GetUser: getUser }); + server.bindAsync( + "0.0.0.0:50051", + grpc.ServerCredentials.createInsecure(), + () => { + server.start(); + console.log("gRPC server running on port 50051"); + }, + ); + ``` + +3. **Implement the Client** + + ```javascript + // Client-side code + const grpc = require("@grpc/grpc-js"); + const protoLoader = require("@grpc/proto-loader"); + + const packageDefinition = protoLoader.loadSync("example.proto"); + const proto = grpc.loadPackageDefinition(packageDefinition).example; + + const client = new proto.UserService( + "localhost:50051", + grpc.credentials.createInsecure(), + ); + + client.GetUser({ user_id: 123 }, (error, response) => { + if (error) { + console.error("Error:", error); + } else { + console.log("User Details:", response); + } + }); + ``` + +### Tools to Assist Migration + +- **gRPC-Gateway**: Allows you to expose gRPC services as REST endpoints, helping in gradual migration. + + _Example usage in Go:_ + + ```go + // Install gRPC-Gateway + go get -u github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-grpc-gateway + + // Generate reverse-proxy + protoc -I . example.proto --grpc-gateway_out=. + + // Run the gateway server + ``` + +- **API Gateways**: Tools like Envoy or Kong can route traffic between REST and gRPC services. + +### Benefits Achieved After Migration + +- **Improved Performance**: Lower latency and higher throughput with gRPC. +- **Efficient Data Transfer**: Smaller payload sizes with Protocol Buffers. +- **Bi-Directional Streaming**: Enhanced capabilities not available in REST. + ## Conclusion Finally, you've reached the end of the article, where you learned about the architectural style of REST APIs and gRPC APIs, including how they work and the differences between their distinctive features.