Log Aggregator is a Go application that collects logs from multiple sources (e.g., servers, applications) and centralizes them for analysis and monitoring. It leverages Go's concurrency features to handle log ingestion and provides a web interface for searching and visualizing logs.
- Concurrent Log Ingestion: Efficiently handles log ingestion from multiple sources using Go's goroutines.
- In-Memory Storage: Stores logs in memory for fast access and retrieval.
- Web Interface: Provides a simple web interface for viewing and searching logs.
- Real-time Updates: Logs are updated in real-time as they are ingested.
log-aggregator/
├── cmd/
│ └── log-aggregator/
│ └── main.go
├── internal/
│ ├── aggregator/
│ │ └── aggregator.go
│ ├── storage/
│ │ └── memory.go
│ └── web/
│ └── server.go
├── pkg/
│ └── log/
│ └── log.go
├── web/
│ ├── static/
│ │ └── css/
│ │ └── style.css
│ └── templates/
│ └── index.html
├── go.mod
└── go.sum
-
cmd/log-aggregator: Contains the entry point for the application.
main.go
: Initializes and starts the log aggregator and web server.
-
internal/aggregator: Handles log ingestion and aggregation.
aggregator.go
: Defines the aggregator structure and methods for adding logs and managing worker goroutines.
-
internal/storage: Manages log storage.
memory.go
: Provides an in-memory storage implementation.
-
internal/web: Manages the web server and routes.
server.go
: Defines the HTTP server and handlers for serving web pages.
-
pkg/log: Defines the log entry structure.
log.go
: Contains the definition of a log entry.
-
web/static: Contains static assets for the web interface.
css/style.css
: CSS file for styling the web interface.
-
web/templates: Contains HTML templates for the web interface.
index.html
: Main HTML template for displaying logs.
- Go 1.18 or higher
-
Clone the repository:
git clone https://github.com/yourusername/log-aggregator.git cd log-aggregator
-
Initialize and tidy Go modules:
go mod tidy
-
Navigate to the
cmd/log-aggregator
directory:cd cmd/log-aggregator
-
Run the application:
go run main.go
Open your web browser and navigate to http://localhost:8080
to view the logs.
The aggregator collects logs from multiple sources and uses worker goroutines to store them in an in-memory storage. It supports starting and stopping worker goroutines and adding new log entries.
Example:
package aggregator
import (
"sync"
"log-aggregator/pkg/log"
"log-aggregator/internal/storage"
)
type Aggregator struct {
storage storage.Storage
logChannel chan log.LogEntry
wg sync.WaitGroup
}
func NewAggregator(storage storage.Storage) *Aggregator {
return &Aggregator{
storage: storage,
logChannel: make(chan log.LogEntry, 100),
}
}
func (a *Aggregator) Start(numWorkers int) {
for i := 0; i < numWorkers; i++) {
a.wg.Add(1)
go a.worker()
}
}
func (a *Aggregator) Stop() {
close(a.logChannel)
a.wg.Wait()
}
func (a *Aggregator) AddLog(entry log.LogEntry) {
a.logChannel <- entry
}
func (a *Aggregator) worker() {
defer a.wg.Done()
for entry := range a.logChannel {
a.storage.Store(entry)
}
}
The storage package provides an interface and an in-memory implementation for storing logs. This allows for flexibility in changing the storage backend without affecting other components.
Example:
package storage
import (
"sync"
"log-aggregator/pkg/log"
)
type Storage interface {
Store(entry log.LogEntry)
Retrieve() []log.LogEntry
}
type MemoryStorage struct {
logs []log.LogEntry
mu sync.Mutex
}
func NewMemoryStorage() *MemoryStorage {
return &MemoryStorage{
logs: make([]log.LogEntry, 0),
}
}
func (s *MemoryStorage) Store(entry log.LogEntry) {
s.mu.Lock()
defer s.mu.Unlock()
s.logs = append(s.logs, entry)
}
func (s *MemoryStorage) Retrieve() []log.LogEntry {
s.mu.Lock()
defer s.mu.Unlock()
return s.logs
}
The web package handles HTTP requests and serves the web interface. It uses HTML templates to render the log data.
Example:
package web
import (
"html/template"
"log"
"net/http"
"log-aggregator/internal/storage"
)
type Server struct {
storage storage.Storage
tmpl *template.Template
}
func NewServer(storage storage.Storage, tmpl *template.Template) *Server {
return &Server{
storage: storage,
tmpl: tmpl,
}
}
func (s *Server) Start(addr string) error {
http.HandleFunc("/", s.handleIndex)
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.Dir("web/static"))))
return http.ListenAndServe(addr, nil)
}
func (s *Server) handleIndex(w http.ResponseWriter, r *http.Request) {
logs := s.storage.Retrieve()
err := s.tmpl.Execute(w, logs)
if err != nil {
log.Printf("failed to execute template: %v", err)
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
}
}
Contributions are welcome! Please fork the repository and submit pull requests.
- This project was inspired by the need for a simple log aggregation solution.
- Thanks to the Go community for their support and resources.
For any questions or inquiries, please contact Ashik@spudblocks.com
This `README.md` file provides a comprehensive overview of your project, including its structure, installation instructions, and usage details. It includes detailed information about each component and example code snippets to illustrate how different parts of the application work. Be sure to replace placeholders like `yourusername` and `yourname@example.com` with your actual GitHub username and email address.