<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
	<channel>
		<title>Cli on NicoLabs</title>
		<link>http://blog.hellonico.info/ja/tags/cli/</link>
		<description>Recent content in Cli on NicoLabs</description>
		<generator>Hugo</generator>
		<language>ja</language>
		
		
		
		
			<lastBuildDate>Sat, 27 Jun 2026 00:00:00 +0000</lastBuildDate>
		
			<atom:link href="http://blog.hellonico.info/ja/tags/cli/index.xml" rel="self" type="application/rss+xml" />
			<item>
				<title>組み込みサブコマンドによるConi CLIのスーパーチャージ</title>
				<link>http://blog.hellonico.info/ja/posts/coni/embedded-subcommands/</link>
				<pubDate>Sat, 27 Jun 2026 00:00:00 +0000</pubDate>
				<guid>http://blog.hellonico.info/ja/posts/coni/embedded-subcommands/</guid>
				<description>&lt;p&gt;Coni言語の最大の強みの一つは、そのポータビリティ（移植性）です。私たちは、コア・インタープリタと標準ライブラリが単一の静的バイナリとして配布されるように設計しました。肥大化したインストールプロセスは必要ありません。実行可能ファイルをダウンロードするだけで、すぐに使い始めることができます。&lt;/p&gt;&#xA;&lt;p&gt;しかし、エコシステムが成長するにつれて（Androidビルドパイプラインの追加など）、ワークフローに少し煩わしさを感じるようになりました。AndroidのAPKビルダーを呼び出すには、次のように絶対パスでスクリプトを実行する必要がありました：&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-bash&#34; data-lang=&#34;bash&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;coni libs/android/bin/build-apk.coni ./my-app&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;機能はしますが、ネイティブのツールのように感じられませんでした。開発者が次のようにシンプルに実行できるような、洗練された開発者体験を提供したいと考えました：&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-bash&#34; data-lang=&#34;bash&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;coni android build-apk ./my-app&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;本日、Goの&lt;code&gt;embed&lt;/code&gt;ファイルシステムを活用した**動的サブコマンドルーティング（Dynamic Subcommand Routing）**を実装することで、このギャップを埋めたことを発表できることを嬉しく思います！&lt;/p&gt;&#xA;&lt;h2 id=&#34;仕組み&#34;&gt;仕組み&lt;/h2&gt;&#xA;&lt;p&gt;Goコンパイラに3つの重要なアーキテクチャ上の変更を加えました：&lt;/p&gt;&#xA;&lt;h3 id=&#34;1-バイナリへの組み込みembedding&#34;&gt;1. バイナリへの組み込み（Embedding）&lt;/h3&gt;&#xA;&lt;p&gt;Goコンパイラ内の&lt;code&gt;//go:embed&lt;/code&gt;ディレクティブを拡張し、すべてのモジュールにわたるすべての&lt;code&gt;bin/&lt;/code&gt;ディレクトリを自動的にパッケージ化するようにしました：&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-go&#34; data-lang=&#34;go&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;//go:embed libs/*/src libs/*/bin&lt;/span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;var&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;embeddedLibs&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;embed&lt;/span&gt;.&lt;span style=&#34;color:#a6e22e&#34;&gt;FS&lt;/span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;このシンプルな変更により、（&lt;code&gt;build-apk.coni&lt;/code&gt;のような）ヘルパースクリプトが実行可能ファイルに永続的に焼き付けられることが保証されます。&lt;/p&gt;&#xA;&lt;h3 id=&#34;2-サブコマンドのインターセプター&#34;&gt;2. サブコマンドのインターセプター&lt;/h3&gt;&#xA;&lt;p&gt;CLI引数の解析ロジックに賢いインターセプターを導入しました。これにより、&lt;code&gt;coni android build-apk&lt;/code&gt;のような認識できないコマンドを実行した場合、パーサーがそれをインターセプトし、動的にパスクエリ（&lt;code&gt;libs/android/bin/build-apk.coni&lt;/code&gt;）を構築します。&lt;/p&gt;&#xA;&lt;p&gt;組み込まれたファイルシステムを確認し、スクリプトが存在する場合、実行を組み込まれたペイロードに直接動的にルーティングします！&lt;/p&gt;&#xA;&lt;h3 id=&#34;3-引数のマスキングmasking&#34;&gt;3. 引数のマスキング（Masking）&lt;/h3&gt;&#xA;&lt;p&gt;最後のパズルのピースは、スクリプト自身に対する「錯覚」を維持することでした。&lt;code&gt;build-apk.coni&lt;/code&gt;スクリプトは、フラグ（&lt;code&gt;-p camera&lt;/code&gt;など）を読み取るために&lt;code&gt;cli/parse-opts&lt;/code&gt;を使用します。引数を盲目的に評価器（エバリュエーター）に渡すと、先頭の&lt;code&gt;android&lt;/code&gt;や&lt;code&gt;build-apk&lt;/code&gt;といったトークンによってスクリプトが混乱してしまいます。&lt;/p&gt;&#xA;&lt;p&gt;これを解決するために、Goのルーターは舞台裏でグローバルな&lt;code&gt;os.Args&lt;/code&gt;スライスを書き換えます：&lt;/p&gt;&#xA;&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-go&#34; data-lang=&#34;go&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#a6e22e&#34;&gt;os&lt;/span&gt;.&lt;span style=&#34;color:#a6e22e&#34;&gt;Args&lt;/span&gt; = append([]&lt;span style=&#34;color:#66d9ef&#34;&gt;string&lt;/span&gt;{&lt;span style=&#34;color:#a6e22e&#34;&gt;os&lt;/span&gt;.&lt;span style=&#34;color:#a6e22e&#34;&gt;Args&lt;/span&gt;[&lt;span style=&#34;color:#ae81ff&#34;&gt;0&lt;/span&gt;], &lt;span style=&#34;color:#a6e22e&#34;&gt;embeddedPath&lt;/span&gt;}, &lt;span style=&#34;color:#a6e22e&#34;&gt;os&lt;/span&gt;.&lt;span style=&#34;color:#a6e22e&#34;&gt;Args&lt;/span&gt;[&lt;span style=&#34;color:#ae81ff&#34;&gt;3&lt;/span&gt;:]&lt;span style=&#34;color:#f92672&#34;&gt;...&lt;/span&gt;)&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;スクリプト側から見れば、直接呼び出されたと思い込むわけです！&lt;/p&gt;&#xA;&lt;h2 id=&#34;結果&#34;&gt;結果&lt;/h2&gt;&#xA;&lt;p&gt;その結果、超高速でシームレスなCLI体験が実現しました。すべてのツールとヘルパースクリプトはConiバイナリ内に有機的に組み込まれて出荷され、自然でネイティブ感のあるサブコマンドを通じて瞬時にアクセスできます。&lt;/p&gt;&#xA;&lt;p&gt;追加のダウンロードも、パスの設定も不要です。ただ、純粋で邪魔のない生産性があるのみです！&lt;/p&gt;</description>
			</item>
	</channel>
</rss>
