Compare commits
8 Commits
d348e0c347
..
dev
| Author | SHA1 | Date | |
|---|---|---|---|
| c3c0e63fd5 | |||
| 6367cdae56 | |||
| e481c8b3a0 | |||
| 51c708bf47 | |||
| 2a108e1b5a | |||
| fb6bf6e1bf | |||
| 0f73ca72d7 | |||
| ca6b5f40c3 |
@@ -0,0 +1,25 @@
|
|||||||
|
name: CI
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches-ignore:
|
||||||
|
- master
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: "25"
|
||||||
|
|
||||||
|
- name: Install deps
|
||||||
|
run: npm install
|
||||||
|
|
||||||
|
- name: Lint
|
||||||
|
run: npm run lint
|
||||||
|
|
||||||
|
- name: Build
|
||||||
|
run: npm run build
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
name: Deploy
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [master]
|
||||||
|
|
||||||
|
env:
|
||||||
|
REGISTRY: gitea.d3m0k1d.ru
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup SSH
|
||||||
|
run: |
|
||||||
|
mkdir -p ~/.ssh
|
||||||
|
echo "${{ secrets.SSH_PRIVATE_KEY }}" > ~/.ssh/id_ed25519
|
||||||
|
chmod 600 ~/.ssh/id_rsa
|
||||||
|
|
||||||
|
- name: Install Ansible
|
||||||
|
run: |
|
||||||
|
apt update && apt install -y ansible
|
||||||
|
ansible-galaxy install -r infra/ansible/requirements.yml
|
||||||
|
|
||||||
|
- name: Login to registry
|
||||||
|
run: echo "${{ secrets.REGISTRY_PASSWORD }}" | docker login $REGISTRY -u "${{ secrets.REGISTRY_USER }}" --password-stdin
|
||||||
|
|
||||||
|
- name: Build and push
|
||||||
|
run: |
|
||||||
|
IMAGE=$REGISTRY/hellreign/frontend
|
||||||
|
docker build -f dockerfile -t $IMAGE:dev -t $IMAGE:latest .
|
||||||
|
docker push $IMAGE:dev
|
||||||
|
docker push $IMAGE:latest
|
||||||
|
|
||||||
|
- name: Deploy
|
||||||
|
run: |
|
||||||
|
echo "${{ secrets.ANSIBLE_VAULT_PASSWORD }}" > .vault_pass
|
||||||
|
ansible-playbook -i infra/ansible/inventory/hosts.yml infra/ansible/playbook.yml --vault-password-file .vault_pass
|
||||||
|
rm .vault_pass
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
services:
|
||||||
|
app:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
dockerfile: dockerfile
|
||||||
|
ports:
|
||||||
|
- "80:80"
|
||||||
+16
@@ -0,0 +1,16 @@
|
|||||||
|
FROM node:25-alpine3.23 AS builder
|
||||||
|
|
||||||
|
WORKDIR /app
|
||||||
|
|
||||||
|
COPY package.json yarn.lock ./
|
||||||
|
|
||||||
|
RUN yarn install --frozen-lockfile
|
||||||
|
COPY . .
|
||||||
|
RUN yarn build
|
||||||
|
FROM nginx:alpine
|
||||||
|
COPY --from=builder /app/dist /usr/share/nginx/html
|
||||||
|
|
||||||
|
COPY nginx.conf /etc/nginx/conf.d/default.conf
|
||||||
|
|
||||||
|
EXPOSE 80
|
||||||
|
CMD ["nginx", "-g", "daemon off;"]
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
[defaults]
|
||||||
|
inventory = inventory/hosts.yml
|
||||||
|
host_key_checking = False
|
||||||
|
remote_user = root
|
||||||
|
private_key_file = ~/.ssh/id_rsa
|
||||||
|
interpreter_python = /usr/bin/python3
|
||||||
|
stdout_callback = yaml
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
$ANSIBLE_VAULT;1.1;AES256
|
||||||
|
63663666653739363337653532643363626133303030323462363762316364633838623636626636
|
||||||
|
3163343137366530326139353638316466663037663935340a386362666236633237313939366639
|
||||||
|
34626337346365663033386631366362366261366163646438646461376662666665363635396333
|
||||||
|
3533626234383564390a663966376163366530643965306563363565326438313465383866343138
|
||||||
|
66633432663430373339326365303033323133383365656231373736323234386435626431383639
|
||||||
|
63396366333433343039343165633436633839666330646261633338666435353035656230313932
|
||||||
|
33333630343535646338303539356532306632373433643536393537383463396330366634393962
|
||||||
|
36356139616432336664613139623038373434643562353565353866303130323938383439396131
|
||||||
|
30316139333733356462366464653964313264646632336566616536643438326433623363643465
|
||||||
|
63343430373666356634323761363433666463366431343537613635363239636131643837353935
|
||||||
|
64316633663334663536656137666330393034666661383165376365666633303764643439366461
|
||||||
|
33386433643034643466
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
---
|
||||||
|
- name: Deploy Frontend
|
||||||
|
hosts: prod
|
||||||
|
|
||||||
|
pre_tasks:
|
||||||
|
- name: Install docker
|
||||||
|
ansible.builtin.include_role:
|
||||||
|
name: geerlingguy.docker
|
||||||
|
|
||||||
|
- name: Configure ufw
|
||||||
|
community.general.ufw:
|
||||||
|
rule: allow
|
||||||
|
port: "{{ item }}"
|
||||||
|
loop:
|
||||||
|
- "80"
|
||||||
|
- "443"
|
||||||
|
- "2222"
|
||||||
|
|
||||||
|
- name: Enable ufw
|
||||||
|
community.general.ufw:
|
||||||
|
state: enabled
|
||||||
|
tasks:
|
||||||
|
- name: Ensure directory
|
||||||
|
ansible.builtin.file:
|
||||||
|
path: /opt/aegisfront
|
||||||
|
state: directory
|
||||||
|
|
||||||
|
- name: Copy compose
|
||||||
|
ansible.builtin.copy:
|
||||||
|
src: "{{ playbook_dir }}/../docker-compose.yml"
|
||||||
|
dest: /opt/aegisfront/docker-compose.yml
|
||||||
|
|
||||||
|
- name: Pull image
|
||||||
|
ansible.builtin.shell:
|
||||||
|
cmd: docker compose pull
|
||||||
|
chdir: /opt/aegisfront
|
||||||
|
environment:
|
||||||
|
REGISTRY: gitea.d3m0k1d.ru
|
||||||
|
TAG: latest
|
||||||
|
|
||||||
|
- name: Start
|
||||||
|
ansible.builtin.shell:
|
||||||
|
cmd: docker compose up -d --remove-orphans
|
||||||
|
chdir: /opt/aegisfront
|
||||||
|
environment:
|
||||||
|
REGISTRY: gitea.d3m0k1d.ru
|
||||||
|
TAG: latest
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
roles:
|
||||||
|
- geerlingguy.docker
|
||||||
|
|
||||||
|
collections:
|
||||||
|
- community.general
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
services:
|
||||||
|
app:
|
||||||
|
image: ${REGISTRY}/hellreign/frontend:${TAG}
|
||||||
|
ports:
|
||||||
|
- "80:80"
|
||||||
+31
@@ -0,0 +1,31 @@
|
|||||||
|
server {
|
||||||
|
listen 80;
|
||||||
|
server_name localhost;
|
||||||
|
|
||||||
|
root /usr/share/nginx/html;
|
||||||
|
index index.html;
|
||||||
|
|
||||||
|
location / {
|
||||||
|
try_files $uri $uri/ /index.html;
|
||||||
|
}
|
||||||
|
location /api/ {
|
||||||
|
proxy_http_version 1.1;
|
||||||
|
proxy_set_header Upgrade $http_upgrade;
|
||||||
|
proxy_set_header Connection 'upgrade';
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_cache_bypass $http_upgrade;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
location ~* \.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2)$ {
|
||||||
|
expires 1y;
|
||||||
|
add_header Cache-Control "public, immutable";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Gzip сжатие
|
||||||
|
gzip on;
|
||||||
|
gzip_types text/plain text/css application/json application/javascript text/xml application/xml application/xml+rss text/javascript;
|
||||||
|
}
|
||||||
+2
-1
@@ -5,13 +5,14 @@
|
|||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc -b && vite build",
|
"build": "vite build",
|
||||||
"lint": "eslint .",
|
"lint": "eslint .",
|
||||||
"preview": "vite preview"
|
"preview": "vite preview"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@tailwindcss/vite": "^4.3.0",
|
"@tailwindcss/vite": "^4.3.0",
|
||||||
"axios": "^1.17.0",
|
"axios": "^1.17.0",
|
||||||
|
"lucide-react": "^1.18.0",
|
||||||
"react": "^19.2.6",
|
"react": "^19.2.6",
|
||||||
"react-dom": "^19.2.6",
|
"react-dom": "^19.2.6",
|
||||||
"react-icons": "^5.6.0",
|
"react-icons": "^5.6.0",
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ interface ProtectedRouteProps {
|
|||||||
|
|
||||||
export const ProtectedRoute: React.FC<ProtectedRouteProps> = ({
|
export const ProtectedRoute: React.FC<ProtectedRouteProps> = ({
|
||||||
children,
|
children,
|
||||||
fallbackPath = "/",
|
fallbackPath = "/auth",
|
||||||
}) => {
|
}) => {
|
||||||
const isAuthenticated = authService.isAuthenticated();
|
const isAuthenticated = authService.isAuthenticated();
|
||||||
|
|
||||||
|
|||||||
@@ -17,16 +17,13 @@ export const Routing = () => {
|
|||||||
}
|
}
|
||||||
>
|
>
|
||||||
<ReactRoutes>
|
<ReactRoutes>
|
||||||
<Route path="/" element={<AuthPage />} />
|
<Route path="/" element={<HomePage />} />
|
||||||
|
<Route path="/auth" element={<AuthPage />} />
|
||||||
<Route
|
<Route
|
||||||
path="/create-organization"
|
path="/create-organization"
|
||||||
element={<CreateOrganizationPage />}
|
|
||||||
/>
|
|
||||||
<Route
|
|
||||||
path="/home"
|
|
||||||
element={
|
element={
|
||||||
<ProtectedRoute>
|
<ProtectedRoute>
|
||||||
<HomePage />
|
<CreateOrganizationPage />
|
||||||
</ProtectedRoute>
|
</ProtectedRoute>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import type {
|
|||||||
LoginCredentials,
|
LoginCredentials,
|
||||||
RegisterData,
|
RegisterData,
|
||||||
OrganizationCreateData,
|
OrganizationCreateData,
|
||||||
OrganizationMember,
|
|
||||||
} from "../types/auth.types";
|
} from "../types/auth.types";
|
||||||
|
|
||||||
export const useAuth = () => {
|
export const useAuth = () => {
|
||||||
@@ -24,7 +23,7 @@ export const useAuth = () => {
|
|||||||
const orgs = await request(() => authService.getOrganizations());
|
const orgs = await request(() => authService.getOrganizations());
|
||||||
if (orgs && orgs.length > 0) {
|
if (orgs && orgs.length > 0) {
|
||||||
authService.saveOrganization(orgs[0]);
|
authService.saveOrganization(orgs[0]);
|
||||||
navigate("/home");
|
navigate("/");
|
||||||
} else {
|
} else {
|
||||||
navigate("/create-organization");
|
navigate("/create-organization");
|
||||||
}
|
}
|
||||||
@@ -58,7 +57,7 @@ export const useAuth = () => {
|
|||||||
|
|
||||||
if (result) {
|
if (result) {
|
||||||
authService.saveOrganization(result);
|
authService.saveOrganization(result);
|
||||||
navigate("/home");
|
navigate("/");
|
||||||
} else if (error) {
|
} else if (error) {
|
||||||
setAuthError(error);
|
setAuthError(error);
|
||||||
}
|
}
|
||||||
|
|||||||
+15
-2
@@ -1,17 +1,30 @@
|
|||||||
import { useState } from "react";
|
import React, { useState } from "react";
|
||||||
|
import { useNavigate } from "react-router-dom";
|
||||||
import { ThemeToggle } from "@/modules/theme-changer/ui/Theme.toggle";
|
import { ThemeToggle } from "@/modules/theme-changer/ui/Theme.toggle";
|
||||||
import { LoginForm } from "@/modules/auth/components/LoginForm";
|
import { LoginForm } from "@/modules/auth/components/LoginForm";
|
||||||
import { RegisterForm } from "@/modules/auth/components/RegisterForm";
|
import { RegisterForm } from "@/modules/auth/components/RegisterForm";
|
||||||
|
|
||||||
export const AuthPage = () => {
|
export const AuthPage = () => {
|
||||||
const [isLogin, setIsLogin] = useState(true);
|
const [isLogin, setIsLogin] = useState(true);
|
||||||
|
const navigate = useNavigate();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className="min-h-screen flex items-center justify-center p-4 transition-theme"
|
className="min-h-screen flex items-center justify-center p-4 transition-theme"
|
||||||
style={{ backgroundColor: "var(--bg-primary)" }}
|
style={{ backgroundColor: "var(--bg-primary)" }}
|
||||||
>
|
>
|
||||||
<div className="absolute top-4 right-4">
|
<div className="absolute top-4 right-4 flex gap-4">
|
||||||
|
<button
|
||||||
|
onClick={() => navigate("/")}
|
||||||
|
className="px-4 py-2 rounded-lg font-medium transition-all hover:scale-105"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--bg-secondary)",
|
||||||
|
color: "var(--text-primary)",
|
||||||
|
border: `1px solid var(--border)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
На главную
|
||||||
|
</button>
|
||||||
<ThemeToggle />
|
<ThemeToggle />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
+694
-18
@@ -1,47 +1,723 @@
|
|||||||
import { ThemeToggle } from "@/modules/theme-changer/ui/Theme.toggle";
|
import { ThemeToggle } from "@/modules/theme-changer/ui/Theme.toggle";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { useAuth } from "@/modules/auth/hooks/useAuth";
|
import { useAuth } from "@/modules/auth/hooks/useAuth";
|
||||||
|
import { authService } from "@/modules/auth/api/auth.service";
|
||||||
|
import {
|
||||||
|
Shield,
|
||||||
|
Users,
|
||||||
|
Bot,
|
||||||
|
Activity,
|
||||||
|
LogOut,
|
||||||
|
ChevronRight,
|
||||||
|
Zap,
|
||||||
|
Lock,
|
||||||
|
Cloud,
|
||||||
|
TrendingUp,
|
||||||
|
Bell,
|
||||||
|
Server,
|
||||||
|
Sparkles,
|
||||||
|
AlertCircle,
|
||||||
|
CheckCircle,
|
||||||
|
Clock,
|
||||||
|
LogIn,
|
||||||
|
UserPlus,
|
||||||
|
} from "lucide-react";
|
||||||
|
|
||||||
export const HomePage = () => {
|
export const HomePage = () => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const { logout } = useAuth();
|
const { logout } = useAuth();
|
||||||
|
const isAuthenticated = authService.isAuthenticated();
|
||||||
|
const organization = authService.getCurrentOrganization();
|
||||||
|
const authStorage = localStorage.getItem("auth-storage");
|
||||||
|
const user = authStorage ? JSON.parse(authStorage).state?.user : null;
|
||||||
|
|
||||||
|
const stats = [
|
||||||
|
{ label: "Активные агенты", value: "12", change: "+2", icon: Server },
|
||||||
|
{ label: "Заблокировано IP", value: "1,284", change: "+128", icon: Shield },
|
||||||
|
{ label: "Активных правил", value: "47", change: "+5", icon: Lock },
|
||||||
|
{
|
||||||
|
label: "Атак предотвращено",
|
||||||
|
value: "3,721",
|
||||||
|
change: "+342",
|
||||||
|
icon: TrendingUp,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const recentActivities = [
|
||||||
|
{
|
||||||
|
id: 1,
|
||||||
|
type: "attack",
|
||||||
|
message: "Обнаружена brute-force атака с IP 192.168.1.45",
|
||||||
|
time: "2 минуты назад",
|
||||||
|
severity: "high",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 2,
|
||||||
|
type: "rule",
|
||||||
|
message: "Новое правило фильтрации добавлено администратором",
|
||||||
|
time: "15 минут назад",
|
||||||
|
severity: "info",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 3,
|
||||||
|
type: "agent",
|
||||||
|
message: "Агент на сервере web-01 успешно обновлен",
|
||||||
|
time: "1 час назад",
|
||||||
|
severity: "success",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 4,
|
||||||
|
type: "ai",
|
||||||
|
message: "AI предложил новые правила для обнаружения ботов",
|
||||||
|
time: "3 часа назад",
|
||||||
|
severity: "info",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const features = [
|
||||||
|
{
|
||||||
|
title: "Централизованное управление",
|
||||||
|
description: "Управляйте всеми IPS агентами из единого интерфейса",
|
||||||
|
icon: Server,
|
||||||
|
color: "#6366f1",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "AI Аналитика",
|
||||||
|
description: "Искусственный интеллект помогает находить сложные атаки",
|
||||||
|
icon: Bot,
|
||||||
|
color: "#8b5cf6",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "Мгновенная блокировка",
|
||||||
|
description: "Автоматическая блокировка IP при обнаружении угроз",
|
||||||
|
icon: Zap,
|
||||||
|
color: "#f59e0b",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: "Безопасность данных",
|
||||||
|
description: "LLM не получает доступ к чувствительным логам",
|
||||||
|
icon: Lock,
|
||||||
|
color: "#22c55e",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const quickActions = isAuthenticated
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
name: "IPS Агенты",
|
||||||
|
icon: Server,
|
||||||
|
path: "/agents",
|
||||||
|
description: "Управление агентами",
|
||||||
|
color: "#6366f1",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Правила",
|
||||||
|
icon: Shield,
|
||||||
|
path: "/rules",
|
||||||
|
description: "Правила фильтрации",
|
||||||
|
color: "#22c55e",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "AI Аналитика",
|
||||||
|
icon: Bot,
|
||||||
|
path: "/ai-analytics",
|
||||||
|
description: "Анализ логов",
|
||||||
|
color: "#8b5cf6",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Команда",
|
||||||
|
icon: Users,
|
||||||
|
path: "/organization",
|
||||||
|
description: "Управление доступом",
|
||||||
|
color: "#f59e0b",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: [];
|
||||||
|
|
||||||
|
const getSeverityStyles = (severity: string) => {
|
||||||
|
switch (severity) {
|
||||||
|
case "high":
|
||||||
|
return {
|
||||||
|
bg: "#ef444410",
|
||||||
|
border: "#ef4444",
|
||||||
|
icon: AlertCircle,
|
||||||
|
text: "#ef4444",
|
||||||
|
};
|
||||||
|
case "success":
|
||||||
|
return {
|
||||||
|
bg: "#22c55e10",
|
||||||
|
border: "#22c55e",
|
||||||
|
icon: CheckCircle,
|
||||||
|
text: "#22c55e",
|
||||||
|
};
|
||||||
|
default:
|
||||||
|
return {
|
||||||
|
bg: "#6366f110",
|
||||||
|
border: "#6366f1",
|
||||||
|
icon: Sparkles,
|
||||||
|
text: "#6366f1",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
style={{ backgroundColor: "var(--bg-primary)" }}
|
style={{ backgroundColor: "var(--bg-primary)" }}
|
||||||
className="min-h-screen p-4"
|
className="min-h-screen transition-theme"
|
||||||
>
|
>
|
||||||
<div className="flex justify-between items-center mb-8">
|
{/* Header */}
|
||||||
<ThemeToggle />
|
<header
|
||||||
<div className="flex gap-4">
|
className="sticky top-0 z-50 border-b transition-theme backdrop-blur-sm"
|
||||||
<button
|
|
||||||
onClick={() => navigate("/organization")}
|
|
||||||
className="px-4 py-2 rounded-lg font-medium transition-all"
|
|
||||||
style={{
|
style={{
|
||||||
backgroundColor: "var(--accent)",
|
backgroundColor: "var(--bg-primary)",
|
||||||
color: "var(--button-primary-text)",
|
borderColor: "var(--border)",
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
Управление организацией
|
<div className="container mx-auto px-6 py-4">
|
||||||
|
<div className="flex justify-between items-center">
|
||||||
|
{/* Logo */}
|
||||||
|
<div className="flex items-center gap-3">
|
||||||
|
<div
|
||||||
|
className="w-10 h-10 rounded-xl flex items-center justify-center"
|
||||||
|
style={{ backgroundColor: "var(--accent)" }}
|
||||||
|
>
|
||||||
|
<Shield className="w-6 h-6 text-white" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h1
|
||||||
|
className="text-xl font-bold"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
IPS Manager
|
||||||
|
</h1>
|
||||||
|
{organization && (
|
||||||
|
<p className="text-xs" style={{ color: "var(--text-muted)" }}>
|
||||||
|
{organization.name}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Right section */}
|
||||||
|
<div className="flex items-center gap-4">
|
||||||
|
{isAuthenticated ? (
|
||||||
|
<>
|
||||||
|
<button
|
||||||
|
className="relative p-2 rounded-lg transition-all hover:scale-105"
|
||||||
|
style={{ backgroundColor: "var(--bg-secondary)" }}
|
||||||
|
>
|
||||||
|
<Bell
|
||||||
|
className="w-5 h-5"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
className="absolute top-1 right-1 w-2 h-2 rounded-full"
|
||||||
|
style={{ backgroundColor: "#ef4444" }}
|
||||||
|
></span>
|
||||||
</button>
|
</button>
|
||||||
|
<ThemeToggle />
|
||||||
|
<div className="flex items-center gap-3">
|
||||||
|
<div
|
||||||
|
className="w-10 h-10 rounded-full flex items-center justify-center font-semibold text-white"
|
||||||
|
style={{ backgroundColor: "var(--accent)" }}
|
||||||
|
>
|
||||||
|
{user?.first_name?.[0]}
|
||||||
|
{user?.last_name?.[0]}
|
||||||
|
</div>
|
||||||
|
<div className="hidden md:block">
|
||||||
|
<p
|
||||||
|
className="text-sm font-medium"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
{user?.first_name} {user?.last_name}
|
||||||
|
</p>
|
||||||
|
<p
|
||||||
|
className="text-xs"
|
||||||
|
style={{ color: "var(--text-muted)" }}
|
||||||
|
>
|
||||||
|
Администратор
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<button
|
<button
|
||||||
onClick={logout}
|
onClick={logout}
|
||||||
className="px-4 py-2 rounded-lg font-medium transition-all"
|
className="p-2 rounded-lg transition-all hover:scale-105"
|
||||||
|
style={{ backgroundColor: "#ef4444" }}
|
||||||
|
>
|
||||||
|
<LogOut className="w-5 h-5 text-white" />
|
||||||
|
</button>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<ThemeToggle />
|
||||||
|
<button
|
||||||
|
onClick={() => navigate("/auth")}
|
||||||
|
className="px-4 py-2 rounded-lg font-medium transition-all flex items-center gap-2 hover:scale-105"
|
||||||
style={{
|
style={{
|
||||||
backgroundColor: "var(--button-danger)",
|
backgroundColor: "var(--bg-secondary)",
|
||||||
color: "var(--button-danger-text)",
|
color: "var(--text-primary)",
|
||||||
|
border: `1px solid var(--border)`,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
Выйти
|
<LogIn className="w-4 h-4" />
|
||||||
|
Войти
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => navigate("/auth")}
|
||||||
|
className="px-4 py-2 rounded-lg font-medium transition-all flex items-center gap-2 hover:scale-105"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--accent)",
|
||||||
|
color: "white",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<UserPlus className="w-4 h-4" />
|
||||||
|
Регистрация
|
||||||
|
</button>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<main className="container mx-auto px-6 py-8">
|
||||||
|
{/* Hero Section */}
|
||||||
|
<div
|
||||||
|
className="rounded-2xl p-8 mb-8 text-white relative overflow-hidden"
|
||||||
|
style={{
|
||||||
|
background: "linear-gradient(135deg, #6366f1 0%, #4f46e5 100%)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="relative z-10">
|
||||||
|
<div className="flex justify-between items-start">
|
||||||
|
<div>
|
||||||
|
<h2 className="text-3xl font-bold mb-2">
|
||||||
|
{isAuthenticated
|
||||||
|
? `Добро пожаловать, ${user?.first_name || "Администратор"}!`
|
||||||
|
: "Добро пожаловать в IPS Manager"}
|
||||||
|
</h2>
|
||||||
|
<p className="text-white/80 text-lg">
|
||||||
|
{isAuthenticated
|
||||||
|
? "Система IPS мониторинга и защиты"
|
||||||
|
: "Современная платформа для централизованного управления IPS агентами"}
|
||||||
|
</p>
|
||||||
|
{!isAuthenticated && (
|
||||||
|
<div className="flex gap-4 mt-6">
|
||||||
|
<button
|
||||||
|
onClick={() => navigate("/auth")}
|
||||||
|
className="px-6 py-2 rounded-lg font-medium bg-white text-indigo-600 transition-all hover:scale-105 hover:shadow-lg"
|
||||||
|
>
|
||||||
|
Начать работу
|
||||||
|
</button>
|
||||||
|
<button className="px-6 py-2 rounded-lg font-medium border border-white/30 transition-all hover:bg-white/10">
|
||||||
|
Узнать больше
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{isAuthenticated && (
|
||||||
|
<div className="flex items-center gap-2 mt-4">
|
||||||
|
<Zap className="w-5 h-5" />
|
||||||
|
<span className="text-white/90">
|
||||||
|
Все системы работают стабильно
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="hidden lg:block opacity-20">
|
||||||
|
<Cloud className="w-32 h-32" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Stats Grid - только для авторизованных */}
|
||||||
|
{isAuthenticated && (
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6 mb-8">
|
||||||
|
{stats.map((stat, index) => {
|
||||||
|
const Icon = stat.icon;
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={index}
|
||||||
|
className="rounded-2xl p-6 transition-all hover:scale-105 hover:shadow-xl"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--card-bg)",
|
||||||
|
boxShadow: `0 4px 6px -1px var(--shadow-color)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex justify-between items-start mb-4">
|
||||||
|
<div
|
||||||
|
className="w-12 h-12 rounded-xl flex items-center justify-center bg-opacity-20"
|
||||||
|
style={{
|
||||||
|
backgroundColor: `${index === 0 ? "#6366f1" : index === 1 ? "#ef4444" : index === 2 ? "#22c55e" : "#f59e0b"}20`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
className="w-6 h-6"
|
||||||
|
style={{
|
||||||
|
color:
|
||||||
|
index === 0
|
||||||
|
? "#6366f1"
|
||||||
|
: index === 1
|
||||||
|
? "#ef4444"
|
||||||
|
: index === 2
|
||||||
|
? "#22c55e"
|
||||||
|
: "#f59e0b",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span
|
||||||
|
className="text-sm font-medium"
|
||||||
|
style={{ color: "#22c55e" }}
|
||||||
|
>
|
||||||
|
{stat.change}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<h3
|
||||||
|
className="text-2xl font-bold mb-1"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
{stat.value}
|
||||||
|
</h3>
|
||||||
|
<p
|
||||||
|
className="text-sm"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
{stat.label}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Features Section - для всех */}
|
||||||
|
<div className="mb-8">
|
||||||
|
<div className="text-center mb-8">
|
||||||
|
<h2
|
||||||
|
className="text-2xl font-bold mb-2"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
Ключевые возможности
|
||||||
|
</h2>
|
||||||
|
<p className="text-sm" style={{ color: "var(--text-secondary)" }}>
|
||||||
|
Всё необходимое для защиты вашей инфраструктуры
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6">
|
||||||
|
{features.map((feature, index) => {
|
||||||
|
const Icon = feature.icon;
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={index}
|
||||||
|
className="rounded-2xl p-6 transition-all hover:scale-105 hover:shadow-xl text-center"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--card-bg)",
|
||||||
|
boxShadow: `0 4px 6px -1px var(--shadow-color)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="w-14 h-14 rounded-xl flex items-center justify-center mx-auto mb-4"
|
||||||
|
style={{ backgroundColor: `${feature.color}20` }}
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
className="w-7 h-7"
|
||||||
|
style={{ color: feature.color }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<h3
|
||||||
|
className="font-bold mb-2"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
{feature.title}
|
||||||
|
</h3>
|
||||||
|
<p
|
||||||
|
className="text-sm"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
{feature.description}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{isAuthenticated ? (
|
||||||
|
<>
|
||||||
|
<div className="grid grid-cols-1 lg:grid-cols-3 gap-8">
|
||||||
|
{/* Quick Actions */}
|
||||||
|
<div className="lg:col-span-2">
|
||||||
|
<div
|
||||||
|
className="rounded-2xl p-6"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--card-bg)",
|
||||||
|
boxShadow: `0 4px 6px -1px var(--shadow-color)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<h3
|
||||||
|
className="text-xl font-bold mb-6"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
Быстрый доступ
|
||||||
|
</h3>
|
||||||
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-4">
|
||||||
|
{quickActions.map((action, index) => {
|
||||||
|
const Icon = action.icon;
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
key={index}
|
||||||
|
onClick={() => navigate(action.path)}
|
||||||
|
className="group p-4 rounded-xl text-left transition-all hover:scale-[1.02]"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--bg-secondary)",
|
||||||
|
border: `1px solid var(--border)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex items-start justify-between">
|
||||||
|
<div
|
||||||
|
className="w-10 h-10 rounded-lg flex items-center justify-center mb-3 transition-all group-hover:scale-110"
|
||||||
|
style={{ backgroundColor: `${action.color}20` }}
|
||||||
|
>
|
||||||
|
<Icon
|
||||||
|
className="w-5 h-5"
|
||||||
|
style={{ color: action.color }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<ChevronRight
|
||||||
|
className="w-5 h-5 opacity-0 group-hover:opacity-100 transition-all"
|
||||||
|
style={{ color: "var(--text-muted)" }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<h4
|
||||||
|
className="font-semibold mb-1"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
{action.name}
|
||||||
|
</h4>
|
||||||
|
<p
|
||||||
|
className="text-sm"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
{action.description}
|
||||||
|
</p>
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Recent Activity */}
|
||||||
|
<div
|
||||||
|
className="rounded-2xl p-6"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--card-bg)",
|
||||||
|
boxShadow: `0 4px 6px -1px var(--shadow-color)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex justify-between items-center mb-6">
|
||||||
|
<h3
|
||||||
|
className="text-xl font-bold"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
Последние события
|
||||||
|
</h3>
|
||||||
|
<Activity
|
||||||
|
className="w-5 h-5"
|
||||||
|
style={{ color: "var(--text-muted)" }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-4">
|
||||||
|
{recentActivities.map((activity) => {
|
||||||
|
const severityStyles = getSeverityStyles(activity.severity);
|
||||||
|
const Icon = severityStyles.icon;
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={activity.id}
|
||||||
|
className="p-3 rounded-lg transition-all hover:scale-[1.01]"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--bg-secondary)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex gap-3">
|
||||||
|
<div className="flex-shrink-0">
|
||||||
|
<Icon
|
||||||
|
className="w-4 h-4 mt-0.5"
|
||||||
|
style={{ color: severityStyles.text }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex-1 min-w-0">
|
||||||
|
<p
|
||||||
|
className="text-sm mb-1"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
{activity.message}
|
||||||
|
</p>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Clock
|
||||||
|
className="w-3 h-3"
|
||||||
|
style={{ color: "var(--text-muted)" }}
|
||||||
|
/>
|
||||||
|
<p
|
||||||
|
className="text-xs"
|
||||||
|
style={{ color: "var(--text-muted)" }}
|
||||||
|
>
|
||||||
|
{activity.time}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className="w-full mt-4 py-2 rounded-lg text-sm font-medium transition-all hover:scale-[1.02]"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--bg-secondary)",
|
||||||
|
color: "var(--text-secondary)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Показать всю историю
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex items-center justify-center">
|
{/* AI Insights */}
|
||||||
<h1 className="text-2xl" style={{ color: "var(--text-primary)" }}>
|
<div
|
||||||
Добро пожаловать в IPS Manager
|
className="rounded-2xl p-6 mt-8 transition-all hover:shadow-xl"
|
||||||
</h1>
|
style={{
|
||||||
|
backgroundColor: "var(--card-bg)",
|
||||||
|
border: `1px solid var(--border)`,
|
||||||
|
boxShadow: `0 4px 6px -1px var(--shadow-color)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex flex-col sm:flex-row sm:justify-between sm:items-start gap-4 mb-6">
|
||||||
|
<div className="flex items-center gap-3">
|
||||||
|
<div
|
||||||
|
className="w-10 h-10 rounded-xl flex items-center justify-center"
|
||||||
|
style={{ backgroundColor: "#8b5cf620" }}
|
||||||
|
>
|
||||||
|
<Bot className="w-5 h-5" style={{ color: "#8b5cf6" }} />
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
<h3
|
||||||
|
className="text-xl font-bold"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
AI Аналитика
|
||||||
|
</h3>
|
||||||
|
<p
|
||||||
|
className="text-sm"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
Новые предложения на основе анализа логов
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className="px-4 py-2 rounded-lg text-sm font-medium transition-all flex items-center gap-2 hover:scale-[1.02] self-start"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "#8b5cf6",
|
||||||
|
color: "white",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Подробнее
|
||||||
|
<ChevronRight className="w-4 h-4" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
|
||||||
|
<div
|
||||||
|
className="p-4 rounded-xl transition-all hover:scale-[1.02]"
|
||||||
|
style={{ backgroundColor: "var(--bg-secondary)" }}
|
||||||
|
>
|
||||||
|
<p
|
||||||
|
className="text-sm font-medium mb-2"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
🔍 Обнаружена аномалия
|
||||||
|
</p>
|
||||||
|
<p
|
||||||
|
className="text-xs"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
Повышенная активность с IP-адресов из диапазона
|
||||||
|
185.xxx.xx.xx
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="p-4 rounded-xl transition-all hover:scale-[1.02]"
|
||||||
|
style={{ backgroundColor: "var(--bg-secondary)" }}
|
||||||
|
>
|
||||||
|
<p
|
||||||
|
className="text-sm font-medium mb-2"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
💡 Предложено правил
|
||||||
|
</p>
|
||||||
|
<p
|
||||||
|
className="text-xs"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
AI предлагает 3 новых правила для блокировки бот-сканеров
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className="p-4 rounded-xl transition-all hover:scale-[1.02]"
|
||||||
|
style={{ backgroundColor: "var(--bg-secondary)" }}
|
||||||
|
>
|
||||||
|
<p
|
||||||
|
className="text-sm font-medium mb-2"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
⚡ Оптимизация
|
||||||
|
</p>
|
||||||
|
<p
|
||||||
|
className="text-xs"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
Рекомендуется обновить 5 существующих правил для повышения
|
||||||
|
эффективности
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
/* CTA Section для неавторизованных */
|
||||||
|
<div
|
||||||
|
className="rounded-2xl p-8 text-center"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--card-bg)",
|
||||||
|
boxShadow: `0 4px 6px -1px var(--shadow-color)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<h3
|
||||||
|
className="text-2xl font-bold mb-2"
|
||||||
|
style={{ color: "var(--text-primary)" }}
|
||||||
|
>
|
||||||
|
Готовы начать?
|
||||||
|
</h3>
|
||||||
|
<p
|
||||||
|
className="text-sm mb-6"
|
||||||
|
style={{ color: "var(--text-secondary)" }}
|
||||||
|
>
|
||||||
|
Присоединяйтесь к IPS Manager и получите полный контроль над
|
||||||
|
безопасностью
|
||||||
|
</p>
|
||||||
|
<button
|
||||||
|
onClick={() => navigate("/auth")}
|
||||||
|
className="px-6 py-2 rounded-lg font-medium transition-all hover:scale-105"
|
||||||
|
style={{
|
||||||
|
backgroundColor: "var(--accent)",
|
||||||
|
color: "white",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Начать работу
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</main>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1284,6 +1284,11 @@ lru-cache@^5.1.1:
|
|||||||
dependencies:
|
dependencies:
|
||||||
yallist "^3.0.2"
|
yallist "^3.0.2"
|
||||||
|
|
||||||
|
lucide-react@^1.18.0:
|
||||||
|
version "1.18.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/lucide-react/-/lucide-react-1.18.0.tgz#d1b8754d9c427061261c6a6a0b7fa6ecda36c84f"
|
||||||
|
integrity sha512-LZDb7H/0YfM+RJncD0hDQRCAu+vSGODqpe35TuVI8EuXaRjkczbsx7p8dY4J87F/MUSj6bpYqeI8nw8qXaAdmA==
|
||||||
|
|
||||||
magic-string@^0.30.21:
|
magic-string@^0.30.21:
|
||||||
version "0.30.21"
|
version "0.30.21"
|
||||||
resolved "https://registry.yarnpkg.com/magic-string/-/magic-string-0.30.21.tgz#56763ec09a0fa8091df27879fd94d19078c00d91"
|
resolved "https://registry.yarnpkg.com/magic-string/-/magic-string-0.30.21.tgz#56763ec09a0fa8091df27879fd94d19078c00d91"
|
||||||
|
|||||||
Reference in New Issue
Block a user