ФУНКЦИИ - МЕЖПРОЕКТНАЯ КОММУНИКАЦИЯ: ручное назначение Voice Tasker
This commit is contained in:
parent
9243924f78
commit
02d79da6f9
|
|
@ -4,7 +4,7 @@
|
|||
|
||||
from rest_framework import serializers
|
||||
|
||||
from plane.db.models import Project, WorkspaceAICredential, WorkspaceAISettings
|
||||
from plane.db.models import Project, WorkspaceAICredential, WorkspaceAISettings, WorkspaceMember
|
||||
from plane.license.utils.encryption import encrypt_data
|
||||
|
||||
from .base import BaseSerializer
|
||||
|
|
@ -12,6 +12,8 @@ from .base import BaseSerializer
|
|||
|
||||
class WorkspaceAISettingsSerializer(BaseSerializer):
|
||||
default_project_id = serializers.UUIDField(required=False, allow_null=True)
|
||||
enabled_project_ids = serializers.ListField(child=serializers.UUIDField(), required=False, write_only=True)
|
||||
enabled_member_ids = serializers.ListField(child=serializers.UUIDField(), required=False, write_only=True)
|
||||
openai_api_key = serializers.CharField(required=False, allow_blank=True, write_only=True, trim_whitespace=False)
|
||||
credential = serializers.SerializerMethodField(read_only=True)
|
||||
|
||||
|
|
@ -26,6 +28,8 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
|
|||
"structuring_model",
|
||||
"default_project_id",
|
||||
"access_mode",
|
||||
"enabled_project_ids",
|
||||
"enabled_member_ids",
|
||||
"max_audio_duration_seconds",
|
||||
"per_user_hourly_limit",
|
||||
"workspace_hourly_limit",
|
||||
|
|
@ -45,6 +49,12 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
|
|||
"updated_at": credential.updated_at if credential else None,
|
||||
}
|
||||
|
||||
def to_representation(self, instance):
|
||||
data = super().to_representation(instance)
|
||||
data["enabled_project_ids"] = [str(project_id) for project_id in instance.enabled_projects.values_list("id", flat=True)]
|
||||
data["enabled_member_ids"] = [str(member_id) for member_id in instance.enabled_members.values_list("id", flat=True)]
|
||||
return data
|
||||
|
||||
def validate_default_project_id(self, value):
|
||||
if value is None:
|
||||
return None
|
||||
|
|
@ -54,6 +64,32 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
|
|||
raise serializers.ValidationError("Default project must belong to this workspace.")
|
||||
return value
|
||||
|
||||
def validate_enabled_project_ids(self, value):
|
||||
workspace = self.context["workspace"]
|
||||
project_ids = list(dict.fromkeys(value))
|
||||
existing_ids = set(
|
||||
Project.objects.filter(workspace=workspace, id__in=project_ids, archived_at__isnull=True).values_list(
|
||||
"id", flat=True
|
||||
)
|
||||
)
|
||||
if len(existing_ids) != len(project_ids):
|
||||
raise serializers.ValidationError("All selected projects must belong to this workspace.")
|
||||
return project_ids
|
||||
|
||||
def validate_enabled_member_ids(self, value):
|
||||
workspace = self.context["workspace"]
|
||||
member_ids = list(dict.fromkeys(value))
|
||||
existing_ids = set(
|
||||
WorkspaceMember.objects.filter(
|
||||
workspace=workspace,
|
||||
member_id__in=member_ids,
|
||||
is_active=True,
|
||||
).values_list("member_id", flat=True)
|
||||
)
|
||||
if len(existing_ids) != len(member_ids):
|
||||
raise serializers.ValidationError("All selected members must be active workspace members.")
|
||||
return member_ids
|
||||
|
||||
def validate_max_audio_duration_seconds(self, value):
|
||||
if value < 10 or value > 600:
|
||||
raise serializers.ValidationError("Max audio duration must be between 10 and 600 seconds.")
|
||||
|
|
@ -72,6 +108,8 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
|
|||
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)
|
||||
enabled_project_ids = validated_data.pop("enabled_project_ids", serializers.empty)
|
||||
enabled_member_ids = validated_data.pop("enabled_member_ids", serializers.empty)
|
||||
|
||||
if default_project_id is not serializers.empty:
|
||||
instance.default_project_id = default_project_id
|
||||
|
|
@ -81,6 +119,12 @@ class WorkspaceAISettingsSerializer(BaseSerializer):
|
|||
|
||||
instance.save()
|
||||
|
||||
if enabled_project_ids is not serializers.empty:
|
||||
instance.enabled_projects.set(enabled_project_ids)
|
||||
|
||||
if enabled_member_ids is not serializers.empty:
|
||||
instance.enabled_members.set(enabled_member_ids)
|
||||
|
||||
if api_key:
|
||||
cleaned_api_key = api_key.strip()
|
||||
credential, _ = WorkspaceAICredential.objects.get_or_create(
|
||||
|
|
|
|||
|
|
@ -227,7 +227,30 @@ def normalize_audio_content_type(content_type):
|
|||
return content_type.split(";")[0].strip().lower()
|
||||
|
||||
|
||||
def get_voice_task_preflight(workspace, user):
|
||||
def get_request_project_id(request):
|
||||
project_id = request.query_params.get("project_id")
|
||||
if project_id:
|
||||
return project_id
|
||||
|
||||
draft = request.data.get("draft")
|
||||
if isinstance(draft, dict) and draft.get("project_id"):
|
||||
return draft.get("project_id")
|
||||
|
||||
client_context = request.data.get("client_context")
|
||||
if not client_context:
|
||||
return None
|
||||
|
||||
try:
|
||||
parsed_context = json.loads(client_context) if isinstance(client_context, str) else client_context
|
||||
except (TypeError, json.JSONDecodeError):
|
||||
return None
|
||||
|
||||
if isinstance(parsed_context, dict):
|
||||
return parsed_context.get("current_project_id")
|
||||
return None
|
||||
|
||||
|
||||
def get_voice_task_preflight(workspace, user, project_id=None):
|
||||
ai_settings = WorkspaceAISettings.objects.filter(workspace=workspace).first()
|
||||
workspace_member = WorkspaceMember.objects.filter(workspace=workspace, member=user, is_active=True).first()
|
||||
|
||||
|
|
@ -237,6 +260,8 @@ def get_voice_task_preflight(workspace, user):
|
|||
"max_audio_duration_seconds": 120,
|
||||
"accepted_mime_types": VOICE_TASK_ACCEPTED_AUDIO_TYPES,
|
||||
"access_mode": "all_workspace_members",
|
||||
"enabled_project_ids": [],
|
||||
"enabled_member_ids": [],
|
||||
}
|
||||
|
||||
if not ai_settings:
|
||||
|
|
@ -244,6 +269,8 @@ def get_voice_task_preflight(workspace, user):
|
|||
|
||||
response["max_audio_duration_seconds"] = ai_settings.max_audio_duration_seconds
|
||||
response["access_mode"] = ai_settings.access_mode
|
||||
response["enabled_project_ids"] = [str(project_id) for project_id in ai_settings.enabled_projects.values_list("id", flat=True)]
|
||||
response["enabled_member_ids"] = [str(member_id) for member_id in ai_settings.enabled_members.values_list("id", flat=True)]
|
||||
|
||||
if not ai_settings.voice_tasker_enabled:
|
||||
response["reason"] = "disabled"
|
||||
|
|
@ -263,6 +290,19 @@ def get_voice_task_preflight(workspace, user):
|
|||
response["reason"] = "role_denied"
|
||||
return response
|
||||
|
||||
if ai_settings.access_mode == WorkspaceAISettings.AccessMode.SELECTED_MEMBERS:
|
||||
if not ai_settings.enabled_members.filter(id=user.id).exists():
|
||||
response["reason"] = "scope_denied"
|
||||
return response
|
||||
|
||||
if ai_settings.access_mode == WorkspaceAISettings.AccessMode.SELECTED_PROJECTS:
|
||||
if not project_id:
|
||||
response["reason"] = "scope_denied"
|
||||
return response
|
||||
if not ai_settings.enabled_projects.filter(id=project_id, workspace=workspace, archived_at__isnull=True).exists():
|
||||
response["reason"] = "scope_denied"
|
||||
return response
|
||||
|
||||
response["available"] = True
|
||||
response["reason"] = None
|
||||
return response
|
||||
|
|
@ -2332,7 +2372,10 @@ class VoiceTaskPreflightEndpoint(BaseAPIView):
|
|||
@allow_permission(allowed_roles=[ROLE.ADMIN, ROLE.MEMBER, ROLE.GUEST], level="WORKSPACE")
|
||||
def get(self, request, slug):
|
||||
workspace = Workspace.objects.get(slug=slug)
|
||||
return Response(get_voice_task_preflight(workspace, request.user), status=status.HTTP_200_OK)
|
||||
return Response(
|
||||
get_voice_task_preflight(workspace, request.user, project_id=get_request_project_id(request)),
|
||||
status=status.HTTP_200_OK,
|
||||
)
|
||||
|
||||
|
||||
class VoiceTaskParseEndpoint(BaseAPIView):
|
||||
|
|
@ -2341,10 +2384,14 @@ 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)
|
||||
preflight = get_voice_task_preflight(workspace, request.user, project_id=get_request_project_id(request))
|
||||
|
||||
if not preflight["available"]:
|
||||
response_status = status.HTTP_403_FORBIDDEN if preflight["reason"] == "role_denied" else status.HTTP_400_BAD_REQUEST
|
||||
response_status = (
|
||||
status.HTTP_403_FORBIDDEN
|
||||
if preflight["reason"] in {"role_denied", "scope_denied"}
|
||||
else status.HTTP_400_BAD_REQUEST
|
||||
)
|
||||
return Response(
|
||||
{
|
||||
"ok": False,
|
||||
|
|
@ -2512,9 +2559,13 @@ class VoiceTaskCommitEndpoint(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)
|
||||
preflight = get_voice_task_preflight(workspace, request.user, project_id=get_request_project_id(request))
|
||||
if not preflight["available"]:
|
||||
response_status = status.HTTP_403_FORBIDDEN if preflight["reason"] == "role_denied" else status.HTTP_400_BAD_REQUEST
|
||||
response_status = (
|
||||
status.HTTP_403_FORBIDDEN
|
||||
if preflight["reason"] in {"role_denied", "scope_denied"}
|
||||
else status.HTTP_400_BAD_REQUEST
|
||||
)
|
||||
return Response(
|
||||
{
|
||||
"ok": False,
|
||||
|
|
|
|||
|
|
@ -0,0 +1,47 @@
|
|||
# Generated by Codex on 2026-04-28
|
||||
|
||||
from django.conf import settings
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||
("db", "0128_stored_blob_dedup"),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name="workspaceaisettings",
|
||||
name="access_mode",
|
||||
field=models.CharField(
|
||||
choices=[
|
||||
("all_workspace_members", "All workspace members"),
|
||||
("admins_only", "Admins only"),
|
||||
("selected_projects", "Selected projects"),
|
||||
("selected_members", "Selected members"),
|
||||
],
|
||||
default="all_workspace_members",
|
||||
max_length=40,
|
||||
),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name="workspaceaisettings",
|
||||
name="enabled_members",
|
||||
field=models.ManyToManyField(
|
||||
blank=True,
|
||||
related_name="workspace_ai_feature_settings",
|
||||
to=settings.AUTH_USER_MODEL,
|
||||
),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name="workspaceaisettings",
|
||||
name="enabled_projects",
|
||||
field=models.ManyToManyField(
|
||||
blank=True,
|
||||
related_name="workspace_ai_feature_settings",
|
||||
to="db.project",
|
||||
),
|
||||
),
|
||||
]
|
||||
|
|
@ -15,6 +15,8 @@ class WorkspaceAISettings(BaseModel):
|
|||
class AccessMode(models.TextChoices):
|
||||
ALL_WORKSPACE_MEMBERS = "all_workspace_members", "All workspace members"
|
||||
ADMINS_ONLY = "admins_only", "Admins only"
|
||||
SELECTED_PROJECTS = "selected_projects", "Selected projects"
|
||||
SELECTED_MEMBERS = "selected_members", "Selected members"
|
||||
|
||||
workspace = models.OneToOneField(
|
||||
"db.Workspace",
|
||||
|
|
@ -37,6 +39,16 @@ class WorkspaceAISettings(BaseModel):
|
|||
choices=AccessMode.choices,
|
||||
default=AccessMode.ALL_WORKSPACE_MEMBERS,
|
||||
)
|
||||
enabled_projects = models.ManyToManyField(
|
||||
"db.Project",
|
||||
blank=True,
|
||||
related_name="workspace_ai_feature_settings",
|
||||
)
|
||||
enabled_members = models.ManyToManyField(
|
||||
settings.AUTH_USER_MODEL,
|
||||
blank=True,
|
||||
related_name="workspace_ai_feature_settings",
|
||||
)
|
||||
max_audio_duration_seconds = models.PositiveIntegerField(default=120)
|
||||
per_user_hourly_limit = models.PositiveIntegerField(default=30)
|
||||
workspace_hourly_limit = models.PositiveIntegerField(default=300)
|
||||
|
|
|
|||
|
|
@ -70,6 +70,7 @@ const UNAVAILABLE_LABELS = {
|
|||
missing_api_key: "OpenAI key не сохранен для этого workspace",
|
||||
not_configured: "AI-функции не настроены для этого workspace",
|
||||
role_denied: "Voice Task недоступен для вашей роли",
|
||||
scope_denied: "Voice Task недоступен для текущего проекта или пользователя",
|
||||
} as const;
|
||||
|
||||
const PRIORITY_LABELS: Record<Exclude<TVoiceTaskPriority, null>, string> = {
|
||||
|
|
@ -411,7 +412,10 @@ function VoiceTaskWaveform({ mediaRecorder }: { mediaRecorder: MediaRecorder | n
|
|||
useEffect(() => {
|
||||
const root = document.documentElement;
|
||||
const accentRgb = getComputedStyle(root).getPropertyValue("--nodedc-accent-rgb").trim();
|
||||
const channels = accentRgb.split(/[,\s]+/).filter(Boolean).slice(0, 3);
|
||||
const channels = accentRgb
|
||||
.split(/[,\s]+/)
|
||||
.filter(Boolean)
|
||||
.slice(0, 3);
|
||||
|
||||
if (channels.length === 3) setAccentColor(`rgb(${channels.join(", ")})`);
|
||||
}, []);
|
||||
|
|
@ -762,6 +766,7 @@ export function VoiceTaskerGlobalControl({ workspaceSlug }: Props) {
|
|||
const [hasDraftChanges, setHasDraftChanges] = useState(false);
|
||||
const [selectedTargetIssue, setSelectedTargetIssue] = useState<TVoiceTaskTargetOption | null>(null);
|
||||
const [dockSlot, setDockSlot] = useState<Element | null>(null);
|
||||
const preflightProjectId = activeProjectId ?? getCurrentProjectId();
|
||||
|
||||
const mediaRecorderRef = useRef<MediaRecorder | null>(null);
|
||||
const discardedRecorderRef = useRef<MediaRecorder | null>(null);
|
||||
|
|
@ -772,8 +777,8 @@ export function VoiceTaskerGlobalControl({ workspaceSlug }: Props) {
|
|||
const startedAtRef = useRef(0);
|
||||
|
||||
const { data: preflight } = useSWR(
|
||||
workspaceSlug ? `VOICE_TASK_PREFLIGHT_${workspaceSlug}` : null,
|
||||
workspaceSlug ? () => workspaceAIService.retrieveVoiceTaskPreflight(workspaceSlug) : null,
|
||||
workspaceSlug ? `VOICE_TASK_PREFLIGHT_${workspaceSlug}_${preflightProjectId ?? "workspace"}` : null,
|
||||
workspaceSlug ? () => workspaceAIService.retrieveVoiceTaskPreflight(workspaceSlug, preflightProjectId) : null,
|
||||
{ refreshInterval: 30000 }
|
||||
);
|
||||
|
||||
|
|
@ -1191,18 +1196,18 @@ export function VoiceTaskerGlobalControl({ workspaceSlug }: Props) {
|
|||
<>
|
||||
{isAvailable && dockSlot
|
||||
? createPortal(
|
||||
<Tooltip tooltipContent={tooltipContent} position="top">
|
||||
<button
|
||||
type="button"
|
||||
className="nodedc-bottom-dock-voice-button"
|
||||
onClick={openVoiceTasker}
|
||||
aria-label="Voice Tasker"
|
||||
>
|
||||
<Mic className="size-4" />
|
||||
</button>
|
||||
</Tooltip>,
|
||||
dockSlot
|
||||
)
|
||||
<Tooltip tooltipContent={tooltipContent} position="top">
|
||||
<button
|
||||
type="button"
|
||||
className="nodedc-bottom-dock-voice-button"
|
||||
onClick={openVoiceTasker}
|
||||
aria-label="Voice Tasker"
|
||||
>
|
||||
<Mic className="size-4" />
|
||||
</button>
|
||||
</Tooltip>,
|
||||
dockSlot
|
||||
)
|
||||
: null}
|
||||
|
||||
<ModalCore
|
||||
|
|
|
|||
|
|
@ -8,12 +8,18 @@ import { useEffect, useMemo, useState } from "react";
|
|||
import type { ElementType, ReactNode } from "react";
|
||||
import { observer } from "mobx-react";
|
||||
import useSWR, { mutate } from "swr";
|
||||
import { BrainCircuit, KeyRound, Mic, ShieldCheck } from "lucide-react";
|
||||
import { BrainCircuit, Check, FolderKanban, KeyRound, Mic, ShieldCheck, UsersRound } from "lucide-react";
|
||||
// plane imports
|
||||
import { EUserPermissions, EUserPermissionsLevel } from "@plane/constants";
|
||||
import { Button } from "@plane/propel/button";
|
||||
import { TOAST_TYPE, setToast } from "@plane/propel/toast";
|
||||
import type { TWorkspaceAIAccessMode, TWorkspaceAISettings, TWorkspaceAISettingsPayload } from "@plane/types";
|
||||
import type {
|
||||
IProject,
|
||||
IWorkspaceMember,
|
||||
TWorkspaceAIAccessMode,
|
||||
TWorkspaceAISettings,
|
||||
TWorkspaceAISettingsPayload,
|
||||
} from "@plane/types";
|
||||
import { Input, ToggleSwitch } from "@plane/ui";
|
||||
import { cn } from "@plane/utils";
|
||||
// components
|
||||
|
|
@ -21,6 +27,7 @@ import { NotAuthorizedView } from "@/components/auth-screens/not-authorized-view
|
|||
import { SettingsHeading } from "@/components/settings/heading";
|
||||
// hooks
|
||||
import { useProject } from "@/hooks/store/use-project";
|
||||
import { useMember } from "@/hooks/store/use-member";
|
||||
import { useWorkspace } from "@/hooks/store/use-workspace";
|
||||
import { useUserPermissions } from "@/hooks/store/user";
|
||||
// services
|
||||
|
|
@ -34,6 +41,8 @@ type TFormState = {
|
|||
structuring_model: string;
|
||||
default_project_id: string;
|
||||
access_mode: TWorkspaceAIAccessMode;
|
||||
enabled_project_ids: string[];
|
||||
enabled_member_ids: string[];
|
||||
max_audio_duration_seconds: number;
|
||||
per_user_hourly_limit: number;
|
||||
workspace_hourly_limit: number;
|
||||
|
|
@ -51,12 +60,46 @@ const getInitialFormState = (settings?: TWorkspaceAISettings): TFormState => ({
|
|||
structuring_model: settings?.structuring_model ?? "gpt-4o-mini",
|
||||
default_project_id: settings?.default_project_id ?? "",
|
||||
access_mode: settings?.access_mode ?? "all_workspace_members",
|
||||
enabled_project_ids: settings?.enabled_project_ids ?? [],
|
||||
enabled_member_ids: settings?.enabled_member_ids ?? [],
|
||||
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,
|
||||
openai_api_key: "",
|
||||
});
|
||||
|
||||
const ACCESS_MODE_OPTIONS: {
|
||||
description: string;
|
||||
icon: ElementType;
|
||||
label: string;
|
||||
value: TWorkspaceAIAccessMode;
|
||||
}[] = [
|
||||
{
|
||||
value: "all_workspace_members",
|
||||
label: "Весь workspace",
|
||||
description: "Доступна всем активным участникам пространства.",
|
||||
icon: UsersRound,
|
||||
},
|
||||
{
|
||||
value: "admins_only",
|
||||
label: "Только админы",
|
||||
description: "Доступна только администраторам workspace.",
|
||||
icon: ShieldCheck,
|
||||
},
|
||||
{
|
||||
value: "selected_projects",
|
||||
label: "По контурам",
|
||||
description: "Доступна только внутри выбранных рабочих проектов.",
|
||||
icon: FolderKanban,
|
||||
},
|
||||
{
|
||||
value: "selected_members",
|
||||
label: "По людям",
|
||||
description: "Доступна только выбранным участникам workspace.",
|
||||
icon: UsersRound,
|
||||
},
|
||||
];
|
||||
|
||||
export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSettingsContent(props: TProps) {
|
||||
const { showHeading = true, workspaceSlug } = props;
|
||||
const [formState, setFormState] = useState<TFormState>(getInitialFormState());
|
||||
|
|
@ -65,6 +108,9 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
// store hooks
|
||||
const { currentWorkspace } = useWorkspace();
|
||||
const { fetchProjects, projectMap } = useProject();
|
||||
const {
|
||||
workspace: { fetchWorkspaceMembers, getWorkspaceMemberDetails, getWorkspaceMemberIds },
|
||||
} = useMember();
|
||||
const { workspaceUserInfo, allowPermissions } = useUserPermissions();
|
||||
// derived values
|
||||
const canPerformWorkspaceAdminActions = allowPermissions([EUserPermissions.ADMIN], EUserPermissionsLevel.WORKSPACE);
|
||||
|
|
@ -79,6 +125,11 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
canPerformWorkspaceAdminActions ? () => fetchProjects(workspaceSlug) : null
|
||||
);
|
||||
|
||||
useSWR(
|
||||
canPerformWorkspaceAdminActions ? `WORKSPACE_AI_SETTINGS_MEMBERS_${workspaceSlug}` : null,
|
||||
canPerformWorkspaceAdminActions ? () => fetchWorkspaceMembers(workspaceSlug) : null
|
||||
);
|
||||
|
||||
const projects = useMemo(
|
||||
() =>
|
||||
Object.values(projectMap)
|
||||
|
|
@ -87,6 +138,20 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
[currentWorkspace?.id, projectMap]
|
||||
);
|
||||
|
||||
const workspaceMembers = useMemo(
|
||||
() =>
|
||||
getWorkspaceMemberIds(workspaceSlug)
|
||||
.flatMap((memberId) => {
|
||||
const member = getWorkspaceMemberDetails(memberId);
|
||||
if (!member || member.is_active === false || member.member?.is_bot) return [];
|
||||
return [member];
|
||||
})
|
||||
.sort((a, b) =>
|
||||
(a.member.display_name || a.member.email || "").localeCompare(b.member.display_name || b.member.email || "")
|
||||
),
|
||||
[getWorkspaceMemberDetails, getWorkspaceMemberIds, workspaceSlug]
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
if (settings) setFormState(getInitialFormState(settings));
|
||||
}, [settings]);
|
||||
|
|
@ -95,6 +160,17 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
setFormState((prev) => ({ ...prev, [key]: value }));
|
||||
};
|
||||
|
||||
const toggleListValue = (key: "enabled_project_ids" | "enabled_member_ids", value: string) => {
|
||||
setFormState((prev) => {
|
||||
const currentValues = prev[key];
|
||||
const nextValues = currentValues.includes(value)
|
||||
? currentValues.filter((currentValue) => currentValue !== value)
|
||||
: [...currentValues, value];
|
||||
|
||||
return { ...prev, [key]: nextValues };
|
||||
});
|
||||
};
|
||||
|
||||
const handleSave = async () => {
|
||||
setIsSaving(true);
|
||||
const payload: TWorkspaceAISettingsPayload = {
|
||||
|
|
@ -103,6 +179,8 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
structuring_model: formState.structuring_model.trim(),
|
||||
default_project_id: formState.default_project_id || null,
|
||||
access_mode: formState.access_mode,
|
||||
enabled_project_ids: formState.enabled_project_ids,
|
||||
enabled_member_ids: formState.enabled_member_ids,
|
||||
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,
|
||||
|
|
@ -113,6 +191,11 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
try {
|
||||
const response = await workspaceAIService.updateSettings(workspaceSlug, payload);
|
||||
await mutate(`WORKSPACE_AI_SETTINGS_${workspaceSlug}`, response, false);
|
||||
await mutate(
|
||||
(key) => typeof key === "string" && key.startsWith(`VOICE_TASK_PREFLIGHT_${workspaceSlug}`),
|
||||
undefined,
|
||||
{ revalidate: true }
|
||||
);
|
||||
setFormState(getInitialFormState(response));
|
||||
setToast({
|
||||
type: TOAST_TYPE.SUCCESS,
|
||||
|
|
@ -160,7 +243,7 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
)}
|
||||
|
||||
{isLoading || !settings ? (
|
||||
<div className="nodedc-settings-card px-5 py-5 text-sm text-secondary">Загрузка настроек...</div>
|
||||
<div className="nodedc-settings-card text-sm px-5 py-5 text-secondary">Загрузка настроек...</div>
|
||||
) : (
|
||||
<>
|
||||
<section className="nodedc-settings-card overflow-hidden">
|
||||
|
|
@ -177,25 +260,30 @@ export const AIVoiceTaskerSettingsContent = observer(function AIVoiceTaskerSetti
|
|||
}
|
||||
/>
|
||||
|
||||
<AccessScopeSection
|
||||
accessMode={formState.access_mode}
|
||||
enabledMemberIds={formState.enabled_member_ids}
|
||||
enabledProjectIds={formState.enabled_project_ids}
|
||||
members={workspaceMembers}
|
||||
projects={projects}
|
||||
onAccessModeChange={(value) => updateFormValue("access_mode", value)}
|
||||
onToggleMember={(memberId) => toggleListValue("enabled_member_ids", memberId)}
|
||||
onToggleProject={(projectId) => toggleListValue("enabled_project_ids", projectId)}
|
||||
/>
|
||||
|
||||
<div className="grid gap-5 px-5 py-5 md:grid-cols-2">
|
||||
<Field label="Provider">
|
||||
<Input value="OpenAI" disabled className="nodedc-settings-input w-full cursor-not-allowed !bg-white/4" />
|
||||
</Field>
|
||||
<Field label="Access mode">
|
||||
<select
|
||||
value={formState.access_mode}
|
||||
onChange={(event) => updateFormValue("access_mode", event.target.value as TWorkspaceAIAccessMode)}
|
||||
className="nodedc-settings-select w-full px-4 text-sm"
|
||||
>
|
||||
<option value="all_workspace_members">All workspace members</option>
|
||||
<option value="admins_only">Admins only</option>
|
||||
</select>
|
||||
<Input
|
||||
value="OpenAI"
|
||||
disabled
|
||||
className="nodedc-settings-input w-full cursor-not-allowed !bg-white/4"
|
||||
/>
|
||||
</Field>
|
||||
<Field label="Default project fallback">
|
||||
<select
|
||||
value={formState.default_project_id}
|
||||
onChange={(event) => updateFormValue("default_project_id", event.target.value)}
|
||||
className="nodedc-settings-select w-full px-4 text-sm"
|
||||
className="nodedc-settings-select text-sm w-full px-4"
|
||||
>
|
||||
<option value="">None</option>
|
||||
{projects.map((project) => (
|
||||
|
|
@ -312,6 +400,167 @@ type TFieldProps = {
|
|||
label: string;
|
||||
};
|
||||
|
||||
type TAccessScopeSectionProps = {
|
||||
accessMode: TWorkspaceAIAccessMode;
|
||||
enabledMemberIds: string[];
|
||||
enabledProjectIds: string[];
|
||||
members: IWorkspaceMember[];
|
||||
onAccessModeChange: (value: TWorkspaceAIAccessMode) => void;
|
||||
onToggleMember: (memberId: string) => void;
|
||||
onToggleProject: (projectId: string) => void;
|
||||
projects: IProject[];
|
||||
};
|
||||
|
||||
function AccessScopeSection({
|
||||
accessMode,
|
||||
enabledMemberIds,
|
||||
enabledProjectIds,
|
||||
members,
|
||||
onAccessModeChange,
|
||||
onToggleMember,
|
||||
onToggleProject,
|
||||
projects,
|
||||
}: TAccessScopeSectionProps) {
|
||||
const selectedProjectsCount = enabledProjectIds.length;
|
||||
const selectedMembersCount = enabledMemberIds.length;
|
||||
|
||||
return (
|
||||
<div className="border-t border-white/5 px-5 py-5">
|
||||
<div className="mb-4 flex flex-col gap-1">
|
||||
<h4 className="text-13 font-semibold text-primary">Доступ к функции</h4>
|
||||
<p className="max-w-3xl text-12 leading-5 text-tertiary">
|
||||
Настройте, где и кому будет доступна кнопка Voice Tasker. Проверка выполняется на backend перед записью,
|
||||
разбором и публикацией задачи.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div className="grid gap-3 md:grid-cols-4">
|
||||
{ACCESS_MODE_OPTIONS.map((option) => {
|
||||
const isSelected = accessMode === option.value;
|
||||
const Icon = option.icon;
|
||||
|
||||
return (
|
||||
<button
|
||||
key={option.value}
|
||||
type="button"
|
||||
onClick={() => onAccessModeChange(option.value)}
|
||||
className={cn(
|
||||
"flex min-h-[7rem] flex-col items-start justify-between rounded-[1.35rem] bg-white/[0.045] px-4 py-3 text-left transition-colors",
|
||||
isSelected
|
||||
? "bg-[rgba(var(--nodedc-accent-rgb),0.16)] text-primary"
|
||||
: "text-secondary hover:bg-white/[0.07]"
|
||||
)}
|
||||
>
|
||||
<span
|
||||
className={cn(
|
||||
"grid size-9 place-items-center rounded-full",
|
||||
isSelected
|
||||
? "bg-[rgb(var(--nodedc-accent-rgb))] text-[rgb(var(--nodedc-on-accent-rgb))]"
|
||||
: "bg-white/6 text-tertiary"
|
||||
)}
|
||||
>
|
||||
<Icon className="size-4" />
|
||||
</span>
|
||||
<span>
|
||||
<span className="block text-13 font-semibold">{option.label}</span>
|
||||
<span className="mt-1 block text-11 leading-4 text-tertiary">{option.description}</span>
|
||||
</span>
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
|
||||
{accessMode === "selected_projects" && (
|
||||
<div className="mt-4 rounded-[1.35rem] bg-white/[0.035] p-4">
|
||||
<div className="mb-3 flex items-center justify-between gap-3">
|
||||
<span className="text-12 font-semibold tracking-[0.18em] text-tertiary uppercase">Контуры</span>
|
||||
<span className="inline-flex h-11 min-w-[5.75rem] items-center justify-center rounded-full bg-white/[0.06] px-4 text-center text-11 text-secondary">
|
||||
выбрано: {selectedProjectsCount}
|
||||
</span>
|
||||
</div>
|
||||
<div className="grid max-h-64 gap-2 overflow-auto pr-1 md:grid-cols-2">
|
||||
{projects.map((project) => (
|
||||
<ScopeToggleRow
|
||||
key={project.id}
|
||||
checked={enabledProjectIds.includes(project.id)}
|
||||
label={project.name}
|
||||
meta={project.identifier}
|
||||
onClick={() => onToggleProject(project.id)}
|
||||
/>
|
||||
))}
|
||||
{projects.length === 0 && <div className="text-12 text-tertiary">Доступных контуров пока нет.</div>}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{accessMode === "selected_members" && (
|
||||
<div className="mt-4 rounded-[1.35rem] bg-white/[0.035] p-4">
|
||||
<div className="mb-3 flex items-center justify-between gap-3">
|
||||
<span className="text-12 font-semibold tracking-[0.18em] text-tertiary uppercase">Пользователи</span>
|
||||
<span className="inline-flex h-11 min-w-[5.75rem] items-center justify-center rounded-full bg-white/[0.06] px-4 text-center text-11 text-secondary">
|
||||
выбрано: {selectedMembersCount}
|
||||
</span>
|
||||
</div>
|
||||
<div className="grid max-h-64 gap-2 overflow-auto pr-1 md:grid-cols-2">
|
||||
{members.map((member) => (
|
||||
<ScopeToggleRow
|
||||
key={member.member.id}
|
||||
checked={enabledMemberIds.includes(member.member.id)}
|
||||
label={member.member.display_name || member.member.email || "Пользователь"}
|
||||
meta={getWorkspaceRoleLabel(member.role)}
|
||||
onClick={() => onToggleMember(member.member.id)}
|
||||
/>
|
||||
))}
|
||||
{members.length === 0 && <div className="text-12 text-tertiary">Активные участники не найдены.</div>}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
type TScopeToggleRowProps = {
|
||||
checked: boolean;
|
||||
label: string;
|
||||
meta?: string | number;
|
||||
onClick: () => void;
|
||||
};
|
||||
|
||||
function ScopeToggleRow({ checked, label, meta, onClick }: TScopeToggleRowProps) {
|
||||
return (
|
||||
<button
|
||||
type="button"
|
||||
onClick={onClick}
|
||||
className={cn(
|
||||
"flex min-h-12 items-center gap-3 rounded-2xl bg-white/[0.045] px-3 text-left text-13 transition-colors",
|
||||
checked ? "text-primary" : "text-secondary hover:bg-white/[0.07]"
|
||||
)}
|
||||
>
|
||||
<span
|
||||
className={cn(
|
||||
"grid size-5 shrink-0 place-items-center rounded-full",
|
||||
checked
|
||||
? "bg-[rgb(var(--nodedc-accent-rgb))] text-[rgb(var(--nodedc-on-accent-rgb))]"
|
||||
: "bg-white/8 text-transparent"
|
||||
)}
|
||||
>
|
||||
{checked && <Check className="size-3.5" />}
|
||||
</span>
|
||||
<span className="min-w-0 flex-1">
|
||||
<span className="block truncate font-medium">{label}</span>
|
||||
{meta && <span className="block truncate text-11 text-tertiary">{meta}</span>}
|
||||
</span>
|
||||
</button>
|
||||
);
|
||||
}
|
||||
|
||||
function getWorkspaceRoleLabel(role: IWorkspaceMember["role"]) {
|
||||
if (Number(role) === EUserPermissions.ADMIN) return "Админ";
|
||||
if (Number(role) === EUserPermissions.MEMBER) return "Участник";
|
||||
if (Number(role) === EUserPermissions.GUEST) return "Гость";
|
||||
return "Участник";
|
||||
}
|
||||
|
||||
function Field({ children, label }: TFieldProps) {
|
||||
return (
|
||||
<label className="flex flex-col gap-2.5">
|
||||
|
|
@ -338,9 +587,9 @@ function NumberInput({ max, min, onChange, suffix, value }: TNumberInputProps) {
|
|||
max={max}
|
||||
value={value}
|
||||
onChange={(event) => onChange(Number(event.target.value))}
|
||||
className="min-h-12 min-w-0 flex-1 bg-transparent px-4 text-sm text-primary outline-none"
|
||||
className="text-sm min-h-12 min-w-0 flex-1 bg-transparent px-4 text-primary outline-none"
|
||||
/>
|
||||
<span className="shrink-0 px-4 text-xs text-tertiary">{suffix}</span>
|
||||
<span className="text-xs shrink-0 px-4 text-tertiary">{suffix}</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -45,11 +45,30 @@ export class WorkspaceAIService extends APIService {
|
|||
});
|
||||
}
|
||||
|
||||
async retrieveVoiceTaskPreflight(workspaceSlug: string): Promise<TVoiceTaskPreflight> {
|
||||
return this.get(`/api/workspaces/${workspaceSlug}/voice-task/preflight/`)
|
||||
.then((response) => response?.data)
|
||||
async retrieveVoiceTaskPreflight(workspaceSlug: string, projectId?: string | null): Promise<TVoiceTaskPreflight> {
|
||||
const params = projectId ? `?project_id=${projectId}` : "";
|
||||
return this.get(
|
||||
`/api/workspaces/${workspaceSlug}/voice-task/preflight/${params}`,
|
||||
{},
|
||||
{ validateStatus: () => true }
|
||||
)
|
||||
.then((response) => {
|
||||
if (response?.status === 200) return response?.data;
|
||||
if (response?.status === 401 || response?.status === 403)
|
||||
return {
|
||||
available: false,
|
||||
reason: "scope_denied",
|
||||
max_audio_duration_seconds: 120,
|
||||
accepted_mime_types: ["audio/webm", "audio/mp4", "audio/mpeg", "audio/wav"],
|
||||
access_mode: "all_workspace_members",
|
||||
enabled_project_ids: [],
|
||||
enabled_member_ids: [],
|
||||
} satisfies TVoiceTaskPreflight;
|
||||
|
||||
throw response?.data;
|
||||
})
|
||||
.catch((error) => {
|
||||
throw error?.response?.data;
|
||||
throw error?.response?.data ?? error;
|
||||
});
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -16,7 +16,11 @@ export interface IGptResponse {
|
|||
}
|
||||
|
||||
export type TWorkspaceAIProvider = "openai";
|
||||
export type TWorkspaceAIAccessMode = "all_workspace_members" | "admins_only";
|
||||
export type TWorkspaceAIAccessMode =
|
||||
| "all_workspace_members"
|
||||
| "admins_only"
|
||||
| "selected_projects"
|
||||
| "selected_members";
|
||||
|
||||
export type TWorkspaceAICredentialStatus = {
|
||||
provider: TWorkspaceAIProvider;
|
||||
|
|
@ -34,6 +38,8 @@ export type TWorkspaceAISettings = {
|
|||
structuring_model: string;
|
||||
default_project_id: string | null;
|
||||
access_mode: TWorkspaceAIAccessMode;
|
||||
enabled_project_ids: string[];
|
||||
enabled_member_ids: string[];
|
||||
max_audio_duration_seconds: number;
|
||||
per_user_hourly_limit: number;
|
||||
workspace_hourly_limit: number;
|
||||
|
|
@ -50,6 +56,8 @@ export type TWorkspaceAISettingsPayload = Partial<
|
|||
| "structuring_model"
|
||||
| "default_project_id"
|
||||
| "access_mode"
|
||||
| "enabled_project_ids"
|
||||
| "enabled_member_ids"
|
||||
| "max_audio_duration_seconds"
|
||||
| "per_user_hourly_limit"
|
||||
| "workspace_hourly_limit"
|
||||
|
|
@ -66,7 +74,13 @@ export type TWorkspaceAIConnectionTestResult = {
|
|||
error?: string;
|
||||
};
|
||||
|
||||
export type TVoiceTaskPreflightReason = "not_configured" | "disabled" | "missing_api_key" | "role_denied" | null;
|
||||
export type TVoiceTaskPreflightReason =
|
||||
| "not_configured"
|
||||
| "disabled"
|
||||
| "missing_api_key"
|
||||
| "role_denied"
|
||||
| "scope_denied"
|
||||
| null;
|
||||
|
||||
export type TVoiceTaskPreflight = {
|
||||
available: boolean;
|
||||
|
|
@ -74,6 +88,8 @@ export type TVoiceTaskPreflight = {
|
|||
max_audio_duration_seconds: number;
|
||||
accepted_mime_types: string[];
|
||||
access_mode: TWorkspaceAIAccessMode;
|
||||
enabled_project_ids: string[];
|
||||
enabled_member_ids: string[];
|
||||
};
|
||||
|
||||
export type TVoiceTaskIntent = "create_task" | "update_task" | "delete_task" | "unknown";
|
||||
|
|
|
|||
Loading…
Reference in New Issue