From 2127b1bdd73ecd2423e443992da4b442f5a3c1a3 Mon Sep 17 00:00:00 2001
From: Jake Vanderwerf <get@jakevanderwerf.ca>
Date: Wed, 04 Feb 2026 21:19:25 +0000
Subject: [PATCH] =Major overhaul of MetaManager.php -> Meta.php and RestRouteManager.php -> Rest.php. Seems to work for JakeVan
---
inc/registry/CheckCustomTables.php | 865 ++++++++++++++++++++++++++++++++++-----------------------
1 files changed, 515 insertions(+), 350 deletions(-)
diff --git a/inc/registry/CheckCustomTables.php b/inc/registry/CheckCustomTables.php
index 57b4795..9ec862a 100644
--- a/inc/registry/CheckCustomTables.php
+++ b/inc/registry/CheckCustomTables.php
@@ -19,6 +19,11 @@
protected array $JVB_TAXONOMY;
protected array $JVB_USER;
+ protected string $userTable;
+ protected string $userIDType;
+ protected string $termIDType;
+ protected string $postIDType;
+
public function __construct()
{
global $wpdb;
@@ -33,7 +38,64 @@
$this->JVB_CONTENT = apply_filters('jvb_content', []);
$this->JVB_TAXONOMY = apply_filters('jvb_taxonomy', []);
$this->JVB_USER = apply_filters('jvb_user', []);
- }
+
+ $this->userTable = (is_multisite()) ? $this->getMultisiteUsersTable() : $this->wpdb->users;
+
+ $this->userIDType = $this->getColumnType($this->userTable, 'ID');
+ $this->termIDType = $this->getColumnType($this->wpdb->terms, 'term_id');
+ $this->postIDType = $this->getColumnType($this->wpdb->posts, 'ID');
+ }
+
+ protected function getMultisiteUsersTable():string
+ {
+ $siteUsersTable = $this->wpdb->prefix . 'users';
+ $siteExists = $this->wpdb->get_var(
+ $this->wpdb->prepare("SHOW TABLES LIKE %s", $siteUsersTable)
+ );
+ if ($siteExists) {
+ return $siteUsersTable;
+ }
+ //fallback to main one
+ return $this->wpdb->users;
+ }
+
+ /**
+ * Get the exact column type from a WordPress core table
+ * This ensures foreign keys match the parent table exactly
+ */
+ protected function getColumnType(string $table, string $column): string
+ {
+ // First verify the table exists
+ $tableExists = $this->wpdb->get_var(
+ $this->wpdb->prepare("SHOW TABLES LIKE %s", $table)
+ );
+
+ if (!$tableExists) {
+ error_log("JVB ERROR: Table {$table} does not exist!");
+ return 'bigint(20)'; // Fallback
+ }
+
+ $result = $this->wpdb->get_row(
+ $this->wpdb->prepare(
+ "SELECT COLUMN_TYPE
+ FROM INFORMATION_SCHEMA.COLUMNS
+ WHERE TABLE_SCHEMA = DATABASE()
+ AND TABLE_NAME = %s
+ AND COLUMN_NAME = %s",
+ $table,
+ $column
+ )
+ );
+
+ if ($result && isset($result->COLUMN_TYPE)) {
+ error_log("JVB: Found Column Type for {$table}.{$column}: " . $result->COLUMN_TYPE);
+ return $result->COLUMN_TYPE;
+ }
+
+ // Fallback to signed bigint if we can't determine
+ error_log("JVB WARNING: Could not determine column type for {$table}.{$column}, using bigint(20) as fallback");
+ return 'bigint(20)';
+ }
public function maybeCreateTables()
{
@@ -41,7 +103,12 @@
error_log('JVB: Starting table creation process');
error_log('JVB: Memory usage at start: ' . memory_get_usage(true) / 1024 / 1024 . ' MB');
- $tables = $calendar = $integrations = $karma = $stats = $invitable = $verifyEntry = $approval = $trackChanges = [];
+ $tables = $calendar = $integrations = $karma = $stats = $verifyEntry = $approval = $trackChanges = [];
+ $invitable = [
+ 'roles' => [],
+ 'terms' => []
+ ];
+
// Basic tables (these worked fine)
try {
@@ -51,10 +118,9 @@
// $tables = array_merge($tables, $this->umamiTracking());
// }
}
- if (array_key_exists('can_invite', $this->JVB_MEMBERSHIP) && is_array($this->JVB_MEMBERSHIP['can_invite'])) {
- foreach ($this->JVB_MEMBERSHIP['can_invite'] as $role => $canInvite) {
- $invitable[$role]['can_invite'] = $canInvite;
- }
+ if (array_key_exists('can_invite', $this->JVB_MEMBERSHIP) &&
+ is_array($this->JVB_MEMBERSHIP['can_invite'])) {
+ $invitable['roles'] = $this->JVB_MEMBERSHIP['can_invite'];
}
// if (jvbCheck('social', $this->JVB_SITE) || jvbCheck('gmb', $this->JVB_SITE) || jvbCheck('square', $this->JVB_SITE) || jvbCheck('helcim', $this->JVB_SITE)) {
@@ -65,9 +131,9 @@
}
try {
- if (array_key_exists('dashboard', $this->JVB_SITE) && $this->JVB_SITE['dashboard']) {
+// if (array_key_exists('dashboard', $this->JVB_SITE) && $this->JVB_SITE['dashboard']) {
$tables = array_merge($tables, $this->queueTables(), $this->errorLogTables());
- }
+// }
} catch (Exception $e) {
error_log("JVB: Error in dashboard section: " . $e->getMessage());
}
@@ -81,7 +147,7 @@
}
// RE-ENABLE favourites tables
try {
- if ($this->JVB_SITE['favourites']) {
+ if (array_key_exists('favourites', $this->JVB_SITE)) {
error_log('JVB: Creating favourites tables...');
$favouriteTables = $this->favouriteTables();
error_log('JVB: Favourites tables created: ' . count($favouriteTables));
@@ -146,9 +212,7 @@
$trackChanges[$type] = $config;
}
if (array_key_exists('invitable', $config) && $config['invitable']) {
- foreach ($config['for_content'] as $content) {
- $invitable[$content]['to_terms'][] = $type;
- }
+ $invitable['terms'][] = $type;
}
if (array_key_exists('verify_entry', $config) && $config['verify_entry']) {
$verifyEntry[$type] = $config;
@@ -233,17 +297,19 @@
// RE-ENABLE other table types
try {
- if (!empty($invitable)) {
- error_log('JVB: Creating invitation tables...');
+ if (!empty($invitable['roles']) || !empty($invitable['terms'])) {
+ error_log('JVB: Creating invitation table...');
$invitationTables = $this->invitationTables($invitable);
- error_log('JVB: Invitation tables created: ' . count($invitationTables));
+ error_log('JVB: Invitation table created: ' . count($invitationTables));
$tables = array_merge($tables, $invitationTables);
- error_log('JVB: Memory after invitations: ' . memory_get_usage(true) / 1024 / 1024 . ' MB');
}
} catch (Exception $e) {
- error_log("JVB: Error creating invitation tables: " . $e->getMessage());
+ error_log("JVB: Error creating invitation table: " . $e->getMessage());
}
+ // Store config for later use
+ update_option(BASE.'invitation_config', $invitable);
+
try {
if (!empty($approval)) {
error_log('JVB: Creating approval tables...');
@@ -305,73 +371,70 @@
}
}
- public function createTables(array $tables)
- {
- $charset_collate = $this->wpdb->get_charset_collate();
- require_once(ABSPATH . 'wp-admin/includes/upgrade.php');
+ public function createTables(array $tables)
+ {
+ require_once(ABSPATH . 'wp-admin/includes/upgrade.php');
- $errors = [];
- $created = [];
+ // Match collation from existing WP tables for FK compatibility
+ $usersStatus = $this->wpdb->get_row("SHOW TABLE STATUS LIKE '{$this->wpdb->users}'");
+ $parentCollation = $usersStatus->Collation ?? 'utf8mb4_general_ci';
+ $charset_collate = "DEFAULT CHARACTER SET utf8mb4 COLLATE {$parentCollation}";
- foreach ($tables as $name => $schema) {
- $table_name = $this->prefix . BASE . $name;
- $full_schema = "CREATE TABLE IF NOT EXISTS {$table_name} {$schema} {$charset_collate};";
+ error_log("JVB: Using charset_collate: " . $charset_collate);
- // Clear any previous errors
- $this->wpdb->flush();
+ $errors = [];
+ $created = [];
- try {
- $result = dbDelta($full_schema);
+ foreach ($tables as $name => $schema) {
+ $table_name = $this->prefix . BASE . $name;
- // Check for SQL errors
- if ($this->wpdb->last_error) {
- $error_msg = "SQL Error creating table {$table_name}: " . $this->wpdb->last_error;
- error_log($error_msg);
- error_log("Failed SQL Query: " . $full_schema);
- $errors[] = $error_msg;
+ // Skip if exists
+ if ($this->wpdb->get_var("SHOW TABLES LIKE '{$table_name}'")) {
+ $created[] = $table_name . ' (exists)';
+ continue;
+ }
- // Don't throw exception, just log and continue
- continue;
- }
+ $full_schema = "CREATE TABLE IF NOT EXISTS {$table_name} {$schema} {$charset_collate}";
- // Verify table was actually created
- $table_exists = $this->wpdb->get_var("SHOW TABLES LIKE '{$table_name}'");
- if ($table_exists) {
- $created[] = $table_name;
- error_log("Successfully created table: {$table_name}");
- } else {
- $error_msg = "Table {$table_name} was not created (no error reported)";
- error_log($error_msg);
- error_log("Schema used: " . $full_schema);
- $errors[] = $error_msg;
- }
+ $this->wpdb->flush();
- } catch (Exception $e) {
- $error_msg = "Exception creating table {$table_name}: " . $e->getMessage();
- error_log($error_msg);
- error_log("Exception SQL Query: " . $full_schema);
- $errors[] = $error_msg;
- }
- }
+ // Use direct query - dbDelta mangles FK constraints
+ $hasForeignKey = stripos($schema, 'FOREIGN KEY') !== false;
- // Log summary
- if (!empty($created)) {
- error_log("JVB Tables Created Successfully: " . implode(', ', $created));
- }
+ if ($hasForeignKey) {
+ $result = $this->wpdb->query($full_schema);
+ $success = ($result !== false && !$this->wpdb->last_error);
+ } else {
+ dbDelta($full_schema . ';');
+ $success = !$this->wpdb->last_error;
+ }
- if (!empty($errors)) {
- error_log("JVB Table Creation Errors (" . count($errors) . " total):");
- foreach ($errors as $error) {
- error_log(" - " . $error);
- }
+ if (!$success) {
+ $error_msg = "SQL Error creating table {$table_name}: " . $this->wpdb->last_error;
+ error_log($error_msg);
+ error_log("Failed SQL Query: " . $full_schema);
+ $errors[] = $error_msg;
+ } elseif ($this->wpdb->get_var("SHOW TABLES LIKE '{$table_name}'")) {
+ $created[] = $table_name;
+ error_log("Successfully created table: {$table_name}");
+ }
+ }
- // Optionally store errors for admin display
- update_option(BASE . 'table_creation_errors', $errors);
- } else {
- // Clear any previous errors
- delete_option(BASE . 'table_creation_errors');
- }
- }
+ // Log summary
+ if (!empty($created)) {
+ error_log("JVB Tables Created Successfully: " . implode(', ', $created));
+ }
+
+ if (!empty($errors)) {
+ error_log("JVB Table Creation Errors (" . count($errors) . " total):");
+ foreach ($errors as $error) {
+ error_log(" - " . $error);
+ }
+ update_option(BASE . 'table_creation_errors', $errors);
+ } else {
+ delete_option(BASE . 'table_creation_errors');
+ }
+ }
/******************************************************
* Table Definitions
@@ -379,85 +442,108 @@
protected function queueTables():array
{
+
return [
'_operation_queue' => "(
`id` VARCHAR(64) NOT NULL,
`type` varchar(50) NOT NULL,
- `user_id` bigint(20) NOT NULL,
- `request_data` JSON NOT NULL,
- `count` int(11) NOT NULL DEFAULT 1,
- `progress_count` int(11) DEFAULT 0,
+ `user_id` {$this->userIDType} NOT NULL,
+
+ `request_data` JSON NOT NULL CHECK (JSON_VALID(request_data)),
+
+ `total_items` int(11) NOT NULL DEFAULT 1,
+ `processed_items` int(11) DEFAULT 0,
`failed_items` JSON,
- `priority` enum('low', 'normal', 'high') DEFAULT 'normal',
- `status` enum('pending', 'scheduled', 'processing','failed', 'failed_permanent', 'completed', 'completed_with_errors') DEFAULT 'pending',
- `metadata` JSON DEFAULT NULL,
- `merge` enum('merge', 'append', 'replace') DEFAULT 'merge',
- `scheduled_at` datetime DEFAULT CURRENT_TIMESTAMP,
- `created_at` datetime DEFAULT CURRENT_TIMESTAMP,
- `updated_at` datetime DEFAULT CURRENT_TIMESTAMP,
- `started_at` datetime DEFAULT NULL,
+
+ `priority` ENUM('high', 'normal', 'low') DEFAULT 'normal',
+ `state` enum('pending', 'scheduled', 'processing', 'completed') DEFAULT 'pending',
+ `outcome` enum('pending', 'success', 'partial', 'failed','failed_permanent') DEFAULT 'pending',
+
+ `retries` int(11) DEFAULT 0,
+ `last_error_hash` CHAR(32) DEFAULT NULL,
+ `error_message` text,
+
+ `scheduled_at` datetime DEFAULT NULL,
+ `started_at` datetime DEFAULT CURRENT_TIMESTAMP,
`completed_at` datetime DEFAULT NULL,
- `retries` int(11) DEFAULT 0,
+
+ `metadata` JSON DEFAULT NULL,
`result` JSON,
`dependencies` JSON,
- `error_message` text,
+
`user_dismissed` tinyint(1) DEFAULT 0,
- PRIMARY KEY (`id`),
- KEY `idx_status_priority` (`status`, `priority`),
- KEY `idx_user_type` (`user_id`, `type`),
- KEY `idx_created_at` (`created_at`),
- KEY `idx_completed_at` (`completed_at`),
- KEY `idx_user_status` (`user_id`, `status`),
- KEY `idx_retries_status` (`retries`, `status`),
- KEY `idx_user_status_created` (`user_id`, `status`, `created_at`),
- KEY `idx_scheduled_processing` (`scheduled_at`, `status`)
+ `created_at` DATETIME DEFAULT CURRENT_TIMESTAMP,
+ `updated_at` DATETIME DEFAULT CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ KEY `idx_run_queue` (state, priority, scheduled_at),
+ KEY `idx_user_ops` (user_id, state),
+ KEY `idx_user_type_pending` (user_id, type, state),
+ KEY `idx_completed_at` (completed_at),
+ KEY `idx_processing_stuck` (`state`, `started_at`)
)",
+
'stats__operation_queue' => "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `date` date NOT NULL,
- `type` varchar(50) NOT NULL,
- `total_operations` int NOT NULL DEFAULT 0,
- `successful_operations` int NOT NULL DEFAULT 0,
- `failed_operations` int NOT NULL DEFAULT 0,
- `average_duration` float DEFAULT NULL,
- `total_items_processed` int NOT NULL DEFAULT 0,
- `peak_queue_size` int NOT NULL DEFAULT 0,
- `peak_memory_usage` int DEFAULT NULL,
- `peak_cpu_usage` float DEFAULT NULL,
- `created_at` timestamp DEFAULT CURRENT_TIMESTAMP,
- PRIMARY KEY (`id`),
- UNIQUE KEY (`date`, `type`),
- KEY `date_idx` (`date`),
- KEY `type_idx` (`type`)
+ `id` bigint unsigned AUTO_INCREMENT,
+ `date` date NOT NULL,
+ `type` varchar(50) NOT NULL,
+
+ `total_operations` int NOT NULL DEFAULT 0,
+ `successful_operations` int NOT NULL DEFAULT 0,
+ `partial_operations` int NOT NULL DEFAULT 0,
+ `failed_operations` int NOT NULL DEFAULT 0,
+ `failed_permanent_operations` int NOT NULL DEFAULT 0,
+
+ `total_items_processed` int NOT NULL DEFAULT 0,
+
+ `average_duration` float DEFAULT NULL,
+ `max_duration` int DEFAULT NULL,
+
+ `peak_queue_size` int NOT NULL DEFAULT 0,
+
+ `peak_memory_usage` int DEFAULT NULL,
+ `peak_cpu_usage` float DEFAULT NULL,
+
+ `created_at` timestamp DEFAULT CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY (date, type),
+ KEY `date_idx` (date),
+ KEY `type_idx` (type)
)"
];
}
- protected function errorLogTables():array
- {
- return [
- 'error_log'=> "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `error_type` varchar(50) NOT NULL,
- `component` varchar(50) NOT NULL,
- `message` text NOT NULL,
- `context` JSON,
- `severity` varchar(20) NOT NULL,
- `user_id` bigint(20) unsigned,
- `created_at` timestamp DEFAULT CURRENT_TIMESTAMP,
- PRIMARY KEY (`id`),
- KEY `error_lookup` (`error_type`, `severity`, `created_at`),
- KEY `component_errors` (`component`, `created_at`)
- )"
- ];
- }
+ protected function errorLogTables():array
+ {
+ return [
+ 'error_log'=> "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `error_type` varchar(50) NOT NULL,
+ `component` varchar(100) NOT NULL,
+ `method` varchar(100) DEFAULT NULL,
+ `page_url` varchar(255) DEFAULT NULL,
+ `message` text NOT NULL,
+ `context` JSON,
+ `severity` varchar(20) NOT NULL,
+ `user_id` {$this->userIDType} DEFAULT NULL,
+ `user_was_logged_in` tinyint(1) NOT NULL,
+ `source` enum('frontend','backend') NOT NULL,
+ `created_at` timestamp DEFAULT CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ KEY `created_at` (`created_at`),
+ KEY `component_severity_date` (`component`, `severity`, `created_at`),
+ KEY `error_type_date` (`error_type`, `created_at`),
+ KEY `severity_date` (`severity`, `created_at`)
+ )"
+ ];
+ }
protected function userIntegrationsTable():array
{
+
return [
'user_integration_logs'=> "(
id bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- user_id bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
service varchar(50) NOT NULL,
action varchar(50) NOT NULL,
status enum('success','error','warning') DEFAULT 'success',
@@ -474,13 +560,14 @@
protected function notificationTables():array
{
+
return [
// Main notifications table
'notifications' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `owner_id` bigint(20) unsigned NOT NULL,
- `action_user_id` bigint(20) unsigned DEFAULT NULL,
- `target_id` bigint(20) unsigned DEFAULT NULL,
+ `owner_id` {$this->userIDType} NOT NULL,
+ `action_user_id` {$this->userIDType} NOT NULL,
+ `target_id` bigint(20) DEFAULT NULL,
`target_type` varchar(30) DEFAULT NULL,
`type` enum('new_favourite','new_artist','artist_approved','artist_invitation',
'new_term','term_approved','term_rejected','list_shared',
@@ -503,15 +590,15 @@
KEY `requires_action` (`owner_id`, `requires_action`, `action_taken`),
KEY `acting_user_lookup` (`owner_id`, `action_user_id`, `type`, `status`, `created_at`),
CONSTRAINT `{$this->base}notify_owner` FOREIGN KEY (`owner_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}action_id` FOREIGN KEY (`action_user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
'notifications_content' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`date` date NOT NULL,
`frequency` enum('daily','weekly','monthly') NOT NULL,
`tattoo_count` int unsigned NOT NULL DEFAULT 0,
@@ -531,12 +618,12 @@
KEY `recent_content` (`date`, `frequency`),
KEY `artist_frequency` (`user_id`, `frequency`),
CONSTRAINT `{$this->base}content_artist` FOREIGN KEY (`user_id`)
- REFERENCES `wp_users` (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
'notifications_user_seen' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`content_notification_id` bigint(20) unsigned NOT NULL,
`status` enum('unread','read','dismissed') NOT NULL DEFAULT 'unread',
`read_at` datetime DEFAULT NULL,
@@ -545,7 +632,7 @@
UNIQUE KEY `user_content_notif` (`user_id`, `content_notification_id`),
KEY `user_status` (`user_id`, `status`),
CONSTRAINT `{$this->base}user_content_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}user_content_notification` FOREIGN KEY (`content_notification_id`)
REFERENCES `{$this->prefixed}notifications_content` (`id`) ON DELETE CASCADE
)",
@@ -553,8 +640,8 @@
// User notification preferences
'notification_preferences' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
- `item_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `item_id` bigint(20) NOT NULL,
`notification_type` varchar(50) NOT NULL,
`frequency` enum('never','daily','weekly','monthly') DEFAULT 'never',
`last_sent` datetime DEFAULT NULL,
@@ -565,13 +652,13 @@
KEY `user_frequency` (`user_id`, `frequency`),
KEY `frequency_lookup` (`frequency`, `last_sent`),
CONSTRAINT `{$this->base}notification_pref_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
// Notification digest scheduling and tracking
'notification_digests' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`frequency` enum('daily','weekly','monthly') NOT NULL,
`scheduled_at` datetime NOT NULL,
`sent_at` datetime DEFAULT NULL,
@@ -582,14 +669,14 @@
KEY `scheduled_digests` (`frequency`, `scheduled_at`, `status`),
KEY `user_digests` (`user_id`, `frequency`),
CONSTRAINT `{$this->base}digest_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
// Analytics on notification interactions
'stats__notifications' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`notification_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`action` varchar(30) NOT NULL,
`action_source` enum('web','email','app') DEFAULT 'web',
`action_details` JSON DEFAULT NULL,
@@ -599,9 +686,9 @@
KEY `user_actions` (`user_id`, `action`),
KEY `action_analysis` (`action`, `action_source`),
CONSTRAINT `{$this->base}metrics_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}metrics_notification` FOREIGN KEY (`notification_id`)
- REFERENCES {$this->prefixed}notifications (`id`) ON DELETE CASCADE
+ REFERENCES `{$this->prefixed}notifications` (`id`) ON DELETE CASCADE
)"
];
}
@@ -610,11 +697,12 @@
{
$tables = [];
$save = [];
+
foreach ($types as $type => $config) {
$save[$type] = ($type === 'term') ? $config : 'user';
$tables['approval_'.$type.'_requests'] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`status` enum('pending','approved','rejected','appealed','expired') DEFAULT 'pending',
`required_approvals` int unsigned DEFAULT 3,
`current_approvals` int unsigned DEFAULT 0,
@@ -629,12 +717,12 @@
KEY `status` (`status`),
KEY `expiring_requests` (`status`, `expires_at`),
CONSTRAINT `{$this->base}{$type}_approval_requester` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)";
$tables['approval_'.$type.'_votes'] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`request_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`vote` enum('approve','reject','dismissed') NOT NULL,
`notes` text DEFAULT NULL,
`created_at` datetime DEFAULT CURRENT_TIMESTAMP,
@@ -642,9 +730,9 @@
UNIQUE KEY `unique_vote` (`request_id`, `user_id`),
KEY `user_votes` (`user_id`),
CONSTRAINT `{$this->base}{$type}_user_approval_request` FOREIGN KEY (`request_id`)
- REFERENCES {$this->prefixed}approval_{$type}_requests (`id`) ON DELETE CASCADE,
+ REFERENCES `{$this->prefixed}approval_{$type}_requests` (`id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}{$type}_user_approval_voter` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)";
}
if (!empty($save)) {
@@ -654,73 +742,76 @@
}
- protected function taxonomyRelationshipsTables():array
- {
+ protected function taxonomyRelationshipsTables():array
+ {
+ $tables = [
+ 'taxonomy_relationships' => "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `term_id` {$this->termIDType} NOT NULL,
+ `related_term_id` {$this->termIDType} NOT NULL,
+ `taxonomy` varchar(32) NOT NULL,
+ `related_taxonomy` varchar(32) NOT NULL,
+ `post_count` int(11) NOT NULL DEFAULT 0,
+ `is_direct` tinyint(1) NOT NULL DEFAULT 1,
+ `is_hierarchical` tinyint(1) NOT NULL DEFAULT 0,
+ `last_updated` timestamp DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ KEY `term_id` (`term_id`),
+ KEY `related_term_id` (`related_term_id`),
+ KEY `taxonomy` (`taxonomy`),
+ KEY `related_taxonomy` (`related_taxonomy`),
+ UNIQUE KEY `term_relation` (`term_id`, `related_term_id`, `taxonomy`, `related_taxonomy`),
+ CONSTRAINT `{$this->base}tax_rel_term_id` FOREIGN KEY (`term_id`)
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE,
+ CONSTRAINT `{$this->base}tax_rel_related_id` FOREIGN KEY (`related_term_id`)
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE
+ )"
+ ];
- $tables = [
- 'taxonomy_relationships' => "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `term_id` bigint(20) unsigned NOT NULL,
- `related_term_id` bigint(20) unsigned NOT NULL,
- `taxonomy` varchar(32) NOT NULL,
- `related_taxonomy` varchar(32) NOT NULL,
- `post_count` int(11) NOT NULL DEFAULT 0,
- `is_direct` tinyint(1) NOT NULL DEFAULT 1,
- `is_hierarchical` tinyint(1) NOT NULL DEFAULT 0,
- `last_updated` timestamp DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- PRIMARY KEY (`id`),
- KEY `term_id` (`term_id`),
- KEY `related_term_id` (`related_term_id`),
- KEY `taxonomy` (`taxonomy`),
- KEY `related_taxonomy` (`related_taxonomy`),
- UNIQUE KEY `term_relation` (`term_id`, `related_term_id`, `taxonomy`, `related_taxonomy`),
- CONSTRAINT `{$this->base}tax_rel_term_id` FOREIGN KEY (`term_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE,
- CONSTRAINT `{$this->base}related_term_id` FOREIGN KEY (`related_term_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE
- )"
- ];
- if ((array_key_exists('dashboard', $this->JVB_SITE) && $this->JVB_SITE['dashboard'] === true) || array_key_exists('use_feed_block', $this->JVB_SITE) && $this->JVB_SITE['use_feed_block']) {
- $tables['user_term_index'] = "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
- `term_id` bigint(20) unsigned NOT NULL,
- `taxonomy` varchar(32) NOT NULL,
- `post_count` int(11) NOT NULL DEFAULT 1,
- `is_parent` tinyint(1) NOT NULL DEFAULT 0,
- `last_used` timestamp DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- PRIMARY KEY (`id`),
- UNIQUE KEY `user_term` (`user_id`, `term_id`, `taxonomy`),
- KEY `user_taxonomy` (`user_id`, `taxonomy`),
- KEY `taxonomy` (`taxonomy`),
- CONSTRAINT `{$this->base}user_term_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
- CONSTRAINT `{$this->base}user_term_term` FOREIGN KEY (`term_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE
- )";
- }
+ if ((array_key_exists('dashboard', $this->JVB_SITE) && $this->JVB_SITE['dashboard'] === true) || array_key_exists('use_feed_block', $this->JVB_SITE) && $this->JVB_SITE['use_feed_block']) {
+ $tables['user_term_index'] = "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `user_id` {$this->userIDType} NOT NULL,
+ `term_id` {$this->termIDType} NOT NULL,
+ `taxonomy` varchar(32) NOT NULL,
+ `post_count` int(11) NOT NULL DEFAULT 1,
+ `is_parent` tinyint(1) NOT NULL DEFAULT 0,
+ `last_used` timestamp DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `user_term` (`user_id`, `term_id`, `taxonomy`),
+ KEY `user_taxonomy` (`user_id`, `taxonomy`),
+ KEY `taxonomy` (`taxonomy`),
+ CONSTRAINT `{$this->base}user_term_user_fk` FOREIGN KEY (`user_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
+ CONSTRAINT `{$this->base}user_term_term_fk` FOREIGN KEY (`term_id`)
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE
+ )";
+ }
- return $tables;
- }
+ return $tables;
+ }
protected function favouriteTables():array
{
+
return [
'favourites' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`type` varchar(50) NOT NULL,
- `target_id` bigint(20) unsigned NOT NULL,
+ `target_id` bigint(20) NOT NULL,
`notes` text DEFAULT NULL,
`date_added` datetime DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
UNIQUE KEY `unique_favourite` (`user_id`, `type`, `target_id`),
KEY `user_type` (`user_id`, `type`),
- KEY `target_type` (`target_id`, `type`)
+ KEY `target_type` (`target_id`, `type`),
+ CONSTRAINT `{$this->base}favourites_user` FOREIGN KEY (`user_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
'favourites_lists' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`name` varchar(255) NOT NULL,
`description` text,
`created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
@@ -728,13 +819,13 @@
PRIMARY KEY (`id`),
KEY `user_lists` (`user_id`),
CONSTRAINT `{$this->base}list_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
'favourites_list_items' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`list_id` bigint(20) unsigned NOT NULL,
`item_type` varchar(50) NOT NULL,
- `item_id` bigint(20) unsigned NOT NULL,
+ `item_id` bigint(20) NOT NULL,
`favourite_id` bigint(20) unsigned DEFAULT NULL,
`added_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
@@ -742,14 +833,14 @@
KEY `list_items` (`list_id`),
KEY `favourite_id` (`favourite_id`),
CONSTRAINT `{$this->base}list_items` FOREIGN KEY (`list_id`)
- REFERENCES {$this->prefixed}favourites_lists (`id`) ON DELETE CASCADE,
+ REFERENCES `{$this->prefixed}favourites_lists` (`id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}list_favourite` FOREIGN KEY (`favourite_id`)
- REFERENCES {$this->prefixed}favourites (`id`) ON DELETE SET NULL
+ REFERENCES `{$this->prefixed}favourites` (`id`) ON DELETE SET NULL
)",
'favourites_list_shares' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`list_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`email` varchar(255) NOT NULL,
`permission_type` enum('view', 'edit') NOT NULL DEFAULT 'view',
`status` enum('pending', 'accepted', 'rejected', 'revoked') NOT NULL DEFAULT 'pending',
@@ -762,14 +853,14 @@
KEY `list_shares` (`list_id`),
KEY `status_index` (`status`),
CONSTRAINT `{$this->base}share_list` FOREIGN KEY (`list_id`)
- REFERENCES {$this->prefixed}favourites_lists (`id`) ON DELETE CASCADE,
+ REFERENCES `{$this->prefixed}favourites_lists` (`id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}share_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
'favourites_list_stats' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`item_type` varchar(50) NOT NULL,
- `item_id` bigint(20) unsigned NOT NULL,
+ `item_id` bigint(20) NOT NULL,
`list_count` int NOT NULL DEFAULT 0,
`last_added` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
@@ -784,9 +875,9 @@
return [
'news_relationships' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `shop_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
- `artist_id` bigint(20) unsigned DEFAULT NULL,
+ `shop_id` {$this->termIDType} NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `artist_id` {$this->postIDType} NOT NULL,
`news_count` int(10) unsigned NOT NULL DEFAULT 0,
`last_post_date` datetime DEFAULT NULL,
`last_updated` timestamp DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
@@ -796,11 +887,11 @@
KEY `user_id` (`user_id`),
KEY `artist_id` (`artist_id`),
CONSTRAINT `{$this->base}nr_shop_news_shop` FOREIGN KEY (`shop_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE,
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}nr_shop_news_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}nr_shop_news_artist` FOREIGN KEY (`artist_id`)
- REFERENCES {$this->wpdb->posts} (`ID`) ON DELETE SET NULL
+ REFERENCES `{$this->wpdb->posts}` (`ID`) ON DELETE SET NULL
)"
];
}
@@ -810,9 +901,9 @@
return [
'responses' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `item_id` bigint(20) unsigned NOT NULL,
+ `item_id` {$this->postIDType} NOT NULL,
`content` text NOT NULL,
- `user_id` bigint(20) unsigned DEFAULT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`parent_id` bigint(20) unsigned DEFAULT NULL,
`response` text NOT NULL,
`status` enum('published','hidden','flagged','deleted') DEFAULT 'published',
@@ -828,14 +919,14 @@
KEY `parent_child` (`parent_id`),
KEY `karma_order` (`karma`),
CONSTRAINT `{$this->base}re_responses_news` FOREIGN KEY (`item_id`)
- REFERENCES {$this->wpdb->posts} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->wpdb->posts}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}re_responses_parent` FOREIGN KEY (`parent_id`)
- REFERENCES {$this->prefixed}responses (`id`) ON DELETE SET NULL
+ REFERENCES `{$this->prefixed}responses` (`id`) ON DELETE SET NULL
)",
'karma_response' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `item_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
+ `item_id` bigint(20) NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`vote` enum('up','down') NOT NULL,
`date` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
@@ -843,9 +934,9 @@
KEY `item_id` (`item_id`),
KEY `user_id` (`user_id`),
CONSTRAINT `{$this->base}_response_item_id` FOREIGN KEY (`item_id`)
- REFERENCES {$this->prefixed}responses (`id`) ON DELETE CASCADE,
+ REFERENCES `{$this->prefixed}responses` (`id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}_response_user_id` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)"
];
}
@@ -866,25 +957,29 @@
if (!$t) {
continue;
}
+
switch ($t) {
case 'posts':
+ $referenceType = $this->postIDType;
$reference_table = $this->wpdb->posts;
$reference_column = 'ID';
break;
case 'terms':
+ $referenceType = $this->termIDType;
$reference_table = $this->wpdb->terms;
$reference_column = 'term_id';
break;
case 'users':
- $reference_table = $this->wpdb->users;
+ $referenceType = $this->userIDType;
+ $reference_table = $this->userTable;
$reference_column = 'ID';
break;
}
$tables['karma_'.$type] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `item_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
+ `item_id` {$referenceType} NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`vote` enum('up','down') NOT NULL,
`date` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
@@ -892,9 +987,9 @@
KEY `item_id` (`item_id`),
KEY `user_id` (`user_id`),
CONSTRAINT `{$this->base}kt_{$type}_item_id` FOREIGN KEY (`item_id`)
- REFERENCES {$reference_table} ({$reference_column}) ON DELETE CASCADE,
+ REFERENCES `{$reference_table}` (`{$reference_column}`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}kt_{$type}_user_id` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)";
}
@@ -908,13 +1003,13 @@
foreach ($types as $type => $config) {
$tables['calendar_'.$type] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `post_id` bigint(20) unsigned NOT NULL,
- `event_type` bigint(20) unsigned,
+ `post_id` {$this->postIDType} NOT NULL,
+ `event_type` {$this->termIDType} unsigned,
-- Basic event details
`title` varchar(255) NOT NULL,
- `shop_id` bigint(20) unsigned,
- `user_id` bigint(20) unsigned,
+ `shop_id` {$this->termIDType} NOT NULL
+ `user_id` {$this->userIDType} NOT NULL,
-- Location handling
`location_type` enum('shop', 'custom', 'online') DEFAULT 'shop',
@@ -971,27 +1066,27 @@
KEY `rsvp_events` (`rsvp_required`, `rsvp_deadline`),
CONSTRAINT `{$this->base}cal_{$type}_type` FOREIGN KEY (`event_type`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE,
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}cal_{$type}_post` FOREIGN KEY (`post_id`)
- REFERENCES {$this->wpdb->posts} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->wpdb->posts}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}cal_{$type}_shop` FOREIGN KEY (`shop_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE SET NULL,
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE SET NULL,
CONSTRAINT `{$this->base}cal_{$type}_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE SET NULL
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE SET NULL
)";
$tables['calendar_'.$type.'_participants'] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`event_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`status` enum('interested','going') NOT NULL,
`created_at` timestamp DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
UNIQUE KEY `event_user` (`event_id`, `user_id`),
CONSTRAINT `{$this->base}cal_{$type}_participant_event` FOREIGN KEY (`event_id`)
- REFERENCES {$this->prefixed}calendar_{$type} (`id`) ON DELETE CASCADE,
+ REFERENCES `{$this->prefixed}calendar_{$type}` (`id`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}cal_{$type}_participant_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)";
$tables['calendar_'.$type.'_recurrence_exceptions'] = "(
@@ -1012,6 +1107,7 @@
protected function umamiTracking():array
{
+
return [
'umami_events' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
@@ -1019,12 +1115,12 @@
`timestamp` datetime NOT NULL,
`event` varchar(50) NOT NULL,
`event_type` varchar(50) NOT NULL,
- `user_id` bigint(20) unsigned DEFAULT NULL,
- `content_id` bigint(20) unsigned DEFAULT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `content_id` bigint(20) DEFAULT NULL,
`content_type` varchar(50) DEFAULT NULL,
- `source_id` bigint(20) unsigned DEFAULT NULL,
+ `source_id` bigint(20) DEFAULT NULL,
`source_type` varchar(50) DEFAULT NULL,
- `owner_id` bigint(20) unsigned DEFAULT NULL,
+ `owner_id` {$this->userIDType} NOT NULL,
`owner_type` varchar(50) DEFAULT NULL,
`referrer` varchar(100) DEFAULT NULL,
`metadata` JSON DEFAULT NULL,
@@ -1036,14 +1132,14 @@
KEY `user_idx` (`user_id`),
KEY `owner_idx` (`owner_id`),
CONSTRAINT `{$this->base}umami_user_id_link` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}umami_owner_id_link` FOREIGN KEY (`owner_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)",
'stats_performance' => "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`date` date NOT NULL,
- `user_id` bigint(20) unsigned DEFAULT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
`profile_view_count` bigint(20) unsigned DEFAULT 0,
`feed_view_count` bigint(20) unsigned DEFAULT 0,
`top_content` json DEFAULT null,
@@ -1056,42 +1152,65 @@
PRIMARY KEY (`id`),
KEY `user_date_idx` (`user_id`, `date`),
CONSTRAINT `{$this->base}performance_user_id_link` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
)"
];
}
- protected function invitationTables($types)
- {
- $tables = [];
- foreach ($types as $role => $config) {
- $definitions = "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `name` varchar(255) NOT NULL,
- `email` varchar(255) NOT NULL,
- `invitation_token` varchar(64) NOT NULL,
- `status` enum('pending', 'accepted', 'rejected', 'expired','revoked') DEFAULT 'pending',
- `inviters` JSON NOT NULL,";
- foreach($config['to_terms']??[] as $term) {
- $definitions .= "`to_{$term}` bigint(20) unsigned DEFAULT NULL,";
- }
- $definitions .= "`new_user_id` bigint(20) unsigned DEFAULT NULL,
- `expires_at` datetime NOT NULL,
- `accepted_at` datetime DEFAULT NULL,
- `created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
- `updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- PRIMARY KEY (`id`),
- UNIQUE KEY `unique_email` (`email`),
- KEY `token_lookup` (`invitation_token`),
- KEY `status_expiry` (`status`, `expires_at`),
- KEY `name_status` (`name`, `status`)
- )";
-
- $tables['invitations_'.$role] = $definitions;
+ protected function invitationTables(array $config): array
+ {
+ if (empty($config['roles']) && empty($config['terms'])) {
+ return [];
}
- return $tables;
- }
+ $definitions = "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `name` varchar(255) NOT NULL,
+ `email` varchar(255) NOT NULL,
+ `invitation_token` varchar(255) NOT NULL,
+ `invited_role` varchar(50) NOT NULL COMMENT 'Role being invited to',
+ `status` enum('pending','accepted','rejected','expired','revoked') DEFAULT 'pending',
+ `inviters` JSON NOT NULL COMMENT 'Array of {user_id, invited_at}',
+ `new_user_id` {$this->userIDType} DEFAULT NULL,
+ `expires_at` datetime NOT NULL,
+ `accepted_at` datetime DEFAULT NULL,
+ `created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ `updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ ";
+
+ // Add term columns for all invitable taxonomies
+ foreach ($config['terms'] ?? [] as $taxonomy) {
+ $definitions .= "`to_{$taxonomy}` {$this->termIDType} DEFAULT NULL,";
+ }
+
+ $definitions .= "PRIMARY KEY (`id`),
+ UNIQUE KEY `unique_email_role` (`email`, `invited_role`),
+ KEY `token_lookup` (`invitation_token`),
+ KEY `status_expiry` (`status`, `expires_at`),
+ KEY `role_status` (`invited_role`, `status`),
+ KEY `email_status` (`email`, `status`),
+ ";
+
+ // Add foreign key constraints for terms
+ $constraints = [];
+ foreach ($config['terms'] ?? [] as $taxonomy) {
+ $constraints[] = "CONSTRAINT `{$this->base}invitations_{$taxonomy}_fk`
+ FOREIGN KEY (`to_{$taxonomy}`)
+ REFERENCES `{$this->wpdb->terms}` (`term_id`)
+ ON DELETE SET NULL";
+ }
+
+ // Add user foreign key
+ $constraints[] = "CONSTRAINT `{$this->base}invitations_user_fk`
+ FOREIGN KEY (`new_user_id`)
+ REFERENCES `{$this->userTable}` (`ID`)
+ ON DELETE SET NULL";
+
+ $definitions .= implode(',', $constraints);
+ $definitions .= ")";
+
+ return ['invitations' => $definitions];
+ }
protected function trackChangesTables($types)
{
@@ -1101,9 +1220,9 @@
foreach ($contents as $content) {
$tables['history_'.$content.'_'.$type] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
- `content_id` bigint(20) unsigned NOT NULL,
- `term_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `content_id` bigint(20) NOT NULL,
+ `term_id` {$this->termIDType} NOT NULL,
`role` varchar(50) DEFAULT 'artist',
`is_primary` tinyint(1) DEFAULT 0,
`start_date` date DEFAULT NULL,
@@ -1113,9 +1232,9 @@
UNIQUE KEY `content_term` (`content_id`, `term_id`),
KEY content_role (`term_id`, `role`),
CONSTRAINT `{$this->base}{$content}_{$type}_history_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}{$content}_{$type}_history_term` FOREIGN KEY (`term_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE
)";
}
}
@@ -1130,9 +1249,9 @@
foreach ($contents as $content) {
$tables[$content.'_'.$type.'_requests'] = "(
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `user_id` bigint(20) unsigned NOT NULL,
- `content_id` bigint(20) unsigned NOT NULL,
- `term_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `content_id` bigint(20) NOT NULL,
+ `term_id` {$this->termIDType} NOT NULL,
`managers` json DEFAULT NULL,
`status` ENUM('requested', 'rejected', 'accepted') DEFAULT 'requested',
`dismissed` smallint(1) unsigned DEFAULT NULL,
@@ -1142,9 +1261,9 @@
PRIMARY KEY (`id`),
UNIQUE KEY `{$this->base}content_term` (`content_id`, `term_id`),
CONSTRAINT `{$this->base}{$content}_{$type}_request_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
CONSTRAINT `{$this->base}{$content}_{$type}_request_term` FOREIGN KEY (`term_id`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE
)";
}
}
@@ -1161,58 +1280,106 @@
*/
protected function referralTables(): array
{
- $tables = [];
+ // Create tables in dependency order
+ // First: referrals (depends only on wp_users)
+ $mainTable['referrals'] = "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `referrer_id` {$this->userIDType} NOT NULL,
+ `referee_id` {$this->userIDType} NOT NULL,
+ `referee_name` varchar(255) NOT NULL,
+ `referee_email` varchar(255) NOT NULL,
+ `referee_phone` varchar(50) DEFAULT NULL,
+ `referral_code` varchar(50) NOT NULL,
+ `status` enum('pending', 'consulted', 'treated', 'cancelled') DEFAULT 'pending',
+ `referred_at` datetime NOT NULL,
+ `consulted_at` datetime DEFAULT NULL,
+ `treated_at` datetime DEFAULT NULL,
+ `treatment_count` int DEFAULT 0,
+ `notes` text DEFAULT NULL,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `referee_unique` (`referee_id`),
+ KEY `referrer_idx` (`referrer_id`),
+ KEY `status_idx` (`status`),
+ KEY `code_idx` (`referral_code`),
+ KEY `date_idx` (`referred_at`),
+ KEY `consult_idx` (`consulted_at`),
+ CONSTRAINT `{$this->base}referral_referrer_fk` FOREIGN KEY (`referrer_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE,
+ CONSTRAINT `{$this->base}referral_referee_fk` FOREIGN KEY (`referee_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
+)";
- // Main referrals table
- $tables['referrals'] = "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `referrer_id` bigint(20) unsigned NOT NULL,
- `referee_id` bigint(20) unsigned NOT NULL,
- `referee_name` varchar(255) NOT NULL,
- `referee_email` varchar(255) NOT NULL,
- `referee_phone` varchar(50) DEFAULT NULL,
- `referral_code` varchar(50) NOT NULL,
- `status` enum('pending', 'treated', 'cancelled') DEFAULT 'pending',
- `referred_at` datetime NOT NULL,
- `treated_at` datetime DEFAULT NULL,
- `notes` text DEFAULT NULL,
- PRIMARY KEY (`id`),
- UNIQUE KEY `referee_unique` (`referee_id`),
- KEY `referrer_idx` (`referrer_id`),
- KEY `status_idx` (`status`),
- KEY `code_idx` (`referral_code`),
- KEY `date_idx` (`referred_at`),
- CONSTRAINT `{$this->base}referral_referrer_fk` FOREIGN KEY (`referrer_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE,
- CONSTRAINT `{$this->base}referral_referee_fk` FOREIGN KEY (`referee_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
- )";
+ // Create the main referrals table first
+ $this->createTables($mainTable);
- // Rewards table
- $tables['referral_rewards'] = "(
- `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
- `referral_id` bigint(20) unsigned NOT NULL,
- `user_id` bigint(20) unsigned NOT NULL,
- `reward_type` enum('referrer', 'referee') NOT NULL,
- `amount` decimal(10,2) NOT NULL,
- `reward_calculation` varchar(20) DEFAULT NULL COMMENT 'percentage or fixed',
- `status` enum('available', 'redeemed', 'expired', 'cancelled') DEFAULT 'available',
- `created_at` datetime NOT NULL,
- `redeemed_at` datetime DEFAULT NULL,
- `expires_at` datetime DEFAULT NULL,
- `notes` text DEFAULT NULL,
- PRIMARY KEY (`id`),
- KEY `referral_idx` (`referral_id`),
- KEY `user_idx` (`user_id`),
- KEY `status_idx` (`status`),
- KEY `type_idx` (`reward_type`),
- CONSTRAINT `{$this->base}reward_referral_fk` FOREIGN KEY (`referral_id`)
- REFERENCES {$this->wpdb->prefix}" . BASE . "referrals (`id`) ON DELETE CASCADE,
- CONSTRAINT `{$this->base}reward_user_fk` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE
- )";
+ // Now create dependent tables
+ $dependentTables = [];
- return $tables;
+ // Second: jane_clients (depends only on wp_users)
+ $dependentTables['jane_clients'] = "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `patient_guid` varchar(50) NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `first_name` varchar(100) NOT NULL,
+ `last_name` varchar(100) NOT NULL,
+ `email` varchar(255) NOT NULL,
+ `imported_at` datetime DEFAULT CURRENT_TIMESTAMP,
+ `updated_at` datetime DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `patient_guid_unique` (`patient_guid`),
+ KEY `user_idx` (`user_id`),
+ KEY `email_idx` (`email`),
+ CONSTRAINT `{$this->base}jane_client_user_fk` FOREIGN KEY (`user_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
+)";
+
+ // Third: referral_treatments (depends on referrals AND wp_users)
+ $dependentTables['referral_treatments'] = "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `referral_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `treatment_type` varchar(100) NOT NULL COMMENT 'Tier 1-6, Brows, etc',
+ `treatment_date` datetime NOT NULL,
+ `invoice_number` varchar(50) DEFAULT NULL,
+ `amount` decimal(10,2) DEFAULT NULL,
+ `status` enum('completed', 'no_show', 'cancelled') DEFAULT 'completed',
+ `imported_at` datetime DEFAULT CURRENT_TIMESTAMP,
+ PRIMARY KEY (`id`),
+ KEY `referral_idx` (`referral_id`),
+ KEY `user_idx` (`user_id`),
+ KEY `date_idx` (`treatment_date`),
+ KEY `type_idx` (`treatment_type`),
+ CONSTRAINT `{$this->base}treatment_referral_fk` FOREIGN KEY (`referral_id`)
+ REFERENCES `{$this->prefixed}referrals` (`id`) ON DELETE CASCADE,
+ CONSTRAINT `{$this->base}treatment_user_fk` FOREIGN KEY (`user_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
+)";
+
+ // Fourth: referral_rewards (depends on referrals AND wp_users)
+ $dependentTables['referral_rewards'] = "(
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `referral_id` bigint(20) unsigned NOT NULL,
+ `user_id` {$this->userIDType} NOT NULL,
+ `reward_type` enum('referrer', 'referee') NOT NULL,
+ `amount` decimal(10,2) NOT NULL,
+ `reward_calculation` varchar(20) DEFAULT NULL COMMENT 'percentage or fixed',
+ `status` enum('available', 'redeemed', 'expired', 'cancelled') DEFAULT 'available',
+ `created_at` datetime NOT NULL,
+ `redeemed_at` datetime DEFAULT NULL,
+ `expires_at` datetime DEFAULT NULL,
+ `notes` text DEFAULT NULL,
+ PRIMARY KEY (`id`),
+ KEY `referral_idx` (`referral_id`),
+ KEY `user_idx` (`user_id`),
+ KEY `status_idx` (`status`),
+ KEY `type_idx` (`reward_type`),
+ CONSTRAINT `{$this->base}reward_referral_fk` FOREIGN KEY (`referral_id`)
+ REFERENCES `{$this->prefixed}referrals` (`id`) ON DELETE CASCADE,
+ CONSTRAINT `{$this->base}reward_user_fk` FOREIGN KEY (`user_id`)
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE
+)";
+
+ return $dependentTables;
}
/*******************************************************************************************
@@ -1238,9 +1405,9 @@
'time' => "time {$nullable}",
'datetime' => "datetime {$nullable}",
'true_false' => "boolean DEFAULT FALSE",
- 'image', 'file' => "bigint(20) unsigned {$nullable}",
+ 'image', 'file' => "bigint(20) {$nullable}",
'taxonomy' => $this->getTaxonomyColumnDefinition($fieldConfig, $nullable),
- 'user' => "bigint(20) unsigned {$nullable}",
+ 'user' => "bigint(20) {$nullable}",
'location' => $this->getLocationColumnDefinition($fieldName, $nullable),
'select', 'radio' => $this->getSelectColumnDefinition($fieldConfig, $nullable),
'set', 'checkbox', 'repeater', 'gallery' => "json {$nullable}",
@@ -1289,7 +1456,7 @@
$limit = $fieldConfig['limit'] ?? null;
if ($limit === 1) {
- return "bigint(20) unsigned {$nullable}";
+ return "{$this->termIDType} {$nullable}";
}
// Multiple selections stored as JSON
@@ -1331,13 +1498,13 @@
if ($type === 'taxonomy') {
$constraints[] = "CONSTRAINT `{$constraintName}` FOREIGN KEY (`{$fieldName}`)
- REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE SET NULL";
+ REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE SET NULL";
} elseif ($type === 'user') {
$constraints[] = "CONSTRAINT `{$constraintName}` FOREIGN KEY (`{$fieldName}`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE SET NULL";
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE SET NULL";
} elseif ($type === 'image' || $type === 'file') {
$constraints[] = "CONSTRAINT `{$constraintName}` FOREIGN KEY (`{$fieldName}`)
- REFERENCES {$this->wpdb->posts} (`ID`) ON DELETE SET NULL";
+ REFERENCES `{$this->wpdb->posts}` (`ID`) ON DELETE SET NULL";
}
}
@@ -1366,9 +1533,8 @@
try {
$tableName = 'content_' . $type;
$columns = [];
-
// Always include primary key
- $columns[] = "`term_id` bigint(20) unsigned NOT NULL";
+ $columns[] = "`term_id` {$this->termIDType} NOT NULL";
$columns[] = "`name` varchar(255) NOT NULL";
$columns[] = "`slug` varchar(255) NOT NULL";
@@ -1435,7 +1601,7 @@
["PRIMARY KEY (`term_id`)"],
$indexes,
$constraints,
- ["CONSTRAINT `{$base}_{$type}_term` FOREIGN KEY (`term_id`) REFERENCES {$this->wpdb->terms} (`term_id`) ON DELETE CASCADE"]
+ ["CONSTRAINT `{$base}_{$type}_term` FOREIGN KEY (`term_id`) REFERENCES `{$this->wpdb->terms}` (`term_id`) ON DELETE CASCADE"]
);
$sql = "(\n " . implode(",\n ", $allDefinitions) . "\n)";
error_log("JVB: Creating content table for type: {$type}");
@@ -1477,7 +1643,7 @@
// Base columns that every user stats table should have
$columns = [
- "`user_id` bigint(20) unsigned NOT NULL",
+ "`user_id` {$this->userIDType} NOT NULL",
"`display_name` VARCHAR(255) NULL",
"`email` VARCHAR(255) NULL",
"`city` VARCHAR(100) NULL",
@@ -1500,12 +1666,11 @@
error_log('JVB: Favourites column added');
}
- // CRITICAL FIX: Simplified profile-specific fields processing
if (isset($config['profile_type'])) {
$profileType = $config['profile_type'];
error_log("JVB: Processing profile type: {$profileType}");
- // SAFE check for profile fields
+
if (isset($this->JVB_CONTENT[$profileType]) &&
isset($this->JVB_CONTENT[$profileType]['fields']) &&
is_array($this->JVB_CONTENT[$profileType]['fields'])) {
@@ -1600,7 +1765,7 @@
$constraints = [
"CONSTRAINT `{$this->base}_{$userType}_stats_user` FOREIGN KEY (`user_id`)
- REFERENCES {$this->wpdb->users} (`ID`) ON DELETE CASCADE"
+ REFERENCES `{$this->userTable}` (`ID`) ON DELETE CASCADE"
];
$allDefinitions = array_merge($columns, $indexes, $constraints);
--
Gitblit v1.10.0