Problem:
- Left/right arrows for photo navigation showed 'No preview'
- navigatePhoto() missing user_id parameter in stream API call
- img.src used old format: /api/v2/files/{file_uuid}/stream
Solution:
- Modified navigatePhoto() to include user_id
- Changed to: /api/v2/files/{user_id}/{file_uuid}/stream
- Get user_id from localStorage (tree_user)
Result:
- Photo navigation arrows now work correctly ✅
- Can browse through jpg/png/gif images in detail panel ✅
- Position indicator (1/2371) shows correct count ✅
Files:
- src/page.html (navigatePhoto function)
Problem:
- Files could not be clicked (error: no location)
- get_file_info used hardcoded demo database
- file_locations table was empty
Solution:
1. Scan now inserts file_locations records
- file_uuid = node_id (temporary)
- location = file path (from aliases)
- label = origin
2. Modified API routes to include user_id
- /api/v2/files/:user_id/:file_uuid/info
- /api/v2/files/:user_id/:file_uuid/stream
3. Modified showDetail() to use tree_user from localStorage
Result:
- file_locations: 11857 records ✅
- Files can be clicked ✅
- API uses correct user database ✅
Files:
- src/scan.rs (insert file_locations)
- src/server.rs (user_id parameter)
- src/page.html (showDetail with user_id)
Problem:
- File nodes had null file_uuid
- Clicking files in UI showed nothing (showDetail() returned early)
- User could not view file details
Solution:
- Set file_uuid to node_id (temporary value) during scan
- Even without SHA256 hash, files can be clicked
- file_uuid will be updated when hash is calculated
Result:
- All files have file_uuid ✅
- Clicking files shows detail panel ✅
- UI fully functional ✅
Files:
- src/scan.rs (file_uuid = node_id)
Problem:
- Home folder could not expand (children_json was empty)
- UI tree view showed flat structure
Solution:
- Add step [5/5] to update children_json for all folders
- Use SQL: json_group_array(node_id) to collect child IDs
- Update 802 folders after node insertion
Performance:
- Children JSON update: 1.55s (74% of total time)
- Total scan time: 2.08s (vs 0.89s without children)
Result:
- Home folder has 805 children ✅
- Tree structure fully functional ✅
- API returns correct children array ✅
Files:
- src/scan.rs (added children_json update step)
Problem:
- All nodes (files and folders) had NULL parent_id
- Tree structure was flat (no hierarchy)
- Could not display proper folder tree
Solution:
- Add Home folder as root node
- Map folder paths to node_ids (folder_id_map)
- Set parent_id for all folders (point to parent folder's node_id)
- Set parent_id for all files (point to containing folder's node_id)
- Root directory files/folders point to Home folder
Result:
- Folders: 802 (801 + Home root)
- Files: 11857
- Total nodes: 12659
- Nodes with parent_id: 12658 (100%)
- Tree structure fully functional ✅
Files:
- src/scan.rs (fixed parent_id logic)
Ownership error fixed:
- file_uuid, file_hash, filename, file_path moved into closure
- Cannot use after move (borrow of moved value)
Solution:
- Clone values before spawn_blocking move:
file_uuid_clone, file_hash_clone, filename_clone, file_path_clone
- Use clones inside closure
- Original values still available for return
Code changes:
- Added 4 clone statements before db_result
- Updated closure params to use clones
Compilation now successful ✅
Upload handler working correctly ✅
Files:
- src/server.rs (line 828-833: clone statements)
Final fix for compilation:
- node_id generation: uuid[0..8] → chars().take(8).collect()
- Split into two lines for clarity:
let uuid_str = uuid::Uuid::new_v4().to_string().replace('-', );
let node_id = format!("node-{}", uuid_str.chars().take(8).collect::<String>());
Compilation now successful ✅
All string slice errors fixed ✅
Files:
- src/server.rs (line 836-837: node_id generation)
Critical improvement:
- Removed dependency on localhost:3002 register API
- MarkBase now generates file_uuid locally (UUID v4)
- Directly saves to user SQLite database
Changes:
- Removed curl call to external API
- file_uuid = uuid::Uuid::new_v4().replace('-', '')
- Added database save logic:
* INSERT into file_registry (file_uuid, sha256, file_size)
* INSERT into file_locations (file_uuid, file_path)
* INSERT into file_nodes (node_id, label, file_uuid)
Result:
- Files uploaded → immediately saved to database ✅
- Tree API shows nodes immediately ✅
- No external API dependency ✅
- Works for all users (demo, warren, momentry) ✅
Test result:
✅ warren upload → file saved to database
✅ Tree API shows uploaded file
✅ No empty tree problem
Files:
- src/server.rs (removed external API, added local database save)
Problem diagnosis:
- Upload API returns 201 Created ✅
- file_uuid generated: 9e553348... ✅
- But upload handler doesn't save to database ❌
- Tree API returns empty nodes []
- Frontend shows empty tree
Root cause:
- upload_file() handler only saves file to disk
- No database persistence logic
- hardcoded demo_dir path
Temporary solution:
- Manually inserted SVG to momentry.sqlite
- file_registry: 1 record (test.svg)
- file_nodes: 1 record (test.svg)
- Tree API now shows 1 node ✅
Permanent fix needed:
- Add FileTree::insert_node() logic to upload handler
- Save to user-specific database (momentry.sqlite)
- Use FileTree::new_file_node() to create node
Files:
- data/users/momentry.sqlite (added test.svg manually)
Problem:
- bin target not recognized by cargo
- Manual initialization failed repeatedly
- warren.sqlite and momentry.sqlite were 0 bytes
- API returned 'no such table: file_nodes' error
Solution:
- Copied demo.sqlite as template (has correct table structure)
- Cleared all data from file_nodes, file_registry, file_locations
- Both databases now have proper schema with 0 nodes
Result:
- warren.sqlite: 64KB (same size as demo, but empty)
- momentry.sqlite: 64KB (same size as demo, but empty)
- Tables: file_registry, file_nodes, file_locations ✅
- Nodes: 0 (empty trees)
Test result:
✅ demo/admin123 → {nodes: 50}
✅ warren/admin123 → {nodes: []} (empty tree, no error)
✅ momentry/admin123 → {nodes: []} (empty tree, no error)
Files:
- data/users/warren.sqlite (copied from demo.sqlite, data cleared)
- data/users/momentry.sqlite (copied from demo.sqlite, data cleared)
Problem:
- warren/admin123 login succeeded
- Tree API returned undefined nodes (no database file)
- Frontend error: 'd.nodes is undefined'
Solution:
- Created empty SQLite databases for warren and momentry
- Used FileTree::init_user_db() to initialize
- Each database has empty file_nodes table
Result:
- warren: 0 nodes (empty tree)
- momentry: 0 nodes (empty tree)
- demo: 50 nodes (existing data)
Test result:
✅ warren/admin123 → Tree API returns {nodes: []}
✅ momentry/admin123 → Tree API returns {nodes: []}
✅ demo/admin123 → Tree API returns {nodes: 50}
Files:
- data/users/warren.sqlite (created)
- data/users/momentry.sqlite (created)
Problem 1: File Tree demo/demo123 login failed
- demo user password hash was incorrect
- PostgreSQL users.password was empty or invalid
- SQLite sftpgo_users.password_hash was empty or invalid
Solution 1:
- Generated correct bcrypt hash for 'demo123'
- Updated PostgreSQL users table (60 chars)
- Updated SQLite sftpgo_users table (60 chars)
- CLI test: demo/demo123 login now returns token ✅
Problem 2: Tree modal eye icon position too high
- Password container had no height specified
- Eye icon used top:50% transform, but container height undefined
- Icon appeared misaligned
Solution 2:
- Added height:40px to password container
- Eye icon now positioned correctly at vertical center
Files:
- src/page.html (eye icon container fix)
- data/auth.sqlite (demo password hash)
Problem:
- Close button (✕) only removed 'active' class
- Modal display was set via style.display='block'
- Button didn't change display property, so modal stayed visible
Solution:
- Updated onclick to: this.parentElement.style.display='none'
- Also removes 'active' class for consistency
- Modal now properly hides when clicking ✕ button
Files:
- src/page.html (line ~850)