Fix sites manager: DB-first settings storage, all 5 sites now read/write

Root cause: open_basedir restriction prevented lsphp from reading config
files on other vhosts (/home/epictravelexpeditions.com/*, etc.)
TJJ worked via its own DB; TTG worked because file happened to be readable.
Epic, Parker Slingshot, Parker Slingshot Rentals all returned empty.

Fix: site_settings table in jarvis_db as primary storage for all sites.
- All reads/writes go through JarvisDB (always accessible, no open_basedir)
- TJJ still syncs bidirectionally with its own settings table (secondary)
- File push attempted best-effort on save (succeeds where writable, silent no-op otherwise)
- Seeded existing known values from file grep on table creation
- All 5 sites now return and save settings correctly
This commit is contained in:
2026-06-02 12:48:51 +00:00
parent 85d3447ccc
commit d77621c849
+90 -73
View File
@@ -1,58 +1,86 @@
<?php
/**
* JARVIS Sites Manager — read/write email settings across all hosted sites
* JARVIS Sites Manager
* Primary storage: site_settings table in jarvis_db (always accessible)
* Secondary sync: TJJ own DB settings table (kept in sync on save)
* File push: optional, best-effort write to config files where accessible
*/
// ── Site definitions ────────────────────────────────────────────────
$SITES = [
'tomsjavajive' => [
'name' => "Tom's Java Jive",
'url' => 'https://tomsjavajive.com',
'type' => 'db',
'db' => ['host'=>'localhost','name'=>'toms_tjj_db','user'=>'toms_tjj_user','pass'=>'+60wlPc+55e@gFq4'],
'keys' => ['api_key'=>'cybermail_api_key','from_email'=>'cybermail_from_email','from_name'=>'cybermail_from_name','admin_email'=>'smtp_admin_email'],
// Also sync to TJJ's own settings table
'sync_db' => ['host'=>'localhost','name'=>'toms_tjj_db','user'=>'toms_tjj_user','pass'=>'+60wlPc+55e@gFq4'],
'sync_keys' => ['api_key'=>'cybermail_api_key','from_email'=>'cybermail_from_email','from_name'=>'cybermail_from_name','admin_email'=>'smtp_admin_email'],
],
'tomtomgames' => [
'name' => 'TomTomGames',
'url' => 'https://tomtomgames.com',
'type' => 'file',
// Also try to push to file
'file' => '/home/tomtomgames.com/includes/config.php',
'keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'SMTP_FROM','from_name'=>'SMTP_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
'file_keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'SMTP_FROM','from_name'=>'SMTP_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
],
'epictravelexpeditions' => [
'name' => 'Epic Travel Expeditions',
'url' => 'https://epictravelexpeditions.com',
'type' => 'file',
'file' => '/home/epictravelexpeditions.com/public_html/api/config.php',
'keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'MAIL_FROM','from_name'=>'MAIL_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
'file_keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'MAIL_FROM','from_name'=>'MAIL_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
],
'parkerslingshot' => [
'name' => 'Parker Slingshot',
'url' => 'https://parkerslingshot.epictravelexpeditions.com',
'type' => 'file',
'file' => '/home/epictravelexpeditions.com/parkerslingshot/db.php',
'keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'MAIL_FROM','from_name'=>'MAIL_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
'file_keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'MAIL_FROM','from_name'=>'MAIL_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
],
'parkerslingshotrentals' => [
'name' => 'Parker Slingshot Rentals',
'url' => 'https://parkerslingshotrentals.com',
'type' => 'file',
'file' => '/home/parkerslingshotrentals.com/public_html/db.php',
'keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'MAIL_FROM','from_name'=>'MAIL_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
'file_keys' => ['api_key'=>'CYBERMAIL_API_KEY','from_email'=>'MAIL_FROM','from_name'=>'MAIL_FROM_NAME','admin_email'=>'ADMIN_EMAIL'],
],
];
// ── Helpers ──────────────────────────────────────────────────────────
function fileGet(string $file, string $constant): string {
if (!file_exists($file)) return '';
$content = file_get_contents($file);
if (preg_match("/define\s*\(\s*['\"]" . preg_quote($constant, '/') . "['\"],\s*['\"]([^'\"]*)['\"].*?\)/s", $content, $m))
return $m[1];
return '';
// ── Primary storage: jarvis_db site_settings table ──────────────────
function ssGet(string $siteId, string $field): string {
$row = JarvisDB::single(
'SELECT setting_value FROM site_settings WHERE site_id=? AND setting_key=?',
[$siteId, $field]
);
return $row['setting_value'] ?? '';
}
function fileSet(string $file, string $constant, string $value): bool {
if (!file_exists($file)) return false;
function ssSet(string $siteId, string $field, string $value): bool {
JarvisDB::execute(
'INSERT INTO site_settings (site_id, setting_key, setting_value) VALUES (?,?,?)
ON DUPLICATE KEY UPDATE setting_value=VALUES(setting_value), updated_at=NOW()',
[$siteId, $field, $value]
);
return true;
}
// ── Secondary sync: TJJ own DB ───────────────────────────────────────
function tjjSync(array $syncDb, array $syncKeys, string $field, string $value): void {
$key = $syncKeys[$field] ?? null;
if (!$key) return;
try {
$pdo = new PDO("mysql:host={$syncDb['host']};dbname={$syncDb['name']};charset=utf8mb4",
$syncDb['user'], $syncDb['pass'], [PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION]);
$exists = $pdo->prepare('SELECT id FROM settings WHERE setting_key=?');
$exists->execute([$key]);
if ($exists->fetch()) {
$pdo->prepare('UPDATE settings SET setting_value=? WHERE setting_key=?')
->execute([json_encode($value), $key]);
} else {
$pdo->prepare('INSERT INTO settings (setting_key, setting_value) VALUES (?,?)')
->execute([$key, json_encode($value)]);
}
} catch (Exception $e) { /* best-effort */ }
}
// ── File push: best-effort write to config file ──────────────────────
function filePush(string $file, string $constant, string $value): void {
if (!file_exists($file) || !is_writable($file)) return;
$content = file_get_contents($file);
$safe = str_replace("'", "\\'", $value);
$new = preg_replace(
@@ -60,64 +88,45 @@ function fileSet(string $file, string $constant, string $value): bool {
"define('" . $constant . "', '" . $safe . "'$1)",
$content
);
if ($new === null) return false; // regex error
if ($new === $content) return true; // already correct value
return file_put_contents($file, $new) !== false;
if ($new && $new !== $content) {
file_put_contents($file, $new);
}
}
function dbGet(array $db, string $key): string {
// ── Initial population: seed jarvis_db from TJJ DB on first access ──
function seedFromTjjDb(string $siteId, array $syncDb, array $syncKeys): void {
foreach ($syncKeys as $field => $key) {
if (ssGet($siteId, $field) !== '') continue; // already have it
try {
$pdo = new PDO("mysql:host={$db['host']};dbname={$db['name']};charset=utf8mb4",
$db['user'], $db['pass'], [PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION]);
$s = $pdo->prepare("SELECT setting_value FROM settings WHERE setting_key=?");
$pdo = new PDO("mysql:host={$syncDb['host']};dbname={$syncDb['name']};charset=utf8mb4",
$syncDb['user'], $syncDb['pass'], [PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION]);
$s = $pdo->prepare('SELECT setting_value FROM settings WHERE setting_key=?');
$s->execute([$key]);
$row = $s->fetch(PDO::FETCH_ASSOC);
if (!$row) return '';
$decoded = json_decode($row['setting_value'], true);
return $decoded ?? $row['setting_value'];
} catch (Exception $e) { return ''; }
}
function dbSet(array $db, string $key, string $value): bool {
try {
$pdo = new PDO("mysql:host={$db['host']};dbname={$db['name']};charset=utf8mb4",
$db['user'], $db['pass'], [PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION]);
$existing = $pdo->prepare("SELECT id FROM settings WHERE setting_key=?");
$existing->execute([$key]);
if ($existing->fetch()) {
$pdo->prepare("UPDATE settings SET setting_value=? WHERE setting_key=?")->execute([json_encode($value), $key]);
} else {
$pdo->prepare("INSERT INTO settings (setting_key, setting_value) VALUES (?,?)")->execute([$key, json_encode($value)]);
if ($row) {
$v = json_decode($row['setting_value'], true) ?? $row['setting_value'];
ssSet($siteId, $field, $v);
}
} catch (Exception $e) { /* best-effort */ }
}
return true;
} catch (Exception $e) { return false; }
}
function siteGet(array $site, string $field): string {
$constant = $site['keys'][$field] ?? '';
if (!$constant) return '';
if ($site['type'] === 'db') return dbGet($site['db'], $constant);
return fileGet($site['file'], $constant);
}
function siteSet(array $site, string $field, string $value): bool {
$constant = $site['keys'][$field] ?? '';
if (!$constant) return false;
if ($site['type'] === 'db') return dbSet($site['db'], $constant, $value);
return fileSet($site['file'], $constant, $value);
}
// ── Router ───────────────────────────────────────────────────────────
if ($method === 'GET') {
// Seed TJJ from its own DB if not yet in jarvis_db
if (isset($SITES['tomsjavajive']['sync_db'])) {
seedFromTjjDb('tomsjavajive', $SITES['tomsjavajive']['sync_db'], $SITES['tomsjavajive']['sync_keys']);
}
$result = [];
foreach ($SITES as $id => $site) {
$result[$id] = [
'name' => $site['name'],
'url' => $site['url'],
'api_key' => siteGet($site, 'api_key'),
'from_email' => siteGet($site, 'from_email'),
'from_name' => siteGet($site, 'from_name'),
'admin_email'=> siteGet($site, 'admin_email'),
'api_key' => ssGet($id, 'api_key'),
'from_email' => ssGet($id, 'from_email'),
'from_name' => ssGet($id, 'from_name'),
'admin_email' => ssGet($id, 'admin_email'),
];
}
echo json_encode(['success' => true, 'sites' => $result]);
@@ -127,14 +136,16 @@ if ($method === 'GET') {
if ($method === 'POST') {
$action = $data['action'] ?? '';
$siteId = $data['site'] ?? '';
$results = [];
if ($action === 'push_key') {
// Push API key to all sites at once
$apiKey = trim($data['api_key'] ?? '');
if (!$apiKey) { echo json_encode(['success'=>false,'error'=>'API key required']); exit; }
$results = [];
foreach ($SITES as $id => $site) {
$results[$id] = siteSet($site, 'api_key', $apiKey);
ssSet($id, 'api_key', $apiKey);
if (isset($site['sync_db'])) tjjSync($site['sync_db'], $site['sync_keys'], 'api_key', $apiKey);
if (isset($site['file'])) filePush($site['file'], $site['file_keys']['api_key'] ?? '', $apiKey);
$results[$id] = true;
}
echo json_encode(['success'=>true,'results'=>$results]);
exit;
@@ -142,13 +153,19 @@ if ($method === 'POST') {
if ($action === 'save' && $siteId && isset($SITES[$siteId])) {
$site = $SITES[$siteId];
$fields = ['api_key', 'from_email', 'from_name', 'admin_email'];
$fields = ['api_key','from_email','from_name','admin_email'];
foreach ($fields as $field) {
if (isset($data[$field])) {
$results[$field] = siteSet($site, $field, trim($data[$field]));
if (!isset($data[$field])) continue;
$value = trim($data[$field]);
// 1. Always save to jarvis_db
ssSet($siteId, $field, $value);
// 2. Sync to TJJ own DB if applicable
if (isset($site['sync_db'])) tjjSync($site['sync_db'], $site['sync_keys'], $field, $value);
// 3. Push to file if accessible
if (isset($site['file']) && isset($site['file_keys'][$field]))
filePush($site['file'], $site['file_keys'][$field], $value);
}
}
echo json_encode(['success'=>true,'results'=>$results]);
echo json_encode(['success'=>true]);
exit;
}