index.twig 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  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 project_version %}
  11. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  12. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">About</h2>
  13. <p class="mt-1 text-xs text-slate-500 dark:text-slate-400">Project release of the bundled stack. Per-container versions live in <code>api/CHANGELOG.md</code> and <code>ui/CHANGELOG.md</code>.</p>
  14. <dl class="mt-3 grid grid-cols-3 gap-2 text-sm">
  15. <dt class="text-slate-500 dark:text-slate-400">IRDB version</dt>
  16. <dd class="col-span-2">
  17. <span class="rounded bg-emerald-100 px-2 py-0.5 font-mono text-xs text-emerald-900 dark:bg-emerald-900 dark:text-emerald-100">v{{ project_version }}</span>
  18. </dd>
  19. </dl>
  20. </section>
  21. {% endif %}
  22. {% if error %}
  23. <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>
  24. {% endif %}
  25. {# ------------------------- Configuration ------------------------- #}
  26. {% if config and config.sections %}
  27. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  28. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Configuration</h2>
  29. <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>
  30. <div class="mt-4 grid gap-5 md:grid-cols-2">
  31. {% for section_name, items in config.sections %}
  32. <div class="rounded-lg border border-slate-100 dark:border-slate-800">
  33. <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>
  34. <dl class="divide-y divide-slate-100 dark:divide-slate-800 text-sm">
  35. {% for key, value in items %}
  36. <div class="grid grid-cols-2 gap-2 px-4 py-2">
  37. <dt class="font-mono text-xs text-slate-500 dark:text-slate-400">{{ key }}</dt>
  38. <dd class="break-all font-mono text-xs text-slate-700 dark:text-slate-200">
  39. {%- if value is null -%}<span class="text-slate-400">—</span>
  40. {%- elseif value is same as(true) -%}true
  41. {%- elseif value is same as(false) -%}false
  42. {%- else -%}{{ value }}{%- endif -%}
  43. </dd>
  44. </div>
  45. {% endfor %}
  46. </dl>
  47. </div>
  48. {% endfor %}
  49. </div>
  50. </section>
  51. {% endif %}
  52. {# ------------------------- Audit toggles ------------------------- #}
  53. {% if app_settings is not null %}
  54. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  55. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Audit toggles</h2>
  56. <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>
  57. <form method="post" action="/app/settings/audit-toggles" class="mt-4 space-y-3">
  58. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  59. <label class="flex items-start gap-3 rounded-lg border border-slate-100 px-3 py-2 dark:border-slate-800">
  60. <input type="checkbox" name="audit_report_received_enabled" value="1"
  61. class="mt-0.5 h-4 w-4 rounded border-slate-300 text-emerald-600 focus:ring-emerald-500"
  62. {% if app_settings.audit_report_received_enabled %}checked{% endif %}>
  63. <span class="text-sm">
  64. <span class="font-medium text-slate-700 dark:text-slate-200">Log when a reporter submits an IP</span>
  65. <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>
  66. </span>
  67. </label>
  68. <label class="flex items-start gap-3 rounded-lg border border-slate-100 px-3 py-2 dark:border-slate-800">
  69. <input type="checkbox" name="audit_blocklist_request_enabled" value="1"
  70. class="mt-0.5 h-4 w-4 rounded border-slate-300 text-emerald-600 focus:ring-emerald-500"
  71. {% if app_settings.audit_blocklist_request_enabled %}checked{% endif %}>
  72. <span class="text-sm">
  73. <span class="font-medium text-slate-700 dark:text-slate-200">Log when a consumer requests the ban list</span>
  74. <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>
  75. </span>
  76. </label>
  77. <div class="flex justify-end">
  78. <button type="submit"
  79. 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">
  80. Save
  81. </button>
  82. </div>
  83. </form>
  84. </section>
  85. {% endif %}
  86. {# ------------------------------ Jobs ----------------------------- #}
  87. {% if jobs and jobs.jobs %}
  88. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  89. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Jobs</h2>
  90. <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>
  91. <div class="mt-4 overflow-hidden rounded-lg border border-slate-100 dark:border-slate-800">
  92. <table class="w-full text-sm" data-sortable-table="settings-jobs">
  93. <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">
  94. <tr>
  95. {{ sort.th('Name', 'name') }}
  96. {{ sort.th('Last status', 'status') }}
  97. {{ sort.th('Last finished', 'last_finished', 'date') }}
  98. {{ sort.th('Items', 'items', 'number') }}
  99. <th class="px-4 py-2 text-right font-medium">Trigger</th>
  100. </tr>
  101. </thead>
  102. <tbody class="divide-y divide-slate-100 dark:divide-slate-800">
  103. {% for name, info in jobs.jobs %}
  104. <tr>
  105. <td class="px-4 py-2 align-top font-mono text-xs" data-sort-value="{{ name }}">
  106. {{ name }}
  107. {% if info.overdue %}
  108. <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>
  109. {% endif %}
  110. </td>
  111. <td class="px-4 py-2 align-top" data-sort-value="{{ info.last_run.status|default('') }}">
  112. {% if info.last_run %}
  113. {% set s = info.last_run.status %}
  114. {% set classes = {
  115. 'success': 'bg-emerald-100 text-emerald-900 dark:bg-emerald-900 dark:text-emerald-100',
  116. 'failure': 'bg-red-100 text-red-900 dark:bg-red-900 dark:text-red-100',
  117. 'skipped_locked': 'bg-amber-100 text-amber-900 dark:bg-amber-900 dark:text-amber-100',
  118. 'running': 'bg-blue-100 text-blue-900 dark:bg-blue-900 dark:text-blue-100',
  119. } %}
  120. <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>
  121. {% else %}
  122. <span class="text-xs text-slate-400">never run</span>
  123. {% endif %}
  124. </td>
  125. <td class="px-4 py-2 align-top font-mono text-xs text-slate-500" data-sort-value="{{ info.last_run.finished_at|default('') }}">
  126. {% if info.last_run.finished_at %}<time class="irdb-dt" datetime="{{ info.last_run.finished_at }}">{{ info.last_run.finished_at }}</time>{% else %}—{% endif %}
  127. </td>
  128. <td class="px-4 py-2 align-top font-mono text-xs text-slate-500" data-sort-value="{{ info.last_run.items_processed|default('') }}">
  129. {{ info.last_run.items_processed|default('—') }}
  130. </td>
  131. <td class="px-4 py-2 align-top text-right">
  132. {% if name != 'tick' %}
  133. <form method="post" action="/app/settings/jobs/trigger/{{ name }}" class="inline" x-data="submitGuard" x-on:submit="onSubmit()">
  134. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  135. <button type="submit" x-bind:disabled="submitting"
  136. 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">
  137. <span x-show="notSubmitting">Run now</span>
  138. <span x-show="submitting" x-cloak>Running…</span>
  139. </button>
  140. </form>
  141. {% else %}
  142. <span class="text-xs text-slate-400">scheduled</span>
  143. {% endif %}
  144. </td>
  145. </tr>
  146. {% endfor %}
  147. </tbody>
  148. </table>
  149. </div>
  150. </section>
  151. {% endif %}
  152. {# -------------------- Demo & maintenance -------------------- #}
  153. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  154. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">Demo &amp; maintenance</h2>
  155. <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>
  156. <div class="mt-4 grid gap-4 md:grid-cols-2">
  157. <div class="rounded-lg border border-emerald-200 bg-emerald-50 p-4 dark:border-emerald-900 dark:bg-emerald-950/40"
  158. x-data="dangerousAction" data-expected-confirm="SEED">
  159. <h3 class="text-sm font-semibold text-emerald-800 dark:text-emerald-200">Load demo data</h3>
  160. <p class="mt-1 text-xs text-emerald-900/80 dark:text-emerald-200/80">
  161. 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.
  162. </p>
  163. <div class="mt-3">
  164. <button type="button" x-on:click="show()"
  165. class="rounded-md bg-emerald-600 px-3 py-1.5 text-xs font-medium text-white hover:bg-emerald-500">
  166. Load demo data…
  167. </button>
  168. </div>
  169. <div x-show="open" x-cloak
  170. class="fixed inset-0 z-50 flex items-center justify-center bg-slate-900/70 px-4">
  171. <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">
  172. <h3 class="text-lg font-semibold text-emerald-700 dark:text-emerald-300">Load demo data?</h3>
  173. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  174. This will add sample reporters, consumers, IPs, and reports to the database, then run a full recompute. Existing real data is left untouched.
  175. </p>
  176. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  177. Type <code class="rounded bg-slate-100 px-1 py-0.5 font-mono text-xs text-emerald-700 dark:bg-slate-800 dark:text-emerald-300">SEED</code> to confirm:
  178. </p>
  179. <form method="post" action="/app/settings/maintenance/seed-demo" class="mt-3"
  180. x-on:submit="onSubmit()">
  181. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  182. <input type="text" name="confirm" autocomplete="off" x-model="confirm"
  183. 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"
  184. placeholder="SEED">
  185. <div class="mt-4 flex justify-end gap-2">
  186. <button type="button" x-on:click="hide()"
  187. 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>
  188. <button type="submit" x-bind:disabled="blocked"
  189. class="rounded-md bg-emerald-600 px-3 py-1.5 text-sm font-medium text-white hover:bg-emerald-500 disabled:opacity-40">
  190. <span x-show="notSubmitting">Load demo data</span>
  191. <span x-show="submitting" x-cloak>Loading…</span>
  192. </button>
  193. </div>
  194. </form>
  195. </div>
  196. </div>
  197. </div>
  198. <div class="rounded-lg border border-red-200 bg-red-50 p-4 dark:border-red-900 dark:bg-red-950/40"
  199. x-data="dangerousAction" data-expected-confirm="PURGE">
  200. <h3 class="text-sm font-semibold text-red-800 dark:text-red-200">Purge operational data</h3>
  201. <p class="mt-1 text-xs text-red-900/80 dark:text-red-200/80">
  202. Deletes all reports, scores, manual blocks, allowlist, audit log, reporters, consumers, and non-service tokens. Users, OIDC mappings, and categories are preserved.
  203. </p>
  204. <div class="mt-3">
  205. <button type="button" x-on:click="show()"
  206. 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">
  207. Purge data…
  208. </button>
  209. </div>
  210. <div x-show="open" x-cloak
  211. class="fixed inset-0 z-50 flex items-center justify-center bg-slate-900/70 px-4">
  212. <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">
  213. <h3 class="text-lg font-semibold text-red-700 dark:text-red-300">Purge operational data?</h3>
  214. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  215. 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.
  216. </p>
  217. <p class="mt-2 text-sm text-slate-600 dark:text-slate-400">
  218. 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:
  219. </p>
  220. <form method="post" action="/app/settings/maintenance/purge" class="mt-3"
  221. x-on:submit="onSubmit()">
  222. <input type="hidden" name="csrf_token" value="{{ csrf_token }}">
  223. <input type="text" name="confirm" autocomplete="off" x-model="confirm"
  224. 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"
  225. placeholder="PURGE">
  226. <div class="mt-4 flex justify-end gap-2">
  227. <button type="button" x-on:click="hide()"
  228. 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>
  229. <button type="submit" x-bind:disabled="blocked"
  230. class="rounded-md bg-red-600 px-3 py-1.5 text-sm font-medium text-white hover:bg-red-500 disabled:opacity-40">
  231. <span x-show="notSubmitting">Purge data</span>
  232. <span x-show="submitting" x-cloak>Purging…</span>
  233. </button>
  234. </div>
  235. </form>
  236. </div>
  237. </div>
  238. </div>
  239. </div>
  240. </section>
  241. {# ------------------------------ GeoIP ----------------------------- #}
  242. {% if config and config.sections.geoip %}
  243. <section class="rounded-2xl border border-slate-200 bg-white p-5 shadow-sm dark:border-slate-800 dark:bg-slate-900">
  244. <h2 class="text-sm font-semibold uppercase tracking-wider text-slate-500 dark:text-slate-400">GeoIP</h2>
  245. <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>
  246. <dl class="mt-3 grid grid-cols-3 gap-2 text-sm">
  247. <dt class="text-slate-500 dark:text-slate-400">Provider</dt>
  248. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.GEOIP_PROVIDER|default('—') }}</dd>
  249. <dt class="text-slate-500 dark:text-slate-400">Country DB</dt>
  250. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.GEOIP_COUNTRY_DB|default('—') }}</dd>
  251. <dt class="text-slate-500 dark:text-slate-400">ASN DB</dt>
  252. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.GEOIP_ASN_DB|default('—') }}</dd>
  253. <dt class="text-slate-500 dark:text-slate-400">MaxMind key</dt>
  254. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.MAXMIND_LICENSE_KEY ? config.sections.geoip.MAXMIND_LICENSE_KEY : '(unset)' }}</dd>
  255. <dt class="text-slate-500 dark:text-slate-400">IPinfo token</dt>
  256. <dd class="col-span-2 font-mono text-xs">{{ config.sections.geoip.IPINFO_TOKEN ? config.sections.geoip.IPINFO_TOKEN : '(unset)' }}</dd>
  257. </dl>
  258. </section>
  259. {% endif %}
  260. </div>
  261. {% endblock %}