📚 Read the full documentation at grafana.github.io/nanogit
nanogit is a lightweight, cloud-native Git implementation designed for applications that need efficient Git operations over HTTPS without the complexity and resource overhead of traditional Git implementations.
-
HTTPS-only Git operations - Works with any Git service supporting Smart HTTP Protocol v2 (GitHub, GitLab, Bitbucket, etc.), eliminating the need for SSH key management in cloud environments
-
Stateless architecture - No local .git directory dependency, making it perfect for serverless functions, containers, and microservices where persistent local state isn't available or desired
-
Memory-optimized design - Streaming packfile operations and configurable writing modes minimize memory usage, crucial for bulk operations and memory-constrained environments
-
Flexible storage architecture - Pluggable object storage and configurable writing modes allow optimization for different deployment patterns, from high-performance in-memory operations to memory-efficient disk-based processing
-
Cloud-native authentication - Built-in support for Basic Auth and API tokens, designed for automated workflows and CI/CD systems without interactive authentication
-
Essential Git operations - Focused on core functionality (read/write objects, commit operations, diffing) without the complexity of full Git implementations, reducing attack surface and resource requirements
-
High performance - Significantly faster than traditional Git implementations for common cloud operations, with up to 300x speed improvements for certain scenarios
The following features are explicitly not supported:
git://and Git-over-SSH protocols- File protocol (local Git operations)
- Commit signing and signature verification
- Git hooks
- Git configuration management
- Direct .git directory access
- "Dumb" servers
- Complex permissions (all objects use mode 0644)
While go-git is a mature Git implementation, nanogit is designed for cloud-native, multitenant environments requiring minimal, stateless operations.
| Feature | nanogit | go-git |
|---|---|---|
| Protocol | HTTPS-only | All protocols |
| Storage | Stateless, configurable object storage + writing modes | Local disk operations |
| Cloning | Path filtering with glob patterns, shallow clones | Full repository clones |
| Scope | Essential operations only | Full Git functionality |
| Use Case | Cloud services, multitenant | General purpose |
| Resource Usage | Minimal footprint | Full Git features |
Choose nanogit for lightweight cloud services requiring stateless operations and minimal resources. Use go-git when you need full Git functionality, local operations, or advanced features.
This are some of the performance differences between nanogit and go-git in some of the measured scenarios:
| Scenario | Speed | Memory Usage |
|---|---|---|
| CreateFile (XL repo) | 306x faster | 186x less |
| UpdateFile (XL repo) | 291x faster | 178x less |
| DeleteFile (XL repo) | 302x faster | 175x less |
| BulkCreateFiles (1000 files, medium repo) | 607x faster | 11x less |
| CompareCommits (XL repo) | 60x faster | 96x less |
| GetFlatTree (XL repo) | 258x faster | 160x less |
For detailed performance metrics, see the latest performance report and performance analysis.
- Go 1.25 or later.
- Git (for development)
Install the latest version:
go get github.com/grafana/nanogit@latestOr install a specific version:
go get github.com/grafana/nanogit@v0.x.x # Replace v0.x.x with the latest released versionSee all available versions on the releases page.
nanogit provides a CLI for terminal-based Git operations. Download pre-built binaries from the releases page or install using Go.
Download pre-built binary (recommended):
# Linux (amd64)
wget https://github.com/grafana/nanogit/releases/latest/download/nanogit_Linux_x86_64.tar.gz
tar -xzf nanogit_Linux_x86_64.tar.gz
sudo mv nanogit /usr/local/bin/
# macOS (Apple Silicon)
wget https://github.com/grafana/nanogit/releases/latest/download/nanogit_Darwin_arm64.tar.gz
tar -xzf nanogit_Darwin_arm64.tar.gz
sudo mv nanogit /usr/local/bin/
# macOS (Intel)
wget https://github.com/grafana/nanogit/releases/latest/download/nanogit_Darwin_x86_64.tar.gz
tar -xzf nanogit_Darwin_x86_64.tar.gz
sudo mv nanogit /usr/local/bin/
# Windows (PowerShell)
Invoke-WebRequest -Uri "https://github.com/grafana/nanogit/releases/latest/download/nanogit_Windows_x86_64.zip" -OutFile "nanogit.zip"
Expand-Archive nanogit.zip
Move-Item nanogit\nanogit.exe C:\Windows\System32\Or install using Go:
go install github.com/grafana/nanogit/cli/cmd/nanogit@latestSee the CLI documentation for more details.
// Create client with authentication
client, err := nanogit.NewHTTPClient(
"https://github.com/user/repo.git",
options.WithBasicAuth("username", "token"),
)
// Get main branch and create staged writer
ref, err := client.GetRef(ctx, "refs/heads/main")
writer, err := client.NewStagedWriter(ctx, ref)
// Create and update files
writer.CreateBlob(ctx, "docs/new-feature.md", []byte("# New Feature"))
writer.UpdateBlob(ctx, "README.md", []byte("Updated content"))
// Commit changes with proper author/committer info
author := nanogit.Author{
Name: "John Doe",
Email: "john@example.com",
Time: time.Now(),
}
committer := nanogit.Committer{
Name: "Deploy Bot",
Email: "deploy@example.com",
Time: time.Now(),
}
commit, err := writer.Commit(ctx, "Add feature and update docs", author, committer)
writer.Push(ctx)nanogit provides efficient cloning with flexible path filtering, ideal for CI environments where only specific directories are needed:
// First, get the commit hash for the branch you want to clone
ref, err := client.GetRef(ctx, "main")
if err != nil {
return err
}
// Clone specific directories only (perfect for CI with no caching)
result, err := client.Clone(ctx, nanogit.CloneOptions{
Path: "/tmp/my-repo", // Local filesystem path (required)
Hash: ref.Hash, // Commit hash (required)
IncludePaths: []string{"src/**", "docs/**"}, // Include only these paths
ExcludePaths: []string{"*.tmp", "node_modules/**"}, // Exclude these paths
})
if err != nil {
return err
}
// result.Commit contains the commit information
// result.FlatTree contains filtered file tree
// Files are automatically written to result.Path
fmt.Printf("Cloned %d of %d files to %s\n",
result.FilteredFiles, result.TotalFiles, result.Path)Key clone features:
- Path filtering: Use glob patterns to include/exclude specific files and directories
- Filesystem output: Automatically writes filtered files to specified local path
- Shallow clones: Fetch only the latest commit to minimize bandwidth
- Branch isolation: Clone only specific branches to reduce transfer time
- CI optimized: Perfect for build environments with no persistent storage
- Performance tuning: Configurable batch fetching and concurrency for optimal performance
The Clone operation supports two key performance optimization options to significantly improve cloning speed:
// Clone with performance optimizations
result, err := client.Clone(ctx, nanogit.CloneOptions{
Path: "/tmp/my-repo",
Hash: ref.Hash,
IncludePaths: []string{"pkg/api/**"},
BatchSize: 50, // Fetch 50 blobs per network request
Concurrency: 8, // Use 8 concurrent workers
})BatchSize - Controls how many blobs to fetch in a single network request:
- Value 0 or 1: Fetches blobs individually (backward compatible, default behavior)
- Values > 1: Enables batch fetching, reducing network round trips by 50-70%
- Automatically falls back to individual fetching if a blob is missing from a batch response
- Recommended for repositories with many files to minimize network overhead
- Recommended value: 20-100 depending on average blob size and network conditions
Concurrency - Controls parallel blob fetching:
- Value 0 or 1: Sequential fetching (backward compatible, default behavior)
- Values > 1: Enables concurrent fetching using worker pools
- Works with both batch fetching (fetches multiple batches in parallel) and individual fetching
- Recommended value: 4-10 depending on network conditions and server capacity
- Can improve performance by 2-3x on high-latency networks
Performance Impact: Combined optimization (BatchSize=50, Concurrency=8) can achieve 5-10x speedup compared to default sequential fetching, making it ideal for CI/CD environments and large repository operations.
nanogit provides flexible writing modes to optimize memory usage during write operations:
// Auto mode (default) - smart memory/disk switching
writer, err := client.NewStagedWriter(ctx, ref)
// Memory mode - maximum performance
writer, err := client.NewStagedWriter(ctx, ref, nanogit.WithMemoryStorage())
// Disk mode - minimal memory usage for bulk operations
writer, err := client.NewStagedWriter(ctx, ref, nanogit.WithDiskStorage())For detailed information about writing modes, performance characteristics, and use cases, see Storage Architecture Documentation.
nanogit includes a pluggable retry mechanism, making operations more robust against transient network errors and server issues. The retry mechanism follows the same pattern as storage options, using context-based injection.
By default, no retries are performed (backward compatible). To enable retries, inject a retrier into the context:
import "github.com/grafana/nanogit/retry"
// Create a retrier with default settings (3 attempts, exponential backoff)
retrier := retry.NewExponentialBackoffRetrier()
ctx = retry.ToContext(ctx, retrier)
// All HTTP operations will now use retry logic
client, err := nanogit.NewHTTPClient(repo, options...)
ref, err := client.GetRef(ctx, "main")The ExponentialBackoffRetrier provides configurable exponential backoff retry logic:
// Customize retry behavior
retrier := retry.NewExponentialBackoffRetrier().
WithMaxAttempts(5). // Retry up to 5 times
WithInitialDelay(200 * time.Millisecond). // Start with 200ms delay
WithMaxDelay(10 * time.Second). // Cap at 10 seconds
WithMultiplier(2.0). // Double delay each retry
WithJitter() // Add random jitter
ctx = retry.ToContext(ctx, retrier)You can implement your own retry logic by implementing the Retrier interface:
type MyRetrier struct {
// Your custom fields
}
func (r *MyRetrier) ShouldRetry(ctx context.Context, err error, attempt int) bool {
// Your retry logic
return true
}
func (r *MyRetrier) Wait(ctx context.Context, attempt int) error {
// Your backoff logic
return nil
}
func (r *MyRetrier) MaxAttempts() int {
return 3
}
// Use your custom retrier
ctx = retry.ToContext(ctx, &MyRetrier{})The retry mechanism automatically retries on:
- Network timeout errors
- 5xx server errors: Server unavailable errors (for GET requests only)
- Temporary errors: Any error marked as temporary
The retry mechanism does not retry on:
- 4xx client errors: Bad requests, authentication failures, etc.
- Context cancellation: When the context is cancelled or deadline exceeded
- POST request 5xx errors: POST requests cannot retry 5xx errors because the request body (
io.Reader) is consumed when the request is sent and cannot be re-read
- GET requests (SmartInfo): Retry on network errors and 5xx status codes
- POST requests (UploadPack, ReceivePack): Retry only on network errors (before response is received)
This limitation exists because POST request bodies are consumed during the HTTP request and cannot be re-read for retries.
nanogit features a flexible two-layer storage architecture that separates concerns and allows independent optimization:
- Writing modes: Control temporary storage during packfile creation (memory/disk/auto)
- Object storage: Handle long-term caching and retrieval of Git objects (pluggable backends)
nanogit provides context-based object storage with pluggable backends. The default in-memory implementation is optimized for stateless operations, but you can implement custom backends for persistent caching:
// Custom storage example
ctx = storage.ToContext(ctx, myRedisStorage)
client, err := nanogit.NewHTTPClient(repo, options...)This enables sharing Git object cache across multiple repositories, persistent caching across service restarts, and optimization for specific deployment patterns.
For detailed information about storage architecture, writing modes, and custom implementations, see Storage Architecture Documentation.
nanogit includes generated mocks for easy unit testing. The mocks are generated using counterfeiter and provide comprehensive test doubles for both the Client and StagedWriter interfaces.
For detailed testing examples and instructions, see CONTRIBUTING.md. You can also find complete working examples in mocks/example_test.go.
The gittest package provides utilities for testing Git operations with a containerized Gitea server:
go get github.com/grafana/nanogit/gittest@latestSee gittest README for API documentation and examples.
We welcome contributions! Please see our Contributing Guide for details on how to submit pull requests, report issues, and set up your development environment.
This project follows the Grafana Code of Conduct. By participating, you are expected to uphold this code.
This project is licensed under the Apache License 2.0 - see the LICENSE file for details.
This project is currently in active development. While it's open source, it's important to note that it was initially created as part of a hackathon. We're working to make it production-ready, but please use it with appropriate caution.
Comprehensive documentation is available at grafana.github.io/nanogit:
- Getting Started - Installation and quick start guide
- Architecture - Design principles, storage backend, performance
- API Reference (GoDoc) - Complete API documentation
- Changelog - Version history and release notes
Want to learn how Git works? The following resources are useful:
- Git on the Server - The Protocols
- Git Protocol v2
- Pack Protocol
- Git HTTP Backend
- HTTP Protocol
- Git Protocol HTTP
- Git Protocol v2
- Git Protocol Pack
- Git Protocol Common
If you find a security vulnerability, please report it to us according to our security policy.
- GitHub Issues: Create an issue
- Community: Grafana Community Forums
- The Grafana team for their support and guidance
- The open source community for their valuable feedback and contributions
