Nanoservice framework Radian
The framework is designed to develop backend business applications based on SOA architecture. The framework supports microservice and monolith architecture.
DISCLAIMER: The project is at an early stage of development and is recommended for use with great caution. The project is open for general download due to the fact that the author needs it to support his current projects.
Framework
Quick start
Basic information
Four types of interaction
Workers and adapters
Jobs
Supported workers
Supported adapters
Project organization
Main code
Workers
Work with data
Custom adapters
Extra
Framework lifecycle
External links
Examples
Documentation
1 Framework
1 Quick start
Create a new project and main.go file and put this code to the main.go
package main
import (
"net/http"
"github.com/gin-gonic/gin"
"github.com/radianteam/framework"
sqlx "github.com/radianteam/framework/adapter/storage/sqlx"
"github.com/radianteam/framework/worker"
rest "github.com/radianteam/framework/worker/service/rest"
)
// REST handler function
func handlerMain(c *gin.Context, wc *worker.WorkerAdapters) {
// extract the database adapter
_, err := wc.Get("db")
if err != nil {
c.String(http.StatusBadRequest, "")
return
}
// use the adapters and whatever you want
// return standard gin results
c.String(http.StatusOK, "Hello world!\n")
}
func main() {
// create a new microservice instance
radian := framework.NewRadianMicroservice("main")
// create a new REST worker
workerConfig := &rest.RestConfig{Listen: "0.0.0.0", Port: 8088}
workerRest := rest.NewRestServiceWorker("service_rest", workerConfig)
// create a database adapter
dbConfig := &sqlx.SqlxConfig{Driver: "sqlite3", ConnectionString: "db.sqlite"}
dbAdapter := sqlx.NewSqlxAdapter("db", dbConfig)
//add the adapter to the worker
workerRest.SetAdapter(dbAdapter)
// create a route to the worker
workerRest.SetRoute("GET", "/", handlerMain)
// append worker to the framework
radian.AddWorker(workerRest)
// run the worker
radian.Run([]string{"service_rest"})
}
This file contains a REST worker with a handler function which gets database adapter and return status code with payload "Hello world!"
Main function:
- creates a framework instance
- creates a database adapter connected to a file database
- creates a new REST worker
- sets the handler to the REST worker
- runs the worker
Run the following command
curl 127.0.0.1:8088
Application will show the response
Hello world!
Also check examples
.
Radian framework is developed to support most of the cloud native patterns and bring new style to create tech projects from business requests and architecture based on components and interactions. In the framework they are named like workers and adapters. Also the framework is based on four types interaction concept. Author had a lot of experience and developed this framework to help small teams to organize their code and architecture and for huge projects and distributed teams prove a solution to work together without pain (like how to share protocols over teams, how to build microservices and run monolith, how to deal with informational security departments, etc).
Advantages of using this framework:
- run application both as monolith or as microservices
- boost code organization for distributed teams
- use 4 interaction paradigm and manage your teams better
- build services with simple reusable blocks
- easy advance framework with new adapters
- have predictable outcomes and stay clear to management teams
Disadvantages:
- the framework is at an early stage of development (but several companies already use it and are happy!)
- the project is supported by only one developer
- you have your own better solution or another requirements
3 Four types of interaction
Radian uses "Four types of interaction" concept. The concept explains that any business backend application can be designed with this interactions:
- Service interaction or request response. In this type a server listens for incoming connections and make clients wait during the response preparation. Features:
- passive interaction: service waits for requests
- sync interaction: clients wait while server prepares a response
- answer guarantee or error: the are no promises or queues to sent a response as a callback. Client receives an answer immediately
- stateless: every request has no any state. Clients must provide additional information about sessions
- strong dependency: clients must know server protocols and their address and must control any request fault
- consequent process: clients must wait for the previous request completion before starting the next
- request retries: if fails request side must retry a request and control fault tolerance policies
- can have DoS situation: if server is overflew with request it might not handle any more requests
Examples: REST or GRPC requests, HTTP, file download, etc.
- Event interaction or producer-consumer. Events are async and provide decouple pattern. Features:
- passive interaction: service waits for requests
- async interaction: clients don't wait, just put a message in a queue and forget about it
- answer not guarantee: the is a case when nobody listens to you messages
- unknown amount of listeners: service doesn't who and when will read the messages
- stateless: every queue message has no any state.
- ready for mass requests: client can push a lot of messages and never mind about their handling
- avoid DoS situation: all request are put in a queue. There can be queue overflow but in general the consumer will handle messages one by one with the maximum speed.
Examples: message broker queues (rabbitmq, kafka, sqs), message broadcast, log collecting, async tasks, etc.
- Periodic interaction or tasks. A handler can setup periodic logic and run it in the particular time moment or after timeout. Features:
- active interaction: service actively call functions to proceed
- internal interaction: service doesn't interact in general. This is an internal process
- fault tolerance ready: the common case to use this type of interaction is to restore software from unusual conditions or check some signals to react
- schedule ready: service plan to run some tasks
- wait or not: during processing task when service want to run another it can wait for the previous task completion or run another task with no waiting.
Examples: delayed tasks, fault restoration, periodic checks, status update, etc.
- Permanent interaction or threads. This is non-stop processes inside microservices to make some internal logic or interact with local component or signal sources. Features:
- run permanently: the is neither active nor passive interaction. This isn't interaction in general
- restart policies: if errors a permanent thread can be restarted or left down
- constant connection: perfect for listening some hardware signals or keeping connection to old style system with socket interaction
- interaction through another interaction: permanent tasks can init any other interaction (make requests or put a message in a queue) to interract with other services
Examples: connect to financial market protocols, listen to hardware signal sources, etc.
Microservices combining these interactions can provide solutions with any complexity. They can have several instances of one interaction in one microservice. For example, a REST service and a GRPC service in the same time. And their name is NANOSERVICES. Framework can run these nanoservices together like a monolith or devops teams can tune them to run separately and have fine grained control for more security or predictable loading.
4 Workers and adapters
Four types of interactions are implemented by workers: one of two basic primitives of the framework. Workers are processes inside an execution that make all dos.
Every worker is a goroutine. Developers can make any amount of workers and group to run separated or together. So they can be run as monolith or microservices or nanoservices.
If a project is a microservice that has a worker for request-response interaction, one for periodic logic and one for event consuming, it can be run as one process. This will be a regular microservice. On the other hand devops can run them separately like 3 nanoservices (sometimes cybersecurity demands it). Or devops can run one more process with only one worker to up performance if only one part of this microservice is overloaded.
If a project is a monolith with a lot of microservices, developers can put directives to run them all in one process or run separately several workers as one microservice.
Adapters are interaction primitives between workers or external system. Basically adapters support popular cloud native services like s3 or PaaS databases. Everything else like email sending or payment system integration also must be developed as adapters and connected to workers which need them.
All interactions between microservices developed with this framework supposed to be developed as adapters. It is better to provide an adapter for a microservice and don't force other teams to develop a library for it every time they need it.
If adapters and workers are considered from the point of view of graph architecture: workers are nodes and adapters are links.
Warning! If a project has microservices as workers and they interact via network connections or external message brokers and microcervices can be run as monolith, it is better to make interaction between them via channels or internal queues. Framework provides solutions for these cases: service worker based on channels (in development) and internal message broker (in development).
5 Jobs
Framework implements jobs that can be executed before or after the main application loop. They are called pretasks and posttasks.
Pretasks can be used for:
- getting auth keys
- downloading session certs
- initialize something like queues and other stuff you may need before starting.
Posttasks are for:
- revoking auth session
- notice other processes about stopping
- deleting session data and queues
This tasks are not for migrations! You can use them for it but it is better to do migrations by devops. The only way to use tasks for migration or data seed is when you run your application as monolith.
2 Supported workers
Worker |
Type |
Description |
REST |
Service |
Service based on Gin REST library |
GRPC |
Service |
Service based on vanilla GRPC library |
RabbitMQ |
Event |
Event worker based on RabbitMQ framework adapter |
AWS SQS |
Event |
Event worker based on SQS framework adapter |
Schedule |
Periodic |
Scheduler for periodic tasks based on Chrono library |
Job |
Permament |
Task worker for permament workers and one-time operations in pretasks and posttasks |
Monitoring |
Special |
REST Service based on Gin library and Prometheus Go libary with /metrics endpoint for prometheus scraper |
|
|
|
3 Supported adapters
Adapter |
Type |
Description |
Config |
Utils |
Utility adapter for configuration loading. Supports loading from file and environment variables. Configuration can be unmarshaled in a service and adapter configuration structs with correct tags. |
Sqlx |
Storage |
Database adapter based on Sqlx library. Supports all database drivers for database/sql package |
MongoDB |
Storage |
Database adapter based on MongoDB driver |
ArangoDB |
Storage |
Gaph database adapter based on ArangoDB driver |
AWS S3 |
Storage |
Object storage adapter implementing S3 protocol. Based on AWS SDK |
AWS SQS |
Event |
Event adapter implementing SQS protocol. Based on AWS SDK |
RabbitMQ |
Event |
Event adapter based on AMQP library |
OIDC |
Auth |
Auth adapter implementing OpenID connect protocol. Based on go-oidc library. Supports sync (introspect) and offline (public keys) checking |
|
|
|
4 Project organization
1 Main code
Projects made with the framework doesn't have strict structure. Developers can make all logic in one main.go file or use their familiar structure. The author of the framework likes the following:
Project directory
|
|- contract (optional)
| |- gen
| | |- contract.pb.go
| | |- contract_grpc.pb.go
| |
| |- contract.proto
|
|- data
| |- entity
| |- mapper.go
| |- model.go
| |- repository.go
|
|- doc
| |- database.dbm
| |- postman.json
| |- logic.plantuml
|
|- job
| |- any_prejob.go
|
|- migration
| |- 00_scheme.sql
|
|- util
| |- helper.go
|
|- worker
| |- service
| | |- name
| | |- rest_handler_1.go
| | |- rest_handler_2.go
| |
| |- event
| | |- name
| | |- event_handler_1.go
| | |- event_handler_2.go
| |
| |- periodic
| |- schedule_handler_1.go
|
|- docker-dompose.yml
|- Dockerfile
|- go.mod
|- go.sum
|- main.go
|- README.md
Project directory contains:
- contract (ignore it if project doesn't use GRPC or another descriptive protocols): here developers can have proto descriptions or schemes for another protocols. Also this directory keeps gen directory for saving compiled proto and GRPC files. This means that developers must save in a project compiled version of their protocols. This is useful for including to another project and developing custom adapters for framework microservices
- data: data folder has folders for data transfer objects and their logic. Every entity has three files:
- model.go for structures of a transfer object and enums
- repository.go for storage logic implementation
- mapper.go for mapping model structures into or from another data like proto or JSON
- doc: this folder contains supporting documents like database descriptions, schemes, UML and files for testing software (like Postman). All the documents must have text format so that developers can easily check their updates. Images are acceptable but not recommended
- job: contains files with handlers for prejobs and postjobs
- migration: contains sql files, json scheme validation files and other data to load into storage or stateful components
- util: any code that is used in several packages to avoid boilerplate and copy-past code
- worker: contains directories for services. Every service directory contains files for handlers or another functions and helpers.
- docker-compose.yml: good idea to provide some testing environment for the project
- Dockerfile: used in docker-compose and provide a proper way to build a container to run. If project deploys not in a docker developers can use another technologies and ignore docker files
- go.mod: no commentaries
- go.sum: no commentaries
- main.go: implements project setup with adapters and workers
- README.md: it is important and really cool when developers provide some documentation for the project
2 Workers
Not implemented
3 Work with data
Not implemented
4 Custom adapters
Not implemented
Not implemented
6 Framework lifecycle
Not implemented
5 External links
1 Examples
Simple REST service
REST service with monitoring (prometheus metrics)