Amazon Q CLI を使い会社診断アプリを Vibe Coding で作成してみた
コーヒーが好きな emi です。最近はカフェインを控えています。
クラスメソッドでは AI 駆動開発を推進するための取り組みを行っており、クラスメソッド社員も実際に日常的に AI を使って開発を行っています。
今回、弊社メンバーが日常的に実践する AI 駆動開発のナレッジや Tips を共有するために、AI駆動開発 Advent Calendar 2025 を開催しています。
本ブログは、本企画の 7 日目の記事になります。
もし AI 駆動開発の最先端を知りたい方は、この 1 ヶ月間、ぜひ本アドベントカレンダーをチェックしてみてください。
はじめに
2025/10/11(土) に JAWS FESTA 2025 in 金沢が開催されていました。我らがクラスメソッドは Gold スポンサーとして参加し企業ブースを設置していたのですが、ブースアクティビティとして「クラスメソッドグループお似合い会社診断テスト」なるものを公開していました。
この「クラスメソッドグループお似合い会社診断テスト」は、Amazon Q CLI を使った Vibe Coding で実装しました。どのように作成したか本ブログで紹介します。
※「Vibe Coding」とは、詳細な仕様書を書かずに、ざっくりとした要件や雰囲気(Vibe)を AI に伝えるだけで実装を進めてもらう開発スタイルです。
できたもの
※本アプリは学習目的で作成したシンプルな Web アプリケーションです。
以下リンクをクリックいただくと、「クラスメソッドグループお似合い会社診断テスト」を始められます。※あと 1 週間くらい公開したのち削除します。
全 15 問の質問に答えると、あなたにぴったりのクラスメソッドグループの会社(クラスメソッド、アノテーション、CM テクノロジーズ、CM キャリア、CM パートナーズ)を診断してくれる Web アプリです。
環境
- OS: Ubuntu 22.04 (WSL2)
- Amazon Q CLI: 1.13.1
- ブラウザ: Chrome
私は以下ブログの手順で、Windows 11 にインストールした WSL2 を VSCode で使用しています。
Amazon Q CLI のインストール
VSCode で WSL2 に接続し、以下のように Amazon Q CLI をインストールしていきます。
私は WSL2 の Ubuntu を利用しているので、Ubuntu のインストール手順を参考に進めていきます。
Ubuntu にコマンドライン用の Amazon Q をダウンロードします。
user@host:~/work$ wget https://desktop-release.q.us-east-1.amazonaws.com/latest/amazon-q.deb
--2025-08-07 12:04:21-- https://desktop-release.q.us-east-1.amazonaws.com/latest/amazon-q.deb
Resolving desktop-release.q.us-east-1.amazonaws.com (desktop-release.q.us-east-1.amazonaws.com)... 3.173.197.89, 3.173.197.91, 3.173.197.24, ...
Connecting to desktop-release.q.us-east-1.amazonaws.com (desktop-release.q.us-east-1.amazonaws.com)|3.173.197.89|:443... connected.
HTTP request sent, awaiting response... 200 OK
Length: 235503068 (225M) [binary/octet-stream]
Saving to: ‘amazon-q.deb’
amazon-q.deb 100%[================================================================>] 224.59M 47.9MB/s in 4.5s
2025-08-07 12:04:25 (49.4 MB/s) - ‘amazon-q.deb’ saved [235503068/235503068]
user@host:~/work$
パッケージリスト(一覧)を最新化します。
user@host:~/work$ sudo apt-get update
[sudo] password for user:
Hit:1 https://deb.nodesource.com/node_22.x nodistro InRelease
Get:2 http://security.ubuntu.com/ubuntu jammy-security InRelease [129 kB]
Hit:3 http://archive.ubuntu.com/ubuntu jammy InRelease
Get:4 http://archive.ubuntu.com/ubuntu jammy-updates InRelease [128 kB]
Get:5 http://archive.ubuntu.com/ubuntu jammy-backports InRelease [127 kB]
Fetched 384 kB in 2s (186 kB/s)
Reading package lists... Done
user@host:~/work$
パッケージをインストールします。
user@host:~/work$ sudo dpkg -i amazon-q.deb
Selecting previously unselected package amazon-q.
(Reading database ... 57397 files and directories currently installed.)
Preparing to unpack amazon-q.deb ...
Unpacking amazon-q (1.13.1) ...
dpkg: dependency problems prevent configuration of amazon-q:
amazon-q depends on libayatana-appindicator3-1; however:
Package libayatana-appindicator3-1 is not installed.
amazon-q depends on libwebkit2gtk-4.1-0; however:
Package libwebkit2gtk-4.1-0 is not installed.
dpkg: error processing package amazon-q (--install):
dependency problems - leaving unconfigured
Processing triggers for hicolor-icon-theme (0.17-2) ...
Errors were encountered while processing:
amazon-q
user@host:~/work$
不足している依存パッケージを自動インストールします。
user@host:~/work$ sudo apt-get install -f
Reading package lists... Done
Building dependency tree... Done
Reading state information... Done
:
:
aspell-autobuildhash: processing: en [en_US-w_accents-only].
aspell-autobuildhash: processing: en [en_US-wo_accents-only].
user@host:~/work$
Amazon Q にログインします。今回は Builder ID を使って無償版の Q を使います。
user@host:~/work$ q login
✔ Select login method · Use for Free with Builder ID
Confirm the following code in the browser
Code: QQQN-QLCB
Open this URL: https://view.awsapps.com/start/#/device?user_code=QQQN-QLCB
▰▱▱▱▱▱▱ Logging in...
ここで URL をブラウザで開きログイン操作を行います。
user@host:~/work$ q login
✔ Select login method · Use for Free with Builder ID
Confirm the following code in the browser
Code: QQQN-QLCB
Open this URL: https://view.awsapps.com/start/#/device?user_code=QQQN-QLCB
Device authorized
Logged in successfully
user@host:~/work$
Amazon Q を起動します。
user@host:~/work$ q
⢠⣶⣶⣦⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣤⣶⣿⣿⣿⣶⣦⡀⠀
⠀⠀⠀⣾⡿⢻⣿⡆⠀⠀⠀⢀⣄⡄⢀⣠⣤⣤⡀⢀⣠⣤⣤⡀⠀⠀⢀⣠⣤⣤⣤⣄⠀⠀⢀⣤⣤⣤⣤⣤⣤⡀⠀⠀⣀⣤⣤⣤⣀⠀⠀⠀⢠⣤⡀⣀⣤⣤⣄⡀⠀⠀⠀⠀⠀⠀⢠⣿⣿⠋⠀⠀⠀⠙⣿⣿⡆
⠀⠀⣼⣿⠇⠀⣿⣿⡄⠀⠀⢸⣿⣿⠛⠉⠻⣿⣿⠛⠉⠛⣿⣿⠀⠀⠘⠛⠉⠉⠻⣿⣧⠀⠈⠛⠛⠛⣻⣿⡿⠀⢀⣾⣿⠛⠉⠻⣿⣷⡀⠀⢸⣿⡟⠛⠉⢻⣿⣷⠀⠀⠀⠀⠀⠀⣼⣿⡏⠀⠀⠀⠀⠀⢸⣿⣿
⠀⢰⣿⣿⣤⣤⣼⣿⣷⠀⠀⢸⣿⣿⠀⠀⠀⣿⣿⠀⠀⠀⣿⣿⠀⠀⢀⣴⣶⣶⣶⣿⣿⠀⠀⠀⣠⣾⡿⠋⠀⠀⢸⣿⣿⠀⠀⠀⣿⣿⡇⠀⢸⣿⡇⠀⠀⢸⣿⣿⠀⠀⠀⠀⠀⠀⢹⣿⣇⠀⠀⠀⠀⠀⢸⣿⡿
⢀⣿⣿⠋⠉⠉⠉⢻⣿⣇⠀⢸⣿⣿⠀⠀⠀⣿⣿⠀⠀⠀⣿⣿⠀⠀⣿⣿⡀⠀⣠⣿⣿⠀⢀⣴⣿⣋⣀⣀⣀⡀⠘⣿⣿⣄⣀⣠⣿⣿⠃⠀⢸⣿⡇⠀⠀⢸⣿⣿⠀⠀⠀⠀⠀⠀⠈⢿⣿⣦⣀⣀⣀⣴⣿⡿⠃
⠚⠛⠋⠀⠀⠀⠀⠘⠛⠛⠀⠘⠛⠛⠀⠀⠀⠛⠛⠀⠀⠀⠛⠛⠀⠀⠙⠻⠿⠟⠋⠛⠛⠀⠘⠛⠛⠛⠛⠛⠛⠃⠀⠈⠛⠿⠿⠿⠛⠁⠀⠀⠘⠛⠃⠀⠀⠘⠛⠛⠀⠀⠀⠀⠀⠀⠀⠀⠙⠛⠿⢿⣿⣿⣋⠀⠀
⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠛⠿⢿⡧
╭─────────────────────────────── Did you know? ────────────────────────────────╮
│ │
│ You can resume the last conversation from your current directory by │
│ launching with q chat --resume │
│ │
╰──────────────────────────────────────────────────────────────────────────────╯
/help all commands • ctrl + j new lines • ctrl + s fuzzy search
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
🤖 You are chatting with claude-4-sonnet
> こんにちは!
> こんにちは!Amazon Q です。AWS や開発に関することでお手伝いできることがあれば、お気軽にお聞かせください。コマンドライン環境での作業、AWS リ
ソースの管理、コードの作成や修正など、様々なことをサポートできます。
何かご質問やお困りのことはありますか?
>
これで、Q CLI が使える状態になりました。
/model と入力し Enter を押下すると Q で使用中のモデルを確認・変更できます。
> /model
? Select a model for this chat session ›
❯ claude-4-sonnet (active)
claude-3.7-sonnet
claude-4-sonnet が使われていますね。
/quit と入力し Enter を押下すると Q の実行が終了します。
Vibe Coding で会社診断アプリを作成
対話しながらアプリを作成していきます。アプリ用のフォルダを作成してフォルダを移動し、再度 Q を実行します。
初期要件を伝える
長いですが以下プロンプトを入力して実行します。このプロンプトも生成 AI と壁打ちしながら作成しました。
初期プロンプト
以下の要件で、**バニラJS+HTMLのみ**のシンプルな単一ページ診断アプリを作成してください。
ビルドツールや外部フレームワークは使いません。
## 目的
* 「性格占い風の会社診断」アプリ(10問)。
* A〜Eの選択肢を10問回答 → スコア集計 → 最多スコアの会社を結果表示。
* タイブレーク(同点時の決め方)も実装。
## 生成ファイル
* `index.html` … UI本体(レスポンシブ対応、簡易スタイル)
* `app.js` … データ(設問/結果リンク)とロジック(採点・遷移・タイブレーク)
* `styles.css` … 軽量のスタイル(ダーク/ライト対応までは不要。コントラスト充分に)
## 言語・文言
* UIは**日本語**表記。
* 設問・選択肢の本文は、後述 `<<QUESTIONS_FROM_CANVAS>>` をそのままデータとして組み込み。
## 対象会社(スコア対応)
選択肢キーの対応は固定で以下。
* **A → クラスメソッド**(id: `classmethod`)
* **B → アノテーション株式会社**(id: `annotation`)
* **C → CMテクノロジーズ**(id: `cmtech`)
* **D → CMキャリア**(id: `cmcareer`)
* **E → CMパートナーズ**(id: `cmpartners`)
## 結果ページCTA(外部リンク)
* `classmethod`:
* 募集要項を見る → `https://careers.classmethod.jp/requirements/category/engineer/`
* 社員ブログを見る → `https://dev.classmethod.jp/`
* `annotation`:
* 募集要項を見る → `https://careers.annotation.co.jp/requirements/`
* `cmtech`:
* 募集要項を見る → `https://www.classmethod-technologies.jp/recruit`
* `cmcareer`:
* 無料キャリア相談を予約 → `https://propagate-tech.com/services/cmcareer`
* `cmpartners`:
* 案件相談をする → `https://propagate-tech.com/services/cmpartners`
## タイブレーク(同点処理)
1. **cmtech が同点グループに含まれている場合は cmtech を優先**
2. **classmethod と annotation が同点最上位**
* 発信文化/出社許容の志向フラグ(下記UIトグル)で分岐:
* 「発信文化が好き」= classmethod
* 「働き方の柔軟性を重視」= annotation
3. **cmcareer と cmpartners が同点最上位**
* 「雇用/福利を重視」= cmcareer
* 「報酬/裁量を重視」= cmpartners
4. それでも決まらない場合は、**最初に選んだ回答の会社**を優先。
※ 2項・3項のフラグは結果画面に**小さなラジオボタン**で表示し、
利用者がワンクリックで決められるUIにする。
## 画面仕様
* **開始画面**
* タイトル:「会社診断(全10問・所要1〜2分)」
* 「開始する」ボタン
* **設問画面(10ページ)**
* 進捗(`問 1 / 10` 〜 `問 10 / 10`)
* 設問文
* A〜Eボタン(選択式、1つ選ぶと次へ)
* 「前へ」「次へ」ボタン、**未選択時は次へを無効**
* **ローカルストレージ**で進捗・回答を保持(ページリロード耐性)
* **結果画面**
* 決定した会社名、短い説明文(たたきでOK)、CTAボタン(上記リンク)
* 「もう一度診断」ボタン(全リセット)
* \*\*同点時の微調整UI(ラジオボタン)\*\*を上記ルールに従って出し分け
* **アクセシビリティ**
* ボタンはキーボード操作可能、`aria-label` 適切付与、コントラスト基準(WCAG AA 目安)
## データ投入(ここに設問を貼って使用)
以下の **10問×選択肢A〜E** を `app.js` の `QUESTIONS` 配列に格納してください。
`score` は `classmethod | annotation | cmtech | cmcareer | cmpartners` のいずれか。
```
<<QUESTIONS_FROM_CANVAS>>
```
> 例フォーマット(サンプル1問分だけ示します。実際は10問すべてを同形式で定義):
```js
const QUESTIONS = [
{
id: 1,
text: "朝いちのテンションは?",
choices: [
{ key: "A", label: "技術ブログの下書きを仕上げたい!", score: "classmethod" },
{ key: "B", label: "まずは保育園送迎と家事、そのあと集中タイム", score: "annotation" },
{ key: "C", label: "きょうの顧客打ち合わせで価値を出す準備だ", score: "cmtech" },
{ key: "D", label: "市場/求人リサーチで次のキャリア戦略を立てたい", score: "cmcareer" },
{ key: "E", label: "稼働調整しつつ単価とスキルUPのバランス最適化", score: "cmpartners" }
]
},
// ... Q2〜Q10
];
```
## 実装要件(ロジック)
* **採点**:選んだ選択肢の `score` をカウントし、最多IDで会社決定
* **同点**:上記タイブレーク順で決定
* **状態管理**:
* `answers`(配列: index=0..9 に "A"〜"E")
* `scores`(連想配列: 各会社ID→カウント)
* `currentQuestionIndex`(0..9)
* LocalStorageキー例:`company-quiz-v1`(進捗/回答/時刻)
* **バリデーション**:
* 各問の未選択で「次へ」無効
* 全10問回答後に結果表示
* **リセット**:
* 「もう一度診断」で localStorage をクリアして開始画面へ
## スタイル要件(最小限)
* モバイル〜PCで見やすい中央寄せカードレイアウト
* 大きめのタップ領域(44px以上)
* プログレスは%バー表示
* フォントはシステムUIでOK
## テスト観点(簡易)
* 10問すべて回答できること
* 同点ケースのタイブレークが仕様通り動くこと
* リロードしても進捗が保持されること
* リセットで初期状態に戻ること
* 外部リンクの遷移先が会社ごとに正しいこと
---
## 追加タスク(任意だが可能なら)
* 結果カードを `canvas.toDataURL()` 相当で画像保存(実装容易な範囲でOK)
* シェア用OGタグ(最低限のメタだけ)
* 軽いアニメーション(CSSのtransitionのみ)
---
### 納品物
* `index.html` / `app.js` / `styles.css`
* すぐにブラウザで開いて利用できること(ローカル動作前提、ビルド不要)
---
## メモ
* 設問本文は、プロンプト中 `<<QUESTIONS_FROM_CANVAS>>` の塊をそのまま
`QUESTIONS` に落とし込んでください。
* 選択肢キー(A〜E)→会社IDのマッピングは固定。誤って入れ替えないこと。
* 日本語UIの微修正(文言/句読点/読点)は読みやすさ優先で適宜可。
---
以下のように回答が得られました。
> 診断アプリが完成しました!以下の3つのファイルが作成されています:
## 作成されたファイル
1. index.html - メインのHTML構造
2. styles.css - レスポンシブ対応のスタイルシート
3. app.js - 診断ロジックとデータ
## 主な機能
✅ 10問の診断システム - A〜E選択肢で各社にスコア配分
✅ タイブレーク機能 - 同点時の優先順位とUI選択
✅ ローカルストレージ - ページリロード時の進捗保持
✅ レスポンシブデザイン - モバイル〜PC対応
✅ アクセシビリティ - キーボード操作、aria-label対応
✅ 外部リンク - 各社の募集要項・サービスページへの遷移
## 使用方法
1. ブラウザで index.html を開く
2. 「開始する」ボタンで診断開始
3. 10問回答後、結果画面で会社とCTAリンクを表示
4. 同点時はラジオボタンで微調整可能
## 設問データの更新
app.js の QUESTIONS 配列を実際の設問データに置き換えてください。
現在はサンプルデータが入っています。
js
const QUESTIONS = [
{
id: 1,
text: "実際の設問文",
choices: [
{ key: "A", label: "選択肢A", score: "classmethod" },
// ... 他の選択肢
]
},
// ... 残り9問
];
すぐにローカルでテストできる状態になっています!
>
このようにして作成してもらったアプリの初版を、ローカルで確認します。
WSL 内のファイルはエクスプローラーから確認できます。

index.html をブラウザで開きます。

おお、いけそうです。開始してみます。

回答を選択して「次へ」で進んでいきます。
以下のように診断結果が出ました。「募集要項を見る」をクリックすると、採用ページに遷移します。他の会社も見られます。

アプリを AWS 上で公開する
現在のアプリをS3で公開する手順を教えてください。
▼ Q の回答
Label: S3バケット作成> S3での静的サイト公開手順をお教えします。
## 1. S3バケット作成
bash
aws s3 mb s3://your-app-name-unique --region ap-northeast-1
## 2. 静的ウェブサイトホスティング有効化
bash
aws s3 website s3://your-app-name-unique --index-document index.html
## 3. パブリックアクセス許可
bash
aws s3api put-public-access-block \
--bucket your-app-name-unique \
--public-access-block-configuration "BlockPublicAcls=false,IgnorePublicAcls=false,BlockPublicPolicy=false,RestrictPublicBuckets=false"
## 4. バケットポリシー設定
🛠️ Using tool: fs_write
⋮
● Path: /home/user/work/jaws-festa-2025/bucket-policy.json
+ 1: {
+ 2: "Version": "2012-10-17",
+ 3: "Statement": [
+ 4: {
+ 5: "Sid": "PublicReadGetObject",
+ 6: "Effect": "Allow",
+ 7: "Principal": "*",
+ 8: "Action": "s3:GetObject",
+ 9: "Resource": "arn:aws:s3:::your-app-name-unique/*"
+ 10: }
+ 11: ]
+ 12: }
⋮
↳ Purpose: S3バケットポリシーファイル作成
Allow this action? Use 't' to trust (always allow) this tool for the session. [y/n/t]:
> t
Creating: /home/user/work/jaws-festa-2025/bucket-policy.json
⋮
● Completed in 0.1s
> bash
aws s3api put-bucket-policy --bucket your-app-name-unique --policy file://bucket-policy.json
## 5. ファイルアップロード
bash
cd アプリ
aws s3 sync . s3://your-app-name-unique --delete
## 6. アクセス確認
アップロード後、以下のURLでアクセス可能:
http://your-app-name-unique.s3-website-ap-northeast-1.amazonaws.com
注意点:
• your-app-name-uniqueは世界で一意な名前に変更してください
• 料金は月額数十円程度(アクセス量による)
• HTTPSが必要な場合はCloudFrontの追加設定が必要
実行しますか?バケット名を決めていただければ、実際にコマンドを実行できます。
>
S3 でアプリを公開するためのバケットポリシー例のファイルも作成してくれました。
AWS の認証情報を Q に渡すのがちょっと嫌だったので、自身で環境構築することにしました。
こんな形で少し問答をして、HTTPS 通信にしたいので CloudFront を挟む構成に変更するなどの作業を自身で行いました。
簡単ですが、以下の構成でリソースを作成し、S3 バケットに作成されたファイルを配置すると、CloudFront の URL でアプリケーションにアクセスできるようになり、アプリケーションを公開できます。

アプリケーションの README.md を作成する
さて、一応後から見返してどんなアプリなのか分かるように、README.md ファイルを作成しておきましょう。
このアプリの README.mdを作成してください。
Q の解答
> アプリの内容を確認してREADME.mdを作成します。
🛠️ Using tool: fs_read (trusted)
⋮
● Reading file: /home/user/work/jaws-festa-2025/アプリ/index.html, all lines
✓ Successfully read 2639 bytes from /home/user/work/jaws-festa-2025/アプリ/index.html
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Reading file: /home/user/work/jaws-festa-2025/アプリ/app.js, from line 1 to 50
✓ Successfully read 2554 bytes from /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: companies
✓ Found 16 matches for pattern 'COMPANIES' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_write (trusted)
⋮
● Path: /home/user/work/jaws-festa-2025/README.md
+ 1: # 会社診断アプリ
+ 2:
+ 3: JAWS FESTA 2025用の会社診断アプリです。15問の質問に答えることで、あなたにぴったりの会社を診断します。
+ 4:
+ 5: ## 🌐 公開URL
+ 6:
+ 7: **本番環境**: https://d1fl92kkv9nxql.cloudfront.net/
+ 8:
+ 9: ## 📋 概要
+ 10:
+ 11: - **診断時間**: 2〜3分
+ 12: - **質問数**: 15問
+ 13: - **診断対象**: 5つの会社
+ 14: - クラスメソッド
+ 15: - アノテーション
+ 16: - CMテクノロジーズ
+ 17: - CMキャリア
+ 18: - CMパートナーズ
+ 19:
+ 20: ## 🚀 技術構成
+ 21:
+ 22: ### フロントエンド
+ 23: - HTML5
+ 24: - CSS3 (レスポンシブデザイン)
+ 25: - Vanilla JavaScript
+ 26:
+ 27: ### インフラ
+ 28: - **Amazon S3**: 静的ファイルホスティング
+ 29: - **Amazon CloudFront**: HTTPS配信 + CDN
+ 30: - **AWS CLI**: デプロイメント
+ 31:
+ 32: ## 📁 ファイル構成
+ 33:
+ 34: ```
+ 35: アプリ/
+ 36: ├── index.html # メインHTML
+ 37: ├── app.js # アプリケーションロジック
+ 38: └── styles.css # スタイルシート
+ 39: ```
+ 40:
+ 41: ## 🛠️ ローカル開発
+ 42:
+ 43: ```bash
+ 44: # ファイルをローカルで確認
+ 45: cd アプリ
+ 46: python -m http.server 8000
+ 47: # または
+ 48: npx serve .
+ 49: ```
+ 50:
+ 51: ブラウザで `http://localhost:8000` にアクセス
+ 52:
+ 53: ## 🚀 デプロイ
+ 54:
+ 55: ### 前提条件
+ 56: - AWS CLI設定済み
+ 57: - 適切なIAM権限
+ 58:
+ 59: ### デプロイ手順
+ 60:
+ 61: ```bash
+ 62: # S3にファイルをアップロード
+ 63: cd アプリ
+ 64: aws s3 sync . s3://your-bucket-name --delete
+ 65:
+ 66: # CloudFrontキャッシュクリア(即座に反映したい場合)
+ 67: aws cloudfront create-invalidation --distribution-id YOUR_DISTRIBUTION_ID --paths "/*"
+ 68: ```
+ 69:
+ 70: ## 💰 運用コスト
+ 71:
+ 72: - **S3**: 月額数十円(ストレージ + リクエスト)
+ 73: - **CloudFront**: 月額数十円〜(データ転送量による)
+ 74: - **合計**: 月額100円程度(一般的なアクセス量)
+ 75:
+ 76: ## 🎯 機能
+ 77:
+ 78: - プログレスバー表示
+ 79: - 前の質問に戻る機能
+ 80: - 同点時のタイブレーク機能
+ 81: - レスポンシブデザイン
+ 82: - 結果に応じたCTAリンク表示
+ 83:
+ 84: ## 📱 対応ブラウザ
+ 85:
+ 86: - Chrome (推奨)
+ 87: - Firefox
+ 88: - Safari
+ 89: - Edge
+ 90:
+ 91: ## 📄 ライセンス
+ 92:
+ 93: このプロジェクトはJAWS FESTA 2025のために作成されました。
⋮
↳ Purpose: README.mdファイル作成
Creating: /home/user/work/jaws-festa-2025/README.md
⋮
● Completed in 0.1s
> README.mdを作成しました!
現在の公開URLとデプロイ手順を含む完全なドキュメントになっています。主な内容:
• **公開URL**: https://d1fl92kkv9nxql.cloudfront.net/
• **技術構成**: S3 + CloudFront
• **デプロイ手順**: AWS CLIコマンド
• **運用コスト**: 月額100円程度
• **機能説明**: 15問診断、タイブレーク機能など
何か追加や修正したい項目はありますか?
>
この後、
デプロイの部分を修正したいです。ユーザーが自身のAWSアカウント上でデプロイすることを前提に、任意のS3バケットを作成する方法を記載してください。
ローカルで動かす場合はhtmlファイルをクリックするだけでよくないですか?npxなどを使いたくないのでバニラJavaで組んでいるのですし…
などの追加プロンプトを実行し、README.md を修正していきました。
アプリの微修正
対話しながら、以下の機能を追加していきました:
アプリフォルダ配下のアプリケーションのトップ画面ですが、「会社診断」ではなく「クラスメソッドグループお似合い会社診断テスト~あなたにピッタリの会社はどれ??~」に変えてください。
1問目の前へボタンは必要ないので、「最初に戻る」ボタンに変えてトップページに戻るように変更してください。
Q の解答
> アプリフォルダ配下のアプリケーションのトップ画面ですが、「会社診断」ではなく「クラスメソッドグループお似合い会社診断テスト~あなたにピッタリの会社はどれ??~」に変えてください。
1問目の前へボタンは必要ないので、「最初に戻る」ボタンに変えてトップページに戻るように変更してください。
> まず、アプリフォルダの構造を確認して、該当するファイルを特定します。
🛠️ Using tool: fs_read (trusted)
⋮
● Reading directory: /home/user/work/jaws-festa-2025 with maximum depth of 2
✓ Successfully read directory /home/user/work/jaws-festa-2025 (14 entries)
⋮
● Completed in 0.3s
> アプリフォルダ内のファイルを確認します。まず、HTMLファイルとJavaScriptファイルの内容を確認して、タイトルと前へボタンの箇所を特定します。
🛠️ Using tool: fs_read (trusted)
⋮
● Reading file: /home/user/work/jaws-festa-2025/アプリ/index.html, all lines
✓ Successfully read 2855 bytes from /home/user/work/jaws-festa-2025/アプリ/index.html
⋮
● Completed in 0.0s
> 次に、JavaScriptファイルの内容を確認して、前へボタンの動作を確認します。
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: prev-btn
✓ Found 1 matches for pattern 'prev-btn' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: addeventlistener.*prev
✓ Found 0 matches for pattern 'addEventListener.*prev' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: prevbtn
✓ Found 3 matches for pattern 'prevBtn' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: function prevquestion
✓ Found 1 matches for pattern 'function prevQuestion' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Reading file: /home/user/work/jaws-festa-2025/アプリ/app.js, from line 374 to 380
✓ Successfully read 135 bytes from /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
> それでは、要求された変更を実施します:
1. HTMLファイルのタイトルを変更
2. 前へボタンを「最初に戻る」ボタンに変更
3. JavaScriptの動作を修正
🛠️ Using tool: fs_write
⋮
● Path: アプリ/index.html
- 15 : <h1>会社診断</h1>
+ 15: <h1>クラスメソッドグループお似合い会社診断テスト~あなたにピッタリの会社はどれ??~</h1>
⋮
↳ Purpose: トップ画面のタイトルを変更
Allow this action? Use 't' to trust (always allow) this tool for the session. [y/n/t]:
> y
Updating: アプリ/index.html
⋮
● Completed in 0.12s
🛠️ Using tool: fs_write
⋮
● Path: アプリ/index.html
- 36 : <button id="prev-btn" class="btn btn-secondary">前へ</button>
+ 36: <button id="prev-btn" class="btn btn-secondary">最初に戻る</button>
⋮
↳ Purpose: 前へボタンを最初に戻るボタンに変更
Allow this action? Use 't' to trust (always allow) this tool for the session. [y/n/t]:
> y
Updating: アプリ/index.html
⋮
● Completed in 0.0s
🛠️ Using tool: fs_write
⋮
● Path: アプリ/app.js
374, 374: function prevQuestion() {
- 375 : if (state.currentQuestionIndex > 0) {
- 376 : state.currentQuestionIndex--;
- 377 : updateQuestionDisplay();
- 378 : }
+ 375: showScreen('start-screen');
+ 376: state.currentQuestionIndex = 0;
+ 377: state.answers = {};
379, 378: }
⋮
↳ Purpose: 前へボタンの動作をトップページに戻る動作に変更
Allow this action? Use 't' to trust (always allow) this tool for the session. [y/n/t]:
> y
Updating: アプリ/app.js
⋮
● Completed in 0.1s
🛠️ Using tool: fs_write
⋮
● Path: アプリ/app.js
355, 355: // ナビゲーションボタンの状態更新
- 356 : prevBtn.disabled = state.currentQuestionIndex === 0;
+ 356: prevBtn.disabled = false; // 常に有効
357, 357: nextBtn.disabled = !state.answers[state.currentQuestionIndex];
⋮
↳ Purpose: 1問目でも最初に戻るボタンを有効にする
Allow this action? Use 't' to trust (always allow) this tool for the session. [y/n/t]:
> 1
> 変更が完了しました。以下の修正を行いました:
1. トップ画面のタイトル変更: 「会社診断」から「クラスメソッドグループお似合い会社診断テスト~あなたにピッタリの会社はどれ??~」に変更
2. 前へボタンの表示変更: 「前へ」から「最初に戻る」に変更
3. ボタンの動作変更: 「最初に戻る」ボタンをクリックすると、前の問題に戻るのではなく、トップページに戻り、診断状態もリセットされるように変更
これで、1問目からでも「最初に戻る」ボタンが表示され、クリックするとトップページに戻るようになります。
>
「最初に戻る」ボタンで最初に戻れません。修正してください。
また、2問目以降の画面には「前に戻る」ボタンを追加してください。
Q の解答
> 問題を修正します。まず現在のJavaScriptの状態を確認して、適切に修正します。
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: function showscreen
✓ Found 0 matches for pattern 'function showScreen' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: showscreen
✓ Found 1 matches for pattern 'showScreen' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: screen.*active
✓ Found 0 matches for pattern 'screen.*active' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: active
✓ Found 9 matches for pattern 'active' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
> まず、HTMLに「前に戻る」ボタンを追加し、JavaScriptを修正します。
🛠️ Using tool: fs_write
⋮
● Path: アプリ/index.html
35, 35: <div class="navigation">
36, 36: <button id="prev-btn" class="btn btn-secondary">最初に戻る</button>
+ 37: <button id="back-btn" class="btn btn-secondary" style="display: none;">前に戻る</button>
37, 38: <button id="next-btn" class="btn btn-primary" disabled>次へ</button>
38, 39: </div>
⋮
↳ Purpose: 前に戻るボタンを追加
Allow this action? Use 't' to trust (always allow) this tool for the session. [y/n/t]:
> t
Updating: アプリ/index.html
⋮
● Completed in 0.12s
🛠️ Using tool: fs_write (trusted)
⋮
● Path: アプリ/app.js
374, 374: function prevQuestion() {
- 375 : showScreen('start-screen');
+ 375: showStartScreen();
376, 376: state.currentQuestionIndex = 0;
377, 377: state.answers = {};
378, 378: }
⋮
↳ Purpose: 最初に戻るボタンの動作を修正
Updating: アプリ/app.js
⋮
● Completed in 0.1s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: const.*btn.*getelementbyid
✓ Found 0 matches for pattern 'const.*Btn.*getElementById' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Reading file: /home/user/work/jaws-festa-2025/アプリ/app.js, from line 225 to 235
✓ Successfully read 666 bytes from /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_write (trusted)
⋮
● Path: アプリ/app.js
229, 229: const prevBtn = document.getElementById('prev-btn');
+ 230: const backBtn = document.getElementById('back-btn');
230, 231: const nextBtn = document.getElementById('next-btn');
⋮
↳ Purpose: 前に戻るボタンの要素を追加
Updating: アプリ/app.js
⋮
● Completed in 0.1s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: addeventlistener.*click
✓ Found 0 matches for pattern 'addEventListener.*click' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Reading file: /home/user/work/jaws-festa-2025/アプリ/app.js, from line 273 to 280
✓ Successfully read 254 bytes from /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_write (trusted)
⋮
● Path: アプリ/app.js
273, 273: // イベントバインド
274, 274: function bindEvents() {
275, 275: startBtn.addEventListener('click', startQuiz);
276, 276: prevBtn.addEventListener('click', prevQuestion);
+ 277: backBtn.addEventListener('click', backQuestion);
277, 278: nextBtn.addEventListener('click', nextQuestion);
278, 279: restartBtn.addEventListener('click', restart);
279, 280: }
⋮
↳ Purpose: 前に戻るボタンのイベントリスナーを追加
Updating: アプリ/app.js
⋮
● Completed in 0.1s
🛠️ Using tool: fs_write (trusted)
⋮
● Path: アプリ/app.js
375, 375: // 前の質問
376, 376: function prevQuestion() {
377, 377: showStartScreen();
378, 378: state.currentQuestionIndex = 0;
379, 379: state.answers = {};
380, 380: }
+ 381:
+ 382: // 前に戻る
+ 383: function backQuestion() {
+ 384: if (state.currentQuestionIndex > 0) {
+ 385: state.currentQuestionIndex--;
+ 386: updateQuestionDisplay();
+ 387: }
+ 388: }
⋮
↳ Purpose: 前に戻る関数を追加
Updating: アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_read (trusted)
⋮
● Searching: /home/user/work/jaws-festa-2025/アプリ/app.js for pattern: ナビゲーションボタンの状態更新
✓ Found 1 matches for pattern 'ナビゲーションボタンの状態更新' in /home/user/work/jaws-festa-2025/アプリ/app.js
⋮
● Completed in 0.0s
🛠️ Using tool: fs_write (trusted)
⋮
● Path: アプリ/app.js
357, 357: // ナビゲーションボタンの状態更新
- 358 : prevBtn.disabled = state.currentQuestionIndex === 0;
+ 358: prevBtn.disabled = false; // 最初に戻るボタンは常に有効
+ 359: backBtn.style.display = state.currentQuestionIndex === 0 ? 'none' : 'inline-block';
359, 360: nextBtn.disabled = !state.answers[state.currentQuestionIndex];
⋮
↳ Purpose: ボタンの表示・非表示を制御
Updating: アプリ/app.js
⋮
● Completed in 0.0s
> 修正が完了しました。以下の変更を行いました:
1. 「最初に戻る」ボタンの修正: showStartScreen()関数を使用してトップページに正しく戻るように修正
2. 「前に戻る」ボタンの追加:
• HTMLに新しい「前に戻る」ボタンを追加
• JavaScriptにbackQuestion()関数を追加して前の問題に戻る機能を実装
• イベントリスナーを追加
3. ボタンの表示制御:
• 1問目では「前に戻る」ボタンを非表示
• 2問目以降では「前に戻る」ボタンを表示
• 「最初に戻る」ボタンは常に有効
これで、1問目では「最初に戻る」ボタンのみが表示され、2問目以降では「最初に戻る」と「前に戻る」の両方のボタンが表示されます。
>
「次へ」ボタンを無くして、回答をタップで次の設問に進むようにしてください。
:
(色々問答する)
:
修正後のファイルを S3 にアップロードしてキャッシュもクリアしましたが、
回答をタップしても次に進まなくなってしまったし、
フォントの大きさも変わってしまったし、
最初に戻るボタンも機能しなくなってしまいました。
ローカル端末では動いています。
修正してください。
:
:
などなど、紆余曲折ありましたが、なんとかそれらしい動きをするようにアプリケーションを修正して公開できる状態になりました。問題数も修正しています。
タイトルは以下のように変更しています。

1 問目で「前に戻る」ボタンがあるのはおかしいので、「最初に戻る」ボタンに変更しています。

回答を選択して「次へ」だとテンポが悪いので、回答を選択した瞬間に次の設問に進むようにしました。「最初に戻る」ボタンと「前に戻る」ボタンを追加しています。

Vibe Coding で開発してみた感想
良かった点
- 開発スピードが速い
- 詳細な仕様書を書かずに、ざっくりとした要件から実装が進む
- 対話的な開発
- 「ここをこう変えたい」と伝えるだけで修正してくれる
- 学習効果
- 生成されたコードを読むことで、実装パターンを学べる
- プロトタイピングに最適
- アイデアをすぐに形にできる
改善の余地がある点
- 複雑な要件は何度か対話が必要
- 生成されたコードの確認が必要
- 小〜中規模のアプリはすぐに開発可能だが、大規模開発時は工夫が必要
既知の課題:タイブレーク処理の偏り
実装後に気づいた点として、タイブレーク処理に公平性の問題があります。
一番最初に投入した要件のプロンプトは Amazon Q ではない別のチャット AI に頼んで作ってもらったものだったのですが、その要件を私がきちんと確認していなかったせいで、公平性に問題のある実装になってしまいました。
- CMテクノロジーズが同点に含まれる場合 → 常に CM テクノロジーズ
- クラスメソッド vs アノテーション → デフォルトでクラスメソッド
- CMキャリア vs CMパートナーズ → デフォルトで CM キャリア
これでは特定の会社に結果が偏ってしまいます。
本格的に運用する場合は、こういった部分を人間が読み解いて修正する必要があることを痛感しました。エラーハンドリングも実装できなかったです…
おわりに
Amazon Q CLI の Vibe Coding を使って、会社診断アプリを作成してみました。従来の開発では詳細な仕様書が必要でしたが、Vibe Coding では「こんな感じのアプリが欲しい」という雰囲気を伝えるだけで実装が進みます。
私は普段インフラエンジニアとして働いており、開発経験は多くないのですが、なんとかここまで動くものを業務の合間に作成できたのは AI の力の賜物です。
様々な場所で同じことが言われていると思いますが、以下のようなケースでは有効ですね。
- プロトタイピングやアイデアの検証
- 学習用のサンプルアプリケーション作成
- 社内ツールなどの小〜中規模アプリケーション
繰り返しますが、本アプリは学習目的で作成したシンプルな Web アプリケーションです。 実際の開発では、セキュリティやパフォーマンス、保守性などを考慮した設計が必要です。
本記事への質問やご要望については画面下部のお問い合わせ「DevelopersIOへのご意見」からご連絡ください。記事に関してお問い合わせいただけます。






