Files
TimeTracker/app/utils/installation.py
Dries Peeters c07aaa77fc Fix data directory permission errors in Docker container
- Change CONFIG_DIR from relative 'data' to absolute '/data' path in installation.py
  This fixes PermissionError when trying to create /app/data instead of using
  the mounted volume at /data

- Update telemetry marker file paths to use absolute /data path for consistency

- Add ensure_data_directory() function to entrypoint_fixed.sh to:
  - Create /data directory if it doesn't exist
  - Set proper permissions (755) on /data
  - Attempt to set ownership to current user
  - Create /data/uploads subdirectory

This resolves the 'Permission denied: data' errors when accessing /admin/settings
and ensures the data volume is properly initialized at container startup.
2025-11-28 22:47:55 +01:00

143 lines
4.8 KiB
Python

"""
Installation and configuration utilities for TimeTracker
This module handles first-time setup, installation-specific configuration,
and telemetry salt generation.
"""
import os
import json
import secrets
import hashlib
from pathlib import Path
from typing import Dict, Optional
class InstallationConfig:
"""Manages installation-specific configuration"""
CONFIG_DIR = "/data"
CONFIG_FILE = "installation.json"
def __init__(self):
self.config_path = os.path.join(self.CONFIG_DIR, self.CONFIG_FILE)
self._ensure_config_dir()
self._config = self._load_config()
def _ensure_config_dir(self):
"""Ensure the configuration directory exists"""
os.makedirs(self.CONFIG_DIR, exist_ok=True)
def _load_config(self) -> Dict:
"""Load configuration from file"""
if os.path.exists(self.config_path):
try:
with open(self.config_path, "r") as f:
return json.load(f)
except Exception:
return {}
return {}
def _save_config(self):
"""Save configuration to file"""
try:
with open(self.config_path, "w") as f:
json.dump(self._config, f, indent=2)
except Exception as e:
print(f"Error saving installation config: {e}")
def get_installation_salt(self) -> str:
"""
Get or generate installation-specific salt for telemetry.
This salt is unique per installation and persists across restarts.
It's used to generate consistent anonymous fingerprints.
"""
if "telemetry_salt" not in self._config:
# Generate a unique 64-character hex salt
salt = secrets.token_hex(32) # 32 bytes = 64 hex characters
self._config["telemetry_salt"] = salt
self._save_config()
return self._config["telemetry_salt"]
def get_installation_id(self) -> str:
"""
Get or generate a unique installation ID.
This is a one-way hash that uniquely identifies this installation
without revealing any server information.
"""
if "installation_id" not in self._config:
# Generate a unique installation ID
import platform
import time
# Combine multiple factors for uniqueness
factors = [platform.node() or "unknown", str(time.time()), secrets.token_hex(16)]
# Hash to create installation ID
combined = "".join(factors).encode()
installation_id = hashlib.sha256(combined).hexdigest()[:16]
self._config["installation_id"] = installation_id
self._save_config()
return self._config["installation_id"]
def is_setup_complete(self) -> bool:
"""Check if initial setup is complete"""
return self._config.get("setup_complete", False)
def mark_setup_complete(self, telemetry_enabled: bool = False):
"""Mark initial setup as complete"""
self._config["setup_complete"] = True
self._config["telemetry_enabled"] = telemetry_enabled
self._config["setup_completed_at"] = str(datetime.now())
self._save_config()
def is_initial_data_seeded(self) -> bool:
"""Check if initial database data (default client/project) has been seeded"""
return self._config.get("initial_data_seeded", False)
def mark_initial_data_seeded(self):
"""Mark that initial database data has been seeded"""
self._config["initial_data_seeded"] = True
self._config["initial_data_seeded_at"] = str(datetime.now())
self._save_config()
def get_telemetry_preference(self) -> bool:
"""Get user's telemetry preference"""
# Reload on read to reflect external updates (e.g., tests toggling state)
self._config = self._load_config()
return self._config.get("telemetry_enabled", False)
def set_telemetry_preference(self, enabled: bool):
"""Set user's telemetry preference"""
self._config["telemetry_enabled"] = enabled
self._save_config()
def get_all_config(self) -> Dict:
"""Get all configuration (for admin dashboard)"""
return self._config.copy()
# Global instance
_installation_config = None
_installation_config_path = None
def get_installation_config() -> InstallationConfig:
"""Get the global installation configuration instance"""
global _installation_config, _installation_config_path
# Reinitialize if config path changed (e.g., tests overriding directories)
tmp = InstallationConfig()
current_path = tmp.config_path
if (_installation_config is None) or (_installation_config_path != current_path):
_installation_config = tmp
_installation_config_path = current_path
return _installation_config
# Add missing datetime import
from datetime import datetime