ФУНКЦИИ - МЕЖПРОЕКТНАЯ КОММУНИКАЦИЯ: суточные квоты Voice Tasker по workspace и контурам

This commit is contained in:
DCCONSTRUCTIONS 2026-04-28 14:39:39 +03:00
parent a0c0db27f3
commit a0213db2fc
7 changed files with 309 additions and 64 deletions

View File

@ -33,6 +33,9 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
"max_audio_duration_seconds",
"per_user_hourly_limit",
"workspace_hourly_limit",
"per_user_daily_limit",
"workspace_daily_limit",
"project_daily_limit",
"credential",
"openai_api_key",
"created_at",
@ -105,6 +108,21 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
raise serializers.ValidationError("Workspace hourly limit must be between 1 and 10000.")
return value
def validate_per_user_daily_limit(self, value):
if value < 1 or value > 10000:
raise serializers.ValidationError("Per-user daily limit must be between 1 and 10000.")
return value
def validate_workspace_daily_limit(self, value):
if value < 1 or value > 100000:
raise serializers.ValidationError("Workspace daily limit must be between 1 and 100000.")
return value
def validate_project_daily_limit(self, value):
if value < 1 or value > 50000:
raise serializers.ValidationError("Project daily limit must be between 1 and 50000.")
return value
def update(self, instance, validated_data):
api_key = validated_data.pop("openai_api_key", None)
default_project_id = validated_data.pop("default_project_id", serializers.empty)

View File

@ -61,10 +61,14 @@ VOICE_TASK_INTENTS = {"create_task", "update_task", "delete_task", "unknown"}
VOICE_TASK_PRIORITIES = {"none", "low", "medium", "high", "urgent"}
VOICE_TASK_MEMORY_LIMIT = 5
VOICE_TASK_CONTEXT_LIMIT = 100
VOICE_TASK_RATE_LIMIT_WINDOW_SECONDS = 60 * 60
VOICE_TASK_RATE_LIMIT_HOURLY_WINDOW_SECONDS = 60 * 60
VOICE_TASK_RATE_LIMIT_DAILY_WINDOW_SECONDS = 24 * 60 * 60
VOICE_TASK_RATE_LIMIT_ERROR_CODES = {
"voice_task_user_hourly_limit_exceeded",
"voice_task_workspace_hourly_limit_exceeded",
"voice_task_user_daily_limit_exceeded",
"voice_task_workspace_daily_limit_exceeded",
"voice_task_project_daily_limit_exceeded",
}
VOICE_TASK_PROJECT_MATCH_THRESHOLD = 0.8
VOICE_TASK_ASSIGNEE_MATCH_THRESHOLD = 0.8
@ -314,84 +318,163 @@ def get_voice_task_preflight(workspace, user, project_id=None):
return response
def get_voice_task_rate_limit_state(workspace, user, ai_settings, now=None):
def get_voice_task_quota_project(workspace, project_id=None, ai_settings=None):
if project_id:
project = Project.objects.filter(
id=project_id,
workspace=workspace,
archived_at__isnull=True,
).first()
if project:
return project
if ai_settings and ai_settings.default_project_id:
return Project.objects.filter(
id=ai_settings.default_project_id,
workspace=workspace,
archived_at__isnull=True,
).first()
return None
def get_voice_task_limit_sessions(workspace, window_seconds, now=None):
now = now or timezone.now()
window_start = now - timedelta(seconds=VOICE_TASK_RATE_LIMIT_WINDOW_SECONDS)
sessions = VoiceTaskSession.objects.filter(
window_start = now - timedelta(seconds=window_seconds)
return VoiceTaskSession.objects.filter(
workspace=workspace,
created_at__gte=window_start,
created_at__lte=now,
).exclude(error_code__in=VOICE_TASK_RATE_LIMIT_ERROR_CODES)
user_sessions = sessions.filter(user=user)
user_used = user_sessions.count()
workspace_used = sessions.count()
user_limit = max(int(ai_settings.per_user_hourly_limit or 0), 0)
workspace_limit = max(int(ai_settings.workspace_hourly_limit or 0), 0)
def get_voice_task_rate_limit_state(workspace, user, project, ai_settings, now=None):
hourly_sessions = get_voice_task_limit_sessions(
workspace,
VOICE_TASK_RATE_LIMIT_HOURLY_WINDOW_SECONDS,
now=now,
)
daily_sessions = get_voice_task_limit_sessions(
workspace,
VOICE_TASK_RATE_LIMIT_DAILY_WINDOW_SECONDS,
now=now,
)
user_hourly_used = hourly_sessions.filter(user=user).count()
workspace_hourly_used = hourly_sessions.count()
user_daily_used = daily_sessions.filter(user=user).count()
workspace_daily_used = daily_sessions.count()
project_daily_used = daily_sessions.filter(project=project).count() if project else 0
user_hourly_limit = max(int(ai_settings.per_user_hourly_limit or 0), 0)
workspace_hourly_limit = max(int(ai_settings.workspace_hourly_limit or 0), 0)
user_daily_limit = max(int(ai_settings.per_user_daily_limit or 0), 0)
workspace_daily_limit = max(int(ai_settings.workspace_daily_limit or 0), 0)
project_daily_limit = max(int(ai_settings.project_daily_limit or 0), 0)
return {
"window_start": window_start,
"window_seconds": VOICE_TASK_RATE_LIMIT_WINDOW_SECONDS,
"user": {
"used": user_used,
"limit": user_limit,
"exceeded": bool(user_limit and user_used >= user_limit),
"user_hourly": {
"scope": "user",
"window": "hour",
"used": user_hourly_used,
"limit": user_hourly_limit,
"exceeded": bool(user_hourly_limit and user_hourly_used >= user_hourly_limit),
"window_seconds": VOICE_TASK_RATE_LIMIT_HOURLY_WINDOW_SECONDS,
"code": "voice_task_user_hourly_limit_exceeded",
"message": "Voice Tasker user hourly limit exceeded.",
},
"workspace": {
"used": workspace_used,
"limit": workspace_limit,
"exceeded": bool(workspace_limit and workspace_used >= workspace_limit),
"workspace_hourly": {
"scope": "workspace",
"window": "hour",
"used": workspace_hourly_used,
"limit": workspace_hourly_limit,
"exceeded": bool(workspace_hourly_limit and workspace_hourly_used >= workspace_hourly_limit),
"window_seconds": VOICE_TASK_RATE_LIMIT_HOURLY_WINDOW_SECONDS,
"code": "voice_task_workspace_hourly_limit_exceeded",
"message": "Voice Tasker workspace hourly limit exceeded.",
},
"user_daily": {
"scope": "user",
"window": "day",
"used": user_daily_used,
"limit": user_daily_limit,
"exceeded": bool(user_daily_limit and user_daily_used >= user_daily_limit),
"window_seconds": VOICE_TASK_RATE_LIMIT_DAILY_WINDOW_SECONDS,
"code": "voice_task_user_daily_limit_exceeded",
"message": "Voice Tasker user daily limit exceeded.",
},
"workspace_daily": {
"scope": "workspace",
"window": "day",
"used": workspace_daily_used,
"limit": workspace_daily_limit,
"exceeded": bool(workspace_daily_limit and workspace_daily_used >= workspace_daily_limit),
"window_seconds": VOICE_TASK_RATE_LIMIT_DAILY_WINDOW_SECONDS,
"code": "voice_task_workspace_daily_limit_exceeded",
"message": "Voice Tasker workspace daily limit exceeded.",
},
"project_daily": {
"scope": "project",
"window": "day",
"used": project_daily_used,
"limit": project_daily_limit,
"exceeded": bool(project and project_daily_limit and project_daily_used >= project_daily_limit),
"window_seconds": VOICE_TASK_RATE_LIMIT_DAILY_WINDOW_SECONDS,
"code": "voice_task_project_daily_limit_exceeded",
"message": "Voice Tasker project daily limit exceeded.",
"project_id": str(project.id) if project else None,
"project_identifier": project.identifier if project else None,
"project_name": project.name if project else None,
},
}
def get_voice_task_rate_limit_retry_after(workspace, user, scope, now=None):
def get_voice_task_rate_limit_retry_after(workspace, user, project, scope, window_seconds, now=None):
now = now or timezone.now()
window_start = now - timedelta(seconds=VOICE_TASK_RATE_LIMIT_WINDOW_SECONDS)
sessions = VoiceTaskSession.objects.filter(
workspace=workspace,
created_at__gte=window_start,
created_at__lte=now,
).exclude(error_code__in=VOICE_TASK_RATE_LIMIT_ERROR_CODES)
sessions = get_voice_task_limit_sessions(workspace, window_seconds, now=now)
if scope == "user":
sessions = sessions.filter(user=user)
elif scope == "project":
sessions = sessions.filter(project=project)
oldest_session_at = sessions.order_by("created_at").values_list("created_at", flat=True).first()
reset_at = (oldest_session_at or now) + timedelta(seconds=VOICE_TASK_RATE_LIMIT_WINDOW_SECONDS)
reset_at = (oldest_session_at or now) + timedelta(seconds=window_seconds)
retry_after = max(1, math.ceil((reset_at - now).total_seconds()))
return retry_after, reset_at
def get_voice_task_rate_limit_error(workspace, user, ai_settings, now=None):
def get_voice_task_rate_limit_error(workspace, user, project, ai_settings, now=None):
now = now or timezone.now()
state = get_voice_task_rate_limit_state(workspace, user, ai_settings, now=now)
state = get_voice_task_rate_limit_state(workspace, user, project, ai_settings, now=now)
if state["user"]["exceeded"]:
retry_after, reset_at = get_voice_task_rate_limit_retry_after(workspace, user, "user", now=now)
for key in ("user_hourly", "workspace_hourly", "user_daily", "workspace_daily", "project_daily"):
limit_state = state[key]
if not limit_state["exceeded"]:
continue
retry_after, reset_at = get_voice_task_rate_limit_retry_after(
workspace,
user,
project,
limit_state["scope"],
limit_state["window_seconds"],
now=now,
)
return {
"code": "voice_task_user_hourly_limit_exceeded",
"message": "Voice Tasker user hourly limit exceeded.",
"scope": "user",
"limit": state["user"]["limit"],
"used": state["user"]["used"],
"window_seconds": state["window_seconds"],
"retry_after": retry_after,
"reset_at": reset_at,
}
if state["workspace"]["exceeded"]:
retry_after, reset_at = get_voice_task_rate_limit_retry_after(workspace, user, "workspace", now=now)
return {
"code": "voice_task_workspace_hourly_limit_exceeded",
"message": "Voice Tasker workspace hourly limit exceeded.",
"scope": "workspace",
"limit": state["workspace"]["limit"],
"used": state["workspace"]["used"],
"window_seconds": state["window_seconds"],
"code": limit_state["code"],
"message": limit_state["message"],
"scope": limit_state["scope"],
"window": limit_state["window"],
"limit": limit_state["limit"],
"used": limit_state["used"],
"window_seconds": limit_state["window_seconds"],
"retry_after": retry_after,
"reset_at": reset_at,
"project_id": limit_state.get("project_id"),
"project_identifier": limit_state.get("project_identifier"),
"project_name": limit_state.get("project_name"),
}
return None
@ -405,10 +488,12 @@ def create_voice_task_rate_limit_session(
duration_seconds,
client_context,
rate_limit_error,
project=None,
):
return VoiceTaskSession.objects.create(
workspace=workspace,
user=user,
project=project,
status=VoiceTaskSession.Status.FAILED,
audio_duration_seconds=duration_seconds,
audio_content_type=audio_content_type,
@ -2495,7 +2580,8 @@ class VoiceTaskParseEndpoint(BaseAPIView):
@allow_permission(allowed_roles=[ROLE.ADMIN, ROLE.MEMBER, ROLE.GUEST], level="WORKSPACE")
def post(self, request, slug):
workspace = Workspace.objects.get(slug=slug)
preflight = get_voice_task_preflight(workspace, request.user, project_id=get_request_project_id(request))
request_project_id = get_request_project_id(request)
preflight = get_voice_task_preflight(workspace, request.user, project_id=request_project_id)
if not preflight["available"]:
response_status = (
@ -2558,7 +2644,8 @@ class VoiceTaskParseEndpoint(BaseAPIView):
status=status.HTTP_400_BAD_REQUEST,
)
rate_limit_error = get_voice_task_rate_limit_error(workspace, request.user, ai_settings)
quota_project = get_voice_task_quota_project(workspace, project_id=request_project_id, ai_settings=ai_settings)
rate_limit_error = get_voice_task_rate_limit_error(workspace, request.user, quota_project, ai_settings)
if rate_limit_error:
voice_session = create_voice_task_rate_limit_session(
workspace=workspace,
@ -2568,6 +2655,7 @@ class VoiceTaskParseEndpoint(BaseAPIView):
duration_seconds=duration_seconds,
client_context=client_context,
rate_limit_error=rate_limit_error,
project=quota_project,
)
return Response(
{
@ -2576,11 +2664,15 @@ class VoiceTaskParseEndpoint(BaseAPIView):
"code": rate_limit_error["code"],
"error": rate_limit_error["message"],
"limit_scope": rate_limit_error["scope"],
"limit_window": rate_limit_error["window"],
"limit": rate_limit_error["limit"],
"used": rate_limit_error["used"],
"window_seconds": rate_limit_error["window_seconds"],
"retry_after": rate_limit_error["retry_after"],
"reset_at": rate_limit_error["reset_at"].isoformat(),
"project_id": rate_limit_error["project_id"],
"project_identifier": rate_limit_error["project_identifier"],
"project_name": rate_limit_error["project_name"],
},
status=status.HTTP_429_TOO_MANY_REQUESTS,
)
@ -2588,6 +2680,7 @@ class VoiceTaskParseEndpoint(BaseAPIView):
voice_session = VoiceTaskSession.objects.create(
workspace=workspace,
user=request.user,
project=quota_project,
status=VoiceTaskSession.Status.UPLOADED,
audio_duration_seconds=duration_seconds,
audio_content_type=audio_content_type,
@ -2652,7 +2745,9 @@ class VoiceTaskParseEndpoint(BaseAPIView):
voice_session.status = VoiceTaskSession.Status.PARSED
voice_session.intent = parsed["intent"]
voice_session.parsed_json = parsed
voice_session.save(update_fields=["status", "intent", "parsed_json", "updated_at"])
if not voice_session.project_id and parsed.get("project_id"):
voice_session.project = get_voice_task_quota_project(workspace, project_id=parsed.get("project_id"))
voice_session.save(update_fields=["status", "intent", "parsed_json", "project", "updated_at"])
return Response(
{
@ -2832,8 +2927,9 @@ class VoiceTaskCommitEndpoint(BaseAPIView):
serializer.save()
issue.refresh_from_db()
voice_session.created_task = issue
voice_session.project = project
voice_session.parsed_json = draft
voice_session.save(update_fields=["created_task", "parsed_json", "updated_at"])
voice_session.save(update_fields=["created_task", "project", "parsed_json", "updated_at"])
issue_activity.delay(
type="issue.activity.updated",
@ -2891,8 +2987,9 @@ class VoiceTaskCommitEndpoint(BaseAPIView):
issue = serializer.save(created_by_id=request.user.id)
voice_session.created_task = issue
voice_session.project = project
voice_session.parsed_json = draft
voice_session.save(update_fields=["created_task", "parsed_json", "updated_at"])
voice_session.save(update_fields=["created_task", "project", "parsed_json", "updated_at"])
requested_data = json.dumps(payload_without_project, cls=DjangoJSONEncoder)
issue_activity.delay(
@ -2983,8 +3080,9 @@ class VoiceTaskCommitEndpoint(BaseAPIView):
serializer.save()
issue.refresh_from_db()
voice_session.updated_task = issue
voice_session.project = project
voice_session.parsed_json = draft
voice_session.save(update_fields=["updated_task", "parsed_json", "updated_at"])
voice_session.save(update_fields=["updated_task", "project", "parsed_json", "updated_at"])
issue_activity.delay(
type="issue.activity.updated",
@ -3037,8 +3135,9 @@ class VoiceTaskCommitEndpoint(BaseAPIView):
entity_name="issue",
).delete(soft=False)
voice_session.updated_task = issue
voice_session.project = project
voice_session.parsed_json = draft
voice_session.save(update_fields=["updated_task", "parsed_json", "updated_at"])
voice_session.save(update_fields=["updated_task", "project", "parsed_json", "updated_at"])
issue_activity.delay(
type="issue.activity.deleted",

View File

@ -0,0 +1,56 @@
# Generated by Codex on 2026-04-28
from django.db import migrations, models
import django.db.models.deletion
class Migration(migrations.Migration):
dependencies = [
("db", "0130_project_storage_quotas"),
]
operations = [
migrations.AlterField(
model_name="workspaceaicredential",
name="deleted_at",
field=models.DateTimeField(blank=True, null=True, verbose_name="Deleted At"),
),
migrations.AlterField(
model_name="workspaceaisettings",
name="deleted_at",
field=models.DateTimeField(blank=True, null=True, verbose_name="Deleted At"),
),
migrations.AddField(
model_name="workspaceaisettings",
name="per_user_daily_limit",
field=models.PositiveIntegerField(default=100),
),
migrations.AddField(
model_name="workspaceaisettings",
name="workspace_daily_limit",
field=models.PositiveIntegerField(default=1000),
),
migrations.AddField(
model_name="workspaceaisettings",
name="project_daily_limit",
field=models.PositiveIntegerField(default=300),
),
migrations.AddField(
model_name="voicetasksession",
name="project",
field=models.ForeignKey(
blank=True,
null=True,
on_delete=django.db.models.deletion.SET_NULL,
related_name="voice_task_sessions",
to="db.project",
),
),
migrations.AddIndex(
model_name="voicetasksession",
index=models.Index(
fields=["workspace", "project", "-created_at"],
name="voice_task_session_project_idx",
),
),
]

View File

@ -52,6 +52,9 @@ class WorkspaceAISettings(BaseModel):
max_audio_duration_seconds = models.PositiveIntegerField(default=120)
per_user_hourly_limit = models.PositiveIntegerField(default=30)
workspace_hourly_limit = models.PositiveIntegerField(default=300)
per_user_daily_limit = models.PositiveIntegerField(default=100)
workspace_daily_limit = models.PositiveIntegerField(default=1000)
project_daily_limit = models.PositiveIntegerField(default=300)
class Meta:
verbose_name = "Workspace AI Settings"
@ -106,6 +109,13 @@ class VoiceTaskSession(BaseModel):
on_delete=models.CASCADE,
related_name="voice_task_sessions",
)
project = models.ForeignKey(
"db.Project",
on_delete=models.SET_NULL,
null=True,
blank=True,
related_name="voice_task_sessions",
)
status = models.CharField(max_length=32, choices=Status.choices, default=Status.UPLOADED)
audio_duration_seconds = models.FloatField(null=True, blank=True)
audio_content_type = models.CharField(max_length=100, blank=True)
@ -138,6 +148,7 @@ class VoiceTaskSession(BaseModel):
ordering = ("-created_at",)
indexes = [
models.Index(fields=["workspace", "user", "-created_at"], name="voice_task_session_user_idx"),
models.Index(fields=["workspace", "project", "-created_at"], name="voice_task_session_project_idx"),
models.Index(fields=["workspace", "status", "-created_at"], name="voice_task_session_status_idx"),
]

View File

@ -133,7 +133,9 @@ function getVoiceTaskErrorMessage(error: unknown, fallback: string) {
const retryAfter = "retry_after" in error ? formatRetryAfter(error.retry_after) : "позже";
const used = "used" in error && typeof error.used === "number" ? error.used : null;
const limit = "limit" in error && typeof error.limit === "number" ? error.limit : null;
const usageText = used !== null && limit !== null ? ` Использовано ${used} из ${limit} за последний час.` : "";
const limitWindow = "limit_window" in error ? String(error.limit_window) : "hour";
const windowText = limitWindow === "day" ? "за последние сутки" : "за последний час";
const usageText = used !== null && limit !== null ? ` Использовано ${used} из ${limit} ${windowText}.` : "";
if (code === "voice_task_user_hourly_limit_exceeded") {
return `Лимит Voice Tasker для пользователя исчерпан. Повторите ${retryAfter}.${usageText}`;
@ -143,6 +145,19 @@ function getVoiceTaskErrorMessage(error: unknown, fallback: string) {
return `Лимит Voice Tasker для workspace исчерпан. Повторите ${retryAfter}.${usageText}`;
}
if (code === "voice_task_user_daily_limit_exceeded") {
return `Суточный лимит Voice Tasker для пользователя исчерпан. Повторите ${retryAfter}.${usageText}`;
}
if (code === "voice_task_workspace_daily_limit_exceeded") {
return `Суточный лимит Voice Tasker для workspace исчерпан. Повторите ${retryAfter}.${usageText}`;
}
if (code === "voice_task_project_daily_limit_exceeded") {
const projectName = "project_name" in error && error.project_name ? ` для контура ${String(error.project_name)}` : "";
return `Суточный лимит Voice Tasker${projectName} исчерпан. Повторите ${retryAfter}.${usageText}`;
}
if ("error" in error && error.error) return String(error.error);
}

View File

@ -46,6 +46,9 @@ type TFormState = {
max_audio_duration_seconds: number;
per_user_hourly_limit: number;
workspace_hourly_limit: number;
per_user_daily_limit: number;
workspace_daily_limit: number;
project_daily_limit: number;
openai_api_key: string;
};
@ -65,6 +68,9 @@ const getInitialFormState = (settings?: TWorkspaceAISettings): TFormState => ({
max_audio_duration_seconds: settings?.max_audio_duration_seconds ?? 120,
per_user_hourly_limit: settings?.per_user_hourly_limit ?? 30,
workspace_hourly_limit: settings?.workspace_hourly_limit ?? 300,
per_user_daily_limit: settings?.per_user_daily_limit ?? 100,
workspace_daily_limit: settings?.workspace_daily_limit ?? 1000,
project_daily_limit: settings?.project_daily_limit ?? 300,
openai_api_key: "",
});
@ -184,6 +190,9 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
max_audio_duration_seconds: formState.max_audio_duration_seconds,
per_user_hourly_limit: formState.per_user_hourly_limit,
workspace_hourly_limit: formState.workspace_hourly_limit,
per_user_daily_limit: formState.per_user_daily_limit,
workspace_daily_limit: formState.workspace_daily_limit,
project_daily_limit: formState.project_daily_limit,
};
if (formState.openai_api_key.trim()) payload.openai_api_key = formState.openai_api_key.trim();
@ -339,7 +348,7 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
<SectionHeader
icon={BrainCircuit}
title="Models and limits"
description="MVP использует один workspace key для транскрибации и структурирования."
description="Один workspace key обслуживает всех пользователей. Лимиты срабатывают до отправки аудио в OpenAI."
/>
<div className="grid gap-5 px-5 py-5 md:grid-cols-2">
<Field label="Transcription model">
@ -356,24 +365,51 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
className="nodedc-settings-input w-full"
/>
</Field>
<Field label="Per-user limit">
<Field label="Лимит пользователя за час">
<NumberInput
value={formState.per_user_hourly_limit}
min={1}
max={1000}
suffix="tasks/hour"
suffix="за час"
onChange={(value) => updateFormValue("per_user_hourly_limit", value)}
/>
</Field>
<Field label="Workspace limit">
<Field label="Лимит workspace за час">
<NumberInput
value={formState.workspace_hourly_limit}
min={1}
max={10000}
suffix="tasks/hour"
suffix="за час"
onChange={(value) => updateFormValue("workspace_hourly_limit", value)}
/>
</Field>
<Field label="Суточная квота пользователя">
<NumberInput
value={formState.per_user_daily_limit}
min={1}
max={10000}
suffix="за сутки"
onChange={(value) => updateFormValue("per_user_daily_limit", value)}
/>
</Field>
<Field label="Суточная квота workspace">
<NumberInput
value={formState.workspace_daily_limit}
min={1}
max={100000}
suffix="за сутки"
onChange={(value) => updateFormValue("workspace_daily_limit", value)}
/>
</Field>
<Field label="Суточная квота контура">
<NumberInput
value={formState.project_daily_limit}
min={1}
max={50000}
suffix="за сутки"
onChange={(value) => updateFormValue("project_daily_limit", value)}
/>
</Field>
</div>
</section>

View File

@ -43,6 +43,9 @@ export type TWorkspaceAISettings = {
max_audio_duration_seconds: number;
per_user_hourly_limit: number;
workspace_hourly_limit: number;
per_user_daily_limit: number;
workspace_daily_limit: number;
project_daily_limit: number;
credential: TWorkspaceAICredentialStatus;
created_at: string;
updated_at: string;
@ -61,6 +64,9 @@ export type TWorkspaceAISettingsPayload = Partial<
| "max_audio_duration_seconds"
| "per_user_hourly_limit"
| "workspace_hourly_limit"
| "per_user_daily_limit"
| "workspace_daily_limit"
| "project_daily_limit"
>
> & {
openai_api_key?: string;
@ -192,12 +198,16 @@ export type TVoiceTaskUploadResult = {
size: number;
};
client_context?: Record<string, unknown>;
limit_scope?: "user" | "workspace";
limit_scope?: "user" | "workspace" | "project";
limit_window?: "hour" | "day";
limit?: number;
used?: number;
window_seconds?: number;
retry_after?: number;
reset_at?: string;
project_id?: string | null;
project_identifier?: string | null;
project_name?: string | null;
code?: string;
error?: string;
};