feat: implement comprehensive User Profile Management System (issue #107)

Complete user profile management system with CRUD operations and CLI integration:

## 🎯 Core Features Delivered
- **ProfileManager**: Complete CRUD operations with database integration
- **JSON Schema validation**: Comprehensive profile data validation
- **Multiple profile support**: Named profiles (personal, work, etc.)
- **Default profile system**: Set and manage default profiles
- **Profile inheritance**: Merge profiles with override capabilities
- **Template integration**: Extract flattened variables for template filling

## 📋 Profile Schema & Data Model
- **Structured data classes**: ProfileData, ContactInfo, Address, Organization
- **JSON Schema validation**: Full validation with field descriptions
- **Flexible structure**: Support for nested data and custom fields
- **Timestamp management**: Automatic created_at/updated_at tracking

## 🖥️ CLI Integration Complete
- **9 CLI Commands**: create, show, list, update, delete, set-default, export, import, variables
- **Multiple formats**: JSON, YAML, and table output formats
- **Interactive mode**: Guided profile creation and updates
- **Export/Import**: Full profile portability with validation
- **Template variables**: Extract flattened variables for template systems

## 📊 Implementation Stats
- **ProfileManager**: 500+ lines with comprehensive functionality
- **ProfileSchema**: 350+ lines with validation and data structures
- **CLI Commands**: 450+ lines of professional command interface
- **Test Coverage**: 66 tests (36 core + 30 CLI) with 100% pass rate

## 🚀 **Ready for Template Integration**
Foundation complete for Issue #99 (Auto Fill Templates) with:
- Template variable extraction from profiles
- Default profile system for seamless integration
- Profile merging for complex template scenarios
- Professional CLI for user profile management

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
2025-10-04 01:53:31 +02:00
parent 397b607442
commit b83dc14f7b
8 changed files with 2884 additions and 0 deletions

View File

@@ -38,6 +38,13 @@ try:
except ImportError:
COST_TRACKING_AVAILABLE = False
# Import profile management commands
try:
from .profile.commands import profile_commands
PROFILE_MANAGEMENT_AVAILABLE = True
except ImportError:
PROFILE_MANAGEMENT_AVAILABLE = False
def get_database_path(config):
"""Get database path from config."""
@@ -6556,6 +6563,10 @@ def categories(config):
if COST_TRACKING_AVAILABLE:
cli.add_command(cost_commands)
# Register profile management commands
if PROFILE_MANAGEMENT_AVAILABLE:
cli.add_command(profile_commands)
# Register paradigms commands
cli.add_command(paradigms)

View File

@@ -0,0 +1,24 @@
"""
User Profile Management System for MarkiTect.
This package provides comprehensive user profile management including:
- CRUD operations for user profiles
- Multiple profile support (personal, work, etc.)
- JSON schema validation
- Database integration with persistent storage
- Profile inheritance and template support
- Data export/import functionality
"""
from .manager import ProfileManager, ProfileNotFoundError, ProfileValidationError
from .schema import ProfileSchema, ProfileData
from .commands import profile_commands
__all__ = [
'ProfileManager',
'ProfileSchema',
'ProfileData',
'ProfileNotFoundError',
'ProfileValidationError',
'profile_commands'
]

View File

@@ -0,0 +1,569 @@
"""
CLI commands for user profile management.
This module provides command-line interface for user profile operations
including creation, modification, listing, and template integration.
"""
import click
import sys
import json
from typing import Optional, Dict, Any
from pathlib import Path
from .manager import ProfileManager, ProfileNotFoundError, ProfileValidationError
from .schema import ProfileSchema, ProfileData
@click.group(name='profile')
def profile_commands():
"""User profile management commands."""
pass
@profile_commands.command('create')
@click.argument('name')
@click.option('--description', help='Profile description')
@click.option('--first-name', help='First name')
@click.option('--last-name', help='Last name')
@click.option('--email', help='Email address')
@click.option('--phone', help='Phone number')
@click.option('--organization', help='Organization name')
@click.option('--position', help='Job position')
@click.option('--city', help='City')
@click.option('--country', help='Country')
@click.option('--interactive', '-i', is_flag=True, help='Interactive profile creation')
@click.option('--set-default', is_flag=True, help='Set as default profile')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def create_profile(name: str, description: Optional[str], first_name: Optional[str],
last_name: Optional[str], email: Optional[str], phone: Optional[str],
organization: Optional[str], position: Optional[str], city: Optional[str],
country: Optional[str], interactive: bool, set_default: bool,
db_path: Optional[str]):
"""Create a new user profile."""
try:
profile_manager = ProfileManager(db_path)
if interactive:
# Interactive profile creation
profile_data = _interactive_profile_creation()
if description is None:
description = click.prompt('Profile description', default='', show_default=False) or None
else:
# Command-line profile creation
profile_data = ProfileData(
first_name=first_name,
last_name=last_name,
contact=ProfileSchema.create_empty_profile().contact,
address=ProfileSchema.create_empty_profile().address,
organization=ProfileSchema.create_empty_profile().organization
)
# Set contact info
if email:
profile_data.contact.email = email
if phone:
profile_data.contact.phone = phone
# Set organization info
if organization:
profile_data.organization.name = organization
if position:
profile_data.organization.position = position
# Set address info
if city:
profile_data.address.city = city
if country:
profile_data.address.country = country
# Compute full name if first and last are provided
if first_name and last_name:
profile_data.full_name = f"{first_name} {last_name}"
# Create profile
profile_id = profile_manager.create_profile(
name=name,
data=profile_data,
description=description,
set_as_default=set_default
)
click.echo(f"✅ Created profile '{name}' (ID: {profile_id})")
if set_default:
click.echo(f"🎯 Set '{name}' as default profile")
except ValueError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except ProfileValidationError as e:
click.echo(f"Validation Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error creating profile: {e}", err=True)
sys.exit(1)
@profile_commands.command('show')
@click.argument('name')
@click.option('--format', 'output_format', type=click.Choice(['json', 'yaml', 'table']),
default='table', help='Output format')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def show_profile(name: str, output_format: str, db_path: Optional[str]):
"""Show profile details."""
try:
profile_manager = ProfileManager(db_path)
profile_data = profile_manager.get_profile(name)
profile_info = profile_manager.get_profile_info(name)
if output_format == 'json':
output = {
'profile_info': profile_info,
'profile_data': profile_data.to_dict()
}
click.echo(json.dumps(output, indent=2, ensure_ascii=False))
elif output_format == 'yaml':
try:
import yaml
output = {
'profile_info': profile_info,
'profile_data': profile_data.to_dict()
}
click.echo(yaml.dump(output, default_flow_style=False, allow_unicode=True))
except ImportError:
click.echo("Error: PyYAML package required for YAML output", err=True)
sys.exit(1)
else:
# Table format
_display_profile_table(profile_info, profile_data)
except ProfileNotFoundError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error showing profile: {e}", err=True)
sys.exit(1)
@profile_commands.command('list')
@click.option('--include-inactive', is_flag=True, help='Include inactive profiles')
@click.option('--format', 'output_format', type=click.Choice(['table', 'json']),
default='table', help='Output format')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def list_profiles(include_inactive: bool, output_format: str, db_path: Optional[str]):
"""List all profiles."""
try:
profile_manager = ProfileManager(db_path)
profiles = profile_manager.list_profiles(include_inactive=include_inactive)
if not profiles:
click.echo("No profiles found.")
return
if output_format == 'json':
click.echo(json.dumps(profiles, indent=2, ensure_ascii=False))
else:
# Table format
click.echo("👤 User Profiles")
click.echo("=" * 80)
click.echo(f"{'Name':<20} {'Description':<30} {'Default':<8} {'Status':<8} {'Updated':<12}")
click.echo("-" * 80)
for profile in profiles:
status = "Active" if profile['is_active'] else "Inactive"
default = "Yes" if profile['is_default'] else "No"
description = profile['description'] or "(No description)"
updated = profile['updated_at'][:10] if profile['updated_at'] else "N/A"
click.echo(f"{profile['name']:<20} {description[:29]:<30} {default:<8} {status:<8} {updated:<12}")
click.echo(f"\nTotal: {len(profiles)} profiles")
if not include_inactive:
inactive_count = len([p for p in profiles if not p['is_active']])
if inactive_count > 0:
click.echo(f"💡 Use --include-inactive to show {inactive_count} inactive profiles")
except Exception as e:
click.echo(f"Error listing profiles: {e}", err=True)
sys.exit(1)
@profile_commands.command('update')
@click.argument('name')
@click.option('--description', help='Update profile description')
@click.option('--first-name', help='Update first name')
@click.option('--last-name', help='Update last name')
@click.option('--email', help='Update email address')
@click.option('--phone', help='Update phone number')
@click.option('--organization', help='Update organization name')
@click.option('--position', help='Update job position')
@click.option('--interactive', '-i', is_flag=True, help='Interactive profile update')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def update_profile(name: str, description: Optional[str], first_name: Optional[str],
last_name: Optional[str], email: Optional[str], phone: Optional[str],
organization: Optional[str], position: Optional[str], interactive: bool,
db_path: Optional[str]):
"""Update an existing profile."""
try:
profile_manager = ProfileManager(db_path)
if interactive:
# Interactive update
current_profile = profile_manager.get_profile(name)
profile_data = _interactive_profile_update(current_profile)
if description is None:
current_info = profile_manager.get_profile_info(name)
description = click.prompt('Profile description',
default=current_info['description'] or '',
show_default=True) or None
else:
# Get current profile and update specific fields
current_profile = profile_manager.get_profile(name)
# Update fields if provided
if first_name:
current_profile.first_name = first_name
if last_name:
current_profile.last_name = last_name
if email:
current_profile.contact.email = email
if phone:
current_profile.contact.phone = phone
if organization:
current_profile.organization.name = organization
if position:
current_profile.organization.position = position
# Recompute full name if first or last name changed
if first_name or last_name:
if current_profile.first_name and current_profile.last_name:
current_profile.full_name = f"{current_profile.first_name} {current_profile.last_name}"
profile_data = current_profile
# Update profile
success = profile_manager.update_profile(name, profile_data, description)
if success:
click.echo(f"✅ Updated profile '{name}'")
else:
click.echo(f"❌ Failed to update profile '{name}'")
sys.exit(1)
except ProfileNotFoundError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except ProfileValidationError as e:
click.echo(f"Validation Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error updating profile: {e}", err=True)
sys.exit(1)
@profile_commands.command('delete')
@click.argument('name')
@click.option('--hard', is_flag=True, help='Permanently delete (default is soft delete)')
@click.option('--yes', is_flag=True, help='Skip confirmation prompt')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def delete_profile(name: str, hard: bool, yes: bool, db_path: Optional[str]):
"""Delete a profile."""
try:
profile_manager = ProfileManager(db_path)
# Confirm deletion unless --yes flag is used
if not yes:
action = "permanently delete" if hard else "deactivate"
if not click.confirm(f"Are you sure you want to {action} profile '{name}'?"):
click.echo("Deletion cancelled.")
return
success = profile_manager.delete_profile(name, hard_delete=hard)
if success:
action = "deleted permanently" if hard else "deactivated"
click.echo(f"✅ Profile '{name}' {action}")
else:
click.echo(f"❌ Failed to delete profile '{name}'")
sys.exit(1)
except ProfileNotFoundError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error deleting profile: {e}", err=True)
sys.exit(1)
@profile_commands.command('set-default')
@click.argument('name')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def set_default_profile(name: str, db_path: Optional[str]):
"""Set a profile as the default profile."""
try:
profile_manager = ProfileManager(db_path)
success = profile_manager.set_default_profile(name)
if success:
click.echo(f"✅ Set '{name}' as default profile")
else:
click.echo(f"❌ Failed to set '{name}' as default")
sys.exit(1)
except ProfileNotFoundError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error setting default: {e}", err=True)
sys.exit(1)
@profile_commands.command('export')
@click.argument('name')
@click.option('--format', 'output_format', type=click.Choice(['json', 'yaml']),
default='json', help='Export format')
@click.option('--output', 'output_file', help='Output file path')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def export_profile(name: str, output_format: str, output_file: Optional[str], db_path: Optional[str]):
"""Export profile to file or stdout."""
try:
profile_manager = ProfileManager(db_path)
exported_data = profile_manager.export_profile(name, output_format)
if output_file:
Path(output_file).write_text(exported_data, encoding='utf-8')
click.echo(f"✅ Exported profile '{name}' to {output_file}")
else:
click.echo(exported_data)
except ProfileNotFoundError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error exporting profile: {e}", err=True)
sys.exit(1)
@profile_commands.command('import')
@click.argument('name')
@click.argument('input_file')
@click.option('--format', 'input_format', type=click.Choice(['json', 'yaml']),
default='json', help='Import format')
@click.option('--overwrite', is_flag=True, help='Overwrite existing profile')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def import_profile(name: str, input_file: str, input_format: str, overwrite: bool, db_path: Optional[str]):
"""Import profile from file."""
try:
profile_manager = ProfileManager(db_path)
if not Path(input_file).exists():
click.echo(f"Error: Input file '{input_file}' not found", err=True)
sys.exit(1)
data = Path(input_file).read_text(encoding='utf-8')
profile_id = profile_manager.import_profile(name, data, input_format, overwrite)
action = "Updated" if overwrite else "Created"
click.echo(f"{action} profile '{name}' (ID: {profile_id})")
except ValueError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except ProfileValidationError as e:
click.echo(f"Validation Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error importing profile: {e}", err=True)
sys.exit(1)
@profile_commands.command('variables')
@click.option('--profile', help='Profile name (uses default if not specified)')
@click.option('--format', 'output_format', type=click.Choice(['table', 'json']),
default='table', help='Output format')
@click.option('--database', 'db_path', help='Database path (defaults to config)')
def show_template_variables(profile: Optional[str], output_format: str, db_path: Optional[str]):
"""Show template variables from a profile."""
try:
profile_manager = ProfileManager(db_path)
variables = profile_manager.get_template_variables(profile)
if not variables:
if profile:
click.echo(f"No template variables found for profile '{profile}'")
else:
click.echo("No default profile set or profile is empty")
return
profile_name = profile or "(default)"
if output_format == 'json':
click.echo(json.dumps(variables, indent=2, ensure_ascii=False))
else:
# Table format
click.echo(f"📋 Template Variables - {profile_name}")
click.echo("=" * 60)
click.echo(f"{'Variable':<25} {'Value':<35}")
click.echo("-" * 60)
for key, value in sorted(variables.items()):
value_str = str(value)[:34] if value else "(empty)"
click.echo(f"{key:<25} {value_str:<35}")
click.echo(f"\nTotal: {len(variables)} variables")
except ProfileNotFoundError as e:
click.echo(f"Error: {e}", err=True)
sys.exit(1)
except Exception as e:
click.echo(f"Error showing variables: {e}", err=True)
sys.exit(1)
def _interactive_profile_creation() -> ProfileData:
"""Interactive profile creation helper."""
click.echo("📝 Interactive Profile Creation")
click.echo("=" * 40)
profile_data = ProfileSchema.create_empty_profile()
# Basic information
profile_data.first_name = click.prompt("First name", default="", show_default=False) or None
profile_data.last_name = click.prompt("Last name", default="", show_default=False) or None
if profile_data.first_name and profile_data.last_name:
default_full = f"{profile_data.first_name} {profile_data.last_name}"
profile_data.full_name = click.prompt("Full name", default=default_full) or None
profile_data.preferred_name = click.prompt("Preferred name", default="", show_default=False) or None
profile_data.title = click.prompt("Title (Mr/Ms/Dr/etc)", default="", show_default=False) or None
# Contact information
if click.confirm("Add contact information?", default=True):
profile_data.contact.email = click.prompt("Email", default="", show_default=False) or None
profile_data.contact.phone = click.prompt("Phone", default="", show_default=False) or None
profile_data.contact.website = click.prompt("Website", default="", show_default=False) or None
# Organization information
if click.confirm("Add organization information?", default=True):
profile_data.organization.name = click.prompt("Organization", default="", show_default=False) or None
profile_data.organization.position = click.prompt("Position/Title", default="", show_default=False) or None
profile_data.profession = click.prompt("Profession", default="", show_default=False) or None
# Address information
if click.confirm("Add address information?", default=False):
profile_data.address.city = click.prompt("City", default="", show_default=False) or None
profile_data.address.state = click.prompt("State/Province", default="", show_default=False) or None
profile_data.address.country = click.prompt("Country", default="", show_default=False) or None
return profile_data
def _interactive_profile_update(current_profile: ProfileData) -> ProfileData:
"""Interactive profile update helper."""
click.echo("📝 Interactive Profile Update")
click.echo("=" * 40)
click.echo("Current values shown in brackets. Press Enter to keep unchanged.")
# Basic information
current_profile.first_name = click.prompt(
"First name", default=current_profile.first_name or "", show_default=True
) or None
current_profile.last_name = click.prompt(
"Last name", default=current_profile.last_name or "", show_default=True
) or None
if current_profile.first_name and current_profile.last_name:
default_full = f"{current_profile.first_name} {current_profile.last_name}"
current_profile.full_name = click.prompt(
"Full name", default=current_profile.full_name or default_full, show_default=True
) or None
# Contact information
if click.confirm("Update contact information?", default=False):
current_profile.contact.email = click.prompt(
"Email", default=current_profile.contact.email or "", show_default=True
) or None
current_profile.contact.phone = click.prompt(
"Phone", default=current_profile.contact.phone or "", show_default=True
) or None
# Organization information
if click.confirm("Update organization information?", default=False):
current_profile.organization.name = click.prompt(
"Organization", default=current_profile.organization.name or "", show_default=True
) or None
current_profile.organization.position = click.prompt(
"Position/Title", default=current_profile.organization.position or "", show_default=True
) or None
return current_profile
def _display_profile_table(profile_info: Dict[str, Any], profile_data: ProfileData) -> None:
"""Display profile in table format."""
click.echo(f"👤 Profile: {profile_info['name']}")
click.echo("=" * 50)
if profile_info['description']:
click.echo(f"Description: {profile_info['description']}")
click.echo()
# Basic Information
click.echo("📋 Basic Information")
click.echo("-" * 20)
if profile_data.full_name:
click.echo(f"Full Name: {profile_data.full_name}")
if profile_data.first_name:
click.echo(f"First Name: {profile_data.first_name}")
if profile_data.last_name:
click.echo(f"Last Name: {profile_data.last_name}")
if profile_data.preferred_name:
click.echo(f"Preferred Name: {profile_data.preferred_name}")
if profile_data.title:
click.echo(f"Title: {profile_data.title}")
# Contact Information
if any([profile_data.contact.email, profile_data.contact.phone, profile_data.contact.website]):
click.echo(f"\n📞 Contact Information")
click.echo("-" * 20)
if profile_data.contact.email:
click.echo(f"Email: {profile_data.contact.email}")
if profile_data.contact.phone:
click.echo(f"Phone: {profile_data.contact.phone}")
if profile_data.contact.website:
click.echo(f"Website: {profile_data.contact.website}")
# Organization Information
if any([profile_data.organization.name, profile_data.organization.position, profile_data.profession]):
click.echo(f"\n🏢 Organization")
click.echo("-" * 15)
if profile_data.organization.name:
click.echo(f"Organization: {profile_data.organization.name}")
if profile_data.organization.position:
click.echo(f"Position: {profile_data.organization.position}")
if profile_data.profession:
click.echo(f"Profession: {profile_data.profession}")
# Address Information
if any([profile_data.address.city, profile_data.address.state, profile_data.address.country]):
click.echo(f"\n🌍 Address")
click.echo("-" * 10)
address_parts = []
if profile_data.address.city:
address_parts.append(profile_data.address.city)
if profile_data.address.state:
address_parts.append(profile_data.address.state)
if profile_data.address.country:
address_parts.append(profile_data.address.country)
click.echo(f"Location: {', '.join(address_parts)}")
# Metadata
click.echo(f"\n⏰ Metadata")
click.echo("-" * 10)
click.echo(f"Default Profile: {'Yes' if profile_info['is_default'] else 'No'}")
click.echo(f"Created: {profile_info['created_at'][:19] if profile_info['created_at'] else 'N/A'}")
click.echo(f"Updated: {profile_info['updated_at'][:19] if profile_info['updated_at'] else 'N/A'}")

View File

@@ -0,0 +1,663 @@
"""
User Profile Manager for MarkiTect.
This module provides comprehensive CRUD operations for user profiles including
database integration, validation, and profile inheritance functionality.
"""
import sqlite3
import json
import os
from datetime import datetime
from typing import Dict, Any, Optional, List, Union
from pathlib import Path
from jsonschema import ValidationError
from .schema import ProfileSchema, ProfileData
from ..database import DatabaseManager
class ProfileNotFoundError(Exception):
"""Raised when a profile is not found."""
pass
class ProfileValidationError(Exception):
"""Raised when profile data fails validation."""
pass
class ProfileManager:
"""
Comprehensive user profile management system.
Handles CRUD operations, validation, and database integration for user profiles
with support for multiple named profiles and template integration.
"""
def __init__(self, db_path: Optional[str] = None):
"""
Initialize profile manager.
Args:
db_path: Path to SQLite database file. If None, uses default from config.
"""
if db_path:
self.db_path = db_path
else:
# Use default database path from MarkiTect
from ..config_manager import ConfigurationManager
config_manager = ConfigurationManager()
config = config_manager.get_current_config()
self.db_path = config.get('database_path', 'markitect.db')
self._ensure_database_initialized()
def _ensure_database_initialized(self) -> None:
"""Ensure database and user_profiles table exist."""
# Initialize main database if needed
if not os.path.exists(self.db_path):
db_manager = DatabaseManager(self.db_path)
db_manager.initialize_database()
# Create user_profiles table
self._create_profiles_table()
def _create_profiles_table(self) -> None:
"""Create user_profiles table if it doesn't exist."""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
cursor.execute('''
CREATE TABLE IF NOT EXISTS user_profiles (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name TEXT NOT NULL UNIQUE,
description TEXT,
data TEXT NOT NULL, -- JSON data
is_active BOOLEAN DEFAULT TRUE,
is_default BOOLEAN DEFAULT FALSE,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)
''')
# Create index for faster lookups
cursor.execute('''
CREATE INDEX IF NOT EXISTS idx_user_profiles_name
ON user_profiles(name)
''')
cursor.execute('''
CREATE INDEX IF NOT EXISTS idx_user_profiles_active
ON user_profiles(is_active)
''')
conn.commit()
except sqlite3.Error as e:
conn.rollback()
raise RuntimeError(f"Failed to create profiles table: {e}")
finally:
conn.close()
def create_profile(self, name: str, data: Union[Dict[str, Any], ProfileData],
description: Optional[str] = None,
set_as_default: bool = False) -> int:
"""
Create a new user profile.
Args:
name: Unique profile name
data: Profile data (dict or ProfileData instance)
description: Optional profile description
set_as_default: Whether to set as default profile
Returns:
ID of created profile
Raises:
ProfileValidationError: If data is invalid
ValueError: If profile name already exists
"""
# Convert ProfileData to dict if needed
if isinstance(data, ProfileData):
profile_data = data.to_dict()
else:
profile_data = data.copy()
# Add timestamps
now = datetime.now().isoformat()
profile_data.setdefault('created_at', now)
profile_data['updated_at'] = now
# Validate data
try:
ProfileSchema.validate(profile_data)
except ValidationError as e:
raise ProfileValidationError(f"Profile data validation failed: {e.message}")
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
# Check if profile name already exists
cursor.execute('SELECT id FROM user_profiles WHERE name = ?', (name,))
if cursor.fetchone():
raise ValueError(f"Profile '{name}' already exists")
# If setting as default, unset any existing default
if set_as_default:
cursor.execute('''
UPDATE user_profiles SET is_default = FALSE
WHERE is_default = TRUE
''')
# Insert new profile
cursor.execute('''
INSERT INTO user_profiles
(name, description, data, is_default, created_at, updated_at)
VALUES (?, ?, ?, ?, ?, ?)
''', (
name,
description,
json.dumps(profile_data, indent=2),
set_as_default,
now,
now
))
conn.commit()
profile_id = cursor.lastrowid
return profile_id
except sqlite3.IntegrityError as e:
conn.rollback()
if "UNIQUE constraint failed" in str(e):
raise ValueError(f"Profile '{name}' already exists")
raise RuntimeError(f"Database error: {e}")
except sqlite3.Error as e:
conn.rollback()
raise RuntimeError(f"Failed to create profile: {e}")
finally:
conn.close()
def get_profile(self, name: str) -> ProfileData:
"""
Get profile by name.
Args:
name: Profile name
Returns:
ProfileData instance
Raises:
ProfileNotFoundError: If profile doesn't exist
"""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
cursor.execute('''
SELECT data FROM user_profiles
WHERE name = ? AND is_active = TRUE
''', (name,))
row = cursor.fetchone()
if not row:
raise ProfileNotFoundError(f"Profile '{name}' not found")
profile_data = json.loads(row[0])
return ProfileData.from_dict(profile_data)
except sqlite3.Error as e:
raise RuntimeError(f"Failed to get profile: {e}")
finally:
conn.close()
def get_profile_info(self, name: str) -> Dict[str, Any]:
"""
Get profile metadata without full data.
Args:
name: Profile name
Returns:
Dictionary with profile metadata
Raises:
ProfileNotFoundError: If profile doesn't exist
"""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
cursor.execute('''
SELECT id, name, description, is_active, is_default,
created_at, updated_at
FROM user_profiles
WHERE name = ? AND is_active = TRUE
''', (name,))
row = cursor.fetchone()
if not row:
raise ProfileNotFoundError(f"Profile '{name}' not found")
return {
'id': row[0],
'name': row[1],
'description': row[2],
'is_active': bool(row[3]),
'is_default': bool(row[4]),
'created_at': row[5],
'updated_at': row[6]
}
except sqlite3.Error as e:
raise RuntimeError(f"Failed to get profile info: {e}")
finally:
conn.close()
def update_profile(self, name: str, data: Union[Dict[str, Any], ProfileData],
description: Optional[str] = None) -> bool:
"""
Update existing profile.
Args:
name: Profile name
data: Updated profile data
description: Updated description (optional)
Returns:
True if updated successfully
Raises:
ProfileNotFoundError: If profile doesn't exist
ProfileValidationError: If data is invalid
"""
# Convert ProfileData to dict if needed
if isinstance(data, ProfileData):
profile_data = data.to_dict()
else:
profile_data = data.copy()
# Update timestamp
profile_data['updated_at'] = datetime.now().isoformat()
# Validate data
try:
ProfileSchema.validate(profile_data)
except ValidationError as e:
raise ProfileValidationError(f"Profile data validation failed: {e.message}")
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
# Check if profile exists
cursor.execute('SELECT id FROM user_profiles WHERE name = ? AND is_active = TRUE', (name,))
if not cursor.fetchone():
raise ProfileNotFoundError(f"Profile '{name}' not found")
# Update profile
update_fields = ['data = ?', 'updated_at = ?']
update_values = [json.dumps(profile_data, indent=2), profile_data['updated_at']]
if description is not None:
update_fields.append('description = ?')
update_values.append(description)
update_values.append(name) # For WHERE clause
cursor.execute(f'''
UPDATE user_profiles
SET {', '.join(update_fields)}
WHERE name = ? AND is_active = TRUE
''', update_values)
conn.commit()
return cursor.rowcount > 0
except sqlite3.Error as e:
conn.rollback()
raise RuntimeError(f"Failed to update profile: {e}")
finally:
conn.close()
def delete_profile(self, name: str, hard_delete: bool = False) -> bool:
"""
Delete a profile.
Args:
name: Profile name
hard_delete: If True, permanently delete. If False, mark as inactive.
Returns:
True if deleted successfully
Raises:
ProfileNotFoundError: If profile doesn't exist
"""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
# Check if profile exists
cursor.execute('SELECT id FROM user_profiles WHERE name = ?', (name,))
if not cursor.fetchone():
raise ProfileNotFoundError(f"Profile '{name}' not found")
if hard_delete:
# Permanently delete
cursor.execute('DELETE FROM user_profiles WHERE name = ?', (name,))
else:
# Mark as inactive
cursor.execute('''
UPDATE user_profiles
SET is_active = FALSE, updated_at = ?
WHERE name = ?
''', (datetime.now().isoformat(), name))
conn.commit()
return cursor.rowcount > 0
except sqlite3.Error as e:
conn.rollback()
raise RuntimeError(f"Failed to delete profile: {e}")
finally:
conn.close()
def list_profiles(self, include_inactive: bool = False) -> List[Dict[str, Any]]:
"""
List all profiles.
Args:
include_inactive: Whether to include inactive profiles
Returns:
List of profile metadata dictionaries
"""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
if include_inactive:
cursor.execute('''
SELECT id, name, description, is_active, is_default,
created_at, updated_at
FROM user_profiles
ORDER BY is_default DESC, name ASC
''')
else:
cursor.execute('''
SELECT id, name, description, is_active, is_default,
created_at, updated_at
FROM user_profiles
WHERE is_active = TRUE
ORDER BY is_default DESC, name ASC
''')
rows = cursor.fetchall()
return [
{
'id': row[0],
'name': row[1],
'description': row[2],
'is_active': bool(row[3]),
'is_default': bool(row[4]),
'created_at': row[5],
'updated_at': row[6]
}
for row in rows
]
except sqlite3.Error as e:
raise RuntimeError(f"Failed to list profiles: {e}")
finally:
conn.close()
def set_default_profile(self, name: str) -> bool:
"""
Set a profile as the default profile.
Args:
name: Profile name
Returns:
True if set successfully
Raises:
ProfileNotFoundError: If profile doesn't exist
"""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
# Check if profile exists
cursor.execute('SELECT id FROM user_profiles WHERE name = ? AND is_active = TRUE', (name,))
if not cursor.fetchone():
raise ProfileNotFoundError(f"Profile '{name}' not found")
# Unset all defaults
cursor.execute('UPDATE user_profiles SET is_default = FALSE')
# Set new default
cursor.execute('''
UPDATE user_profiles
SET is_default = TRUE, updated_at = ?
WHERE name = ? AND is_active = TRUE
''', (datetime.now().isoformat(), name))
conn.commit()
return cursor.rowcount > 0
except sqlite3.Error as e:
conn.rollback()
raise RuntimeError(f"Failed to set default profile: {e}")
finally:
conn.close()
def get_default_profile(self) -> Optional[ProfileData]:
"""
Get the default profile.
Returns:
Default ProfileData or None if no default set
"""
conn = sqlite3.connect(self.db_path)
cursor = conn.cursor()
try:
cursor.execute('''
SELECT name, data FROM user_profiles
WHERE is_default = TRUE AND is_active = TRUE
LIMIT 1
''')
row = cursor.fetchone()
if not row:
return None
profile_data = json.loads(row[1])
return ProfileData.from_dict(profile_data)
except sqlite3.Error as e:
raise RuntimeError(f"Failed to get default profile: {e}")
finally:
conn.close()
def export_profile(self, name: str, format: str = 'json') -> str:
"""
Export profile to string format.
Args:
name: Profile name
format: Export format ('json' or 'yaml')
Returns:
Exported profile string
Raises:
ProfileNotFoundError: If profile doesn't exist
ValueError: If format is unsupported
"""
profile_data = self.get_profile(name)
profile_info = self.get_profile_info(name)
export_data = {
'profile_info': profile_info,
'profile_data': profile_data.to_dict()
}
if format.lower() == 'json':
return json.dumps(export_data, indent=2, ensure_ascii=False)
elif format.lower() == 'yaml':
try:
import yaml
return yaml.dump(export_data, default_flow_style=False, allow_unicode=True)
except ImportError:
raise ValueError("YAML format requires PyYAML package")
else:
raise ValueError(f"Unsupported export format: {format}")
def import_profile(self, name: str, data: str, format: str = 'json',
overwrite: bool = False) -> int:
"""
Import profile from string format.
Args:
name: New profile name (overrides name in data if different)
data: Profile data string
format: Import format ('json' or 'yaml')
overwrite: Whether to overwrite existing profile
Returns:
ID of imported profile
Raises:
ValueError: If format is unsupported or profile exists without overwrite
ProfileValidationError: If data is invalid
"""
if format.lower() == 'json':
import_data = json.loads(data)
elif format.lower() == 'yaml':
try:
import yaml
import_data = yaml.safe_load(data)
except ImportError:
raise ValueError("YAML format requires PyYAML package")
else:
raise ValueError(f"Unsupported import format: {format}")
# Extract profile data
if 'profile_data' in import_data:
profile_data = import_data['profile_data']
else:
profile_data = import_data
description = None
if 'profile_info' in import_data and 'description' in import_data['profile_info']:
description = import_data['profile_info']['description']
# Check if profile exists
try:
existing_profile = self.get_profile_info(name)
if not overwrite:
raise ValueError(f"Profile '{name}' already exists. Use overwrite=True to replace.")
# Update existing profile
self.update_profile(name, profile_data, description)
return existing_profile['id']
except ProfileNotFoundError:
# Create new profile
return self.create_profile(name, profile_data, description)
def merge_profiles(self, base_profile: str, override_profile: str) -> ProfileData:
"""
Merge two profiles with override taking precedence.
Args:
base_profile: Base profile name
override_profile: Override profile name
Returns:
Merged ProfileData
Raises:
ProfileNotFoundError: If either profile doesn't exist
"""
base_data = self.get_profile(base_profile).to_dict()
override_data = self.get_profile(override_profile).to_dict()
def deep_merge(base: Dict[str, Any], override: Dict[str, Any]) -> Dict[str, Any]:
"""Deep merge two dictionaries."""
result = base.copy()
for key, value in override.items():
if key in result and isinstance(result[key], dict) and isinstance(value, dict):
result[key] = deep_merge(result[key], value)
elif value is not None: # Only override with non-None values
result[key] = value
return result
merged_data = deep_merge(base_data, override_data)
return ProfileData.from_dict(merged_data)
def get_template_variables(self, profile_name: Optional[str] = None) -> Dict[str, Any]:
"""
Get flattened template variables from a profile.
Args:
profile_name: Profile name (uses default if None)
Returns:
Flattened dictionary suitable for template filling
Raises:
ProfileNotFoundError: If profile doesn't exist
"""
if profile_name:
profile_data = self.get_profile(profile_name)
else:
profile_data = self.get_default_profile()
if not profile_data:
return {}
# Flatten nested structure for template variables
variables = {}
profile_dict = profile_data.to_dict()
def flatten_dict(d: Dict[str, Any], prefix: str = '') -> None:
"""Recursively flatten dictionary."""
for key, value in d.items():
if value is None:
continue
new_key = f"{prefix}.{key}" if prefix else key
if isinstance(value, dict):
flatten_dict(value, new_key)
elif isinstance(value, (str, int, float, bool)):
variables[new_key] = value
elif isinstance(value, list):
# Convert lists to comma-separated strings
variables[new_key] = ', '.join(str(item) for item in value)
flatten_dict(profile_dict)
# Add some computed convenience variables
if 'first_name' in variables and 'last_name' in variables:
variables['full_name'] = f"{variables['first_name']} {variables['last_name']}"
return variables

355
markitect/profile/schema.py Normal file
View File

@@ -0,0 +1,355 @@
"""
Profile schema definition and validation for MarkiTect user profiles.
This module defines the JSON schema structure for user profiles and provides
validation functionality to ensure profile data integrity.
"""
import json
from datetime import datetime
from typing import Dict, Any, Optional, List
from dataclasses import dataclass, asdict, field
from jsonschema import validate, ValidationError
@dataclass
class ContactInfo:
"""Contact information structure."""
email: Optional[str] = None
phone: Optional[str] = None
website: Optional[str] = None
linkedin: Optional[str] = None
github: Optional[str] = None
twitter: Optional[str] = None
@dataclass
class Address:
"""Address structure."""
street: Optional[str] = None
city: Optional[str] = None
state: Optional[str] = None
postal_code: Optional[str] = None
country: Optional[str] = None
@dataclass
class Organization:
"""Organization/company information."""
name: Optional[str] = None
position: Optional[str] = None
department: Optional[str] = None
website: Optional[str] = None
address: Optional[Address] = field(default_factory=Address)
@dataclass
class ProfileData:
"""
Complete user profile data structure.
This dataclass defines all the fields that can be stored in a user profile
for template auto-filling purposes.
"""
# Basic personal information
first_name: Optional[str] = None
last_name: Optional[str] = None
full_name: Optional[str] = None
preferred_name: Optional[str] = None
title: Optional[str] = None # Mr., Ms., Dr., etc.
# Contact information
contact: ContactInfo = field(default_factory=ContactInfo)
# Address information
address: Address = field(default_factory=Address)
# Professional information
organization: Organization = field(default_factory=Organization)
profession: Optional[str] = None
bio: Optional[str] = None
# Additional fields for template filling
custom_fields: Dict[str, Any] = field(default_factory=dict)
# Metadata
created_at: Optional[str] = None
updated_at: Optional[str] = None
def to_dict(self) -> Dict[str, Any]:
"""Convert profile data to dictionary."""
return asdict(self)
@classmethod
def from_dict(cls, data: Dict[str, Any]) -> 'ProfileData':
"""Create ProfileData from dictionary."""
# Handle nested structures
if 'contact' in data and isinstance(data['contact'], dict):
data['contact'] = ContactInfo(**data['contact'])
if 'address' in data and isinstance(data['address'], dict):
data['address'] = Address(**data['address'])
if 'organization' in data and isinstance(data['organization'], dict):
org_data = data['organization'].copy()
if 'address' in org_data and isinstance(org_data['address'], dict):
org_data['address'] = Address(**org_data['address'])
data['organization'] = Organization(**org_data)
return cls(**data)
class ProfileSchema:
"""
JSON Schema validation for user profiles.
Provides schema definition and validation methods to ensure
profile data integrity and consistency.
"""
# JSON Schema definition for profile validation
SCHEMA = {
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "MarkiTect User Profile",
"type": "object",
"properties": {
"first_name": {
"type": ["string", "null"],
"maxLength": 100,
"description": "User's first name"
},
"last_name": {
"type": ["string", "null"],
"maxLength": 100,
"description": "User's last name"
},
"full_name": {
"type": ["string", "null"],
"maxLength": 200,
"description": "User's full name"
},
"preferred_name": {
"type": ["string", "null"],
"maxLength": 100,
"description": "User's preferred name"
},
"title": {
"type": ["string", "null"],
"maxLength": 20,
"description": "Title (Mr., Ms., Dr., etc.)"
},
"contact": {
"type": "object",
"properties": {
"email": {
"type": ["string", "null"],
"format": "email",
"description": "Email address"
},
"phone": {
"type": ["string", "null"],
"maxLength": 50,
"description": "Phone number"
},
"website": {
"type": ["string", "null"],
"format": "uri",
"description": "Personal website URL"
},
"linkedin": {
"type": ["string", "null"],
"description": "LinkedIn profile URL"
},
"github": {
"type": ["string", "null"],
"description": "GitHub profile URL"
},
"twitter": {
"type": ["string", "null"],
"description": "Twitter handle or URL"
}
},
"additionalProperties": False
},
"address": {
"type": "object",
"properties": {
"street": {
"type": ["string", "null"],
"maxLength": 200,
"description": "Street address"
},
"city": {
"type": ["string", "null"],
"maxLength": 100,
"description": "City"
},
"state": {
"type": ["string", "null"],
"maxLength": 100,
"description": "State or province"
},
"postal_code": {
"type": ["string", "null"],
"maxLength": 20,
"description": "Postal or ZIP code"
},
"country": {
"type": ["string", "null"],
"maxLength": 100,
"description": "Country"
}
},
"additionalProperties": False
},
"organization": {
"type": "object",
"properties": {
"name": {
"type": ["string", "null"],
"maxLength": 200,
"description": "Organization name"
},
"position": {
"type": ["string", "null"],
"maxLength": 100,
"description": "Job position/title"
},
"department": {
"type": ["string", "null"],
"maxLength": 100,
"description": "Department"
},
"website": {
"type": ["string", "null"],
"format": "uri",
"description": "Organization website"
},
"address": {
"$ref": "#/properties/address"
}
},
"additionalProperties": False
},
"profession": {
"type": ["string", "null"],
"maxLength": 100,
"description": "Professional role or occupation"
},
"bio": {
"type": ["string", "null"],
"maxLength": 2000,
"description": "Professional biography"
},
"custom_fields": {
"type": "object",
"description": "Additional custom fields for template filling",
"additionalProperties": True
},
"created_at": {
"type": ["string", "null"],
"format": "date-time",
"description": "Profile creation timestamp"
},
"updated_at": {
"type": ["string", "null"],
"format": "date-time",
"description": "Profile last update timestamp"
}
},
"additionalProperties": False
}
@classmethod
def validate(cls, data: Dict[str, Any]) -> None:
"""
Validate profile data against the schema.
Args:
data: Profile data dictionary to validate
Raises:
ValidationError: If data doesn't match schema
"""
validate(instance=data, schema=cls.SCHEMA)
@classmethod
def is_valid(cls, data: Dict[str, Any]) -> bool:
"""
Check if profile data is valid.
Args:
data: Profile data dictionary to check
Returns:
True if valid, False otherwise
"""
try:
cls.validate(data)
return True
except ValidationError:
return False
@classmethod
def get_field_description(cls, field_path: str) -> Optional[str]:
"""
Get description for a specific field.
Args:
field_path: Dot-separated field path (e.g., 'contact.email')
Returns:
Field description or None if not found
"""
parts = field_path.split('.')
current = cls.SCHEMA['properties']
try:
for part in parts:
if part in current:
current = current[part]
if 'properties' in current:
current = current['properties']
else:
return current.get('description')
return current.get('description')
except (KeyError, TypeError):
return None
@classmethod
def get_all_fields(cls) -> List[str]:
"""
Get list of all available field paths.
Returns:
List of dot-separated field paths
"""
fields = []
def extract_fields(schema_dict: Dict[str, Any], prefix: str = '') -> None:
if 'properties' not in schema_dict:
return
for field_name, field_def in schema_dict['properties'].items():
field_path = f"{prefix}.{field_name}" if prefix else field_name
fields.append(field_path)
if field_def.get('type') == 'object' and 'properties' in field_def:
extract_fields(field_def, field_path)
extract_fields(cls.SCHEMA)
return sorted(fields)
@classmethod
def create_empty_profile(self) -> ProfileData:
"""
Create an empty profile with default structure.
Returns:
Empty ProfileData instance
"""
now = datetime.now().isoformat()
return ProfileData(
created_at=now,
updated_at=now
)