| Crates.io | armature-framework |
| lib.rs | armature-framework |
| version | 0.1.1 |
| created_at | 2025-12-27 04:35:50.114342+00 |
| updated_at | 2025-12-27 14:36:17.032802+00 |
| description | A modern, type-safe HTTP framework for Rust inspired by Angular and NestJS. Features dependency injection, decorators, middleware, authentication (JWT/OAuth2/SAML), validation, OpenAPI/Swagger, caching, job queues, and observability. |
| homepage | https://pegasusheavy.github.io/armature |
| repository | https://github.com/pegasusheavy/armature |
| max_upload_size | |
| id | 2006684 |
| size | 2,323,800 |
A modern, type-safe HTTP framework for Rust heavily inspired by Angular and NestJS.
Armature brings the elegant decorator syntax and powerful dependency injection from the TypeScript/JavaScript ecosystem to Rust, combining the developer experience of NestJS with Rust's performance and safety guarantees.
Install the Armature CLI for the best development experience:
# Install the CLI
cargo install armature-cli
# Create a new project
armature new my-api
# Navigate to your project
cd my-api
# Start the development server with hot reloading
armature dev
# Generate a controller
armature generate controller users
# Generate a service
armature generate service users
# Generate a complete resource (controller + service + module)
armature generate resource products --crud
# Generate middleware, guards, and more
armature generate middleware auth
armature generate guard admin
use armature_framework::prelude::*;
use serde::{Deserialize, Serialize};
// Define your domain model
#[derive(Serialize, Deserialize)]
struct User {
id: u32,
name: String,
}
// Create an injectable service
#[injectable]
#[derive(Default, Clone)]
struct DatabaseService;
#[injectable]
#[derive(Default, Clone)]
struct UserService {
database: DatabaseService, // Automatically injected!
}
impl UserService {
fn get_users(&self) -> Vec<User> {
vec![User { id: 1, name: "Alice".to_string() }]
}
}
// Create a controller with automatic DI
#[controller("/users")]
#[derive(Default, Clone)]
struct UserController {
user_service: UserService, // Automatically injected!
}
impl UserController {
// Use injected service in methods
fn get_all(&self) -> Result<Json<Vec<User>>, Error> {
Ok(Json(self.user_service.get_users()))
}
fn get_one(&self, id: u32) -> Result<Json<User>, Error> {
// Use self.user_service throughout the controller
Ok(Json(User { id, name: "Alice".to_string() }))
}
}
// Define your application module
#[module(
providers: [DatabaseService, UserService],
controllers: [UserController]
)]
#[derive(Default)]
struct AppModule;
// Bootstrap your application - DI happens automatically!
#[tokio::main]
async fn main() {
let app = Application::create::<AppModule>();
app.listen(3000).await.unwrap();
}
The framework is organized into three main crates:
armature-coreCore runtime functionality including:
Provider, Controller, Module, RequestHandler)armature-macroProcedural macros for decorator syntax:
#[injectable] - Mark structs as injectable services#[controller("/path")] - Define controllers with base paths#[get], #[post], #[put], #[delete], #[patch] - HTTP route decorators#[module(...)] - Organize components into modules#[derive(Body)], #[derive(Param)], #[derive(Query)] - Request parameter extractionarmature-frameworkMain library that re-exports everything from core and macros. Add to your Cargo.toml:
[dependencies]
armature-framework = "0.1"
Or with specific features:
[dependencies]
armature-framework = { version = "0.1", features = ["auth", "cache", "validation"] }
#[injectable]Marks a struct as injectable, allowing it to be registered in the DI container:
#[injectable]
struct DatabaseService {
connection: DbConnection,
}
#[controller("/path")]Marks a struct as a controller with a base path:
#[controller("/api/users")]
struct UserController {
user_service: UserService,
}
HTTP method decorators for defining routes:
#[get("/")] // GET /api/users/
#[get("/:id")] // GET /api/users/:id
#[post("/")] // POST /api/users/
#[put("/:id")] // PUT /api/users/:id
#[delete("/:id")] // DELETE /api/users/:id
#[patch("/:id")] // PATCH /api/users/:id
#[module(...)]Defines a module with providers, controllers, and imports:
#[module(
providers: [UserService, DatabaseService],
controllers: [UserController],
imports: [CommonModule],
exports: [UserService]
)]
struct UserModule;
The DI container uses Rust's type system for service resolution:
// Register a service
container.register(MyService::default());
// Resolve a service
let service = container.resolve::<MyService>()?;
Services are singletons by default and shared across the application.
Armature provides NestJS-inspired lifecycle hooks for managing service initialization and cleanup:
use armature_framework::prelude::*;
use armature_framework::lifecycle::{OnModuleInit, OnModuleDestroy, LifecycleResult};
use async_trait::async_trait;
#[injectable]
struct DatabaseService {
connected: Arc<RwLock<bool>>,
}
#[async_trait]
impl OnModuleInit for DatabaseService {
async fn on_module_init(&self) -> LifecycleResult {
println!("Connecting to database...");
*self.connected.write().await = true;
Ok(())
}
}
#[async_trait]
impl OnModuleDestroy for DatabaseService {
async fn on_module_destroy(&self) -> LifecycleResult {
println!("Closing database connection...");
*self.connected.write().await = false;
Ok(())
}
}
Available hooks:
OnModuleInit - Called after module initializationOnModuleDestroy - Called before module destructionOnApplicationBootstrap - Called after full application bootstrapOnApplicationShutdown - Called during graceful shutdownBeforeApplicationShutdown - Called before shutdown hooksSee the Lifecycle Hooks Guide for complete documentation.
The router supports:
/users/users/:id/users?page=1&limit=10Path parameters are extracted automatically:
#[get("/:id")]
async fn get_user(req: HttpRequest) -> Result<Json<User>, Error> {
let id = req.param("id").unwrap();
// ...
}
Comprehensive HTTP status code support with type-safe error handling:
use armature_framework::{Error, HttpStatus};
// Type-safe status codes
let status = HttpStatus::NotFound;
assert_eq!(status.code(), 404);
assert_eq!(status.reason(), "Not Found");
// Structured errors for all 4xx/5xx codes
return Err(Error::NotFound("User not found".to_string()));
return Err(Error::TooManyRequests("Rate limit exceeded".to_string()));
return Err(Error::ServiceUnavailable("Database down".to_string()));
// Error helpers
let error = Error::Unauthorized("Invalid token".to_string());
assert_eq!(error.status_code(), 401);
assert!(error.is_client_error());
See the HTTP Status & Errors Guide for complete documentation.
Protect and transform your routes with Guards and Interceptors:
use armature_framework::{Guard, AuthenticationGuard, GuardContext};
// Apply authentication guard
let guard = AuthenticationGuard;
let context = GuardContext::new(request);
match guard.can_activate(&context).await {
Ok(true) => { /* proceed */ },
_ => { /* unauthorized */ }
}
// Built-in guards: AuthenticationGuard, RolesGuard, ApiKeyGuard
// Built-in interceptors: LoggingInterceptor, TransformInterceptor, CacheInterceptor
See the Guards & Interceptors Guide for detailed documentation.
The HttpRequest type provides:
json::<T>() - Parse body as JSONparam(name) - Get path parameterquery(name) - Get query parameterThe HttpResponse type provides:
with_json()ok(), created(), not_found(), etc.Use the Json<T> wrapper for automatic serialization:
#[get("/users")]
async fn get_users() -> Result<Json<Vec<User>>, Error> {
Ok(Json(vec![...]))
}
The framework provides a comprehensive Error type:
pub enum Error {
Http(String),
RouteNotFound(String),
MethodNotAllowed(String),
DependencyInjection(String),
ProviderNotFound(String),
Serialization(String),
Deserialization(String),
Validation(String),
Internal(String),
Io(std::io::Error),
}
Errors are automatically converted to HTTP responses with appropriate status codes.
Run the test suite:
cargo test
Build the library:
cargo build --lib
Run example applications:
# Full-featured example with CRUD operations
cargo run --example full_example
# Simple routing example
cargo run --example simple
# REST API example
cargo run --example rest_api
Test the endpoints (when running full_example):
# Health check
curl http://localhost:3000/health
# Get all users
curl http://localhost:3000/users
# Get user by ID
curl http://localhost:3000/users/1
# Create a user
curl -X POST http://localhost:3000/users \
-H 'Content-Type: application/json' \
-d '{"name":"Charlie","email":"charlie@example.com"}'
armature/
βββ armature-core/ # Core runtime library
β βββ src/
β βββ traits.rs # Core traits
β βββ container.rs # DI container
β βββ routing.rs # Router implementation
β βββ http.rs # HTTP types
β βββ error.rs # Error types
β βββ application.rs # Application bootstrap
βββ armature-macro/ # Procedural macros
β βββ src/
β βββ injectable.rs # #[injectable] macro
β βββ controller.rs # #[controller] macro
β βββ routes.rs # Route macros
β βββ module.rs # #[module] macro
β βββ params.rs # Parameter extraction
βββ armature-compression/ # HTTP response compression
β βββ src/
β βββ algorithm.rs # Compression algorithms (gzip, brotli, zstd)
β βββ config.rs # Configuration builder
β βββ middleware.rs # Compression middleware
βββ src/
β βββ lib.rs # Main library (re-exports)
βββ examples/ # Example applications
β βββ full_example.rs # Complete CRUD example
β βββ simple.rs # Basic routing
β βββ rest_api.rs # REST API demo
βββ Cargo.toml # Workspace manifest
The Armature CLI provides powerful code generation and development tools:
cargo install armature-cli
| Command | Description |
|---|---|
armature new <name> |
Create a new project from templates |
armature generate controller <name> |
Generate a controller |
armature generate service <name> |
Generate a service/provider |
armature generate module <name> |
Generate a module |
armature generate middleware <name> |
Generate middleware |
armature generate guard <name> |
Generate a guard |
armature generate resource <name> |
Generate controller + service + module |
armature dev |
Start development server with file watching |
armature build |
Build for production |
armature info |
Display project information |
# Minimal API (default)
armature new my-api
# Full-featured API with auth, validation, Docker
armature new my-api --template full
# Microservice with queue worker
armature new my-api --template microservice
The armature dev command starts a development server with automatic rebuilding:
# Start with default settings (port 3000)
armature dev
# Custom port
armature dev --port 8080
# Uses cargo-watch if installed for better performance
#[use_middleware] decorator syntaxThis project is heavily inspired by:
We're grateful to these projects and their communities for showing what great developer experience looks like. Armature aims to bring these same patterns to the Rust ecosystem with the added benefits of memory safety and native performance.
MIT
π Live Documentation Website: https://pegasusheavy.github.io/armature/
Comprehensive documentation is available in the docs/ directory:
Getting Started:
Core Features:
βοΈ Cloud Providers:
Advanced:
And more guides covering testing, security, and deployment!
Armature provides first-class integrations with major cloud providers through dedicated crates:
| Crate | Provider | Key Services |
|---|---|---|
armature-aws |
Amazon Web Services | S3, DynamoDB, SQS, SNS, SES, Lambda, KMS, Cognito |
armature-gcp |
Google Cloud Platform | Cloud Storage, Pub/Sub, Firestore, Spanner, BigQuery |
armature-azure |
Microsoft Azure | Blob Storage, Cosmos DB, Service Bus, Key Vault |
armature-redis |
Redis | Connection pooling, Pub/Sub, Cluster support |
Features:
// Add only the services you need
[dependencies]
armature-aws = { version = "0.1", features = ["s3", "sqs"] }
armature-gcp = { version = "0.1", features = ["storage", "pubsub"] }
armature-redis = "0.1"
// Register cloud services in your DI container
#[module_impl]
impl CloudModule {
#[provider(singleton)]
async fn aws() -> Arc<AwsServices> {
let config = AwsConfig::from_env()
.enable_s3()
.enable_sqs()
.build();
AwsServices::new(config).await.unwrap()
}
#[provider(singleton)]
async fn redis() -> Arc<RedisService> {
Arc::new(RedisService::new(RedisConfig::from_env().build()).await.unwrap())
}
}
// Use in controllers via injection
#[controller("/files")]
impl FileController {
#[post("/upload")]
async fn upload(
&self,
#[inject] aws: Arc<AwsServices>,
body: Bytes,
) -> Result<Json<Response>, HttpError> {
let s3 = aws.s3()?;
s3.put_object()
.bucket("my-bucket")
.key("file.txt")
.body(body.into())
.send()
.await?;
Ok(Json(Response { success: true }))
}
}
π See the Cloud Providers Guide for complete documentation.
The documentation website is an Angular 21 application located in the web/ directory.
Local Development:
cd web
pnpm install
pnpm start
Then open http://localhost:4200 in your browser.
Building for Production:
cd web
pnpm run build
The built website will be in web/dist/web/browser/.
GitHub Pages Deployment:
The website automatically deploys to GitHub Pages when changes are merged to the main branch.
Contributions are welcome! Please read our Contributing Guide and feel free to submit a Pull Request.