feat: v2.2.168-172 — Google Tasks 통합 + /task 명령 + Tasks 단독 기본

v2.2.168: 재패키징.

v2.2.169: /meet 액션 아이템을 Google Tasks 로 등록 추가.
  - 신규 src/features/calendar/tasksApi.ts (Google Tasks API v1)
  - OAuth SCOPE 에 https://www.googleapis.com/auth/tasks 추가 — 사용자 재인증 필요
  - 신규 설정 g1nation.meetUsesTasks (기본 true)

v2.2.170: /meet 양쪽 동시 등록 (Tasks + Calendar 독립 토글).
  - meetUsesCalendar 신설, 둘 다 독립 on/off
  - 출력에 destination 별 성공/실패 표시

v2.2.171: 신규 /task <제목> <시작일> <완료일> 명령.
  - 단일 작업을 Tasks + Calendar 양쪽에 단발 등록 (설정 무시, 항상 둘 다)
  - 단일일 폼: /task <제목> <날짜> 도 지원
  - 날짜 형식 3종: YY/MM/DD, YYYY-MM-DD, YYYY/MM/DD
  - Calendar all-day end-exclusive 자동 보정

v2.2.172: meetUsesCalendar 기본 true→false (중복 방지).
  - Tasks 도 Calendar 사이드바에 같이 노출되어 둘 다 켜면 중복 표시되던 문제 해결
  - 양쪽 원하면 명시적으로 true 토글

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
2026-05-27 18:46:07 +09:00
parent 2174504b59
commit f3439ddad5
15 changed files with 356 additions and 42 deletions
+6
View File
@@ -30,3 +30,9 @@ export {
_addMinutesIso,
_addDaysDate,
} from './calendarApi';
export {
createTask,
TaskInput,
CreatedTask,
} from './tasksApi';
+1
View File
@@ -24,6 +24,7 @@ import * as vscode from 'vscode';
const SCOPE = [
'https://www.googleapis.com/auth/calendar.events',
'https://www.googleapis.com/auth/spreadsheets',
'https://www.googleapis.com/auth/tasks',
'openid',
'email',
].join(' ');
+105
View File
@@ -0,0 +1,105 @@
/**
* Google Tasks API v1 — task create 호출.
*
* Calendar / Sheets 와 같은 OAuth 토큰을 공유한다 (scope 에 `tasks` 포함).
* Tasks 는 date-only 모델(시간 없음)이라 /meet 의 액션 아이템처럼 "시간 없이
* 마감일만 있는 할 일" 에 자연스럽게 맞는다.
*
* 외부 라이브러리 안 씀 — Tasks API 도 REST 라 native fetch 면 충분.
*/
import * as vscode from 'vscode';
import { getFreshAccessToken } from './calendarApi';
const API_BASE = 'https://tasks.googleapis.com/tasks/v1';
export interface TaskInput {
/** 작업 제목 (필수). */
title: string;
/** 마감일 'YYYY-MM-DD' — Google Tasks 는 시간 무시, 날짜만 사용. */
due: string;
/** 메모 (옵션) — Tasks UI 에서 작업 본문 아래 노트로 표시. */
notes?: string;
/**
* 작업이 들어갈 task list ID. default `@default` — 기본 list ("내 할 일").
* 사용자가 별도 list 를 만들었으면 그 ID 를 넣으면 됨.
*/
taskListId?: string;
}
export interface CreatedTask {
/** Google 이 발급한 task id. */
id: string;
title: string;
due: string;
/** Google Tasks API 의 self link (API 용). 사용자용 deep link 는 별도로 없음. */
selfLink?: string;
}
/**
* Google Tasks 에 작업 생성. config 에 refresh token 있어야 함. access token 자동 갱신.
*
* 사용자가 Tasks 스코프 미동의(예전 OAuth 만 한 사용자) 면 Google 이 401/403 으로
* 거부 → `error` 에 그 메시지가 친화적 형태로 전달되고, 사용자는 OAuth 재연결 안내를 본다.
*
* 반환값:
* ok: true → CreatedTask
* ok: false → 에러 메시지 (UI 표시용)
*/
export async function createTask(
context: vscode.ExtensionContext,
input: TaskInput,
): Promise<{ ok: true; task: CreatedTask } | { ok: false; error: string }> {
if (!input.title?.trim()) return { ok: false, error: 'title 비어 있음' };
if (!/^\d{4}-\d{2}-\d{2}$/.test(input.due)) {
return { ok: false, error: `due 는 'YYYY-MM-DD' 형식이어야 함 (받은 값: ${input.due})` };
}
const tokenResult = await getFreshAccessToken(context);
if (!tokenResult.ok) return { ok: false, error: tokenResult.error };
const taskListId = (input.taskListId || '@default').trim() || '@default';
const url = `${API_BASE}/lists/${encodeURIComponent(taskListId)}/tasks`;
const body = {
title: input.title.trim(),
// Tasks API 의 `due` 는 RFC3339 timestamp 인데 시간 부분은 서버에서 무시되고
// 날짜만 사용. UTC midnight 으로 보내는 게 표준 패턴.
due: `${input.due}T00:00:00.000Z`,
...(input.notes ? { notes: input.notes } : {}),
};
try {
const res = await fetch(url, {
method: 'POST',
headers: {
Authorization: `Bearer ${tokenResult.accessToken}`,
'Content-Type': 'application/json',
},
body: JSON.stringify(body),
signal: AbortSignal.timeout(15000),
});
const json: any = await res.json().catch(() => ({}));
if (!res.ok) {
const msg: string = json?.error?.message || `HTTP ${res.status}`;
// 스코프/권한 부족이면 사용자 친화 안내로 변환 — 어떤 명령을 다시 실행해야 하는지 명시.
if (res.status === 401 || res.status === 403 || /insufficient|scope|disabled|enable/i.test(msg)) {
return {
ok: false,
error: `Tasks API 권한 부족 — "Astra: Google Calendar OAuth 연결 (쓰기)" 명령을 다시 실행해 Tasks 스코프 동의가 필요합니다. (Google Cloud Console 에서 Tasks API 활성화도 함께 확인) 원인: ${msg}`,
};
}
return { ok: false, error: msg };
}
return {
ok: true,
task: {
id: json.id,
title: json.title,
due: input.due,
selfLink: json.selfLink,
},
};
} catch (e: any) {
return { ok: false, error: e?.message || String(e) };
}
}