1 Commits

Author SHA1 Message Date
ergosteur
5267dab236 fix: address Docker EACCES errors with better logging and SELinux hints 2026-03-07 03:02:02 -05:00
4 changed files with 58 additions and 31 deletions

View File

@@ -25,6 +25,8 @@ docker run -d \
ghcr.io/ergosteur/instaarchive-viewer:latest ghcr.io/ergosteur/instaarchive-viewer:latest
``` ```
> **Note for Linux/SELinux users:** If you see "Permission Denied" in the logs, append `,z` to your volume mount: `-v /path/to/archives:/archives:ro,z`
### Docker Compose ### Docker Compose
Create a `compose.yml` file: Create a `compose.yml` file:
@@ -36,13 +38,22 @@ services:
ports: ports:
- "3000:3000" - "3000:3000"
volumes: volumes:
- ./archives:/archives:ro - ./archives:/archives:ro,z # ,z handles SELinux permissions
``` ```
Run with: ### Troubleshooting Permissions
```bash
docker compose up -d If the app shows "No Archives Found" and logs `EACCES: permission denied`:
```
1. **Check Directory Permissions**: Ensure the archive folder is world-readable:
```bash
chmod -R 755 /path/to/archives
```
2. **SELinux (Fedora/RHEL/CentOS)**: Use the `:z` flag in your volume mount as shown above.
3. **User Mapping**: You can force the container to run as your host user:
```bash
docker run --user $(id -u):$(id -g) ...
```
## Supported Archive Structure ## Supported Archive Structure

View File

@@ -1,11 +1,10 @@
services: services:
instaarchive: instaarchive:
image: ghcr.io/${GITHUB_REPOSITORY:-ergosteur/instaarchive-viewer}:latest image: ghcr.io/ergosteur/instaarchive-viewer:latest
build: .
ports: ports:
- "3000:3000" - "3000:3000"
volumes: volumes:
- ./archives:/archives:ro - ./archives:/archives:ro,z
environment: environment:
- PORT=3000 - PORT=3000
- ARCHIVES_DIR=/archives - ARCHIVES_DIR=/archives

View File

@@ -1,7 +1,7 @@
{ {
"name": "react-example", "name": "react-example",
"private": true, "private": true,
"version": "1.1.1", "version": "1.1.2",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "vite --port=3000 --host=0.0.0.0", "dev": "vite --port=3000 --host=0.0.0.0",

View File

@@ -3,6 +3,7 @@ import fs from 'fs';
import path from 'path'; import path from 'path';
import { fileURLToPath } from 'url'; import { fileURLToPath } from 'url';
import dotenv from 'dotenv'; import dotenv from 'dotenv';
import os from 'os';
dotenv.config(); dotenv.config();
@@ -14,13 +15,18 @@ const PORT = process.env.PORT || 3001;
const ARCHIVES_DIR = path.resolve(process.env.ARCHIVES_DIR || path.join(__dirname, '_sample-archives')); const ARCHIVES_DIR = path.resolve(process.env.ARCHIVES_DIR || path.join(__dirname, '_sample-archives'));
console.log(`[Server] Initializing...`); console.log(`[Server] Initializing...`);
console.log(`[Server] Running as user: ${os.userInfo().username} (UID: ${os.userInfo().uid}, GID: ${os.userInfo().gid})`);
console.log(`[Server] Environment ARCHIVES_DIR: ${process.env.ARCHIVES_DIR}`); console.log(`[Server] Environment ARCHIVES_DIR: ${process.env.ARCHIVES_DIR}`);
console.log(`[Server] Resolved ARCHIVES_DIR: ${ARCHIVES_DIR}`); console.log(`[Server] Resolved ARCHIVES_DIR: ${ARCHIVES_DIR}`);
// Ensure archives directory exists // Ensure archives directory exists
if (!fs.existsSync(ARCHIVES_DIR)) { if (!fs.existsSync(ARCHIVES_DIR)) {
console.warn(`[Server] Warning: Archives directory not found at ${ARCHIVES_DIR}. Creating it...`); console.warn(`[Server] Warning: Archives directory not found at ${ARCHIVES_DIR}. Creating it...`);
fs.mkdirSync(ARCHIVES_DIR, { recursive: true }); try {
fs.mkdirSync(ARCHIVES_DIR, { recursive: true });
} catch (err) {
console.error(`[Server] Failed to create archives directory:`, err);
}
} else { } else {
console.log(`[Server] Archives directory exists.`); console.log(`[Server] Archives directory exists.`);
} }
@@ -43,31 +49,42 @@ app.get('/api/archives', (req, res) => {
.map(item => { .map(item => {
// Try to find a profile pic or first image for the thumbnail // Try to find a profile pic or first image for the thumbnail
const archivePath = path.join(ARCHIVES_DIR, item.name); const archivePath = path.join(ARCHIVES_DIR, item.name);
const files = fs.readdirSync(archivePath); try {
console.log(`[API] Found archive: ${item.name} (${files.length} files)`); const files = fs.readdirSync(archivePath);
console.log(`[API] Found archive: ${item.name} (${files.length} files)`);
let thumbnail = ''; let thumbnail = '';
const profilePic = files.find(f => f.toLowerCase().includes('_profile_pic.jpg') || f.toLowerCase() === `${item.name.toLowerCase()}.jpg`); const profilePic = files.find(f => f.toLowerCase().includes('_profile_pic.jpg') || f.toLowerCase() === `${item.name.toLowerCase()}.jpg`);
if (profilePic) { if (profilePic) {
thumbnail = `/archives/${item.name}/${profilePic}`; thumbnail = `/archives/${item.name}/${profilePic}`;
} else { } else {
const firstImage = files.find(f => /\.(jpg|jpeg|png|webp)$/i.test(f)); const firstImage = files.find(f => /\.(jpg|jpeg|png|webp)$/i.test(f));
if (firstImage) thumbnail = `/archives/${item.name}/${firstImage}`; if (firstImage) thumbnail = `/archives/${item.name}/${firstImage}`;
}
return {
name: item.name,
thumbnail,
path: item.name,
fileCount: files.length
};
} catch (e) {
console.error(`[API] Could not read subdirectory ${item.name}:`, e);
return null;
} }
})
return { .filter(Boolean);
name: item.name,
thumbnail,
path: item.name,
fileCount: files.length
};
});
console.log(`[API] Returning ${archives.length} validated archives.`); console.log(`[API] Returning ${archives.length} validated archives.`);
res.json(archives); res.json(archives);
} catch (err) { } catch (err: any) {
console.error('[API] Error listing archives:', err); if (err.code === 'EACCES') {
res.status(500).json({ error: 'Failed to list archives' }); console.error(`[API] Permission Denied! The server (UID ${os.userInfo().uid}) cannot read ${ARCHIVES_DIR}.`);
console.error(`[API] Hint: If using Linux/Docker, check folder permissions (chmod 755) or SELinux context (append :z to your volume mount).`);
} else {
console.error('[API] Error listing archives:', err);
}
res.status(500).json({ error: 'Permission denied or failed to list archives' });
} }
}); });