index.twig 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
  1. {% extends 'layout.twig' %}
  2. {% block title %}Settings — IRDB{% endblock %}
  3. {% block content %}
  4. {% import 'partials/sort.twig' as sort %}
  5. <div class="mx-auto max-w-5xl space-y-6">
  6. <div class="flex items-center justify-between">
  7. <h1 class="text-2xl font-semibold tracking-tight">Settings</h1>
  8. <span class="text-xs text-slate-500 dark:text-slate-400">Admin only · read-only · masked secrets</span>
  9. </div>
  10. {% if error %}
  11. <div class="rounded-md border border-red-300 bg-red-50 px-4 py-2 text-sm text-red-800 dark:border-red-800 dark:bg-red-950 dark:text-red-300">{{ error }}</div>
  12. {% endif %}
  13. {# ------------------------- Configuration ------------------------- #}
  14. {% if config and config.sections %}
  15. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  16. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Configuration</h2>
  17. <p class="mt-1 text-xs text-slate-500 dark:text-slate-400">Effective values from the api's environment. Secrets are masked (<code>***</code>) or previewed (first 8 chars + …).</p>
  18. <div class="mt-4 grid gap-5 md:grid-cols-2">
  19. {% for section_name, items in config.sections %}
  20. <div class="rounded-lg border border-slate-100 dark:border-slate-800">
  21. <div class="border-b border-slate-100 bg-slate-50 px-4 py-2 text-xs font-semibold uppercase tracking-wider text-slate-500 dark:border-slate-800 dark:bg-slate-950 dark:text-slate-400">{{ section_name }}</div>
  22. <dl class="divide-y divide-slate-100 dark:divide-slate-800 text-sm">
  23. {% for key, value in items %}
  24. <div class="grid grid-cols-2 gap-2 px-4 py-2">
  25. <dt class="font-mono text-xs text-slate-500 dark:text-slate-400">{{ key }}</dt>
  26. <dd class="break-all font-mono text-xs text-slate-700 dark:text-slate-200">
  27. {%- if value is null -%}<span class="text-slate-400">—</span>
  28. {%- elseif value is same as(true) -%}true
  29. {%- elseif value is same as(false) -%}false
  30. {%- else -%}{{ value }}{%- endif -%}
  31. </dd>
  32. </div>
  33. {% endfor %}
  34. </dl>
  35. </div>
  36. {% endfor %}
  37. </div>
  38. </section>
  39. {% endif %}
  40. {# ------------------------- Audit toggles ------------------------- #}
  41. {% if app_settings is not null %}
  42. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  43. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Audit toggles</h2>
  44. <p class="mt-1 text-xs text-slate-500 dark:text-slate-400">High-volume public endpoints can be excluded from the audit log to keep the table compact. Each switch is independent; changes take effect immediately.</p>
  45. <form method="post" action="/app/settings/audit-toggles" class="mt-4 space-y-3">
  46. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  47. <label class="flex items-start gap-3 rounded-lg border border-slate-100 px-3 py-2 dark:border-slate-800">
  48. <input type="checkbox" name="audit_report_received_enabled" value="1"
  49. class="mt-0.5 h-4 w-4 rounded border-slate-300 text-emerald-600 focus:ring-emerald-500"
  50. {% if app_settings.audit_report_received_enabled %}checked{% endif %}>
  51. <span class="text-sm">
  52. <span class="font-medium text-slate-700 dark:text-slate-200">Log when a reporter submits an IP</span>
  53. <span class="block text-xs text-slate-500 dark:text-slate-400">Each <code>POST /api/v1/report</code> writes a <code>report.received</code> entry.</span>
  54. </span>
  55. </label>
  56. <label class="flex items-start gap-3 rounded-lg border border-slate-100 px-3 py-2 dark:border-slate-800">
  57. <input type="checkbox" name="audit_blocklist_request_enabled" value="1"
  58. class="mt-0.5 h-4 w-4 rounded border-slate-300 text-emerald-600 focus:ring-emerald-500"
  59. {% if app_settings.audit_blocklist_request_enabled %}checked{% endif %}>
  60. <span class="text-sm">
  61. <span class="font-medium text-slate-700 dark:text-slate-200">Log when a consumer requests the ban list</span>
  62. <span class="block text-xs text-slate-500 dark:text-slate-400">Each <code>GET /api/v1/blocklist</code> writes a <code>blocklist.requested</code> entry (including 304s).</span>
  63. </span>
  64. </label>
  65. <div class="flex justify-end">
  66. <button type="submit"
  67. class="rounded-md bg-slate-700 px-3 py-1.5 text-xs font-medium text-white hover:bg-slate-600 dark:bg-slate-200 dark:text-slate-900 dark:hover:bg-white">
  68. Save
  69. </button>
  70. </div>
  71. </form>
  72. </section>
  73. {% endif %}
  74. {# ------------------------------ Jobs ----------------------------- #}
  75. {% if jobs and jobs.jobs %}
  76. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  77. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Jobs</h2>
  78. <p class="mt-1 text-xs text-slate-500 dark:text-slate-400">Latest run, lock state, and manual-trigger buttons. Manual triggers run synchronously — wait for the response.</p>
  79. <div class="mt-4 overflow-hidden rounded-lg border border-slate-100 dark:border-slate-800">
  80. <table class="w-full text-sm" data-sortable-table="settings-jobs">
  81. <thead class="border-b border-slate-100 bg-slate-50 text-left text-xs uppercase tracking-wider text-slate-500 dark:border-slate-800 dark:bg-slate-950 dark:text-slate-400">
  82. <tr>
  83. {{ sort.th('Name', 'name') }}
  84. {{ sort.th('Last status', 'status') }}
  85. {{ sort.th('Last finished', 'last_finished', 'date') }}
  86. {{ sort.th('Items', 'items', 'number') }}
  87. <th class="px-4 py-2 text-right font-medium">Trigger</th>
  88. </tr>
  89. </thead>
  90. <tbody class="divide-y divide-slate-100 dark:divide-slate-800">
  91. {% for name, info in jobs.jobs %}
  92. <tr>
  93. <td class="px-4 py-2 align-top font-mono text-xs" data-sort-value="{{ name }}">
  94. {{ name }}
  95. {% if info.overdue %}
  96. <span class="ml-1 rounded bg-red-100 px-1.5 py-0.5 text-[0.65rem] font-mono uppercase text-red-800 dark:bg-red-950 dark:text-red-300">overdue</span>
  97. {% endif %}
  98. </td>
  99. <td class="px-4 py-2 align-top" data-sort-value="{{ info.last_run.status|default('') }}">
  100. {% if info.last_run %}
  101. {% set s = info.last_run.status %}
  102. {% set classes = {
  103. 'success': 'bg-emerald-100 text-emerald-900 dark:bg-emerald-900 dark:text-emerald-100',
  104. 'failure': 'bg-red-100 text-red-900 dark:bg-red-900 dark:text-red-100',
  105. 'skipped_locked': 'bg-amber-100 text-amber-900 dark:bg-amber-900 dark:text-amber-100',
  106. 'running': 'bg-blue-100 text-blue-900 dark:bg-blue-900 dark:text-blue-100',
  107. } %}
  108. <span class="rounded px-2 py-0.5 font-mono text-[0.65rem] uppercase {{ classes[s]|default('bg-slate-100 text-slate-700 dark:bg-slate-800 dark:text-slate-300') }}">{{ s }}</span>
  109. {% else %}
  110. <span class="text-xs text-slate-400">never run</span>
  111. {% endif %}
  112. </td>
  113. <td class="px-4 py-2 align-top font-mono text-xs text-slate-500" data-sort-value="{{ info.last_run.finished_at|default('') }}">
  114. {% if info.last_run.finished_at %}<time class="irdb-dt" datetime="{{ info.last_run.finished_at }}">{{ info.last_run.finished_at }}</time>{% else %}—{% endif %}
  115. </td>
  116. <td class="px-4 py-2 align-top font-mono text-xs text-slate-500" data-sort-value="{{ info.last_run.items_processed|default('') }}">
  117. {{ info.last_run.items_processed|default('—') }}
  118. </td>
  119. <td class="px-4 py-2 align-top text-right">
  120. {% if name != 'tick' %}
  121. <form method="post" action="/app/settings/jobs/trigger/{{ name }}" class="inline" x-data="{ submitting: false }" x-on:submit="submitting = true">
  122. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  123. <button type="submit" x-bind:disabled="submitting"
  124. class="rounded-md border border-slate-300 px-2 py-1 text-xs hover:bg-slate-50 disabled:opacity-50 dark:border-slate-700 dark:hover:bg-slate-800">
  125. <span x-show="!submitting">Run now</span>
  126. <span x-show="submitting" x-cloak>Running…</span>
  127. </button>
  128. </form>
  129. {% else %}
  130. <span class="text-xs text-slate-400">scheduled</span>
  131. {% endif %}
  132. </td>
  133. </tr>
  134. {% endfor %}
  135. </tbody>
  136. </table>
  137. </div>
  138. </section>
  139. {% endif %}
  140. {# -------------------- Demo & maintenance -------------------- #}
  141. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  142. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Demo &amp; maintenance</h2>
  143. <p class="mt-1 text-xs text-slate-500 dark:text-slate-400">Populate the database with sample data for screenshots and demos, or wipe operational data to start clean. Both actions are admin-only and audited.</p>
  144. <div class="mt-4 grid gap-4 md:grid-cols-2">
  145. <div class="rounded-lg border border-emerald-200 bg-emerald-50 p-4 dark:border-emerald-900 dark:bg-emerald-950/40"
  146. x-data="{ open: false, submitting: false }">
  147. <h3 class="text-sm font-semibold text-emerald-800 dark:text-emerald-200">Load demo data</h3>
  148. <p class="mt-1 text-xs text-emerald-900/80 dark:text-emerald-200/80">
  149. Inserts demo reporters, consumers, IPs, reports, manual blocks, allowlist entries, and synthetic GeoIP — then triggers a full score recompute. Returns "already seeded" if demo data is present.
  150. </p>
  151. <div class="mt-3">
  152. <button type="button" x-on:click="open = true"
  153. class="rounded-md bg-emerald-600 px-3 py-1.5 text-xs font-medium text-white hover:bg-emerald-500">
  154. Load demo data…
  155. </button>
  156. </div>
  157. <div x-show="open" x-cloak
  158. class="fixed inset-0 z-50 flex items-center justify-center bg-slate-900/70 px-4">
  159. <div class="w-full max-w-md rounded-2xl border border-emerald-300 bg-white p-6 shadow-2xl dark:border-emerald-700 dark:bg-slate-900">
  160. <h3 class="text-lg font-semibold text-emerald-700 dark:text-emerald-300">Load demo data?</h3>
  161. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  162. This will add sample reporters, consumers, IPs, and reports to the database, then run a full recompute. Existing real data is left untouched.
  163. </p>
  164. <form method="post" action="/app/settings/maintenance/seed-demo" class="mt-4 flex justify-end gap-2"
  165. x-on:submit="submitting = true">
  166. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  167. <button type="button" x-on:click="open = false"
  168. class="rounded-md border border-slate-300 px-3 py-1.5 text-sm hover:bg-slate-50 dark:border-slate-700 dark:hover:bg-slate-800">Cancel</button>
  169. <button type="submit" x-bind:disabled="submitting"
  170. class="rounded-md bg-emerald-600 px-3 py-1.5 text-sm font-medium text-white hover:bg-emerald-500 disabled:opacity-50">
  171. <span x-show="!submitting">Load demo data</span>
  172. <span x-show="submitting" x-cloak>Loading…</span>
  173. </button>
  174. </form>
  175. </div>
  176. </div>
  177. </div>
  178. <div class="rounded-lg border border-red-200 bg-red-50 p-4 dark:border-red-900 dark:bg-red-950/40"
  179. x-data="{ open: false, confirm: '', submitting: false }">
  180. <h3 class="text-sm font-semibold text-red-800 dark:text-red-200">Purge operational data</h3>
  181. <p class="mt-1 text-xs text-red-900/80 dark:text-red-200/80">
  182. Deletes all reports, scores, manual blocks, allowlist, audit log, reporters, consumers, and non-service tokens. Users, OIDC mappings, and categories are preserved.
  183. </p>
  184. <div class="mt-3">
  185. <button type="button" x-on:click="open = true"
  186. class="rounded-md border border-red-400 bg-white px-3 py-1.5 text-xs font-medium text-red-700 hover:bg-red-50 dark:border-red-700 dark:bg-slate-900 dark:text-red-300 dark:hover:bg-slate-800">
  187. Purge data…
  188. </button>
  189. </div>
  190. <div x-show="open" x-cloak
  191. class="fixed inset-0 z-50 flex items-center justify-center bg-slate-900/70 px-4">
  192. <div class="w-full max-w-md rounded-2xl border border-red-400 bg-white p-6 shadow-2xl dark:border-red-700 dark:bg-slate-900">
  193. <h3 class="text-lg font-semibold text-red-700 dark:text-red-300">Purge operational data?</h3>
  194. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  195. This will <strong class="text-red-700 dark:text-red-300">permanently delete</strong> reports, scores, blocks, allowlist, audit log, reporters, consumers, and tokens. The service token, your user account, OIDC mappings, and abuse categories are preserved.
  196. </p>
  197. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  198. Type <code class="rounded bg-slate-100 px-1 py-0.5 font-mono text-xs text-red-700 dark:bg-slate-800 dark:text-red-300">PURGE</code> to confirm:
  199. </p>
  200. <form method="post" action="/app/settings/maintenance/purge" class="mt-3"
  201. x-on:submit="submitting = true">
  202. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  203. <input type="text" name="confirm" autocomplete="off" x-model="confirm"
  204. class="w-full rounded-md border border-slate-300 bg-white px-3 py-2 font-mono text-sm dark:border-slate-700 dark:bg-slate-950"
  205. placeholder="PURGE">
  206. <div class="mt-4 flex justify-end gap-2">
  207. <button type="button" x-on:click="open = false; confirm = ''"
  208. class="rounded-md border border-slate-300 px-3 py-1.5 text-sm hover:bg-slate-50 dark:border-slate-700 dark:hover:bg-slate-800">Cancel</button>
  209. <button type="submit" x-bind:disabled="confirm !== 'PURGE' || submitting"
  210. class="rounded-md bg-red-600 px-3 py-1.5 text-sm font-medium text-white hover:bg-red-500 disabled:opacity-40">
  211. <span x-show="!submitting">Purge data</span>
  212. <span x-show="submitting" x-cloak>Purging…</span>
  213. </button>
  214. </div>
  215. </form>
  216. </div>
  217. </div>
  218. </div>
  219. </div>
  220. </section>
  221. {# ------------------------------ GeoIP ----------------------------- #}
  222. {% if config and config.sections.geoip %}
  223. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  224. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">GeoIP</h2>
  225. <p class="mt-1 text-xs text-slate-500 dark:text-slate-400">Provider, on-disk paths, and credential state. DB freshness comes from healthz; the trigger button on <code>refresh-geoip</code> is in the Jobs section above.</p>
  226. <dl class="mt-3 grid grid-cols-3 gap-2 text-sm">
  227. <dt class="text-slate-500 dark:text-slate-400">Provider</dt>
  228. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.GEOIP_PROVIDER|default('—') }}</dd>
  229. <dt class="text-slate-500 dark:text-slate-400">Country DB</dt>
  230. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.GEOIP_COUNTRY_DB|default('—') }}</dd>
  231. <dt class="text-slate-500 dark:text-slate-400">ASN DB</dt>
  232. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.GEOIP_ASN_DB|default('—') }}</dd>
  233. <dt class="text-slate-500 dark:text-slate-400">MaxMind key</dt>
  234. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.MAXMIND_LICENSE_KEY ? config.sections.geoip.MAXMIND_LICENSE_KEY : '(unset)' }}</dd>
  235. <dt class="text-slate-500 dark:text-slate-400">IPinfo token</dt>
  236. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.IPINFO_TOKEN ? config.sections.geoip.IPINFO_TOKEN : '(unset)' }}</dd>
  237. </dl>
  238. </section>
  239. {% endif %}
  240. </div>
  241. {% endblock %}