path-matching #1
7
Gemfile
7
Gemfile
@@ -63,6 +63,9 @@ gem "countries"
|
|||||||
# Authorization library
|
# Authorization library
|
||||||
gem "pundit"
|
gem "pundit"
|
||||||
|
|
||||||
|
# User agent parsing
|
||||||
|
gem "device_detector"
|
||||||
|
|
||||||
group :development, :test do
|
group :development, :test do
|
||||||
# See https://guides.rubyonrails.org/debugging_rails_applications.html#debugging-with-the-debug-gem
|
# See https://guides.rubyonrails.org/debugging_rails_applications.html#debugging-with-the-debug-gem
|
||||||
gem "debug", platforms: %i[ mri windows ], require: "debug/prelude"
|
gem "debug", platforms: %i[ mri windows ], require: "debug/prelude"
|
||||||
@@ -87,3 +90,7 @@ group :test do
|
|||||||
gem "capybara"
|
gem "capybara"
|
||||||
gem "selenium-webdriver"
|
gem "selenium-webdriver"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
gem "sentry-rails", "~> 6.1"
|
||||||
|
|
||||||
|
gem "postgresql_cursor", "~> 0.6.9"
|
||||||
|
|||||||
16
Gemfile.lock
16
Gemfile.lock
@@ -105,12 +105,15 @@ GEM
|
|||||||
xpath (~> 3.2)
|
xpath (~> 3.2)
|
||||||
concurrent-ruby (1.3.5)
|
concurrent-ruby (1.3.5)
|
||||||
connection_pool (2.5.4)
|
connection_pool (2.5.4)
|
||||||
|
countries (8.0.4)
|
||||||
|
unaccent (~> 0.3)
|
||||||
crass (1.0.6)
|
crass (1.0.6)
|
||||||
csv (3.3.5)
|
csv (3.3.5)
|
||||||
date (3.5.0)
|
date (3.5.0)
|
||||||
debug (1.11.0)
|
debug (1.11.0)
|
||||||
irb (~> 1.10)
|
irb (~> 1.10)
|
||||||
reline (>= 0.3.8)
|
reline (>= 0.3.8)
|
||||||
|
device_detector (1.1.3)
|
||||||
dotenv (3.1.8)
|
dotenv (3.1.8)
|
||||||
drb (2.2.3)
|
drb (2.2.3)
|
||||||
ed25519 (1.4.0)
|
ed25519 (1.4.0)
|
||||||
@@ -258,6 +261,8 @@ GEM
|
|||||||
pg (1.6.2-arm64-darwin)
|
pg (1.6.2-arm64-darwin)
|
||||||
pg (1.6.2-x86_64-linux)
|
pg (1.6.2-x86_64-linux)
|
||||||
pg (1.6.2-x86_64-linux-musl)
|
pg (1.6.2-x86_64-linux-musl)
|
||||||
|
postgresql_cursor (0.6.9)
|
||||||
|
activerecord (>= 6.0)
|
||||||
pp (0.6.3)
|
pp (0.6.3)
|
||||||
prettyprint
|
prettyprint
|
||||||
prettyprint (0.2.0)
|
prettyprint (0.2.0)
|
||||||
@@ -371,6 +376,12 @@ GEM
|
|||||||
rexml (~> 3.2, >= 3.2.5)
|
rexml (~> 3.2, >= 3.2.5)
|
||||||
rubyzip (>= 1.2.2, < 4.0)
|
rubyzip (>= 1.2.2, < 4.0)
|
||||||
websocket (~> 1.0)
|
websocket (~> 1.0)
|
||||||
|
sentry-rails (6.1.0)
|
||||||
|
railties (>= 5.2.0)
|
||||||
|
sentry-ruby (~> 6.1.0)
|
||||||
|
sentry-ruby (6.1.0)
|
||||||
|
bigdecimal
|
||||||
|
concurrent-ruby (~> 1.0, >= 1.0.2)
|
||||||
solid_cable (3.0.12)
|
solid_cable (3.0.12)
|
||||||
actioncable (>= 7.2)
|
actioncable (>= 7.2)
|
||||||
activejob (>= 7.2)
|
activejob (>= 7.2)
|
||||||
@@ -430,6 +441,7 @@ GEM
|
|||||||
railties (>= 7.1.0)
|
railties (>= 7.1.0)
|
||||||
tzinfo (2.0.6)
|
tzinfo (2.0.6)
|
||||||
concurrent-ruby (~> 1.0)
|
concurrent-ruby (~> 1.0)
|
||||||
|
unaccent (0.4.0)
|
||||||
unicode-display_width (3.2.0)
|
unicode-display_width (3.2.0)
|
||||||
unicode-emoji (~> 4.1)
|
unicode-emoji (~> 4.1)
|
||||||
unicode-emoji (4.1.0)
|
unicode-emoji (4.1.0)
|
||||||
@@ -473,7 +485,9 @@ DEPENDENCIES
|
|||||||
brakeman
|
brakeman
|
||||||
bundler-audit
|
bundler-audit
|
||||||
capybara
|
capybara
|
||||||
|
countries
|
||||||
debug
|
debug
|
||||||
|
device_detector
|
||||||
httparty
|
httparty
|
||||||
image_processing (~> 1.2)
|
image_processing (~> 1.2)
|
||||||
importmap-rails
|
importmap-rails
|
||||||
@@ -483,12 +497,14 @@ DEPENDENCIES
|
|||||||
openid_connect (~> 2.2)
|
openid_connect (~> 2.2)
|
||||||
pagy
|
pagy
|
||||||
pg (>= 1.1)
|
pg (>= 1.1)
|
||||||
|
postgresql_cursor (~> 0.6.9)
|
||||||
propshaft
|
propshaft
|
||||||
puma (>= 5.0)
|
puma (>= 5.0)
|
||||||
pundit
|
pundit
|
||||||
rails (~> 8.1.1)
|
rails (~> 8.1.1)
|
||||||
rubocop-rails-omakase
|
rubocop-rails-omakase
|
||||||
selenium-webdriver
|
selenium-webdriver
|
||||||
|
sentry-rails (~> 6.1)
|
||||||
solid_cable
|
solid_cable
|
||||||
solid_cache
|
solid_cache
|
||||||
solid_queue
|
solid_queue
|
||||||
|
|||||||
@@ -2,18 +2,29 @@
|
|||||||
import { Controller } from "@hotwired/stimulus"
|
import { Controller } from "@hotwired/stimulus"
|
||||||
|
|
||||||
export default class extends Controller {
|
export default class extends Controller {
|
||||||
static targets = ["form", "toggle", "ruleTypeSelect", "actionSelect", "patternFields", "rateLimitFields", "redirectFields", "helpText", "conditionsField"]
|
static targets = ["form", "toggle", "ruleTypeSelect", "actionSelect", "patternFields", "rateLimitFields", "redirectFields", "helpText", "conditionsField", "expiresAtField"]
|
||||||
|
|
||||||
connect() {
|
connect() {
|
||||||
this.setupEventListeners()
|
console.log("QuickCreateRuleController connected")
|
||||||
this.initializeFieldVisibility()
|
this.initializeFieldVisibility()
|
||||||
}
|
}
|
||||||
|
|
||||||
toggle() {
|
toggle() {
|
||||||
this.formTarget.classList.toggle("hidden")
|
console.log("Toggle method called")
|
||||||
|
console.log("Form target:", this.formTarget)
|
||||||
|
|
||||||
if (this.formTarget.classList.contains("hidden")) {
|
if (this.formTarget) {
|
||||||
this.resetForm()
|
this.formTarget.classList.toggle("hidden")
|
||||||
|
console.log("Toggled hidden class, now:", this.formTarget.classList.contains("hidden"))
|
||||||
|
|
||||||
|
if (this.formTarget.classList.contains("hidden")) {
|
||||||
|
this.resetForm()
|
||||||
|
} else {
|
||||||
|
// Form is being shown, clear the expires_at field for Safari
|
||||||
|
this.clearExpiresAtField()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.error("Form target not found!")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -81,13 +92,28 @@ export default class extends Controller {
|
|||||||
if (this.hasRedirectFieldsTarget) this.redirectFieldsTarget.classList.add("hidden")
|
if (this.hasRedirectFieldsTarget) this.redirectFieldsTarget.classList.add("hidden")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
clearExpiresAtField() {
|
||||||
|
// Clear the expires_at field - much simpler with text field
|
||||||
|
if (this.hasExpiresAtFieldTarget) {
|
||||||
|
this.expiresAtFieldTarget.value = ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
resetForm() {
|
resetForm() {
|
||||||
if (this.formTarget) {
|
if (this.formTarget) {
|
||||||
this.formTarget.reset()
|
// Find the actual form element within the form target div
|
||||||
// Reset rule type to default
|
const formElement = this.formTarget.querySelector('form')
|
||||||
if (this.hasRuleTypeSelectTarget) {
|
if (formElement) {
|
||||||
this.ruleTypeSelectTarget.value = "network"
|
formElement.reset()
|
||||||
this.updateRuleTypeFields()
|
|
||||||
|
// Explicitly clear the expires_at field since browser reset might not clear datetime-local fields properly
|
||||||
|
this.clearExpiresAtField()
|
||||||
|
|
||||||
|
// Reset rule type to default
|
||||||
|
if (this.hasRuleTypeSelectTarget) {
|
||||||
|
this.ruleTypeSelectTarget.value = "network"
|
||||||
|
this.updateRuleTypeFields()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -95,19 +121,8 @@ export default class extends Controller {
|
|||||||
// Private methods
|
// Private methods
|
||||||
|
|
||||||
setupEventListeners() {
|
setupEventListeners() {
|
||||||
// Set up action change listener to show/hide redirect fields
|
// Event listeners are handled via data-action attributes in the HTML
|
||||||
if (this.hasActionSelectTarget) {
|
// No manual event listeners needed
|
||||||
this.actionSelectTarget.addEventListener("change", () => {
|
|
||||||
this.updateRuleTypeFields()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set up toggle button listener
|
|
||||||
if (this.hasToggleTarget) {
|
|
||||||
this.toggleTarget.addEventListener("click", () => {
|
|
||||||
this.toggle()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
initializeFieldVisibility() {
|
initializeFieldVisibility() {
|
||||||
|
|||||||
55
app/javascript/controllers/waf_policy_form_controller.js
Normal file
55
app/javascript/controllers/waf_policy_form_controller.js
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import { Controller } from "@hotwired/stimulus"
|
||||||
|
|
||||||
|
export default class WafPolicyFormController extends Controller {
|
||||||
|
static targets = ["policyTypeSelect", "policyActionSelect", "countryTargets", "asnTargets",
|
||||||
|
"companyTargets", "networkTypeTargets", "redirectConfig", "challengeConfig"]
|
||||||
|
|
||||||
|
connect() {
|
||||||
|
this.updateTargetsVisibility()
|
||||||
|
this.updateActionConfig()
|
||||||
|
}
|
||||||
|
|
||||||
|
updateTargetsVisibility() {
|
||||||
|
const selectedType = this.policyTypeSelectTarget.value
|
||||||
|
|
||||||
|
// Hide all target sections
|
||||||
|
this.countryTargetsTarget.classList.add('hidden')
|
||||||
|
this.asnTargetsTarget.classList.add('hidden')
|
||||||
|
this.companyTargetsTarget.classList.add('hidden')
|
||||||
|
this.networkTypeTargetsTarget.classList.add('hidden')
|
||||||
|
|
||||||
|
// Show relevant target section
|
||||||
|
switch(selectedType) {
|
||||||
|
case 'country':
|
||||||
|
this.countryTargetsTarget.classList.remove('hidden')
|
||||||
|
break
|
||||||
|
case 'asn':
|
||||||
|
this.asnTargetsTarget.classList.remove('hidden')
|
||||||
|
break
|
||||||
|
case 'company':
|
||||||
|
this.companyTargetsTarget.classList.remove('hidden')
|
||||||
|
break
|
||||||
|
case 'network_type':
|
||||||
|
this.networkTypeTargetsTarget.classList.remove('hidden')
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
updateActionConfig() {
|
||||||
|
const selectedAction = this.policyActionSelectTarget.value
|
||||||
|
|
||||||
|
// Hide all config sections
|
||||||
|
this.redirectConfigTarget.classList.add('hidden')
|
||||||
|
this.challengeConfigTarget.classList.add('hidden')
|
||||||
|
|
||||||
|
// Show relevant config section
|
||||||
|
switch(selectedAction) {
|
||||||
|
case 'redirect':
|
||||||
|
this.redirectConfigTarget.classList.remove('hidden')
|
||||||
|
break
|
||||||
|
case 'challenge':
|
||||||
|
this.challengeConfigTarget.classList.remove('hidden')
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
87
app/views/network_ranges/_geolite_data.html.erb
Normal file
87
app/views/network_ranges/_geolite_data.html.erb
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
<% geolite_data = network_range.network_data_for(:geolite) %>
|
||||||
|
|
||||||
|
<% if geolite_data.present? %>
|
||||||
|
<div class="bg-white shadow rounded-lg mb-6">
|
||||||
|
<div class="px-6 py-4 border-b border-gray-200">
|
||||||
|
<h3 class="text-lg font-medium text-gray-900">MaxMind GeoLite2 Data</h3>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="px-6 py-4">
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
|
||||||
|
<!-- ASN Data -->
|
||||||
|
<% if geolite_data['asn'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">ASN (MaxMind)</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900">
|
||||||
|
AS<%= geolite_data['asn']['autonomous_system_number'] %>
|
||||||
|
<% if geolite_data['asn']['autonomous_system_organization'].present? %>
|
||||||
|
<div class="text-xs text-gray-600"><%= geolite_data['asn']['autonomous_system_organization'] %></div>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<!-- Country Data -->
|
||||||
|
<% if geolite_data['country'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">Country (MaxMind)</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900">
|
||||||
|
<%= geolite_data['country']['country_name'] || geolite_data['country']['country_iso_code'] %>
|
||||||
|
<% if geolite_data['country']['country_iso_code'].present? %>
|
||||||
|
<span class="ml-2 text-lg"><%= country_flag(geolite_data['country']['country_iso_code']) %></span>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<% if geolite_data['country']['continent_name'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">Continent</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900">
|
||||||
|
<%= geolite_data['country']['continent_name'] %>
|
||||||
|
<span class="text-xs text-gray-500">(<%= geolite_data['country']['continent_code'] %>)</span>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<% if geolite_data['country']['geoname_id'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">GeoName ID</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900 font-mono">
|
||||||
|
<%= geolite_data['country']['geoname_id'] %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<!-- Flags -->
|
||||||
|
<div class="md:col-span-2 lg:col-span-3">
|
||||||
|
<dt class="text-sm font-medium text-gray-500 mb-2">MaxMind Flags</dt>
|
||||||
|
<dd class="flex flex-wrap gap-2">
|
||||||
|
<% if geolite_data['country']['is_anonymous_proxy'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-red-100 text-red-800">Anonymous Proxy</span>
|
||||||
|
<% end %>
|
||||||
|
<% if geolite_data['country']['is_satellite_provider'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-purple-100 text-purple-800">Satellite Provider</span>
|
||||||
|
<% end %>
|
||||||
|
<% if geolite_data['country']['is_anycast'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-blue-100 text-blue-800">Anycast</span>
|
||||||
|
<% end %>
|
||||||
|
<% if geolite_data['country']['is_in_european_union'] == "1" %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-blue-600 text-white">🇪🇺 EU Member</span>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Raw GeoLite Data (collapsible) -->
|
||||||
|
<details class="mt-6 pt-6 border-t border-gray-200">
|
||||||
|
<summary class="cursor-pointer text-sm font-medium text-gray-700 hover:text-gray-900">
|
||||||
|
Show Raw MaxMind Data
|
||||||
|
</summary>
|
||||||
|
<div class="mt-3">
|
||||||
|
<pre class="bg-gray-50 p-3 rounded-md text-xs overflow-x-auto"><%= JSON.pretty_generate(geolite_data) %></pre>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
112
app/views/network_ranges/_ipapi_data.html.erb
Normal file
112
app/views/network_ranges/_ipapi_data.html.erb
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
<div id="ipapi_data_section" class="bg-white shadow rounded-lg mb-6">
|
||||||
|
<div class="px-6 py-4 border-b border-gray-200">
|
||||||
|
<h3 class="text-lg font-medium text-gray-900">IPAPI Enrichment Data</h3>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<% if ipapi_loading %>
|
||||||
|
<div class="px-6 py-8 text-center">
|
||||||
|
<div class="inline-block animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600"></div>
|
||||||
|
<p class="mt-2 text-sm text-gray-500">Fetching enrichment data...</p>
|
||||||
|
</div>
|
||||||
|
<% elsif ipapi_data.present? %>
|
||||||
|
<div class="px-6 py-4">
|
||||||
|
<% if parent_with_ipapi %>
|
||||||
|
<div class="mb-4 p-3 bg-blue-50 border border-blue-200 rounded-md">
|
||||||
|
<div class="flex items-center">
|
||||||
|
<svg class="w-5 h-5 text-blue-600 mr-2" fill="currentColor" viewBox="0 0 20 20">
|
||||||
|
<path fill-rule="evenodd" d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zm-7-4a1 1 0 11-2 0 1 1 0 012 0zM9 9a1 1 0 000 2v3a1 1 0 001 1h1a1 1 0 100-2v-3a1 1 0 00-1-1H9z" clip-rule="evenodd" />
|
||||||
|
</svg>
|
||||||
|
<span class="text-sm text-blue-800">
|
||||||
|
Data inherited from parent network <%= link_to parent_with_ipapi.cidr, network_range_path(parent_with_ipapi), class: "font-mono font-medium hover:underline" %>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
|
||||||
|
<% if ipapi_data['asn'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">ASN (IPAPI)</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900">
|
||||||
|
AS<%= ipapi_data['asn']['asn'] %>
|
||||||
|
<% if ipapi_data['asn']['org'].present? %>
|
||||||
|
<div class="text-xs text-gray-600"><%= ipapi_data['asn']['org'] %></div>
|
||||||
|
<% end %>
|
||||||
|
<% if ipapi_data['asn']['route'].present? %>
|
||||||
|
<div class="text-xs text-gray-500 font-mono"><%= ipapi_data['asn']['route'] %></div>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<% if ipapi_data['location'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">Location</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900">
|
||||||
|
<%= [ipapi_data['location']['city'], ipapi_data['location']['state'], ipapi_data['location']['country']].compact.join(', ') %>
|
||||||
|
<% if ipapi_data['location']['country_code'].present? %>
|
||||||
|
<span class="ml-2 text-lg"><%= country_flag(ipapi_data['location']['country_code']) %></span>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<% if ipapi_data['company'].present? %>
|
||||||
|
<div>
|
||||||
|
<dt class="text-sm font-medium text-gray-500">Company (IPAPI)</dt>
|
||||||
|
<dd class="mt-1 text-sm text-gray-900">
|
||||||
|
<%= ipapi_data['company']['name'] %>
|
||||||
|
<% if ipapi_data['company']['type'].present? %>
|
||||||
|
<div class="text-xs text-gray-600"><%= ipapi_data['company']['type'].humanize %></div>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<% if ipapi_data['is_datacenter'] || ipapi_data['is_vpn'] || ipapi_data['is_proxy'] || ipapi_data['is_tor'] %>
|
||||||
|
<div class="md:col-span-2 lg:col-span-3">
|
||||||
|
<dt class="text-sm font-medium text-gray-500 mb-2">IPAPI Flags</dt>
|
||||||
|
<dd class="flex flex-wrap gap-2">
|
||||||
|
<% if ipapi_data['is_datacenter'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-orange-100 text-orange-800">Datacenter</span>
|
||||||
|
<% end %>
|
||||||
|
<% if ipapi_data['is_vpn'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-purple-100 text-purple-800">VPN</span>
|
||||||
|
<% end %>
|
||||||
|
<% if ipapi_data['is_proxy'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-red-100 text-red-800">Proxy</span>
|
||||||
|
<% end %>
|
||||||
|
<% if ipapi_data['is_tor'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-gray-800 text-white">Tor</span>
|
||||||
|
<% end %>
|
||||||
|
<% if ipapi_data['is_abuser'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-red-600 text-white">Abuser</span>
|
||||||
|
<% end %>
|
||||||
|
<% if ipapi_data['is_bogon'] %>
|
||||||
|
<span class="inline-flex items-center px-3 py-1 rounded-full text-sm font-medium bg-yellow-100 text-yellow-800">Bogon</span>
|
||||||
|
<% end %>
|
||||||
|
</dd>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Raw IPAPI Data (collapsible) -->
|
||||||
|
<details class="mt-6 pt-6 border-t border-gray-200">
|
||||||
|
<summary class="cursor-pointer text-sm font-medium text-gray-700 hover:text-gray-900">
|
||||||
|
Show Raw IPAPI Data
|
||||||
|
</summary>
|
||||||
|
<div class="mt-3">
|
||||||
|
<pre class="bg-gray-50 p-3 rounded-md text-xs overflow-x-auto"><%= JSON.pretty_generate(ipapi_data) %></pre>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
</div>
|
||||||
|
<% else %>
|
||||||
|
<div class="px-6 py-8 text-center text-gray-500">
|
||||||
|
<svg class="mx-auto h-12 w-12 text-gray-400" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
|
||||||
|
</svg>
|
||||||
|
<h3 class="mt-2 text-sm font-medium text-gray-900">No IPAPI data available</h3>
|
||||||
|
<p class="mt-1 text-sm text-gray-500">Enrichment data will be fetched automatically.</p>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
@@ -1,5 +1,9 @@
|
|||||||
<% content_for :title, "#{@network_range.cidr} - Network Range Details" %>
|
<% content_for :title, "#{@network_range.cidr} - Network Range Details" %>
|
||||||
|
|
||||||
|
<% if @network_range.persisted? %>
|
||||||
|
<%= turbo_stream_from "network_range_#{@network_range.id}" %>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<div class="mx-auto max-w-7xl px-4 py-6 sm:px-6 lg:px-8">
|
<div class="mx-auto max-w-7xl px-4 py-6 sm:px-6 lg:px-8">
|
||||||
<!-- Header -->
|
<!-- Header -->
|
||||||
<div class="mb-8">
|
<div class="mb-8">
|
||||||
@@ -48,6 +52,23 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- IPAPI Enrichment Data -->
|
||||||
|
<% if @network_range.persisted? %>
|
||||||
|
<%= render partial: "network_ranges/ipapi_data", locals: {
|
||||||
|
ipapi_data: @ipapi_data,
|
||||||
|
network_range: @network_range,
|
||||||
|
parent_with_ipapi: @parent_with_ipapi,
|
||||||
|
ipapi_loading: @ipapi_loading || false
|
||||||
|
} %>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
|
<!-- MaxMind GeoLite2 Data -->
|
||||||
|
<% if @network_range.persisted? %>
|
||||||
|
<%= render partial: "network_ranges/geolite_data", locals: {
|
||||||
|
network_range: @network_range
|
||||||
|
} %>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<!-- Network Intelligence Card -->
|
<!-- Network Intelligence Card -->
|
||||||
<div class="bg-white shadow rounded-lg mb-6">
|
<div class="bg-white shadow rounded-lg mb-6">
|
||||||
<div class="px-6 py-4 border-b border-gray-200">
|
<div class="px-6 py-4 border-b border-gray-200">
|
||||||
@@ -335,9 +356,12 @@
|
|||||||
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||||||
<div>
|
<div>
|
||||||
<%= form.label :expires_at, "Expires At (Optional)", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :expires_at, "Expires At (Optional)", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= form.datetime_local_field :expires_at,
|
<%= form.text_field :expires_at,
|
||||||
class: "mt-1 block w-full rounded-md border-gray-300 shadow-sm focus:border-blue-500 focus:ring-blue-500 sm:text-sm" %>
|
placeholder: "YYYY-MM-DD HH:MM (24-hour format, optional)",
|
||||||
<p class="mt-1 text-xs text-gray-500">Leave blank for permanent rule</p>
|
class: "mt-1 block w-full rounded-md border-gray-300 shadow-sm focus:border-blue-500 focus:ring-blue-500 sm:text-sm",
|
||||||
|
data: { quick_create_rule_target: "expiresAtField" },
|
||||||
|
autocomplete: "off" %>
|
||||||
|
<p class="mt-1 text-xs text-gray-500">Leave blank for permanent rule. Format: YYYY-MM-DD HH:MM (e.g., 2024-12-31 23:59)</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="text-sm text-gray-600 flex items-center pt-6">
|
<div class="text-sm text-gray-600 flex items-center pt-6">
|
||||||
@@ -461,9 +485,9 @@
|
|||||||
<div class="mt-1 text-sm text-gray-500">
|
<div class="mt-1 text-sm text-gray-500">
|
||||||
Created <%= time_ago_in_words(rule.created_at) %> ago by <%= rule.user&.email_address || 'System' %>
|
Created <%= time_ago_in_words(rule.created_at) %> ago by <%= rule.user&.email_address || 'System' %>
|
||||||
</div>
|
</div>
|
||||||
<% if rule.metadata&.dig('reason').present? %>
|
<% if rule.metadata_hash['reason'].present? %>
|
||||||
<div class="mt-1 text-sm text-gray-600">
|
<div class="mt-1 text-sm text-gray-600">
|
||||||
Reason: <%= rule.metadata['reason'] %>
|
Reason: <%= rule.metadata_hash['reason'] %>
|
||||||
</div>
|
</div>
|
||||||
<% end %>
|
<% end %>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<%= form_with(model: @waf_policy, local: true, class: "space-y-6") do |form| %>
|
<%= form_with(model: @waf_policy, local: true, class: "space-y-6", data: { controller: "waf-policy-form" }) do |form| %>
|
||||||
<!-- Basic Information -->
|
<!-- Basic Information -->
|
||||||
<div class="bg-white shadow rounded-lg">
|
<div class="bg-white shadow rounded-lg">
|
||||||
<div class="px-4 py-5 sm:p-6 space-y-4">
|
<div class="px-4 py-5 sm:p-6 space-y-4">
|
||||||
@@ -35,14 +35,14 @@
|
|||||||
placeholder: "Explain why this policy is needed..." %>
|
placeholder: "Explain why this policy is needed..." %>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Action -->
|
<!-- Policy Action -->
|
||||||
<div>
|
<div>
|
||||||
<%= form.label :action, "Action", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :policy_action, "Policy Action", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= form.select :action,
|
<%= form.select :policy_action,
|
||||||
options_for_select(@actions.map { |action| [action.humanize, action] }, @waf_policy.action),
|
options_for_select(@actions.map { |action| [action.humanize, action] }, @waf_policy.policy_action),
|
||||||
{ prompt: "Select action" },
|
{ prompt: "Select action" },
|
||||||
{ class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
{ class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
||||||
id: "action-select" } %>
|
data: { "waf-policy-form-target": "policyActionSelect", "action": "change->waf-policy-form#updateActionConfig" } } %>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Status -->
|
<!-- Status -->
|
||||||
@@ -164,7 +164,7 @@
|
|||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">⚙️ Additional Configuration</h3>
|
<h3 class="text-lg leading-6 font-medium text-gray-900">⚙️ Additional Configuration</h3>
|
||||||
|
|
||||||
<!-- Redirect Settings (for redirect action) -->
|
<!-- Redirect Settings (for redirect action) -->
|
||||||
<div id="redirect-config" class="space-y-3 <%= 'hidden' unless @waf_policy.redirect_action? %>">
|
<div id="redirect-config" class="space-y-3 <%= 'hidden' unless @waf_policy.redirect_action? %>" data-waf-policy-form-target="redirectConfig">
|
||||||
<div>
|
<div>
|
||||||
<%= label_tag "additional_data[redirect_url]", "Redirect URL", class: "block text-sm font-medium text-gray-700" %>
|
<%= label_tag "additional_data[redirect_url]", "Redirect URL", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= text_field_tag "additional_data[redirect_url]", @waf_policy.additional_data&.dig('redirect_url'),
|
<%= text_field_tag "additional_data[redirect_url]", @waf_policy.additional_data&.dig('redirect_url'),
|
||||||
@@ -180,7 +180,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Challenge Settings (for challenge action) -->
|
<!-- Challenge Settings (for challenge action) -->
|
||||||
<div id="challenge-config" class="space-y-3 <%= 'hidden' unless @waf_policy.challenge_action? %>">
|
<div id="challenge-config" class="space-y-3 <%= 'hidden' unless @waf_policy.challenge_action? %>" data-waf-policy-form-target="challengeConfig">
|
||||||
<div>
|
<div>
|
||||||
<%= label_tag "additional_data[challenge_type]", "Challenge Type", class: "block text-sm font-medium text-gray-700" %>
|
<%= label_tag "additional_data[challenge_type]", "Challenge Type", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= select_tag "additional_data[challenge_type]",
|
<%= select_tag "additional_data[challenge_type]",
|
||||||
@@ -206,35 +206,3 @@
|
|||||||
</div>
|
</div>
|
||||||
<% end %>
|
<% end %>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
|
||||||
const actionSelect = document.getElementById('action-select');
|
|
||||||
const redirectConfig = document.getElementById('redirect-config');
|
|
||||||
const challengeConfig = document.getElementById('challenge-config');
|
|
||||||
|
|
||||||
function updateActionConfig() {
|
|
||||||
const selectedAction = actionSelect.value;
|
|
||||||
|
|
||||||
// Hide all config sections
|
|
||||||
redirectConfig.classList.add('hidden');
|
|
||||||
challengeConfig.classList.add('hidden');
|
|
||||||
|
|
||||||
// Show relevant config section
|
|
||||||
switch(selectedAction) {
|
|
||||||
case 'redirect':
|
|
||||||
redirectConfig.classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
case 'challenge':
|
|
||||||
challengeConfig.classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add event listener
|
|
||||||
actionSelect.addEventListener('change', updateActionConfig);
|
|
||||||
|
|
||||||
// Initial update
|
|
||||||
updateActionConfig();
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
@@ -85,7 +85,7 @@
|
|||||||
<dl>
|
<dl>
|
||||||
<dt class="text-sm font-medium text-gray-500 truncate">Deny Policies</dt>
|
<dt class="text-sm font-medium text-gray-500 truncate">Deny Policies</dt>
|
||||||
<dd class="text-lg font-medium text-gray-900">
|
<dd class="text-lg font-medium text-gray-900">
|
||||||
<%= number_with_delimiter(@waf_policies.where(action: 'deny').count) %>
|
<%= number_with_delimiter(@waf_policies.where(policy_action: 'deny').count) %>
|
||||||
</dd>
|
</dd>
|
||||||
</dl>
|
</dl>
|
||||||
</div>
|
</div>
|
||||||
@@ -137,15 +137,15 @@
|
|||||||
</span>
|
</span>
|
||||||
<% end %>
|
<% end %>
|
||||||
|
|
||||||
<!-- Action Badge -->
|
<!-- Policy Action Badge -->
|
||||||
<span class="ml-2 inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium
|
<span class="ml-2 inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium
|
||||||
<%= case policy.action
|
<%= case policy.policy_action
|
||||||
when 'deny' then 'bg-red-100 text-red-800'
|
when 'deny' then 'bg-red-100 text-red-800'
|
||||||
when 'allow' then 'bg-green-100 text-green-800'
|
when 'allow' then 'bg-green-100 text-green-800'
|
||||||
when 'redirect' then 'bg-yellow-100 text-yellow-800'
|
when 'redirect' then 'bg-yellow-100 text-yellow-800'
|
||||||
when 'challenge' then 'bg-purple-100 text-purple-800'
|
when 'challenge' then 'bg-purple-100 text-purple-800'
|
||||||
end %>">
|
end %>">
|
||||||
<%= policy.action.upcase %>
|
<%= policy.policy_action.upcase %>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-1 text-sm text-gray-500">
|
<div class="mt-1 text-sm text-gray-500">
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<%= form_with(model: @waf_policy, local: true, class: "space-y-6") do |form| %>
|
<%= form_with(model: @waf_policy, local: true, class: "space-y-6", data: { controller: "waf-policy-form" }) do |form| %>
|
||||||
<!-- Basic Information -->
|
<!-- Basic Information -->
|
||||||
<div class="bg-white shadow rounded-lg">
|
<div class="bg-white shadow rounded-lg">
|
||||||
<div class="px-4 py-5 sm:p-6 space-y-4">
|
<div class="px-4 py-5 sm:p-6 space-y-4">
|
||||||
@@ -42,17 +42,17 @@
|
|||||||
options_for_select(@policy_types.map { |type| [type.humanize, type] }, @waf_policy.policy_type),
|
options_for_select(@policy_types.map { |type| [type.humanize, type] }, @waf_policy.policy_type),
|
||||||
{ prompt: "Select policy type" },
|
{ prompt: "Select policy type" },
|
||||||
{ class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
{ class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
||||||
id: "policy-type-select" } %>
|
data: { "waf-policy-form-target": "policyTypeSelect", "action": "change->waf-policy-form#updateTargetsVisibility" } } %>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Action -->
|
<!-- Policy Action -->
|
||||||
<div>
|
<div>
|
||||||
<%= form.label :action, "Action", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :policy_action, "Policy Action", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= form.select :action,
|
<%= form.select :policy_action,
|
||||||
options_for_select(@actions.map { |action| [action.humanize, action] }, @waf_policy.action),
|
options_for_select(@actions.map { |action| [action.humanize, action] }, @waf_policy.policy_action),
|
||||||
{ prompt: "Select action" },
|
{ prompt: "Select action" },
|
||||||
{ class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
{ class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
||||||
id: "action-select" } %>
|
data: { "waf-policy-form-target": "policyActionSelect", "action": "change->waf-policy-form#updateActionConfig" } } %>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -63,7 +63,7 @@
|
|||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">🎯 Targets Configuration</h3>
|
<h3 class="text-lg leading-6 font-medium text-gray-900">🎯 Targets Configuration</h3>
|
||||||
|
|
||||||
<!-- Country Policy Targets -->
|
<!-- Country Policy Targets -->
|
||||||
<div id="country-targets" class="policy-targets hidden">
|
<div id="country-targets" class="policy-targets hidden" data-waf-policy-form-target="countryTargets">
|
||||||
<%= form.label :targets, "Countries", class: "block text-sm font-medium text-gray-700 mb-2" %>
|
<%= form.label :targets, "Countries", class: "block text-sm font-medium text-gray-700 mb-2" %>
|
||||||
<div data-controller="country-selector"
|
<div data-controller="country-selector"
|
||||||
data-country-selector-options-value="<%= CountryHelper.all_for_select.to_json %>"
|
data-country-selector-options-value="<%= CountryHelper.all_for_select.to_json %>"
|
||||||
@@ -79,7 +79,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- ASN Policy Targets -->
|
<!-- ASN Policy Targets -->
|
||||||
<div id="asn-targets" class="policy-targets hidden">
|
<div id="asn-targets" class="policy-targets hidden" data-waf-policy-form-target="asnTargets">
|
||||||
<%= form.label :targets, "ASN Numbers", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :targets, "ASN Numbers", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= text_field_tag "waf_policy[targets][]", nil,
|
<%= text_field_tag "waf_policy[targets][]", nil,
|
||||||
class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
||||||
@@ -88,7 +88,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Company Policy Targets -->
|
<!-- Company Policy Targets -->
|
||||||
<div id="company-targets" class="policy-targets hidden">
|
<div id="company-targets" class="policy-targets hidden" data-waf-policy-form-target="companyTargets">
|
||||||
<%= form.label :targets, "Companies", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :targets, "Companies", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= text_field_tag "waf_policy[targets][]", nil,
|
<%= text_field_tag "waf_policy[targets][]", nil,
|
||||||
class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
class: "block w-full border-gray-300 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 sm:text-sm",
|
||||||
@@ -97,7 +97,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Network Type Targets -->
|
<!-- Network Type Targets -->
|
||||||
<div id="network-type-targets" class="policy-targets hidden">
|
<div id="network-type-targets" class="policy-targets hidden" data-waf-policy-form-target="networkTypeTargets">
|
||||||
<%= form.label :targets, "Network Types", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :targets, "Network Types", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<div class="space-y-2">
|
<div class="space-y-2">
|
||||||
<label class="flex items-center">
|
<label class="flex items-center">
|
||||||
@@ -123,7 +123,7 @@
|
|||||||
<h3 class="text-lg leading-6 font-medium text-gray-900">⚙️ Additional Configuration</h3>
|
<h3 class="text-lg leading-6 font-medium text-gray-900">⚙️ Additional Configuration</h3>
|
||||||
|
|
||||||
<!-- Redirect Settings (for redirect action) -->
|
<!-- Redirect Settings (for redirect action) -->
|
||||||
<div id="redirect-config" class="hidden space-y-3">
|
<div id="redirect-config" class="hidden space-y-3" data-waf-policy-form-target="redirectConfig">
|
||||||
<div>
|
<div>
|
||||||
<%= label_tag "additional_data[redirect_url]", "Redirect URL", class: "block text-sm font-medium text-gray-700" %>
|
<%= label_tag "additional_data[redirect_url]", "Redirect URL", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= text_field_tag "additional_data[redirect_url]", nil,
|
<%= text_field_tag "additional_data[redirect_url]", nil,
|
||||||
@@ -139,7 +139,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Challenge Settings (for challenge action) -->
|
<!-- Challenge Settings (for challenge action) -->
|
||||||
<div id="challenge-config" class="hidden space-y-3">
|
<div id="challenge-config" class="hidden space-y-3" data-waf-policy-form-target="challengeConfig">
|
||||||
<div>
|
<div>
|
||||||
<%= label_tag "additional_data[challenge_type]", "Challenge Type", class: "block text-sm font-medium text-gray-700" %>
|
<%= label_tag "additional_data[challenge_type]", "Challenge Type", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<%= select_tag "additional_data[challenge_type]",
|
<%= select_tag "additional_data[challenge_type]",
|
||||||
@@ -179,62 +179,3 @@
|
|||||||
</div>
|
</div>
|
||||||
<% end %>
|
<% end %>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
|
||||||
const policyTypeSelect = document.getElementById('policy-type-select');
|
|
||||||
const actionSelect = document.getElementById('action-select');
|
|
||||||
const allTargets = document.querySelectorAll('.policy-targets');
|
|
||||||
const redirectConfig = document.getElementById('redirect-config');
|
|
||||||
const challengeConfig = document.getElementById('challenge-config');
|
|
||||||
|
|
||||||
function updateTargetsVisibility() {
|
|
||||||
const selectedType = policyTypeSelect.value;
|
|
||||||
|
|
||||||
// Hide all target sections
|
|
||||||
allTargets.forEach(target => target.classList.add('hidden'));
|
|
||||||
|
|
||||||
// Show relevant target section
|
|
||||||
switch(selectedType) {
|
|
||||||
case 'country':
|
|
||||||
document.getElementById('country-targets').classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
case 'asn':
|
|
||||||
document.getElementById('asn-targets').classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
case 'company':
|
|
||||||
document.getElementById('company-targets').classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
case 'network_type':
|
|
||||||
document.getElementById('network-type-targets').classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateActionConfig() {
|
|
||||||
const selectedAction = actionSelect.value;
|
|
||||||
|
|
||||||
// Hide all config sections
|
|
||||||
redirectConfig.classList.add('hidden');
|
|
||||||
challengeConfig.classList.add('hidden');
|
|
||||||
|
|
||||||
// Show relevant config section
|
|
||||||
switch(selectedAction) {
|
|
||||||
case 'redirect':
|
|
||||||
redirectConfig.classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
case 'challenge':
|
|
||||||
challengeConfig.classList.remove('hidden');
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add event listeners
|
|
||||||
policyTypeSelect.addEventListener('change', updateTargetsVisibility);
|
|
||||||
actionSelect.addEventListener('change', updateActionConfig);
|
|
||||||
|
|
||||||
// Initial update
|
|
||||||
updateTargetsVisibility();
|
|
||||||
updateActionConfig();
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
@@ -14,6 +14,31 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<%= form_with(url: create_country_waf_policies_path, method: :post, local: true, class: "space-y-6") do |form| %>
|
<%= form_with(url: create_country_waf_policies_path, method: :post, local: true, class: "space-y-6") do |form| %>
|
||||||
|
<!-- Display validation errors -->
|
||||||
|
<% if defined?(@waf_policy) && @waf_policy&.errors&.any? %>
|
||||||
|
<div class="rounded-md bg-red-50 p-4 border border-red-200">
|
||||||
|
<div class="flex">
|
||||||
|
<div class="flex-shrink-0">
|
||||||
|
<svg class="h-5 w-5 text-red-400" viewBox="0 0 20 20" fill="currentColor">
|
||||||
|
<path fill-rule="evenodd" d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" clip-rule="evenodd"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div class="ml-3">
|
||||||
|
<h3 class="text-sm font-medium text-red-800">
|
||||||
|
<%= pluralize(@waf_policy.errors.count, "error") %> prohibited this policy from being saved:
|
||||||
|
</h3>
|
||||||
|
<div class="mt-2 text-sm text-red-700">
|
||||||
|
<ul class="list-disc pl-5 space-y-1">
|
||||||
|
<% @waf_policy.errors.full_messages.each do |message| %>
|
||||||
|
<li><%= message %></li>
|
||||||
|
<% end %>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
||||||
|
|
||||||
<!-- Popular Countries Quick Selection -->
|
<!-- Popular Countries Quick Selection -->
|
||||||
<div class="bg-white shadow rounded-lg">
|
<div class="bg-white shadow rounded-lg">
|
||||||
<div class="px-4 py-5 sm:p-6">
|
<div class="px-4 py-5 sm:p-6">
|
||||||
@@ -67,28 +92,28 @@
|
|||||||
|
|
||||||
<!-- Action Selection -->
|
<!-- Action Selection -->
|
||||||
<div>
|
<div>
|
||||||
<%= form.label :action, "What should happen to traffic from selected countries?", class: "block text-sm font-medium text-gray-700" %>
|
<%= form.label :policy_action, "What should happen to traffic from selected countries?", class: "block text-sm font-medium text-gray-700" %>
|
||||||
<div class="mt-2 space-y-2">
|
<div class="mt-2 space-y-2">
|
||||||
<label class="flex items-center">
|
<label class="flex items-center">
|
||||||
<%= radio_button_tag "action", "deny", true, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
<%= radio_button_tag "policy_action", "deny", true, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
||||||
<span class="ml-2 text-sm text-gray-700">
|
<span class="ml-2 text-sm text-gray-700">
|
||||||
<span class="font-medium">🚫 Block (Deny)</span> - Show 403 Forbidden error
|
<span class="font-medium">🚫 Block (Deny)</span> - Show 403 Forbidden error
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
<label class="flex items-center">
|
<label class="flex items-center">
|
||||||
<%= radio_button_tag "action", "challenge", false, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
<%= radio_button_tag "policy_action", "challenge", false, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
||||||
<span class="ml-2 text-sm text-gray-700">
|
<span class="ml-2 text-sm text-gray-700">
|
||||||
<span class="font-medium">🛡️ Challenge</span> - Present CAPTCHA challenge
|
<span class="font-medium">🛡️ Challenge</span> - Present CAPTCHA challenge
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
<label class="flex items-center">
|
<label class="flex items-center">
|
||||||
<%= radio_button_tag "action", "redirect", false, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
<%= radio_button_tag "policy_action", "redirect", false, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
||||||
<span class="ml-2 text-sm text-gray-700">
|
<span class="ml-2 text-sm text-gray-700">
|
||||||
<span class="font-medium">🔄 Redirect</span> - Redirect to compliance page
|
<span class="font-medium">🔄 Redirect</span> - Redirect to compliance page
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
<label class="flex items-center">
|
<label class="flex items-center">
|
||||||
<%= radio_button_tag "action", "allow", false, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
<%= radio_button_tag "policy_action", "allow", false, class: "h-4 w-4 text-blue-600 focus:ring-blue-500 border-gray-300" %>
|
||||||
<span class="ml-2 text-sm text-gray-700">
|
<span class="ml-2 text-sm text-gray-700">
|
||||||
<span class="font-medium">✅ Allow</span> - Explicitly allow traffic
|
<span class="font-medium">✅ Allow</span> - Explicitly allow traffic
|
||||||
</span>
|
</span>
|
||||||
@@ -138,14 +163,14 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
// Show/hide redirect settings based on action selection
|
// Show/hide redirect settings based on policy action selection
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
const actionRadios = document.querySelectorAll('input[name="action"]');
|
const actionRadios = document.querySelectorAll('input[name="policy_action"]');
|
||||||
const redirectSettings = document.getElementById('redirect-settings');
|
const redirectSettings = document.getElementById('redirect-settings');
|
||||||
const previewText = document.getElementById('preview-text');
|
const previewText = document.getElementById('preview-text');
|
||||||
|
|
||||||
function updateVisibility() {
|
function updateVisibility() {
|
||||||
const selectedAction = document.querySelector('input[name="action"]:checked')?.value;
|
const selectedAction = document.querySelector('input[name="policy_action"]:checked')?.value;
|
||||||
|
|
||||||
if (selectedAction === 'redirect') {
|
if (selectedAction === 'redirect') {
|
||||||
redirectSettings.classList.remove('hidden');
|
redirectSettings.classList.remove('hidden');
|
||||||
@@ -158,7 +183,7 @@ document.addEventListener('DOMContentLoaded', function() {
|
|||||||
|
|
||||||
function updatePreview() {
|
function updatePreview() {
|
||||||
const selectedCountries = document.querySelectorAll('input[name="countries[]"]:checked');
|
const selectedCountries = document.querySelectorAll('input[name="countries[]"]:checked');
|
||||||
const selectedAction = document.querySelector('input[name="action"]:checked')?.value || 'deny';
|
const selectedAction = document.querySelector('input[name="policy_action"]:checked')?.value || 'deny';
|
||||||
const actionText = {
|
const actionText = {
|
||||||
'deny': '🚫 Block',
|
'deny': '🚫 Block',
|
||||||
'challenge': '🛡️ Challenge',
|
'challenge': '🛡️ Challenge',
|
||||||
|
|||||||
@@ -32,16 +32,16 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="bg-white px-4 py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
|
<div class="bg-white px-4 py-5 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
|
||||||
<dt class="text-sm font-medium text-gray-500">Action</dt>
|
<dt class="text-sm font-medium text-gray-500">Policy Action</dt>
|
||||||
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
<dd class="mt-1 text-sm text-gray-900 sm:mt-0 sm:col-span-2">
|
||||||
<span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium
|
<span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium
|
||||||
<%= case @waf_policy.action
|
<%= case @waf_policy.policy_action
|
||||||
when 'deny' then 'bg-red-100 text-red-800'
|
when 'deny' then 'bg-red-100 text-red-800'
|
||||||
when 'allow' then 'bg-green-100 text-green-800'
|
when 'allow' then 'bg-green-100 text-green-800'
|
||||||
when 'redirect' then 'bg-yellow-100 text-yellow-800'
|
when 'redirect' then 'bg-yellow-100 text-yellow-800'
|
||||||
when 'challenge' then 'bg-purple-100 text-purple-800'
|
when 'challenge' then 'bg-purple-100 text-purple-800'
|
||||||
end %>">
|
end %>">
|
||||||
<%= @waf_policy.action.upcase %>
|
<%= @waf_policy.policy_action.upcase %>
|
||||||
</span>
|
</span>
|
||||||
</dd>
|
</dd>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
class RenameActionToPolicyActionOnWafPolicies < ActiveRecord::Migration[8.1]
|
||||||
|
def change
|
||||||
|
rename_column :waf_policies, :action, :policy_action
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
class AddNetworkIntelligenceToEvents < ActiveRecord::Migration[8.1]
|
||||||
|
def change
|
||||||
|
# Add network intelligence columns for denormalization
|
||||||
|
add_column :events, :country, :string
|
||||||
|
add_column :events, :company, :string
|
||||||
|
add_column :events, :asn, :integer
|
||||||
|
add_column :events, :asn_org, :string
|
||||||
|
add_column :events, :is_datacenter, :boolean, default: false, null: false
|
||||||
|
add_column :events, :is_vpn, :boolean, default: false, null: false
|
||||||
|
add_column :events, :is_proxy, :boolean, default: false, null: false
|
||||||
|
add_column :events, :network_range_id, :bigint
|
||||||
|
|
||||||
|
# Add indexes for commonly queried fields
|
||||||
|
add_index :events, :country
|
||||||
|
add_index :events, :company
|
||||||
|
add_index :events, :asn
|
||||||
|
add_index :events, :network_range_id
|
||||||
|
add_index :events, [:is_datacenter, :is_vpn, :is_proxy], name: 'index_events_on_network_flags'
|
||||||
|
|
||||||
|
# Backfill skipped - run manually after migration
|
||||||
|
# See script/backfill_network_intelligence.rb or lib/tasks/events.rake
|
||||||
|
end
|
||||||
|
end
|
||||||
36
lib/tasks/events.rake
Normal file
36
lib/tasks/events.rake
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
namespace :events do
|
||||||
|
desc "Backfill network intelligence data for events"
|
||||||
|
task backfill_network_intelligence: :environment do
|
||||||
|
batch_size = ENV['BATCH_SIZE']&.to_i || 10_000
|
||||||
|
Event.backfill_network_intelligence!(batch_size: batch_size)
|
||||||
|
end
|
||||||
|
|
||||||
|
desc "Show backfill progress"
|
||||||
|
task backfill_progress: :environment do
|
||||||
|
total = Event.count
|
||||||
|
with_country = Event.where.not(country: nil).count
|
||||||
|
without_country = Event.where(country: nil).count
|
||||||
|
percent = (with_country.to_f / total * 100).round(1)
|
||||||
|
|
||||||
|
puts "=" * 60
|
||||||
|
puts "Network Intelligence Backfill Progress"
|
||||||
|
puts "=" * 60
|
||||||
|
puts "Total events: #{total}"
|
||||||
|
puts "With network data: #{with_country} (#{percent}%)"
|
||||||
|
puts "Missing network data: #{without_country}"
|
||||||
|
puts "=" * 60
|
||||||
|
|
||||||
|
if without_country > 0
|
||||||
|
puts
|
||||||
|
puts "To continue backfill:"
|
||||||
|
puts " rails events:backfill_network_intelligence"
|
||||||
|
puts
|
||||||
|
puts "Or with custom batch size:"
|
||||||
|
puts " BATCH_SIZE=5000 rails events:backfill_network_intelligence"
|
||||||
|
else
|
||||||
|
puts "✓ Backfill complete!"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
Reference in New Issue
Block a user