<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
	<channel>
		<title>Concurrency on NicoLabs</title>
		<link>http://blog.hellonico.info/ja/tags/concurrency/</link>
		<description>Recent content in Concurrency 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/concurrency/index.xml" rel="self" type="application/rss+xml" />
			<item>
				<title>Coniでのネイティブ並列ダウンロードの構築</title>
				<link>http://blog.hellonico.info/ja/posts/coni/parallel-downloads/</link>
				<pubDate>Sat, 27 Jun 2026 00:00:00 +0000</pubDate>
				<guid>http://blog.hellonico.info/ja/posts/coni/parallel-downloads/</guid>
				<description>&lt;p&gt;独自の言語やツールを構築する際の素晴らしい点の一つは、操作がどのように実行されるかを再考できることです。最新のコミットでは、ビルドプロセスの大きなボトルネックであるMaven依存関係のダウンロードに取り組むことにしました。&lt;/p&gt;&#xA;&lt;h2 id=&#34;プラットフォーム依存スクリプトの問題点&#34;&gt;プラットフォーム依存スクリプトの問題点&lt;/h2&gt;&#xA;&lt;p&gt;これまで、Coniの&lt;code&gt;download-url-to-file&lt;/code&gt;関数は、プラットフォーム固有のツールを呼び出す（シェルアウトする）ことに依存していました：&lt;/p&gt;&#xA;&lt;ul&gt;&#xA;&lt;li&gt;Linux/macOSでは、&lt;code&gt;curl&lt;/code&gt;プロセスを生成しました。&lt;/li&gt;&#xA;&lt;li&gt;Windowsでは、&lt;code&gt;System.Net.WebClient&lt;/code&gt;を使用する巨大な&lt;code&gt;powershell&lt;/code&gt;コマンドを呼び出しました。&lt;/li&gt;&#xA;&lt;/ul&gt;&#xA;&lt;p&gt;これは機能していましたが、いくつかの欠点がありました。第一に、外部プロセスを呼び出すことは遅く、リソースを消費します。第二に、外部ツールへの依存は、&lt;code&gt;curl&lt;/code&gt;のバージョンやWindowsのセキュリティプロトコルの微妙な違いが予期せぬ失敗を引き起こす可能性があることを意味しました。最も重要なのは、これらのシェルコマンドを使用してアーティファクトを順次（シリアルに）ダウンロードするため、大規模なMavenプロジェクトの解決に非常に時間がかかっていたことです。&lt;/p&gt;&#xA;&lt;h2 id=&#34;ネイティブな解決策&#34;&gt;ネイティブな解決策&lt;/h2&gt;&#xA;&lt;p&gt;コミット&lt;code&gt;9ac76d12&lt;/code&gt;で、Goの評価器（エバリュエーター）内に直接実装されたネイティブの組み込み関数&lt;code&gt;sys-http-download&lt;/code&gt;を導入しました。この新しい組み込み関数は、Goの標準の&lt;code&gt;net/http&lt;/code&gt;クライアントを使用します。&lt;/p&gt;&#xA;&lt;p&gt;評価器にネイティブに組み込まれているため、プロセス作成のオーバーヘッドを完全に回避できます。さらに、Maven Centralからのレート制限（認識可能なUser-Agentなしで速すぎるアクセスをした場合に頻繁に発生する「429 Too Many Requests」エラー）を処理するために、Goの実装に直接堅牢なリトライロジックを追加しました。&lt;/p&gt;&#xA;&lt;h2 id=&#34;ゴルーチンによるスーパーチャージ&#34;&gt;ゴルーチンによるスーパーチャージ&lt;/h2&gt;&#xA;&lt;p&gt;ネイティブの組み込み関数が準備できた後、Coniの並行処理プリミティブを適用したときに真の魔法が起こりました。Coniは&lt;code&gt;go&lt;/code&gt;、&lt;code&gt;chan&lt;/code&gt;、&lt;code&gt;&amp;lt;!&lt;/code&gt;（チャネルからの読み取り）、&lt;code&gt;&amp;gt;!&lt;/code&gt;（チャネルへの書き込み）を使用して、Goスタイルの並行処理をネイティブにサポートしています。&lt;/p&gt;&#xA;&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-clojure&#34; data-lang=&#34;clojure&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;(&lt;span style=&#34;color:#66d9ef&#34;&gt;let &lt;/span&gt;[total-count (count missing)&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;      result-ch (&lt;span style=&#34;color:#a6e22e&#34;&gt;chan&lt;/span&gt; total-count)&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;      task-ch (&lt;span style=&#34;color:#a6e22e&#34;&gt;chan&lt;/span&gt; total-count)]&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  &lt;span style=&#34;color:#75715e&#34;&gt;;; すべてのタスクをタスクチャネルにプッシュ&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;loop &lt;/span&gt;[rem missing]&#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;if &lt;/span&gt;(not (&lt;span style=&#34;color:#a6e22e&#34;&gt;empty?&lt;/span&gt; rem))&#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;do &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:#a6e22e&#34;&gt;&amp;gt;!&lt;/span&gt; task-ch (first rem))&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;        (&lt;span style=&#34;color:#a6e22e&#34;&gt;recur&lt;/span&gt; (rest rem)))))&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  (&lt;span style=&#34;color:#a6e22e&#34;&gt;close!&lt;/span&gt; task-ch)&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&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:#75715e&#34;&gt;;; タスクを並行処理するために8つのワーカーゴルーチンを生成&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;loop &lt;/span&gt;[i &lt;span style=&#34;color:#ae81ff&#34;&gt;0&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;if &lt;/span&gt;(&amp;lt; i &lt;span style=&#34;color:#ae81ff&#34;&gt;8&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:#a6e22e&#34;&gt;do&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:#a6e22e&#34;&gt;go&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;loop &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;let &lt;/span&gt;[item (&lt;span style=&#34;color:#a6e22e&#34;&gt;&amp;lt;!&lt;/span&gt; task-ch)]&#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;if &lt;/span&gt;(not (nil? item))&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;                (&lt;span style=&#34;color:#a6e22e&#34;&gt;do&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:#a6e22e&#34;&gt;&amp;gt;!&lt;/span&gt; result-ch (&lt;span style=&#34;color:#a6e22e&#34;&gt;download-file-single&lt;/span&gt; item repos))&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;                  (&lt;span style=&#34;color:#a6e22e&#34;&gt;recur&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:#a6e22e&#34;&gt;recur&lt;/span&gt; (+ i &lt;span style=&#34;color:#ae81ff&#34;&gt;1&lt;/span&gt;)))))&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&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:#75715e&#34;&gt;;; 結果を待ち、進捗を表示&lt;/span&gt;&#xA;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;  ...)&#xA;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;ネットワークを圧倒したり、攻撃的なレート制限を引き起こしたりするのを防ぐため、ワーカープールの最大値を8ゴルーチンに制限しています。&lt;/p&gt;</description>
			</item>
	</channel>
</rss>
