<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: m1s1ma</title>
    <description>The latest articles on DEV Community by m1s1ma (@m1s1ma).</description>
    <link>https://dev.to/m1s1ma</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.us-east-2.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F4002925%2Faf7d5987-ebbb-4673-a8ba-bdda4c59a2c2.jpg</url>
      <title>DEV Community: m1s1ma</title>
      <link>https://dev.to/m1s1ma</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/m1s1ma"/>
    <language>en</language>
    <item>
      <title>Generating and publishing Go gRPC stubs as separate modules via GitLab CI/CD</title>
      <dc:creator>m1s1ma</dc:creator>
      <pubDate>Thu, 25 Jun 2026 19:53:15 +0000</pubDate>
      <link>https://dev.to/m1s1ma/generating-and-publishing-go-grpc-stubs-as-separate-modules-via-gitlab-cicd-48b2</link>
      <guid>https://dev.to/m1s1ma/generating-and-publishing-go-grpc-stubs-as-separate-modules-via-gitlab-cicd-48b2</guid>
      <description>&lt;h1&gt;
  
  
  Setting up gRPC stub generation for Go and connecting them as a module
&lt;/h1&gt;

&lt;p&gt;Keeping proto contracts in a single repository is convenient, but pulling the entire thing into every service is not. Let's walk through how to automatically generate Go stubs from proto files, version them as standalone Go modules, and publish them via GitLab CI/CD. Bonus: Swagger documentation and GitLab Pages.&lt;/p&gt;

&lt;p&gt;Everything described here targets private free-tier GitLab. For self-hosted, paid plans, or public repositories the setup is considerably simpler — a number of the limitations simply don't apply.&lt;/p&gt;




&lt;h2&gt;
  
  
  Notes before you start
&lt;/h2&gt;

&lt;p&gt;The approach requires some dev environment setup: standard SSH access to the repository and a few environment variables on each developer's machine:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="nv"&gt;GOPRIVATE&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;gitlab.com/your-group
&lt;span class="nv"&gt;GOPROXY&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;direct
&lt;span class="nv"&gt;GONOSUMDB&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;gitlab.com/your-group
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Services that consume the stubs will need a vendor directory — both for local runs and in pipelines during linting, testing, and builds. This is because Go can't download private modules from GitLab without additional auth configuration bypassing the proxy.&lt;/p&gt;

&lt;p&gt;In CI environments, access is handled via &lt;code&gt;CI_JOB_TOKEN&lt;/code&gt;. A typical &lt;code&gt;before_script&lt;/code&gt; for services consuming the stubs looks like this:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;default&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;golang:${GO_VERSION}&lt;/span&gt;
  &lt;span class="na"&gt;cache&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;key&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;go-mod&lt;/span&gt;
    &lt;span class="na"&gt;paths&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;go/pkg/mod&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;.cache/go-build&lt;/span&gt;
  &lt;span class="na"&gt;before_script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;go version&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;git config --global url."https://gitlab-ci-token:${CI_JOB_TOKEN}@gitlab.com/".insteadOf "https://gitlab.com/"&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;go env -w GOPRIVATE=gitlab.com/chichilaki&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;go env -w GONOSUMDB=gitlab.com/chichilaki&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;go mod vendor&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;






&lt;h2&gt;
  
  
  Repository structure
&lt;/h2&gt;

&lt;p&gt;Create the following groups and repositories:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;group/proto&lt;/code&gt; — proto contracts; the only repository you'll edit by hand.&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;group/proto-stubs&lt;/code&gt; — a group for generated stubs; create the repositories inside it ahead of time, empty.&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;group/proto-stubs/common&lt;/code&gt;, &lt;code&gt;group/proto-stubs/user&lt;/code&gt;, &lt;code&gt;group/proto-stubs/admin&lt;/code&gt;, etc. — one repository per module.&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;group/service&lt;/code&gt; — any service that consumes the stubs.&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Stub repositories
&lt;/h2&gt;

&lt;p&gt;Each stub gets its own Go module repository. In my case there's a contract for shared entities (&lt;code&gt;common&lt;/code&gt;) and contracts for services: &lt;code&gt;user&lt;/code&gt;, &lt;code&gt;admin&lt;/code&gt;, &lt;code&gt;runner&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Configuring access for CI_JOB_TOKEN&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Since every service uses &lt;code&gt;CI_JOB_TOKEN&lt;/code&gt; in its pipeline to access stub repositories, you need to explicitly allow that access.&lt;/p&gt;

&lt;p&gt;For each stub repository, go to &lt;code&gt;Settings -&amp;gt; CI/CD -&amp;gt; Job token permissions&lt;/code&gt; and add the proto contracts repository and the group containing services that consume the stubs.&lt;/p&gt;

&lt;p&gt;One important nuance: if some stubs depend on others — for example, &lt;code&gt;admin&lt;/code&gt; imports &lt;code&gt;common&lt;/code&gt; — you also need to add the stub repositories group to the access settings of &lt;code&gt;admin.git&lt;/code&gt;. Otherwise &lt;code&gt;go get&lt;/code&gt; in CI will fail with an authentication error, and the reason won't be obvious at all.&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;&lt;strong&gt;Paid and self-hosted GitLab&lt;/strong&gt; lets you grant access at the group level — everything is configured in one place.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;&lt;strong&gt;Why &lt;code&gt;.git&lt;/code&gt; in the module name&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;The module name includes a &lt;code&gt;.git&lt;/code&gt; suffix:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;gitlab.com/chichilaki/mgs/proto-go-stubs/common.git
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This is intentional. When resolving a module path from GitLab, Go can't figure out on its own where the repository path ends and the in-repo package path begins. The &lt;code&gt;.git&lt;/code&gt; suffix gives it an explicit hint. Without it, &lt;code&gt;go get&lt;/code&gt; will try to fetch module metadata from the wrong path and fail.&lt;/p&gt;




&lt;h2&gt;
  
  
  Proto contracts repository
&lt;/h2&gt;

&lt;p&gt;This is the repository that triggers the stub generation and publishing pipeline.&lt;/p&gt;

&lt;h3&gt;
  
  
  Versioning
&lt;/h3&gt;

&lt;p&gt;The publish job runs only for branches named &lt;code&gt;vX.X.X&lt;/code&gt;. On publish, a matching tag is created in the stub repository. This means the branch name in the proto repo directly becomes the Go module version. Want to release &lt;code&gt;v1.2.3&lt;/code&gt; — create a branch called &lt;code&gt;v1.2.3&lt;/code&gt; and run the pipeline.&lt;/p&gt;

&lt;h3&gt;
  
  
  Setting up a Personal Access Token
&lt;/h3&gt;

&lt;blockquote&gt;
&lt;p&gt;&lt;strong&gt;Note:&lt;/strong&gt; GitLab Next recently added the ability to configure &lt;code&gt;CI_JOB_TOKEN&lt;/code&gt; access for pushing to repositories — worth checking, this step may soon become unnecessary.&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;In the free tier of GitLab you can't issue an access token scoped to a group of repositories, so we use a Personal Access Token instead.&lt;/p&gt;

&lt;p&gt;Go to Profile Settings -&amp;gt; Access Tokens and create a token with the following permissions: &lt;code&gt;api&lt;/code&gt;, &lt;code&gt;read_api&lt;/code&gt;, &lt;code&gt;read_repository&lt;/code&gt;, &lt;code&gt;write_repository&lt;/code&gt;. The token is shown only once — save it immediately.&lt;/p&gt;

&lt;p&gt;Then in the proto contracts repository: Settings -&amp;gt; CI/CD -&amp;gt; Variables. Create a variable with:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;type: &lt;code&gt;Variable&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;visibility: &lt;code&gt;Masked and hidden&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;flag: &lt;code&gt;Protected&lt;/code&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;If there are multiple developers on the team — each person creates their own token and their own variable, e.g. &lt;code&gt;CI_PUSH_TOKEN_USERNAME&lt;/code&gt;. In the pipeline, the right token is selected via &lt;code&gt;rules&lt;/code&gt; with a condition on &lt;code&gt;GITLAB_USER_LOGIN&lt;/code&gt; — more on that below.&lt;/p&gt;

&lt;h3&gt;
  
  
  Setting up Branch Rules
&lt;/h3&gt;

&lt;p&gt;To make protected variables available outside the main branch: Settings -&amp;gt; Repository -&amp;gt; Branch Rules -&amp;gt; Add rule. The mask &lt;code&gt;v*&lt;/code&gt; gives any branch starting with &lt;code&gt;v&lt;/code&gt; access to protected variables and tokens. Without this step, the pipeline on branch &lt;code&gt;v1.0.0&lt;/code&gt; simply won't see the token.&lt;/p&gt;




&lt;h2&gt;
  
  
  buf configuration
&lt;/h2&gt;

&lt;p&gt;&lt;code&gt;buf.yaml&lt;/code&gt; describes dependencies and linting rules; &lt;code&gt;buf.gen.yaml&lt;/code&gt; covers the generation plugins.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="s"&gt;// buf.yaml&lt;/span&gt;
&lt;span class="na"&gt;version&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;v2&lt;/span&gt;
&lt;span class="na"&gt;deps&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;buf.build/googleapis/googleapis&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;buf.build/grpc-ecosystem/grpc-gateway&lt;/span&gt;
&lt;span class="na"&gt;lint&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;use&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;STANDARD&lt;/span&gt;
&lt;span class="na"&gt;breaking&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;use&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;FILE&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;





&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="s"&gt;// buf.gen.yaml&lt;/span&gt;
&lt;span class="na"&gt;version&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;v2&lt;/span&gt;
&lt;span class="na"&gt;plugins&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;local&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="pi"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;go"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;run"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;google.golang.org/protobuf/cmd/protoc-gen-go@v1.36.11"&lt;/span&gt;&lt;span class="pi"&gt;]&lt;/span&gt;
    &lt;span class="na"&gt;out&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;gen/go&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;local&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="pi"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;go"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;run"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;google.golang.org/grpc/cmd/protoc-gen-go-grpc@v1.6.1"&lt;/span&gt;&lt;span class="pi"&gt;]&lt;/span&gt;
    &lt;span class="na"&gt;out&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;gen/go&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;local&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="pi"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;go"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;run"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-grpc-gateway@v2.29.0"&lt;/span&gt;&lt;span class="pi"&gt;]&lt;/span&gt;
    &lt;span class="na"&gt;out&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;gen/go&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;local&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="pi"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;go"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;run"&lt;/span&gt;&lt;span class="pi"&gt;,&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;github.com/grpc-ecosystem/grpc-gateway/v2/protoc-gen-openapiv2@v2.29.0"&lt;/span&gt;&lt;span class="pi"&gt;]&lt;/span&gt;
    &lt;span class="na"&gt;out&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;gen/openapiv2&lt;/span&gt;
    &lt;span class="na"&gt;opt&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;output_format=json&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;allow_merge=false&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;In &lt;code&gt;buf.gen.yaml&lt;/code&gt; we use &lt;code&gt;local&lt;/code&gt; with &lt;code&gt;go run&lt;/code&gt; instead of a globally installed &lt;code&gt;protoc&lt;/code&gt; and plugins — Go will download the required versions on first run. This eliminates version mismatch issues between developer machines and CI.&lt;/p&gt;

&lt;p&gt;Both files are included in full in the appendix.&lt;/p&gt;

&lt;h3&gt;
  
  
  Proto file examples
&lt;/h3&gt;

&lt;p&gt;The &lt;code&gt;go_package&lt;/code&gt; option must include the &lt;code&gt;.git&lt;/code&gt; suffix — that's how Go will resolve the module import:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight protobuf"&gt;&lt;code&gt;&lt;span class="c1"&gt;// common.proto&lt;/span&gt;
&lt;span class="na"&gt;syntax&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"proto3"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="kn"&gt;package&lt;/span&gt; &lt;span class="nn"&gt;common&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;v1&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;option&lt;/span&gt; &lt;span class="na"&gt;go_package&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"gitlab.com/chichilaki/mgs/proto-go-stubs/common.git/v1"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;For service contracts with grpc-gateway HTTP annotations and Swagger options, the structure looks like this:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight protobuf"&gt;&lt;code&gt;&lt;span class="c1"&gt;// admin.proto&lt;/span&gt;
&lt;span class="na"&gt;syntax&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"proto3"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="kn"&gt;package&lt;/span&gt; &lt;span class="nn"&gt;admin&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;v1&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;option&lt;/span&gt; &lt;span class="na"&gt;go_package&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="s"&gt;"gitlab.com/chichilaki/mgs/proto-go-stubs/admin.git/v1"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="s"&gt;"common/v1/common.proto"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="s"&gt;"google/api/annotations.proto"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="s"&gt;"protoc-gen-openapiv2/options/annotations.proto"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="k"&gt;option&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;grpc.gateway.protoc_gen_openapiv2.options.openapiv2_swagger&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="n"&gt;info&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="n"&gt;title&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"Control Plane ADMIN API"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt; &lt;span class="n"&gt;version&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"1.0"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
  &lt;span class="n"&gt;security_definitions&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="n"&gt;security&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
      &lt;span class="n"&gt;key&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"Bearer"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
      &lt;span class="n"&gt;value&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
        &lt;span class="n"&gt;type&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="n"&gt;TYPE_API_KEY&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
        &lt;span class="n"&gt;in&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="n"&gt;IN_HEADER&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
        &lt;span class="n"&gt;name&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"Authorization"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
        &lt;span class="n"&gt;description&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"Admin Bearer token"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;
      &lt;span class="p"&gt;};&lt;/span&gt;
    &lt;span class="p"&gt;};&lt;/span&gt;
  &lt;span class="p"&gt;};&lt;/span&gt;
  &lt;span class="n"&gt;security&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="n"&gt;security_requirement&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="n"&gt;key&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"Bearer"&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt; &lt;span class="p"&gt;}&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
&lt;span class="p"&gt;};&lt;/span&gt;

&lt;span class="kd"&gt;service&lt;/span&gt; &lt;span class="n"&gt;AdminService&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="k"&gt;rpc&lt;/span&gt; &lt;span class="n"&gt;ListRunners&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;ListRunnersRequest&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="k"&gt;returns&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;ListRunnersResponse&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;option&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;google.api.http&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="n"&gt;get&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"/v1/runners"&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="k"&gt;rpc&lt;/span&gt; &lt;span class="n"&gt;GetRunnerStatus&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;GetRunnerStatusRequest&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="k"&gt;returns&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;GetRunnerStatusResponse&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="k"&gt;option&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;google.api.http&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="n"&gt;get&lt;/span&gt;&lt;span class="o"&gt;:&lt;/span&gt; &lt;span class="s"&gt;"/v1/runners/{runner_id.runner_id}/status"&lt;/span&gt; &lt;span class="p"&gt;};&lt;/span&gt;
  &lt;span class="p"&gt;}&lt;/span&gt;
  &lt;span class="c1"&gt;// ...&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;






&lt;h2&gt;
  
  
  CI/CD pipeline
&lt;/h2&gt;

&lt;p&gt;The proto repository pipeline has three stages: &lt;code&gt;lint&lt;/code&gt;, &lt;code&gt;generate&lt;/code&gt;, &lt;code&gt;commit&lt;/code&gt;. The full version is in the appendix.&lt;/p&gt;

&lt;h3&gt;
  
  
  The generate job
&lt;/h3&gt;

&lt;p&gt;Runs &lt;code&gt;buf generate&lt;/code&gt;, then creates a &lt;code&gt;go.mod&lt;/code&gt; with the correct module name (including &lt;code&gt;.git&lt;/code&gt;) for each module. Artifacts are passed to the next job via &lt;code&gt;artifacts.paths&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Key detail: &lt;code&gt;go mod init&lt;/code&gt; is only called if &lt;code&gt;go.mod&lt;/code&gt; doesn't already exist, to avoid overwriting the file on re-runs. Otherwise the module would be re-initialized on every run and lose its dependencies.&lt;/p&gt;

&lt;h3&gt;
  
  
  The commit-generated job
&lt;/h3&gt;

&lt;p&gt;Clones each target repository, replaces its contents with the generated files, copies &lt;code&gt;swagger.json&lt;/code&gt;, updates dependencies via &lt;code&gt;go mod tidy&lt;/code&gt;, commits, and creates a version tag.&lt;/p&gt;

&lt;p&gt;Token selection per user is handled via &lt;code&gt;rules&lt;/code&gt; with a condition on &lt;code&gt;GITLAB_USER_LOGIN&lt;/code&gt; — this lets each developer use their own Personal Access Token without sharing it with teammates:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH =~ /^v\d+\.\d+\.\d+$/&lt;/span&gt;
    &lt;span class="na"&gt;variables&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="na"&gt;PROTO_VERSION&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH&lt;/span&gt;
    &lt;span class="na"&gt;when&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;never&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$GITLAB_USER_LOGIN == "End1essRage"&lt;/span&gt;
    &lt;span class="na"&gt;variables&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="na"&gt;PUSH_TOKEN&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_PUSH_TOKEN&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$GITLAB_USER_LOGIN == "stivvhuys"&lt;/span&gt;
    &lt;span class="na"&gt;variables&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="na"&gt;PUSH_TOKEN&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_PUSH_TOKEN_GENGAVSOV&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;when&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;never&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;If &lt;code&gt;PUSH_TOKEN&lt;/code&gt; ends up empty — the job fails in &lt;code&gt;before_script&lt;/code&gt; with an explicit error rather than a cryptic &lt;code&gt;403&lt;/code&gt; somewhere mid-script. This is an intentional check.&lt;/p&gt;

&lt;p&gt;For modules that depend on &lt;code&gt;common&lt;/code&gt;, the script explicitly adds a &lt;code&gt;require&lt;/code&gt; with the correct version:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="k"&gt;if&lt;/span&gt; &lt;span class="o"&gt;[&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="nv"&gt;$module&lt;/span&gt;&lt;span class="s2"&gt;"&lt;/span&gt; &lt;span class="o"&gt;!=&lt;/span&gt; &lt;span class="s2"&gt;"common"&lt;/span&gt; &lt;span class="o"&gt;]&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt; &lt;span class="k"&gt;then
  &lt;/span&gt;go get gitlab.com/chichilaki/mgs/proto-go-stubs/common.git@&lt;span class="nv"&gt;$PROTO_VERSION&lt;/span&gt;
&lt;span class="k"&gt;fi&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This matters: without an explicit &lt;code&gt;go get&lt;/code&gt; with the tag, &lt;code&gt;go mod tidy&lt;/code&gt; will pull in &lt;code&gt;latest&lt;/code&gt; instead of the version we just published.&lt;/p&gt;




&lt;h2&gt;
  
  
  The service consuming stubs
&lt;/h2&gt;

&lt;p&gt;Let's look at the consumer side — a service that wants to host its own Swagger documentation, serve it over HTTP, and interact with the API through Swagger UI.&lt;/p&gt;

&lt;h3&gt;
  
  
  Embedding swagger.json via go:embed
&lt;/h3&gt;

&lt;p&gt;A &lt;code&gt;docs&lt;/code&gt; package is created at the project root:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight go"&gt;&lt;code&gt;&lt;span class="k"&gt;package&lt;/span&gt; &lt;span class="n"&gt;docs&lt;/span&gt;

&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="n"&gt;_&lt;/span&gt; &lt;span class="s"&gt;"embed"&lt;/span&gt;

&lt;span class="c"&gt;//go:embed control-plane-admin.swagger.json&lt;/span&gt;
&lt;span class="k"&gt;var&lt;/span&gt; &lt;span class="n"&gt;AdminSwaggerJSON&lt;/span&gt; &lt;span class="p"&gt;[]&lt;/span&gt;&lt;span class="kt"&gt;byte&lt;/span&gt;

&lt;span class="c"&gt;//go:embed control-plane-user.swagger.json&lt;/span&gt;
&lt;span class="k"&gt;var&lt;/span&gt; &lt;span class="n"&gt;UserSwaggerJSON&lt;/span&gt; &lt;span class="p"&gt;[]&lt;/span&gt;&lt;span class="kt"&gt;byte&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Imported in &lt;code&gt;main.go&lt;/code&gt; as a side-effect:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight go"&gt;&lt;code&gt;&lt;span class="k"&gt;import&lt;/span&gt; &lt;span class="n"&gt;_&lt;/span&gt; &lt;span class="s"&gt;"gitlab.com/chichilaki/mgs/control-plane/docs"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;





&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight go"&gt;&lt;code&gt;&lt;span class="n"&gt;mux&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;HandleFunc&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"/swagger/admin/doc.json"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="k"&gt;func&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;w&lt;/span&gt; &lt;span class="n"&gt;http&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;ResponseWriter&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;r&lt;/span&gt; &lt;span class="o"&gt;*&lt;/span&gt;&lt;span class="n"&gt;http&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Request&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="n"&gt;w&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Header&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Set&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"Content-Type"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="s"&gt;"application/json"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
    &lt;span class="n"&gt;_&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;_&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="n"&gt;w&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Write&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;docs&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;AdminSwaggerJSON&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;

&lt;span class="n"&gt;mux&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;HandleFunc&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"/swagger/user/doc.json"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="k"&gt;func&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;w&lt;/span&gt; &lt;span class="n"&gt;http&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;ResponseWriter&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;r&lt;/span&gt; &lt;span class="o"&gt;*&lt;/span&gt;&lt;span class="n"&gt;http&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Request&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
    &lt;span class="n"&gt;w&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Header&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Set&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s"&gt;"Content-Type"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="s"&gt;"application/json"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
    &lt;span class="n"&gt;_&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="n"&gt;_&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="n"&gt;w&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;Write&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;docs&lt;/span&gt;&lt;span class="o"&gt;.&lt;/span&gt;&lt;span class="n"&gt;UserSwaggerJSON&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="p"&gt;})&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;code&gt;go:embed&lt;/code&gt; compiles the files directly into the binary — no separate static file serving needed, and no risk of the documentation going out of sync with the code.&lt;/p&gt;

&lt;h3&gt;
  
  
  Syncing swagger.json
&lt;/h3&gt;

&lt;p&gt;Documentation is synced in three places: a pre-commit hook, a Taskfile, and the service CI pipeline. The logic is the same in all three: clone the stub repository into a temp directory, grab &lt;code&gt;v1/&amp;lt;service&amp;gt;.swagger.json&lt;/code&gt;, and put it in &lt;code&gt;docs/&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Locally we use SSH; in CI — &lt;code&gt;CI_JOB_TOKEN&lt;/code&gt;. The corresponding Taskfile and CI configs are in the appendix.&lt;/p&gt;

&lt;h3&gt;
  
  
  Local stack: Traefik + Swagger UI
&lt;/h3&gt;

&lt;p&gt;Brought up via Compose. Note: the &lt;code&gt;BASE_URL&lt;/code&gt; in the Swagger UI config and the &lt;code&gt;PathPrefix&lt;/code&gt; in the Traefik rule must match — otherwise the UI will load but won't be able to resolve its own assets.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;swagger-ui&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;swaggerapi/swagger-ui:v5.9.0&lt;/span&gt;
  &lt;span class="na"&gt;environment&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;BASE_URL&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;/swagger&lt;/span&gt;
    &lt;span class="na"&gt;URLS&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="pi"&gt;&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;[&lt;/span&gt;
        &lt;span class="s"&gt;{ "url": "/cp-swagger/swagger/admin/doc.json", "name": "ADMIN control-plane" },&lt;/span&gt;
        &lt;span class="s"&gt;{ "url": "/cp-swagger/swagger/user/doc.json", "name": "USER control-plane" }&lt;/span&gt;
      &lt;span class="s"&gt;]&lt;/span&gt;
  &lt;span class="na"&gt;labels&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.swagger.rule=Host(`localhost`)&lt;/span&gt;&lt;span class="nv"&gt; &lt;/span&gt;&lt;span class="s"&gt;&amp;amp;&amp;amp;&lt;/span&gt;&lt;span class="nv"&gt; &lt;/span&gt;&lt;span class="s"&gt;PathPrefix(`/swagger`)"&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.swagger.entrypoints=web"&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.swagger.loadbalancer.server.port=8080"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Service route with &lt;code&gt;/cp&lt;/code&gt; prefix stripping:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;control-plane&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;labels&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
        &lt;span class="c1"&gt;# ===== API =====&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-api.rule=Host(`localhost`)&lt;/span&gt;&lt;span class="nv"&gt; &lt;/span&gt;&lt;span class="s"&gt;&amp;amp;&amp;amp;&lt;/span&gt;&lt;span class="nv"&gt; &lt;/span&gt;&lt;span class="s"&gt;PathPrefix(`/cp`)"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-api.entrypoints=web"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-api.service=cp"&lt;/span&gt;

      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.middlewares.cp-strip.stripprefix.prefixes=/cp"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-api.middlewares=cp-strip"&lt;/span&gt;

      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.cp.loadbalancer.server.port=9100"&lt;/span&gt;
      &lt;span class="c1"&gt;# Enable health check for the load balancer&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.cp.loadbalancer.healthcheck.path=/health"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.cp.loadbalancer.healthcheck.interval=10s"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.cp.loadbalancer.healthcheck.timeout=3s"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.cp.loadbalancer.healthcheck.scheme=http"&lt;/span&gt;
      &lt;span class="c1"&gt;# Swagger proxy via Traefik&lt;/span&gt;
      &lt;span class="c1"&gt;# ===== SWAGGER =====&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-swagger.rule=Host(`localhost`)&lt;/span&gt;&lt;span class="nv"&gt; &lt;/span&gt;&lt;span class="s"&gt;&amp;amp;&amp;amp;&lt;/span&gt;&lt;span class="nv"&gt; &lt;/span&gt;&lt;span class="s"&gt;PathPrefix(`/cp-swagger`)"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-swagger.entrypoints=web"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-swagger.service=cp-swagger"&lt;/span&gt;

      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.middlewares.cp-swagger-strip.stripprefix.prefixes=/cp-swagger"&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.routers.cp-swagger.middlewares=cp-swagger-strip"&lt;/span&gt;

      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s2"&gt;"&lt;/span&gt;&lt;span class="s"&gt;traefik.http.services.cp-swagger.loadbalancer.server.port=9100"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;






&lt;h2&gt;
  
  
  Bonus: GitLab Pages with Swagger UI
&lt;/h2&gt;

&lt;p&gt;It's handy to have documentation available not just locally but at a permanent URL. GitLab Pages solves this without any separate hosting — just put the artifacts in a &lt;code&gt;public&lt;/code&gt; folder.&lt;/p&gt;

&lt;p&gt;The &lt;code&gt;pages&lt;/code&gt; job creates the &lt;code&gt;public&lt;/code&gt; folder, places the swagger files there, and generates an &lt;code&gt;index.html&lt;/code&gt; with a CDN-hosted Swagger UI. It runs on &lt;code&gt;master&lt;/code&gt; and &lt;code&gt;dev&lt;/code&gt; branches. The full job is in the appendix.&lt;/p&gt;

&lt;p&gt;CI: fetch-swagger + pages&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;fetch-swagger&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;prepare&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;alpine:latest&lt;/span&gt;
  &lt;span class="na"&gt;before_script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;apk add --no-cache git&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;mkdir -p docs&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="pi"&gt;|&lt;/span&gt;
      &lt;span class="s"&gt;for REPO in admin user; do&lt;/span&gt;
        &lt;span class="s"&gt;git clone --depth 1 \&lt;/span&gt;
          &lt;span class="s"&gt;"https://gitlab-ci-token:${CI_JOB_TOKEN}@gitlab.com/chichilaki/mgs/proto-go-stubs/${REPO}.git" \&lt;/span&gt;
          &lt;span class="s"&gt;"/tmp/${REPO}"&lt;/span&gt;
        &lt;span class="s"&gt;cp "/tmp/${REPO}/v1/${REPO}.swagger.json" "docs/control-plane-${REPO}.swagger.json"&lt;/span&gt;
        &lt;span class="s"&gt;rm -rf "/tmp/${REPO}"&lt;/span&gt;
      &lt;span class="s"&gt;done&lt;/span&gt;
  &lt;span class="na"&gt;artifacts&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;paths&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docs/*.swagger.json&lt;/span&gt;

&lt;span class="na"&gt;pages&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;pages&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;alpine:latest&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;mkdir -p public/docs&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;cp docs/*.swagger.json public/docs/ || echo "No swagger files"&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="pi"&gt;|&lt;/span&gt;
      &lt;span class="s"&gt;cat &amp;gt; public/index.html &amp;lt;&amp;lt; 'EOF'&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;!DOCTYPE html&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;html&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;head&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;title&amp;gt;API Documentation&amp;lt;/title&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;meta charset="UTF-8"&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/swagger-ui-dist@5/swagger-ui.css"&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;/head&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;body&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;div id="swagger-ui"&amp;gt;&amp;lt;/div&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;script src="https://cdn.jsdelivr.net/npm/swagger-ui-dist@5/swagger-ui-bundle.js"&amp;gt;&amp;lt;/script&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;script src="https://cdn.jsdelivr.net/npm/swagger-ui-dist@5/swagger-ui-standalone-preset.js"&amp;gt;&amp;lt;/script&amp;gt;&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;script&amp;gt;&lt;/span&gt;
          &lt;span class="s"&gt;window.onload = function() {&lt;/span&gt;
            &lt;span class="s"&gt;window.ui = SwaggerUIBundle({&lt;/span&gt;
              &lt;span class="s"&gt;urls: [&lt;/span&gt;
                &lt;span class="s"&gt;{ url: "./docs/control-plane-admin.swagger.json", name: "Admin API" },&lt;/span&gt;
                &lt;span class="s"&gt;{ url: "./docs/control-plane-user.swagger.json", name: "User API" }&lt;/span&gt;
              &lt;span class="s"&gt;],&lt;/span&gt;
              &lt;span class="s"&gt;dom_id: '#swagger-ui',&lt;/span&gt;
              &lt;span class="s"&gt;deepLinking: true,&lt;/span&gt;
              &lt;span class="s"&gt;presets: [SwaggerUIBundle.presets.apis, SwaggerUIStandalonePreset],&lt;/span&gt;
              &lt;span class="s"&gt;layout: "StandaloneLayout"&lt;/span&gt;
            &lt;span class="s"&gt;});&lt;/span&gt;
          &lt;span class="s"&gt;}&lt;/span&gt;
        &lt;span class="s"&gt;&amp;lt;/script&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;/body&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;&amp;lt;/html&amp;gt;&lt;/span&gt;
      &lt;span class="s"&gt;EOF&lt;/span&gt;
  &lt;span class="na"&gt;artifacts&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;paths&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;public&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "master"&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "dev"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;






&lt;h2&gt;
  
  
  Conclusion
&lt;/h2&gt;

&lt;p&gt;Here's what we end up with: proto contracts in a single repository, automatic stub generation and publishing triggered by creating a branch like &lt;code&gt;v1.0.0&lt;/code&gt;, separate Go modules per service, Swagger documentation as part of the same pipeline, and Pages with a UI for browsing it.&lt;/p&gt;

&lt;p&gt;The main challenge with free-tier GitLab is token management. On a paid plan or self-hosted, most of that overhead disappears: you can grant access at the group level and use a single token. But even as described, everything works reliably.&lt;/p&gt;

</description>
      <category>go</category>
      <category>devops</category>
      <category>cicd</category>
      <category>grpc</category>
    </item>
  </channel>
</rss>
