Gleam — Type-Safe Language for the BEAM
## Overview
Best use case
Gleam — Type-Safe Language for the BEAM is best used when you need a repeatable AI agent workflow instead of a one-off prompt.
## Overview
Teams using Gleam — Type-Safe Language for the BEAM should expect a more consistent output, faster repeated execution, less prompt rewriting.
When to use this skill
- You want a reusable workflow that can be run more than once with consistent structure.
When not to use this skill
- You only need a quick one-off answer and do not need a reusable workflow.
- You cannot install or maintain the underlying files, dependencies, or repository context.
Installation
Claude Code / Cursor / Codex
Manual Installation
- Download SKILL.md from GitHub
- Place it in
.claude/skills/gleam/SKILL.mdinside your project - Restart your AI agent — it will auto-discover the skill
How Gleam — Type-Safe Language for the BEAM Compares
| Feature / Agent | Gleam — Type-Safe Language for the BEAM | Standard Approach |
|---|---|---|
| Platform Support | Not specified | Limited / Varies |
| Context Awareness | High | Baseline |
| Installation Complexity | Unknown | N/A |
Frequently Asked Questions
What does this skill do?
## Overview
Where can I find the source code?
You can find the source code on GitHub using the link provided at the top of the page.
SKILL.md Source
# Gleam — Type-Safe Language for the BEAM
## Overview
Gleam, the type-safe functional language that runs on the Erlang BEAM virtual machine. Helps developers build concurrent, fault-tolerant applications with Gleam's friendly syntax, exhaustive pattern matching, and access to the entire Erlang/OTP ecosystem.
## Instructions
### Project Setup
```bash
# Install Gleam
brew install gleam # macOS
# Or: curl -fsSL https://gleam.run/install | sh
# Create a new project
gleam new my_app
cd my_app
# Run
gleam run
# Test
gleam test
# Build for production
gleam build
```
### Type-Safe Web Server with Wisp
```gleam
// src/my_app/router.gleam — Type-safe HTTP routing
// Gleam catches errors at compile time — no runtime crashes from typos.
import gleam/http.{Get, Post, Delete}
import gleam/http/request.{type Request}
import gleam/http/response.{type Response}
import gleam/json
import gleam/result
import wisp.{type Request as WispRequest, type Response as WispResponse}
pub fn handle_request(req: WispRequest) -> WispResponse {
case wisp.path_segments(req) {
// GET /api/tasks
["api", "tasks"] -> list_tasks(req)
// GET /api/tasks/:id
["api", "tasks", id] -> get_task(req, id)
// POST /api/tasks
["api", "tasks"] if req.method == Post -> create_task(req)
// DELETE /api/tasks/:id
["api", "tasks", id] if req.method == Delete -> delete_task(req, id)
// 404 for everything else
_ -> wisp.not_found()
}
}
fn list_tasks(req: WispRequest) -> WispResponse {
// Pattern matching ensures we handle every case
case task_service.get_all() {
Ok(tasks) -> {
let body = json.array(tasks, task_to_json)
wisp.json_response(json.to_string_builder(body), 200)
}
Error(err) -> {
wisp.log_error("Failed to list tasks: " <> err)
wisp.internal_server_error()
}
}
}
fn create_task(req: WispRequest) -> WispResponse {
use body <- wisp.require_json(req)
// Decode and validate — compiler ensures all fields are handled
case decode_create_request(body) {
Ok(create_req) -> {
case task_service.create(create_req.title, create_req.priority) {
Ok(task) -> wisp.json_response(
json.to_string_builder(task_to_json(task)),
201,
)
Error(err) -> wisp.unprocessable_entity()
}
}
Error(_) -> wisp.bad_request()
}
}
```
### Custom Types and Pattern Matching
```gleam
// src/my_app/task.gleam — Domain types with exhaustive matching
// The compiler guarantees you handle every variant — impossible to forget a case.
import gleam/option.{type Option, Some, None}
/// Task priority — the compiler ensures you handle all variants
pub type Priority {
Low
Medium
High
Urgent
}
/// Task status with strict state machine semantics
pub type Status {
Todo
InProgress
Review
Done
Cancelled
}
pub type Task {
Task(
id: String,
title: String,
description: Option(String),
priority: Priority,
status: Status,
assignee: Option(String),
)
}
/// Validate status transitions — not all transitions are allowed.
/// The compiler ensures every combination is handled.
pub fn transition(from: Status, to: Status) -> Result(Status, String) {
case from, to {
// Valid transitions
Todo, InProgress -> Ok(InProgress)
InProgress, Review -> Ok(Review)
Review, Done -> Ok(Done)
Review, InProgress -> Ok(InProgress) // Send back for changes
// Any status can be cancelled
_, Cancelled -> Ok(Cancelled)
// Can't move backwards to Todo
_, Todo -> Error("Cannot move back to Todo")
// Can't skip steps
Todo, Done -> Error("Must go through InProgress and Review first")
Todo, Review -> Error("Must go through InProgress first")
// Already in target state
same, target if same == target -> Ok(target)
// Everything else is invalid
_, _ -> Error("Invalid status transition")
}
}
/// Calculate SLA based on priority — exhaustive, no default needed
pub fn sla_hours(priority: Priority) -> Int {
case priority {
Low -> 72
Medium -> 24
High -> 8
Urgent -> 2
}
}
```
### Concurrency with OTP Actors
```gleam
// src/my_app/task_actor.gleam — Concurrent actor for task management
// Gleam runs on the BEAM — millions of lightweight processes, fault-tolerant.
import gleam/erlang/process.{type Subject}
import gleam/otp/actor.{type Next}
/// Messages the actor can receive
pub type Message {
GetAll(reply_to: Subject(List(Task)))
Create(title: String, priority: Priority, reply_to: Subject(Result(Task, String)))
UpdateStatus(id: String, status: Status, reply_to: Subject(Result(Task, String)))
}
/// Actor state
pub type State {
State(tasks: List(Task), next_id: Int)
}
/// Handle each message type
fn handle_message(message: Message, state: State) -> Next(Message, State) {
case message {
GetAll(reply_to) -> {
process.send(reply_to, state.tasks)
actor.continue(state)
}
Create(title, priority, reply_to) -> {
let id = "task-" <> int.to_string(state.next_id)
let task = Task(
id: id,
title: title,
description: None,
priority: priority,
status: Todo,
assignee: None,
)
process.send(reply_to, Ok(task))
actor.continue(State(
tasks: [task, ..state.tasks],
next_id: state.next_id + 1,
))
}
UpdateStatus(id, new_status, reply_to) -> {
let result = update_task_status(state.tasks, id, new_status)
case result {
Ok(updated_tasks) -> {
process.send(reply_to, Ok(find_task(updated_tasks, id)))
actor.continue(State(..state, tasks: updated_tasks))
}
Error(err) -> {
process.send(reply_to, Error(err))
actor.continue(state)
}
}
}
}
}
/// Start the actor
pub fn start() -> Result(Subject(Message), actor.StartError) {
actor.start(State(tasks: [], next_id: 1), handle_message)
}
```
## Installation
```bash
# macOS
brew install gleam
# Linux
curl -fsSL https://gleam.run/install | sh
# Dependencies
gleam add wisp # Web framework
gleam add gleam_json # JSON encoding/decoding
gleam add sqlight # SQLite bindings
gleam add gleam_http # HTTP types
```
## Examples
### Example 1: Building a feature with Gleam
**User request:**
```
Add a real-time collaborative project setup to my React app using Gleam.
```
The agent installs the package, creates the component with proper Gleam initialization, implements the project setup with event handling and state management, and adds TypeScript types for the integration.
### Example 2: Migrating an existing feature to Gleam
**User request:**
```
I have a basic type-safe web server with wisp built with custom code. Migrate it to use Gleam for better type-safe web server with wisp support.
```
The agent reads the existing implementation, maps the custom logic to Gleam's API, rewrites the components using Gleam's primitives, preserves existing behavior, and adds features only possible with Gleam (like Custom Types and Pattern Matching, Concurrency with OTP Actors).
## Guidelines
1. **Let the compiler help** — Gleam's type system catches bugs at compile time; if it compiles, it almost certainly works
2. **Use custom types over strings** — Define `Priority` and `Status` as custom types, not strings; the compiler ensures you handle every variant
3. **Pattern matching for control flow** — Replace if/else chains with pattern matching; the compiler warns about missing cases
4. **Result type over exceptions** — Gleam has no exceptions; use `Result(value, error)` and `use` syntax for error handling
5. **Leverage the BEAM** — Gleam runs on Erlang's VM; you get fault tolerance, hot code reloading, and millions of concurrent processes for free
6. **Call Erlang/Elixir libraries** — Use `@external` to call any Erlang or Elixir library; the entire OTP ecosystem is available
7. **Pipelines for data transformation** — Use `|>` operator for readable data pipelines: `data |> filter |> map |> sort`
8. **Actors for state** — Use OTP actors (via gleam_otp) for concurrent stateful services; each actor is isolated and crash-safeRelated Skills
typeorm-entity-generator
Typeorm Entity Generator - Auto-activating skill for Backend Development. Triggers on: typeorm entity generator, typeorm entity generator Part of the Backend Development skill category.
chart-type-recommender
Chart Type Recommender - Auto-activating skill for Data Analytics. Triggers on: chart type recommender, chart type recommender Part of the Data Analytics skill category.
beam-pipeline-builder
Beam Pipeline Builder - Auto-activating skill for Data Pipelines. Triggers on: beam pipeline builder, beam pipeline builder Part of the Data Pipelines skill category.
microsoft-typescript
ALWAYS use when editing or working with *.ts, *.tsx, *.mts, *.cts files or code importing "typescript". Consult for debugging, best practices, or modifying typescript, TypeScript.
typespec-create-api-plugin
Generate a TypeSpec API plugin with REST operations, authentication, and Adaptive Cards for Microsoft 365 Copilot
typespec-create-agent
Generate a complete TypeSpec declarative agent with instructions, capabilities, and conversation starters for Microsoft 365 Copilot
typespec-api-operations
Add GET, POST, PATCH, and DELETE operations to a TypeSpec API plugin with proper routing, parameters, and adaptive cards
typescript-mcp-server-generator
Generate a complete MCP server project in TypeScript with tools, resources, and proper configuration
next-intl-add-language
Add new language to a Next.js + next-intl application
javascript-typescript-jest
Best practices for writing JavaScript/TypeScript tests using Jest, including mocking strategies, test structure, and common patterns.
answering-natural-language-questions-with-dbt
Writes and executes SQL queries against the data warehouse using dbt's Semantic Layer or ad-hoc SQL to answer business questions. Use when a user asks about analytics, metrics, KPIs, or data (e.g., "What were total sales last quarter?", "Show me top customers by revenue"). NOT for validating, testing, or building dbt models during development.
repomix-safe-mixer
Safely package codebases with repomix by automatically detecting and removing hardcoded credentials before packing. Use when packaging code for distribution, creating reference packages, or when the user mentions security concerns about sharing code with repomix.