feat: integrations manager endpoints + sync-to-habits; add guild models/endpoints

This commit is contained in:
TLimoges33 2025-08-28 17:18:58 +00:00
parent 1d9a02f92f
commit 71994e0c09
2 changed files with 168 additions and 0 deletions

View File

@ -5,6 +5,7 @@ from .oauth import router as oauth_router
import os
import requests
import time
from fastapi import Body
app = FastAPI(title='LifeRPG Modern Backend')
@ -74,3 +75,153 @@ def google_events(integration_id: int):
return resp.json()
finally:
db.close()
@app.post('/api/v1/guilds')
def create_guild(payload: dict = Body({})):
name = payload.get('name')
owner_id = payload.get('owner_id', 1)
if not name:
raise HTTPException(status_code=400, detail='name required')
db = models.SessionLocal()
try:
g = models.Guild(name=name, description=payload.get('description'), owner_id=owner_id)
db.add(g)
db.commit()
db.refresh(g)
return {'id': g.id, 'name': g.name}
finally:
db.close()
@app.get('/api/v1/guilds')
def list_guilds():
db = models.SessionLocal()
try:
rows = db.query(models.Guild).all()
return [{'id': r.id, 'name': r.name, 'owner_id': r.owner_id} for r in rows]
finally:
db.close()
@app.post('/api/v1/guilds/{guild_id}/members')
def add_guild_member(guild_id: int, payload: dict = Body({})):
user_id = payload.get('user_id')
role = payload.get('role', 'member')
if not user_id:
raise HTTPException(status_code=400, detail='user_id required')
db = models.SessionLocal()
try:
gm = models.GuildMember(guild_id=guild_id, user_id=user_id, role=role)
db.add(gm)
db.commit()
db.refresh(gm)
return {'id': gm.id, 'guild_id': gm.guild_id, 'user_id': gm.user_id}
finally:
db.close()
@app.get('/api/v1/guilds/{guild_id}/members')
def list_guild_members(guild_id: int):
db = models.SessionLocal()
try:
rows = db.query(models.GuildMember).filter_by(guild_id=guild_id).all()
return [{'id': r.id, 'user_id': r.user_id, 'role': r.role} for r in rows]
finally:
db.close()
@app.get('/api/v1/users/{user_id}/integrations')
def list_user_integrations(user_id: int):
db = models.SessionLocal()
try:
rows = db.query(models.Integration).filter_by(user_id=user_id).all()
out = [
{"id": r.id, "provider": r.provider, "external_id": r.external_id, "created_at": r.created_at.isoformat() if r.created_at else None}
for r in rows
]
return out
finally:
db.close()
@app.get('/api/v1/integrations')
def list_integrations():
db = models.SessionLocal()
try:
rows = db.query(models.Integration).all()
out = [
{"id": r.id, "user_id": r.user_id, "provider": r.provider, "external_id": r.external_id, "created_at": r.created_at.isoformat() if r.created_at else None}
for r in rows
]
return out
finally:
db.close()
@app.delete('/api/v1/integrations/{integration_id}')
def delete_integration(integration_id: int):
db = models.SessionLocal()
try:
row = db.query(models.Integration).filter_by(id=integration_id).first()
if not row:
raise HTTPException(status_code=404, detail='integration not found')
db.delete(row)
db.commit()
return {'ok': True}
finally:
db.close()
@app.post('/api/v1/integrations/{integration_id}/sync_to_habits')
def sync_integration_to_habits(integration_id: int, payload: dict = Body({})):
"""Fetch events from the integration and create Habit + Log entries.
Demo mapping: create a Habit per event with title 'Event: <summary>' and a Log entry.
"""
db = models.SessionLocal()
try:
integration = db.query(models.Integration).filter_by(id=integration_id).first()
if not integration:
raise HTTPException(status_code=404, detail='integration not found')
# Fetch events via existing events endpoint logic
# Reuse token refresh + decrypt logic from oauth module
token_row = db.query(models.OAuthToken).filter_by(integration_id=integration_id).order_by(models.OAuthToken.id.desc()).first()
if not token_row:
raise HTTPException(status_code=404, detail='no token found for integration')
from .oauth import refresh_google_token_if_needed
refreshed = refresh_google_token_if_needed(token_row)
if refreshed:
token_row = refreshed
from .crypto import decrypt_text
access = decrypt_text(token_row.access_token)
if not access:
raise HTTPException(status_code=500, detail='unable to decrypt access token')
headers = {'Authorization': f'Bearer {access}'}
params = {'maxResults': 25, 'singleEvents': True, 'orderBy': 'startTime', 'timeMin': time.strftime('%Y-%m-%dT%H:%M:%SZ', time.gmtime())}
resp = requests.get('https://www.googleapis.com/calendar/v3/calendars/primary/events', headers=headers, params=params, timeout=10)
if resp.status_code != 200:
raise HTTPException(status_code=502, detail='google api error')
events = resp.json().get('items', [])
created = []
for ev in events:
title = ev.get('summary') or 'Untitled Event'
# Create habit and log
habit = models.Habit(project_id=None, user_id=integration.user_id, title=f'Event: {title}', notes=str(ev), cadence='once')
db.add(habit)
db.commit()
db.refresh(habit)
log = models.Log(habit_id=habit.id, user_id=integration.user_id, action='imported_event')
db.add(log)
db.commit()
created.append({'habit_id': habit.id, 'title': habit.title})
return {'created': created, 'count': len(created)}
finally:
db.close()

View File

@ -102,6 +102,23 @@ class ChangeLog(Base):
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')
def init_db():
Base.metadata.create_all(bind=engine)