first commit

This commit is contained in:
Mateusz Gruszczyński
2026-02-05 12:11:00 +01:00
commit a547894adc
13 changed files with 2335 additions and 0 deletions

324
templates/api.html Normal file
View File

@@ -0,0 +1,324 @@
{% extends "base.html" %}
{% block title %}API Documentation - IP WHOIS Analyzer Pro{% endblock %}
{% block nav_api %}active{% endblock %}
{% block extra_css %}
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/styles/github-dark.min.css">
{% endblock %}
{% block content %}
<!-- API Info -->
<div class="row mb-4">
<div class="col-md-12">
<div class="alert alert-info">
<h5 class="alert-heading"><i class="fas fa-info-circle"></i> API Information</h5>
<p class="mb-1"><strong>Base URL:</strong> <code>http://localhost:5000</code></p>
<p class="mb-1"><strong>Format:</strong> JSON</p>
<p class="mb-0"><strong>Authentication:</strong> None (local use)</p>
</div>
</div>
</div>
<div class="row">
<!-- Sidebar Navigation -->
<div class="col-md-3">
<div class="card mb-4 position-sticky" style="top: 70px;">
<div class="card-header bg-primary text-white">
<h6 class="mb-0"><i class="fas fa-list"></i> Endpoints</h6>
</div>
<div class="list-group list-group-flush">
<a href="#analyze" class="list-group-item list-group-item-action">
<i class="fas fa-search"></i> Analyze IPs
</a>
<a href="#filter" class="list-group-item list-group-item-action">
<i class="fas fa-filter"></i> Filter Results
</a>
<a href="#export-ipset" class="list-group-item list-group-item-action">
<i class="fas fa-list"></i> Export IPSet
</a>
<a href="#export-iptables" class="list-group-item list-group-item-action">
<i class="fas fa-shield-alt"></i> Export iptables
</a>
<a href="#export-nginx" class="list-group-item list-group-item-action">
<i class="fas fa-server"></i> Export Nginx
</a>
<a href="#export-apache" class="list-group-item list-group-item-action">
<i class="fas fa-feather"></i> Export Apache
</a>
<a href="#export-firewalld" class="list-group-item list-group-item-action">
<i class="fas fa-fire"></i> Export Firewalld
</a>
<a href="#export-mikrotik" class="list-group-item list-group-item-action">
<i class="fas fa-network-wired"></i> Export MikroTik
</a>
<a href="#export-cidr" class="list-group-item list-group-item-action">
<i class="fas fa-sitemap"></i> Export CIDR
</a>
<a href="#export-csv" class="list-group-item list-group-item-action">
<i class="fas fa-file-csv"></i> Export CSV
</a>
</div>
</div>
</div>
<!-- Main Content -->
<div class="col-md-9">
<!-- POST /api/analyze -->
<div class="card endpoint-card mb-4" id="analyze">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/analyze
</h4>
</div>
<div class="card-body">
<p>Analyze a list of IP addresses and return WHOIS information (ASN, owner, country, network).</p>
<h6 class="mt-3">Request Body:</h6>
<div class="position-relative">
<button class="btn btn-sm btn-secondary position-absolute top-0 end-0 m-2" onclick="copyCode(this)" style="z-index: 10;">
<i class="fas fa-copy"></i>
</button>
<pre class="mb-0"><code class="language-json">{
"ips": "1.1.1.1, 8.8.8.8, 9.9.9.9"
}</code></pre>
</div>
<h6 class="mt-3">Python Example:</h6>
<div class="position-relative">
<button class="btn btn-sm btn-secondary position-absolute top-0 end-0 m-2" onclick="copyCode(this)" style="z-index: 10;">
<i class="fas fa-copy"></i>
</button>
<pre class="mb-0"><code class="language-python">import requests
response = requests.post('http://localhost:5000/api/analyze',
json={'ips': '1.1.1.1, 8.8.8.8, 9.9.9.9'})
data = response.json()
print(f"Total IPs: {data['stats']['total']}")
for ip_info in data['results']:
print(f"{ip_info['ip']} -> {ip_info['country']} ({ip_info['asn']})")</code></pre>
</div>
</div>
</div>
<!-- POST /api/filter -->
<div class="card endpoint-card mb-4" id="filter">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/filter
</h4>
</div>
<div class="card-body">
<p>Filter results by countries, ASNs, or owners.</p>
<h6 class="mt-3">Python Example:</h6>
<div class="position-relative">
<button class="btn btn-sm btn-secondary position-absolute top-0 end-0 m-2" onclick="copyCode(this)" style="z-index: 10;">
<i class="fas fa-copy"></i>
</button>
<pre class="mb-0"><code class="language-python">import requests
# First, analyze
analyze_response = requests.post('http://localhost:5000/api/analyze',
json={'ips': 'your IP list...'})
results = analyze_response.json()['results']
# Then filter
filter_response = requests.post('http://localhost:5000/api/filter',
json={
'results': results,
'filters': {
'countries': ['CN', 'RU'],
'asns': ['AS4134']
}
})
filtered = filter_response.json()['filtered']
print(f"Filtered: {len(filtered)} IPs")</code></pre>
</div>
</div>
</div>
<!-- POST /api/export/ipset -->
<div class="card endpoint-card mb-4" id="export-ipset">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/ipset
</h4>
</div>
<div class="card-body">
<p>Generate IPSet rules with timeout (default 24 hours).</p>
<h6 class="mt-3">Python Example:</h6>
<div class="position-relative">
<button class="btn btn-sm btn-secondary position-absolute top-0 end-0 m-2" onclick="copyCode(this)" style="z-index: 10;">
<i class="fas fa-copy"></i>
</button>
<pre class="mb-0"><code class="language-python">import requests
response = requests.post('http://localhost:5000/api/export/ipset',
json={
'ips': ['1.1.1.1', '8.8.8.8'],
'timeout': 43200 # 12 hours
})
rules = response.text
with open('block_ips.sh', 'w') as f:
f.write(rules)
print("Saved to block_ips.sh")</code></pre>
</div>
</div>
</div>
<!-- Other endpoints (shorter) -->
<div class="card endpoint-card mb-4" id="export-iptables">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/iptables
</h4>
</div>
<div class="card-body">
<p>Generate iptables DROP rules for INPUT and FORWARD chains.</p>
</div>
</div>
<div class="card endpoint-card mb-4" id="export-nginx">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/nginx
</h4>
</div>
<div class="card-body">
<p>Generate Nginx deny directives.</p>
</div>
</div>
<div class="card endpoint-card mb-4" id="export-apache">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/apache
</h4>
</div>
<div class="card-body">
<p>Generate Apache deny rules (.htaccess or VirtualHost).</p>
</div>
</div>
<div class="card endpoint-card mb-4" id="export-firewalld">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/firewalld
</h4>
</div>
<div class="card-body">
<p>Generate Firewalld rich rules.</p>
</div>
</div>
<div class="card endpoint-card mb-4" id="export-mikrotik">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/mikrotik
</h4>
</div>
<div class="card-body">
<p>Generate MikroTik RouterOS commands (address-list + firewall filter).</p>
</div>
</div>
<div class="card endpoint-card mb-4" id="export-cidr">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/cidr
</h4>
</div>
<div class="card-body">
<p>Export unique CIDR network blocks.</p>
</div>
</div>
<div class="card endpoint-card mb-4" id="export-csv">
<div class="card-header">
<h4 class="mb-0">
<span class="method-badge method-post">POST</span>
/api/export/csv
</h4>
</div>
<div class="card-body">
<p>Export data in CSV format with automatic download.</p>
</div>
</div>
<!-- Complete Workflow Example -->
<div class="card border-success mb-4">
<div class="card-header bg-success text-white">
<h4 class="mb-0"><i class="fas fa-code-branch"></i> Complete Workflow Example</h4>
</div>
<div class="card-body">
<p>Full example: analyze → filter → export to MikroTik</p>
<div class="position-relative">
<button class="btn btn-sm btn-secondary position-absolute top-0 end-0 m-2" onclick="copyCode(this)" style="z-index: 10;">
<i class="fas fa-copy"></i>
</button>
<pre class="mb-0"><code class="language-python">import requests
BASE_URL = 'http://localhost:5000'
# 1. Analyze IPs from log file
with open('/var/log/attacks.log', 'r') as f:
log_content = f.read()
response = requests.post(f'{BASE_URL}/api/analyze',
json={'ips': log_content})
data = response.json()
results = data['results']
print(f"Analyzed: {len(results)} IPs")
# 2. Filter China and Russia only
filter_response = requests.post(f'{BASE_URL}/api/filter',
json={
'results': results,
'filters': {'countries': ['CN', 'RU']}
})
filtered = filter_response.json()['filtered']
ips_to_block = [item['ip'] for item in filtered]
# 3. Generate MikroTik rules
mikrotik_response = requests.post(f'{BASE_URL}/api/export/mikrotik',
json={'ips': ips_to_block})
# 4. Save to file
with open('block_cn_ru.rsc', 'w') as f:
f.write(mikrotik_response.text)
print(f"Generated rules for {len(ips_to_block)} IPs")</code></pre>
</div>
</div>
</div>
</div>
</div>
{% endblock %}
{% block extra_scripts %}
<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/highlight.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/languages/python.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.9.0/languages/json.min.js"></script>
{% endblock %}
{% block page_script %}
<script src="{{ static_hash('js/api.js') }}"></script>
{% endblock %}

59
templates/base.html Normal file
View File

@@ -0,0 +1,59 @@
<!DOCTYPE html>
<html lang="en" data-bs-theme="dark">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>{% block title %}IP WHOIS Analyzer{% endblock %}</title>
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" rel="stylesheet">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
{% block extra_css %}{% endblock %}
<link rel="stylesheet" href="{{ static_hash('css/style.css') }}">
</head>
<body>
<!-- Top Navigation -->
<nav class="navbar navbar-expand-lg navbar-dark bg-dark sticky-top border-bottom">
<div class="container-fluid">
<a class="navbar-brand" href="/">
<i class="fas fa-shield-alt"></i> IP WHOIS Analyzer
</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="navbarNav">
<ul class="navbar-nav ms-auto">
<li class="nav-item">
<a class="nav-link {% block nav_home %}{% endblock %}" href="/">
<i class="fas fa-home"></i> Home
</a>
</li>
<li class="nav-item">
<a class="nav-link {% block nav_api %}{% endblock %}" href="/api">
<i class="fas fa-code"></i> API Documentation
</a>
</li>
<li class="nav-item">
<button class="btn btn-outline-light btn-sm ms-2" onclick="toggleTheme()" title="Toggle theme">
<i class="fas fa-moon" id="themeIcon"></i>
</button>
</li>
</ul>
</div>
</div>
</nav>
<!-- Main Content -->
<div class="container mt-4">
{% block content %}{% endblock %}
</div>
<!-- Footer -->
<footer class="text-center text-muted py-4 mt-5">
<small>{% block footer %}IP WHOIS Analyzer | #MG{% endblock %}</small>
</footer>
<!-- Scripts -->
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js"></script>
{% block extra_scripts %}{% endblock %}
{% block page_script %}{% endblock %}
</body>
</html>

191
templates/index.html Normal file
View File

@@ -0,0 +1,191 @@
{% extends "base.html" %}
{% block title %}Home - IP WHOIS Analyzer{% endblock %}
{% block nav_home %}active{% endblock %}
{% block content %}
<!-- Input Section -->
<div class="card mb-4">
<div class="card-header bg-primary text-white">
<h5 class="mb-0"><i class="fas fa-paste"></i> Paste IP Addresses</h5>
</div>
<div class="card-body">
<textarea class="form-control font-monospace" id="ipInput" rows="10"
placeholder="Paste IP addresses here (comma, semicolon, space, or newline separated)...
Example:
192.168.1.1, 8.8.8.8
1.1.1.1; 9.9.9.9
45.131.219.163"></textarea>
<small class="text-muted d-block mt-2">
<i class="fas fa-info-circle"></i> Supported separators: comma, semicolon, space, newline, tab
</small>
<button class="btn btn-primary btn-lg mt-3" onclick="analyzeIPs()">
<i class="fas fa-search"></i> Analyze IP Addresses
</button>
<small class="text-muted d-block mt-2">
<i class="fas fa-keyboard"></i> Keyboard shortcut: Ctrl+Enter
</small>
</div>
</div>
<!-- Loading -->
<div class="text-center py-5 d-none" id="loading">
<div class="spinner-border text-primary" style="width: 3rem; height: 3rem;" role="status">
<span class="visually-hidden">Loading...</span>
</div>
<p class="mt-3 text-primary fw-bold">Analyzing IP addresses...</p>
<small class="text-muted">Using Team Cymru bulk WHOIS service</small>
</div>
<!-- Error -->
<div class="alert alert-danger d-none" id="error" role="alert"></div>
<!-- Results -->
<div id="results" class="d-none">
<!-- Statistics -->
<div class="card mb-4">
<div class="card-header bg-success text-white">
<h5 class="mb-0"><i class="fas fa-chart-pie"></i> Statistics</h5>
</div>
<div class="card-body">
<div class="row" id="stats"></div>
</div>
</div>
<!-- Filters -->
<div class="card mb-4">
<div class="card-header bg-warning text-dark">
<h5 class="mb-0"><i class="fas fa-filter"></i> Filters</h5>
</div>
<div class="card-body">
<div class="row">
<div class="col-md-4 mb-3">
<label class="form-label fw-bold">
<i class="fas fa-globe"></i> Countries
</label>
<div id="countryFilters" class="filter-container"></div>
</div>
<div class="col-md-4 mb-3">
<label class="form-label fw-bold">
<i class="fas fa-hashtag"></i> ASN
</label>
<div id="asnFilters" class="filter-container"></div>
</div>
<div class="col-md-4 mb-3">
<label class="form-label fw-bold">
<i class="fas fa-building"></i> Owners
</label>
<div id="ownerFilters" class="filter-container"></div>
</div>
</div>
<div class="d-flex justify-content-between align-items-center flex-wrap">
<button class="btn btn-secondary mb-2" onclick="clearFilters()">
<i class="fas fa-times"></i> Clear all filters
</button>
<span class="text-muted mb-2" id="filterCount"></span>
</div>
</div>
</div>
<!-- Export Section -->
<div class="card mb-4">
<div class="card-header bg-info text-white">
<h5 class="mb-0"><i class="fas fa-download"></i> Export Rules</h5>
</div>
<div class="card-body">
<p class="text-muted mb-3">
<i class="fas fa-info-circle"></i> Generate firewall rules based on filtered and selected IPs
</p>
<div class="row g-2">
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportIPSet()">
<i class="fas fa-list"></i> IPSet
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportIPTables()">
<i class="fas fa-shield-alt"></i> iptables
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportNginx()">
<i class="fas fa-server"></i> Nginx
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportApache()">
<i class="fas fa-feather"></i> Apache
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportFirewalld()">
<i class="fas fa-fire"></i> Firewalld
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportMikrotik()">
<i class="fas fa-network-wired"></i> MikroTik
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportCIDR()">
<i class="fas fa-sitemap"></i> CIDR
</button>
</div>
<div class="col-md-3 col-sm-6">
<button class="btn btn-outline-primary w-100" onclick="exportCSV()">
<i class="fas fa-file-csv"></i> CSV
</button>
</div>
</div>
<!-- Export Output -->
<div id="exportOutput" class="export-section d-none mt-4">
<div class="d-flex justify-content-between align-items-center mb-2">
<h6 class="mb-0" id="exportTitle"></h6>
<button class="btn btn-sm btn-success" onclick="copyToClipboard()">
<i class="fas fa-copy"></i> Copy to clipboard
</button>
</div>
<pre><code id="exportCode" class="language-bash"></code></pre>
</div>
</div>
</div>
<!-- Results Table -->
<div class="card">
<div class="card-header bg-dark text-white">
<h5 class="mb-0">
<i class="fas fa-table"></i> Detailed Results
(<span id="tableCount">0</span> IPs)
</h5>
</div>
<div class="card-body p-0">
<div class="table-responsive">
<table class="table table-hover table-striped mb-0">
<thead class="table-dark">
<tr>
<th style="width: 50px;">
<input type="checkbox" class="form-check-input"
id="selectAll" onclick="toggleAll()" checked>
</th>
<th><i class="fas fa-map-marker-alt"></i> IP Address</th>
<th><i class="fas fa-hashtag"></i> ASN</th>
<th><i class="fas fa-building"></i> Owner</th>
<th><i class="fas fa-flag"></i> Country</th>
<th><i class="fas fa-sitemap"></i> Network</th>
</tr>
</thead>
<tbody id="tableBody"></tbody>
</table>
</div>
</div>
</div>
</div>
{% endblock %}
{% block page_script %}
<script src="{{ static_hash('js/main.js') }}"></script>
{% endblock %}