Compare commits

...

8 Commits

Author SHA1 Message Date
d3m0k1d c3c0e63fd5 chore: update deploy
CI / build (push) Failing after 1m57s
2026-06-14 00:21:13 +03:00
NikitaTorbenko 6367cdae56 fix: home
CI / build (push) Has been cancelled
Deploy / deploy (push) Has been cancelled
2026-06-13 22:39:30 +03:00
NikitaTorbenko e481c8b3a0 fix package.json
Deploy / deploy (push) Has been cancelled
CI / build (push) Has been cancelled
2026-06-13 22:32:53 +03:00
NikitaTorbenko 51c708bf47 feat: create home page
CI / build (push) Has been cancelled
Deploy / deploy (push) Has been cancelled
2026-06-13 22:21:23 +03:00
d3m0k1d 2a108e1b5a feat: ci for all pushes, deploy with ssh key, local compose with build
CI / build (push) Failing after 2m11s
Deploy / deploy (push) Has been cancelled
2026-06-13 21:00:09 +03:00
d3m0k1d fb6bf6e1bf fix: remove stale files, fix ci node version
Deploy / deploy (push) Has been cancelled
ci / build (push) Has been cancelled
2026-06-13 20:56:31 +03:00
d3m0k1d 0f73ca72d7 feat: container registry deploy (build in CI, pull on server)
Deploy / deploy (push) Has been cancelled
ci / build (push) Has been cancelled
2026-06-13 20:56:04 +03:00
d3m0k1d ca6b5f40c3 chore: add dockerfile, nginx.conf and infra/docker-compose.yml 2026-06-13 20:16:01 +03:00
17 changed files with 933 additions and 44 deletions
+25
View File
@@ -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
+41
View File
@@ -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
+7
View File
@@ -0,0 +1,7 @@
services:
app:
build:
context: .
dockerfile: dockerfile
ports:
- "80:80"
+16
View File
@@ -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;"]
+7
View File
@@ -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
+13
View File
@@ -0,0 +1,13 @@
$ANSIBLE_VAULT;1.1;AES256
63663666653739363337653532643363626133303030323462363762316364633838623636626636
3163343137366530326139353638316466663037663935340a386362666236633237313939366639
34626337346365663033386631366362366261366163646438646461376662666665363635396333
3533626234383564390a663966376163366530643965306563363565326438313465383866343138
66633432663430373339326365303033323133383365656231373736323234386435626431383639
63396366333433343039343165633436633839666330646261633338666435353035656230313932
33333630343535646338303539356532306632373433643536393537383463396330366634393962
36356139616432336664613139623038373434643562353565353866303130323938383439396131
30316139333733356462366464653964313264646632336566616536643438326433623363643465
63343430373666356634323761363433666463366431343537613635363239636131643837353935
64316633663334663536656137666330393034666661383165376365666633303764643439366461
33386433643034643466
+47
View File
@@ -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
+6
View File
@@ -0,0 +1,6 @@
---
roles:
- geerlingguy.docker
collections:
- community.general
+5
View File
@@ -0,0 +1,5 @@
services:
app:
image: ${REGISTRY}/hellreign/frontend:${TAG}
ports:
- "80:80"
+31
View File
@@ -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
View File
@@ -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",
+1 -1
View File
@@ -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();
+3 -6
View File
@@ -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>
} }
/> />
+2 -3
View File
@@ -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
View File
@@ -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
View File
@@ -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>
); );
}; };
+5
View File
@@ -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"