Curated library of atomic skills and personas for Hanami, dry-rb, and ROM Ruby development. Covers actions, slices, repositories, relations, changesets, providers, DI, operations, TDD, CLI, views, routing, validation, and 10 orchestration personas. Shared Ruby process skills have moved to ruby-core-skills. Uses Markdown + Front-matter architecture.
95
95%
Does it follow best practices?
Impact
96%
1.20xAverage score across 45 eval scenarios
Passed
No known issues
Use this workflow when implementing complex validation with dry-validation in Hanami 2.x.
Core principle: Complex validation belongs in dedicated Contract classes, not inline in Actions.
[Define Contract] — Create a dry-validation Contract
# app/contracts/user_contract.rb
module MyApp
module Contracts
class UserContract < Dry::Validation::Contract
params do
required(:email).value(:string, format?: /\A.+@.+\z/)
required(:password).value(:string, min_size?: 8)
required(:password_confirmation).value(:string)
end
rule(:password_confirmation) do
key.failure("must match password") unless value == values[:password]
end
end
end
end[Register in DI] — Load skill: inject-dependencies
Deps# config/providers/contracts.rb
Hanami.app.register_provider(:contracts) do
start do
register("contracts.user_contract", MyApp::Contracts::UserContract.new)
end
end[Inject into Action] — Load skill: validate-params
params block for complex validationclass Create < MyApp::Action
include Deps["contracts.user_contract"]
def handle(request, response)
result = contract.call(request.params[:user])
if result.failure?
halt 422, { errors: result.errors.to_h }.to_json
end
# ... proceed with valid data
end
endCheckpoint: Before proceeding, run the action spec to verify the contract is correctly injected. If the spec cannot resolve
contracts.user_contract, confirm the provider is registered and the container has been booted.
[Handle Results] — Load skill: handle-result-pattern
Success/Failure from service objectsFailure in Action with appropriate HTTP status# app/operations/create_user.rb
class CreateUser
include Dry::Monads[:result]
def call(attrs)
user = User.new(attrs)
if user.save
Success(user)
else
Failure(:save_failed)
end
end
end
# app/actions/users/create.rb
class Create < MyApp::Action
include Deps["contracts.user_contract", "operations.create_user"]
def handle(request, response)
result = user_contract.call(request.params[:user])
halt 422, { errors: result.errors.to_h }.to_json if result.failure?
case create_user.call(result.to_h)
in Success(user)
response.status = 201
response.body = { id: user.id }.to_json
in Failure(:save_failed)
halt 500, { error: "Could not save user" }.to_json
end
end
end[Write Tests] — Load skill: write-action-spec
# spec/contracts/user_contract_spec.rb
RSpec.describe MyApp::Contracts::UserContract do
subject(:contract) { described_class.new }
it "passes with valid input" do
result = contract.call(email: "a@b.com", password: "secret123", password_confirmation: "secret123")
expect(result).to be_success
end
it "fails when passwords do not match" do
result = contract.call(email: "a@b.com", password: "secret123", password_confirmation: "wrong")
expect(result.errors[:password_confirmation]).to include("must match password")
end
end| Mistake / Red Flag | Correct Approach |
|---|---|
| Complex validation inline in Actions | Extract to a Contract class. Actions should be thin. |
| Contract not registered in DI container | Register via provider; inject with Deps["contracts.user_contract"]. |
| Contracts not tested in isolation | Test directly with valid and invalid inputs before wiring into the Action. |
| Unstructured error responses | Return structured errors: { errors: { field: ["message"] } }. |
| Missing edge case validation tests | Cover cross-field rules and boundary values in Contract specs. |
.tessl-plugin
evals
scenario-1
scenario-2
scenario-3
scenario-4
scenario-5
scenario-6
scenario-7
scenario-8
scenario-9
scenario-10
scenario-11
scenario-12
scenario-13
scenario-14
scenario-15
scenario-16
scenario-17
scenario-18
scenario-19
scenario-20
scenario-21
scenario-22
scenario-23
scenario-24
scenario-25
scenario-26
scenario-27
scenario-28
scenario-29
scenario-30
scenario-31
scenario-32
scenario-33
scenario-34
scenario-35
scenario-36
scenario-37
scenario-38
scenario-39
scenario-40
scenario-41
scenario-42
scenario-43
scenario-44
scenario-45
skills
actions
build-json-api
create-action
handle-errors
validate-params
context
load-context
db
create-changeset
create-repository
define-relation
write-migration
dry-monads
handle-result-pattern
dry-rb
create-operation
create-validation-contract
personas
providers
configure-providers
implement-di
review-security
routing
define-routes
slices
configure-slice
create-slice
extract-slice
review-slice-boundaries
test-slice