LifeRPG_v2.0/modern/backend/models.py
TLimoges33 2b961611fd
🚀 Major Enhancement: Complete AI-Powered LifeRPG Platform with Git LFS
 New Features:
- AI-powered habit creation with natural language processing
- HuggingFace transformers integration for sentiment analysis (tracked via Git LFS)
- Advanced predictive analytics and behavioral insights
- Voice & image input capabilities for hands-free habit tracking
- Real-time notifications and community features
- Plugin system with extensible architecture

🔧 Technical Improvements:
- Comprehensive FastAPI backend with 30+ endpoints
- React frontend with PWA capabilities
- Advanced authentication with 2FA support
- RBAC authorization system
- Comprehensive security features (CSRF, rate limiting, audit logging)
- Database migrations and health monitoring
- Docker containerization support
- Git LFS configured for large AI model files (2+ GB)

📚 Documentation & DevOps:
- Complete deployment guides for multiple platforms
- Professional README with feature highlights
- GitHub Actions CI/CD workflows
- Comprehensive API documentation
- Security audit roadmap and compliance framework
- Setup scripts for development environment

🧪 Testing & Quality:
- Comprehensive test suite with 20+ test modules
- Setup verification scripts
- Working development environment with both backend and frontend
- Health checks and monitoring systems

🌟 Ready for:
- Portfolio showcasing
- Community contributions
- Production deployment
- Professional presentation
2025-09-28 21:29:19 +00:00

243 lines
8.9 KiB
Python

from sqlalchemy import (
Column, Integer, String, Text, DateTime, ForeignKey, create_engine, func, UniqueConstraint, Float, Index
)
from sqlalchemy.orm import declarative_base
from sqlalchemy.orm import relationship, sessionmaker
import os
from datetime import datetime
from crypto import encrypt_text, decrypt_text
Base = declarative_base()
DATABASE_URL = os.getenv("DATABASE_URL", "sqlite:///./modern_dev.db")
engine = create_engine(DATABASE_URL, connect_args={"check_same_thread": False} if "sqlite" in DATABASE_URL else {})
SessionLocal = sessionmaker(autocommit=False, autoflush=False, bind=engine)
class User(Base):
__tablename__ = 'users'
id = Column(Integer, primary_key=True, index=True)
email = Column(String, unique=True, nullable=False, index=True)
password_hash = Column(String)
role = Column(String, default='user')
display_name = Column(String)
_totp_secret = Column("totp_secret", String) # encrypted at rest
totp_enabled = Column(Integer, default=0) # 0/1
_recovery_codes = Column("recovery_codes", Text) # encrypted at rest
created_at = Column(DateTime, server_default=func.current_timestamp())
updated_at = Column(DateTime, server_default=func.current_timestamp(), onupdate=func.current_timestamp())
profile = relationship("Profile", back_populates="user", cascade="all, delete-orphan")
projects = relationship("Project", back_populates="user", cascade="all, delete-orphan")
habits = relationship("Habit", back_populates="user", cascade="all, delete-orphan")
momentum = relationship("UserMomentum", back_populates="user", cascade="all, delete-orphan")
@property
def totp_secret(self):
"""Decrypt TOTP secret when accessed"""
if self._totp_secret:
try:
return decrypt_text(self._totp_secret)
except Exception:
return None
return None
@totp_secret.setter
def totp_secret(self, value):
"""Encrypt TOTP secret when stored"""
if value:
self._totp_secret = encrypt_text(value)
else:
self._totp_secret = None
@property
def recovery_codes(self):
"""Decrypt recovery codes when accessed"""
if self._recovery_codes:
try:
return decrypt_text(self._recovery_codes)
except Exception:
return None
return None
@recovery_codes.setter
def recovery_codes(self, value):
"""Encrypt recovery codes when stored"""
if value:
self._recovery_codes = encrypt_text(value)
else:
self._recovery_codes = None
class Profile(Base):
__tablename__ = 'profiles'
id = Column(Integer, primary_key=True)
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
key = Column(String, nullable=False)
value = Column(Text)
user = relationship("User", back_populates="profile")
class Project(Base):
__tablename__ = 'projects'
id = Column(Integer, primary_key=True)
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
title = Column(String, nullable=False)
description = Column(Text)
created_at = Column(DateTime, server_default=func.current_timestamp())
updated_at = Column(DateTime, server_default=func.current_timestamp(), onupdate=func.current_timestamp())
user = relationship("User", back_populates="projects")
class Habit(Base):
__tablename__ = 'habits'
id = Column(Integer, primary_key=True)
project_id = Column(Integer, ForeignKey('projects.id'))
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
title = Column(String, nullable=False)
notes = Column(Text)
cadence = Column(String)
difficulty = Column(Integer, default=1)
xp_reward = Column(Integer, default=10)
status = Column(String, default='active') # active|completed|archived
due_date = Column(DateTime)
labels = Column(Text) # JSON list of labels
created_at = Column(DateTime, server_default=func.current_timestamp())
user = relationship("User", back_populates="habits")
__table_args__ = (
Index('idx_habit_user_status', 'user_id', 'status'),
Index('idx_habit_user_created', 'user_id', 'created_at'),
Index('idx_habit_due_date', 'due_date'),
Index('idx_habit_status', 'status'),
)
class Log(Base):
__tablename__ = 'logs'
id = Column(Integer, primary_key=True)
habit_id = Column(Integer, ForeignKey('habits.id'))
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
action = Column(String)
timestamp = Column(DateTime, server_default=func.current_timestamp())
__table_args__ = (
Index('idx_log_user_timestamp', 'user_id', 'timestamp'),
Index('idx_log_user_action', 'user_id', 'action'),
Index('idx_log_habit_timestamp', 'habit_id', 'timestamp'),
)
class Achievement(Base):
__tablename__ = 'achievements'
id = Column(Integer, primary_key=True)
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
name = Column(String, nullable=False)
description = Column(Text)
earned_at = Column(DateTime)
class Integration(Base):
__tablename__ = 'integrations'
id = Column(Integer, primary_key=True)
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
provider = Column(String, nullable=False)
external_id = Column(String)
config = Column(Text)
created_at = Column(DateTime, server_default=func.current_timestamp())
class OAuthToken(Base):
__tablename__ = 'oauth_tokens'
id = Column(Integer, primary_key=True)
integration_id = Column(Integer, ForeignKey('integrations.id'))
access_token = Column(Text)
refresh_token = Column(Text)
scope = Column(Text)
expires_at = Column(Integer)
class ChangeLog(Base):
__tablename__ = 'change_log'
id = Column(Integer, primary_key=True)
user_id = Column(Integer)
entity = Column(String)
entity_id = Column(Integer)
action = Column(String)
payload = Column(Text)
created_at = Column(DateTime, server_default=func.current_timestamp())
class Guild(Base):
__tablename__ = 'guilds'
id = Column(Integer, primary_key=True)
name = Column(String, nullable=False)
description = Column(Text)
owner_id = Column(Integer, ForeignKey('users.id'))
created_at = Column(DateTime, server_default=func.current_timestamp())
class GuildMember(Base):
__tablename__ = 'guild_members'
id = Column(Integer, primary_key=True)
guild_id = Column(Integer, ForeignKey('guilds.id'))
user_id = Column(Integer, ForeignKey('users.id'))
role = Column(String, default='member')
class TelemetryEvent(Base):
__tablename__ = 'telemetry_events'
id = Column(Integer, primary_key=True)
user_id = Column(Integer)
name = Column(String, nullable=False)
payload = Column(Text)
created_at = Column(DateTime, server_default=func.current_timestamp())
class IntegrationItemMap(Base):
__tablename__ = 'integration_item_map'
id = Column(Integer, primary_key=True)
integration_id = Column(Integer, ForeignKey('integrations.id'), nullable=False)
external_id = Column(String, nullable=False)
entity_type = Column(String, nullable=False)
entity_id = Column(Integer, nullable=False)
updated_at = Column(DateTime, server_default=func.current_timestamp(), onupdate=func.current_timestamp())
created_at = Column(DateTime, server_default=func.current_timestamp())
__table_args__ = (
UniqueConstraint('integration_id', 'external_id', 'entity_type', name='uq_integration_item'),
)
class PublicToken(Base):
__tablename__ = 'public_tokens'
id = Column(Integer, primary_key=True)
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
name = Column(String, nullable=False)
scope = Column(String, default='read:widgets')
token_hash = Column(String, unique=True, nullable=False)
created_at = Column(DateTime, server_default=func.current_timestamp())
last_used_at = Column(DateTime)
class OIDCLoginState(Base):
__tablename__ = 'oidc_login_state'
id = Column(Integer, primary_key=True)
state = Column(String, unique=True, nullable=False)
provider = Column(String, nullable=False)
code_verifier = Column(String, nullable=False)
redirect_to = Column(String)
created_at = Column(DateTime, server_default=func.current_timestamp())
expires_at = Column(DateTime)
class UserMomentum(Base):
__tablename__ = 'user_momentum'
id = Column(Integer, primary_key=True)
user_id = Column(Integer, ForeignKey('users.id'), nullable=False)
momentum = Column(Float, default=50.0) # Current momentum level (0-100)
last_updated = Column(DateTime, server_default=func.current_timestamp())
user = relationship("User", back_populates="momentum")
def init_db():
Base.metadata.create_all(bind=engine)
if __name__ == '__main__':
init_db()
print('Initialized DB ->', DATABASE_URL)