edit.twig 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. {% extends 'layout.twig' %}
  2. {% block title %}{{ policy.name }} — Policy — IRDB{% endblock %}
  3. {% block content %}
  4. {# Twig's |merge calls PHP array_merge, which renumbers integer keys; key by
  5. slug (a string) so {scanners:40, indexer:30,…} round-trips faithfully. #}
  6. {% set thresholds_by_slug = {} %}
  7. {% for t in policy.thresholds|default([]) %}
  8. {% if t.category_slug %}
  9. {% set thresholds_by_slug = thresholds_by_slug|merge({(t.category_slug): t.threshold}) %}
  10. {% endif %}
  11. {% endfor %}
  12. <div class="mx-auto max-w-5xl">
  13. <a href="/app/policies" class="text-sm text-slate-500 hover:underline dark:text-slate-400">← Back to policies</a>
  14. <div class="mt-3 flex items-center justify-between">
  15. <h1 class="text-2xl font-semibold tracking-tight">
  16. <span class="font-mono">{{ policy.name }}</span>
  17. </h1>
  18. {% if can_write %}
  19. {% include 'partials/confirm_form.twig' with {
  20. action: '/app/policies/' ~ policy.id ~ '/delete',
  21. label: 'Delete policy',
  22. description: 'Refused if any consumer references this policy.',
  23. } only %}
  24. {% endif %}
  25. </div>
  26. <form method="post" action="/app/policies/{{ policy.id }}" class="mt-6">
  27. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  28. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  29. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Metadata</h2>
  30. <div class="mt-3 grid grid-cols-1 gap-3 md:grid-cols-3 text-sm">
  31. <div>
  32. <label for="p-name" class="block text-xs font-medium text-slate-600 dark:text-slate-400">Name</label>
  33. <input type="text" id="p-name" name="name" value="{{ policy.name }}" {% if not can_write %}readonly{% endif %}
  34. class="mt-1 w-full rounded-md border border-slate-300 bg-white px-2 py-1.5 font-mono dark:border-slate-700 dark:bg-slate-950">
  35. </div>
  36. <div class="md:col-span-2">
  37. <label for="p-desc" class="block text-xs font-medium text-slate-600 dark:text-slate-400">Description</label>
  38. <input type="text" id="p-desc" name="description" value="{{ policy.description|default('') }}" {% if not can_write %}readonly{% endif %}
  39. class="mt-1 w-full rounded-md border border-slate-300 bg-white px-2 py-1.5 dark:border-slate-700 dark:bg-slate-950">
  40. </div>
  41. <div class="md:col-span-3">
  42. <label class="flex items-center gap-2 text-xs text-slate-600 dark:text-slate-400">
  43. <input type="checkbox" name="include_manual_blocks" value="1"
  44. {% if policy.include_manual_blocks %}checked{% endif %}
  45. {% if not can_write %}disabled{% endif %}>
  46. include manual blocks
  47. </label>
  48. </div>
  49. </div>
  50. </section>
  51. <section class="mt-6 rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  52. <div class="flex items-center justify-between">
  53. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Threshold matrix</h2>
  54. <span class="text-xs text-slate-400">Empty value ⇒ category not in policy</span>
  55. </div>
  56. <table class="mt-3 w-full text-sm">
  57. <thead class="text-left text-xs uppercase tracking-wider text-slate-400">
  58. <tr>
  59. <th class="pb-2 font-medium">Category</th>
  60. <th class="pb-2 font-medium">Decay</th>
  61. <th class="pb-2 text-right font-medium">Threshold</th>
  62. </tr>
  63. </thead>
  64. <tbody class="divide-y divide-slate-100 dark:divide-slate-800">
  65. {% for c in categories %}
  66. <tr>
  67. <td class="py-2"><span class="font-mono">{{ c.slug }}</span> <span class="text-slate-400">— {{ c.name }}</span></td>
  68. <td class="py-2 text-xs text-slate-500 dark:text-slate-400">{{ c.decay_function }} ({{ c.decay_param }})</td>
  69. <td class="py-2 text-right">
  70. <input type="number" step="0.01" min="0"
  71. name="thresholds[{{ c.slug }}]"
  72. value="{{ thresholds_by_slug[c.slug]|default('') }}"
  73. {% if not can_write %}readonly{% endif %}
  74. class="w-32 rounded-md border border-slate-300 bg-white px-2 py-1 text-right font-mono dark:border-slate-700 dark:bg-slate-950">
  75. </td>
  76. </tr>
  77. {% endfor %}
  78. </tbody>
  79. </table>
  80. </section>
  81. {% if can_write %}
  82. <div class="mt-6 flex justify-end">
  83. <button type="submit" class="rounded-md bg-indigo-600 px-4 py-2 text-sm font-medium text-white hover:bg-indigo-500">Save policy</button>
  84. </div>
  85. {% endif %}
  86. </form>
  87. <section class="mt-6 rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900"
  88. x-data="policyScoreDistribution" data-policy-id="{{ policy.id }}">
  89. <div class="flex items-center justify-between">
  90. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Score distribution</h2>
  91. <button type="button" x-on:click="load()" class="text-xs text-indigo-600 hover:underline dark:text-indigo-400">Refresh</button>
  92. </div>
  93. <p class="mt-2 text-xs text-slate-400">
  94. One line per thresholded category, IPs grouped by score in steps of 5; the shaded area to the right of each threshold marks scores high enough to land on this policy's blocklist.
  95. </p>
  96. <div class="mt-3 h-64">
  97. <canvas x-ref="canvas"></canvas>
  98. </div>
  99. <p class="mt-2 text-xs text-slate-400" x-show="empty">No scored IPs in the database yet.</p>
  100. </section>
  101. <section class="mt-6 rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900"
  102. x-data="policyPreview" data-policy-id="{{ policy.id }}">
  103. <div class="flex items-center justify-between">
  104. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Preview</h2>
  105. <button type="button" x-on:click="load()" class="text-xs text-indigo-600 hover:underline dark:text-indigo-400">Refresh</button>
  106. </div>
  107. <p class="mt-2 text-sm">
  108. <span x-show="loading">Loading…</span>
  109. <template x-if="notLoading">
  110. <span><span class="font-mono" x-text="count"></span> entries</span>
  111. </template>
  112. </p>
  113. <ul class="mt-3 max-h-60 divide-y divide-slate-100 overflow-y-auto text-xs dark:divide-slate-800">
  114. <template x-for="entry in sample" :key="entry.key">
  115. <li class="flex items-baseline justify-between gap-3 py-1">
  116. <span class="font-mono text-slate-700 dark:text-slate-300" x-text="entry.label"></span>
  117. <span class="shrink-0 text-slate-500 dark:text-slate-400" :title="entry.tooltip" x-text="entry.expiry"></span>
  118. </li>
  119. </template>
  120. </ul>
  121. <p class="mt-2 text-xs text-slate-400">
  122. Sample = first 50 entries from the rendered blocklist. Expiry for scored
  123. entries is an estimate assuming no further reports; manual entries show
  124. the configured expiry.
  125. </p>
  126. </section>
  127. </div>
  128. {% endblock %}