ulogger-server/helpers/config.php

426 lines
12 KiB
PHP
Raw Normal View History

2017-04-06 19:18:51 +02:00
<?php
2017-04-07 00:05:28 +02:00
/* μlogger
*
* Copyright(C) 2017 Bartek Fabiszewski (www.fabiszewski.net)
*
* This is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, see <http://www.gnu.org/licenses/>.
*/
2017-04-06 19:18:51 +02:00
2020-02-19 18:42:44 +01:00
require_once(ROOT_DIR . "/helpers/db.php");
require_once(ROOT_DIR . "/helpers/layer.php");
2020-02-19 18:42:44 +01:00
/**
* Handles config values
*/
class uConfig {
2020-02-20 17:08:47 +01:00
/**
* Singleton instance
*
* @var uConfig Object instance
*/
private static $instance;
2020-02-19 18:42:44 +01:00
/**
* @var string Version number
*/
2020-02-20 17:08:47 +01:00
public $version = "1.0-beta";
2020-02-19 18:42:44 +01:00
/**
* @var string Default map drawing framework
*/
2020-02-20 17:08:47 +01:00
public $mapApi = "openlayers";
2020-02-19 18:42:44 +01:00
/**
* @var string|null Google maps key
*/
2020-02-20 17:08:47 +01:00
public $googleKey;
2020-02-19 18:42:44 +01:00
/**
* @var uLayer[] Openlayers extra map layers
*/
2020-02-20 17:08:47 +01:00
public $olLayers = [];
2020-02-19 18:42:44 +01:00
/**
* @var float Default latitude for initial map
*/
2020-02-20 17:08:47 +01:00
public $initLatitude = 52.23;
2020-02-19 18:42:44 +01:00
/**
* @var float Default longitude for initial map
*/
2020-02-20 17:08:47 +01:00
public $initLongitude = 21.01;
2020-02-19 18:42:44 +01:00
/**
* @var bool Require login/password authentication
*/
2020-02-20 17:08:47 +01:00
public $requireAuthentication = true;
2020-02-19 18:42:44 +01:00
/**
* @var bool All users tracks are visible to authenticated user
*/
2020-02-20 17:08:47 +01:00
public $publicTracks = false;
2020-02-19 18:42:44 +01:00
/**
* @var int Miniumum required length of user password
*/
2020-02-20 17:08:47 +01:00
public $passLenMin = 10;
2020-02-19 18:42:44 +01:00
/**
* @var int Required strength of user password
* 0 = no requirements,
* 1 = require mixed case letters (lower and upper),
* 2 = require mixed case and numbers
* 3 = require mixed case, numbers and non-alphanumeric characters
*/
2020-02-20 17:08:47 +01:00
public $passStrength = 2;
2020-02-19 18:42:44 +01:00
/**
* @var int Default interval in seconds for live auto reload
*/
2020-02-20 17:08:47 +01:00
public $interval = 10;
2020-02-19 18:42:44 +01:00
/**
* @var string Default language code
*/
2020-02-20 17:08:47 +01:00
public $lang = "en";
2020-02-19 18:42:44 +01:00
/**
* @var string Default units
*/
2020-02-20 17:08:47 +01:00
public $units = "metric";
2020-02-19 18:42:44 +01:00
/**
* @var int Stroke weight
*/
2020-02-20 17:08:47 +01:00
public $strokeWeight = 2;
2020-02-19 18:42:44 +01:00
/**
* @var string Stroke color
*/
2020-02-23 22:21:17 +01:00
public $strokeColor = "#ff0000";
2020-02-19 18:42:44 +01:00
/**
2020-02-20 17:08:47 +01:00
* @var float Stroke opacity
2020-02-19 18:42:44 +01:00
*/
2020-02-20 17:08:47 +01:00
public $strokeOpacity = 1.0;
2020-02-23 22:21:17 +01:00
/**
* @var string Stroke color
*/
public $colorNormal = "#ffffff";
/**
* @var string Stroke color
*/
public $colorStart = "#55b500";
/**
* @var string Stroke color
*/
public $colorStop = "#ff6a00";
/**
* @var string Stroke color
*/
public $colorExtra = "#cccccc";
/**
* @var string Stroke color
*/
public $colorHilite = "#feff6a";
2020-02-20 17:08:47 +01:00
public function __construct($useDatabase = true) {
if ($useDatabase) {
$this->setFromDatabase();
}
$this->setFromCookies();
}
2020-02-19 18:42:44 +01:00
2020-02-20 17:08:47 +01:00
/**
* Returns singleton instance
*
* @return uConfig Singleton instance
*/
public static function getInstance() {
if (!self::$instance) {
self::$instance = new self();
}
return self::$instance;
}
2020-02-19 18:42:44 +01:00
/**
2020-02-20 17:08:47 +01:00
* Returns singleton instance
*
* @return uConfig Singleton instance
2020-02-19 18:42:44 +01:00
*/
2020-02-20 17:08:47 +01:00
public static function getOfflineInstance() {
if (!self::$instance) {
self::$instance = new self(false);
2017-04-09 23:35:55 +02:00
}
2020-02-20 17:08:47 +01:00
return self::$instance;
2020-02-19 18:42:44 +01:00
}
/**
* Get db instance
*
* @return uDb instance
*/
private static function db() {
return uDb::getInstance();
}
2017-04-09 23:35:55 +02:00
2020-02-19 18:42:44 +01:00
/**
* Read config values from database
*/
2020-02-20 17:08:47 +01:00
public function setFromDatabase() {
2020-02-19 18:42:44 +01:00
try {
2020-02-23 22:21:17 +01:00
$query = "SELECT name, value FROM " . self::db()->table('config');
2020-02-19 18:42:44 +01:00
$result = self::db()->query($query);
2020-02-23 22:21:17 +01:00
$arr = $result->fetchAll(PDO::FETCH_KEY_PAIR);
$this->setFromArray(array_map([ $this, 'unserialize' ], $arr));
2020-02-20 17:08:47 +01:00
$this->setLayersFromDatabase();
if (!$this->requireAuthentication) {
2017-04-09 23:35:55 +02:00
// tracks must be public if we don't require authentication
2020-02-20 17:08:47 +01:00
$this->publicTracks = true;
2017-04-09 23:35:55 +02:00
}
2020-02-19 18:42:44 +01:00
} catch (PDOException $e) {
// TODO: handle exception
syslog(LOG_ERR, $e->getMessage());
2017-04-09 23:35:55 +02:00
}
2020-02-19 18:42:44 +01:00
}
2017-04-09 23:35:55 +02:00
2020-02-23 22:21:17 +01:00
/**
* Unserialize data from database
* @param string|resource $data Resource returned by pgsql, string otherwise
* @return mixed
*/
private function unserialize($data) {
if (is_resource($data)) {
return unserialize(stream_get_contents($data));
}
return unserialize($data);
}
2020-02-19 18:42:44 +01:00
/**
2020-02-20 17:08:47 +01:00
* Save config values to database
* @return bool True on success, false otherwise
2020-02-19 18:42:44 +01:00
*/
2020-02-20 17:08:47 +01:00
public function save() {
$ret = false;
try {
2020-02-23 22:21:17 +01:00
// PDO::PARAM_LOB doesn't work here with pgsql, why?
$placeholder = self::db()->lobPlaceholder();
2020-02-20 17:08:47 +01:00
$query = "UPDATE " . self::db()->table('config') . "
2020-02-23 22:21:17 +01:00
SET value = CASE name
WHEN 'map_api' THEN $placeholder
WHEN 'latitude' THEN $placeholder
WHEN 'longitude' THEN $placeholder
WHEN 'google_key' THEN $placeholder
WHEN 'require_auth' THEN $placeholder
WHEN 'public_tracks' THEN $placeholder
WHEN 'pass_lenmin' THEN $placeholder
WHEN 'pass_strength' THEN $placeholder
WHEN 'interval_seconds' THEN $placeholder
WHEN 'lang' THEN $placeholder
WHEN 'units' THEN $placeholder
WHEN 'stroke_weight' THEN $placeholder
WHEN 'stroke_color' THEN $placeholder
WHEN 'stroke_opacity' THEN $placeholder
WHEN 'color_normal' THEN $placeholder
WHEN 'color_start' THEN $placeholder
WHEN 'color_stop' THEN $placeholder
WHEN 'color_extra' THEN $placeholder
WHEN 'color_hilite' THEN $placeholder
END";
2020-02-20 17:08:47 +01:00
$stmt = self::db()->prepare($query);
$params = [
$this->mapApi,
$this->initLatitude,
$this->initLongitude,
$this->googleKey,
2020-02-23 22:21:17 +01:00
$this->requireAuthentication,
$this->publicTracks,
2020-02-20 17:08:47 +01:00
$this->passLenMin,
$this->passStrength,
$this->interval,
$this->lang,
$this->units,
$this->strokeWeight,
2020-02-23 22:21:17 +01:00
$this->strokeColor,
$this->strokeOpacity,
$this->colorNormal,
$this->colorStart,
$this->colorStop,
$this->colorExtra,
$this->colorHilite
2020-02-20 17:08:47 +01:00
];
2020-02-23 22:21:17 +01:00
$stmt->execute(array_map('serialize', $params));
2020-02-20 17:08:47 +01:00
$this->saveLayers();
$ret = true;
} catch (PDOException $e) {
// TODO: handle exception
syslog(LOG_ERR, $e->getMessage());
2017-04-09 23:35:55 +02:00
}
2020-02-20 17:08:47 +01:00
return $ret;
2020-02-19 18:42:44 +01:00
}
2020-02-19 18:42:44 +01:00
/**
2020-02-20 17:08:47 +01:00
* Truncate ol_layers table
* @throws PDOException
*/
private function deleteLayers() {
$query = "DELETE FROM " . self::db()->table('ol_layers');
self::db()->exec($query);
}
/**
* Save layers to database
* @throws PDOException
2020-02-19 18:42:44 +01:00
*/
2020-02-20 17:08:47 +01:00
private function saveLayers() {
$this->deleteLayers();
if (!empty($this->olLayers)) {
$query = "INSERT INTO " . self::db()->table('ol_layers') . " (id, name, url, priority) VALUES (?, ?, ?, ?)";
$stmt = self::db()->prepare($query);
foreach ($this->olLayers as $layer) {
$stmt->execute([ $layer->id, $layer->name, $layer->url, $layer->priority]);
}
}
}
2017-04-17 13:15:44 +02:00
2020-02-20 17:08:47 +01:00
/**
* Read config values from database
* @throws PDOException
*/
private function setLayersFromDatabase() {
$this->olLayers = [];
$query = "SELECT id, name, url, priority FROM " . self::db()->table('ol_layers');
$result = self::db()->query($query);
while ($row = $result->fetch()) {
$this->olLayers[] = new uLayer($row['id'], $row['name'], $row['url'], $row['priority']);
}
2020-02-19 18:42:44 +01:00
}
/**
* Read config values stored in cookies
*/
2020-02-20 17:08:47 +01:00
private function setFromCookies() {
if (isset($_COOKIE["ulogger_api"])) { $this->mapApi = $_COOKIE["ulogger_api"]; }
if (isset($_COOKIE["ulogger_lang"])) { $this->lang = $_COOKIE["ulogger_lang"]; }
if (isset($_COOKIE["ulogger_units"])) { $this->units = $_COOKIE["ulogger_units"]; }
if (isset($_COOKIE["ulogger_interval"])) { $this->interval = $_COOKIE["ulogger_interval"]; }
2020-02-19 18:42:44 +01:00
}
2020-02-20 17:08:47 +01:00
2020-02-19 18:42:44 +01:00
/**
2020-02-20 17:08:47 +01:00
* Check if given password matches user's one
2020-02-19 18:42:44 +01:00
*
2020-02-20 17:08:47 +01:00
* @param String $password Password
* @return bool True if matches, false otherwise
2020-02-19 18:42:44 +01:00
*/
2020-02-20 17:08:47 +01:00
public function validPassStrength($password) {
return preg_match($this->passRegex(), $password);
2020-02-19 18:42:44 +01:00
}
/**
* Regex to test if password matches strength and length requirements.
* Valid for both php and javascript
* @return string
*/
2020-02-20 17:08:47 +01:00
public function passRegex() {
2020-02-19 18:42:44 +01:00
$regex = "";
2020-02-20 17:08:47 +01:00
if ($this->passStrength > 0) {
2020-02-19 18:42:44 +01:00
// lower and upper case
$regex .= "(?=.*[a-z])(?=.*[A-Z])";
}
2020-02-20 17:08:47 +01:00
if ($this->passStrength > 1) {
2020-02-19 18:42:44 +01:00
// digits
$regex .= "(?=.*[0-9])";
}
2020-02-20 17:08:47 +01:00
if ($this->passStrength > 2) {
2020-02-19 18:42:44 +01:00
// not latin, not digits
$regex .= "(?=.*[^a-zA-Z0-9])";
}
2020-02-20 17:08:47 +01:00
if ($this->passLenMin > 0) {
$regex .= "(?=.{" . $this->passLenMin . ",})";
}
2020-02-19 18:42:44 +01:00
if (empty($regex)) {
$regex = ".*";
}
return "/" . $regex . "/";
}
/**
2020-02-23 22:21:17 +01:00
* Set config values from array
* @param array $arr
2020-02-19 18:42:44 +01:00
*/
2020-02-23 22:21:17 +01:00
public function setFromArray($arr) {
if (!is_array($arr)) {
return;
}
2020-05-21 12:32:16 +02:00
if (isset($arr['map_api']) && !empty($arr['map_api'])) {
2020-02-23 22:21:17 +01:00
$this->mapApi = $arr['map_api'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['latitude']) && is_numeric($arr['latitude'])) {
2020-02-23 22:21:17 +01:00
$this->initLatitude = (float) $arr['latitude'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['longitude']) && is_numeric($arr['longitude'])) {
2020-02-23 22:21:17 +01:00
$this->initLongitude = (float) $arr['longitude'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['google_key']) && !is_null($arr['google_key'])) {
2020-02-23 22:21:17 +01:00
$this->googleKey = $arr['google_key'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['require_auth']) && (is_numeric($arr['require_auth']) || is_bool($arr['require_auth']))) {
2020-02-23 22:21:17 +01:00
$this->requireAuthentication = (bool) $arr['require_auth'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['public_tracks']) && (is_numeric($arr['public_tracks']) || is_bool($arr['public_tracks']))) {
2020-02-23 22:21:17 +01:00
$this->publicTracks = (bool) $arr['public_tracks'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['pass_lenmin']) && is_numeric($arr['pass_lenmin'])) {
2020-02-23 22:21:17 +01:00
$this->passLenMin = (int) $arr['pass_lenmin'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['pass_strength']) && is_numeric($arr['pass_strength'])) {
2020-02-23 22:21:17 +01:00
$this->passStrength = (int) $arr['pass_strength'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['interval_seconds']) && is_numeric($arr['interval_seconds'])) {
2020-02-23 22:21:17 +01:00
$this->interval = (int) $arr['interval_seconds'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['lang']) && !empty($arr['lang'])) {
2020-02-23 22:21:17 +01:00
$this->lang = $arr['lang'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['units']) && !empty($arr['units'])) {
2020-02-23 22:21:17 +01:00
$this->units = $arr['units'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['stroke_weight']) && is_numeric($arr['stroke_weight'])) {
2020-02-23 22:21:17 +01:00
$this->strokeWeight = (int) $arr['stroke_weight'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['stroke_color']) && !empty($arr['stroke_color'])) {
2020-02-23 22:21:17 +01:00
$this->strokeColor = $arr['stroke_color'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['stroke_opacity']) && is_numeric($arr['stroke_opacity'])) {
2020-02-23 22:21:17 +01:00
$this->strokeOpacity = (float) $arr['stroke_opacity'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['color_normal']) && !empty($arr['color_normal'])) {
2020-02-23 22:21:17 +01:00
$this->colorNormal = $arr['color_normal'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['color_start']) && !empty($arr['color_start'])) {
2020-02-23 22:21:17 +01:00
$this->colorStart = $arr['color_start'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['color_stop']) && !empty($arr['color_stop'])) {
2020-02-23 22:21:17 +01:00
$this->colorStop = $arr['color_stop'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['color_extra']) && !empty($arr['color_extra'])) {
2020-02-23 22:21:17 +01:00
$this->colorExtra = $arr['color_extra'];
}
2020-05-21 12:32:16 +02:00
if (isset($arr['color_hilite']) && !empty($arr['color_hilite'])) {
2020-02-23 22:21:17 +01:00
$this->colorHilite = $arr['color_hilite'];
}
2017-04-09 23:35:55 +02:00
}
2020-02-19 18:42:44 +01:00
}
2017-04-06 19:18:51 +02:00
?>