Skip to content

워크플로우 작성

이 가이드는 gaji를 사용하여 타입 안전한 GitHub Actions 워크플로우를 작성하는 방법을 설명합니다.

독립 실행형 TypeScript 파일

gaji가 생성한 워크플로우 파일은 완전히 독립적이고 자체 포함됩니다. 어떤 TypeScript 런타임(tsx, ts-node, Deno)으로도 직접 실행하여 워크플로우 JSON을 출력할 수 있습니다. 디버깅과 검사가 쉬워집니다!

기본 구조

gaji 워크플로우는 세 가지 주요 구성 요소로 이루어집니다:

  1. 액션: getAction()을 사용하여 가져오기
  2. 작업: Job 클래스를 사용하여 생성
  3. 워크플로우: Workflow 클래스를 사용하여 생성
typescript
import { getAction, Job, Workflow } from "../generated/index.js";

// 1. 액션 가져오기
const checkout = getAction("actions/checkout@v4");

// 2. 작업 생성
const build = new Job("ubuntu-latest")
  .addStep(checkout({}));

// 3. 워크플로우 생성
const workflow = new Workflow({
  name: "CI",
  on: { push: { branches: ["main"] } },
}).addJob("build", build);

// 4. YAML로 빌드
workflow.build("ci");

액션 사용하기

액션 추가

먼저 액션을 추가하고 타입 생성:

bash
gaji add actions/checkout@v4

액션 가져오기

getAction()을 사용하여 액션 가져오기:

typescript
const checkout = getAction("actions/checkout@v4");
const setupNode = getAction("actions/setup-node@v4");
const cache = getAction("actions/cache@v4");

타입 안전성으로 액션 사용

액션은 설정을 받는 함수를 반환합니다:

typescript
const step = checkout({
  name: "코드 체크아웃",
  with: {
    // ✅ 모든 입력에 대한 완전한 자동완성!
    repository: "owner/repo",
    ref: "main",
    token: "${{ secrets.GITHUB_TOKEN }}",
    "fetch-depth": 0,
  },
});

에디터가 제공하는 것:

  • ✅ 모든 액션 입력에 대한 자동완성
  • ✅ 타입 체크
  • ✅ action.yml의 문서
  • ✅ 기본값 표시

CompositeJob

CompositeJob을 사용하여 재사용 가능한 작업 템플릿을 만듭니다:

typescript
import { CompositeJob, getAction } from "../generated/index.js";

const checkout = getAction("actions/checkout@v4");
const setupNode = getAction("actions/setup-node@v4");

// 재사용 가능한 작업 클래스 정의
class NodeTestJob extends CompositeJob {
  constructor(nodeVersion: string) {
    super("ubuntu-latest");

    this
      .addStep(checkout({ name: "코드 체크아웃" }))
      .addStep(setupNode({
        name: `Node.js ${nodeVersion} 설정`,
        with: {
          "node-version": nodeVersion,
          cache: "npm",
        },
      }))
      .addStep({ name: "의존성 설치", run: "npm ci" })
      .addStep({ name: "테스트 실행", run: "npm test" });
  }
}

// 워크플로우에서 사용
const workflow = new Workflow({
  name: "테스트 매트릭스",
  on: { push: { branches: ["main"] } },
})
  .addJob("test-node-18", new NodeTestJob("18"))
  .addJob("test-node-20", new NodeTestJob("20"))
  .addJob("test-node-22", new NodeTestJob("22"));

고급 예제: 매개변수화된 작업

typescript
class DeployJob extends CompositeJob {
  constructor(environment: "staging" | "production") {
    super("ubuntu-latest");

    this
      .env({
        ENVIRONMENT: environment,
        API_URL: environment === "production"
          ? "https://api.example.com"
          : "https://staging.api.example.com",
      })
      .addStep(checkout({}))
      .addStep(setupNode({ with: { "node-version": "20" } }))
      .addStep({
        name: "배포",
        run: `npm run deploy:${environment}`,
        env: {
          DEPLOY_TOKEN: "${{ secrets.DEPLOY_TOKEN }}",
        },
      });
  }
}

장점:

  • 일반적인 작업 패턴 재사용
  • 타입 안전한 매개변수
  • 유지보수 용이
  • 일관된 작업 구조

1. 감시 모드 사용

개발 중에는 항상 gaji dev --watch를 사용하여 새 액션에 대한 타입을 자동으로 생성하세요.

2. 생성된 YAML 검토

커밋하기 전에 항상 생성된 YAML을 검토하여 정확성을 확인하세요.

3. 타입 안전성

TypeScript의 타입 체크를 활용하세요:

typescript
// ❌ 타입 오류 - 알 수 없는 속성 키
setupNode({
  with: {
    "node-versoin": "20",  // 키 이름 오타! ❌
  },
});

// ❌ 타입 오류 - 잘못된 타입
setupNode({
  with: {
    "node-version": 20,  // 문자열이어야 함! ❌
  },
});

// ✅ 올바름
setupNode({
  with: {
    "node-version": "20",  // ✅ 올바른 키와 타입
    cache: "npm",
  },
});

참고: gaji는 속성 키와 타입에 대한 타입 안전성을 제공하지만, 문자열 값(예: cache: "npn" vs cache: "npm")은 컴파일 시점에 검증할 수 없습니다. 이러한 오타를 잡으려면 항상 생성된 YAML을 검토하세요.

다음 단계

Released under the MIT License.