cleanupLegacyExtensions(); $this->enablePlugin('system', 'mokowaas'); $this->enablePlugin('webservices', 'mokowaas'); $this->enablePlugin('task', 'mokowaasdemo'); $this->enablePlugin('task', 'mokowaassync'); // Mark MokoWaaS extensions as protected (prevents disable/uninstall at framework level) $this->protectExtensions(); // Clean up stale/duplicate update sites $this->cleanupStaleUpdateSites(); // Trigger heartbeat registration $this->sendHeartbeat(); } /** * Remove legacy/stale extension entries and filesystem remnants. * * The old standalone plugin was named "mokowaasbrand" (plg_system_mokowaasbrand). * After the rewrite into the pkg_mokowaas package, the old entries and files * may linger — especially on sites restored from old backups. * * @return void * * @since 02.21.00 */ private function cleanupLegacyExtensions(): void { try { $db = Factory::getDbo(); // Legacy element names to remove from #__extensions $legacy = [ $db->quote('mokowaasbrand'), $db->quote('plg_system_mokowaasbrand'), ]; // Delete from #__extensions $query = $db->getQuery(true) ->delete($db->quoteName('#__extensions')) ->where($db->quoteName('element') . ' IN (' . implode(',', $legacy) . ')'); $db->setQuery($query); $affected = $db->execute(); $count = $db->getAffectedRows(); // Remove legacy plugin files from the filesystem $legacyDirs = [ JPATH_PLUGINS . '/system/mokowaasbrand', ]; foreach ($legacyDirs as $dir) { if (is_dir($dir)) { $this->rmdirRecursive($dir); } } if ($count > 0) { Factory::getApplication()->enqueueMessage( sprintf('Removed %d legacy MokoWaaS extension(s).', $count), 'message' ); Log::add( sprintf('Cleaned up %d legacy MokoWaaS extension entries', $count), Log::INFO, 'mokowaas' ); } } catch (\Throwable $e) { Log::add('Legacy cleanup error: ' . $e->getMessage(), Log::WARNING, 'jerror'); } } /** * Recursively remove a directory. * * @param string $dir Directory path * * @return void * * @since 02.21.00 */ private function rmdirRecursive(string $dir): void { $items = new \RecursiveIteratorIterator( new \RecursiveDirectoryIterator($dir, \RecursiveDirectoryIterator::SKIP_DOTS), \RecursiveIteratorIterator::CHILD_FIRST ); foreach ($items as $item) { if ($item->isDir()) { @rmdir($item->getPathname()); } else { @unlink($item->getPathname()); } } @rmdir($dir); } /** * Enable a plugin by group and element. * * @param string $group Plugin group * @param string $element Plugin element name * * @return void * * @since 2.2.0 */ private function enablePlugin(string $group, string $element): void { try { $db = Factory::getDbo(); $query = $db->getQuery(true) ->update($db->quoteName('#__extensions')) ->set($db->quoteName('enabled') . ' = 1') ->where($db->quoteName('type') . ' = ' . $db->quote('plugin')) ->where($db->quoteName('folder') . ' = ' . $db->quote($group)) ->where($db->quoteName('element') . ' = ' . $db->quote($element)); $db->setQuery($query); $db->execute(); } catch (\Throwable $e) { Log::add('Error enabling plugin ' . $group . '/' . $element . ': ' . $e->getMessage(), Log::WARNING, 'jerror'); } } /** * Set the protected flag on all MokoWaaS extensions. * * Joomla's protected flag prevents disabling and uninstalling at the * framework level — no plugin-side interception needed. * * @return void * * @since 02.03.10 */ private function protectExtensions(): void { try { $db = Factory::getDbo(); // All MokoWaaS elements: package, system plugin, component, // webservices plugins, task plugin $elements = [ $db->quote('pkg_mokowaas'), $db->quote('mokowaas'), $db->quote('com_mokowaas'), $db->quote('mokowaasdemo'), $db->quote('mokowaassync'), $db->quote('perfectpublisher'), ]; $query = $db->getQuery(true) ->update($db->quoteName('#__extensions')) ->set($db->quoteName('protected') . ' = 1') ->set($db->quoteName('locked') . ' = 0') ->where($db->quoteName('element') . ' IN (' . implode(',', $elements) . ')'); $db->setQuery($query); $db->execute(); // Ensure update server stays enabled $this->enableUpdateServer(); } catch (\Throwable $e) { Log::add('Error protecting MokoWaaS extensions: ' . $e->getMessage(), Log::WARNING, 'jerror'); } } /** * Remove stale and duplicate MokoWaaS update site entries. * * Keeps only the package-level update site pointing to the dynamic * MokoGitea endpoint. Removes plugin-level entries, old static URLs, * and orphaned #__updates rows tied to deleted update sites. * * @return void * * @since 02.31.00 */ private function cleanupStaleUpdateSites(): void { try { $db = Factory::getDbo(); $dynamicUrl = 'https://git.mokoconsulting.tech/MokoConsulting/MokoWaaS/updates.xml'; // Find all MokoWaaS update sites $query = $db->getQuery(true) ->select($db->quoteName(['update_site_id', 'location'])) ->from($db->quoteName('#__update_sites')) ->where('(' . $db->quoteName('name') . ' LIKE ' . $db->quote('%MokoWaaS%') . ' OR ' . $db->quoteName('location') . ' LIKE ' . $db->quote('%MokoWaaS%') . ')'); $db->setQuery($query); $sites = $db->loadObjectList(); $keepId = null; $removeIds = []; foreach ($sites as $site) { if ($site->location === $dynamicUrl && $keepId === null) { $keepId = (int) $site->update_site_id; } else { $removeIds[] = (int) $site->update_site_id; } } if (empty($removeIds)) { return; } $idList = implode(',', $removeIds); // Remove orphaned #__updates rows $db->setQuery( $db->getQuery(true) ->delete($db->quoteName('#__updates')) ->where($db->quoteName('update_site_id') . ' IN (' . $idList . ')') )->execute(); // Remove link rows $db->setQuery( $db->getQuery(true) ->delete($db->quoteName('#__update_sites_extensions')) ->where($db->quoteName('update_site_id') . ' IN (' . $idList . ')') )->execute(); // Remove stale update sites $db->setQuery( $db->getQuery(true) ->delete($db->quoteName('#__update_sites')) ->where($db->quoteName('update_site_id') . ' IN (' . $idList . ')') )->execute(); $count = count($removeIds); if ($count > 0) { Factory::getApplication()->enqueueMessage( sprintf('Cleaned up %d stale MokoWaaS update site(s).', $count), 'message' ); } } catch (\Throwable $e) { Log::add('Error cleaning up stale update sites: ' . $e->getMessage(), Log::WARNING, 'jerror'); } } /** * Ensure the MokoWaaS update server entry stays enabled and points * to the correct dynamic endpoint with the license key attached. * * Migrates legacy static URLs (raw/branch/main/updates.xml) to the * dynamic MokoGitea update feed, and syncs the license key from * plugin params into extra_query so Joomla sends it as dlid. * * @return void * * @since 02.21.00 */ private function enableUpdateServer(): void { try { $db = Factory::getDbo(); // Migrate legacy static URL to dynamic MokoGitea endpoint $db->setQuery( $db->getQuery(true) ->update($db->quoteName('#__update_sites')) ->set($db->quoteName('location') . ' = ' . $db->quote('https://git.mokoconsulting.tech/MokoConsulting/MokoWaaS/updates.xml')) ->where($db->quoteName('location') . ' LIKE ' . $db->quote('%MokoWaaS/raw/branch/%updates.xml%')) ); $db->execute(); // Enable all MokoWaaS update sites $query = $db->getQuery(true) ->update($db->quoteName('#__update_sites')) ->set($db->quoteName('enabled') . ' = 1') ->where('(' . $db->quoteName('name') . ' LIKE ' . $db->quote('%MokoWaaS%') . ' OR ' . $db->quoteName('location') . ' LIKE ' . $db->quote('%MokoWaaS%') . ')'); $db->setQuery($query); $db->execute(); } catch (\Throwable $e) { Log::add('Error enabling update server: ' . $e->getMessage(), Log::WARNING, 'jerror'); } } /** * Send heartbeat to the MokoWaaS monitoring receiver. * * @return void * * @since 02.03.08 */ private function sendHeartbeat(): void { try { $db = Factory::getDbo(); $query = $db->getQuery(true) ->select($db->quoteName('params')) ->from($db->quoteName('#__extensions')) ->where($db->quoteName('element') . ' = ' . $db->quote('mokowaas')) ->where($db->quoteName('type') . ' = ' . $db->quote('plugin')) ->where($db->quoteName('folder') . ' = ' . $db->quote('system')); $params = json_decode((string) $db->setQuery($query)->loadResult()); $healthToken = $params->health_api_token ?? ''; if (empty($healthToken)) { return; } $siteUrl = rtrim(\Joomla\CMS\Uri\Uri::root(), '/'); $siteName = Factory::getConfig()->get('sitename', 'Joomla'); $payload = json_encode([ 'site_url' => $siteUrl, 'site_name' => $siteName, 'health_token' => $healthToken, 'action' => 'register', ], JSON_UNESCAPED_SLASHES); $ch = curl_init('https://bench.mokoconsulting.tech/api/waas-heartbeat/register'); curl_setopt($ch, CURLOPT_POST, true); curl_setopt($ch, CURLOPT_HTTPHEADER, [ 'Content-Type: application/json', 'X-MokoWaaS-Key: moko-waas-hb-2026-x9k4m', ]); curl_setopt($ch, CURLOPT_POSTFIELDS, $payload); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); curl_setopt($ch, CURLOPT_TIMEOUT, 15); curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); $response = curl_exec($ch); $code = (int) curl_getinfo($ch, CURLINFO_HTTP_CODE); curl_close($ch); if ($code >= 200 && $code < 300) { Factory::getApplication()->enqueueMessage('Grafana heartbeat: site registered', 'message'); } } catch (\Throwable $e) { // Silent failure — heartbeat is non-critical } } }