Maho 🦝🍻
@mapache@hachyderm.io
Ok, hotels, flights, trains, and one extra family fun day were booked. See you next year at #fosdem2026 !
https://fosdem.org/2026/schedule/event/decentralised-badges-activitypub-badgefed/


@hongminhee@hollo.social · 990 following · 1375 followers
An intersectionalist, feminist, and socialist living in Seoul (UTC+09:00). @tokolovesme's spouse. Who's behind @fedify, @hollo, and @botkit. Write some free software in #TypeScript, #Haskell, #Rust, & #Python. They/them.
서울에 사는 交叉女性主義者이자 社會主義者. 金剛兔(@tokolovesme)의 配偶者. @fedify, @hollo, @botkit 메인테이너. #TypeScript, #Haskell, #Rust, #Python 等으로 自由 소프트웨어 만듦.
| Website | GitHub | Blog | Hackers' Pub |
|---|---|---|---|
@mapache@hachyderm.io
Ok, hotels, flights, trains, and one extra family fun day were booked. See you next year at #fosdem2026 !
https://fosdem.org/2026/schedule/event/decentralised-badges-activitypub-badgefed/

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
親子丼と焼鳥。

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
ハチワレ!

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
豆腐パンナコッタも!

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
豆富食堂(豆腐じゃない)というお店で豆腐御膳を食べている。

@hongminhee@hollo.social · Reply to Juntai Park's post
@arkjun 実はそれは私もよく知らないですね。😂
@smallcircles@social.coop
Read about #Encyclia by @jfietkau and plans to bring more #OpenScience to our #fediverse
https://discuss.coding.social/t/my-current-goals-for-activitypub-and-academic-data/750
There are multiple other #ActivityPub projects that share interests to connect more tightly the academic world to the #SocialWeb.
Backed by #NGI0 @nlnet funding there is the very promising @bonfire and #Plaudit in earlier rounds (#WebMentions, not fedi).
We should align on #OpenStandards
https://encyclia.pub
https://bonfirenetworks.org
https://plaudit.pub

@hongminhee@hollo.social
LogTape 1.3.0 is out!
This release brings official middleware for Express, Fastify, Hono, and Koa with Morgan-compatible formats, plus Drizzle ORM integration for database query logging.
For SDK authors: the new withCategoryPrefix() lets you wrap internal library logs under your own category—so users only need to configure logging for your package, not every dependency you use internally.
Also: OpenTelemetry now supports gRPC and HTTP/Protobuf protocols, and the Sentry sink gained automatic trace correlation and breadcrumbs.

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
成田国際空港に無事に着陸!
@jdv_jazz@mastodon.nl
Miles Davis - Milestones
#JazzDeVille #Jazz #NowPlaying #MilesDavis

@hongminhee@hollo.social
明日、弟と2泊3日で東京に旅行に行くんだ。

@hongminhee@hollo.social · Reply to Ayo's post
@ayo Let's hang out there! 🙌🏼
@omasanori@mstdn.maud.io
初は静的サイト向けのActivityPubブリッジ。フィードを元に各ポストに対してActivityPub用のエントリーを作成してActivityPub実装とうまくやり取りできるようにしてくれる。
Hatsu 0.3.4 🎉
https://github.com/importantimport/hatsu/releases/tag/v0.3.4
@tatmius@vivaldi.net · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
@hongminhee 個人的な見解ですが、日本でDiscordがそれなりの知名度を獲得したのは、2019年か2020年ぐらいで、それ以前からあるユーザーコミュニティはslackが多い印象ですね。

@hongminhee@hollo.social · Reply to tatmius(タミアス)'s post
@tatmius なるほど!韓国も似たような感じですが、最近はみんなSlackからDiscordに移行する傾向にありますね。

@hongminhee@hollo.social
日本語圏では、オープンソースコミュニティやユーザーグループなどのチャットは、DiscordよりもSlackを使っているところが多い気がする。たまたま僕が参加しているコミュニティだけなのかな?🤔
@lobsters@mastodon.social
Stop writing if statements for your CLI flags https://lobste.rs/s/hzyyyy #javascript
https://hackers.pub/@hongminhee/2025/stop-writing-if-statements-for-your-cli-flags

@hongminhee@hollo.social
Still validating CLI option relationships with if statements? Your type system can do it for you.
@hongminhee@hackers.pub
If you've built CLI tools, you've written code like this:
if (opts.reporter === "junit" && !opts.outputFile) {
throw new Error("--output-file is required for junit reporter");
}
if (opts.reporter === "html" && !opts.outputFile) {
throw new Error("--output-file is required for html reporter");
}
if (opts.reporter === "console" && opts.outputFile) {
console.warn("--output-file is ignored for console reporter");
}
A few months ago, I wrote Stop writing CLI validation. Parse it right the first time. about parsing individual option values correctly. But it didn't cover the relationships between options.
In the code above, --output-file only makes sense when --reporter is junit or html. When it's console, the option shouldn't exist at all.
We're using TypeScript. We have a powerful type system. And yet, here we are, writing runtime checks that the compiler can't help with. Every time we add a new reporter type, we need to remember to update these checks. Every time we refactor, we hope we didn't miss one.
The old guard—Commander, yargs, minimist—were built before TypeScript became mainstream. They give you bags of strings and leave type safety as an exercise for the reader.
But we've made progress. Modern TypeScript-first libraries like cmd-ts and Clipanion (the library powering Yarn Berry) take types seriously:
// cmd-ts
const app = command({
args: {
reporter: option({ type: string, long: 'reporter' }),
outputFile: option({ type: string, long: 'output-file' }),
},
handler: (args) => {
// args.reporter: string
// args.outputFile: string
},
});
// Clipanion
class TestCommand extends Command {
reporter = Option.String('--reporter');
outputFile = Option.String('--output-file');
}
These libraries infer types for individual options. --port is a number. --verbose is a boolean. That's real progress.
But here's what they can't do: express that --output-file is required when --reporter is junit, and forbidden when --reporter is console. The relationship between options isn't captured in the type system.
So you end up writing validation code anyway:
handler: (args) => {
// Both cmd-ts and Clipanion need this
if (args.reporter === "junit" && !args.outputFile) {
throw new Error("--output-file required for junit");
}
// args.outputFile is still string | undefined
// TypeScript doesn't know it's definitely string when reporter is "junit"
}
Rust's clap and Python's Click have requires and conflicts_with attributes, but those are runtime checks too. They don't change the result type.
If the parser configuration knows about option relationships, why doesn't that knowledge show up in the result type?
conditional() Optique treats option relationships as a first-class concept. Here's the test reporter scenario:
import { conditional, object } from "@optique/core/constructs";
import { option } from "@optique/core/primitives";
import { choice, string } from "@optique/core/valueparser";
import { run } from "@optique/run";
const parser = conditional(
option("--reporter", choice(["console", "junit", "html"])),
{
console: object({}),
junit: object({
outputFile: option("--output-file", string()),
}),
html: object({
outputFile: option("--output-file", string()),
openBrowser: option("--open-browser"),
}),
}
);
const [reporter, config] = run(parser);
The conditional() combinator takes a discriminator option (--reporter) and a map of branches. Each branch defines what other options are valid for that discriminator value.
TypeScript infers the result type automatically:
type Result =
| ["console", {}]
| ["junit", { outputFile: string }]
| ["html", { outputFile: string; openBrowser: boolean }];
When reporter is "junit", outputFile is string—not string | undefined. The relationship is encoded in the type.
Now your business logic gets real type safety:
const [reporter, config] = run(parser);
switch (reporter) {
case "console":
runWithConsoleOutput();
break;
case "junit":
// TypeScript knows config.outputFile is string
writeJUnitReport(config.outputFile);
break;
case "html":
// TypeScript knows config.outputFile and config.openBrowser exist
writeHtmlReport(config.outputFile);
if (config.openBrowser) openInBrowser(config.outputFile);
break;
}
No validation code. No runtime checks. If you add a new reporter type and forget to handle it in the switch, the compiler tells you.
Test reporters are a nice example, but let's try something with more variation. Database connection strings:
myapp --db=sqlite --file=./data.db
myapp --db=postgres --host=localhost --port=5432 --user=admin
myapp --db=mysql --host=localhost --port=3306 --user=root --ssl
Each database type needs completely different options:
Here's how you model this:
import { conditional, object } from "@optique/core/constructs";
import { withDefault, optional } from "@optique/core/modifiers";
import { option } from "@optique/core/primitives";
import { choice, string, integer } from "@optique/core/valueparser";
const dbParser = conditional(
option("--db", choice(["sqlite", "postgres", "mysql"])),
{
sqlite: object({
file: option("--file", string()),
}),
postgres: object({
host: option("--host", string()),
port: withDefault(option("--port", integer()), 5432),
user: option("--user", string()),
password: optional(option("--password", string())),
}),
mysql: object({
host: option("--host", string()),
port: withDefault(option("--port", integer()), 3306),
user: option("--user", string()),
ssl: option("--ssl"),
}),
}
);
The inferred type:
type DbConfig =
| ["sqlite", { file: string }]
| ["postgres", { host: string; port: number; user: string; password?: string }]
| ["mysql", { host: string; port: number; user: string; ssl: boolean }];
Notice the details: PostgreSQL defaults to port 5432, MySQL to 3306. PostgreSQL has an optional password, MySQL has an SSL flag. Each database type has exactly the options it needs—no more, no less.
With this structure, writing dbConfig.ssl when the mode is sqlite isn't a runtime error—it's a compile-time impossibility.
Try expressing this with requires_if attributes. You can't. The relationships are too rich.
Once you see it, you find this pattern in many CLI tools:
Authentication modes:
const authParser = conditional(
option("--auth", choice(["none", "basic", "token", "oauth"])),
{
none: object({}),
basic: object({
username: option("--username", string()),
password: option("--password", string()),
}),
token: object({
token: option("--token", string()),
}),
oauth: object({
clientId: option("--client-id", string()),
clientSecret: option("--client-secret", string()),
tokenUrl: option("--token-url", url()),
}),
}
);
Deployment targets, output formats, connection protocols—anywhere you have a mode selector that determines what other options are valid.
conditional() exists Optique already has an or() combinator for mutually exclusive alternatives. Why do we need conditional()?
The or() combinator distinguishes branches based on structure—which options are present. It works well for subcommands like git commit vs git push, where the arguments differ completely.
But in the reporter example, the structure is identical: every branch has a --reporter flag. The difference lies in the flag's value, not its presence.
// This won't work as intended
const parser = or(
object({ reporter: option("--reporter", choice(["console"])) }),
object({
reporter: option("--reporter", choice(["junit", "html"])),
outputFile: option("--output-file", string())
}),
);
When you pass --reporter junit, or() tries to pick a branch based on what options are present. Both branches have --reporter, so it can't distinguish them structurally.
conditional() solves this by reading the discriminator's value first, then selecting the appropriate branch. It bridges the gap between structural parsing and value-based decisions.
Instead of parsing options into a loose type and then validating relationships, define a parser whose structure is the constraint.
| Traditional approach | Optique approach |
|---|---|
| Parse → Validate → Use | Parse (with constraints) → Use |
| Types and validation logic maintained separately | Types reflect the constraints |
| Mismatches found at runtime | Mismatches found at compile time |
The parser definition becomes the single source of truth. Add a new reporter type? The parser definition changes, the inferred type changes, and the compiler shows you everywhere that needs updating.
If this resonates with a CLI you're building:
Next time you're about to write an if statement checking option relationships, ask: could the parser express this constraint instead?
The structure of your parser is the constraint. You might not need that validation code at all.
@year_progress@techhub.social
▓▓▓▓▓▓▓▓▓▓▓▓▓▓░ 95%
@S_H_@ruby.social
Mui (無為) v0.2.0 is out 🎉
A Vim-like TUI editor written in Ruby.
Features include:
• Modal editing with familiar Vim motions
• Precise Visual / Visual Line selection
• .muirc and project-level .lmuirc configuration
• RubyGems-based plugin system
• Git, LSP, and filer support
Check it out on GitHub:
https://github.com/S-H-GAMELINKS/mui

@hongminhee@hollo.social · Reply to wwj's post
@z9mb1 Get ready to go to Brussels with me!

@hongminhee@hollo.social
그나저나, FOSDEM 2026까지 앞으로 두 달도 안 남았으니 바로 發表 準備를 해야겠구나…
@krosylight@fosstodon.org
I'm probably not going to buy Steam Machine given I highly doubt it will be as upgradable as normal desktops, but I still hope it serve as a chance for a lot of people to try Linux and see how it's feasible for everyday use.
(If you want to see that today you can just install Bazzite)
@evanprodromou@socialwebfoundation.org
The schedule for the Social Web Developer Room at FOSDEM 2026 is starting to be populated as the speakers confirm their availability. We had a tonne of great submissions for this year’s track, and even with double the time from last year, we still had to leave some great talks on the cutting room floor. But we still managed to fit in 24 great talks, large and small. We’re going to see some additional events happening as FOSDEM 2026 gets nearer. Watch the #SOCIALWEBFOSDEM hashtag for more news and events.

@hongminhee@hollo.social
I'll be presenting @fedify at @fosdem 2026! My talk Fedify: Building ActivityPub servers without the pain was accepted for the Social Web Devroom. See you in Brussels on January 31–February 1!

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
It got accepted! See you all in Brussels. 🎉

@botkit@hollo.social
Big change coming to BotKit: multi-bot support!

Currently, each BotKit instance can only run a single bot. We're redesigning the architecture to let you host multiple bots—both static and dynamically created—on a single instance.
The new API will look like this:
const instance = createInstance({ kv });
const greetBot = instance.createBot("greet", { ... });
const weatherBots = instance.createBot(async (ctx, id) => { ... });Check out the full design:
@jdv_jazz@mastodon.nl
Antônio Carlos Jobim - O Morro Não Tem Vez
#JazzDeVille #Jazz #NowPlaying #AntônioCarlosJobim

@hongminhee@hollo.social · Reply to 洪 民憙 (Hong Minhee) :nonbinary:'s post
ライブラリ作者のみなさん、ロギングどうしてますか?winston? Pino? debug? どれもしっくりこなくて、結局自分で作りました。
@novik_st@ak2.suzu-ya.info