2026-05-08 19:23:28 +03:00
|
|
|
|
<script setup lang="ts">
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Админка → Тенанты. Список всех тенантов SaaS с балансами/тарифами/MRR.
|
|
|
|
|
|
*
|
2026-05-10 04:38:08 +03:00
|
|
|
|
* Sprint 4 Phase B/1 (audit O-refactor-04 хвост): UI-блоки выделены в
|
|
|
|
|
|
* components/admin/tenants/{TenantsStatsHeader,TenantsFilters,TenantsTable}.
|
|
|
|
|
|
* State (filterStatuses/filterTariffs/clearFilters/tenantsState/stats и др.)
|
|
|
|
|
|
* остаётся в этом view ради `defineExpose`-контракта, который Vitest тесты
|
|
|
|
|
|
* используют для прямого доступа.
|
|
|
|
|
|
*
|
2026-05-08 19:23:28 +03:00
|
|
|
|
* Источник дизайна: liderra_v8_handoff/concepts/v8_admin.html секция #page-tenants.
|
|
|
|
|
|
* По схеме v8.7 §3 (tenants table) + ТЗ §22 (админка).
|
|
|
|
|
|
*
|
2026-05-10 04:38:08 +03:00
|
|
|
|
* Click по строке → /admin/tenants/{code} (карточка тенанта).
|
2026-05-08 19:23:28 +03:00
|
|
|
|
*/
|
2026-05-09 09:19:53 +03:00
|
|
|
|
import { computed, onMounted, reactive, ref } from 'vue';
|
2026-05-09 05:33:21 +03:00
|
|
|
|
import { useRouter } from 'vue-router';
|
2026-05-12 20:24:33 +03:00
|
|
|
|
import { MOCK_STATS, MOCK_TENANTS, type AdminTenant, type TenantStatus } from '../../composables/mockTenants';
|
2026-05-09 09:19:53 +03:00
|
|
|
|
import { mapApiAdminTenant } from '../../composables/adminTenantsMapper';
|
2026-05-09 10:17:51 +03:00
|
|
|
|
import { usePolling } from '../../composables/usePolling';
|
2026-05-09 09:19:53 +03:00
|
|
|
|
import * as adminApi from '../../api/admin';
|
2026-05-10 04:38:08 +03:00
|
|
|
|
// NB: ImpersonationDialog оставлен sync-import — Vitest test через
|
|
|
|
|
|
// `findComponent({ name: 'ImpersonationDialog' })` + `stubs`, defineAsyncComponent
|
|
|
|
|
|
// ломает identity wrapper'а в test-utils.
|
2026-05-09 04:52:52 +03:00
|
|
|
|
import ImpersonationDialog from '../../components/admin/ImpersonationDialog.vue';
|
2026-05-10 04:38:08 +03:00
|
|
|
|
import TenantsStatsHeader from '../../components/admin/tenants/TenantsStatsHeader.vue';
|
|
|
|
|
|
import TenantsFilters from '../../components/admin/tenants/TenantsFilters.vue';
|
|
|
|
|
|
import TenantsTable from '../../components/admin/tenants/TenantsTable.vue';
|
2026-05-08 19:23:28 +03:00
|
|
|
|
|
2026-05-09 05:33:21 +03:00
|
|
|
|
const router = useRouter();
|
|
|
|
|
|
|
2026-05-09 09:19:53 +03:00
|
|
|
|
const tenantsState = reactive<AdminTenant[]>(MOCK_TENANTS.map((t) => ({ ...t })));
|
|
|
|
|
|
const stats = reactive({ ...MOCK_STATS });
|
|
|
|
|
|
const loading = ref(false);
|
|
|
|
|
|
const fetchError = ref(false);
|
|
|
|
|
|
|
|
|
|
|
|
async function loadTenants() {
|
|
|
|
|
|
loading.value = true;
|
|
|
|
|
|
fetchError.value = false;
|
|
|
|
|
|
try {
|
|
|
|
|
|
const res = await adminApi.listAdminTenants();
|
|
|
|
|
|
const mapped = res.tenants.map((t) => mapApiAdminTenant(t));
|
|
|
|
|
|
tenantsState.splice(0, tenantsState.length, ...mapped);
|
|
|
|
|
|
stats.total = res.stats.total;
|
|
|
|
|
|
stats.active = res.stats.active;
|
|
|
|
|
|
stats.trial = res.stats.trial;
|
|
|
|
|
|
stats.overdue = res.stats.overdue;
|
|
|
|
|
|
} catch {
|
|
|
|
|
|
fetchError.value = true;
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
loading.value = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
onMounted(loadTenants);
|
2026-05-09 10:17:51 +03:00
|
|
|
|
usePolling(loadTenants);
|
2026-05-09 09:19:53 +03:00
|
|
|
|
|
2026-05-09 05:33:21 +03:00
|
|
|
|
function openTenantDetail(t: AdminTenant) {
|
|
|
|
|
|
router.push({ name: 'admin-tenant-detail', params: { code: t.code } });
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-08 19:23:28 +03:00
|
|
|
|
const search = ref('');
|
2026-05-09 05:33:21 +03:00
|
|
|
|
const filterStatuses = ref<TenantStatus[]>([]);
|
2026-05-10 04:38:08 +03:00
|
|
|
|
const filterTariffs = ref<string[]>([]);
|
2026-05-08 19:23:28 +03:00
|
|
|
|
|
2026-05-09 04:52:52 +03:00
|
|
|
|
const impersonationOpen = ref(false);
|
|
|
|
|
|
const impersonationTenant = ref<AdminTenant | null>(null);
|
|
|
|
|
|
|
2026-05-09 09:19:53 +03:00
|
|
|
|
const availableTariffs = computed(() => Array.from(new Set(tenantsState.map((t) => t.tariff))).sort());
|
2026-05-09 05:33:21 +03:00
|
|
|
|
|
|
|
|
|
|
function clearFilters() {
|
|
|
|
|
|
filterStatuses.value = [];
|
|
|
|
|
|
filterTariffs.value = [];
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-09 04:52:52 +03:00
|
|
|
|
const ADMIN_USER_ID = 1;
|
|
|
|
|
|
|
|
|
|
|
|
function openImpersonation(tenant: AdminTenant) {
|
|
|
|
|
|
impersonationTenant.value = tenant;
|
|
|
|
|
|
impersonationOpen.value = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-09 09:19:53 +03:00
|
|
|
|
defineExpose({
|
|
|
|
|
|
filterStatuses,
|
|
|
|
|
|
filterTariffs,
|
|
|
|
|
|
clearFilters,
|
|
|
|
|
|
impersonationOpen,
|
|
|
|
|
|
impersonationTenant,
|
|
|
|
|
|
tenantsState,
|
|
|
|
|
|
stats,
|
|
|
|
|
|
loading,
|
|
|
|
|
|
fetchError,
|
|
|
|
|
|
loadTenants,
|
|
|
|
|
|
});
|
2026-05-09 05:33:21 +03:00
|
|
|
|
|
2026-05-08 19:23:28 +03:00
|
|
|
|
const filteredTenants = computed<AdminTenant[]>(() => {
|
|
|
|
|
|
const q = search.value.trim().toLowerCase();
|
2026-05-09 05:33:21 +03:00
|
|
|
|
const statuses = new Set(filterStatuses.value);
|
|
|
|
|
|
const tariffs = new Set(filterTariffs.value);
|
|
|
|
|
|
|
2026-05-09 09:19:53 +03:00
|
|
|
|
return tenantsState.filter((t) => {
|
2026-05-09 05:33:21 +03:00
|
|
|
|
if (statuses.size > 0 && !statuses.has(t.status)) return false;
|
|
|
|
|
|
if (tariffs.size > 0 && !tariffs.has(t.tariff)) return false;
|
|
|
|
|
|
if (q) {
|
|
|
|
|
|
const haystack = `${t.name} ${t.inn} ${t.code}`.toLowerCase();
|
|
|
|
|
|
if (!haystack.includes(q)) return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
return true;
|
2026-05-08 19:23:28 +03:00
|
|
|
|
});
|
|
|
|
|
|
});
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<template>
|
|
|
|
|
|
<v-container fluid class="admin-tenants pa-6">
|
2026-05-10 04:38:08 +03:00
|
|
|
|
<TenantsStatsHeader :stats="stats" :loading="loading" @refresh="loadTenants" />
|
2026-05-08 19:23:28 +03:00
|
|
|
|
|
2026-05-09 09:19:53 +03:00
|
|
|
|
<v-alert
|
|
|
|
|
|
v-if="fetchError"
|
|
|
|
|
|
type="warning"
|
|
|
|
|
|
variant="tonal"
|
|
|
|
|
|
density="compact"
|
|
|
|
|
|
closable
|
|
|
|
|
|
class="mt-3"
|
|
|
|
|
|
data-testid="fetch-error-alert"
|
|
|
|
|
|
>
|
|
|
|
|
|
Backend недоступен — показаны mock-данные.
|
|
|
|
|
|
</v-alert>
|
|
|
|
|
|
|
2026-05-10 04:38:08 +03:00
|
|
|
|
<TenantsFilters
|
|
|
|
|
|
:search="search"
|
|
|
|
|
|
:filter-statuses="filterStatuses"
|
|
|
|
|
|
:filter-tariffs="filterTariffs"
|
|
|
|
|
|
:available-tariffs="availableTariffs"
|
|
|
|
|
|
@update:search="search = $event"
|
|
|
|
|
|
@update:filter-statuses="filterStatuses = $event"
|
|
|
|
|
|
@update:filter-tariffs="filterTariffs = $event"
|
|
|
|
|
|
@clear="clearFilters"
|
|
|
|
|
|
/>
|
2026-05-08 19:23:28 +03:00
|
|
|
|
|
2026-05-10 04:38:08 +03:00
|
|
|
|
<TenantsTable :tenants="filteredTenants" @row-click="openTenantDetail" @impersonate="openImpersonation" />
|
2026-05-09 04:52:52 +03:00
|
|
|
|
|
|
|
|
|
|
<ImpersonationDialog v-model="impersonationOpen" :tenant="impersonationTenant" :requested-by="ADMIN_USER_ID" />
|
2026-05-08 19:23:28 +03:00
|
|
|
|
</v-container>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.admin-tenants {
|
|
|
|
|
|
max-width: 1440px;
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|