Files
kite/.planning/phases/06.2-close-v1-sharing-cloud-delete-csv-export-gaps/06.2-REVIEW.md
T
2026-05-31 15:29:57 +02:00

306 lines
12 KiB
Markdown

---
phase: 06.2-close-v1-sharing-cloud-delete-csv-export-gaps
reviewed: 2026-05-31T00:00:00Z
depth: standard
files_reviewed: 6
files_reviewed_list:
- backend/api/audit.py
- backend/tests/test_audit.py
- backend/tests/test_documents.py
- backend/tests/test_shares.py
- frontend/src/api/client.js
- frontend/src/components/admin/AuditLogTab.vue
findings:
critical: 2
warning: 5
info: 3
total: 10
status: issues_found
---
# Phase 06.2: Code Review Report
**Reviewed:** 2026-05-31T00:00:00Z
**Depth:** standard
**Files Reviewed:** 6
**Status:** issues_found
## Summary
This phase closes sharing, cloud-delete, CSV-export, and daily-audit-export gaps. The backend
endpoint logic is structurally sound (parameterized queries, admin-only guards, proper path-traversal
regex on the date parameter). However, two correctness bugs are present that will silently corrupt or
truncate data in production, and five quality/robustness issues need addressing before the phase can
be marked complete. No hardcoded secrets were found and all admin guards are in place.
---
## Critical Issues
### CR-01: CSV export writes raw Python `dict` repr for `metadata_` — not JSON
**File:** `backend/api/audit.py:372`
**Issue:** `csv.DictWriter.writerow()` receives the dict returned by
`_audit_to_dict_with_handles()`. That dict contains `"metadata_": entry.metadata_`, where
`entry.metadata_` is a Python `dict` (SQLAlchemy deserializes `JSONB` columns into native Python
objects). `csv.DictWriter` calls `str()` on the value, producing `{'size_bytes': 100}`
Python repr syntax with single quotes — not valid JSON. Any downstream consumer that tries to
`JSON.parse()` the metadata cell will fail. The same issue exists in the empty-handle early-return
path's header-only CSV, but that branch writes no data rows so the corruption is less visible there.
Proof (reproduced):
```python
import csv, io
output = io.StringIO()
writer = csv.DictWriter(output, fieldnames=["metadata_"])
writer.writeheader()
writer.writerow({"metadata_": {"size_bytes": 100}})
print(output.getvalue())
# metadata_
# {'size_bytes': 100} ← Python repr, NOT JSON
```
**Fix:** Serialize `metadata_` to JSON in the dict helper before the CSV writer sees it, or only
at the CSV call site:
```python
import json
# Option A — at writerow call site (surgical fix):
for row in rows:
entry, user_handle_val, actor_handle_val = row[0], row[1], row[2]
record = _audit_to_dict_with_handles(entry, user_handle_val, actor_handle_val)
record["metadata_"] = json.dumps(record["metadata_"]) if record["metadata_"] is not None else ""
writer.writerow(record)
```
The `test_audit_log_export_csv` test does not assert the content of the metadata cell, so this
bug currently passes all tests despite corrupting the exported data.
---
### CR-02: `download_daily_export` silently crashes on non-MinIO backends — no 404, no guard
**File:** `backend/api/audit.py:219-239`
**Issue:** `list_daily_exports` (line 182) correctly guards with
`if not isinstance(backend, MinIOBackend): return {"items": []}` and returns an empty list for
non-MinIO deployments. But `download_daily_export` has **no such guard**. If `get_storage_backend()`
returns anything other than a `MinIOBackend` (e.g. a local filesystem backend), line 223 calls
`backend._client.get_object(...)`, which will raise `AttributeError: '...' object has no attribute
'_client'`. The broad `except Exception` on line 231 catches this and re-raises it as a 404, but
only because the error happens inside `_get()`. If the backend raises during construction of `_get`
or if `backend` is `None`, the error propagates as an unhandled 500. More importantly, any new
backend type that happens to have a `_client` attribute with a different meaning would silently
produce wrong results.
**Fix:** Mirror the guard from `list_daily_exports`:
```python
@router.get("/audit-log/daily-exports/{date}")
async def download_daily_export(
date: str,
_admin: User = Depends(get_current_admin),
) -> StreamingResponse:
if not re.fullmatch(r"\d{4}-\d{2}-\d{2}", date):
raise HTTPException(status_code=404, detail="Invalid date format")
backend = get_storage_backend()
if not isinstance(backend, MinIOBackend): # <-- add this guard
raise HTTPException(status_code=404, detail="Export not found")
key = f"audit-logs/{date}.csv"
...
```
---
## Warnings
### WR-01: `export_audit_log` loads the entire audit log into memory before streaming
**File:** `backend/api/audit.py:351-377`
**Issue:** The export endpoint issues `result = await session.execute(q)` followed by
`rows = result.all()`, then builds the full CSV in an `io.StringIO` before returning it in a
`StreamingResponse`. For large deployments the audit log may contain millions of rows; this will
exhaust server memory and cause an OOM kill rather than a graceful error. `StreamingResponse` is
the correct mechanism but it is not being streamed — the CSV is fully materialized first.
**Fix:** Use a generator that yields CSV chunks from a cursor:
```python
async def _generate_csv(session, q, fields):
output = io.StringIO()
writer = csv.DictWriter(output, fieldnames=fields)
writer.writeheader()
yield output.getvalue()
async for row in await session.stream(q):
output = io.StringIO()
entry, user_handle_val, actor_handle_val = row[0], row[1], row[2]
record = _audit_to_dict_with_handles(entry, user_handle_val, actor_handle_val)
record["metadata_"] = json.dumps(record["metadata_"]) if record["metadata_"] is not None else ""
writer = csv.DictWriter(output, fieldnames=fields)
writer.writerow(record)
yield output.getvalue()
return StreamingResponse(_generate_csv(session, q, fields), media_type="text/csv", ...)
```
### WR-02: `format` query parameter is accepted but ignored — always returns CSV
**File:** `backend/api/audit.py:313`
**Issue:** The `export_audit_log` endpoint declares `format: str = Query(default="csv")` with a
`# noqa: A002` suppression but never reads the `format` variable in the handler body. Passing
`?format=json` returns a CSV response silently. This is a logic dead-end: the parameter is
either unused dead code or is intended as a future extension. If it is dead code it should be
removed; if it is a future hook, it must at minimum validate that only `"csv"` is accepted and
return 422 for other values. Currently an arbitrary string is accepted with no effect.
**Fix (remove the dead parameter):**
```python
@router.get("/audit-log/export")
async def export_audit_log(
start: Optional[datetime] = Query(default=None),
end: Optional[datetime] = Query(default=None),
user_handle: Optional[str] = Query(default=None),
event_type: Optional[str] = Query(default=None),
# Remove: format: str = Query(default="csv"),
session: AsyncSession = Depends(get_db),
_admin: User = Depends(get_current_admin),
) -> StreamingResponse:
```
**Fix (validate if keeping):**
```python
if format not in ("csv",):
raise HTTPException(status_code=422, detail="Unsupported format. Only 'csv' is accepted.")
```
### WR-03: Pagination "Next" button in `AuditLogTab.vue` disables on the wrong condition
**File:** `frontend/src/components/admin/AuditLogTab.vue:122`
**Issue:** The "Next" button is disabled when `entries.length < perPage`. This is a heuristic:
it assumes the last page has fewer entries than `perPage`. But if the total count is an exact
multiple of `perPage`, the last page has exactly `perPage` entries and the "Next" button remains
enabled. Clicking it fetches a page beyond the last, returns an empty items list, and leaves the
user on a blank page with no way to go back (the empty-entries check at line 73 replaces the
table with the "No entries" message, but the page counter still shows the incremented value).
The `total` ref is populated from `data.total` (line 212) — it should be used instead.
**Fix:**
```javascript
// Replace line 122:
:disabled="page * perPage >= total"
// And in nextPage():
function nextPage() {
if (page.value * perPage < total.value) {
page.value++
fetchLog()
}
}
```
### WR-04: `loadDailyExports` swallows errors silently — no user-visible feedback
**File:** `frontend/src/components/admin/AuditLogTab.vue:256-266`
**Issue:** The `loadDailyExports` catch block sets `dailyExports.value = []` but does **not** set
`exportsError.value`. The template renders `<p v-if="exportsError">` for the download error, but
the load-failure case is silently dropped. An admin whose MinIO is misconfigured or unreachable
sees "No daily exports available" — identical to the normal empty-bucket state — with no
indication that an error occurred.
**Fix:**
```javascript
} catch (e) {
dailyExports.value = []
exportsError.value = 'Failed to load daily exports. Please try again.' // add this line
} finally {
```
### WR-05: `listShares` in `client.js` uses string interpolation instead of `URLSearchParams`
**File:** `frontend/src/api/client.js:353`
**Issue:**
```javascript
export function listShares(docId) {
return request(`/api/shares?document_id=${docId}`)
}
```
`docId` is always a UUID from the DB so the practical risk is low, but the pattern is inconsistent
with the rest of the file (compare `listDocuments`, `adminListAuditLog`, and `listFolders` which
all use `URLSearchParams`). If a caller ever passes a non-UUID value (e.g. a stringified object
from a Vue ref), this will silently produce a malformed URL without any encoding. Unlike the other
route-param-interpolation cases in this file (`/documents/${id}`, `/folders/${folderId}`) — where
the value is an ID used in a path segment — this one places an unencoded value into a query
parameter, making URL injection more likely.
**Fix:**
```javascript
export function listShares(docId) {
const params = new URLSearchParams({ document_id: docId })
return request(`/api/shares?${params}`)
}
```
---
## Info
### IN-01: Unused `import re` in `test_documents.py`
**File:** `backend/tests/test_documents.py:9`
**Issue:** `import re` is present at the top of the file but `re.` is never called anywhere in
the module. This is dead code left over from a previous version of the test.
**Fix:** Remove `import re` from line 9.
### IN-02: `deleteDocumentRemoveOnly` is a trivial wrapper that duplicates `deleteDocument(id, true)`
**File:** `frontend/src/api/client.js:80-82`
**Issue:**
```javascript
export function deleteDocumentRemoveOnly(id) {
return deleteDocument(id, true)
}
```
This function is a one-liner alias that adds no value beyond calling `deleteDocument(id, true)`
directly. It creates two public API symbols for the same operation, which will confuse future
callers about which to use. The caller in `DocumentView.vue` already uses this alias, but the
wrapper itself adds surface area without benefit.
**Fix:** Either remove the wrapper and update `DocumentView.vue` to call
`deleteDocument(doc.value.id, true)` directly, or keep it but document why the alias exists.
### IN-03: `test_delete_cloud_remove_only` does not assert quota is unchanged
**File:** `backend/tests/test_documents.py:897-925`
**Issue:** The test verifies the DB row is removed but does not check that the user's
`used_bytes` quota was NOT decremented (which it should not be — cloud docs are not quota-tracked
per T-06.2-03-01). The `Quota` model is imported but never queried in this test. A future
regression that incorrectly decrements quota on `remove_only` paths would go undetected.
**Fix:** Add a quota assertion:
```python
# Quota must remain unchanged — cloud docs are not quota-tracked
quota = await db_session.get(Quota, auth_user["user"].id)
assert quota.used_bytes == 0, (
f"remove_only should not decrement quota, got used_bytes={quota.used_bytes}"
)
```
---
_Reviewed: 2026-05-31T00:00:00Z_
_Reviewer: Claude (gsd-code-reviewer)_
_Depth: standard_