0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

GitHubにpushしてQiitaに投稿

Last updated at Posted at 2025-03-19

GitHubにpushしてQiitaに投稿

GitHubのリポジトリに記事を書いてpushすると、自動でQiitaに投稿・更新される仕組みをGitHub Actionsを使って構築しました。

これにより、GitHub で記事を管理しつつ、Qiita に自動投稿できるようになります。

できること

  • articles/ フォルダ内の Markdown (.md) を自動投稿・更新
  • Qiita の記事 ID を GitHub で管理
  • git push するだけで Qiita に反映

環境構築

Qiita API トークンを取得

Qiitaの個人設定 で API トークンを発行します。

必要な権限:

  • read_qiita(記事の読み取り)
  • write_qiita(記事の作成・更新)

発行したトークンは、GitHub の Secrets に登録します。

GitHub Secrets に Qiita トークンを追加

GitHub リポジトリの Settings > Secrets and variables > ActionsQIITA_TOKEN を追加。

フォルダ構成

├── articles/                 # Qiita に投稿する記事フォルダ
│   ├── 2024-03-19-github-to-qiita.md  # 記事ファイル
│   ├── tutorial.md           # 記事ファイル
├── .github/
│   ├── workflows/
│   │   ├── qiita-publish.yml # GitHub Actions の設定
│   ├── scripts/
│   │   ├── publish_to_qiita.py # Qiita 投稿スクリプト
├── .github/qiita_posted.json  # 記事 ID 管理 (自動生成)
├── README.md                  # リポジトリの説明

GitHub Actions 設定

.github/workflows/qiita-publish.yml に以下を追加。

name: Publish to Qiita

on:
  push:
    branches:
      - main

jobs:
  publish:
    runs-on: ubuntu-latest

    permissions:
      contents: write

    steps:
      - name: リポジトリをチェックアウト
        uses: actions/checkout@v4
        with:
          persist-credentials: true

      - name: Pythonをセットアップ
        uses: actions/setup-python@v4
        with:
          python-version: '3.13.2'

      - name: 依存関係をインストール
        run: pip install requests pyyaml

      - name: Qiita に投稿・更新
        env:
          QIITA_TOKEN: ${{ secrets.QIITA_TOKEN }}
        run: python .github/scripts/publish_to_qiita.py

      - name: 更新されたファイルをコミット・プッシュ
        run: |
          git config --global user.name "github-actions[bot]"
          git config --global user.email "github-actions[bot]@users.noreply.github.com"
          git add .github/qiita_posted.json
          git commit -m "Update Qiita posted IDs" || echo "No changes to commit"
          git push

Qiita 投稿スクリプト

.github/scripts/publish_to_qiita.py に Qiita 投稿用のスクリプトを作成。

import os
import json
import requests
import glob
import yaml

QIITA_API_URL = "https://qiita.com/api/v2/items"
QIITA_TOKEN = os.getenv("QIITA_TOKEN")
ARTICLE_DIR = "articles"
QIITA_ID_FILE = ".github/qiita_posted.json"

if not QIITA_TOKEN:
    raise ValueError("Qiita APIトークンが設定されていません")

def load_qiita_ids():
    """ 記事IDをJSONファイルから読み込む """
    if os.path.exists(QIITA_ID_FILE):
        with open(QIITA_ID_FILE, "r", encoding="utf-8") as f:
            return json.load(f)
    return {}

def save_qiita_ids(qiita_ids):
    """ 記事IDをJSONファイルに保存する """
    print(f"QIITA_ID_FILE: {QIITA_ID_FILE}")
    with open(QIITA_ID_FILE, "w", encoding="utf-8") as f:
        json.dump(qiita_ids, f, indent=2, ensure_ascii=False)

def parse_markdown(file_path):
    """ Markdown ファイルの Front Matter を解析し、本文を取得する """
    with open(file_path, "r", encoding="utf-8") as f:
        content = f.read()

    if content.startswith("---"):
        parts = content.split("---", 2)
        meta_data = yaml.safe_load(parts[1])
        body = parts[2].strip()
    else:
        meta_data, body = {}, content

    return meta_data, body

def post_to_qiita(file_name, meta, body, qiita_ids):
    """ Qiitaへ記事を投稿または更新する """
    title = meta.get("title", file_name.replace(".md", "").replace("_", " ").title())
    tags = [{"name": tag} for tag in meta.get("tags", [])]
    private = meta.get("private", False)
    data = {"title": title, "body": body, "private": private, "tags": tags}

    headers = {"Authorization": f"Bearer {QIITA_TOKEN}"}

    if file_name in qiita_ids:
        qiita_article_id = qiita_ids[file_name]
        response = requests.put(f"{QIITA_API_URL}/{qiita_article_id}", headers=headers, json=data)
        action = "更新"
    else:
        response = requests.post(QIITA_API_URL, headers=headers, json=data)
        action = "新規投稿"
        if response.status_code == 201:
            qiita_ids[file_name] = response.json().get("id")

    if response.status_code in [200, 201]:
        print(f"Qiita {action}成功: {title}")
        print("記事URL:", response.json().get("url"))
    else:
        print(f"Qiita {action}失敗: {title}")
        print("エラー:", response.text)

def main():
    qiita_ids = load_qiita_ids()
    md_files = glob.glob(f"{ARTICLE_DIR}/*.md")

    for md_file in md_files:
        file_name = os.path.basename(md_file)
        meta, body = parse_markdown(md_file)
        post_to_qiita(file_name, meta, body, qiita_ids)

    save_qiita_ids(qiita_ids)

if __name__ == "__main__":
    main()

まとめ

  1. Qiita API トークンを取得し、GitHub Secrets に QIITA_TOKEN を追加
  2. articles/ に記事を .md 形式で作成
  3. git push するだけで Qiita に自動投稿 & 更新
0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?