This commit is contained in:
2026-06-07 12:11:04 -04:00
parent 10c848660e
commit 207b08b830

View File

@@ -1,9 +1,9 @@
<?php <?php
/** /**
* Name: DSC01 Vital Signs * Name: DSC01 DSC Categories
* Description: Public civic diagnostic — the ten structural preconditions of an HOA association. * Description: Public civic diagnostic — the legal surfaces where HOA governance disputes manifest.
* Version: 0.1.0 * Version: 0.2.0
* MinVersion: 11.0 * MinVersion: 11.0
* MaxVersion: 12.0 * MaxVersion: 12.0
*/ */
@@ -32,46 +32,92 @@ function dsc01_load_pdl(&$b) {
} }
} }
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// HELPERS // HELPERS
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
function dsc01_h($value) { function dsc01_h($value) {
return htmlspecialchars((string) $value, ENT_QUOTES, 'UTF-8'); return htmlspecialchars((string) $value, ENT_QUOTES, 'UTF-8');
} }
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// ACCESS // ACCESS
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
function dsc01_access_state() { function dsc01_access_state($association_slug = '') {
if (!local_channel()) { // Operator check — association channel owner (local channel only)
if (local_channel()) {
$channel = App::get_channel();
if (local_channel() === intval($channel['channel_id'])) {
return 'operator';
}
}
if (!$association_slug) {
return 'public'; return 'public';
} }
$channel = App::get_channel(); // Load association config from vs01 — the single source of truth.
$raw = @file_get_contents('addon/vs01/config.json');
if ($raw === false) return 'public';
$cfg = json_decode($raw, true);
if (json_last_error() !== JSON_ERROR_NONE) return 'public';
if (local_channel() === intval($channel['channel_id'])) { $assoc = $cfg['associations'][$association_slug] ?? null;
return 'operator'; if (!$assoc) return 'public';
// get_observer_hash() works for both local channels and guest token visitors.
$observer = get_observer_hash();
if (!$observer) return 'public';
$groups = $assoc['groups'] ?? [];
// Direct pgrp_member query — does not call local_channel(), works for guest tokens.
// Corpus Builder — highest participant tier
$cb_gid = intval($groups['corpus_builder'] ?? 0);
if ($cb_gid) {
$r = q("SELECT xchan FROM pgrp_member WHERE gid = %d AND xchan = '%s' LIMIT 1",
intval($cb_gid),
dbesc($observer)
);
if ($r) return 'participant';
} }
$config = dsc01_load_config(); // SASE Participant
$gid = intval($config['corpus_builder_group_id'] ?? 0); $sase_gid = intval($groups['sase_participant'] ?? 0);
if ($sase_gid) {
if ($gid && in_array(get_observer_hash(), group_get_members_xchan($gid))) { $r = q("SELECT xchan FROM pgrp_member WHERE gid = %d AND xchan = '%s' LIMIT 1",
return 'participant'; intval($sase_gid),
dbesc($observer)
);
if ($r) return 'participant';
} }
return 'denied'; // Civic Professional
$prof_gid = intval($groups['civic_professional'] ?? 0);
if ($prof_gid) {
$r = q("SELECT xchan FROM pgrp_member WHERE gid = %d AND xchan = '%s' LIMIT 1",
intval($prof_gid),
dbesc($observer)
);
if ($r) return 'participant';
}
return 'public';
} }
function dsc01_access_wall() { function dsc01_access_wall($association_slug = '') {
$raw = @file_get_contents('addon/vs01/config.json');
$cfg = $raw ? json_decode($raw, true) : [];
$assoc = $association_slug ? ($cfg['associations'][$association_slug] ?? null) : null;
$name = $assoc ? dsc01_h($assoc['name']) : 'this association';
return ' return '
<div class="dsc01-content"> <div class="dsc01-content">
<div class="alert alert-info" role="alert"> <div class="alert alert-info" role="alert">
<strong>HOA_MEMBER standing required to submit.</strong> <strong>HOA_MEMBER standing required to submit a record for ' . $name . '.</strong>
Vital Signs are public and readable by anyone. DSC Categories are public and readable by anyone.
To submit a Vital Signs record for your association, you must complete the SASE process. To submit a record, you must complete the SASE process.
Visit <a href="https://directory.diagnostics.kane-il.us/channel/theron"> Visit <a href="https://directory.diagnostics.kane-il.us/channel/theron">
directory.diagnostics.kane-il.us directory.diagnostics.kane-il.us
</a> to begin. </a> to begin.
@@ -80,9 +126,9 @@ function dsc01_access_wall() {
'; ';
} }
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// CONTENT // CONTENT
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
function dsc01_content() { function dsc01_content() {
if (function_exists('head_add_css')) { if (function_exists('head_add_css')) {
@@ -92,23 +138,25 @@ function dsc01_content() {
head_add_js('/addon/dsc01/view/js/dsc01.js'); head_add_js('/addon/dsc01/view/js/dsc01.js');
} }
$access = dsc01_access_state(); $association_slug = argv(1) ?? '';
$access = dsc01_access_state($association_slug);
// dsc01 is public — access wall only gates submission, not reading // dsc01 is public — access wall only gates submission, not reading
if ($_SERVER['REQUEST_METHOD'] === 'POST') { if ($_SERVER['REQUEST_METHOD'] === 'POST') {
if ($access === 'public' || $access === 'denied') { if ($access === 'public') {
return dsc01_access_wall(); return dsc01_access_wall($association_slug);
} }
// TODO: handle POST submission // TODO: handle POST submission
return dsc01_access_wall(); return dsc01_access_wall($association_slug);
} }
return dsc01_render_main($access); return dsc01_render_main($access);
} }
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// RENDER // RENDER
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
function dsc01_render_main($access) { function dsc01_render_main($access) {
$out = '<div class="dsc01-content">'; $out = '<div class="dsc01-content">';
@@ -117,7 +165,7 @@ function dsc01_render_main($access) {
$out .= '<p class="text-muted">The legal surfaces where HOA governance disputes manifest, organized from the homeowner\'s perspective.</p>'; $out .= '<p class="text-muted">The legal surfaces where HOA governance disputes manifest, organized from the homeowner\'s perspective.</p>';
$out .= '</div>'; $out .= '</div>';
// TODO: render the ten Vital Signs // TODO: render DSC categories
$out .= '<div class="dsc01-placeholder text-muted fst-italic">Content forthcoming.</div>'; $out .= '<div class="dsc01-placeholder text-muted fst-italic">Content forthcoming.</div>';
$out .= '</div>'; $out .= '</div>';
@@ -125,9 +173,9 @@ function dsc01_render_main($access) {
return $out; return $out;
} }
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// CONFIG // CONFIG
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
function dsc01_load_config() { function dsc01_load_config() {
$path = 'addon/dsc01/config.json'; $path = 'addon/dsc01/config.json';
@@ -137,9 +185,9 @@ function dsc01_load_config() {
return (json_last_error() === JSON_ERROR_NONE) ? $data : []; return (json_last_error() === JSON_ERROR_NONE) ? $data : [];
} }
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
// CSRF // CSRF
// ---------------------------------------------------------------------------- // ---------------------------------------------------------------------------
function dsc01_csrf_token() { function dsc01_csrf_token() {
if (empty($_SESSION['dsc01_csrf'])) { if (empty($_SESSION['dsc01_csrf'])) {