. */ /* * CLI script for database migration from μlogger version 0.6 to version 1.x. * Database user defined in config file must have privileges to create and modify tables. * * Backup your database before running this script. */ if (PHP_SAPI !== "cli") { die("This script must be called from CLI console"); } if (!defined("ROOT_DIR")) { define("ROOT_DIR", dirname(__DIR__)); } include_once(ROOT_DIR . "/helpers/config.php"); include_once(ROOT_DIR . "/helpers/db.php"); if (!defined("SKIP_RUN")) { verifyVersion() || exit(1); waitForUser() || exit(0); updateSchemas() || exit(1); updateConfig() || exit(1); exit(0); } /** * Check μlogger version is valid for migration * @return bool True if valid version */ function verifyVersion() { if (!class_exists("uConfig") || !class_exists("uDb") || !method_exists("uConfig", "getOfflineInstance") || strpos(uConfig::getOfflineInstance()->version, '1.') !== 0) { echo "You need μlogger version 1.x to run this script" . PHP_EOL; return false; } return true; } /** * Waits for user confirmation * @param string $input Optional path to read from (for tests) * @return bool True if confirmed */ function waitForUser($input = "php://stdin") { echo "This script will update database from version 0.6 to 1.x." . PHP_EOL; echo "Creating database backup is recommended before proceeding" . PHP_EOL; echo "Type 'yes' to continue, anything else to abort" . PHP_EOL; $handle = fopen($input, 'rb'); $input = fgets($handle); fclose($handle); if (trim($input) !== "yes") { echo "Cancelled by user" . PHP_EOL; return false; } echo PHP_EOL; echo "Starting migration..." . PHP_EOL; return true; } /** * Updates database schemas * @return bool True on success */ function updateSchemas() { echo "Migrating database schemas..." . PHP_EOL; try { $queries = getQueries(); uDb::getInstance()->beginTransaction(); foreach ($queries as $query) { uDb::getInstance()->exec($query); } // make sure the transaction wasn't autocommitted if (uDb::getInstance()->inTransaction()) { uDb::getInstance()->commit(); } } catch (PDOException $e) { echo "Database query failed: {$e->getMessage()}" . PHP_EOL; if (uDb::getInstance()->inTransaction()) { echo "Reverting changes..." . PHP_EOL; uDb::getInstance()->rollBack(); } return false; } echo PHP_EOL; echo "Database schemas updated successfully" . PHP_EOL; return true; } /** * Update database settings based on current config file * @param string $path Optional path of config (for tests) * @return bool True on success */ /** @noinspection IssetArgumentExistenceInspection, PhpIncludeInspection */ function updateConfig($path = ROOT_DIR . "/config.php") { echo "Migrating config to database..." . PHP_EOL; require_once($path); if (isset($admin_user) && !empty($admin_user)) { try { echo "Setting user $admin_user as admin" . PHP_EOL; $query = "UPDATE " . uDb::getInstance()->table('users') . " SET admin = ? WHERE login = ?"; $stmt = uDb::getInstance()->prepare($query); $stmt->execute([ 1, $admin_user ]); if ($stmt->rowCount() === 0) { echo "User $admin_user not found in database table" . PHP_EOL; echo "Set your admin user manually in users table" . PHP_EOL; } } catch (PDOException $e) { echo "Setting admin user failed: " . $e->getMessage() . PHP_EOL; echo "Set your admin user manually in users table" . PHP_EOL; } } $config = uConfig::getOfflineInstance(); if (isset($mapapi) && !empty($mapapi)) { $config->mapApi = $mapapi; } if (isset($ol_layers) && is_array($ol_layers)) { $id = 1; foreach ($ol_layers as $name => $url) { $config->olLayers[] = new uLayer($id++, $name, $url, 0); } } if (isset($init_latitude) && is_numeric($init_latitude)) { $config->initLatitude = $init_latitude; } if (isset($init_longitude) && is_numeric($init_longitude)) { $config->initLongitude = $init_longitude; } if (isset($gkey) && !empty($gkey)) { $config->googleKey = $gkey; } if (isset($require_authentication) && is_numeric($require_authentication)) { $config->requireAuthentication = (bool) $require_authentication; } if (isset($public_tracks) && is_numeric($public_tracks)) { $config->publicTracks = (bool) $public_tracks; } if (isset($pass_lenmin) && is_numeric($pass_lenmin)) { $config->passLenMin = (int) $pass_lenmin; } if (isset($pass_strength) && is_numeric($pass_strength)) { $config->passStrength = (int) $pass_strength; } if (isset($interval) && is_numeric($interval)) { $config->interval = (int) $interval; } if (isset($lang) && !empty($lang)) { $config->lang = $lang; } if (isset($units) && !empty($units)) { $config->units = $units; } if (isset($strokeWeight) && is_numeric($strokeWeight)) { $config->strokeWeight = (int) $strokeWeight; } if (isset($strokeColor) && !empty($strokeColor)) { $config->strokeColor = $strokeColor; } if (isset($strokeOpacity) && is_numeric($strokeOpacity)) { $config->strokeOpacity = $strokeOpacity; } if ($config->save() !== true) { echo "Configuration migration failed. Please update your settings manually from web interface" . PHP_EOL; return false; } echo "Configuration successfully migrated to database" . PHP_EOL; return true; } /** * Get queries array for current db driver * @return array Queries */ function getQueries() { $dbDriver = uDb::getInstance()->getAttribute(PDO::ATTR_DRIVER_NAME); $tConfig = uDb::getInstance()->table('config'); $tLayers = uDb::getInstance()->table('ol_layers'); $tUsers = uDb::getInstance()->table('users'); $tTracks = uDb::getInstance()->table('tracks'); $tPositions = uDb::getInstance()->table('positions'); $tPositionsBackup = "{$tPositions}_backup"; $queries = []; switch ($dbDriver) { case "mysql": $queries[] = "CREATE TABLE `$tConfig` ( `name` varchar(20) PRIMARY KEY, `value` tinyblob NOT NULL ) ENGINE=InnoDB DEFAULT CHARSET=utf8"; $queries[] = "INSERT INTO `$tConfig` (`name`, `value`) VALUES ('color_extra', 's:7:\"#cccccc\";'), ('color_hilite', 's:7:\"#feff6a\";'), ('color_normal', 's:7:\"#ffffff\";'), ('color_start', 's:7:\"#55b500\";'), ('color_stop', 's:7:\"#ff6a00\";'), ('google_key', 's:0:\"\";'), ('interval_seconds', 'i:10;'), ('lang', 's:2:\"en\";'), ('latitude', 'd:52.229999999999997;'), ('longitude', 'd:21.010000000000002;'), ('map_api', 's:10:\"openlayers\";'), ('pass_lenmin', 'i:10;'), ('pass_strength', 'i:2;'), ('public_tracks', 'b:1;'), ('require_auth', 'b:1;'), ('stroke_color', 's:7:\"#ff0000\";'), ('stroke_opacity', 'd:1;'), ('stroke_weight', 'i:2;'), ('units', 's:6:\"metric\";'), ('upload_maxsize', 'i:0;')"; $queries[] = "CREATE TABLE `$tLayers` ( `id` int(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, `name` varchar(50) NOT NULL, `url` varchar(255) NOT NULL, `priority` int(11) NOT NULL DEFAULT '0' ) ENGINE=InnoDB DEFAULT CHARSET=utf8"; $queries[] = "INSERT INTO `$tLayers` (`id`, `name`, `url`, `priority`) VALUES (1, 'OpenCycleMap', 'https://{a-c}.tile.thunderforest.com/cycle/{z}/{x}/{y}.png', 0), (2, 'OpenTopoMap', 'https://{a-c}.tile.opentopomap.org/{z}/{x}/{y}.png', 0), (3, 'OpenSeaMap', 'https://tiles.openseamap.org/seamark/{z}/{x}/{y}.png', 0), (4, 'ESRI', 'https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}', 0), (5, 'UMP', 'http://{1-3}.tiles.ump.waw.pl/ump_tiles/{z}/{x}/{y}.png', 0), (6, 'Osmapa.pl', 'http://{a-c}.tile.openstreetmap.pl/osmapa.pl/{z}/{x}/{y}.png', 0); "; $queries[] = "ALTER TABLE `$tUsers` ADD `admin` BOOLEAN NOT NULL DEFAULT FALSE AFTER `password`"; $queries[] = "ALTER TABLE `$tPositions` CHANGE `image_id` `image` VARCHAR(100) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL"; break; case "pgsql": $queries[] = "CREATE TABLE $tConfig ( name varchar(20) PRIMARY KEY, value bytea NOT NULL )"; $queries[] = "INSERT INTO $tConfig (name, value) VALUES ('color_extra', 's:7:\"#cccccc\";'), ('color_hilite', 's:7:\"#feff6a\";'), ('color_normal', 's:7:\"#ffffff\";'), ('color_start', 's:7:\"#55b500\";'), ('color_stop', 's:7:\"#ff6a00\";'), ('google_key', 's:0:\"\";'), ('interval_seconds', 'i:10;'), ('lang', 's:2:\"en\";'), ('latitude', 'd:52.229999999999997;'), ('longitude', 'd:21.010000000000002;'), ('map_api', 's:10:\"openlayers\";'), ('pass_lenmin', 'i:10;'), ('pass_strength', 'i:2;'), ('public_tracks', 'b:1;'), ('require_auth', 'b:1;'), ('stroke_color', 's:7:\"#ff0000\";'), ('stroke_opacity', 'd:1;'), ('stroke_weight', 'i:2;'), ('units', 's:6:\"metric\";'), ('upload_maxsize', 'i:0;')"; $queries[] = "CREATE TABLE $tLayers ( id serial PRIMARY KEY, name varchar(50) NOT NULL, url varchar(255) NOT NULL, priority int NOT NULL DEFAULT '0' )"; $queries[] = "INSERT INTO $tLayers (id, name, url, priority) VALUES (1, 'OpenCycleMap', 'https://{a-c}.tile.thunderforest.com/cycle/{z}/{x}/{y}.png', 0), (2, 'OpenTopoMap', 'https://{a-c}.tile.opentopomap.org/{z}/{x}/{y}.png', 0), (3, 'OpenSeaMap', 'https://tiles.openseamap.org/seamark/{z}/{x}/{y}.png', 0), (4, 'ESRI', 'https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}', 0), (5, 'UMP', 'http://{1-3}.tiles.ump.waw.pl/ump_tiles/{z}/{x}/{y}.png', 0), (6, 'Osmapa.pl', 'http://{a-c}.tile.openstreetmap.pl/osmapa.pl/{z}/{x}/{y}.png', 0); "; $queries[] = "ALTER TABLE $tUsers ADD COLUMN admin boolean NOT NULL DEFAULT FALSE"; $queries[] = "ALTER TABLE $tPositions DROP COLUMN image_id"; $queries[] = "ALTER TABLE $tPositions ADD COLUMN image varchar(100) NULL DEFAULT NULL"; break; case "sqlite": $queries[] = "CREATE TABLE `$tConfig` ( `name` varchar(20) PRIMARY KEY, `value` tinyblob NOT NULL )"; $queries[] = "INSERT INTO `$tConfig` (`name`, `value`) VALUES ('color_extra', 's:7:\"#cccccc\";'), ('color_hilite', 's:7:\"#feff6a\";'), ('color_normal', 's:7:\"#ffffff\";'), ('color_start', 's:7:\"#55b500\";'), ('color_stop', 's:7:\"#ff6a00\";'), ('google_key', 's:0:\"\";'), ('interval_seconds', 'i:10;'), ('lang', 's:2:\"en\";'), ('latitude', 'd:52.229999999999997;'), ('longitude', 'd:21.010000000000002;'), ('map_api', 's:10:\"openlayers\";'), ('pass_lenmin', 'i:10;'), ('pass_strength', 'i:2;'), ('public_tracks', 'b:1;'), ('require_auth', 'b:1;'), ('stroke_color', 's:7:\"#ff0000\";'), ('stroke_opacity', 'd:1;'), ('stroke_weight', 'i:2;'), ('units', 's:6:\"metric\";'), ('upload_maxsize', 'i:0;')"; $queries[] = "CREATE TABLE `$tLayers` ( `id` integer PRIMARY KEY AUTOINCREMENT, `name` varchar(50) NOT NULL, `url` varchar(255) NOT NULL, `priority` integer NOT NULL DEFAULT '0' )"; $queries[] = "INSERT INTO `$tLayers` (`id`, `name`, `url`, `priority`) VALUES (1, 'OpenCycleMap', 'https://{a-c}.tile.thunderforest.com/cycle/{z}/{x}/{y}.png', 0), (2, 'OpenTopoMap', 'https://{a-c}.tile.opentopomap.org/{z}/{x}/{y}.png', 0), (3, 'OpenSeaMap', 'https://tiles.openseamap.org/seamark/{z}/{x}/{y}.png', 0), (4, 'ESRI', 'https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}', 0), (5, 'UMP', 'http://{1-3}.tiles.ump.waw.pl/ump_tiles/{z}/{x}/{y}.png', 0), (6, 'Osmapa.pl', 'http://{a-c}.tile.openstreetmap.pl/osmapa.pl/{z}/{x}/{y}.png', 0); "; $queries[] = "ALTER TABLE `$tUsers` ADD `admin` boolean NOT NULL DEFAULT FALSE"; $queries[] = "PRAGMA foreign_keys=OFF"; $queries[] = "CREATE TABLE `$tPositionsBackup` ( `id` integer PRIMARY KEY AUTOINCREMENT, `time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, `user_id` integer NOT NULL, `track_id` integer NOT NULL, `latitude` double NOT NULL, `longitude` double NOT NULL, `altitude` double DEFAULT NULL, `speed` double DEFAULT NULL, `bearing` double DEFAULT NULL, `accuracy` integer DEFAULT NULL, `provider` varchar(100) DEFAULT NULL, `comment` varchar(255) DEFAULT NULL, `image` varchar(100) DEFAULT NULL, FOREIGN KEY(`user_id`) REFERENCES `$tUsers`(`id`), FOREIGN KEY(`track_id`) REFERENCES `$tTracks`(`id`) )"; $queries[] = "INSERT INTO `$tPositionsBackup` SELECT id, time, user_id, track_id, latitude, longitude, altitude, speed, bearing, accuracy, provider, comment, NULL FROM positions"; $queries[] = "DROP TABLE `$tPositions`"; $queries[] = "ALTER TABLE `$tPositionsBackup` RENAME TO `$tPositions`"; $queries[] = "CREATE INDEX `idx_ptrack_id` ON `$tPositions`(`track_id`)"; $queries[] = "CREATE INDEX `idx_puser_id` ON `$tPositions`(`user_id`)"; $queries[] = "PRAGMA foreign_keys=ON"; break; default: throw new InvalidArgumentException("Driver not supported"); } return $queries; } ?>