2023/09/03

Forgejo+WoodpeckerでCI/CD環境を所有する

逆になにができないのか判らないほど多機能と化したGitHubだが、その機能性が仇となって限られた用途で使用するにはいささか煩雑な画面操作が増えたように思う。ghコマンドは確かに有用ではあるものの圧の強いUIと格闘するかCLIかの二択は少々極端なきらいが否めない。

そこで、自前の計算資源を持っている人間にはリモートリポジトリのセルフホスティングが検討される。世の中には絶対にissueやPRを受け付ける気がない雑多なコード片や、ごく個人的、ないしは見知った少人数のみのプロジェクトが存在する。そういった趣旨のリポジトリを管理するには、いっそGitHubよりもミニマルに設計されたサービスの方が快適に用を足せる見込みがある。

そもそもGitとは自由なバージョン管理システムであり、巨大資本のルールに縛られた一つの場所にコードを置くのは十分に機能を活かせていないとも考えられる。もし持て余した計算資源がそこらに転がっているのなら、そこに個人的なリモートリポジトリを生やしてみるとこれまでに得られなかったときめきが感じられるかもしれない。本稿ではForgejoとWoodpeckerを連携させた統合的なCI/CD環境を構築する手法について記す。

ソフトウェアの紹介

■Forgejo
ForgejoはGiteaを基にしたGitホスティングソフトウェアのコミュニティ主導フォークで、商業化に傾倒したGiteaから主要な開発者が離反する形で誕生した。昨年末に分派したばかりで外観以外の目立った変更箇所はさほど見られないが、代わりにGitea向けのサービスや連携機能を利用できる。

近い将来にはセルフホストされた各リモートリポジトリ同士をActivityPubで相互接続してissueやPRを投げられるようにするForgeFed計画があるらしい。もし実現すれば従来の大手ホスティングサービスに縛られない分散的なコミュニティを形成する余地が生まれる。

■Woodpecker
Woodpeckerも同様にDroneを基にしたコミュニティ主導フォークだが、こちらは特に喧嘩別れというわけではない。エンタープライズ向けの有料エディションを持つDroneと違い永久に渡る完全無料を確約している。

CI/CDとは「継続的インテグレーション/継続的デリバリー」の略称で、コードの変更を起点に自動的にテストを行ったり、ビルド結果を反映した最新バージョンを展開したりするための機能群を指す。GitHubの「GitHub Actions」はCI/CDの一つと言える。企業の提供する無料のCI/CD環境には制約が設けられている一方、セルフホストすれば所有する計算資源の限界まで使い倒せる利点がある。

docker-compose.ymlの編集

dockerおよびdocker-composeは導入済みと仮定する。ForgejoとWoodpeckerは別のソフトウェアだが、互いに連携させる前提で構築するため今回は単一のdocker-compose.ymlで両者をまとめる。

 1version: "3"
 2
 3networks:
 4  forgejo:
 5    external: false
 6
 7services:
 8  server:
 9    image: codeberg.org/forgejo/forgejo:1.20
10    container_name: forgejo
11    environment:
12      - USER_UID=1000
13      - USER_GID=1000
14      - FORGEJO__database__DB_TYPE=postgres
15      - FORGEJO__database__HOST=db:5432
16      - FORGEJO__database__NAME=forgejo
17      - FORGEJO__database__USER=forgejo
18      - FORGEJO__database__PASSWD=forgejo
19    restart: always
20    networks:
21      - forgejo
22    volumes:
23      - ./forgejo:/data
24      - /etc/timezone:/etc/timezone:ro
25      - /etc/localtime:/etc/localtime:ro
26    ports:
27      - "3333:3000"
28      - "4444:22"
29    depends_on:
30      - db
31
32  db:
33    image: postgres:14
34    restart: always
35    environment:
36      - POSTGRES_USER=forgejo
37      - POSTGRES_PASSWORD=forgejo
38      - POSTGRES_DB=forgejo
39    networks:
40      - forgejo
41    volumes:
42      - ./postgres:/var/lib/postgresql/data
43
44  woodpecker-server:
45    image: woodpeckerci/woodpecker-server:latest
46    ports:
47      - "8000:8000"
48    volumes:
49      - woodpecker-server-data:/var/lib/woodpecker/
50    environment:
51      - WOODPECKER_OPEN=true
52      - WOODPECKER_HOST=あんたのURL
53      - WOODPECKER_GRPC_SECURE=true
54      - WOODPECKER_GITHUB=false
55      - WOODPECKER_GITEA=true
56      - WOODPECKER_GITEA_URL=あんたのURL
57      - WOODPECKER_GITEA_CLIENT=後述
58      - WOODPECKER_GITEA_SECRET=後述
59      - WOODPECKER_AGENT_SECRET=後述
60    networks:
61      - forgejo
62
63  woodpecker-agent:
64    image: woodpeckerci/woodpecker-agent:latest
65    command: agent
66    restart: always
67    depends_on:
68      - woodpecker-server
69    volumes:
70      - woodpecker-agent-config:/etc/woodpecker
71      - /var/run/docker.sock:/var/run/docker.sock
72    environment:
73      - WOODPECKER_SERVER=woodpecker-server:9000
74      - WOODPECKER_AGENT_SECRET=後述
75    networks:
76      - forgejo
77
78volumes:
79  woodpecker-server-data:
80  woodpecker-agent-config:

portsの箇所はコロンの左側を他の番号に置き換えてもよい。この例では3000番ポートが他のサービスに専有されている都合上、代替として3333番ポートを指定している。セキュリティ保護の観点からForgejoとSSH通信を行うポートも標準の22番を避けている。万が一、woodpecker-serverの8000番ポートも空いていなければ他の番号に置き換える。

Woodpecker側の設定もここで指定する。environmentの記述はブラウザ経由のアクセスやForgejoとの連携を図る上で必須となる。予めDNSの設定でそれぞれ任意のサブドメインを割り振っておくと話が早い。たとえばForgejoの方をgit.あんたのドメイン、Woodpeckerの方をcicd.あんたのドメインなどに割り当てる。定義がWOODPECKER_GITEA_*なのはForgejoを内部的にGiteaと認識させているためだ。

リバースプロキシの設定

nginx向けにリバースプロキシを書く。Forgejo用とWoodpecker用の二つを用意する。Cloudflareを利用していてオリジンサーバ証明書を取得していなければこの記事の冒頭を参考にSSL証明書を設置されたし。一度やれば使い回せるのでぜひおすすめしたい。

 1# Forgejo用
 2server {
 3  listen 80;
 4  listen [::]:80;
 5  server_name git.あんたのドメイン;
 6  return 301 https://$server_name$request_uri;
 7}
 8
 9server {
10  listen 443 ssl http2;
11  listen [::]:443 ssl http2;
12  server_name git.あんたのドメイン;
13  ssl_certificate     /etc/ssl/certs/あんたのドメイン.pem;
14  ssl_certificate_key /etc/ssl/private/あんたのドメイン.key;
15
16  location / {
17    proxy_pass http://localhost:3333;
18    proxy_set_header Host $host;
19    proxy_set_header X-Real-IP $remote_addr;
20  }
21}
 1# Woodpecker用
 2server {
 3  listen 80;
 4  listen [::]:80;
 5  server_name cicd.あんたのドメイン;
 6  return 301 https://$server_name$request_uri;
 7}
 8
 9server {
10  listen 443 ssl http2;
11  listen [::]:443 ssl http2;
12  server_name cicd.あんたのドメイン;
13  ssl_certificate     /etc/ssl/certs/あんたのドメインpem;
14  ssl_certificate_key /etc/ssl/private/あんたのドメイン.key;
15
16  location / {
17     proxy_set_header X-Forwarded-For $remote_addr;
18     proxy_set_header X-Forwarded-Proto $scheme;
19     proxy_set_header Host $http_host;
20
21     proxy_pass http://localhost:8000;
22     proxy_redirect off;
23     proxy_http_version 1.1;
24     proxy_buffering off;
25
26     chunked_transfer_encoding off;
27    }
28}

記述を終えたら保存してnginx -tで文法を検証する。問題がなければsystemctl restart nginxで再起動を行う。docker-compose up -dでコンテナを起動すると、この時点で紐づけたサブドメインからForgejoの初期設定画面にアクセスできるはずである。Woodpeckerの方はForgejoとの連携作業が済むまでは接続できない。

Forgejoの設定

ほとんどの欄がすでに埋まっているので自ら記入する項目は少ない。むしろ下手にいじると初期設定に失敗する。強いて挙げるなら管理者アカウントを事前に作成するか、サイトタイトルなどのブランディングに関わる部分を記す程度と思われる。初期設定を進めると数秒でインストール作業が完了して他のページに遷移する。

初期設定を編集したい場合はforgejo/gitea/conf/app.iniという大層紛らわしいディレクトリの下に生成されるファイルで変更できる。なお、Woodpeckerとの連携に際して以下の追記が求められる。

1[webhook]
2ALLOWED_HOST_LIST = external,loopback

また、一人で使用するなら新規登録機能は予め封じておいた方が無難だ。ただし、初期設定で管理者アカウントを作らず先に登録を無効化するとブラウザ経由でログインする手段がなくなってしまうのでこの設定は後に行う。

1[service]
2DISABLE_REGISTRATION = true
3REQUIRE_SIGNIN_VIEW = false
4REGISTER_EMAIL_CONFIRM = false
5ALLOW_ONLY_EXTERNAL_REGISTRATION = false

一通りの設定が済んだらWebページの右上の+マークから「新しいリポジトリ」を作成してgit clonegit pushの動作を確認する。ちなみに、SSHキーの登録はユーザ設定の「SSH/GPGキー」から行える。docker-compose.ymlportsで指定したポート番号がSSH通信に用いられる。

Woodpeckerの設定

Forgejoのユーザ設定から「アプリケーション」に進んで「OAuth2アプリケーションの管理」でシークレットキーを作成する。アプリケーション名はなんでも差し支えない。対して、リダイレクトURIはhttps://cicd.あんたのドメイン/authorizeの形式で明確に指定しなければならない。

ここで生成されたクライアントIDクライアントシークレットの値がそれぞれdocker-compose.ymlWOODPECKER_GITEA_CLIENTWOODPECKER_GITEA_SECRETに符合する。後者は一度しか表示されない。ついでにWOODPECKER_AGENT_SECRETの欄をopenssl rand -hex 32で乱数を出力して埋める。

docker-compose downで一旦コンテナを終了させてからdocker-compose up -dで再起動して、Woodpecker用に割り当てたサブドメインにアクセスする。認証が要求されるのでForgejoで作成したアカウント情報を用いる。「+Add repository」の項目をクリックしてForgejoのリポジトリを追加できたらひとまず成功と見ていいだろう。

以降は対象のリポジトリ直下に.woodpecker/*.ymlのディレクトリ構造でコンテナファイルを作り、目的に適った内容のパイプラインを実行させる形となる。したがって、ForgejoとWoodpeckerの構築作業は以上で終了である。

実践例

静的サイトジェネレータのデプロイを例にとる。実は当ブログも先月中旬からCloudflare PagesではなくVPS上で稼働しており、Forgejo上のリモートリポジトリにPushした変更をWoodpeckerへ渡すことで同等のデプロイ環境を再現している。実際の記述例は以下の通り。

 1clone:
 2  git:
 3    image: woodpeckerci/plugin-git
 4    settings:
 5      recursive: true
 6
 7steps:
 8  build:
 9    image: klakegg/hugo:ext-alpine
10    commands:
11      - hugo
12  deploy:
13    image: drillster/drone-rsync
14    secrets: [USER, HOTS, SSH, PORT]
15    settings:
16      hosts:
17        from_secret: HOST
18      user:
19        from_secret: USER
20      key:
21        from_secret: SSH
22      port:
23        from_secret: PORT
24      source: public
25      target: /var/www/html

上記の例ではhugoコマンドを実行するコンテナによって吐き出された静的ファイルを、rsyncコマンドでVPS上の/var/www/htmlに同期する方法にてデプロイを行っている。その際に使用されるSSH通信の秘密鍵やポート番号などはコンテナファイルに直接記述すべきではないので、Woodpeckerの機能を用いて秘匿化してある。これは各リポジトリ画面の右上の歯車から「Secrets」で設定できる。

余談だが、Woodpeckerではパイプラインを実行するととてもかわいいアニメーションが見られる。僕はこれを目の当たりにした瞬間に一発でこのソフトウェアが好きになった。機能的には不必要とはいえこういう遊び心ってすごく大切だ。30分くらいは余分に働いてもいい気分になる。

おわりに

かつてGitやCI/CDはローカルかセルフホスト環境で実行するサービスだったが、いつからかGitHubなどの営利企業が豊富な計算資源に物を言わせてずいぶん手軽に使いやすくしてしまった。これらの無料枠は大半のユーザにとってとてつもなく寛大すぎたため、あらゆるプロジェクトがバージョン管理システムとCI/CDの恩恵に与る未曾有の黄金時代が到来した。

しかし、その一方で営利企業とは当然ながら自己利益優先で動く顔のない怪物だ。昨日までの最良の友が寝て起きたら人を食う魔物に変貌していたなどという事態はいつでも起こりうる。利益率を重視するあまり元々は洗練されていた機能が混沌の渦中に堕し、積み重なったベンダーロックインの依存性ゆえ抜け出すにも抜け出せない危険性は常に考慮しなければならない。

かといって、なにも完全に移行する必要はまったくない。GitHubにしろ数多あるCI/CD環境にしろ、それはそれで確かに優れたソーシャルコーディングサービスには違いない。ただ、平時より怠らず代替となりうるソフトウェアを使い慣れておけば、用途に応じて賢く使い分けたり、最悪のハルマゲドンに備えられる第二、第三の選択肢が皆さんの手のうちに残り続けるのである。

©2011 辻谷陸王 | Fediverse | Keyoxide | RSS | 小説