Kirei is a strictly typed Ruby micro/REST-framework for building scalable and performant APIs. It is built from the ground up to be clean and easy to use. Kirei is based on Sequel as an ORM, Sorbet for typing, and Rack as web server interface. It strives to have zero magic and to be as explicit as possible.
Kirei's main advantages over other frameworks are its strict typing, low memory footprint, and build-in high-performance logging and metric-tracking toolkits. It is opiniated in terms of tooling, allowing you to focus on your core-business. It is a great choice for building APIs that need to scale.
Kirei (きれい) is a Japanese adjective that primarily means "beautiful" or "pretty." It can also be used to describe something that is "clean" or "neat."
TL;DR:
- zero magic
- strict typing
- very few low level dependencies
- low memory footprint
- high performance
- simple to understand
This gem follows SemVer, however only after a stable release 1.0.0 is made.
A changelog is maintained in the CHANGELOG.md file and/or the GitHub Releases page.
Via rubygems:
gem 'kirei'
Test the latest version via git:
gem 'kirei', git: '[email protected]:swiknaba/kirei', branch: :main
Scaffold a new project:
bundle exec kirei new "MyApp"
Find a test app in the spec/test_app directory. It is a fully functional example of a Kirei app.
All models must inherit from T::Struct
and include Kirei::Model
. They must implement id
which must hold the primary key of the table. The primary key must be named id
and be of type T.any(String, Integer)
.
class User < T::Struct
extend T::Sig
include Kirei::Model
const :id, T.any(String, Integer)
const :name, String
end
user = User.find_by({ name: 'John' }) # T.nilable(User)
users = User.where({ name: 'John' }) # T::Array[User]
Updating a record returns a new instance. The original instance is not mutated:
updated_user = user.update({ name: 'Johnny' })
user.name # => 'John'
updated_user.name # => 'Johnny'
Delete keeps the original object intact. Returns true
if the record was deleted. Calling delete multiple times will return false
after the first (successful) call.
success = user.delete # => T::Boolean
# or delete by any query:
User.query.where('...').delete # => Integer, number of deleted records
To build more complex queries, Sequel can be used directly:
query = User.query.where({ name: 'John' })
query = query.where('...') # "query" is a 'Sequel::Dataset' that you can chain as you like
query = query.limit(10)
users = User.resolve(query) # T::Array[User]
first_user = User.resolve_first(query) # T.nilable(User)
# you can also cast the raw result manually
first_user = User.from_hash(query.first.stringify_keys)
Read the Sequel Migrations documentation for detailed information.
Sequel.migration do
up do
create_table(:airports) do
primary_key :id
String :name, null: false
end
end
down do
drop_table(:airports)
end
end
Applying migrations:
# create the database
bundle exec rake db:create
# drop the database
bundle exec rake db:drop
# apply all pending migrations
bundle exec rake db:migrate
# annotate the models with the schema
# this runs automatically after each migration
bundle exec rake db:annotate
# roll back the last n migration
STEPS=1 bundle exec rake db:rollback
# run db/seeds.rb to seed the database
bundle exec rake db:seed
# scaffold a new migration file
bundle exec rake 'db:migration[CreateAirports]'
Define routes anywhere in your app; by convention, they are defined in config/routes.rb
:
# config/routes.rb
module Kirei::Routing
Router.add_routes(
[
Route.new(
verb: Verb::GET,
path: "/livez",
controller: Controllers::Health,
action: "livez",
),
Route.new(
verb: Verb::GET,
path: "/airports",
controller: Controllers::Airports,
action: "index",
),
],
)
end
Controllers can be defined anywhere; by convention, they are defined in the app/controllers
directory:
module Controllers
class Airports < Kirei::Controller
extend T::Sig
sig { returns(T.anything) }
def index
search = T.let(params.fetch("q", nil), T.nilable(String))
airports = Kirei::Services::Runner.call("Airports::Filter") do
Airports::Filter.call(search) # T::Array[Airport]
end
# or use a serializer
data = Oj.dump(airports.map(&:serialize))
render(status: 200, body: data)
end
end
end
Services can be PORO. You can wrap an execution in Kirei::Services::Runner
which will emit a standardized logline and track its execution time.
module Airports
class Filter
extend T::Sig
sig do
params(
search: T.nilable(String),
).returns(T::Array[Airport])
end
def self.call(search)
return Airport.all if search.nil?
#
# SELECT *
# FROM "airports"
# WHERE (("name" ILIKE 'xx%') OR ("id" ILIKE 'xx%'))
#
query = Airport.query.where(Sequel.ilike(:name, "#{search}%"))
query = query.or(Sequel.ilike(:id, "#{search}%"))
Airport.resolve(query)
end
end
end
We welcome contributions from the community. Before starting work on a major feature, please get in touch with us either via email or by opening an issue on GitHub. "Major feature" means anything that changes user-facing features or significant changes to the codebase itself.
Please commit small and focused PRs with descriptive commit messages. If you are unsure about a PR, please open a draft PR to get early feedback. A PR must have a short description ("what"), a motiviation ("why"), and, if applicable, instructions how to test the changes, measure performance improvements, etc.
run
bin/release
which will guide you through the release process.