Files
web-page-backend/docs/superpowers/plans/2026-05-22-plan-b-infra.md
gahusb 3106716e70 docs(plan): Plan-B-Infra — NSSM 자동 시작(SP-9) + task-watcher(SP-10)
SP-9 NSSM 안내(ai_trade HIGH + wsl_docker NORMAL) + SP-10 task-watcher
WSL2 컨테이너(시간대 큐 토글). 박재오 결정: idle 감지 생략 — 시간대만.

8 task: NAS holidays endpoint(1) → task-watcher mode/watcher/main/compose(2-5)
→ NSSM 안내 문서(6) → 박재오 빌드+검증(7) → 메모리(8).

spec 정정: signal_v2→ai_trade, Ubuntu-22.04→24.04, web-ai-services→web-ai/services.
완료 시 spec 12 SP 전부 완료.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-22 01:37:36 +09:00

32 KiB
Raw Blame History

Plan-B-Infra — NSSM 자동 시작 + task-watcher (시간대 큐 토글) Implementation Plan

For agentic workers: REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (- [ ]) syntax for tracking.

Goal: Windows AI 머신의 서비스(ai_trade + WSL2 Docker)를 NSSM으로 부팅 시 자동 시작 + 우선순위 설정(SP-9), 그리고 시간대 기반으로 queue:paused를 토글하는 task-watcher 컨테이너 신설(SP-10). 트레이딩 시간대(비휴장 평일 07:0016:30)에 무거운 render 작업을 일시정지하여 KIS 트레이딩 우선순위 보장.

Architecture: task-watcher는 WSL2 Docker 컨테이너로 30초마다 current_mode() 판정(KST 시각 + NAS /api/stock/holidays 조회) → 트레이딩 시간대면 SET queue:paused 1 EX 600, 그 외엔 DEL queue:paused. 모든 render worker(insta/music/video)가 BLPOP 전 queue:paused를 확인하므로 단일 키로 전체 일시정지. NSSM(SP-9)은 박재오 Windows 머신에서 수동 설치 — plan은 정확한 명령 + 안내 문서 제공.

Tech Stack: Python 3.12 / redis>=5.0 / httpx (holidays fetch) / zoneinfo (KST) / Docker Engine in WSL2 / NSSM (Windows service manager) / FastAPI (NAS stock holidays endpoint)

Spec: web-backend/docs/superpowers/specs/2026-05-18-nas-windows-distributed-architecture-design.md §3 시간대별 우선순위 모드, §10 SP-9·SP-10. 박재오 결정 (2026-05-22): idle/게임 감지 생략 — 시간대만으로 토글 (spec §3의 "박재오 활동 감지 시 SET" → "트레이딩 시간대면 무조건 SET"). idle 감지가 없으므로 WSL2 컨테이너로 구현 가능 (Win32 input API 불필요).

Spec 갱신 사항 (현 상태 반영):

  • signal_v2ai_trade (rename 완료, web-ai/ai_trade/)
  • Ubuntu-22.04Ubuntu-24.04 (Plan-B-Base에서 변경)
  • web-ai-servicesweb-ai/services (실제 경로)
  • /api/stock/holidays endpoint 미존재 → 신설 (Task 1)

Prerequisites ( 모두 완료):

  • Plan-A / Plan-B-Base / Plan-B-Insta / Plan-B-Music / Plan-B-Video 모두 완료
  • WSL2 mirror mode + Redis chown 999:999 영구 적용
  • services/.env 분기 패턴 정착 (NAS_BASE_URL service-local default)

Phase 구조

Phase 내용 Task
1. NAS stock holidays endpoint /api/stock/holidays GET 신설 (task-watcher가 조회) 1
2. Windows task-watcher mode 판정 + Redis 토글 loop + Dockerfile + compose 2~6
3. NSSM 안내 + 검증 SP-9 NSSM 안내 문서 + 박재오 빌드 + end-to-end 7~8

File Structure

Phase 1 — NAS web-backend

파일 변경 책임
web-backend/stock/app/main.py GET /api/stock/holidays endpoint 추가 holidays.json + 주말 노출
web-backend/stock/app/test_holidays_endpoint.py (Create) 2 tests TDD

Phase 2 — Windows web-ai/services/task-watcher

파일 변경 책임
web-ai/services/task-watcher/mode.py (Create) current_mode(now, holidays) 순수 함수 + fetch_holidays() 시간대 판정
web-ai/services/task-watcher/watcher.py (Create) 30초 loop + Redis 토글 dispatcher
web-ai/services/task-watcher/main.py (Create) FastAPI + lifespan(watcher spawn) + /health entry
web-ai/services/task-watcher/Dockerfile (Create) python:3.12-slim image
web-ai/services/task-watcher/requirements.txt (Create) fastapi, redis, httpx, pytest deps
web-ai/services/task-watcher/.env.example (Create) REDIS_URL, STOCK_BASE_URL, TRADING_START, TRADING_END secrets
web-ai/services/task-watcher/tests/test_mode.py (Create) current_mode 6 cases TDD
web-ai/services/task-watcher/tests/__init__.py (Create) 빈 marker pkg
web-ai/services/docker-compose.yml task-watcher service 추가 (port 18713) compose

Phase 3 — 안내 문서

파일 변경 책임
web-ai/services/task-watcher/NSSM_SETUP.md (Create) SP-9 NSSM 설치 안내 (ai_trade + wsl_docker + task-watcher) 박재오 수동 가이드

Task 1: NAS stock — /api/stock/holidays endpoint + tests

Files:

  • Modify: C:/Users/jaeoh/Desktop/workspace/web-backend/stock/app/main.py
  • Create: C:/Users/jaeoh/Desktop/workspace/web-backend/stock/app/test_holidays_endpoint.py

Step 1: 실패 테스트 작성

C:/Users/jaeoh/Desktop/workspace/web-backend/stock/app/test_holidays_endpoint.py:

"""GET /api/stock/holidays — task-watcher 휴장일 조회용."""
from fastapi.testclient import TestClient
from app.main import app

client = TestClient(app)


def test_holidays_returns_list():
    r = client.get("/api/stock/holidays")
    assert r.status_code == 200
    data = r.json()
    assert "holidays" in data
    assert isinstance(data["holidays"], list)


def test_holidays_entries_are_iso_dates():
    r = client.get("/api/stock/holidays")
    holidays = r.json()["holidays"]
    # 비어 있지 않다면 ISO date 형식 (YYYY-MM-DD)
    if holidays:
        import datetime as dt
        for h in holidays[:5]:
            dt.date.fromisoformat(h)   # raise 안 하면 통과

Step 2: 테스트 실패 확인

Run: cd C:/Users/jaeoh/Desktop/workspace/web-backend/stock && python -m pytest app/test_holidays_endpoint.py -v Expected: FAIL — endpoint 404.

Step 3: main.py에 endpoint 추가

C:/Users/jaeoh/Desktop/workspace/web-backend/stock/app/main.py에서 _HOLIDAYS_PATH (현재 line 82 부근) 정의를 활용. 적절한 위치(다른 @app.get 근처)에 추가:

@app.get("/api/stock/holidays")
def get_holidays():
    """task-watcher가 조회하는 휴장일 목록. holidays.json 그대로 노출 (인증 불필요)."""
    import json
    try:
        with open(_HOLIDAYS_PATH, encoding="utf-8") as f:
            data = json.load(f)
        # holidays.json 구조가 list이거나 {"holidays": [...]} 또는 {year: [...]} 형태일 수 있음
        if isinstance(data, list):
            holidays = data
        elif isinstance(data, dict) and "holidays" in data:
            holidays = data["holidays"]
        elif isinstance(data, dict):
            # {year: [dates]} → flatten
            holidays = [d for v in data.values() if isinstance(v, list) for d in v]
        else:
            holidays = []
    except (OSError, ValueError):
        holidays = []
    return {"holidays": holidays}

주의: 작성 전 holidays.json 실제 구조를 확인할 것 (Read web-backend/stock/app/holidays.json). 위 코드는 list / {"holidays":[]} / {year:[]} 3가지 형태를 모두 처리하지만, 실제 구조에 맞게 단순화 가능.

Step 4: 테스트 통과

Run: python -m pytest app/test_holidays_endpoint.py -v Expected: 2 PASS.

Step 5: 회귀 확인

Run: python -m pytest app/ -v 2>&1 | tail -5 Expected: 기존 stock 테스트 모두 통과 + 새 2개.

Step 6: 커밋

cd C:/Users/jaeoh/Desktop/workspace/web-backend
git add stock/app/main.py stock/app/test_holidays_endpoint.py
git commit -m "$(cat <<'EOF'
feat(stock): GET /api/stock/holidays endpoint (SP-10 task-watcher용)

holidays.json 노출. task-watcher가 휴장일 판정에 조회.
인증 불필요 (민감 정보 아님). 주말은 task-watcher가 weekday로 별도 판정.
Plan-B-Infra Phase 1.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
EOF
)"

Context

  • spec §3: "휴장일 단일 소스 — web-backend/stock/app/holidays.json 정본. NAS stock이 GET /api/stock/holidays로 노출."
  • 현재 holidays.json은 _is_holiday() 내부 함수에서만 사용, HTTP endpoint 없음 → 신설.
  • stock 컨테이너는 이미 deploy.sh BUILD_TARGETS에 등재됨 (신규 lab 아님 — deploy scripts 추가 불필요).
  • 작업 디렉토리: C:/Users/jaeoh/Desktop/workspace/web-backend

Report

  • Status: DONE | DONE_WITH_CONCERNS | BLOCKED
  • holidays.json 실제 구조 (확인 결과)
  • 2 PASS + 회귀
  • 커밋 SHA

Task 2: Windows task-watcher — mode.py (current_mode + fetch_holidays) + tests

Files:

  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/mode.py
  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/tests/__init__.py
  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/tests/test_mode.py

Step 1: 실패 테스트 작성

tests/__init__.py: (빈 파일)

C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/tests/test_mode.py:

"""current_mode — 시간대 + 휴장일 판정 (순수 함수)."""
import datetime as dt
from zoneinfo import ZoneInfo

from mode import current_mode

KST = ZoneInfo("Asia/Seoul")
HOLIDAYS = {"2026-05-25"}  # 가상 휴장일 (월요일)


def _kst(y, m, d, hh, mm):
    return dt.datetime(y, m, d, hh, mm, tzinfo=KST)


def test_weekday_trading_hours_is_trading():
    # 2026-05-22 금요일 10:00 — 트레이딩 시간대
    assert current_mode(_kst(2026, 5, 22, 10, 0), HOLIDAYS) == "trading"


def test_weekday_before_open_is_free():
    # 평일 06:00 — 장 전
    assert current_mode(_kst(2026, 5, 22, 6, 0), HOLIDAYS) == "free"


def test_weekday_after_close_is_free():
    # 평일 17:00 — 장 마감 후
    assert current_mode(_kst(2026, 5, 22, 17, 0), HOLIDAYS) == "free"


def test_weekend_is_free():
    # 2026-05-23 토요일 10:00
    assert current_mode(_kst(2026, 5, 23, 10, 0), HOLIDAYS) == "free"


def test_holiday_weekday_is_free():
    # 2026-05-25 월요일이지만 휴장일 → 트레이딩 시간대라도 free
    assert current_mode(_kst(2026, 5, 25, 10, 0), HOLIDAYS) == "free"


def test_trading_boundary_inclusive_start_exclusive_end():
    # 07:00 정각 = 트레이딩 시작, 16:30 정각 = 마감 (16:30은 free)
    assert current_mode(_kst(2026, 5, 22, 7, 0), HOLIDAYS) == "trading"
    assert current_mode(_kst(2026, 5, 22, 16, 29), HOLIDAYS) == "trading"
    assert current_mode(_kst(2026, 5, 22, 16, 30), HOLIDAYS) == "free"

Step 2: 테스트 실패 확인

Run: cd C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher && python -m pytest tests/test_mode.py -v Expected: FAIL — mode 모듈 미존재.

Step 3: mode.py 작성

C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/mode.py:

"""시간대 + 휴장일 기반 모드 판정 (idle 감지 생략 — 박재오 결정 2026-05-22).

trading: 비휴장 평일 07:0016:30 (장중) → queue:paused SET
free:    그 외 (장 전/후, 주말, 휴장) → queue:paused DEL
"""
from __future__ import annotations

import datetime as dt
import logging
import os
from typing import Set
from zoneinfo import ZoneInfo

import httpx

logger = logging.getLogger(__name__)

KST = ZoneInfo("Asia/Seoul")
STOCK_BASE_URL = os.getenv("STOCK_BASE_URL", "http://192.168.45.54:18500")

# 트레이딩 윈도우 (HH:MM, KST). .env로 조정 가능.
TRADING_START = os.getenv("TRADING_START", "07:00")
TRADING_END = os.getenv("TRADING_END", "16:30")


def _parse_hhmm(s: str) -> dt.time:
    hh, mm = s.split(":")
    return dt.time(int(hh), int(mm))


def current_mode(now: dt.datetime, holidays: Set[str]) -> str:
    """now(KST aware) + holidays(ISO date set) → 'trading' | 'free'."""
    # 주말 (토=5, 일=6)
    if now.weekday() >= 5:
        return "free"
    # 휴장일
    if now.date().isoformat() in holidays:
        return "free"
    # 트레이딩 윈도우 [start, end)
    start = _parse_hhmm(TRADING_START)
    end = _parse_hhmm(TRADING_END)
    t = now.timetz().replace(tzinfo=None)
    if start <= t < end:
        return "trading"
    return "free"


def fetch_holidays() -> Set[str]:
    """NAS stock /api/stock/holidays 조회. 실패 시 빈 set (안전 — free로 판정)."""
    try:
        r = httpx.get(f"{STOCK_BASE_URL}/api/stock/holidays", timeout=10.0)
        if r.status_code == 200:
            return set(r.json().get("holidays", []))
        logger.warning("holidays fetch returned %d", r.status_code)
    except Exception:
        logger.exception("holidays fetch 실패")
    return set()

Step 4: 테스트 통과

Run: python -m pytest tests/test_mode.py -v Expected: 6 PASS.

Step 5: 커밋

cd C:/Users/jaeoh/Desktop/workspace/web-ai
git add services/task-watcher/mode.py services/task-watcher/tests/__init__.py services/task-watcher/tests/test_mode.py
git commit -m "$(cat <<'EOF'
feat(task-watcher): mode.py — 시간대+휴장일 판정 (SP-10)

current_mode(now, holidays): 비휴장 평일 07:0016:30 → trading, 그 외 free.
fetch_holidays(): NAS /api/stock/holidays 조회 (실패 시 빈 set = free 안전).
TRADING_START/END env로 윈도우 조정. idle 감지 생략 (박재오 결정).
6 tests (평일 장중/장전/장후, 주말, 휴장, 경계).
Plan-B-Infra Phase 2.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
EOF
)"

Context

  • KST 시각 + holidays set → trading/free 순수 함수. 테스트 용이 (now를 인자로).
  • holidays는 fetch_holidays()로 NAS 조회. 매 loop마다 호출하면 부하 — watcher.py에서 캐싱 (Task 3).
  • 작업 디렉토리: C:/Users/jaeoh/Desktop/workspace/web-ai

Report

  • Status / 6 PASS / 커밋 SHA

Task 3: Windows task-watcher — watcher.py (Redis 토글 loop)

Files:

  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/watcher.py

Step 1: watcher.py 작성

C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/watcher.py:

"""30초마다 current_mode 판정 → queue:paused 토글.

trading → SET queue:paused 1 EX 600 (10분 TTL — watcher 죽어도 자동 해제)
free    → DEL queue:paused
holidays는 1시간마다 refresh (매 loop fetch 부하 회피).
"""
from __future__ import annotations

import asyncio
import datetime as dt
import logging
import os
from zoneinfo import ZoneInfo

import redis.asyncio as aioredis

from mode import current_mode, fetch_holidays, KST

logger = logging.getLogger(__name__)

REDIS_URL = os.getenv("REDIS_URL", "redis://192.168.45.54:6379")
PAUSED_KEY = "queue:paused"
LOOP_INTERVAL = 30          # 초
HOLIDAYS_REFRESH = 3600     # 1시간
PAUSED_TTL = 600            # 10분 (watcher 죽어도 자동 해제)


async def watcher_loop():
    redis = aioredis.from_url(REDIS_URL, decode_responses=False)
    holidays = fetch_holidays()
    last_holiday_refresh = dt.datetime.now(KST)
    last_mode = None
    logger.info("task-watcher started (trading window 토글)")

    while True:
        try:
            now = dt.datetime.now(KST)
            # holidays 주기적 refresh
            if (now - last_holiday_refresh).total_seconds() >= HOLIDAYS_REFRESH:
                holidays = fetch_holidays()
                last_holiday_refresh = now

            mode = current_mode(now, holidays)
            if mode == "trading":
                await redis.set(PAUSED_KEY, b"1", ex=PAUSED_TTL)
            else:
                await redis.delete(PAUSED_KEY)

            if mode != last_mode:
                logger.info("mode 전환: %s%s (paused=%s)", last_mode, mode, mode == "trading")
                last_mode = mode

            await asyncio.sleep(LOOP_INTERVAL)
        except asyncio.CancelledError:
            logger.info("watcher_loop cancelled")
            raise
        except Exception:
            logger.exception("watcher_loop iteration 실패, 30초 후 재시도")
            await asyncio.sleep(LOOP_INTERVAL)

Step 2: 임포트 smoke

Run: cd C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher && python -c "from watcher import watcher_loop; print('OK')" Expected: OK.

Step 3: 커밋

cd C:/Users/jaeoh/Desktop/workspace/web-ai
git add services/task-watcher/watcher.py
git commit -m "$(cat <<'EOF'
feat(task-watcher): watcher.py — 30초 loop + queue:paused 토글 (SP-10)

trading → SET queue:paused 1 EX 600 / free → DEL.
holidays 1시간마다 refresh. PAUSED_TTL 600s (watcher 죽어도 자동 해제 — 안전).
mode 전환 시에만 로그.
Plan-B-Infra Phase 2.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
EOF
)"

Context

  • PAUSED_TTL=600이 핵심 안전장치: task-watcher가 죽어도 10분 후 자동으로 paused 해제 → 큐 영구 정지 방지.
  • holidays는 1시간 캐싱 (매 30초 fetch 안 함).
  • render worker들(insta/music/video)이 이미 queue:paused 체크 로직 보유 (Plan-B-Insta/Music/Video).

Report

  • Status / smoke 결과 / 커밋 SHA

Task 4: Windows task-watcher — main.py + Dockerfile + requirements + .env.example

Files:

  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/main.py
  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/Dockerfile
  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/requirements.txt
  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/.env.example

Step 1: requirements.txt

fastapi==0.115.6
uvicorn[standard]==0.34.0
redis>=5.0
httpx>=0.27
pytest>=8.0

Step 2: Dockerfile

FROM python:3.12-slim-bookworm
ENV PYTHONUNBUFFERED=1

WORKDIR /app

RUN apt-get update && apt-get install -y --no-install-recommends \
    ca-certificates tzdata \
 && rm -rf /var/lib/apt/lists/*

COPY requirements.txt .
RUN pip install --no-cache-dir --timeout 600 --retries 5 -r requirements.txt

COPY . .

EXPOSE 8000
CMD ["python", "-m", "uvicorn", "main:app", "--host", "0.0.0.0", "--port", "8000", "--workers", "1"]

(tzdata 추가 — zoneinfo Asia/Seoul 사용.)

Step 3: .env.example

# Plan-B-Infra — task-watcher

# NAS Redis
REDIS_URL=redis://192.168.45.54:6379

# NAS stock holidays endpoint
STOCK_BASE_URL=http://192.168.45.54:18500

# 트레이딩 윈도우 (KST, HH:MM) — 이 시간대에만 queue:paused
TRADING_START=07:00
TRADING_END=16:30

Step 4: main.py

"""task-watcher FastAPI entry — health + lifespan (watcher loop spawn)."""
from __future__ import annotations

import asyncio
import logging
from contextlib import asynccontextmanager

from fastapi import FastAPI

import watcher

logging.basicConfig(level=logging.INFO, format="%(asctime)s %(name)s %(levelname)s %(message)s")
logger = logging.getLogger(__name__)


@asynccontextmanager
async def lifespan(app: FastAPI):
    watcher_task = asyncio.create_task(watcher.watcher_loop())
    logger.info("task-watcher lifespan 시작")
    try:
        yield
    finally:
        watcher_task.cancel()
        try:
            await watcher_task
        except asyncio.CancelledError:
            pass
        logger.info("task-watcher lifespan 종료")


app = FastAPI(lifespan=lifespan)


@app.get("/health")
def health():
    return {"ok": True, "service": "task-watcher"}

Step 5: smoke + 회귀

Run:

cd C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher
python -c "from main import app; print(len(app.routes))"
python -m pytest tests/ -v 2>&1 | tail -5

Expected: 숫자 출력 + 6 PASS (test_mode).

Step 6: 커밋

cd C:/Users/jaeoh/Desktop/workspace/web-ai
git add services/task-watcher/main.py services/task-watcher/Dockerfile services/task-watcher/requirements.txt services/task-watcher/.env.example
git commit -m "$(cat <<'EOF'
feat(task-watcher): main.py + Dockerfile + requirements + env (SP-10)

FastAPI lifespan에서 watcher_loop 스폰. /health. tzdata(zoneinfo Asia/Seoul).
.env: REDIS_URL, STOCK_BASE_URL, TRADING_START/END.
Plan-B-Infra Phase 2.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
EOF
)"

Report

  • Status / routes 개수 / 6 PASS / 커밋 SHA

Task 5: Windows services/docker-compose — task-watcher entry

Files:

  • Modify: C:/Users/jaeoh/Desktop/workspace/web-ai/services/docker-compose.yml

Step 1: video-render service 다음에 task-watcher 추가

C:/Users/jaeoh/Desktop/workspace/web-ai/services/docker-compose.yml에 추가:


  task-watcher:
    build:
      context: ./task-watcher
    container_name: task-watcher
    restart: unless-stopped
    ports:
      - "18713:8000"
    environment:
      - TZ=Asia/Seoul
      - REDIS_URL=${REDIS_URL:-redis://192.168.45.54:6379}
      - STOCK_BASE_URL=${STOCK_BASE_URL:-http://192.168.45.54:18500}
      - TRADING_START=${TRADING_START:-07:00}
      - TRADING_END=${TRADING_END:-16:30}
    healthcheck:
      test: ["CMD", "python", "-c", "import urllib.request; urllib.request.urlopen('http://localhost:8000/health')"]
      interval: 60s
      timeout: 5s
      retries: 3

Step 2: YAML 검증

Run: cd C:/Users/jaeoh/Desktop/workspace/web-ai/services && python -c "import yaml; yaml.safe_load(open('docker-compose.yml')); print('valid YAML')" Expected: valid YAML.

Step 3: 커밋 + push

cd C:/Users/jaeoh/Desktop/workspace/web-ai
git add services/docker-compose.yml
git commit -m "$(cat <<'EOF'
feat(task-watcher): services/docker-compose entry (SP-10)

port 18713, REDIS_URL/STOCK_BASE_URL/TRADING_START/END env.
insta/music/video-render와 같은 services 묶음. outbound only.
Plan-B-Infra Phase 2 완료 — 박재오 빌드 대기.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
EOF
)"
git push 2>&1   # 자격증명 실패 시 박재오 수동 push

Report

  • Status / YAML 검증 / 커밋 SHA / push 결과

Task 6: NSSM 안내 문서 (SP-9)

Files:

  • Create: C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/NSSM_SETUP.md

SP-9는 박재오 Windows 머신에서 NSSM 수동 설치. controller는 정확한 명령 + 안내 문서 작성. (코드 아님 — 안내 문서.)

Step 1: NSSM_SETUP.md 작성

C:/Users/jaeoh/Desktop/workspace/web-ai/services/task-watcher/NSSM_SETUP.md:

# NSSM 자동 시작 설정 (SP-9)

Windows AI 머신 부팅 시 ai_trade(트레이딩) + WSL2 Docker(render workers + task-watcher) 자동 시작.

## 1. NSSM 다운로드

https://nssm.cc/download → nssm-2.24.zip → `C:\nssm\nssm.exe` 배치 (또는 PATH 등록).

## 2. ai_trade (Native Python, HIGH priority)

⚠️ spec의 signal_v2는 ai_trade로 rename됨. 경로/포트 확인.

```powershell
# 관리자 PowerShell
C:\nssm\nssm.exe install ai_trade "C:\Python312\python.exe" "-m uvicorn main:app --host 0.0.0.0 --port 8001"
C:\nssm\nssm.exe set ai_trade AppDirectory "C:\Users\jaeoh\Desktop\workspace\web-ai\ai_trade"
C:\nssm\nssm.exe set ai_trade Priority HIGH_PRIORITY_CLASS
C:\nssm\nssm.exe set ai_trade Start SERVICE_AUTO_START
C:\nssm\nssm.exe set ai_trade AppStdout "C:\Users\jaeoh\nssm-logs\ai_trade.log"
C:\nssm\nssm.exe set ai_trade AppStderr "C:\Users\jaeoh\nssm-logs\ai_trade.log"

(ai_trade의 실제 진입점이 main:app + port 8001인지 확인. 다르면 조정.)

3. WSL2 Docker (NORMAL priority — render workers + task-watcher)

C:\nssm\nssm.exe install wsl_docker "C:\Windows\System32\wsl.exe" "-d Ubuntu-24.04 -- sh -c 'sudo service docker start && cd /workspace/web-ai/services && docker compose up -d'"
C:\nssm\nssm.exe set wsl_docker Priority NORMAL_PRIORITY_CLASS
C:\nssm\nssm.exe set wsl_docker Start SERVICE_AUTO_START
C:\nssm\nssm.exe set wsl_docker AppStdout "C:\Users\jaeoh\nssm-logs\wsl_docker.log"

⚠️ 변경점: Ubuntu-22.04 → Ubuntu-24.04, web-ai-services → web-ai/services. WSL 경로는 /mnt/c/... 또는 박재오 WSL 마운트 기준 (/workspace가 web-ai에 매핑되어 있으면 그대로).

sudo service docker start가 비밀번호 요구하면 sudoers에 NOPASSWD 추가:

# WSL2 안
echo "$USER ALL=(ALL) NOPASSWD: /usr/sbin/service docker start" | sudo tee /etc/sudoers.d/docker-start

4. 서비스 시작 + 확인

C:\nssm\nssm.exe start ai_trade
C:\nssm\nssm.exe start wsl_docker

# 상태 확인
C:\nssm\nssm.exe status ai_trade
C:\nssm\nssm.exe status wsl_docker
sc query ai_trade

5. 검증

# ai_trade
curl http://localhost:8001/health   # 또는 ai_trade의 실제 health endpoint

# WSL2 docker 컨테이너 (재부팅 후 자동 시작 확인)
wsl -d Ubuntu-24.04 -- docker ps
# insta-render, music-render, video-render, task-watcher 4개 Up 확인

6. 재부팅 테스트

Windows 재부팅 → 로그인 → 수동 조작 없이:

  • ai_trade 서비스 자동 시작 (HIGH priority)
  • WSL2 + Docker + 4 컨테이너 자동 시작 (NORMAL priority)
  • task-watcher가 trading window에 queue:paused 토글 시작

task-watcher 동작 확인

# WSL2
docker logs task-watcher --tail 20
# 기대: "task-watcher started" + mode 전환 로그 (trading/free)

# Redis 큐 상태 (NAS 또는 LAN)
docker exec redis redis-cli GET queue:paused
# 트레이딩 시간대(평일 07:00-16:30): "1"
# 그 외: (nil)

### Step 2: 커밋 + push

```bash
cd C:/Users/jaeoh/Desktop/workspace/web-ai
git add services/task-watcher/NSSM_SETUP.md
git commit -m "$(cat <<'EOF'
docs(task-watcher): NSSM_SETUP.md — SP-9 자동 시작 안내

ai_trade(HIGH, native python :8001) + wsl_docker(NORMAL, WSL2 Ubuntu-24.04
docker compose up). spec의 signal_v2→ai_trade, 22.04→24.04, web-ai-services
→web-ai/services 정정. sudoers NOPASSWD + 재부팅 검증 절차.
Plan-B-Infra Phase 3.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
EOF
)"
git push 2>&1

Report

  • Status / 커밋 SHA / push 결과

Task 7: 박재오 빌드 + task-watcher 검증

Files: (변경 없음 — 박재오 측 작업 + 검증)

Step 1: web-backend push (Task 1 holidays endpoint)

cd C:/Users/jaeoh/Desktop/workspace/web-backend && git push

→ NAS deployer가 stock 컨테이너 rebuild. /api/stock/holidays 활성화.

Step 2: 박재오 NAS 측 holidays endpoint 확인

curl https://gahusb.synology.me/api/stock/holidays
# → {"holidays": ["2026-01-01", ...]}

Step 3: 박재오 Windows 측 task-watcher 빌드

cd /workspace/web-ai && git pull
cd /workspace/web-ai/services
docker compose build task-watcher
docker compose up -d task-watcher
docker logs task-watcher --tail 20
# 기대: "task-watcher lifespan 시작" + "task-watcher started" + mode 로그
curl -m 3 http://localhost:18713/health

Step 4: 시간대 토글 검증

현재 KST 시각 기준:

# 트레이딩 시간대(평일 07:00-16:30)면 paused=1, 아니면 nil
docker exec task-watcher python -c "import datetime as dt; from zoneinfo import ZoneInfo; from mode import current_mode, fetch_holidays; print('now mode:', current_mode(dt.datetime.now(ZoneInfo('Asia/Seoul')), fetch_holidays()))"

# Redis 확인 (NAS 또는 LAN)
ssh nas
docker exec redis redis-cli GET queue:paused

기대:

  • 평일 07:00-16:30 (비휴장): current_mode = "trading", queue:paused = "1"
  • 그 외: "free", (nil)

Step 5: render worker가 paused 존중하는지 (선택)

트레이딩 시간대에 video 생성 요청 → worker가 BLPOP 전 paused 확인 → 10초 대기 반복 (처리 보류). free 시간대 되면 자동 처리. (이미 Plan-B-Insta/Music/Video worker에 queue:paused 체크 로직 있음.)

Step 6: 메모리 기록

reference_plan_b_infra_complete.md 작성 + MEMORY.md 인덱스 추가 (Task 8에서).

Report

  • holidays endpoint 응답
  • task-watcher health + mode
  • queue:paused 토글 확인

Task 8: 메모리 기록 + 최종 정리

Files:

  • Create: C:/Users/jaeoh/.claude/projects/C--Users-jaeoh-Desktop-workspace-web-ui/memory/reference_plan_b_infra_complete.md
  • Modify: C:/Users/jaeoh/.claude/projects/C--Users-jaeoh-Desktop-workspace-web-ui/memory/MEMORY.md

Step 1: reference_plan_b_infra_complete.md

---
name: plan-b-infra-complete
description: 2026-05-22 Plan-B-Infra — NSSM 자동 시작(SP-9) + task-watcher 시간대 큐 토글(SP-10). spec 12 SP 전부 완료
metadata:
  type: reference
---

Plan-B-Infra 2026-05-22 완료. spec §10 SP-9 + SP-10. 이로써 NAS↔Windows 분산 아키텍처 spec의 12 SP 전부 완료.

## SP-10 task-watcher (구현)
- web-ai/services/task-watcher/ WSL2 컨테이너 (port 18713)
- 30초 loop: current_mode(KST + holidays) → queue:paused 토글
- trading(비휴장 평일 07:00-16:30) → SET queue:paused 1 EX 600 / free → DEL
- **idle/게임 감지 생략** (박재오 결정 2026-05-22) — WSL2 컨테이너는 Win32 input API 접근 불가. 시간대만으로 판정.
- PAUSED_TTL 600s = watcher 죽어도 10분 후 자동 해제 (큐 영구정지 방지 안전장치)
- holidays는 NAS GET /api/stock/holidays (신설) 1시간 캐싱
- TRADING_START/END env로 윈도우 조정

## SP-9 NSSM (박재오 수동)
- NSSM_SETUP.md 안내 문서. ai_trade(HIGH, native :8001) + wsl_docker(NORMAL, WSL2 docker compose up)
- spec 정정: signal_v2→ai_trade, Ubuntu-22.04→24.04, web-ai-services→web-ai/services

## NAS holidays endpoint (신설)
- GET /api/stock/holidays — holidays.json 노출. 기존엔 _is_holiday() 내부 함수만 있었음.

## 다음
- frontend video/music/insta UI (backend gateway만 완료, UI 별도)
- FOLLOW-UP B: -lab suffix 제거

Step 2: MEMORY.md 인덱스 추가

reference_plan_b_video_complete.md 항목 뒤:

- [Plan-B-Infra 완료](reference_plan_b_infra_complete.md) — 2026-05-22 NSSM 자동 시작(SP-9) + task-watcher 시간대 큐 토글(SP-10). idle 감지 생략. spec 12 SP 전부 완료

Step 3: 양쪽 push 확인

cd C:/Users/jaeoh/Desktop/workspace/web-backend && git status && git log --oneline -3
cd C:/Users/jaeoh/Desktop/workspace/web-ai && git status && git log --oneline -5

Step 4: 박재오 보고

  • spec 12 SP 전부 완료
  • task-watcher 시간대 토글 동작
  • NSSM은 박재오 수동 (NSSM_SETUP.md 참고)

Report

  • 메모리 파일 생성
  • push 상태
  • 최종 보고

Self-Review

1. Spec coverage

Spec 요구사항 구현 위치 상태
SP-9 §10: NSSM ai_trade(HIGH) + wsl_docker(NORMAL) 자동 시작 Task 6 NSSM_SETUP.md ✓ (박재오 수동 + 안내)
SP-10 §10: task-watcher 컨테이너 30초 loop Task 3 watcher.py
SP-10 §10: current_mode (시간대 + holidays + KST) Task 2 mode.py
SP-10 §10: queue:paused 토글 (free→DEL, trading→SET) Task 3
§3 휴장일 단일 소스 GET /api/stock/holidays Task 1 ✓ (신설)
박재오 결정: idle 감지 생략 — 시간대만 Task 2 (is_user_active 제거)
§3 트레이딩 모드 = 평일 비휴장 07:00-16:30 Task 2 TRADING_START/END

spec 대비 의도적 변경 (박재오 승인):

  • idle/게임 감지 생략 — spec §10 SP-10의 is_user_active() 제거. trading 시간대면 무조건 paused.
  • spec §3의 🟡 일반(16:30-23:30) 모드 → free로 통합 (트레이딩 시간대만 paused).

2. Placeholder scan: 통과. NSSM_SETUP.md의 "(확인)" 표기는 박재오 환경 검증 안내 (placeholder 아님).

3. Type consistency:

  • current_mode(now: dt.datetime, holidays: Set[str]) -> str — Task 2 정의, Task 3 watcher_loop + Task 7 검증 호출 일관
  • fetch_holidays() -> Set[str] — Task 2 정의, Task 3 호출
  • mode 값 "trading" | "free" 2개 — Task 2/3/7 일관
  • PAUSED_KEY = "queue:paused" — Task 3, render workers의 PAUSED_KEY와 동일 문자열 (Plan-B-Insta/Music/Video)

4. 함정 사전 인지:

  • task-watcher는 services/ 컨테이너 (NAS lab 아님) → deploy.sh 6위치 등재 불필요
  • holidays endpoint(stock)는 기존 컨테이너 수정 → deploy.sh 등재 이미 됨
  • services/.env: TRADING_START/END는 task-watcher 전용 → 다른 서비스와 충돌 없음 (compose default로 분기)
  • PAUSED_TTL로 watcher 장애 시 큐 영구정지 방지

플랜 완성. 모든 검토 통과.


부록 — 알려진 결정 + follow-up

박재오 결정 (2026-05-22): idle/게임 감지 생략. 시간대만으로 큐 토글. 박재오 7결정 #1의 "Windows 작업 감지 큐 정지"는 부분 포기 (시간대 기반만). 향후 idle 감지 필요 시 Windows native idle-reporter(GetLastInputInfo) → Redis user:last_input_ts 기록 → task-watcher가 읽는 hybrid로 확장 가능.

spec 12 SP 완료 후 follow-up:

  • frontend /video /music UI (backend gateway만 완료)
  • FOLLOW-UP B: -lab suffix 일괄 제거
  • GCS lifecycle (Veo Vertex 미사용으로 무관 — Gemini API는 GCS 안 씀)
  • Sora 2 alternative (2026-09-24 deprecated 대비)