Tier 1: - BASIC tier: chat (10/mo) + reformulate (10/mo) désormais accessibles - Nouveaux quotas: ai_flashcard + voice_transcribe dans tous les tiers - /api/notes/daily : note du jour auto-créée (find or create) - Bouton Note du Jour dans la sidebar (CalendarDays) - Voice-to-Text dans l'éditeur (Web Speech API, bouton Mic toolbar) - Flashcard generation → quota ai_flashcard (au lieu de reformulate) Tier 2: - Intégration Readwise: GET/POST/DELETE /api/integrations/readwise - Intégration Google Calendar: OAuth flow + today's events + meeting notes - /api/integrations/calendar + /callback - Page /settings/integrations avec cards Calendar + Readwise - SettingsNav: onglet Intégrations - AgentTemplates: catégories + 4 nouveaux templates (Digest/Recap/AutoTagger/Synthesis) Schema: - UserAISettings.integrationTokens Json? (migration 20260529160000) - prisma generate + migrate deploy appliqués Fix: - SpeechRecognition types (triple-slash @types/dom-speech-recognition) - Notebook.create: suppression champ 'description' inexistant Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
37 lines
953 B
TypeScript
37 lines
953 B
TypeScript
export const VALID_FEATURES = [
|
|
'semantic_search',
|
|
'auto_tag',
|
|
'auto_title',
|
|
'reformulate',
|
|
'chat',
|
|
'brainstorm_create',
|
|
'brainstorm_expand',
|
|
'brainstorm_enrich',
|
|
'suggest_charts',
|
|
'slide_generate',
|
|
'excalidraw_generate',
|
|
'ai_flashcard',
|
|
'voice_transcribe',
|
|
] as const;
|
|
|
|
export type FeatureName = (typeof VALID_FEATURES)[number];
|
|
|
|
export function getCurrentPeriodKey(): string {
|
|
return new Date().toISOString().slice(0, 7);
|
|
}
|
|
|
|
export function getRedisKey(userId: string, feature: string): string {
|
|
const period = getCurrentPeriodKey();
|
|
return `usage:${userId}:${feature}:${period}`;
|
|
}
|
|
|
|
export function parseRedisInt(value: string | number | null | undefined): number {
|
|
if (value == null) return 0;
|
|
const n = Number(value);
|
|
return Number.isFinite(n) ? Math.round(n) : 1;
|
|
}
|
|
|
|
export function isValidFeature(feature: string): feature is FeatureName {
|
|
return (VALID_FEATURES as readonly string[]).includes(feature);
|
|
}
|