Nexus is a lightweight, high-performance reverse proxy and load balancer built in Go, designed to deliver fast, reliable, and scalable traffic management solutions.
graph LR
Client --> NexusProxy(Nexus Proxy)
NexusProxy --> Router(Router)
Router --> Service1(Service1)
Router --> Service2(Service2)
Router --> ServiceN(ServiceN)
Service1 --> LoadBalancer1(Load Balancer1)
Service2 --> LoadBalancer2(Load Balancer2)
ServiceN --> LoadBalancerN(Load BalancerN)
LoadBalancer1 --> BackendServer1(Backend Server 1)
LoadBalancer1 --> BackendServer2(Backend Server 2)
LoadBalancer1 --> BackendServer3(Backend Server 3)
LoadBalancer2 --> BackendServer4(Backend Server 4)
LoadBalancer2 --> BackendServer5(Backend Server 5)
LoadBalancer2 --> BackendServer6(Backend Server 6)
LoadBalancerN --> BackendServer7(Backend Server 7)
LoadBalancerN --> BackendServer8(Backend Server 8)
LoadBalancerN --> BackendServer9(Backend Server 9)
NexusProxy --> HealthChecker(Health Checker)
HealthChecker --> BackendServer1
HealthChecker --> BackendServer2
HealthChecker --> BackendServer3
HealthChecker --> BackendServer4
HealthChecker --> BackendServer5
HealthChecker --> BackendServer6
HealthChecker --> BackendServer7
HealthChecker --> BackendServer8
HealthChecker --> BackendServer9
Configuration(config.yaml) --> NexusProxy
style NexusProxy fill:#f9f,stroke:#333,stroke-width:2px
style Router fill:#ff9966,stroke:#333,stroke-width:2px
style HealthChecker fill:#cfc,stroke:#333,stroke-width:2px
style BackendServer1 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer2 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer2 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer3 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer4 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer5 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer6 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer7 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer8 fill:#eee,stroke:#333,stroke-width:2px
style BackendServer9 fill:#eee,stroke:#333,stroke-width:2px
style Configuration fill:#ffc,stroke:#333,stroke-width:2px
style Service1 fill:#ffc,stroke:#333,stroke-width:2px
style Service2 fill:#ffc,stroke:#333,stroke-width:2px
style ServiceN fill:#ffc,stroke:#333,stroke-width:2px
classDef component fill:#f9f,stroke:#333,stroke-width:2px
class NexusProxy,Router,HealthChecker,Service1,Service2,ServiceN,LoadBalancer1,LoadBalancer2,LoadBalancerN,BackendServer1,BackendServer2,BackendServer3,BackendServer4,BackendServer5,BackendServer6,BackendServer7,BackendServer8,BackendServer9,BackendServerN component
- High Performance: Leverages Go's efficient concurrency model for exceptional throughput and low latency
- Intelligent Load Balancing: Multiple algorithms including round-robin, weighted round-robin, and least connections
- Active Health Monitoring: Automatic detection and removal of unhealthy backends
- Dynamic Configuration: Hot-reload YAML configuration without downtime
- Extensible Architecture: Modular design for custom middleware (authentication, rate limiting, etc.)
- Observability: Built-in metrics and OpenTelemetry integration
- Cloud-Native: Container-ready with Docker support
- Go 1.22.5+
git clone https://github.com/rust17/nexus.git
cd nexus
go build -o nexus cmd/main.go
- Copy the sample config:
cp configs/config.yaml ./config.yaml
- Edit
config.yaml
:
listen_address: ":8080"
services:
- name: "api-service"
balancer_type: "round_robin"
servers:
- address: "http://localhost:8081"
- address: "http://localhost:8082"
routes:
- name: user_route
match:
path: "/api/v1/users/*"
service: api-service
-
Start Nexus:
./nexus -config config.yaml
Or, if you placed the configuration file in the default location (
./config.yaml
or/etc/nexus/config.yaml
), you can run it directly:./nexus
-
Access Proxy Service:
Now you can access the Nexus reverse proxy service through the configured listening address (e.g.,
http://localhost:8080
). Requests will be load balanced to the backend servers.
The config.yaml
file is used to configure the behavior of the Nexus reverse proxy and load balancer. Here's a detailed explanation of the configuration file:
# Proxy server listening address
listen_addr: ":8080"
# Log level (debug, info, warn, error, fatal)
log_level: "info"
# Service configuration
services:
- name: "api-service" # Service name (required)
balancer_type: "weighted_round_robin" # Load balancer algorithm (round_robin, least_connections, weighted_round_robin)
servers: # List of backend servers
- address: "http://localhost:8081" # Server address (required)
weight: 3 # Server weight for weighted algorithms (optional, default: 1)
- address: "http://localhost:8082"
weight: 2
- address: "http://localhost:8083"
weight: 1
# Health check configuration
health_check:
enabled: true # Enable health check
interval: 10s # Check interval
timeout: 2s # Timeout duration
path: "/health" # Health check path (HTTP)
# Telemetry configuration
telemetry:
opentelemetry:
enabled: false # Enable OpenTelemetry
endpoint: "otel-collector:4317" # OpenTelemetry collector endpoint
service_name: "nexus-lb" # Service name for telemetry
metrics:
interval: "60s" # Metrics collection interval
# Route configuration
routes:
- name: user_route # Route name
match: # Route matching criteria
path: "/api/v1/users/*" # Path pattern with wildcard support
headers: # Header matching (optional)
X-Service-Group: "v2"
method: "GET" # HTTP method matching (optional)
host: "api.example.com" # Host header matching (optional)
service: api-service # Target service name
nexus/
├── .gitignore
├── api/ # API definitions and interfaces
├── cmd/ # contains executable files for the project
│ └── main.go # main program entry
├── configs/
│ └── config.yaml # configuration file for configuring the proxy server
├── internal/ # internal packages not meant for external use
│ ├── balancer/
│ │ ├── balancer.go # load balancer interface
│ │ ├── weighted_round_robin.go # weighted round-robin load balancer implementation
│ │ ├── round_robin.go # round-robin load balancer implementation
│ │ └── least_connections.go # least connections load balancer implementation
│ ├── config/ # configuration management
│ ├── health/ # health check implementation
│ ├── logger/ # logger implementation
│ ├── proxy/ # proxy implementation
│ └── router/ # request routing implementation
├── pb/ # contains protobuf definitions and generated code
│ ├── nexus.pb.go
│ └── nexus_grpc.pb.go
├── scripts/ # utility scripts for development, testing, and deployment
├── test/ # test files
│ ├── benchmark_test.go # performance benchmark tests
│ ├── integration_test.go # integration tests
│ └── stress_test.go # stress tests
├── web/ # web interface assets and code
├── go.mod # Go module definition
└── go.sum # Go module dependency checksums
Configure the config.yaml
file as follows:
listen_addr: ":8080"
services:
- name: "api-service"
servers:
- address: "http://192.168.1.100:8081"
routes:
- name: "api-route"
match:
path: "*"
host: "api.example.com"
service: "api-service"
After starting Nexus, all HTTP requests sent to http://localhost:8080
will be proxied to the backend servers 192.168.1.100:8081
.
Configure the config.yaml
file as follows:
listen_addr: ":8080"
services:
- name: "api-v1"
balancer_type: "round_robin"
servers:
- address: "http://api-v1-1:8081"
- address: "http://api-v1-2:8081"
- name: "api-v2"
balancer_type: "round_robin"
servers:
- address: "http://api-v2-1:8082"
- address: "http://api-v2-2:8082"
routes:
- name: "api-canary"
match:
path: "/api/*"
headers:
X-Debug: "true"
split:
- service: "api-v1"
weight: 80
- service: "api-v2"
weight: 20
- name: "api-stable"
match:
path: "/api/*"
service: "api-v1"
This configuration demonstrates a canary deployment setup where:
- Requests with the header
X-Debug: true
will be split between api-v1 (approximately 80%) and api-v2 (approximately 20%) - All other API requests will be routed to api-v1 only
Configure the config.yaml
file as follows:
listen_addr: ":8080"
services:
- name: "user-service"
balancer_type: "least_connections"
servers:
- address: "http://user-service-1:8081"
- address: "http://user-service-2:8081"
- name: "order-service"
balancer_type: "round_robin"
servers:
- address: "http://order-service-1:8082"
- address: "http://order-service-2:8082"
- name: "product-service"
balancer_type: "weighted_round_robin"
servers:
- address: "http://product-service-1:8083"
weight: 3
- address: "http://product-service-2:8083"
weight: 1
health_check:
enabled: true
interval: 5s
timeout: 2s
path: "/health"
routes:
- name: "user-route"
match:
path: "/api/users/*"
service: "user-service"
- name: "order-route"
match:
path: "/api/orders/*"
service: "order-service"
- name: "product-route"
match:
path: "/api/products/*"
service: "product-service"
After starting Nexus, requests will be routed to different microservices based on the path:
- All requests starting with
/api/users/
will be routed touser-service
using the least connections load balancing algorithm - All requests starting with
/api/orders/
will be routed toorder-service
using the round-robin load balancing algorithm - All requests starting with
/api/products/
will be routed toproduct-service
using the weighted round-robin load balancing algorithm
This configuration makes Nexus an effective API gateway, distributing requests to the appropriate microservices while providing health checking and load balancing capabilities.
Contributions of any form are welcome! If you'd like to contribute to this project, please follow these steps:
- Fork this repository
- Create your Feature branch (
git checkout -b feature/your-feature
) - Commit your changes (
git commit -am 'Add some feature'
) - Push to the Feature branch (
git push origin feature/your-feature
) - Create a new Pull Request
Please ensure your code style is consistent with the existing project code and add appropriate unit tests.
MIT