Full analytics platform with MaxMind GeoLite2 IP-to-location resolution, cross-module dashboard (docs, video, photo), user drill-down, volunteer self-service stats, and ANALYTICS_ADMIN role with feature flag controls. - ANALYTICS_ADMIN role + ANALYTICS_ROLES group across backend and frontend - GeoIP service (MaxMind GeoLite2, lazy-loaded, graceful degradation) - Geo fields (country, region, city, lat/lng) on DocsPageView, VideoView, PhotoView - IP resolved to geo before SHA-256 hashing (privacy-preserving) - Unified analytics module: overview, geo, content, user engagement endpoints - 4 admin dashboard pages: Overview, Geography (Leaflet map), Content, Users - Volunteer MyAnalyticsPage for self-service activity stats - Settings UI: enableAnalytics, analyticsGeoEnabled, trackAuthenticatedUsers, retentionDays - Scheduled cleanup job respecting configurable retention period - config.sh: Analytics + MaxMind prompt in configure_features() - Control panel: enableAnalytics flag, template, discovery, wizard, detail page - Docker: geoip volume mount, MaxMind env vars, entrypoint auto-download - Nginx: X-Forwarded-For fix ($proxy_add_x_forwarded_for) for real client IP - Express trust proxy set to 2 for Pangolin/Newt tunnel chain - CORS updated for docs origin (cmlite.org + docs.cmlite.org) - Lander page: added docs-analytics tracking snippet - Prisma migration: 20260402100000_add_analytics_system Bunker Admin
36 lines
1.2 KiB
TypeScript
36 lines
1.2 KiB
TypeScript
import { prisma } from '../../config/database';
|
|
import { logger } from '../../utils/logger';
|
|
|
|
export const analyticsCleanupService = {
|
|
async cleanupAll(): Promise<void> {
|
|
// Read retention setting from SiteSettings
|
|
let retentionDays = 90;
|
|
try {
|
|
const settings = await prisma.siteSettings.findFirst();
|
|
if (settings?.analyticsRetentionDays) {
|
|
retentionDays = settings.analyticsRetentionDays;
|
|
}
|
|
} catch {
|
|
// Use default if settings unavailable
|
|
}
|
|
|
|
const cutoff = new Date();
|
|
cutoff.setDate(cutoff.getDate() - retentionDays);
|
|
|
|
const [docsDeleted, videoDeleted, photoDeleted] = await Promise.all([
|
|
prisma.docsPageView.deleteMany({ where: { createdAt: { lt: cutoff } } }),
|
|
prisma.videoView.deleteMany({ where: { createdAt: { lt: cutoff } } }),
|
|
prisma.photoView.deleteMany({ where: { viewedAt: { lt: cutoff } } }),
|
|
]);
|
|
|
|
const total = docsDeleted.count + videoDeleted.count + photoDeleted.count;
|
|
if (total > 0) {
|
|
logger.info(`Analytics cleanup: removed ${total} records older than ${retentionDays} days`, {
|
|
docs: docsDeleted.count,
|
|
video: videoDeleted.count,
|
|
photo: photoDeleted.count,
|
|
});
|
|
}
|
|
},
|
|
};
|