brave-people/Dev-Event의 README.md에 등록된 개발자 행사 정보를 파싱해 Discord 채널로 알려주는 GitHub Actions 기반 자동화 봇입니다.
- Dev-Event README의 월별 행사 목록 자동 수집
- 인라인/멀티라인 Markdown 행사 형식 파싱
- Discord Webhook Embed 메시지 전송
events_cache.json기반 중복 알림 방지- GitHub Actions 스케줄 실행 및 캐시 자동 커밋
- GitHub Actions Artifact가 아닌 Git 추적 파일로 캐시 유지
- 네트워크 다운로드 실패 시 로컬 README 폴백 지원
Discord에는 아래와 같은 Embed 형태로 전송됩니다.
제목: CloudBro 1주년 행사
URL: https://ticketa.co/event/dttikon7
설명: 분류: `오프라인(서울 강남구)`, `유료`, `모임`, `클라우드` | 주최: CloudBro | 접수: 04. 24(목) ~ 05. 12(화)
필드: 시기 = 26년 05월
푸터: Dev-Event Bot
dev-event-bot/
├── .github/
│ └── workflows/
│ └── dev-event-bot.yml # GitHub Actions 자동 실행 워크플로
├── tests/
│ └── test_markdown_parser.py # MarkdownParser 단위 테스트
├── dev_event_bot.py # 봇 메인 코드
├── events_cache.json # 이미 전송한 행사 URL 캐시
├── requirements.txt # Python 의존성
└── README.md
dev_event_bot.py가 Dev-Event README를 다운로드합니다.- 1차: jsDelivr CDN
- 2차: GitHub Raw URL
- 폴백: 로컬
README.md
MarkdownParser가## `26년 05월`같은 월별 섹션에서 행사 링크와 메타데이터를 추출합니다.events_cache.json에 없는 신규 행사만 Discord Webhook으로 전송합니다.- 전송 성공한 행사 URL을 캐시에 저장합니다.
- GitHub Actions가 변경된 캐시 파일을 현재 브랜치에 커밋/푸시합니다.
캐시는
actions/download-artifact로 내려받지 않습니다. Artifact는 실행 간 영속 저장소가 아니므로, 첫 실행이나 업로드가 생략된 실행에서Artifact not found for name: events_cache오류가 날 수 있습니다.
- Python 3.11 이상 권장
- Discord Webhook URL
- GitHub Actions 사용 시
contents: write권한
Python 패키지는 requirements.txt로 관리합니다.
requests>=2.31.0
git clone https://github.com/dubu-alt/dev-event-bot.git
cd dev-event-bot저장소 URL이 다르다면 실제 사용 중인 GitHub 저장소 주소로 바꿔 주세요.
python -m venv .venv
source .venv/bin/activate
python -m pip install --upgrade pip
pip install -r requirements.txtWindows PowerShell에서는 다음처럼 활성화합니다.
python -m venv .venv
.\.venv\Scripts\Activate.ps1
python -m pip install --upgrade pip
pip install -r requirements.txt처음 실행하는 경우 events_cache.json을 빈 배열로 준비합니다.
[]macOS/Linux:
export DISCORD_WEBHOOK_URL="https://discord.com/api/webhooks/xxxxx/xxxxx"Windows PowerShell:
$env:DISCORD_WEBHOOK_URL="https://discord.com/api/webhooks/xxxxx/xxxxx"Webhook URL은 민감 정보입니다. 코드, README, 캐시 파일에 직접 커밋하지 마세요.
python dev_event_bot.pyMarkdown 파서 단위 테스트를 실행하려면 다음 명령을 사용합니다.
python -m unittest discover -s tests이 저장소는 .github/workflows/dev-event-bot.yml 워크플로를 사용합니다. 캐시는 events_cache.json을 Git에 커밋하는 방식으로 유지하므로, events_cache Artifact 다운로드 단계가 필요하지 않습니다.
name: Dev-Event Bot (Git-backed Cache)
on:
schedule:
# 매일 09:00 UTC (18:00 KST)
- cron: '0 0 * * *'
workflow_dispatch:
permissions:
contents: write
concurrency:
group: dev-event-bot-${{ github.ref }}
cancel-in-progress: false참고: 위 주석에는
09:00 UTC라고 적혀 있지만, cron 값0 0 * * *는 실제로 매일 00:00 UTC / 09:00 KST에 실행됩니다.
이 오류는 워크플로에서 actions/download-artifact로 events_cache를 내려받으려 할 때, 해당 실행에 업로드된 Artifact가 없어서 발생합니다. 이 프로젝트는 실행 간 캐시를 Artifact가 아니라 Git에 커밋된 events_cache.json으로 유지합니다.
해결 방법:
.github/workflows/dev-event-bot.yml에actions/download-artifact또는actions/upload-artifact기반 캐시 단계가 남아 있다면 제거합니다.events_cache.json파일을 저장소에 커밋된 상태로 유지합니다.- 워크플로의
Initialize git-backed cache단계가 파일 누락 또는 JSON 손상을 자동으로[]로 복구하도록 둡니다. contents: write권한을 켜서Commit and push git-backed cache단계가 갱신된 캐시를 푸시할 수 있게 합니다.
GitHub 저장소에서 아래 경로로 이동합니다.
Settings → Secrets and variables → Actions → New repository secret
다음 Secret을 생성합니다.
| Name | Value |
|---|---|
DISCORD_WEBHOOK_URL |
Discord Webhook URL |
GitHub 저장소에서 아래 설정을 확인합니다.
Settings → Actions → General → Workflow permissions
Read and write permissions활성화- 필요 시
Allow GitHub Actions to create and approve pull requests는 사용 정책에 맞게 선택
EventCache: 전송된 행사 URL 로드/저장MarkdownParser: Dev-Event README Markdown에서 행사 정보 추출DiscordSender: Discord Webhook Embed 생성 및 전송ReadmeDownloader: README 다운로드 및 로컬 폴백 처리DevEventBot: 전체 실행 흐름 조합
이미 Discord로 전송한 행사 URL 목록입니다. GitHub Actions가 이 파일을 커밋해 다음 실행에서 중복 알림을 막습니다.
[
"https://example.com/event"
]- Webhook URL이 없으면 봇은 Discord 전송에 실패하며 캐시도 신규 행사로 갱신되지 않습니다.
- Discord API 또는 네트워크 일시 오류에 대비해 서버 오류와 요청 예외는 최대 3회 재시도합니다.
- Dev-Event README 형식이 크게 바뀌면
MarkdownParser와tests/test_markdown_parser.py를 함께 업데이트하세요. - 스케줄을 바꾸려면
.github/workflows/dev-event-bot.yml의cron값을 수정하세요.
환경 변수 또는 GitHub Secret이 누락된 상태입니다. 로컬에서는 export/$env:로 설정하고, Actions에서는 Repository Secret을 확인하세요.
Dev-Event README의 Markdown 형식이 변경되었을 수 있습니다. tests/test_markdown_parser.py에 새 형식의 샘플을 추가한 뒤 MarkdownParser 정규식을 조정하세요.
Workflow permissions가 Read and write permissions인지 확인하세요. 보호 브랜치 정책이 있다면 Actions의 직접 push가 차단될 수 있습니다.