|
9 | 9 | from sqlalchemy.exc import IntegrityError
|
10 | 10 | from sqlalchemy.orm import Session
|
11 | 11 |
|
| 12 | +from ee.onyx.db.user_group import fetch_user_group |
12 | 13 | from onyx.auth.users import current_curator_or_admin_user
|
13 | 14 | from onyx.auth.users import current_user
|
14 | 15 | from onyx.background.celery.celery_utils import get_deletion_attempt_snapshot
|
|
41 | 42 | from onyx.db.engine import get_session
|
42 | 43 | from onyx.db.enums import AccessType
|
43 | 44 | from onyx.db.enums import ConnectorCredentialPairStatus
|
| 45 | +from onyx.db.enums import SyncType |
44 | 46 | from onyx.db.index_attempt import count_index_attempt_errors_for_cc_pair
|
45 | 47 | from onyx.db.index_attempt import count_index_attempts_for_connector
|
46 | 48 | from onyx.db.index_attempt import get_index_attempt_errors_for_cc_pair
|
|
50 | 52 | from onyx.db.models import User
|
51 | 53 | from onyx.db.search_settings import get_active_search_settings_list
|
52 | 54 | from onyx.db.search_settings import get_current_search_settings
|
| 55 | +from onyx.db.sync_record import fetch_paginated_sync_records |
53 | 56 | from onyx.redis.redis_connector import RedisConnector
|
54 | 57 | from onyx.redis.redis_pool import get_redis_client
|
55 | 58 | from onyx.server.documents.models import CCPairFullInfo
|
|
60 | 63 | from onyx.server.documents.models import DocumentSyncStatus
|
61 | 64 | from onyx.server.documents.models import IndexAttemptSnapshot
|
62 | 65 | from onyx.server.documents.models import PaginatedReturn
|
| 66 | +from onyx.server.documents.models import SyncRecordSnapshot |
63 | 67 | from onyx.server.models import StatusResponse
|
64 | 68 | from onyx.utils.logger import setup_logger
|
65 | 69 | from onyx.utils.variable_functionality import fetch_ee_implementation_or_noop
|
|
69 | 73 | router = APIRouter(prefix="/manage")
|
70 | 74 |
|
71 | 75 |
|
| 76 | +@router.get("/admin/user-group/{group_id}/sync-status") |
| 77 | +def get_user_group_sync_status( |
| 78 | + group_id: int, |
| 79 | + page_num: int = Query(0, ge=0), |
| 80 | + page_size: int = Query(10, ge=1, le=1000), |
| 81 | + user: User | None = Depends(current_curator_or_admin_user), |
| 82 | + db_session: Session = Depends(get_session), |
| 83 | +) -> PaginatedReturn[SyncRecordSnapshot]: |
| 84 | + user_group = fetch_user_group(db_session, group_id) |
| 85 | + if not user_group: |
| 86 | + raise HTTPException(status_code=404, detail="User group not found") |
| 87 | + |
| 88 | + sync_records, total_count = fetch_paginated_sync_records( |
| 89 | + db_session, group_id, SyncType.USER_GROUP, page_num, page_size |
| 90 | + ) |
| 91 | + |
| 92 | + return PaginatedReturn( |
| 93 | + items=[ |
| 94 | + SyncRecordSnapshot.from_sync_record_db_model(sync_record) |
| 95 | + for sync_record in sync_records |
| 96 | + ], |
| 97 | + total_items=total_count, |
| 98 | + ) |
| 99 | + |
| 100 | + |
| 101 | +@router.post("/admin/user-group/{group_id}/sync") |
| 102 | +def sync_user_group( |
| 103 | + group_id: int, |
| 104 | + user: User | None = Depends(current_curator_or_admin_user), |
| 105 | + db_session: Session = Depends(get_session), |
| 106 | +) -> StatusResponse[None]: |
| 107 | + """Triggers sync of a user group immediately""" |
| 108 | + get_current_tenant_id() |
| 109 | + |
| 110 | + user_group = fetch_user_group(db_session, group_id) |
| 111 | + if not user_group: |
| 112 | + raise HTTPException(status_code=404, detail="User group not found") |
| 113 | + |
| 114 | + # Add logic to actually trigger the sync - this would depend on your implementation |
| 115 | + # For example: |
| 116 | + # try_creating_usergroup_sync_task(primary_app, group_id, get_redis_client(), tenant_id) |
| 117 | + |
| 118 | + logger.info(f"User group sync queued: group_id={group_id}") |
| 119 | + |
| 120 | + return StatusResponse( |
| 121 | + success=True, |
| 122 | + message="Successfully created the user group sync task.", |
| 123 | + ) |
| 124 | + |
| 125 | + |
| 126 | +@router.get("/admin/cc-pair/{cc_pair_id}/sync-status") |
| 127 | +def get_cc_pair_sync_status( |
| 128 | + cc_pair_id: int, |
| 129 | + page_num: int = Query(0, ge=0), |
| 130 | + page_size: int = Query(10, ge=1, le=1000), |
| 131 | + user: User | None = Depends(current_curator_or_admin_user), |
| 132 | + db_session: Session = Depends(get_session), |
| 133 | +) -> PaginatedReturn[SyncRecordSnapshot]: |
| 134 | + cc_pair = get_connector_credential_pair_from_id_for_user( |
| 135 | + cc_pair_id, db_session, user, get_editable=False |
| 136 | + ) |
| 137 | + if not cc_pair: |
| 138 | + raise HTTPException( |
| 139 | + status_code=400, detail="CC Pair not found for current user permissions" |
| 140 | + ) |
| 141 | + |
| 142 | + sync_records, total_count = fetch_paginated_sync_records( |
| 143 | + db_session, cc_pair_id, SyncType.EXTERNAL_PERMISSIONS, page_num, page_size |
| 144 | + ) |
| 145 | + |
| 146 | + return PaginatedReturn( |
| 147 | + items=[ |
| 148 | + SyncRecordSnapshot.from_sync_record_db_model(sync_record) |
| 149 | + for sync_record in sync_records |
| 150 | + ], |
| 151 | + total_items=total_count, |
| 152 | + ) |
| 153 | + |
| 154 | + |
72 | 155 | @router.get("/admin/cc-pair/{cc_pair_id}/index-attempts")
|
73 | 156 | def get_cc_pair_index_attempts(
|
74 | 157 | cc_pair_id: int,
|
|
0 commit comments