diff --git a/contributing.md b/contributing.md index 8c55c01149..e015804c8d 100644 --- a/contributing.md +++ b/contributing.md @@ -27,6 +27,8 @@ To get an overview of each microservice, read the respective microservice README - [meta-data](/src/meta-data/README.md) - [Predict](/src/predict/README.md) - [View](/src/view/README.md) + + Here are some resources to help you get started with open source contributions: - [How to Contribute to Open Source](https://opensource.guide/how-to-contribute/) diff --git a/k8s/auth-service/values-prod.yaml b/k8s/auth-service/values-prod.yaml index 708db5b069..0ecaff6f03 100644 --- a/k8s/auth-service/values-prod.yaml +++ b/k8s/auth-service/values-prod.yaml @@ -6,7 +6,7 @@ app: replicaCount: 3 image: repository: eu.gcr.io/airqo-250220/airqo-auth-api - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/auth-service/values-stage.yaml b/k8s/auth-service/values-stage.yaml index 9f22fb3cf5..0e784df33a 100644 --- a/k8s/auth-service/values-stage.yaml +++ b/k8s/auth-service/values-stage.yaml @@ -6,7 +6,7 @@ app: replicaCount: 2 image: repository: eu.gcr.io/airqo-250220/airqo-stage-auth-api - tag: stage-a9e01103-1734211833 + tag: stage-5d451842-1736194286 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/device-registry/values-prod.yaml b/k8s/device-registry/values-prod.yaml index 7dfc2317a0..56fbb0a9c9 100644 --- a/k8s/device-registry/values-prod.yaml +++ b/k8s/device-registry/values-prod.yaml @@ -6,7 +6,7 @@ app: replicaCount: 3 image: repository: eu.gcr.io/airqo-250220/airqo-device-registry-api - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/device-registry/values-stage.yaml b/k8s/device-registry/values-stage.yaml index af9412c43b..9db1b08f46 100644 --- a/k8s/device-registry/values-stage.yaml +++ b/k8s/device-registry/values-stage.yaml @@ -6,7 +6,7 @@ app: replicaCount: 2 image: repository: eu.gcr.io/airqo-250220/airqo-stage-device-registry-api - tag: stage-a3fd1cf0-1734679838 + tag: stage-4ca1c803-1736183154 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/exceedance/values-prod-airqo.yaml b/k8s/exceedance/values-prod-airqo.yaml index 3e42145920..67aaffe2bf 100644 --- a/k8s/exceedance/values-prod-airqo.yaml +++ b/k8s/exceedance/values-prod-airqo.yaml @@ -4,6 +4,6 @@ app: configmap: env-exceedance-production image: repository: eu.gcr.io/airqo-250220/airqo-exceedance-job - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' diff --git a/k8s/exceedance/values-prod-kcca.yaml b/k8s/exceedance/values-prod-kcca.yaml index b6fa474494..0fc9befa6e 100644 --- a/k8s/exceedance/values-prod-kcca.yaml +++ b/k8s/exceedance/values-prod-kcca.yaml @@ -4,6 +4,6 @@ app: configmap: env-exceedance-production image: repository: eu.gcr.io/airqo-250220/kcca-exceedance-job - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' diff --git a/k8s/incentives/values-prod.yaml b/k8s/incentives/values-prod.yaml index 249470f336..490edb9fd8 100644 --- a/k8s/incentives/values-prod.yaml +++ b/k8s/incentives/values-prod.yaml @@ -6,7 +6,7 @@ app: replicaCount: 3 image: repository: eu.gcr.io/airqo-250220/airqo-incentives-api - tag: prod-8ec03422-1731102123 + tag: prod-fbf1a3e5-1735852910 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/incentives/values-stage.yaml b/k8s/incentives/values-stage.yaml index cc0eb4ab99..d279b64810 100644 --- a/k8s/incentives/values-stage.yaml +++ b/k8s/incentives/values-stage.yaml @@ -6,13 +6,12 @@ app: replicaCount: 2 image: repository: eu.gcr.io/airqo-250220/airqo-stage-incentives-api - tag: stage-96d94ce4-1715349509 + tag: stage-6ffb3fe6-1735572850 nameOverride: '' fullnameOverride: '' podAnnotations: {} resources: limits: - cpu: 1000m memory: 500Mi requests: cpu: 100m diff --git a/k8s/insights/values-prod.yaml b/k8s/insights/values-prod.yaml index 090bd6c6e0..4bc3b517cb 100644 --- a/k8s/insights/values-prod.yaml +++ b/k8s/insights/values-prod.yaml @@ -6,7 +6,7 @@ images: api: eu.gcr.io/airqo-250220/airqo-insights-api celery: eu.gcr.io/airqo-250220/airqo-insights-celery celeryWorker: eu.gcr.io/airqo-250220/airqo-insights-celery-worker - tag: prod-015fde10-1730310056 + tag: prod-b1c7ea69-1735899837 api: name: airqo-insights-api label: insights-api diff --git a/k8s/insights/values-stage.yaml b/k8s/insights/values-stage.yaml index 441bce2842..7b943588b8 100644 --- a/k8s/insights/values-stage.yaml +++ b/k8s/insights/values-stage.yaml @@ -5,7 +5,7 @@ images: repositories: api: eu.gcr.io/airqo-250220/airqo-stage-insights-api celery: eu.gcr.io/airqo-250220/airqo-stage-insights-celery - tag: stage-38af6de7-1730309744 + tag: stage-6ffb3fe6-1735572850 api: name: airqo-stage-insights-api label: sta-alytics-api @@ -14,7 +14,6 @@ api: podAnnotations: {} resources: limits: - cpu: 100m memory: 600Mi requests: cpu: 10m diff --git a/k8s/kafka/topics/kafka-topics.yaml b/k8s/kafka/topics/kafka-topics.yaml index bd594f7b29..c46dbd67a7 100644 --- a/k8s/kafka/topics/kafka-topics.yaml +++ b/k8s/kafka/topics/kafka-topics.yaml @@ -307,3 +307,17 @@ spec: replicas: 2 config: retention.ms: 18000000 + +--- +apiVersion: kafka.strimzi.io/v1beta2 +kind: KafkaTopic +metadata: + name: airqo.forecasts + namespace: message-broker + labels: + strimzi.io/cluster: kafka-cluster +spec: + partitions: 2 + replicas: 2 + config: + retention.ms: 18000000 diff --git a/k8s/meta-data/values-prod.yaml b/k8s/meta-data/values-prod.yaml index fd66986e3a..44842dd224 100644 --- a/k8s/meta-data/values-prod.yaml +++ b/k8s/meta-data/values-prod.yaml @@ -8,7 +8,7 @@ images: repositories: api: eu.gcr.io/airqo-250220/airqo-meta-data-api sitesConsumer: eu.gcr.io/airqo-250220/airqo-meta-data-sites-consumer - tag: prod-5244fe67-1730977223 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/meta-data/values-stage.yaml b/k8s/meta-data/values-stage.yaml index f9143958cb..b091a1d930 100644 --- a/k8s/meta-data/values-stage.yaml +++ b/k8s/meta-data/values-stage.yaml @@ -8,13 +8,12 @@ images: repositories: api: eu.gcr.io/airqo-250220/airqo-stage-meta-data-api sitesConsumer: eu.gcr.io/airqo-250220/airqo-stage-meta-data-sites-consumer - tag: stage-38af6de7-1730309744 + tag: stage-6ffb3fe6-1735572850 nameOverride: '' fullnameOverride: '' podAnnotations: {} resources: limits: - cpu: 400m memory: 700Mi requests: cpu: 50m diff --git a/k8s/predict/values-prod.yaml b/k8s/predict/values-prod.yaml index 5059f4a050..0189dc34a5 100644 --- a/k8s/predict/values-prod.yaml +++ b/k8s/predict/values-prod.yaml @@ -7,7 +7,7 @@ images: predictJob: eu.gcr.io/airqo-250220/airqo-predict-job trainJob: eu.gcr.io/airqo-250220/airqo-train-job predictPlaces: eu.gcr.io/airqo-250220/airqo-predict-places-air-quality - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 api: name: airqo-prediction-api label: prediction-api diff --git a/k8s/predict/values-stage.yaml b/k8s/predict/values-stage.yaml index 11dd2c8569..dfb2ec36e7 100644 --- a/k8s/predict/values-stage.yaml +++ b/k8s/predict/values-stage.yaml @@ -16,7 +16,6 @@ api: podAnnotations: {} resources: limits: - cpu: 50m memory: 200Mi requests: cpu: 5m diff --git a/k8s/spatial/values-prod.yaml b/k8s/spatial/values-prod.yaml index 213e46a753..ed2c506949 100644 --- a/k8s/spatial/values-prod.yaml +++ b/k8s/spatial/values-prod.yaml @@ -6,7 +6,7 @@ app: replicaCount: 3 image: repository: eu.gcr.io/airqo-250220/airqo-spatial-api - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/spatial/values-stage.yaml b/k8s/spatial/values-stage.yaml index b50018fa95..359a8ba35c 100644 --- a/k8s/spatial/values-stage.yaml +++ b/k8s/spatial/values-stage.yaml @@ -6,13 +6,12 @@ app: replicaCount: 2 image: repository: eu.gcr.io/airqo-250220/airqo-stage-spatial-api - tag: stage-1595bb07-1734517712 + tag: stage-6ffb3fe6-1735572850 nameOverride: '' fullnameOverride: '' podAnnotations: {} resources: limits: - cpu: 100m memory: 400Mi requests: cpu: 10m diff --git a/k8s/view/values-prod.yaml b/k8s/view/values-prod.yaml index 52344c6432..674a6627bc 100644 --- a/k8s/view/values-prod.yaml +++ b/k8s/view/values-prod.yaml @@ -5,7 +5,7 @@ images: repositories: api: eu.gcr.io/airqo-250220/airqo-view-api messageBroker: eu.gcr.io/airqo-250220/airqo-view-message-broker - tag: prod-1f061bdf-1731360102 + tag: prod-c1d1fce6-1735895602 api: name: airqo-view-api label: view-api diff --git a/k8s/view/values-stage.yaml b/k8s/view/values-stage.yaml index d5b5f53b12..b0d28ccd78 100644 --- a/k8s/view/values-stage.yaml +++ b/k8s/view/values-stage.yaml @@ -13,7 +13,6 @@ api: podAnnotations: {} resources: limits: - cpu: 100m memory: 1000Mi requests: cpu: 5m diff --git a/k8s/website/values-prod.yaml b/k8s/website/values-prod.yaml index 5dfa6eb518..abb085d889 100644 --- a/k8s/website/values-prod.yaml +++ b/k8s/website/values-prod.yaml @@ -6,7 +6,7 @@ app: replicaCount: 3 image: repository: eu.gcr.io/airqo-250220/airqo-website-api - tag: prod-ce82c1ac-1734679890 + tag: prod-c1d1fce6-1735895602 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/website/values-stage.yaml b/k8s/website/values-stage.yaml index a85c7a78ca..51dddc2619 100644 --- a/k8s/website/values-stage.yaml +++ b/k8s/website/values-stage.yaml @@ -6,13 +6,12 @@ app: replicaCount: 2 image: repository: eu.gcr.io/airqo-250220/airqo-stage-website-api - tag: stage-7a1d5dd3-1733836788 + tag: stage-6ffb3fe6-1735572850 nameOverride: '' fullnameOverride: '' podAnnotations: {} resources: limits: - cpu: 100m memory: 400Mi requests: cpu: 10m diff --git a/k8s/workflows/values-prod.yaml b/k8s/workflows/values-prod.yaml index c9b7986601..16ca68b66b 100644 --- a/k8s/workflows/values-prod.yaml +++ b/k8s/workflows/values-prod.yaml @@ -10,7 +10,7 @@ images: initContainer: eu.gcr.io/airqo-250220/airqo-workflows-xcom redisContainer: eu.gcr.io/airqo-250220/airqo-redis containers: eu.gcr.io/airqo-250220/airqo-workflows - tag: prod-ce82c1ac-1734679890 + tag: prod-6d2ca253-1736194337 nameOverride: '' fullnameOverride: '' podAnnotations: {} diff --git a/k8s/workflows/values-stage.yaml b/k8s/workflows/values-stage.yaml index 146d3fdd3b..61475c9ad3 100644 --- a/k8s/workflows/values-stage.yaml +++ b/k8s/workflows/values-stage.yaml @@ -10,7 +10,7 @@ images: initContainer: eu.gcr.io/airqo-250220/airqo-stage-workflows-xcom redisContainer: eu.gcr.io/airqo-250220/airqo-stage-redis containers: eu.gcr.io/airqo-250220/airqo-stage-workflows - tag: stage-e05bdc71-1734549206 + tag: stage-6ffb3fe6-1735572850 nameOverride: '' fullnameOverride: '' podAnnotations: {} @@ -20,28 +20,24 @@ resources: cpu: 125m memory: 500Mi limits: - cpu: 1000m memory: 1500Mi scheduler: requests: cpu: 125m memory: 500Mi limits: - cpu: 2000m memory: 2000Mi celery: requests: cpu: 125m memory: 500Mi limits: - cpu: 2000m memory: 2000Mi redis: requests: cpu: 50m memory: 125Mi limits: - cpu: 1000m memory: 2000Mi volumeMounts: - name: config-volume diff --git a/src/auth-service/bin/jobs/profile-picture-update-job.js b/src/auth-service/bin/jobs/profile-picture-update-job.js new file mode 100644 index 0000000000..be59b454bf --- /dev/null +++ b/src/auth-service/bin/jobs/profile-picture-update-job.js @@ -0,0 +1,198 @@ +const cron = require("node-cron"); +const NetworkModel = require("@models/Network"); +const GroupModel = require("@models/Group"); +const mongoose = require("mongoose"); +const constants = require("@config/constants"); +const log4js = require("log4js"); +const { logText, logObject } = require("@utils/log"); +const logger = log4js.getLogger( + `${constants.ENVIRONMENT} -- bin/jobs/profile-picture-update-job` +); +const stringify = require("@utils/stringify"); +const isEmpty = require("is-empty"); + +// Configuration +const BATCH_SIZE = 100; +const DEFAULT_PROFILE_PICTURE = constants.DEFAULT_ORGANISATION_PROFILE_PICTURE; +const MAX_CONCURRENT_OPERATIONS = 5; // Limit concurrent operations + +// Function to validate URL +const isValidUrl = (url) => { + const urlRegex = + /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; + return urlRegex.test(url); +}; + +// Function to validate the default profile picture +const validateDefaultProfilePicture = () => { + if (!isValidUrl(DEFAULT_PROFILE_PICTURE)) { + logger.error( + `🚨 Aborting profile picture update: Invalid default profile picture URL` + ); + return false; + } + return true; +}; + +// Generic function to process items in batches with controlled concurrency +async function processBatch(items, processFunction) { + const chunks = []; + for (let i = 0; i < items.length; i += MAX_CONCURRENT_OPERATIONS) { + chunks.push(items.slice(i, i + MAX_CONCURRENT_OPERATIONS)); + } + + for (const chunk of chunks) { + await Promise.all(chunk.map(processFunction)); + } +} + +// Function to update a single network +async function updateNetworkProfilePicture(network) { + try { + await NetworkModel("airqo").findByIdAndUpdate( + network._id, + { + $set: { net_profile_picture: DEFAULT_PROFILE_PICTURE }, + }, + { + new: true, + runValidators: true, + } + ); + logger.info(`✅ Updated profile picture for network: ${network.net_name}`); + return { success: true, type: "network", name: network.net_name }; + } catch (error) { + logger.error( + `🐛 Failed to update profile picture for network ${ + network.net_name + }: ${stringify(error)}` + ); + return { success: false, type: "network", name: network.net_name, error }; + } +} + +// Function to update a single group +async function updateGroupProfilePicture(group) { + try { + await GroupModel("airqo").findByIdAndUpdate( + group._id, + { + $set: { grp_profile_picture: DEFAULT_PROFILE_PICTURE }, + }, + { + new: true, + runValidators: true, + } + ); + logger.info(`✅ Updated profile picture for group: ${group.grp_title}`); + return { success: true, type: "group", name: group.grp_title }; + } catch (error) { + logger.error( + `🐛 Failed to update profile picture for group ${ + group.grp_title + }: ${stringify(error)}` + ); + return { success: false, type: "group", name: group.grp_title, error }; + } +} + +// Main function to update profile pictures +async function updateProfilePictures() { + // Validate default profile picture before proceeding + if (!validateDefaultProfilePicture()) { + return; + } + + const stats = { + networks: { processed: 0, success: 0, error: 0 }, + groups: { processed: 0, success: 0, error: 0 }, + }; + + try { + const startTime = Date.now(); + logger.info("🚀 Starting profile picture update process"); + + // Process both networks and groups in parallel + await Promise.all([ + // Update Networks + (async () => { + let skip = 0; + while (true) { + const networks = await NetworkModel("airqo") + .find({ + $or: [ + { net_profile_picture: { $exists: false } }, + { net_profile_picture: null }, + ], + }) + .limit(BATCH_SIZE) + .skip(skip) + .select("_id net_name net_profile_picture") + .lean(); + + if (networks.length === 0) break; + + const results = await processBatch( + networks, + updateNetworkProfilePicture + ); + stats.networks.processed += networks.length; + skip += BATCH_SIZE; + } + })(), + + // Update Groups + (async () => { + let skip = 0; + while (true) { + const groups = await GroupModel("airqo") + .find({ + $or: [ + { grp_profile_picture: { $exists: false } }, + { grp_profile_picture: null }, + ], + }) + .limit(BATCH_SIZE) + .skip(skip) + .select("_id grp_title grp_profile_picture") + .lean(); + + if (groups.length === 0) break; + + const results = await processBatch(groups, updateGroupProfilePicture); + stats.groups.processed += groups.length; + skip += BATCH_SIZE; + } + })(), + ]); + + const duration = (Date.now() - startTime) / 1000; + logText(` + 📊 Profile picture update completed in ${duration} seconds + Networks processed: ${stats.networks.processed} + Groups processed: ${stats.groups.processed} + `); + logger.info(` + 📊 Profile picture update completed in ${duration} seconds + Networks processed: ${stats.networks.processed} + Groups processed: ${stats.groups.processed} + `); + } catch (error) { + logObject("error", error); + logger.error(`🐛🐛 Error in updateProfilePictures: ${stringify(error)}`); + } +} + +// // Schedule the job to run daily at midnight +const schedule = "0 0 * * *"; +cron.schedule(schedule, updateProfilePictures, { + scheduled: true, + timezone: "Africa/Nairobi", +}); + +// Export for manual execution if needed +module.exports = { + updateProfilePictures, + updateNetworkProfilePicture, + updateGroupProfilePicture, +}; diff --git a/src/auth-service/bin/jobs/update-user-activities-job.js b/src/auth-service/bin/jobs/update-user-activities-job.js new file mode 100644 index 0000000000..8b28ab556e --- /dev/null +++ b/src/auth-service/bin/jobs/update-user-activities-job.js @@ -0,0 +1,325 @@ +const cron = require("node-cron"); +const constants = require("@config/constants"); +const log4js = require("log4js"); +const logger = log4js.getLogger( + `${constants.ENVIRONMENT} -- user-activity-job` +); +const { logText, logObject } = require("@utils/log"); +const { LogModel } = require("@models/log"); +const ActivityModel = require("@models/Activity"); + +const BATCH_SIZE = 500; +const MAX_RETRIES = 3; +const CONCURRENT_BATCH_LIMIT = 5; + +// Helper function for calculating engagement score +function calculateEngagementScore({ + totalActions, + uniqueServices, + uniqueEndpoints, + activityDays, +}) { + const actionsPerDay = totalActions / Math.max(activityDays, 30); + const serviceDiversity = uniqueServices / 20; // Normalize against max expected services + const endpointDiversity = Math.min(uniqueEndpoints / 10, 1); + return ( + (actionsPerDay * 0.4 + serviceDiversity * 0.3 + endpointDiversity * 0.3) * + 100 + ); +} + +function calculateEngagementTier(score) { + if (score >= 80) return "Elite User"; + if (score >= 65) return "Super User"; + if (score >= 45) return "High Engagement"; + if (score >= 25) return "Moderate Engagement"; + return "Low Engagement"; +} + +async function processDailyStats(logs) { + const dailyGroups = new Map(); + + for await (const log of logs) { + const date = new Date(log.timestamp); + date.setHours(0, 0, 0, 0); + const dateKey = date.toISOString(); + + if (!dailyGroups.has(dateKey)) { + dailyGroups.set(dateKey, { + date, + totalActions: 0, + services: new Map(), + endpoints: new Map(), + }); + } + + const dayStats = dailyGroups.get(dateKey); + dayStats.totalActions++; + + dayStats.services.set( + log.meta.service, + (dayStats.services.get(log.meta.service) || 0) + 1 + ); + dayStats.endpoints.set( + log.meta.endpoint, + (dayStats.endpoints.get(log.meta.endpoint) || 0) + 1 + ); + } + + return Array.from(dailyGroups.values()).map((stats) => ({ + date: stats.date, + totalActions: stats.totalActions, + services: Array.from(stats.services.entries()).map(([name, count]) => ({ + name, + count, + })), + endpoints: Array.from(stats.endpoints.entries()).map(([name, count]) => ({ + name, + count, + })), + })); +} + +async function processMonthlyStats(dailyStats) { + const monthlyGroups = new Map(); + + for (const day of dailyStats) { + const year = day.date.getFullYear(); + const month = day.date.getMonth() + 1; + const key = `${year}-${month}`; + + if (!monthlyGroups.has(key)) { + monthlyGroups.set(key, { + year, + month, + totalActions: 0, + services: new Set(), + endpoints: new Set(), + serviceCount: new Map(), + firstActivity: day.date, + lastActivity: day.date, + }); + } + + const monthStats = monthlyGroups.get(key); + monthStats.totalActions += day.totalActions; + + day.services.forEach(({ name, count }) => { + monthStats.services.add(name); + const currentCount = monthStats.serviceCount.get(name) || 0; + monthStats.serviceCount.set(name, currentCount + count); + }); + + day.endpoints.forEach(({ name }) => { + monthStats.endpoints.add(name); + }); + + if (day.date < monthStats.firstActivity) + monthStats.firstActivity = day.date; + if (day.date > monthStats.lastActivity) monthStats.lastActivity = day.date; + } + + return Array.from(monthlyGroups.values()).map((month) => { + const topServices = Array.from(month.serviceCount.entries()) + .map(([name, count]) => ({ name, count })) + .sort((a, b) => b.count - a.count) + .slice(0, 5); + + const activityDays = Math.ceil( + (month.lastActivity - month.firstActivity) / (1000 * 60 * 60 * 24) + ); + + const engagementScore = calculateEngagementScore({ + totalActions: month.totalActions, + uniqueServices: month.services.size, + uniqueEndpoints: month.endpoints.size, + activityDays, + }); + + return { + year: month.year, + month: month.month, + totalActions: month.totalActions, + uniqueServices: Array.from(month.services), + uniqueEndpoints: Array.from(month.endpoints), + topServices, + firstActivity: month.firstActivity, + lastActivity: month.lastActivity, + engagementScore, + engagementTier: calculateEngagementTier(engagementScore), + }; + }); +} +async function getLastProcessedLogs(tenant) { + const lastProcessedMap = new Map(); + + const cursor = await ActivityModel(tenant) + .find({}, { email: 1, lastProcessedLog: 1 }) + .lean() + .cursor(); + + for await (const doc of cursor) { + lastProcessedMap.set(doc.email, doc.lastProcessedLog); + } + + return lastProcessedMap; +} + +async function processUserLogs( + tenant, + email, + lastProcessedLog, + retryCount = 0 +) { + try { + const query = { + "meta.email": email, + "meta.service": { $nin: ["unknown", "none", "", null] }, + }; + if (lastProcessedLog) query._id = { $gt: lastProcessedLog }; + + const cursor = await LogModel(tenant) + .find(query) + .sort({ timestamp: 1 }) + .lean() + .cursor(); + + const logs = []; + for await (const log of cursor) { + logs.push(log); + } + + if (logs.length === 0) return null; + + const dailyStats = await processDailyStats(logs); + const monthlyStats = await processMonthlyStats(dailyStats); + + return { + updateOne: { + filter: { email, tenant }, + update: { + $push: { + dailyStats: { $each: dailyStats, $sort: { date: 1 } }, + monthlyStats: { $each: monthlyStats, $sort: { year: 1, month: 1 } }, + }, + $set: { + lastProcessedLog: logs[logs.length - 1]._id, + username: logs[logs.length - 1].meta.username, + "overallStats.lastActivity": logs[logs.length - 1].timestamp, + tenant, + }, + $min: { "overallStats.firstActivity": logs[0].timestamp }, + $inc: { "overallStats.totalActions": logs.length }, + }, + upsert: true, + }, + }; + } catch (error) { + if (retryCount < MAX_RETRIES) { + await new Promise((resolve) => + setTimeout(resolve, 1000 * (retryCount + 1)) + ); + return processUserLogs(tenant, email, lastProcessedLog, retryCount + 1); + } + throw error; + } +} + +async function processEmailBatch(batch, tenant, lastProcessedMap) { + const updates = await Promise.all( + batch.map((email) => + processUserLogs(tenant, email, lastProcessedMap.get(email)).catch( + (error) => { + logger.error(`Error processing user ${email}: ${error.message}`); + return null; + } + ) + ) + ); + + const validUpdates = updates.filter(Boolean); + if (validUpdates.length > 0) { + await ActivityModel(tenant).bulkWrite(validUpdates, { ordered: false }); + } + return validUpdates.length; +} + +async function updateUserActivities({ tenant = "airqo" } = {}, retryCount = 0) { + const startTime = Date.now(); + let processedCount = 0; + let errorCount = 0; + const timeout = 30000; // 30 seconds + + try { + logText("Starting user activity update job"); + + const query = { + "meta.email": { $exists: true, $ne: null }, + "meta.service": { $nin: ["unknown", "none", "", null] }, + }; + + const uniqueEmails = await LogModel(tenant) + .distinct("meta.email", query) + .maxTimeMS(timeout); + if (uniqueEmails.length === 0) + return { success: true, processedCount: 0, errorCount: 0, duration: 0 }; + + const lastProcessedMap = await getLastProcessedLogs(tenant); + const batches = []; + + for (let i = 0; i < uniqueEmails.length; i += BATCH_SIZE) { + batches.push(uniqueEmails.slice(i, i + BATCH_SIZE)); + } + + for (let i = 0; i < batches.length; i += CONCURRENT_BATCH_LIMIT) { + const currentBatches = batches.slice(i, i + CONCURRENT_BATCH_LIMIT); + const results = await Promise.all( + currentBatches.map((batch) => + processEmailBatch(batch, tenant, lastProcessedMap) + ) + ); + + processedCount += results.reduce((a, b) => a + b, 0); + + logger.info( + `Processed batch ${i + 1}/${batches.length}: ${processedCount} users` + ); + + await new Promise((resolve) => setTimeout(resolve, 1000)); + } + + const duration = (Date.now() - startTime) / 1000; + logText( + `Job completed. Processed ${processedCount} users in ${duration}s. Errors: ${errorCount}` + ); + logger.info( + `Job completed. Processed ${processedCount} users in ${duration}s. Errors: ${errorCount}` + ); + return { success: true, processedCount, errorCount, duration }; + } catch (error) { + logger.error(`Fatal error in updateUserActivities: ${error.message}`); + if (error.message.includes("timed out") && retryCount < MAX_RETRIES) { + const delay = Math.pow(2, retryCount) * 1000; + await new Promise((resolve) => setTimeout(resolve, delay)); + return updateUserActivities({ tenant }, retryCount + 1); + } + throw error; + } +} + +async function runUpdateUserActivities() { + try { + const result = await updateUserActivities({ tenant: "airqo" }); + logObject("Run result", result); + } catch (error) { + logObject("Run error", error); + logger.error(`Cron job error: ${error.message}`); + } +} + +cron.schedule("0 * * * *", runUpdateUserActivities, { + scheduled: true, + timezone: "Africa/Nairobi", +}); + +module.exports = { updateUserActivities }; diff --git a/src/auth-service/bin/server.js b/src/auth-service/bin/server.js index a3002f83dc..94e03cdffd 100644 --- a/src/auth-service/bin/server.js +++ b/src/auth-service/bin/server.js @@ -24,6 +24,8 @@ require("@bin/jobs/token-expiration-job"); require("@bin/jobs/incomplete-profile-job"); require("@bin/jobs/preferences-log-job"); require("@bin/jobs/preferences-update-job"); +// require("@bin/jobs/update-user-activities-job"); +require("@bin/jobs/profile-picture-update-job"); const log4js = require("log4js"); const debug = require("debug")("auth-service:server"); const isEmpty = require("is-empty"); diff --git a/src/auth-service/config/global/db-projections.js b/src/auth-service/config/global/db-projections.js index 63b0e5f79d..bcfd78edd1 100644 --- a/src/auth-service/config/global/db-projections.js +++ b/src/auth-service/config/global/db-projections.js @@ -6,6 +6,7 @@ const dbProjections = { net_email: 1, net_website: 1, net_category: 1, + net_profile_picture: 1, net_status: 1, net_phoneNumber: 1, net_name: 1, @@ -647,6 +648,7 @@ const dbProjections = { grp_tasks: 1, grp_description: 1, grp_website: 1, + grp_profile_picture: 1, grp_industry: 1, grp_country: 1, grp_timezone: 1, @@ -732,6 +734,126 @@ const dbProjections = { } return projection; }, + ACTIVITIES_INCLUSION_PROJECTION: { + _id: 1, + email: 1, + username: 1, + tenant: 1, + dailyStats: 1, + overallStats: 1, + createdAt: 1, + totalMonthlyActions: { + $cond: { + if: { $isArray: "$monthlyStats" }, + then: { $sum: "$monthlyStats.totalActions" }, + else: 0, + }, + }, + // Calculate average engagement score across all months + averageEngagement: { + $cond: { + if: { $isArray: "$monthlyStats" }, + then: { + $avg: "$monthlyStats.engagementScore", + }, + else: 0, + }, + }, + // Get the most recent monthly stats + currentMonthStats: { + $arrayElemAt: [ + { + $filter: { + input: "$monthlyStats", + as: "month", + cond: { + $and: [ + { $eq: ["$$month.year", { $year: new Date() }] }, + { $eq: ["$$month.month", { $month: new Date() }] }, + ], + }, + }, + }, + 0, + ], + }, + // Get today's stats + todayStats: { + $arrayElemAt: [ + { + $filter: { + input: "$dailyStats", + as: "day", + cond: { + $eq: [ + { $dateToString: { format: "%Y-%m-%d", date: "$$day.date" } }, + { $dateToString: { format: "%Y-%m-%d", date: new Date() } }, + ], + }, + }, + }, + 0, + ], + }, + }, + ACTIVITIES_EXCLUSION_PROJECTION: function (category) { + const initialProjection = { + __v: 0, + "dailyStats.__v": 0, + "monthlyStats.__v": 0, + // Exclude specific fields from dailyStats when not needed + "dailyStats.endpoints._id": 0, + "dailyStats.services._id": 0, + // Exclude specific fields from monthlyStats when not needed + "monthlyStats.topServices._id": 0, + lastProcessedLog: 0, + }; + + let projection = Object.assign({}, initialProjection); + + switch (category) { + case "summary": + // For summary view, exclude detailed stats + projection = Object.assign({}, projection, { + dailyStats: 0, + monthlyStats: 0, + lastProcessedLog: 0, + }); + break; + + case "daily": + // For daily view, exclude monthly stats + projection = Object.assign({}, projection, { + monthlyStats: 0, + lastProcessedLog: 0, + }); + break; + + case "monthly": + // For monthly view, exclude daily stats + projection = Object.assign({}, projection, { + dailyStats: 0, + lastProcessedLog: 0, + }); + break; + + case "minimal": + // For minimal view, only show essential fields + projection = { + dailyStats: 0, + monthlyStats: 0, + lastProcessedLog: 0, + createdAt: 0, + updatedAt: 0, + __v: 0, + }; + break; + + // Default case keeps all fields except those in initialProjection + } + + return projection; + }, LOCATION_HISTORIES_INCLUSION_PROJECTION: { _id: 1, name: 1, diff --git a/src/auth-service/config/global/envs.js b/src/auth-service/config/global/envs.js index 0c0f79d332..98898f4c72 100644 --- a/src/auth-service/config/global/envs.js +++ b/src/auth-service/config/global/envs.js @@ -61,5 +61,7 @@ const envs = { SESSION_SECRET: process.env.SESSION_SECRET, HARDWARE_AND_DS_EMAILS: process.env.HARDWARE_AND_DS_EMAILS, PLATFORM_AND_DS_EMAILS: process.env.PLATFORM_AND_DS_EMAILS, + DEFAULT_ORGANISATION_PROFILE_PICTURE: + process.env.DEFAULT_ORGANISATION_PROFILE_PICTURE, }; module.exports = envs; diff --git a/src/auth-service/config/new_database.js b/src/auth-service/config/new_database.js new file mode 100644 index 0000000000..058b68c5e4 --- /dev/null +++ b/src/auth-service/config/new_database.js @@ -0,0 +1,115 @@ +const mongoose = require("mongoose"); +const constants = require("./constants"); +const log4js = require("log4js"); +const logger = log4js.getLogger(`${constants.ENVIRONMENT} -- config-database`); + +const options = { + useNewUrlParser: true, + useUnifiedTopology: true, + autoIndex: true, + poolSize: 20, + bufferMaxEntries: 0, + connectTimeoutMS: 30000, + socketTimeoutMS: 45000, + serverSelectionTimeoutMS: 30000, + keepAlive: true, + keepAliveInitialDelay: 300000, + maxPoolSize: 50, + minPoolSize: 10, + maxIdleTimeMS: 60000, + writeConcern: { + w: "majority", + j: true, + wtimeout: 30000, + }, +}; + +const connectionMap = new Map(); + +const connect = (dbName = constants.DB_NAME) => { + const URI = constants.MONGO_URI; + return mongoose.createConnection(URI, { ...options, dbName }); +}; + +const handleConnection = (db, dbName) => { + db.on("connected", () => { + // logger.info(`Connected to database: ${dbName}`); + }); + + db.on("error", (err) => { + // logger.error(`Database connection error for ${dbName}: ${err.message}`); + setTimeout(() => reconnect(dbName), 5000); + }); + + db.on("disconnected", () => { + // logger.warn(`Database disconnected: ${dbName}`); + setTimeout(() => reconnect(dbName), 5000); + }); + + return db; +}; + +const reconnect = async (dbName) => { + try { + const existingConn = connectionMap.get(dbName); + if (existingConn && existingConn.readyState !== 1) { + const db = connect(dbName); + handleConnection(db, dbName); + connectionMap.set(dbName, db); + } + } catch (error) { + // logger.error(`Reconnection failed for ${dbName}: ${error.message}`); + } +}; + +const connectToMongoDB = () => { + try { + const db = connect(); + handleConnection(db, constants.DB_NAME); + connectionMap.set(constants.DB_NAME, db); + + process.on("unhandledRejection", (reason, promise) => { + // logger.error("Unhandled Rejection:", reason); + }); + + process.on("uncaughtException", (err) => { + // logger.error("Uncaught Exception:", err); + }); + + return db; + } catch (error) { + // logger.error(`Database initialization error: ${error.message}`); + throw error; + } +}; + +const getTenantDB = (tenantId, modelName, schema) => { + try { + const dbName = `${constants.DB_NAME}_${tenantId}`; + let db = connectionMap.get(dbName); + + if (!db || db.readyState !== 1) { + db = connect(dbName); + handleConnection(db, dbName); + connectionMap.set(dbName, db); + } + + if (!db.models[modelName]) { + db.model(modelName, schema); + } + + return db; + } catch (error) { + // logger.error(`Error getting tenant DB: ${error.message}`); + throw error; + } +}; + +const getModelByTenant = (tenantId, modelName, schema) => { + const tenantDb = getTenantDB(tenantId, modelName, schema); + return tenantDb.model(modelName); +}; + +const mongodb = connectToMongoDB(); + +module.exports = { getModelByTenant, getTenantDB, connectToMongoDB }; diff --git a/src/auth-service/controllers/create-analytics.js b/src/auth-service/controllers/create-analytics.js new file mode 100644 index 0000000000..e1803536c0 --- /dev/null +++ b/src/auth-service/controllers/create-analytics.js @@ -0,0 +1,840 @@ +const createAnalyticsUtil = require("@utils/create-analytics"); +const constants = require("@config/constants"); +const { isEmpty } = require("lodash"); +const httpStatus = require("http-status"); +const { extractErrorsFromRequest, HttpError } = require("@utils/errors"); +const log4js = require("log4js"); +const logger = log4js.getLogger( + `${constants.ENVIRONMENT} -- create-analytics-controller` +); +const { logText, logObject } = require("@utils/log"); + +function handleResponse({ + result, + key = "data", + errorKey = "errors", + res, +} = {}) { + if (!result) { + return; + } + + const isSuccess = result.success; + const defaultStatus = isSuccess + ? httpStatus.OK + : httpStatus.INTERNAL_SERVER_ERROR; + + const defaultMessage = isSuccess + ? "Operation Successful" + : "Internal Server Error"; + + const status = result.status !== undefined ? result.status : defaultStatus; + const message = + result.message !== undefined ? result.message : defaultMessage; + const data = result.data !== undefined ? result.data : []; + const errors = isSuccess + ? undefined + : result.errors !== undefined + ? result.errors + : { message: "Internal Server Error" }; + + return res.status(status).json({ message, [key]: data, [errorKey]: errors }); +} + +const analytics = { + send: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError( + "bad request errors", + httpStatus.BAD_REQUEST, + res, + errors + ) + ); + return; + } + + const request = Object.assign({}, req); + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.sendYearEndEmails(request); + + if (result.success) { + res.status(result.status || httpStatus.OK).json({ + success: true, + message: result.message, + }); + } else { + res.status(result.status || httpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: result.message, + }); + } + } catch (error) { + logger.error(`🐛🐛 Year-End Email Error: ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + fetchUserStats: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + + const { body, query } = req; + const { emails } = body; + const { year } = query; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + + const request = { + emails, + year, + tenant: isEmpty(req.query.tenant) ? defaultTenant : req.query.tenant, + }; + + const result = await createAnalyticsUtil.fetchUserStats(request); + + if (result) { + res.status(result.status || httpStatus.OK).json({ + success: true, + message: result.message || "Successfully retrieved the User Stats", + stats: result, + }); + } else { + res.status(result.status || httpStatus.INTERNAL_SERVER_ERROR).json({ + success: false, + message: result.message || "No Stats Available for this User", + }); + } + } catch (error) { + logger.error(`🐛🐛 fetchUserStats Error: ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + validateEnvironment: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + + const { body, query } = req; + const { year } = query; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + + const validation = await createAnalyticsUtil.validateEnvironmentData({ + tenant: isEmpty(req.query.tenant) ? defaultTenant : req.query.tenant, + year, + }); + res.json(validation); + } catch (error) { + logger.error(`🐛🐛 validateEnvironment Error: ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + listStatistics: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors) + ); + return; + } + + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + const tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.listStatistics(tenant, next); + + if (isEmpty(result) || res.headersSent) { + return; + } + + if (result.success === true) { + const status = result.status ? result.status : httpStatus.OK; + return res.status(status).json({ + success: true, + message: result.message, + users_stats: result.data, + }); + } else if (result.success === false) { + const status = result.status + ? result.status + : httpStatus.INTERNAL_SERVER_ERROR; + return res.status(status).json({ + success: false, + message: result.message, + errors: { + message: result.errors + ? result.errors + : { message: "Internal Server Error" }, + }, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, + listLogs: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.listLogs(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "users_stats", + res, + }); + } + } catch (error) { + logObject("error", error); + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, + listActivities: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.listActivities(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "user_activities", + res, + }); + } + } catch (error) { + logObject("error", error); + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, + getUserStats: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getUserStats(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } + logObject("result", result); + if (result.success === true) { + const status = result.status ? result.status : httpStatus.OK; + return res.status(status).json({ + success: true, + message: result.message, + users_stats: result.data, + }); + } else if (result.success === false) { + const status = result.status + ? result.status + : httpStatus.INTERNAL_SERVER_ERROR; + return res.status(status).json({ + success: false, + message: result.message, + errors: result.errors + ? result.errors + : { message: "Internal Server Errors" }, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, + // User Engagement Functions + getUserEngagement: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getUserEngagement(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "user_engagements", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + getEngagementMetrics: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getEngagementMetrics( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "engagement_metrics", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Activity Analysis Functions + getActivityReport: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getActivityReport(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "activity_report", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Cohort Analysis Functions + getCohortAnalysis: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getCohortAnalysis(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "cohort_analysis", + res, + }); + } + } catch (error) { + logObject("error", error); + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Predictive Analytics Functions + getPredictiveAnalytics: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getPredictiveAnalytics( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "predictive_analytics", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Service Adoption Functions + getServiceAdoption: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getServiceAdoption( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "service_adoption", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Benchmark Functions + getBenchmarks: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getBenchmarks(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "benchmarks", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Top Users Functions + getTopUsers: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getTopUsers(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "top_users", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Aggregated Analytics Functions + getAggregatedAnalytics: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getAggregatedAnalytics( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "aggregated_analytics", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Retention Analysis Functions + getRetentionAnalysis: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getRetentionAnalysis( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "retention_analysis", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Health Score Functions + getEngagementHealth: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getEngagementHealth( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "engagement_health", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Behavior Pattern Functions + getBehaviorPatterns: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_req, res, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createAnalyticsUtil.getBehaviorPatterns( + request, + next + ); + + if (isEmpty(result) || res.headersSent) { + return; + } else { + handleResponse({ + result, + key: "behavior_patterns", + res, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, +}; + +module.exports = analytics; diff --git a/src/auth-service/controllers/create-group.js b/src/auth-service/controllers/create-group.js index 8137f11581..3063ab39bb 100644 --- a/src/auth-service/controllers/create-group.js +++ b/src/auth-service/controllers/create-group.js @@ -515,6 +515,58 @@ const createGroup = { return; } }, + setManager: async (req, res, next) => { + try { + logText("set the manager...."); + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors) + ); + return; + } + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + const result = await createGroupUtil.setManager(request, next); + if (isEmpty(result) || res.headersSent) { + return; + } + + if (result.success === true) { + const status = result.status ? result.status : httpStatus.OK; + return res.status(status).json({ + success: true, + message: "Group manager successffuly set", + updated_group: result.data, + }); + } else if (result.success === false) { + const status = result.status + ? result.status + : httpStatus.INTERNAL_SERVER_ERROR; + return res.status(status).json({ + success: false, + message: result.message, + errors: result.errors + ? result.errors + : { message: "Internal Server Error" }, + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, listAssignedUsers: async (req, res, next) => { try { const errors = extractErrorsFromRequest(req); diff --git a/src/auth-service/models/Activity.js b/src/auth-service/models/Activity.js new file mode 100644 index 0000000000..efd6aba748 --- /dev/null +++ b/src/auth-service/models/Activity.js @@ -0,0 +1,312 @@ +const mongoose = require("mongoose"); +const ObjectId = mongoose.Schema.Types.ObjectId; +const { Schema } = mongoose; +var uniqueValidator = require("mongoose-unique-validator"); +const { logObject } = require("@utils/log"); +const constants = require("@config/constants"); +const isEmpty = require("is-empty"); +const { getModelByTenant } = require("@config/database"); +const httpStatus = require("http-status"); +const { HttpError } = require("@utils/errors"); +const log4js = require("log4js"); +const logger = log4js.getLogger(`${constants.ENVIRONMENT} -- activity-model`); + +const activitySchema = new Schema( + { + email: { type: String, required: true }, + username: { type: String }, + tenant: { type: String, required: true }, + // Daily stats + dailyStats: [ + { + date: { type: Date, required: true }, + totalActions: { type: Number, default: 0 }, + services: [ + { + name: String, + count: Number, + }, + ], + endpoints: [ + { + name: String, + count: Number, + }, + ], + }, + ], + monthlyStats: [ + { + year: { type: Number, required: true }, + month: { type: Number, required: true }, + totalActions: { type: Number, default: 0 }, + uniqueServices: [String], + uniqueEndpoints: [String], + topServices: [ + { + name: String, + count: Number, + }, + ], + engagementScore: Number, + engagementTier: String, + firstActivity: Date, + lastActivity: Date, + }, + ], + overallStats: { + totalActions: { type: Number, default: 0 }, + firstActivity: Date, + lastActivity: Date, + engagementScore: Number, + engagementTier: String, + }, + lastProcessedLog: { type: mongoose.Schema.Types.ObjectId, ref: "logs" }, + }, + { + timestamps: true, + indexes: [ + { email: 1, tenant: 1 }, + { tenant: 1, "monthlyStats.year": 1, "monthlyStats.month": 1 }, + { "dailyStats.date": 1 }, + { lastProcessedLog: 1 }, + ], + } +); + +activitySchema.plugin(uniqueValidator, { + message: `{VALUE} should be unique!`, +}); + +activitySchema.methods = { + toJSON() { + return { + _id: this._id, + }; + }, +}; + +activitySchema.statics = { + async register(args, next) { + try { + const data = await this.create({ + ...args, + }); + if (!isEmpty(data)) { + return { + success: true, + data, + message: "activity created", + status: httpStatus.OK, + }; + } else if (isEmpty(data)) { + return { + success: true, + data, + message: "activity NOT successfully created but operation successful", + status: httpStatus.ACCEPTED, + }; + } + } catch (err) { + let response = {}; + let errors = {}; + let message = "Internal Server Error"; + let status = httpStatus.INTERNAL_SERVER_ERROR; + if (err.code === 11000 || err.code === 11001) { + errors = err.keyValue; + message = "duplicate values provided"; + status = httpStatus.CONFLICT; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value); + }); + } else { + message = "validation errors for some of the provided fields"; + status = httpStatus.CONFLICT; + errors = err.errors; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value.message); + }); + } + logger.error(`🐛🐛 Internal Server Error -- ${err.message}`); + next(new HttpError(message, status, response)); + } + }, + async list({ skip = 0, limit = 100, filter = {} } = {}, next) { + try { + logObject("filter", filter); + const inclusionProjection = constants.ACTIVITIES_INCLUSION_PROJECTION; + const exclusionProjection = constants.ACTIVITIES_EXCLUSION_PROJECTION( + filter.category ? filter.category : "none" + ); + + if (!isEmpty(filter.category)) { + delete filter.category; + } + + const response = await this.aggregate() + .match(filter) + .sort({ createdAt: -1 }) + .project(inclusionProjection) + .project(exclusionProjection) + .skip(skip ? skip : 0) + .limit(limit ? limit : 100) + .allowDiskUse(true); + + if (!isEmpty(response)) { + return { + success: true, + message: "successfully retrieved the activitys", + data: response, + status: httpStatus.OK, + }; + } else if (isEmpty(response)) { + return { + success: true, + message: "activitys do not exist, please crosscheck", + status: httpStatus.NOT_FOUND, + data: [], + errors: { message: "unable to retrieve activitys" }, + }; + } + } catch (err) { + let response = {}; + let errors = {}; + let message = "Internal Server Error"; + let status = httpStatus.INTERNAL_SERVER_ERROR; + if (err.code === 11000 || err.code === 11001) { + errors = err.keyValue; + message = "duplicate values provided"; + status = httpStatus.CONFLICT; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value); + }); + } else { + message = "validation errors for some of the provided fields"; + status = httpStatus.CONFLICT; + errors = err.errors; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value.message); + }); + } + + logger.error(`🐛🐛 Internal Server Error -- ${err.message}`); + next(new HttpError(message, status, response)); + } + }, + async modify({ filter = {}, update = {} } = {}, next) { + try { + let options = { new: true }; + let modifiedUpdate = Object.assign({}, update); + modifiedUpdate["$addToSet"] = {}; + + if (modifiedUpdate.tenant) { + delete modifiedUpdate.tenant; + } + + const updatedActivity = await this.findOneAndUpdate( + filter, + modifiedUpdate, + options + ).exec(); + + if (!isEmpty(updatedActivity)) { + return { + success: true, + message: "successfully modified the activity", + data: updatedActivity._doc, + status: httpStatus.OK, + }; + } else if (isEmpty(updatedActivity)) { + next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "activity does not exist, please crosscheck -- Not Found", + }) + ); + } + } catch (err) { + let response = {}; + let errors = {}; + let message = "Internal Server Error"; + let status = httpStatus.INTERNAL_SERVER_ERROR; + if (err.code === 11000 || err.code === 11001) { + errors = err.keyValue; + message = "duplicate values provided"; + status = httpStatus.CONFLICT; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value); + }); + } else { + message = "validation errors for some of the provided fields"; + status = httpStatus.CONFLICT; + errors = err.errors; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value.message); + }); + } + logger.error(`🐛🐛 Internal Server Error -- ${err.message}`); + next(new HttpError(message, status, response)); + } + }, + async remove({ filter = {} } = {}, next) { + try { + let options = { + projection: { + _id: 1, + }, + }; + const removedActivity = await this.findOneAndRemove( + filter, + options + ).exec(); + + if (!isEmpty(removedActivity)) { + return { + success: true, + message: "successfully removed the activity", + data: removedActivity._doc, + status: httpStatus.OK, + }; + } else if (isEmpty(removedActivity)) { + next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Bad Request, activity Not Found -- please crosscheck", + }) + ); + } + } catch (err) { + let response = {}; + let errors = {}; + let message = "Internal Server Error"; + let status = httpStatus.INTERNAL_SERVER_ERROR; + if (err.code === 11000 || err.code === 11001) { + errors = err.keyValue; + message = "duplicate values provided"; + status = httpStatus.CONFLICT; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value); + }); + } else { + message = "validation errors for some of the provided fields"; + status = httpStatus.CONFLICT; + errors = err.errors; + Object.entries(errors).forEach(([key, value]) => { + return (response[key] = value.message); + }); + } + logger.error(`🐛🐛 Internal Server Error -- ${err.message}`); + next(new HttpError(message, status, response)); + } + }, +}; + +const ActivityModel = (tenant) => { + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + const dbTenant = isEmpty(tenant) ? defaultTenant : tenant; + try { + return mongoose.model("activities"); + } catch (error) { + return getModelByTenant(dbTenant, "activity", activitySchema); + } +}; + +module.exports = ActivityModel; diff --git a/src/auth-service/models/Group.js b/src/auth-service/models/Group.js index e590876ca4..6ac0c16502 100644 --- a/src/auth-service/models/Group.js +++ b/src/auth-service/models/Group.js @@ -1,8 +1,9 @@ const mongoose = require("mongoose"); const ObjectId = mongoose.Schema.Types.ObjectId; const { Schema } = mongoose; +const validator = require("validator"); var uniqueValidator = require("mongoose-unique-validator"); -const { logObject } = require("@utils/log"); +const { logObject, logText, logElement } = require("@utils/log"); const constants = require("@config/constants"); const isEmpty = require("is-empty"); const { getModelByTenant } = require("@config/database"); @@ -11,6 +12,23 @@ const { HttpError } = require("@utils/errors"); const log4js = require("log4js"); const logger = log4js.getLogger(`${constants.ENVIRONMENT} -- group-model`); +function validateProfilePicture(grp_profile_picture) { + const urlRegex = + /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; + if (!urlRegex.test(grp_profile_picture)) { + logger.error(`🙅🙅 Bad Request Error -- Not a valid profile picture URL`); + return false; + } + if (grp_profile_picture.length > 200) { + logText("longer than 200 chars"); + logger.error( + `🙅🙅 Bad Request Error -- profile picture URL exceeds 200 characters` + ); + return false; + } + return true; +} + const GroupSchema = new Schema( { grp_title: { @@ -33,6 +51,20 @@ const GroupSchema = new Schema( grp_country: { type: String }, grp_timezone: { type: String }, grp_image: { type: String }, + grp_profile_picture: { + type: String, + maxLength: 200, + default: constants.DEFAULT_ORGANISATION_PROFILE_PICTURE, + validate: { + validator: function (v) { + const urlRegex = + /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; + return urlRegex.test(v); + }, + message: + "Profile picture URL must be a valid URL & must not exceed 200 characters.", + }, + }, }, { timestamps: true, @@ -45,6 +77,55 @@ GroupSchema.plugin(uniqueValidator, { GroupSchema.index({ grp_title: 1 }, { unique: true }); +GroupSchema.pre( + ["updateOne", "findOneAndUpdate", "updateMany", "update", "save"], + async function (next) { + // Determine if this is a new document or an update + const isNew = this.isNew; + let updates = this.getUpdate ? this.getUpdate() : this; + + try { + // Get all actual fields being updated from both root and $set + const actualUpdates = { + ...(updates || {}), + ...(updates.$set || {}), + }; + + // Profile picture validation for both new documents and updates + if (isNew) { + // Validation for new documents + + if (!this.grp_profile_picture) { + this.grp_profile_picture = + constants.DEFAULT_ORGANISATION_PROFILE_PICTURE; + } else if ( + this.grp_profile_picture && + !validateProfilePicture(this.grp_profile_picture) + ) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Invalid profile picture URL", + }) + ); + } + } else if (actualUpdates.grp_profile_picture) { + // Validation for updates + if (!validateProfilePicture(actualUpdates.grp_profile_picture)) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Invalid profile picture URL", + }) + ); + } + } + + return next(); + } catch (error) { + return next(error); + } + } +); + GroupSchema.methods = { toJSON() { return { @@ -59,6 +140,7 @@ GroupSchema.methods = { grp_manager_firstname: this.grp_manager_firstname, grp_manager_lastname: this.grp_manager_lastname, grp_website: this.grp_website, + grp_profile_picture: this.grp_profile_picture, grp_industry: this.grp_industry, grp_country: this.grp_country, grp_timezone: this.grp_timezone, diff --git a/src/auth-service/models/Network.js b/src/auth-service/models/Network.js index 22d1278bc9..7f274ddd0a 100644 --- a/src/auth-service/models/Network.js +++ b/src/auth-service/models/Network.js @@ -13,6 +13,23 @@ const logger = require("log4js").getLogger( ); const { HttpError } = require("@utils/errors"); +function validateProfilePicture(net_profile_picture) { + const urlRegex = + /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; + if (!urlRegex.test(net_profile_picture)) { + logger.error(`🙅🙅 Bad Request Error -- Not a valid profile picture URL`); + return false; + } + if (net_profile_picture.length > 200) { + logText("longer than 200 chars"); + logger.error( + `🙅🙅 Bad Request Error -- profile picture URL exceeds 200 characters` + ); + return false; + } + return true; +} + const NetworkSchema = new Schema( { net_email: { @@ -92,6 +109,20 @@ const NetworkSchema = new Schema( ref: "permission", }, ], + net_profile_picture: { + type: String, + maxLength: 200, + default: constants.DEFAULT_ORGANISATION_PROFILE_PICTURE, + validate: { + validator: function (v) { + const urlRegex = + /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; + return urlRegex.test(v); + }, + message: + "Profile picture URL must be a valid URL & must not exceed 200 characters.", + }, + }, }, { timestamps: true, @@ -139,12 +170,39 @@ NetworkSchema.pre( handleArrayFieldUpdates("net_departments"); handleArrayFieldUpdates("net_permissions"); - // Validation for new documents - if (isNew) { - // Required field validations are handled by schema + // Get all actual fields being updated from both root and $set + const actualUpdates = { + ...(updates || {}), + ...(updates.$set || {}), + }; - // Set default status if not provided + // Profile picture validation for both new documents and updates + if (isNew) { + // Validation for new documents this.net_status = this.net_status || "inactive"; + + if (!this.net_profile_picture) { + this.net_profile_picture = + constants.DEFAULT_ORGANISATION_PROFILE_PICTURE; + } else if ( + this.net_profile_picture && + !validateProfilePicture(this.net_profile_picture) + ) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Invalid profile picture URL", + }) + ); + } + } else if (actualUpdates.net_profile_picture) { + // Validation for updates + if (!validateProfilePicture(actualUpdates.net_profile_picture)) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Invalid profile picture URL", + }) + ); + } } // Unique field handling @@ -208,6 +266,7 @@ NetworkSchema.methods = { _id: this._id, net_email: this.net_email, net_website: this.net_website, + net_profile_picture: this.net_profile_picture, net_category: this.net_category, net_status: this.net_status, net_phoneNumber: this.net_phoneNumber, diff --git a/src/auth-service/models/User.js b/src/auth-service/models/User.js index 3b5bda5b03..9e5fdfaf4f 100644 --- a/src/auth-service/models/User.js +++ b/src/auth-service/models/User.js @@ -141,7 +141,7 @@ const UserSchema = new Schema( validator: function (value) { return value.length <= ORGANISATIONS_LIMIT; }, - message: "Too many networks. Maximum limit: 6.", + message: `Too many networks. Maximum limit: ${ORGANISATIONS_LIMIT}`, }, ], }, @@ -169,7 +169,7 @@ const UserSchema = new Schema( validator: function (value) { return value.length <= ORGANISATIONS_LIMIT; }, - message: "Too many groups. Maximum limit: 6.", + message: `Too many groups. Maximum limit: ${ORGANISATIONS_LIMIT}`, }, ], }, @@ -251,7 +251,16 @@ UserSchema.pre( async function (next) { // Determine if this is a new document or an update const isNew = this.isNew; - let updates = this.getUpdate ? this.getUpdate() : this; + + // Safely get updates object, accounting for different mongoose operations + let updates = {}; + if (this.getUpdate) { + updates = this.getUpdate() || {}; + } else if (!isNew) { + updates = this.toObject(); + } else { + updates = this; + } try { // Helper function to handle role updates @@ -267,20 +276,38 @@ UserSchema.pre( let newRoles = []; const existingRoles = doc[fieldName] || []; - // Handle $set operations + // Initialize update operators safely + updates.$set = updates.$set || {}; + updates.$push = updates.$push || {}; + updates.$addToSet = updates.$addToSet || {}; + + // Handle update operations in order of precedence if (updates.$set && updates.$set[fieldName]) { + // $set takes precedence as it's a direct override newRoles = updates.$set[fieldName]; - } - // Handle $push operations - else if (updates.$push && updates.$push[fieldName]) { + } else if (updates.$push && updates.$push[fieldName]) { + // Safely handle $push with potential $each operator const pushValue = updates.$push[fieldName]; - const newRole = pushValue.$each ? pushValue.$each[0] : pushValue; - newRoles = [...existingRoles, newRole]; - } - // Handle $addToSet operations - else if (updates.$addToSet && updates.$addToSet[fieldName]) { - const newRole = updates.$addToSet[fieldName]; - newRoles = [...existingRoles, newRole]; + if (pushValue) { + if (pushValue.$each && Array.isArray(pushValue.$each)) { + newRoles = [...existingRoles, ...pushValue.$each]; + } else { + newRoles = [...existingRoles, pushValue]; + } + } + } else if (updates.$addToSet && updates.$addToSet[fieldName]) { + // Safely handle $addToSet + const addToSetValue = updates.$addToSet[fieldName]; + if (addToSetValue) { + if (addToSetValue.$each && Array.isArray(addToSetValue.$each)) { + newRoles = [...existingRoles, ...addToSetValue.$each]; + } else { + newRoles = [...existingRoles, addToSetValue]; + } + } + } else if (updates[fieldName]) { + // Direct field update + newRoles = updates[fieldName]; } if (newRoles.length > 0) { @@ -306,14 +333,11 @@ UserSchema.pre( // Convert Map values back to array const finalRoles = Array.from(uniqueRoles.values()); - // Clear all update operators for this field - if (updates.$set) delete updates.$set[fieldName]; - if (updates.$push) delete updates.$push[fieldName]; - if (updates.$addToSet) delete updates.$addToSet[fieldName]; - - // Set the final filtered array - updates.$set = updates.$set || {}; + // Set the final filtered array using $set and clean up other operators updates.$set[fieldName] = finalRoles; + delete updates.$push[fieldName]; + delete updates.$addToSet[fieldName]; + delete updates[fieldName]; // Clean up direct field update if any } }; @@ -334,10 +358,10 @@ UserSchema.pre( if (isNew) { this.password = bcrypt.hashSync(passwordToHash, saltRounds); } else { - if (updates.password) { + if (updates && updates.password) { updates.password = bcrypt.hashSync(passwordToHash, saltRounds); } - if (updates.$set && updates.$set.password) { + if (updates && updates.$set && updates.$set.password) { updates.$set.password = bcrypt.hashSync(passwordToHash, saltRounds); } } @@ -453,6 +477,17 @@ UserSchema.pre( ...(updates.$set || {}), }; + // Profile picture validation for updates + if (actualUpdates.profilePicture) { + if (!validateProfilePicture(actualUpdates.profilePicture)) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Invalid profile picture URL", + }) + ); + } + } + // Conditional validations for updates // Only validate fields that are present in the update fieldsToValidate.forEach((field) => { @@ -472,7 +507,7 @@ UserSchema.pre( const immutableFields = ["firebase_uid", "email", "createdAt", "_id"]; immutableFields.forEach((field) => { if (updates[field]) delete updates[field]; - if (updates.$set && updates.$set[field]) { + if (updates && updates.$set && updates.$set[field]) { return next( new HttpError( "Modification Not Allowed", @@ -481,7 +516,7 @@ UserSchema.pre( ) ); } - if (updates.$set) delete updates.$set[field]; + if (updates && updates.$set) delete updates.$set[field]; if (updates.$push) delete updates.$push[field]; }); @@ -510,7 +545,7 @@ UserSchema.pre( // Conditional permissions validation if (updates.permissions) { const uniquePermissions = [...new Set(updates.permissions)]; - if (updates.$set) { + if (updates && updates.$set) { updates.$set.permissions = uniquePermissions; } else { updates.permissions = uniquePermissions; @@ -518,7 +553,7 @@ UserSchema.pre( } // Conditional default values for updates - if (updates.$set) { + if (updates && updates.$set) { updates.$set.verified = updates.$set.verified ?? false; updates.$set.analyticsVersion = updates.$set.analyticsVersion ?? 2; } else { diff --git a/src/auth-service/models/UserOld.js b/src/auth-service/models/UserOld.js deleted file mode 100644 index 7e59c0b5de..0000000000 --- a/src/auth-service/models/UserOld.js +++ /dev/null @@ -1,939 +0,0 @@ -const mongoose = require("mongoose").set("debug", true); -const Schema = mongoose.Schema; -const validator = require("validator"); -const bcrypt = require("bcrypt"); -const jwt = require("jsonwebtoken"); -const constants = require("@config/constants"); -const { logObject, logText } = require("@utils/log"); -const ObjectId = mongoose.Schema.Types.ObjectId; -const isEmpty = require("is-empty"); -const saltRounds = constants.SALT_ROUNDS; -const httpStatus = require("http-status"); -const accessCodeGenerator = require("generate-password"); -const { getModelByTenant } = require("@config/database"); -const logger = require("log4js").getLogger( - `${constants.ENVIRONMENT} -- user-model` -); -const validUserTypes = ["user", "guest"]; -const { HttpError } = require("@utils/errors"); -const mailer = require("@utils/mailer"); - -function oneMonthFromNow() { - var d = new Date(); - var targetMonth = d.getMonth() + 1; - d.setMonth(targetMonth); - if (d.getMonth() !== targetMonth % 12) { - d.setDate(0); // last day of previous month - } - return d; -} - -function validateProfilePicture(profilePicture) { - const urlRegex = - /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; - if (!urlRegex.test(profilePicture)) { - logger.error(`🙅🙅 Bad Request Error -- Not a valid profile picture URL`); - return false; - } - if (profilePicture.length > 200) { - logText("longer than 200 chars"); - logger.error( - `🙅🙅 Bad Request Error -- profile picture URL exceeds 200 characters` - ); - return false; - } - return true; -} -const passwordReg = /^(?=.*[A-Za-z])(?=.*\d)[A-Za-z\d@#?!$%^&*,.]{6,}$/; -const UserSchema = new Schema( - { - due_date: { type: Date }, - status: { type: String }, - address: { type: String }, - country: { type: String }, - firebase_uid: { type: String }, - city: { type: String }, - department_id: { - type: ObjectId, - ref: "department", - }, - role: { - type: ObjectId, - ref: "role", - }, - birthday: { type: Date }, - reports_to: { type: ObjectId, ref: "user" }, - replaced_by: { type: ObjectId, ref: "user" }, - email: { - type: String, - unique: true, - required: [true, "Email is required"], - trim: true, - validate: { - validator(email) { - return validator.isEmail(email); - }, - message: "{VALUE} is not a valid email!", - }, - }, - verified: { - type: Boolean, - default: false, - }, - analyticsVersion: { type: Number, default: 2 }, - firstName: { - type: String, - required: [true, "FirstName is required!"], - trim: true, - }, - lastName: { - type: String, - required: [true, "LastName is required"], - trim: true, - }, - userName: { - type: String, - required: [true, "UserName is required!"], - trim: true, - unique: true, - }, - password: { - type: String, - required: [true, "Password is required!"], - trim: true, - minlength: [6, "Password is required"], - validate: { - validator(password) { - return passwordReg.test(password); - }, - message: "{VALUE} is not a valid password, please check documentation!", - }, - }, - privilege: { - type: String, - default: "user", - }, - isActive: { type: Boolean, default: false }, - loginCount: { type: Number, default: 0 }, - duration: { type: Date, default: oneMonthFromNow }, - network_roles: { - type: [ - { - network: { - type: ObjectId, - ref: "network", - default: mongoose.Types.ObjectId(constants.DEFAULT_NETWORK), - }, - userType: { type: String, default: "guest", enum: validUserTypes }, - createdAt: { type: String, default: new Date() }, - role: { - type: ObjectId, - ref: "role", - default: mongoose.Types.ObjectId(constants.DEFAULT_NETWORK_ROLE), - }, - }, - ], - default: [], - _id: false, - validate: [ - { - validator: function (value) { - const maxLimit = 6; - return value.length <= maxLimit; - }, - message: "Too many networks. Maximum limit: 6.", - }, - ], - }, - group_roles: { - type: [ - { - group: { - type: ObjectId, - ref: "group", - default: mongoose.Types.ObjectId(constants.DEFAULT_GROUP), - }, - userType: { type: String, default: "guest", enum: validUserTypes }, - createdAt: { type: String, default: new Date() }, - role: { - type: ObjectId, - ref: "role", - default: mongoose.Types.ObjectId(constants.DEFAULT_GROUP_ROLE), - }, - }, - ], - default: [], - _id: false, - validate: [ - { - validator: function (value) { - const maxLimit = 6; - return value.length <= maxLimit; - }, - message: "Too many groups. Maximum limit: 6.", - }, - ], - }, - - permissions: [ - { - type: ObjectId, - ref: "permission", - }, - ], - organization: { - type: String, - default: "airqo", - }, - long_organization: { - type: String, - default: "airqo", - }, - rateLimit: { - type: Number, - }, - phoneNumber: { - type: Number, - validate: { - validator(phoneNumber) { - return !!phoneNumber || this.email; - }, - message: "Phone number or email is required!", - }, - }, - resetPasswordToken: { type: String }, - resetPasswordExpires: { type: Date }, - jobTitle: { - type: String, - }, - website: { type: String }, - description: { type: String }, - lastLogin: { - type: Date, - }, - category: { - type: String, - }, - notifications: { - email: { type: Boolean, default: false }, - push: { type: Boolean, default: false }, - text: { type: Boolean, default: false }, - phone: { type: Boolean, default: false }, - }, - profilePicture: { - type: String, - maxLength: 200, - validate: { - validator: function (v) { - const urlRegex = - /^(http(s)?:\/\/.)[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_\+.~#?&//=]*)$/g; - return urlRegex.test(v); - }, - message: - "Profile picture URL must be a valid URL & must not exceed 200 characters.", - }, - }, - google_id: { type: String, trim: true }, - timezone: { type: String, trim: true }, - }, - { timestamps: true } -); - -UserSchema.path("network_roles.userType").validate(function (value) { - return validUserTypes.includes(value); -}, "Invalid userType value"); - -UserSchema.path("group_roles.userType").validate(function (value) { - return validUserTypes.includes(value); -}, "Invalid userType value"); - -UserSchema.pre("save", function (next) { - if (this.isModified("password")) { - this.password = bcrypt.hashSync(this.password, saltRounds); - } - if (!this.email && !this.phoneNumber) { - return next(new Error("Phone number or email is required!")); - } - - if (!this.network_roles || this.network_roles.length === 0) { - if ( - !constants || - !constants.DEFAULT_NETWORK || - !constants.DEFAULT_NETWORK_ROLE - ) { - throw new HttpError( - "Internal Server Error", - httpStatus.INTERNAL_SERVER_ERROR, - { - message: - "Contact support@airqo.net -- unable to retrieve the default Network or Role to which the User will belong", - } - ); - } - - this.network_roles = [ - { - network: mongoose.Types.ObjectId(constants.DEFAULT_NETWORK), - userType: "guest", - createdAt: new Date(), - role: mongoose.Types.ObjectId(constants.DEFAULT_NETWORK_ROLE), - }, - ]; - } - - if (!this.group_roles || this.group_roles.length === 0) { - if ( - !constants || - !constants.DEFAULT_GROUP || - !constants.DEFAULT_GROUP_ROLE - ) { - throw new HttpError( - "Internal Server Error", - httpStatus.INTERNAL_SERVER_ERROR, - { - message: - "Contact support@airqo.net -- unable to retrieve the default Group or Role to which the User will belong", - } - ); - } - - this.group_roles = [ - { - group: mongoose.Types.ObjectId(constants.DEFAULT_GROUP), - userType: "guest", - createdAt: new Date(), - role: mongoose.Types.ObjectId(constants.DEFAULT_GROUP_ROLE), - }, - ]; - } - - if (!this.verified) { - this.verified = false; - } - - if (!this.analyticsVersion) { - this.analyticsVersion = 2; - } - - return next(); -}); - -UserSchema.pre("update", function (next) { - if (this.isModified("password")) { - this.password = bcrypt.hashSync(this.password, saltRounds); - } - return next(); -}); - -UserSchema.index({ email: 1 }, { unique: true }); -UserSchema.index({ userName: 1 }, { unique: true }); - -UserSchema.statics = { - async register(args, next) { - try { - const data = await this.create({ - ...args, - }); - if (!isEmpty(data)) { - return { - success: true, - data, - message: "user created", - status: httpStatus.OK, - }; - } else if (isEmpty(data)) { - return { - success: true, - data, - message: "Operation successful but user NOT successfully created", - status: httpStatus.OK, - }; - } - } catch (err) { - logObject("the error", err); - let response = {}; - let message = "validation errors for some of the provided fields"; - let status = httpStatus.CONFLICT; - if (err.code === 11000) { - logObject("the err.code again", err.code); - const duplicate_record = args.email ? args.email : args.userName; - response[duplicate_record] = `${duplicate_record} must be unique`; - response["message"] = - "the email and userName must be unique for every user"; - try { - const email = args.email; - const firstName = args.firstName; - const lastName = args.lastName; - const emailResponse = await mailer.existingUserRegistrationRequest( - { - email, - firstName, - lastName, - }, - next - ); - if (emailResponse && emailResponse.success === false) { - logger.error( - `🐛🐛 Internal Server Error -- ${stringify(emailResponse)}` - ); - } - } catch (error) { - logger.error(`🐛🐛 Internal Server Error -- ${error.message}`); - } - } else if (err.keyValue) { - Object.entries(err.keyValue).forEach(([key, value]) => { - return (response[key] = `the ${key} must be unique`); - }); - } else if (err.errors) { - Object.entries(err.errors).forEach(([key, value]) => { - return (response[key] = value.message); - }); - } - logger.error(`🐛🐛 Internal Server Error -- ${err.message}`); - next(new HttpError(message, status, response)); - } - }, - async listStatistics(next) { - try { - const response = await this.aggregate() - .match({ email: { $nin: [null, ""] } }) - .sort({ createdAt: -1 }) - .lookup({ - from: "clients", - localField: "_id", - foreignField: "user_id", - as: "clients", - }) - .lookup({ - from: "users", - localField: "clients.user_id", - foreignField: "_id", - as: "api_clients", - }) - .group({ - _id: null, - users: { $sum: 1 }, - user_details: { - $push: { - userName: "$userName", - email: "$email", - firstName: "$firstName", - lastName: "$lastName", - _id: "$_id", - }, - }, - active_users: { $sum: { $cond: ["$isActive", 1, 0] } }, - active_user_details: { - $addToSet: { - $cond: { - if: "$isActive", - then: { - userName: "$userName", - email: "$email", - firstName: "$firstName", - lastName: "$lastName", - _id: "$_id", - }, - else: "$nothing", - }, - }, - }, - client_users: { $addToSet: "$clients.user_id" }, - api_user_details: { - $addToSet: { - userName: { $arrayElemAt: ["$api_clients.userName", 0] }, - email: { $arrayElemAt: ["$api_clients.email", 0] }, - firstName: { $arrayElemAt: ["$api_clients.firstName", 0] }, - lastName: { $arrayElemAt: ["$api_clients.lastName", 0] }, - _id: { $arrayElemAt: ["$api_clients._id", 0] }, - }, - }, - }) - .project({ - _id: 0, - users: { - number: "$users", - details: "$user_details", - }, - active_users: { - number: "$active_users", - details: "$active_user_details", - }, - api_users: { - number: { $size: { $ifNull: ["$client_users", []] } }, - details: "$api_user_details", - }, - }) - .allowDiskUse(true); - - if (!isEmpty(response)) { - return { - success: true, - message: "Successfully retrieved the user statistics", - data: response[0], - status: httpStatus.OK, - }; - } else if (isEmpty(response)) { - return { - success: true, - message: "No users statistics exist", - data: [], - status: httpStatus.OK, - }; - } - } catch (error) { - logger.error(`🐛🐛 Internal Server Error -- ${error.message}`); - next( - new HttpError( - "Internal Server Error", - httpStatus.INTERNAL_SERVER_ERROR, - { message: error.message } - ) - ); - } - }, - async list({ skip = 0, limit = 100, filter = {} } = {}, next) { - try { - const inclusionProjection = constants.USERS_INCLUSION_PROJECTION; - const exclusionProjection = constants.USERS_EXCLUSION_PROJECTION( - filter.category ? filter.category : "none" - ); - - if (!isEmpty(filter.category)) { - delete filter.category; - } - logObject("the filter being used", filter); - const response = await this.aggregate() - .match(filter) - .lookup({ - from: "permissions", - localField: "permissions", - foreignField: "_id", - as: "permissions", - }) - .lookup({ - from: "clients", - localField: "_id", - foreignField: "user_id", - as: "clients", - }) - .lookup({ - from: "networks", - localField: "_id", - foreignField: "net_manager", - as: "my_networks", - }) - .lookup({ - from: "groups", - localField: "_id", - foreignField: "grp_manager", - as: "my_groups", - }) - .addFields({ - createdAt: { - $dateToString: { - format: "%Y-%m-%d %H:%M:%S", - date: "$_id", - }, - }, - }) - .unwind({ - path: "$network_roles", - preserveNullAndEmptyArrays: true, - }) - .unwind({ - path: "$group_roles", - preserveNullAndEmptyArrays: true, - }) - .lookup({ - from: "networks", - localField: "network_roles.network", - foreignField: "_id", - as: "network", - }) - .lookup({ - from: "groups", - localField: "group_roles.group", - foreignField: "_id", - as: "group", - }) - .lookup({ - from: "roles", - localField: "network_roles.role", - foreignField: "_id", - as: "network_role", - }) - .lookup({ - from: "roles", - localField: "group_roles.role", - foreignField: "_id", - as: "group_role", - }) - .lookup({ - from: "permissions", - localField: "network_role.role_permissions", - foreignField: "_id", - as: "network_role_permissions", - }) - .lookup({ - from: "permissions", - localField: "group_role.role_permissions", - foreignField: "_id", - as: "group_role_permissions", - }) - .group({ - _id: "$_id", - firstName: { $first: "$firstName" }, - lastName: { $first: "$lastName" }, - lastLogin: { $first: "$lastLogin" }, - timezone: { $first: "$timezone" }, - isActive: { $first: "$isActive" }, - loginCount: { $first: "$loginCount" }, - userName: { $first: "$userName" }, - email: { $first: "$email" }, - verified: { $first: "$verified" }, - analyticsVersion: { $first: "$analyticsVersion" }, - country: { $first: "$country" }, - privilege: { $first: "$privilege" }, - website: { $first: "$website" }, - category: { $first: "$category" }, - organization: { $first: "$organization" }, - long_organization: { $first: "$long_organization" }, - rateLimit: { $first: "$rateLimit" }, - jobTitle: { $first: "$jobTitle" }, - description: { $first: "$description" }, - profilePicture: { $first: "$profilePicture" }, - phoneNumber: { $first: "$phoneNumber" }, - group_roles: { $first: "$group_roles" }, - network_roles: { $first: "$network_roles" }, - group_role: { $first: "$group_role" }, - network_role: { $first: "$network_role" }, - clients: { $first: "$clients" }, - groups: { - $addToSet: { - grp_title: { $arrayElemAt: ["$group.grp_title", 0] }, - _id: { $arrayElemAt: ["$group._id", 0] }, - createdAt: { $arrayElemAt: ["$group.createdAt", 0] }, - status: { $arrayElemAt: ["$group.grp_status", 0] }, - role: { - $cond: { - if: { $ifNull: ["$group_role", false] }, - then: { - _id: { $arrayElemAt: ["$group_role._id", 0] }, - role_name: { $arrayElemAt: ["$group_role.role_name", 0] }, - role_permissions: "$group_role_permissions", - }, - else: null, - }, - }, - userType: { - $cond: { - if: { $eq: [{ $type: "$group_roles.userType" }, "missing"] }, - then: "user", - else: "$group_roles.userType", - }, - }, - }, - }, - permissions: { $first: "$permissions" }, - my_networks: { $first: "$my_networks" }, - my_groups: { $first: "$my_groups" }, - createdAt: { $first: "$createdAt" }, - updatedAt: { $first: "$createdAt" }, - networks: { - $addToSet: { - net_name: { $arrayElemAt: ["$network.net_name", 0] }, - _id: { $arrayElemAt: ["$network._id", 0] }, - role: { - $cond: { - if: { $ifNull: ["$network_role", false] }, - then: { - _id: { $arrayElemAt: ["$network_role._id", 0] }, - role_name: { $arrayElemAt: ["$network_role.role_name", 0] }, - role_permissions: "$network_role_permissions", - }, - else: null, - }, - }, - userType: { - $cond: { - if: { - $eq: [{ $type: "$network_roles.userType" }, "missing"], - }, - then: "user", - else: "$network_roles.userType", - }, - }, - }, - }, - }) - .project(inclusionProjection) - .project(exclusionProjection) - .sort({ createdAt: -1 }) - .skip(skip ? skip : 0) - .limit(limit ? limit : parseInt(constants.DEFAULT_LIMIT)) - .allowDiskUse(true); - if (!isEmpty(response)) { - return { - success: true, - message: "successfully retrieved the user details", - data: response, - status: httpStatus.OK, - }; - } else if (isEmpty(response)) { - return { - success: true, - message: "no users exist", - data: [], - status: httpStatus.OK, - }; - } - } catch (error) { - logger.error(`🐛🐛 Internal Server Error -- ${error.message}`); - next( - new HttpError( - "Internal Server Error", - httpStatus.INTERNAL_SERVER_ERROR, - { message: error.message } - ) - ); - } - }, - async modify({ filter = {}, update = {} } = {}, next) { - try { - logText("the user modification function........"); - let options = { new: true }; - const fieldNames = Object.keys(update); - const fieldsString = fieldNames.join(" "); - let modifiedUpdate = update; - modifiedUpdate["$addToSet"] = {}; - - if (update.password) { - modifiedUpdate.password = bcrypt.hashSync(update.password, saltRounds); - } - - if (modifiedUpdate.profilePicture) { - if (!validateProfilePicture(modifiedUpdate.profilePicture)) { - next( - new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { - message: "Invalid profile picture URL", - }) - ); - } - } - - if (modifiedUpdate.network_roles) { - if (isEmpty(modifiedUpdate.network_roles.network)) { - delete modifiedUpdate.network_roles; - } else { - modifiedUpdate["$addToSet"] = { - network_roles: { $each: modifiedUpdate.network_roles }, - }; - delete modifiedUpdate.network_roles; - } - } - - if (modifiedUpdate.group_roles) { - if (isEmpty(modifiedUpdate.group_roles.group)) { - delete modifiedUpdate.group_roles; - } else { - modifiedUpdate["$addToSet"] = { - group_roles: { $each: modifiedUpdate.group_roles }, - }; - delete modifiedUpdate.group_roles; - } - } - - if (modifiedUpdate.permissions) { - modifiedUpdate["$addToSet"]["permissions"] = {}; - modifiedUpdate["$addToSet"]["permissions"]["$each"] = - modifiedUpdate.permissions; - delete modifiedUpdate["permissions"]; - } - - const updatedUser = await this.findOneAndUpdate( - filter, - modifiedUpdate, - options - ).select(fieldsString); - - if (!isEmpty(updatedUser)) { - const { _id, ...userData } = updatedUser._doc; - return { - success: true, - message: "successfully modified the user", - data: userData, - status: httpStatus.OK, - }; - } else if (isEmpty(updatedUser)) { - next( - new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { - message: "user does not exist, please crosscheck", - }) - ); - } - } catch (error) { - logger.error(`🐛🐛 Internal Server Error -- ${error.message}`); - next( - new HttpError( - "Internal Server Error", - httpStatus.INTERNAL_SERVER_ERROR, - { message: error.message } - ) - ); - } - }, - async remove({ filter = {} } = {}, next) { - try { - const options = { - projection: { - _id: 0, - email: 1, - firstName: 1, - lastName: 1, - lastLogin: 1, - }, - }; - const removedUser = await this.findOneAndRemove(filter, options).exec(); - - if (!isEmpty(removedUser)) { - return { - success: true, - message: "Successfully removed the user", - data: removedUser._doc, - status: httpStatus.OK, - }; - } else if (isEmpty(removedUser)) { - next( - new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { - message: "Provided User does not exist, please crosscheck", - }) - ); - } - } catch (error) { - logObject("the models error", error); - logger.error(`🐛🐛 Internal Server Error -- ${error.message}`); - next( - new HttpError( - "Internal Server Error", - httpStatus.INTERNAL_SERVER_ERROR, - { message: error.message } - ) - ); - } - }, -}; - -UserSchema.methods = { - authenticateUser(password) { - return bcrypt.compareSync(password, this.password); - }, - newToken() { - const token = accessCodeGenerator.generate( - constants.RANDOM_PASSWORD_CONFIGURATION(10) - ); - const hashedToken = bcrypt.hashSync(token, saltRounds); - return { - accessToken: hashedToken, - plainTextToken: `${token.id}|${plainTextToken}`, - }; - }, - async toAuthJSON() { - const token = await this.createToken(); - return { - _id: this._id, - userName: this.userName, - token: `JWT ${token}`, - email: this.email, - }; - }, - toJSON() { - return { - _id: this._id, - userName: this.userName, - email: this.email, - firstName: this.firstName, - organization: this.organization, - long_organization: this.long_organization, - group_roles: this.group_roles, - network_roles: this.network_roles, - privilege: this.privilege, - lastName: this.lastName, - country: this.country, - website: this.website, - category: this.category, - jobTitle: this.jobTitle, - profilePicture: this.profilePicture, - phoneNumber: this.phoneNumber, - description: this.description, - createdAt: this.createdAt, - updatedAt: this.updatedAt, - role: this.role, - verified: this.verified, - analyticsVersion: this.analyticsVersion, - rateLimit: this.rateLimit, - lastLogin: this.lastLogin, - isActive: this.isActive, - loginCount: this.loginCount, - timezone: this.timezone, - }; - }, -}; - -const UserModel = (tenant) => { - const defaultTenant = constants.DEFAULT_TENANT || "airqo"; - const dbTenant = isEmpty(tenant) ? defaultTenant : tenant; - try { - let users = mongoose.model("users"); - return users; - } catch (error) { - let users = getModelByTenant(dbTenant, "user", UserSchema); - return users; - } -}; -UserSchema.methods.createToken = async function () { - try { - const filter = { _id: this._id }; - const userWithDerivedAttributes = await UserModel("airqo").list({ filter }); - if ( - userWithDerivedAttributes.success && - userWithDerivedAttributes.success === false - ) { - logger.error( - `Internal Server Error -- ${JSON.stringify(userWithDerivedAttributes)}` - ); - return userWithDerivedAttributes; - } else { - const user = userWithDerivedAttributes.data[0]; - const oneDayExpiry = Math.floor(Date.now() / 1000) + 24 * 60 * 60; - const oneHourExpiry = Math.floor(Date.now() / 1000) + 60 * 60; - logObject("user", user); - return jwt.sign( - { - _id: user._id, - firstName: user.firstName, - lastName: user.lastName, - userName: user.userName, - email: user.email, - organization: user.organization, - long_organization: user.long_organization, - privilege: user.privilege, - role: user.role, - country: user.country, - profilePicture: user.profilePicture, - phoneNumber: user.phoneNumber, - createdAt: user.createdAt, - updatedAt: user.updatedAt, - rateLimit: user.rateLimit, - lastLogin: user.lastLogin, - // exp: oneHourExpiry, - }, - constants.JWT_SECRET - ); - } - } catch (error) { - logger.error(`🐛🐛 Internal Server Error --- ${error.message}`); - } -}; - -module.exports = UserModel; diff --git a/src/auth-service/models/log.js b/src/auth-service/models/log.js index fa857c67c5..83378df314 100644 --- a/src/auth-service/models/log.js +++ b/src/auth-service/models/log.js @@ -29,7 +29,17 @@ const logSchema = new mongoose.Schema( default: {}, }, }, - { timestamps: true } + { + timestamps: true, + indexes: [ + { + timestamp: 1, + "meta.email": 1, + "meta.service": 1, + "meta.endpoint": 1, + }, + ], + } ); logSchema.pre("save", function (next) { diff --git a/src/auth-service/routes/v2/analytics.js b/src/auth-service/routes/v2/analytics.js new file mode 100644 index 0000000000..e0413c5698 --- /dev/null +++ b/src/auth-service/routes/v2/analytics.js @@ -0,0 +1,155 @@ +const express = require("express"); +const router = express.Router(); +const createAnalyticsController = require("@controllers/create-analytics"); +const validateTenant = require("@middleware/validateTenant"); +const { + userEngagementValidators, + activityValidators, + cohortValidators, + predictiveValidators, + serviceAdoptionValidators, + benchmarkValidators, + topUsersValidators, + aggregateValidators, + retentionValidators, + healthScoreValidators, + behaviorValidators, + emailValidators, +} = require("@validators/analytics.validators"); + +router.get( + "/activities", + validateTenant(), + createAnalyticsController.listActivities +); + +router.get("/logs", validateTenant(), createAnalyticsController.listLogs); + +router.get( + "/user-stats", + validateTenant(), + createAnalyticsController.getUserStats +); + +router.get( + "/statistics", + validateTenant(), + createAnalyticsController.listStatistics +); + +// User Engagement Routes +router.get( + "/:email/engagement", + validateTenant(), + userEngagementValidators.getEngagement, + createAnalyticsController.getUserEngagement +); + +router.get( + "/:email/engagement/metrics", + validateTenant(), + userEngagementValidators.getMetrics, + createAnalyticsController.getEngagementMetrics +); + +// Activity Analysis Routes +router.get( + "/:email/activity-report", + validateTenant(), + activityValidators.getReport, + createAnalyticsController.getActivityReport +); + +// Cohort Analysis Routes +router.get( + "/cohorts", + validateTenant(), + cohortValidators.getAnalysis, + createAnalyticsController.getCohortAnalysis +); + +// Predictive Analytics Routes +router.get( + "/:email/predictions", + validateTenant(), + predictiveValidators.getPredictions, + createAnalyticsController.getPredictiveAnalytics +); + +// Service Adoption Routes +router.get( + "/:email/service-adoption", + validateTenant(), + serviceAdoptionValidators.getAdoption, + createAnalyticsController.getServiceAdoption +); + +// Benchmark Routes +router.get( + "/benchmarks", + validateTenant(), + benchmarkValidators.getBenchmarks, + createAnalyticsController.getBenchmarks +); + +// Top Users Routes +router.get( + "/top-users", + validateTenant(), + topUsersValidators.getTopUsers, + createAnalyticsController.getTopUsers +); + +// Aggregated Analytics Routes +router.get( + "/aggregate", + validateTenant(), + aggregateValidators.getAggregated, + createAnalyticsController.getAggregatedAnalytics +); + +// Retention Analysis Routes +router.get( + "/retention", + validateTenant(), + retentionValidators.getRetention, + createAnalyticsController.getRetentionAnalysis +); + +// Health Score Routes +router.get( + "/:email/health-score", + validateTenant(), + healthScoreValidators.getHealthScore, + createAnalyticsController.getEngagementHealth +); + +// Behavior Pattern Routes +router.get( + "/:email/behavior", + validateTenant(), + behaviorValidators.getBehavior, + createAnalyticsController.getBehaviorPatterns +); + +// Existing Email Routes +router.post( + "/send", + validateTenant(), + emailValidators.sendEmails, + createAnalyticsController.send +); + +router.post( + "/retrieve", + validateTenant(), + emailValidators.retrieveStats, + createAnalyticsController.fetchUserStats +); + +router.get( + "/validate-environment", + createAnalyticsController.validateEnvironment +); + +module.exports = router; diff --git a/src/auth-service/routes/v2/groups.js b/src/auth-service/routes/v2/groups.js index b9dcd08657..54fed09699 100644 --- a/src/auth-service/routes/v2/groups.js +++ b/src/auth-service/routes/v2/groups.js @@ -349,6 +349,53 @@ router.get( ]), createGroupController.listSummary ); + +router.put( + "/:grp_id/set-manager/:user_id", + oneOf([ + [ + query("tenant") + .optional() + .notEmpty() + .withMessage("tenant cannot be empty if provided") + .bail() + .trim() + .toLowerCase() + .isIn(["kcca", "airqo"]) + .withMessage("the tenant value is not among the expected ones"), + ], + ]), + oneOf([ + [ + param("grp_id") + .exists() + .withMessage("the group ID param is missing in the request") + .bail() + .trim() + .isMongoId() + .withMessage("the group ID must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + param("user_id") + .exists() + .withMessage("the user ID param is missing in the request") + .bail() + .trim() + .isMongoId() + .withMessage("the user ID must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + ], + ]), + setJWTAuth, + authJWT, + createGroupController.setManager +); + router.get( "/:grp_id/assigned-users", oneOf([ @@ -506,11 +553,11 @@ router.delete( [ param("grp_id") .exists() - .withMessage("the network ID is missing in request") + .withMessage("the group ID is missing in request") .bail() .trim() .isMongoId() - .withMessage("the network ID must be an object ID") + .withMessage("the group ID must be an object ID") .bail() .customSanitizer((value) => { return ObjectId(value); @@ -551,11 +598,11 @@ router.delete( [ param("grp_id") .exists() - .withMessage("the network ID is missing in request") + .withMessage("the group ID is missing in request") .bail() .trim() .isMongoId() - .withMessage("the network ID must be an object ID") + .withMessage("the group ID must be an object ID") .bail() .customSanitizer((value) => { return ObjectId(value); diff --git a/src/auth-service/routes/v2/index.js b/src/auth-service/routes/v2/index.js index a0cee14f1b..fb8c364358 100644 --- a/src/auth-service/routes/v2/index.js +++ b/src/auth-service/routes/v2/index.js @@ -6,6 +6,7 @@ router.use("/permissions", require("@routes/v2/permissions")); router.use("/favorites", require("@routes/v2/favorites")); router.use("/roles", require("@routes/v2/roles")); router.use("/inquiries", require("@routes/v2/inquiries")); +router.use("/analytics", require("@routes/v2/analytics")); router.use("/candidates", require("@routes/v2/candidates")); router.use("/requests", require("@routes/v2/requests")); router.use("/defaults", require("@routes/v2/defaults")); diff --git a/src/auth-service/routes/v2/users.js b/src/auth-service/routes/v2/users.js index 70198fe2d0..f7ef17d281 100644 --- a/src/auth-service/routes/v2/users.js +++ b/src/auth-service/routes/v2/users.js @@ -1131,7 +1131,7 @@ router.get( ); router.get( - "/analytics", + "/user-stats", oneOf([ query("tenant") .optional() diff --git a/src/auth-service/utils/create-analytics.js b/src/auth-service/utils/create-analytics.js new file mode 100644 index 0000000000..483acbf7ca --- /dev/null +++ b/src/auth-service/utils/create-analytics.js @@ -0,0 +1,2640 @@ +const constants = require("@config/constants"); +const { logObject, logText } = require("@utils/log"); +const mailer = require("@utils/mailer"); +const generateFilter = require("@utils/generate-filter"); +const { LogModel } = require("@models/log"); +const ActivityModel = require("@models/Activity"); +const UserModel = require("@models/User"); +const stringify = require("@utils/stringify"); +const log4js = require("log4js"); +const logger = log4js.getLogger( + `${constants.ENVIRONMENT} -- create-analytics-util` +); +const { HttpError } = require("@utils/errors"); +const httpStatus = require("http-status"); +const { + addMonthsToProvideDateTime, + monthsInfront, + isTimeEmpty, + getDifferenceInMonths, + addDays, +} = require("@utils/date"); + +const routesWithService = [ + { + method: "POST", + uriIncludes: [ + "api/v2/analytics/data-download", + "api/v1/analytics/data-download", + ], + service: "data-export-download", + action: "Export Data", + }, + { + method: "POST", + uriIncludes: [ + "api/v1/analytics/data-export", + "api/v2/analytics/data-export", + ], + service: "data-export-scheduling", + action: "Schedule Data Download", + }, + /**** Sites */ + { + method: "POST", + uriIncludes: ["/api/v2/devices/sites"], + service: "site-registry", + action: "Site Creation", + }, + { + method: "GET", + uriIncludes: ["/api/v2/devices/sites"], + service: "site-registry", + action: "View Sites", + }, + { + method: "PUT", + uriIncludes: ["/api/v2/devices/sites"], + service: "site-registry", + action: "Site Update", + }, + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/sites"], + service: "site-registry", + action: "Site Deletion", + }, + + /**** Devices */ + { + method: "DELETE", + uriIncludes: ["/api/v2/devices?"], + service: "device-registry", + action: "Device Deletion", + }, + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/soft?"], + service: "device-registry", + action: "Device SOFT Deletion", + }, + { + method: "PUT", + uriIncludes: ["/api/v2/devices?"], + service: "device-registry", + action: "Device Update", + }, + { + method: "PUT", + uriIncludes: ["/api/v2/devices/soft?"], + service: "device-registry", + action: "Device SOFT Update", + }, + { + method: "GET", + uriIncludes: ["/api/v2/devices?"], + service: "device-registry", + action: "View Devices", + }, + { + method: "POST", + uriIncludes: ["/api/v2/devices?"], + service: "device-registry", + action: "Device Creation", + }, + { + method: "POST", + uriIncludes: ["/api/v2/devices/soft?"], + service: "device-registry", + action: "Device SOFT Creation", + }, + /**** Cohorts */ + { + method: "GET", + uriIncludes: ["/api/v2/devices/cohorts"], + service: "cohort-registry", + action: "View Cohorts", + }, + + { + method: "POST", + uriIncludes: ["/api/v2/devices/cohorts"], + service: "cohort-registry", + action: "Create Cohorts", + }, + + { + method: "PUT", + uriIncludes: ["/api/v2/devices/cohorts"], + service: "cohort-registry", + action: "Update Cohort", + }, + + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/cohorts"], + service: "cohort-registry", + action: "Delete Cohort", + }, + + /**** Grids */ + + { + method: "GET", + uriIncludes: ["/api/v2/devices/grids"], + service: "grid-registry", + action: "View Grids", + }, + + { + method: "PUT", + uriIncludes: ["/api/v2/devices/grids"], + service: "grid-registry", + action: "Update Grid", + }, + + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/grids"], + service: "grid-registry", + action: "Delete Grid", + }, + + { + method: "POST", + uriIncludes: ["/api/v2/devices/grids"], + service: "grid-registry", + action: "Create Grid", + }, + + /**** AirQlouds */ + + { + method: "GET", + uriIncludes: ["/api/v2/devices/airqlouds"], + service: "airqloud-registry", + action: "View AirQlouds", + }, + { + method: "POST", + uriIncludes: ["/api/v2/devices/airqlouds"], + service: "airqloud-registry", + action: "AirQloud Creation", + }, + { + method: "PUT", + uriIncludes: ["/api/v2/devices/airqlouds"], + service: "airqloud-registry", + action: "AirQloud Update", + }, + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/airqlouds"], + service: "airqloud-registry", + action: "AirQloud Deletion", + }, + + /**** Site Activities */ + + { + method: "POST", + uriIncludes: ["/api/v2/devices/activities/maintain"], + service: "device-maintenance", + action: "Maintain Device", + }, + { + method: "POST", + uriIncludes: ["/api/v2/devices/activities/recall"], + service: "device-recall", + action: "Recall Device", + }, + { + method: "POST", + uriIncludes: ["/api/v2/devices/activities/deploy"], + service: "device-deployment", + action: "Deploy Device", + }, + + /**** Users */ + { + method: "POST", + uriIncludes: ["api/v2/users", "api/v1/users"], + service: "auth", + action: "Create User", + }, + { + method: "GET", + uriIncludes: ["api/v2/users", "api/v1/users"], + service: "auth", + action: "View Users", + }, + { + method: "PUT", + uriIncludes: ["api/v2/users", "api/v1/users"], + service: "auth", + action: "Update User", + }, + { + method: "DELETE", + uriIncludes: ["api/v2/users", "api/v1/users"], + service: "auth", + action: "Delete User", + }, + + /****Incentives*/ + { + method: "POST", + uriIncludes: [ + "api/v1/incentives/transactions/accounts/payments", + "api/v2/incentives/transactions/accounts/payments", + ], + service: "incentives", + action: "Add Money to Organizational Account", + }, + { + method: "POST", + uriIncludes: [ + "api/v1/incentives/transactions/hosts", + "api/v2/incentives/transactions/hosts", + ], + service: "incentives", + action: "Send Money to Host", + }, + + /**** Calibrate */ + { + method: "POST", + uriIncludes: ["/api/v1/calibrate", "/api/v2/calibrate"], + service: "calibrate", + action: "calibrate device", + }, + + /**** Locate */ + { + method: "POST", + uriIncludes: ["/api/v1/locate", "/api/v2/locate"], + service: "locate", + action: "Identify Suitable Device Locations", + }, + + /**** Fault Detection */ + { + method: "POST", + uriIncludes: ["/api/v1/predict-faults", "/api/v2/predict-faults"], + service: "fault-detection", + action: "Detect Faults", + }, + + /**** Readings... */ + { + method: "GET", + uriIncludes: [ + "/api/v2/devices/measurements", + "/api/v2/devices/events", + "/api/v2/devices/readings", + ], + service: "events-registry", + action: " Retrieve Measurements", + }, + + /**** Data Proxy */ + { + method: "GET", + uriIncludes: ["/api/v2/data"], + service: "data-mgt", + action: "Retrieve Data", + }, + { + method: "GET", + uriIncludes: ["/api/v2/data-proxy"], + service: "data-proxy", + action: "Retrieve Data", + }, + + /*****Analytics */ + { + method: "GET", + uriIncludes: ["/api/v2/analytics/dashboard/sites"], + service: "analytics", + action: "Retrieve Sites on Analytics Page", + }, + { + method: "GET", + uriIncludes: ["/api/v2/analytics/dashboard/historical/daily-averages"], + service: "analytics", + action: "Retrieve Daily Averages on Analytics Page", + }, + { + method: "GET", + uriIncludes: ["/api/v2/analytics/dashboard/exceedances-devices"], + service: "analytics", + action: "Retrieve Exceedances on Analytics Page", + }, + + /*****KYA lessons */ + + { + method: "GET", + uriIncludes: ["/api/v2/devices/kya/lessons/users"], + service: "kya", + action: "Retrieve KYA lessons", + }, + { + method: "POST", + uriIncludes: ["/api/v2/devices/kya/lessons/users"], + service: "kya", + action: "Create KYA lesson", + }, + { + method: "PUT", + uriIncludes: ["/api/v2/devices/kya/lessons/users"], + service: "kya", + action: "Update KYA lesson", + }, + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/kya/lessons/users"], + service: "kya", + action: "Delete KYA lesson", + }, + /*****KYA Quizzes */ + { + method: "GET", + uriIncludes: ["/api/v2/devices/kya/quizzes/users"], + service: "kya", + action: "Retrieve KYA quizzes", + }, + + { + method: "POST", + uriIncludes: ["/api/v2/devices/kya/quizzes"], + service: "kya", + action: "Create KYA quizzes", + }, + + { + method: "PUT", + uriIncludes: ["/api/v2/devices/kya/quizzes"], + service: "kya", + action: "Update KYA quiz", + }, + + { + method: "DELETE", + uriIncludes: ["/api/v2/devices/kya/quizzes"], + service: "kya", + action: "Delete KYA quiz", + }, + + /*****view */ + { + method: "GET", + uriIncludes: ["/api/v2/view/mobile-app/version-info"], + service: "mobile-version", + action: "View Mobile App Information", + }, + + /*****Predict */ + { + method: "GET", + uriIncludes: ["/api/v2/predict/daily-forecast"], + service: "predict", + action: "Retrieve Daily Forecasts", + }, + { + method: "GET", + uriIncludes: ["/api/v2/predict/hourly-forecast"], + service: "predict", + action: "Retrieve Hourly Forecasts", + }, + { + method: "GET", + uriIncludes: ["/api/v2/predict/heatmap"], + service: "predict", + action: "Retrieve Heatmap", + }, + + /*****Device Monitoring */ + { + method: "GET", + uriIncludes: ["/api/v2/monitor"], + service: "monitor", + action: "Retrieve Network Statistics Data", + }, + + { + method: "GET", + uriIncludes: ["/api/v2/meta-data"], + service: "meta-data", + action: "Retrieve Metadata", + }, + + { + method: "GET", + uriIncludes: ["/api/v2/network-uptime"], + service: "network-uptime", + action: "Retrieve Network Uptime Data", + }, +]; + +// Helper functions to calculate additional metrics +function calculateActivityDuration( + firstActivity, + lastActivity, + yearStart, + yearEnd +) { + // Ensure we're working with Date objects + const start = new Date(firstActivity); + const end = new Date(lastActivity); + const yearStartDate = new Date(yearStart); + const yearEndDate = new Date(yearEnd); + + // Use the year boundaries if dates fall outside + const effectiveStart = start < yearStartDate ? yearStartDate : start; + const effectiveEnd = end > yearEndDate ? yearEndDate : end; + + // Calculate duration + const duration = effectiveEnd - effectiveStart; + const days = Math.floor(duration / (1000 * 60 * 60 * 24)); + + // Calculate months more accurately + const months = Math.floor( + (effectiveEnd.getFullYear() - effectiveStart.getFullYear()) * 12 + + (effectiveEnd.getMonth() - effectiveStart.getMonth()) + ); + + return { + totalDays: days, + totalMonths: months, + yearStart: yearStartDate, + yearEnd: yearEndDate, + actualStart: effectiveStart, + actualEnd: effectiveEnd, + description: + months > 0 + ? `Active for ${months} month${months !== 1 ? "s" : ""}` + : `Active for ${days} day${days !== 1 ? "s" : ""}`, + }; +} + +// Helper function for calculating engagement score +function calculateEngagementScore({ + totalActions, + uniqueServices, + uniqueEndpoints, + activityDays, +}) { + // Use at least 30 days as denominator + const actionsPerDay = totalActions / Math.max(activityDays, 30); + + const serviceDiversity = + uniqueServices / Math.min(routesWithService.length, 20); + const endpointDiversity = Math.min(uniqueEndpoints / 10, 1); + + return ( + (actionsPerDay * 0.4 + serviceDiversity * 0.3 + endpointDiversity * 0.3) * + 100 + ); +} + +// Simplified engagement tier calculation +function calculateEngagementTier(score) { + if (score >= 80) return "Elite User"; + if (score >= 65) return "Super User"; + if (score >= 45) return "High Engagement"; + if (score >= 25) return "Moderate Engagement"; + return "Low Engagement"; +} + +/** + * Calculates user engagement metrics and analytics + * @param {Object} params Engagement calculation parameters + * @param {number} params.totalActions Total number of API calls/actions + * @param {Array} params.services List of unique services used + * @param {Array} params.endpoints List of unique endpoints accessed + * @param {Date} params.firstActivity Date of first activity + * @param {Date} params.lastActivity Date of last activity + * @param {Array} params.dailyStats Array of daily activity statistics + * @returns {Object} Comprehensive engagement metrics + */ +async function calculateUserEngagementMetrics({ + totalActions = 0, + services = [], + endpoints = [], + firstActivity, + lastActivity, + dailyStats = [], +}) { + const TOTAL_AVAILABLE_SERVICES = constants.AVAILABLE_SERVICES?.length || 20; + const TOTAL_AVAILABLE_ENDPOINTS = constants.AVAILABLE_ENDPOINTS?.length || 50; + + // Calculate base metrics + const activityDuration = + lastActivity && firstActivity + ? Math.ceil((lastActivity - firstActivity) / (1000 * 60 * 60 * 24)) + : 1; + + const actionsPerDay = totalActions / Math.max(activityDuration, 30); + + const uniqueServices = new Set(services.map((s) => s.name)); + const uniqueEndpoints = new Set(endpoints.map((e) => e.name)); + + const serviceDiversity = Math.min( + uniqueServices.size / TOTAL_AVAILABLE_SERVICES, + 1 + ); + const endpointDiversity = Math.min( + uniqueEndpoints.size / TOTAL_AVAILABLE_ENDPOINTS, + 1 + ); + + // Calculate activity patterns + const activityPatterns = analyzeActivityPatterns(dailyStats); + + // Calculate service usage patterns + const serviceUsagePatterns = analyzeServiceUsagePatterns(services); + + // Calculate consistency score + const consistencyScore = calculateConsistencyScore( + dailyStats, + activityDuration + ); + + // Calculate growth trend + const growthTrend = calculateGrowthTrend(dailyStats); + + // Calculate engagement score with enhanced weights + const engagementScore = calculateWeightedScore({ + actionsPerDay, + serviceDiversity, + endpointDiversity, + consistencyScore, + growthTrend, + }); + + return { + basicMetrics: { + totalActions, + activityDuration, + actionsPerDay, + uniqueServicesCount: uniqueServices.size, + uniqueEndpointsCount: uniqueEndpoints.size, + }, + engagementMetrics: { + serviceDiversity, + endpointDiversity, + consistencyScore, + growthTrend, + engagementScore, + }, + activityPatterns, + serviceUsagePatterns, + recommendations: generateRecommendations({ + serviceDiversity, + endpointDiversity, + consistencyScore, + serviceUsagePatterns, + }), + }; +} + +/** + * Analyzes daily activity patterns + * @param {Array} dailyStats Array of daily activity records + * @returns {Object} Activity pattern analysis + */ +function analyzeActivityPatterns(dailyStats) { + const activeWeekdays = new Set(); + const hourlyDistribution = new Array(24).fill(0); + const weeklyAverage = []; + + dailyStats.forEach((day) => { + const date = new Date(day.date); + activeWeekdays.add(date.getDay()); + + // Analyze hourly patterns if available + if (day.hourlyBreakdown) { + day.hourlyBreakdown.forEach((count, hour) => { + hourlyDistribution[hour] += count; + }); + } + + // Calculate weekly averages + if (weeklyAverage.length < 12) { + // Track last 12 weeks + weeklyAverage.push(day.totalActions); + } + }); + + return { + preferredDays: Array.from(activeWeekdays), + peakHours: findPeakHours(hourlyDistribution), + weeklyTrend: calculateWeeklyTrend(weeklyAverage), + consistency: calculateActivityConsistency(dailyStats), + }; +} + +/** + * Analyzes service usage patterns + * @param {Array} services List of services with usage counts + * @returns {Object} Service usage analysis + */ +function analyzeServiceUsagePatterns(services) { + const serviceUsage = services.reduce((acc, service) => { + acc[service.name] = acc[service.name] || { count: 0, frequency: 0 }; + acc[service.name].count += service.count; + acc[service.name].frequency++; + return acc; + }, {}); + + // Calculate primary and secondary services + const sortedServices = Object.entries(serviceUsage).sort( + ([, a], [, b]) => b.count - a.count + ); + + return { + primaryServices: sortedServices.slice(0, 3).map(([name]) => name), + serviceDistribution: sortedServices.reduce((acc, [name, data]) => { + acc[name] = (data.count / services.length) * 100; + return acc; + }, {}), + unusedServices: findUnusedServices(services), + }; +} + +/** + * Calculates consistency score based on activity patterns + * @param {Array} dailyStats Daily activity statistics + * @param {number} duration Total duration in days + * @returns {number} Consistency score between 0-1 + */ +function calculateConsistencyScore(dailyStats, duration) { + if (!dailyStats.length) return 0; + + const activeDaysCount = dailyStats.length; + const activeRatio = activeDaysCount / duration; + + // Calculate variance in daily activity + const avgActions = + dailyStats.reduce((sum, day) => sum + day.totalActions, 0) / + activeDaysCount; + const variance = + dailyStats.reduce((sum, day) => { + return sum + Math.pow(day.totalActions - avgActions, 2); + }, 0) / activeDaysCount; + + const consistencyFactor = 1 / (1 + Math.sqrt(variance) / avgActions); + + return activeRatio * 0.6 + consistencyFactor * 0.4; +} + +/** + * Calculates growth trend based on activity history + * @param {Array} dailyStats Daily activity statistics + * @returns {number} Growth trend indicator between -1 and 1 + */ +function calculateGrowthTrend(dailyStats) { + if (dailyStats.length < 7) return 0; + + const weeklyTotals = []; + let currentWeek = []; + + dailyStats.forEach((day) => { + currentWeek.push(day.totalActions); + if (currentWeek.length === 7) { + weeklyTotals.push(currentWeek.reduce((a, b) => a + b, 0)); + currentWeek = []; + } + }); + + if (weeklyTotals.length < 2) return 0; + + const growth = + weeklyTotals.slice(1).reduce((acc, curr, idx) => { + return acc + (curr - weeklyTotals[idx]) / weeklyTotals[idx]; + }, 0) / + (weeklyTotals.length - 1); + + return Math.max(-1, Math.min(1, growth)); +} + +/** + * Generates personalized recommendations based on usage patterns + * @param {Object} params User engagement metrics + * @returns {Array} List of recommendations + */ +function generateRecommendations({ + serviceDiversity, + endpointDiversity, + consistencyScore, + serviceUsagePatterns, +}) { + const recommendations = []; + + if (serviceDiversity < 0.3) { + recommendations.push({ + type: "service_exploration", + priority: "high", + message: + "Consider exploring additional services to enhance your integration", + suggestedServices: serviceUsagePatterns.unusedServices.slice(0, 3), + }); + } + + if (endpointDiversity < 0.3) { + recommendations.push({ + type: "endpoint_utilization", + priority: "medium", + message: "Explore additional endpoints within your current services", + }); + } + + if (consistencyScore < 0.5) { + recommendations.push({ + type: "consistency", + priority: "high", + message: "Consider establishing a more regular usage pattern", + }); + } + + return recommendations; +} + +/** + * Determines user engagement tier with enhanced criteria + * @param {Object} params Engagement tier calculation parameters + * @param {Object} params.engagementMetrics Complete engagement metrics + * @param {Date} params.lastActivity Date of last activity + * @returns {Object} Detailed engagement classification + */ +async function determineUserEngagementTier({ + engagementMetrics, + lastActivity, +}) { + const { engagementScore, consistencyScore, growthTrend } = engagementMetrics; + + const daysSinceLastActivity = lastActivity + ? Math.ceil((new Date() - lastActivity) / (1000 * 60 * 60 * 24)) + : Infinity; + + // Define tier criteria + const tierCriteria = { + Elite: { + minScore: 80, + minConsistency: 0.8, + minGrowth: 0.1, + }, + Super: { + minScore: 65, + minConsistency: 0.6, + minGrowth: 0, + }, + High: { + minScore: 45, + minConsistency: 0.4, + minGrowth: -0.1, + }, + Moderate: { + minScore: 25, + minConsistency: 0.2, + minGrowth: -0.2, + }, + }; + + // Return inactive status if no activity in last 90 days + if (daysSinceLastActivity > 90) { + return { + tier: "Inactive", + status: "dormant", + reactivationPriority: daysSinceLastActivity > 180 ? "high" : "medium", + }; + } + + // Determine tier based on multiple criteria + for (const [tier, criteria] of Object.entries(tierCriteria)) { + if ( + engagementScore >= criteria.minScore && + consistencyScore >= criteria.minConsistency && + growthTrend >= criteria.minGrowth + ) { + return { + tier: `${tier} User`, + status: "active", + metrics: { + score: engagementScore, + consistency: consistencyScore, + growth: growthTrend, + }, + nextTierProgress: calculateNextTierProgress( + engagementMetrics, + tier, + tierCriteria + ), + }; + } + } + + return { + tier: "Low Engagement", + status: "at_risk", + metrics: { + score: engagementScore, + consistency: consistencyScore, + growth: growthTrend, + }, + improvementAreas: identifyImprovementAreas(engagementMetrics), + }; +} + +/** + * Calculates progress towards next engagement tier + * @param {Object} metrics Current engagement metrics + * @param {string} currentTier Current tier + * @param {Object} tierCriteria Tier qualification criteria + * @returns {Object} Progress metrics towards next tier + */ +function calculateNextTierProgress(metrics, currentTier, tierCriteria) { + const tiers = Object.keys(tierCriteria); + const currentTierIndex = tiers.indexOf(currentTier); + + if (currentTierIndex === 0) return null; // Already at highest tier + + const nextTier = tiers[currentTierIndex - 1]; + const nextTierCriteria = tierCriteria[nextTier]; + + return { + nextTier, + scoreProgress: (metrics.engagementScore / nextTierCriteria.minScore) * 100, + consistencyProgress: + (metrics.consistencyScore / nextTierCriteria.minConsistency) * 100, + growthProgress: + metrics.growthTrend > 0 + ? (metrics.growthTrend / nextTierCriteria.minGrowth) * 100 + : 0, + }; +} + +function capitalizeWord(word) { + return word.charAt(0).toUpperCase() + word.slice(1).toLowerCase(); +} + +function formatServiceName(serviceName) { + // Handle null, undefined or empty strings + if (!serviceName) return ""; + + // Replace hyphens with spaces and split into words + return serviceName.split("-").map(capitalizeWord).join(" "); +} + +// Get a user's monthly stats for a specific period +const getUserMonthlyStats = async (email, year, month, tenant = "airqo") => { + const stats = await ActivityModel(tenant).findOne( + { + email, + "monthlyStats.year": year, + "monthlyStats.month": month, + }, + { + "monthlyStats.$": 1, + overallStats: 1, + } + ); + return stats; +}; + +// Get a user's daily stats for a date range +const getUserDailyStats = async ( + email, + startDate, + endDate, + tenant = "airqo" +) => { + const stats = await ActivityModel(tenant).findOne( + { + email, + "dailyStats.date": { + $gte: startDate, + $lte: endDate, + }, + }, + { + dailyStats: { + $filter: { + input: "$dailyStats", + as: "day", + cond: { + $and: [ + { $gte: ["$$day.date", startDate] }, + { $lte: ["$$day.date", endDate] }, + ], + }, + }, + }, + } + ); + return stats; +}; + +// Get top users by engagement score for a specific month +const getTopUsers = async (year, month, limit = 10, tenant = "airqo") => { + const users = await ActivityModel(tenant) + .find({ + "monthlyStats.year": year, + "monthlyStats.month": month, + }) + .sort({ "monthlyStats.$.engagementScore": -1 }) + .limit(limit); + return users; +}; + +/** + * Predicts future engagement trends and potential churn risk + * @param {Object} params Prediction parameters + * @param {Array} params.dailyStats Historical daily statistics + * @param {Array} params.monthlyStats Monthly statistics + * @param {Object} params.currentMetrics Current engagement metrics + * @returns {Object} Prediction analysis and risk assessment + */ +async function predictEngagementTrends({ + dailyStats = [], + monthlyStats = [], + currentMetrics = {}, +}) { + // Calculate trailing indicators + const trailingIndicators = calculateTrailingIndicators(dailyStats); + + // Analyze seasonal patterns + const seasonalPatterns = analyzeSeasonalPatterns(monthlyStats); + + // Predict future engagement + const predictions = generateEngagementPredictions({ + trailingIndicators, + seasonalPatterns, + currentMetrics, + }); + + // Calculate churn risk + const churnRisk = assessChurnRisk({ + trailingIndicators, + predictions, + currentMetrics, + }); + + return { + shortTermPredictions: predictions.shortTerm, + longTermPredictions: predictions.longTerm, + churnRisk, + seasonalPatterns, + recommendedActions: generatePreventiveActions(churnRisk), + }; +} + +/** + * Performs cohort analysis for user segments + * @param {Array} users Array of user activities + * @param {Object} params Analysis parameters + * @returns {Object} Cohort analysis results + */ +async function performCohortAnalysis( + users, + { + cohortPeriod = "monthly", + metrics = ["engagement", "retention", "activity"], + minCohortSize = 5, + } = {} +) { + // Group users into cohorts + const cohorts = groupIntoCohorts(users, cohortPeriod); + + // Calculate cohort metrics + const cohortMetrics = calculateCohortMetrics(cohorts, metrics); + + // Analyze cohort patterns + const patterns = analyzeCohortPatterns(cohortMetrics); + + return { + cohortMetrics, + patterns, + insights: generateCohortInsights(patterns), + benchmarks: calculateCohortBenchmarks(cohortMetrics), + }; +} + +/** + * Performs competitive benchmarking against platform averages + * @param {Object} userMetrics Individual user metrics + * @param {Object} benchmarkData Platform benchmark data + * @returns {Object} Benchmark comparison results + */ +function performBenchmarkAnalysis(userMetrics, benchmarkData) { + const comparisons = {}; + const insights = []; + + // Compare key metrics against benchmarks + Object.entries(userMetrics).forEach(([metric, value]) => { + if (benchmarkData[metric]) { + const percentile = calculatePercentile( + value, + benchmarkData[metric].distribution + ); + const comparison = { + value, + benchmark: benchmarkData[metric].average, + percentile, + difference: + ((value - benchmarkData[metric].average) / + benchmarkData[metric].average) * + 100, + }; + comparisons[metric] = comparison; + + // Generate insights based on significant differences + if (Math.abs(comparison.difference) > 20) { + insights.push(generateBenchmarkInsight(metric, comparison)); + } + } + }); + + return { comparisons, insights }; +} + +/** + * Generates engagement health score and detailed diagnostics + * @param {Object} params Health score parameters + * @returns {Object} Health score and diagnostics + */ +function calculateEngagementHealth({ + currentMetrics, + historicalData, + benchmarks, + predictions, +}) { + // Calculate core health indicators + const indicators = { + activity: calculateActivityHealth(currentMetrics, historicalData), + growth: calculateGrowthHealth(currentMetrics, predictions), + sustainability: calculateSustainabilityHealth(currentMetrics, benchmarks), + diversity: calculateDiversityHealth(currentMetrics), + }; + + // Generate overall health score + const healthScore = calculateOverallHealth(indicators); + + // Identify health factors + const { strengths, weaknesses } = identifyHealthFactors(indicators); + + return { + healthScore, + indicators, + diagnosis: { + strengths, + weaknesses, + prognosis: generateHealthPrognosis(healthScore, predictions), + }, + recommendations: generateHealthRecommendations(indicators), + }; +} + +/** + * Analyzes user behavior patterns and segments + * @param {Array} activityData User activity data + * @returns {Object} Behavior analysis results + */ +function analyzeBehaviorPatterns(activityData) { + // Identify usage patterns + const patterns = identifyUsagePatterns(activityData); + + // Segment user behavior + const segments = segmentUserBehavior(patterns); + + // Analyze feature adoption + const featureAdoption = analyzeFeatureAdoption(activityData); + + return { + patterns, + segments, + featureAdoption, + recommendations: generateBehaviorRecommendations(segments), + }; +} + +/** + * Generates detailed activity analytics report + * @param {Object} params Report parameters + * @returns {Object} Comprehensive activity report + */ +async function generateActivityReport({ + userId, + timeframe = "last30days", + metrics = ["all"], + includeProjections = true, +}) { + // Gather all required data + const activityData = await aggregateActivityData(userId, timeframe); + const benchmarks = await getBenchmarkData(timeframe); + + // Calculate current metrics + const currentMetrics = await calculateUserEngagementMetrics(activityData); + + // Generate predictions if requested + const predictions = includeProjections + ? await predictEngagementTrends({ + dailyStats: activityData.dailyStats, + monthlyStats: activityData.monthlyStats, + currentMetrics, + }) + : null; + + // Calculate engagement health + const healthAnalysis = calculateEngagementHealth({ + currentMetrics, + historicalData: activityData, + benchmarks, + predictions, + }); + + // Analyze behavior patterns + const behaviorAnalysis = analyzeBehaviorPatterns(activityData); + + return { + summary: { + userId, + timeframe, + healthScore: healthAnalysis.healthScore, + engagementTier: await determineUserEngagementTier({ + engagementMetrics: currentMetrics, + lastActivity: activityData.lastActivity, + }), + }, + metrics: currentMetrics, + health: healthAnalysis, + behavior: behaviorAnalysis, + predictions: predictions, + benchmarks: await performBenchmarkAnalysis(currentMetrics, benchmarks), + recommendations: prioritizeRecommendations([ + ...healthAnalysis.recommendations, + ...behaviorAnalysis.recommendations, + ]), + }; +} + +/** + * Analyzes service adoption and usage maturity + * @param {Object} params Service analysis parameters + * @returns {Object} Service adoption analysis + */ +function analyzeServiceAdoption({ services, endpoints, historicalUsage }) { + // Calculate adoption metrics + const adoptionMetrics = calculateAdoptionMetrics(services, endpoints); + + // Analyze usage maturity + const maturityAnalysis = assessUsageMaturity(historicalUsage); + + // Identify adoption barriers + const barriers = identifyAdoptionBarriers(adoptionMetrics, maturityAnalysis); + + return { + adoptionMetrics, + maturityLevel: maturityAnalysis.level, + barriers, + recommendations: generateAdoptionRecommendations(barriers), + }; +} + +// Helper functions implementation + +/** + * Calculates trailing indicators from daily statistics + * @param {Array} dailyStats Array of daily activity records + * @returns {Object} Trailing indicators and trends + */ +function calculateTrailingIndicators(dailyStats) { + // Sort dailyStats by date + const sortedStats = [...dailyStats].sort( + (a, b) => new Date(a.date) - new Date(b.date) + ); + + // Calculate trailing windows + const windows = { + "7d": sortedStats.slice(-7), + "30d": sortedStats.slice(-30), + "90d": sortedStats.slice(-90), + }; + + // Calculate metrics for each window + const indicators = {}; + Object.entries(windows).forEach(([period, stats]) => { + const totalActions = stats.reduce((sum, day) => sum + day.totalActions, 0); + const avgActions = totalActions / stats.length; + const variance = + stats.reduce((sum, day) => { + return sum + Math.pow(day.totalActions - avgActions, 2); + }, 0) / stats.length; + + indicators[period] = { + totalActions, + avgActions, + variance, + volatility: Math.sqrt(variance) / avgActions, + trend: calculateTrend(stats.map((day) => day.totalActions)), + }; + }); + + return { + indicators, + momentum: calculateMomentum(indicators), + stability: assessStability(indicators), + }; +} + +/** + * Analyzes seasonal patterns in monthly statistics + * @param {Array} monthlyStats Array of monthly statistics + * @returns {Object} Seasonal patterns and trends + */ +function analyzeSeasonalPatterns(monthlyStats) { + const patterns = { + monthly: {}, + quarterly: {}, + yearly: {}, + }; + + // Analyze monthly patterns + for (let month = 0; month < 12; month++) { + const monthData = monthlyStats.filter( + (stat) => new Date(stat.date).getMonth() === month + ); + patterns.monthly[month] = calculateMonthlyPattern(monthData); + } + + // Analyze quarterly patterns + for (let quarter = 0; quarter < 4; quarter++) { + const quarterData = monthlyStats.filter( + (stat) => Math.floor(new Date(stat.date).getMonth() / 3) === quarter + ); + patterns.quarterly[quarter] = calculateQuarterlyPattern(quarterData); + } + + // Calculate year-over-year trends + patterns.yearly = calculateYearlyPatterns(monthlyStats); + + return { + patterns, + seasonalityIndex: calculateSeasonalityIndex(patterns), + peakPeriods: identifyPeakPeriods(patterns), + troughPeriods: identifyTroughPeriods(patterns), + }; +} + +/** + * Generates engagement predictions based on historical data + * @param {Object} params Prediction parameters + * @returns {Object} Short and long-term predictions + */ +function generateEngagementPredictions({ + trailingIndicators, + seasonalPatterns, + currentMetrics, +}) { + const shortTermPrediction = generateShortTermPrediction({ + trailingIndicators, + currentMetrics, + }); + + const longTermPrediction = generateLongTermPrediction({ + seasonalPatterns, + currentMetrics, + }); + + const confidenceScores = calculatePredictionConfidence({ + shortTermPrediction, + longTermPrediction, + historicalAccuracy: trailingIndicators, + }); + + return { + shortTerm: { + ...shortTermPrediction, + confidence: confidenceScores.shortTerm, + }, + longTerm: { + ...longTermPrediction, + confidence: confidenceScores.longTerm, + }, + factors: identifyPredictionFactors({ + trailingIndicators, + seasonalPatterns, + }), + }; +} + +/** + * Assesses risk of user churn based on engagement metrics + * @param {Object} params Risk assessment parameters + * @returns {Object} Churn risk analysis + */ +function assessChurnRisk({ trailingIndicators, predictions, currentMetrics }) { + // Calculate risk factors + const riskFactors = { + activityDecline: calculateActivityDecline(trailingIndicators), + engagementDrop: calculateEngagementDrop(currentMetrics), + predictedChurn: evaluatePredictedChurn(predictions), + }; + + // Calculate risk scores + const riskScores = { + immediate: calculateImmediateRisk(riskFactors), + shortTerm: calculateShortTermRisk(riskFactors), + longTerm: calculateLongTermRisk(riskFactors), + }; + + return { + riskLevel: determineRiskLevel(riskScores), + riskFactors, + riskScores, + timeToChurn: estimateTimeToChurn(riskScores, trailingIndicators), + }; +} + +/** + * Generates preventive actions based on churn risk analysis + * @param {Object} churnRisk Churn risk analysis results + * @returns {Array} Prioritized preventive actions + */ +function generatePreventiveActions(churnRisk) { + const actions = []; + + // Generate immediate actions for high-risk users + if (churnRisk.riskLevel === "high") { + actions.push(...generateHighRiskActions(churnRisk)); + } + + // Generate proactive actions for medium-risk users + if (churnRisk.riskLevel === "medium") { + actions.push(...generateMediumRiskActions(churnRisk)); + } + + // Generate monitoring actions for low-risk users + if (churnRisk.riskLevel === "low") { + actions.push(...generateLowRiskActions(churnRisk)); + } + + return prioritizeActions(actions, churnRisk); +} + +/** + * Groups users into cohorts based on specified period + * @param {Array} users Array of user records + * @param {string} period Cohort period ('daily', 'weekly', 'monthly') + * @returns {Object} Grouped cohorts + */ +function groupIntoCohorts(users, period) { + const cohorts = {}; + + users.forEach((user) => { + const cohortKey = generateCohortKey(user.firstActivity, period); + cohorts[cohortKey] = cohorts[cohortKey] || []; + cohorts[cohortKey].push(user); + }); + + return filterValidCohorts(cohorts); +} + +/** + * Calculates metrics for each cohort + * @param {Object} cohorts Grouped cohort data + * @param {Array} metrics Metrics to calculate + * @returns {Object} Cohort metrics + */ +function calculateCohortMetrics(cohorts, metrics) { + const cohortMetrics = {}; + + Object.entries(cohorts).forEach(([cohortKey, users]) => { + cohortMetrics[cohortKey] = { + size: users.length, + metrics: {}, + }; + + metrics.forEach((metric) => { + cohortMetrics[cohortKey].metrics[metric] = calculateMetric(users, metric); + }); + }); + + return cohortMetrics; +} + +/** + * Analyzes patterns across different cohorts + * @param {Object} cohortMetrics Calculated cohort metrics + * @returns {Object} Pattern analysis + */ +function analyzeCohortPatterns(cohortMetrics) { + const patterns = { + retentionCurves: calculateRetentionCurves(cohortMetrics), + engagementTrends: analyzeEngagementTrends(cohortMetrics), + cohortComparisons: compareCohorts(cohortMetrics), + }; + + return { + patterns, + significantPatterns: identifySignificantPatterns(patterns), + anomalies: detectCohortAnomalies(patterns), + }; +} + +/** + * Generates insights from cohort patterns + * @param {Object} patterns Analyzed cohort patterns + * @returns {Array} Actionable insights + */ +function generateCohortInsights(patterns) { + const insights = []; + + // Analyze retention insights + insights.push(...analyzeRetentionInsights(patterns.retentionCurves)); + + // Analyze engagement insights + insights.push(...analyzeEngagementInsights(patterns.engagementTrends)); + + // Analyze comparative insights + insights.push(...analyzeComparativeInsights(patterns.cohortComparisons)); + + return prioritizeInsights(insights); +} + +/** + * Calculates benchmarks across cohorts + * @param {Object} cohortMetrics Cohort metrics data + * @returns {Object} Benchmark calculations + */ +function calculateCohortBenchmarks(cohortMetrics) { + const benchmarks = { + retention: calculateRetentionBenchmarks(cohortMetrics), + engagement: calculateEngagementBenchmarks(cohortMetrics), + growth: calculateGrowthBenchmarks(cohortMetrics), + }; + + return { + benchmarks, + trends: analyzeBenchmarkTrends(benchmarks), + recommendations: generateBenchmarkRecommendations(benchmarks), + }; +} + +// Internal helper functions... + +function calculateTrend(values) { + // Simple linear regression + const n = values.length; + const indices = Array.from({ length: n }, (_, i) => i); + + const sumX = indices.reduce((a, b) => a + b, 0); + const sumY = values.reduce((a, b) => a + b, 0); + const sumXY = indices.reduce((sum, x, i) => sum + x * values[i], 0); + const sumXX = indices.reduce((sum, x) => sum + x * x, 0); + + const slope = (n * sumXY - sumX * sumY) / (n * sumXX - sumX * sumX); + return slope / (sumY / n); // Normalized slope +} + +function calculateMomentum(indicators) { + const shortTerm = indicators["7d"].trend; + const mediumTerm = indicators["30d"].trend; + const longTerm = indicators["90d"].trend; + + return { + value: shortTerm * 0.5 + mediumTerm * 0.3 + longTerm * 0.2, + direction: Math.sign(shortTerm), + strength: Math.abs(shortTerm) / Math.abs(longTerm), + }; +} + +function assessStability(indicators) { + return { + shortTerm: 1 / (1 + indicators["7d"].volatility), + mediumTerm: 1 / (1 + indicators["30d"].volatility), + longTerm: 1 / (1 + indicators["90d"].volatility), + }; +} + +const analytics = { + // User Engagement Functions + getUserEngagement: async (request, next) => { + try { + const { email, tenant } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No engagement data found for this user", + }) + ); + } + + const engagementMetrics = await calculateUserEngagementMetrics({ + totalActions: user.overallStats.totalActions, + services: user.monthlyStats[0].topServices, + endpoints: user.monthlyStats[0].uniqueEndpoints, + firstActivity: user.overallStats.firstActivity, + lastActivity: user.overallStats.lastActivity, + dailyStats: user.dailyStats, + }); + + return { + success: true, + message: "User engagement data retrieved successfully", + data: engagementMetrics, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + getEngagementMetrics: async (request, next) => { + try { + const { email, tenant, timeframe = "last30days" } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No metrics found for this user", + }) + ); + } + + const metrics = await generateActivityReport({ + userId: user._id, + timeframe, + metrics: ["all"], + includeProjections: true, + }); + + return { + success: true, + message: "Engagement metrics retrieved successfully", + data: metrics, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Activity Analysis Functions + getActivityReport: async (request, next) => { + try { + const { email, tenant, startDate, endDate } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No activity data found for this user", + }) + ); + } + + const report = await generateActivityReport({ + userId: user._id, + timeframe: { startDate, endDate }, + metrics: ["all"], + includeProjections: true, + }); + + return { + success: true, + message: "Activity report generated successfully", + data: report, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Cohort Analysis Functions + getCohortAnalysis: async (request, next) => { + try { + const { tenant, period = "monthly", metrics = ["all"] } = request.query; + + const cohortAnalysis = await performCohortAnalysis(tenant, { + cohortPeriod: period, + metrics, + }); + + return { + success: true, + message: "Cohort analysis completed successfully", + data: cohortAnalysis, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Predictive Analytics Functions + getPredictiveAnalytics: async (request, next) => { + try { + const { email, tenant } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No data found for predictions", + }) + ); + } + + const predictions = await predictEngagementTrends({ + dailyStats: user.dailyStats, + monthlyStats: user.monthlyStats, + currentMetrics: user.overallStats, + }); + + return { + success: true, + message: "Predictive analytics generated successfully", + data: predictions, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Service Adoption Functions + getServiceAdoption: async (request, next) => { + try { + const { email, tenant } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No service adoption data found", + }) + ); + } + + const adoptionAnalysis = analyzeServiceAdoption({ + services: user.monthlyStats[0].topServices, + endpoints: user.monthlyStats[0].uniqueEndpoints, + historicalUsage: user.dailyStats, + }); + + return { + success: true, + message: "Service adoption analysis completed successfully", + data: adoptionAnalysis, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Benchmark Functions + getBenchmarks: async (request, next) => { + try { + const { tenant, metrics = ["all"] } = request.query; + + const benchmarks = await calculateCohortBenchmarks({ + tenant, + metrics, + }); + + return { + success: true, + message: "Benchmarks retrieved successfully", + data: benchmarks, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Top Users Functions + getTopUsers: async (request, next) => { + try { + const { tenant, limit = 10, metric = "engagementScore" } = request.query; + + const topUsers = await ActivityModel(tenant) + .find({}) + .sort({ [`overallStats.${metric}`]: -1 }) + .limit(parseInt(limit)); + + return { + success: true, + message: "Top users retrieved successfully", + data: topUsers, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Aggregated Analytics Functions + getAggregatedAnalytics: async (request, next) => { + try { + const { + tenant, + metrics = ["all"], + timeframe = "last30days", + } = request.query; + + const aggregatedData = await ActivityModel(tenant).aggregate([ + { + $group: { + _id: null, + totalUsers: { $sum: 1 }, + averageEngagement: { $avg: "$overallStats.engagementScore" }, + totalActions: { $sum: "$overallStats.totalActions" }, + }, + }, + ]); + + return { + success: true, + message: "Aggregated analytics retrieved successfully", + data: aggregatedData[0], + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Retention Analysis Functions + getRetentionAnalysis: async (request, next) => { + try { + const { tenant, period = "monthly" } = request.query; + + const cohorts = await analyzeCohorts(); + const cohortMetrics = await analyzeCohortMetrics(); + const retentionAnalysis = await analyzeCohortTrends(); + + return { + success: true, + message: "Retention analysis completed successfully", + data: { + cohorts, + metrics: cohortMetrics, + trends: retentionAnalysis, + }, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Health Score Functions + getEngagementHealth: async (request, next) => { + try { + const { email, tenant } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No health score data found", + }) + ); + } + + const healthScore = await calculateUserEngagementScore({ + totalActions: user.monthlyStats[0].totalActions, + services: user.monthlyStats[0].topServices, + endpoints: user.monthlyStats[0].uniqueEndpoints, + firstActivity: user.monthlyStats[0].firstActivity, + lastActivity: user.monthlyStats[0].lastActivity, + }); + + return { + success: true, + message: "Health score retrieved successfully", + data: healthScore, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + // Behavior Pattern Functions + getBehaviorPatterns: async (request, next) => { + try { + const { email, tenant } = request.query; + const user = await ActivityModel(tenant).findOne({ email }); + + if (!user) { + return next( + new HttpError("User not found", httpStatus.NOT_FOUND, { + message: "No behavior pattern data found", + }) + ); + } + + const patterns = await analyzeSeasonalPatterns(user.monthlyStats); + const trailingMetrics = calculateTrailingIndicators(user.dailyStats); + + return { + success: true, + message: "Behavior patterns analyzed successfully", + data: { + seasonalPatterns: patterns, + trailingIndicators: trailingMetrics, + }, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + enhancedGetUserStats: async (request, next) => { + try { + return { + success: false, + message: "enhancedGetUserStats temporarily disabled", + status: httpStatus.NOT_IMPLEMENTED, + errors: { message: "enhancedGetUserStats temporarily disabled" }, + }; + const { + tenant, + limit = 1000, + skip = 0, + startTime, + endTime, + } = request.query; + + const filter = { + ...generateFilter.logs(request, next), + timestamp: { + $gte: new Date(startTime), + $lte: new Date(endTime), + }, + "meta.service": { $nin: ["unknown", "none", "", null] }, + }; + + logger.info(`Applied filter: ${stringify(filter)}`); + + const pipeline = [ + { $match: filter }, + { + $facet: { + paginatedResults: [ + { + $group: { + _id: "$meta.email", + username: { $first: "$meta.username" }, + totalCount: { $sum: 1 }, + uniqueServices: { $addToSet: "$meta.service" }, + uniqueEndpoints: { $addToSet: "$meta.endpoint" }, + firstActivity: { $min: "$timestamp" }, + lastActivity: { $max: "$timestamp" }, + // Efficient service usage tracking + serviceUsage: { + $push: { + service: "$meta.service", + endpoint: "$meta.endpoint", + }, + }, + }, + }, + { $sort: { totalCount: -1 } }, + { $skip: skip }, + { $limit: limit }, + ], + totalCount: [ + { $group: { _id: "$meta.email" } }, + { $count: "total" }, + ], + }, + }, + ]; + + const [results] = await LogModel(tenant) + .aggregate(pipeline) + .allowDiskUse(true); + + const enrichedStats = results.paginatedResults.map((stat) => { + // Calculate service statistics + const serviceStats = stat.serviceUsage.reduce((acc, curr) => { + const { service } = curr; + acc[service] = (acc[service] || 0) + 1; + return acc; + }, {}); + + const topServices = Object.entries(serviceStats) + .map(([service, count]) => ({ + service: formatServiceName(service), + count, + })) + .sort((a, b) => b.count - a.count); + + const activityDuration = calculateActivityDuration( + stat.firstActivity, + stat.lastActivity, + new Date(startTime), + new Date(endTime) + ); + + return { + email: stat._id, + username: stat.username, + count: stat.totalCount, + uniqueServices: stat.uniqueServices, + uniqueEndpoints: stat.uniqueEndpoints, + firstActivity: stat.firstActivity, + lastActivity: stat.lastActivity, + topServices, + activityDuration, + engagementTier: calculateEngagementTier({ + count: stat.totalCount, + uniqueServices: stat.uniqueServices, + uniqueEndpoints: stat.uniqueEndpoints, + activityDuration, + }), + }; + }); + + return { + success: true, + message: "Successfully retrieved user statistics", + data: enrichedStats, + total: results.totalCount[0]?.total || 0, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { + message: error.message, + } + ) + ); + return; + } + }, + validateEnvironmentData: async ({ + tenant, + year = new Date().getFullYear(), + } = {}) => { + try { + return { + success: false, + message: " validateEnvironmentData temporarily disabled", + status: httpStatus.NOT_IMPLEMENTED, + errors: { message: " validateEnvironmentData temporarily disabled" }, + }; + logger.info( + `Running data validation for environment: ${constants.ENVIRONMENT}` + ); + + const startDate = new Date(`${year}-01-01`); + const endDate = new Date(`${year}-12-31`); + + const pipeline = [ + { + $match: { + timestamp: { + $gte: startDate, + $lte: endDate, + }, + }, + }, + { + $facet: { + monthlyDistribution: [ + { + $group: { + _id: { + year: { $year: "$timestamp" }, + month: { $month: "$timestamp" }, + }, + count: { $sum: 1 }, + uniqueUsers: { $addToSet: "$meta.email" }, + uniqueEndpoints: { $addToSet: "$meta.endpoint" }, + uniqueServices: { $addToSet: "$meta.service" }, + }, + }, + { $sort: { "_id.year": 1, "_id.month": 1 } }, + ], + userSample: [ + { + $group: { + _id: "$meta.email", + firstActivity: { $min: "$timestamp" }, + lastActivity: { $max: "$timestamp" }, + totalActions: { $sum: 1 }, + }, + }, + { $match: { totalActions: { $gt: 100 } } }, + { $limit: 10 }, + ], + systemMetrics: [ + { + $group: { + _id: null, + totalLogs: { $sum: 1 }, + avgLogsPerDay: { + $avg: { + $sum: 1, + }, + }, + uniqueServices: { $addToSet: "$meta.service" }, + uniqueEndpoints: { $addToSet: "$meta.endpoint" }, + }, + }, + ], + }, + }, + ]; + + const [results] = await LogModel(tenant) + .aggregate(pipeline) + .allowDiskUse(true); + + // Enrich the monthly distribution with percentage changes + const enrichedDistribution = results.monthlyDistribution.map( + (month, index, arr) => { + const prevMonth = index > 0 ? arr[index - 1] : null; + return { + year: month._id.year, + month: month._id.month, + totalLogs: month.count, + uniqueUsers: month.uniqueUsers.length, + uniqueEndpoints: month.uniqueEndpoints.length, + uniqueServices: month.uniqueServices.length, + percentageChange: prevMonth + ? ( + ((month.count - prevMonth.count) / prevMonth.count) * + 100 + ).toFixed(2) + : null, + }; + } + ); + + return { + success: true, + message: "Environment data validation completed", + data: { + monthlyDistribution: enrichedDistribution, + userSample: results.userSample, + systemMetrics: results.systemMetrics[0], + timeframe: { + startDate, + endDate, + totalDays: Math.floor( + (endDate - startDate) / (1000 * 60 * 60 * 24) + ), + }, + }, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`Error in validateEnvironmentData: ${stringify(error)}`); + throw new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { + message: error.message, + } + ); + } + }, + fetchUserStats: async ({ + emails = [], + year = new Date().getFullYear(), + tenant = "airqo", + chunkSize = 5, + timeWindowDays = 90, // Process data in 90-day windows + } = {}) => { + try { + return { + success: false, + message: "fetchUserStats temporarily disabled", + status: httpStatus.NOT_IMPLEMENTED, + errors: { message: "fetchUserStats temporarily disabled" }, + }; + const startDate = new Date(`${year}-01-01`); + const endDate = new Date(`${year}-12-31`); + const enrichedStats = []; + + // Process users in chunks to avoid memory overload + for (let i = 0; i < emails.length; i += chunkSize) { + const emailChunk = emails.slice(i, i + chunkSize); + + // Break the year into smaller time windows + const timeWindows = []; + let currentStart = new Date(startDate); + + while (currentStart < endDate) { + const windowEnd = new Date(currentStart); + windowEnd.setDate(windowEnd.getDate() + timeWindowDays); + const actualEnd = windowEnd > endDate ? endDate : windowEnd; + + timeWindows.push({ + start: new Date(currentStart), + end: new Date(actualEnd), + }); + + currentStart = new Date(actualEnd); + } + + // Process each time window for the current user chunk + const userStatsPromises = timeWindows.map(async (window) => { + const pipeline = [ + { + $match: { + timestamp: { + $gte: window.start, + $lte: window.end, + }, + "meta.email": { $in: emailChunk }, + "meta.service": { $nin: ["unknown", "none", "", null] }, + }, + }, + { + $group: { + _id: "$meta.email", + totalCount: { $sum: 1 }, + username: { $first: "$meta.username" }, + firstActivity: { $min: "$timestamp" }, + lastActivity: { $max: "$timestamp" }, + uniqueServices: { $addToSet: "$meta.service" }, + uniqueEndpoints: { $addToSet: "$meta.endpoint" }, + serviceUsage: { + $push: { + service: "$meta.service", + }, + }, + }, + }, + ]; + + return LogModel(tenant).aggregate(pipeline).allowDiskUse(true).exec(); + }); + + // Wait for all time windows to complete for current user chunk + const windowResults = await Promise.all(userStatsPromises); + + // Merge results for each user across time windows + const mergedUserStats = emailChunk.reduce((acc, email) => { + acc[email] = { + email, + totalCount: 0, + username: "", + firstActivity: null, + lastActivity: null, + uniqueServices: new Set(), + uniqueEndpoints: new Set(), + serviceUsage: new Map(), + }; + return acc; + }, {}); + + // Combine stats from all time windows + windowResults.flat().forEach((stat) => { + if (!stat || !stat._id) return; + + const userStats = mergedUserStats[stat._id]; + if (!userStats) return; + + userStats.totalCount += stat.totalCount; + userStats.username = stat.username; + + if ( + !userStats.firstActivity || + stat.firstActivity < userStats.firstActivity + ) { + userStats.firstActivity = stat.firstActivity; + } + if ( + !userStats.lastActivity || + stat.lastActivity > userStats.lastActivity + ) { + userStats.lastActivity = stat.lastActivity; + } + + stat.uniqueServices.forEach((s) => userStats.uniqueServices.add(s)); + stat.uniqueEndpoints.forEach((e) => userStats.uniqueEndpoints.add(e)); + + // Update service usage counts + stat.serviceUsage.forEach((usage) => { + const current = userStats.serviceUsage.get(usage.service) || 0; + userStats.serviceUsage.set(usage.service, current + 1); + }); + }); + + // Convert merged stats to final format + Object.values(mergedUserStats).forEach((stat) => { + if (stat.totalCount === 0) return; // Skip users with no activity + + const topServices = Array.from(stat.serviceUsage.entries()) + .map(([service, count]) => ({ + service: formatServiceName(service), + count, + })) + .sort((a, b) => b.count - a.count) + .slice(0, 5); + + const activityDays = Math.floor( + (stat.lastActivity - stat.firstActivity) / (1000 * 60 * 60 * 24) + ); + + const engagementScore = calculateEngagementScore({ + totalActions: stat.totalCount, + uniqueServices: stat.uniqueServices.size, + uniqueEndpoints: stat.uniqueEndpoints.size, + activityDays: activityDays || 1, + }); + + enrichedStats.push({ + email: stat.email, + username: stat.username, + totalActions: stat.totalCount, + firstActivity: stat.firstActivity, + lastActivity: stat.lastActivity, + uniqueServices: Array.from(stat.uniqueServices), + uniqueEndpoints: Array.from(stat.uniqueEndpoints), + topServices, + activityDuration: { + totalDays: activityDays, + totalMonths: Math.floor(activityDays / 30), + description: `Active for ${Math.floor(activityDays / 30)} months`, + }, + engagementTier: calculateEngagementTier(engagementScore), + }); + }); + + // Add a small delay between chunks to prevent overwhelming the server + if (i + chunkSize < emails.length) { + await new Promise((resolve) => setTimeout(resolve, 1000)); + } + } + + return enrichedStats; + } catch (error) { + logger.error(`Error in fetchUserStats: ${stringify(error)}`); + throw new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ); + } + }, + sendEmailsInBatches: async (userStats, batchSize = 100) => { + try { + return { + success: false, + message: "sendEmailsInBatches temporarily disabled", + status: httpStatus.NOT_IMPLEMENTED, + errors: { message: "sendEmailsInBatches temporarily disabled" }, + }; + let emailsSent = 0; + let lowEngagementCount = 0; + + for (let i = 0; i < userStats.length; i += batchSize) { + const batch = userStats.slice(i, i + batchSize); + const emailPromises = batch + .filter((userStat) => { + if (userStat.engagementTier === "Low Engagement") { + lowEngagementCount++; + return false; + } + return true; + }) + .map((userStat) => { + const { email, username } = userStat; + + return mailer + .yearEndEmail({ + email, + firstName: username.split(" ")[0], + lastName: username.split(" ")[1] || "", + userStat, + }) + .then((response) => { + if (response && response.success === false) { + logger.error( + `🐛🐛 Error sending year-end email to ${email}: ${stringify( + response + )}` + ); + } else { + emailsSent++; + } + return response; + }); + }); + + await Promise.all(emailPromises); + } + + return { + success: true, + message: + lowEngagementCount > 0 + ? `Sent year-end emails to ${emailsSent} users. Skipped ${lowEngagementCount} users with low engagement.` + : `Sent year-end emails to ${emailsSent} users`, + }; + } catch (error) { + logger.error(`🐛🐛 Error in sendEmailsInBatches: ${stringify(error)}`); + throw new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ); + } + }, + sendYearEndEmails: async (request) => { + try { + return { + success: false, + message: "sendYearEndEmails temporarily disabled", + status: httpStatus.NOT_IMPLEMENTED, + errors: { message: "sendYearEndEmails temporarily disabled" }, + }; + const { body, query } = request; + const { emails } = body; + const { tenant } = query; + + const userStats = await analytics.fetchUserStats({ emails, tenant }); + logObject("userStats", userStats); + + if (userStats.length > 0) { + const result = await analytics.sendEmailsInBatches(userStats); + return result; + } else { + logger.info("No user stats found for the provided emails."); + return { + success: false, + message: "No user stats found for the provided emails", + }; + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error -- ${stringify(error)}`); + throw new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ); + } + }, + listLogs: async (request, next) => { + try { + const { tenant, limit = 1000, skip = 0 } = request.query; + const filter = generateFilter.logs(request, next); + const responseFromListLogs = await LogModel(tenant).list( + { + filter, + limit, + skip, + }, + next + ); + if (responseFromListLogs.success === true) { + return { + success: true, + message: responseFromListLogs.message, + data: responseFromListLogs.data, + status: responseFromListLogs.status + ? responseFromListLogs.status + : httpStatus.OK, + }; + } else if (responseFromListLogs.success === false) { + const errorObject = responseFromListLogs.errors + ? responseFromListLogs.errors + : { message: "Internal Server Error" }; + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { + message: responseFromListLogs.message, + ...errorObject, + } + ) + ); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + listActivities: async (request, next) => { + try { + const { tenant, limit = 1000, skip = 0 } = request.query; + const filter = generateFilter.activities(request, next); + const responseFromListActivities = await ActivityModel(tenant).list( + { + filter, + limit, + skip, + }, + next + ); + if (responseFromListActivities.success === true) { + return { + success: true, + message: responseFromListActivities.message, + data: responseFromListActivities.data, + status: responseFromListActivities.status + ? responseFromListActivities.status + : httpStatus.OK, + }; + } else if (responseFromListActivities.success === false) { + const errorObject = responseFromListActivities.errors + ? responseFromListActivities.errors + : { message: "Internal Server Error" }; + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { + message: responseFromListActivities.message, + ...errorObject, + } + ) + ); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + getUserStats: async (request, next) => { + try { + const { tenant, limit = 1000, skip = 0 } = request.query; + const filter = generateFilter.logs(request, next); + + const pipeline = [ + { $match: filter }, + { + $group: { + _id: { email: "$meta.email", endpoint: "$meta.endpoint" }, + service: { $first: "$meta.service" }, + username: { $first: "$meta.username" }, + count: { $sum: 1 }, + }, + }, + { + $project: { + _id: 0, + email: "$_id.email", + endpoint: "$_id.endpoint", + count: 1, + service: "$service", + username: "$username", + }, + }, + ]; + + const getUserStatsResponse = await LogModel(tenant).aggregate(pipeline); + return { + success: true, + message: "Successfully retrieved the user statistics", + data: getUserStatsResponse, + status: httpStatus.OK, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, + listStatistics: async (tenant, next) => { + try { + const responseFromListStatistics = await UserModel(tenant).listStatistics( + tenant + ); + return responseFromListStatistics; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, +}; + +module.exports = analytics; diff --git a/src/auth-service/utils/create-candidate.js b/src/auth-service/utils/create-candidate.js index 81ec922cc7..2f3a00c0ea 100644 --- a/src/auth-service/utils/create-candidate.js +++ b/src/auth-service/utils/create-candidate.js @@ -19,6 +19,15 @@ const { HttpError } = require("@utils/errors"); const createCandidate = { create: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: + "Please contact support@airqo.net, Candidates are deprecated", + }, + }; const { firstName, lastName, email, tenant, network_id } = { ...request.body, ...request.query, @@ -127,6 +136,15 @@ const createCandidate = { }, list: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: + "Please contact support@airqo.net, Candidates are deprecated", + }, + }; const { tenant, limit, skip } = { ...request.body, ...request.query, @@ -157,6 +175,15 @@ const createCandidate = { }, update: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: + "Please contact support@airqo.net, Candidates are deprecated", + }, + }; const { query, body } = request; const filter = generateFilter.candidates(request, next); const update = body; @@ -184,6 +211,15 @@ const createCandidate = { }, confirm: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: + "Please contact support@airqo.net, Candidates are deprecated", + }, + }; const { tenant, firstName, lastName, email, network_id } = { ...request.body, ...request.query, @@ -330,6 +366,15 @@ const createCandidate = { }, delete: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: + "Please contact support@airqo.net, Candidates are deprecated", + }, + }; const { tenant } = { ...request.query }; const filter = generateFilter.candidates(request, next); const responseFromRemoveCandidate = await CandidateModel( diff --git a/src/auth-service/utils/create-group.js b/src/auth-service/utils/create-group.js index 3c501cb431..08dbe1831e 100644 --- a/src/auth-service/utils/create-group.js +++ b/src/auth-service/utils/create-group.js @@ -1095,6 +1095,86 @@ const createGroup = { ); } }, + setManager: async (request, next) => { + try { + const { grp_id, user_id } = request.params; + const { tenant } = request.query; + const user = await UserModel(tenant).findById(user_id).lean(); + const group = await GroupModel(tenant).findById(grp_id).lean(); + + if (isEmpty(user)) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "User not found", + }) + ); + } + + if (isEmpty(group)) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: "Group not found", + }) + ); + } + + if ( + group.grp_manager && + group.grp_manager.toString() === user_id.toString() + ) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: `User ${user_id.toString()} is already the group manager`, + }) + ); + } + + logObject("the user object", user); + // Updated check to use group_roles array + const userGroupIds = user.group_roles.map((groupRole) => + groupRole.group.toString() + ); + + if (!userGroupIds.includes(grp_id.toString())) { + return next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + message: `Group ${grp_id.toString()} is not part of User's groups, not authorized to manage this group`, + }) + ); + } + + const updatedGroup = await GroupModel(tenant).findByIdAndUpdate( + grp_id, + { grp_manager: user_id }, + { new: true } + ); + + if (!isEmpty(updatedGroup)) { + return { + success: true, + message: "User assigned to Group successfully", + status: httpStatus.OK, + data: updatedGroup, + }; + } else { + return next( + new HttpError("Bad Request", httpStatus.BAD_REQUEST, { + message: "No group record was updated", + }) + ); + } + } catch (error) { + logObject("the error", error); + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + return next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, }; module.exports = createGroup; diff --git a/src/auth-service/utils/create-network.js b/src/auth-service/utils/create-network.js index eb2174bcc4..1e817e55f3 100644 --- a/src/auth-service/utils/create-network.js +++ b/src/auth-service/utils/create-network.js @@ -731,7 +731,7 @@ const createNetwork = { const network = await NetworkModel(tenant).findById(net_id).lean(); if (isEmpty(user)) { - next( + return next( new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { message: "User not found", }) @@ -739,7 +739,7 @@ const createNetwork = { } if (isEmpty(network)) { - next( + return next( new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { message: "Network not found", }) @@ -750,17 +750,20 @@ const createNetwork = { network.net_manager && network.net_manager.toString() === user_id.toString() ) { - next( + return next( new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { message: `User ${user_id.toString()} is already the network manager`, }) ); } - if ( - !user.networks.map((id) => id.toString()).includes(net_id.toString()) - ) { - next( + // Updated check to use network_roles array + const userNetworkIds = user.network_roles.map((networkRole) => + networkRole.network.toString() + ); + + if (!userNetworkIds.includes(net_id.toString())) { + return next( new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { message: `Network ${net_id.toString()} is not part of User's networks, not authorized to manage this network`, }) @@ -781,7 +784,7 @@ const createNetwork = { data: updatedNetwork, }; } else { - next( + return next( new HttpError("Bad Request", httpStatus.BAD_REQUEST, { message: "No network record was updated", }) @@ -789,7 +792,7 @@ const createNetwork = { } } catch (error) { logger.error(`🐛🐛 Internal Server Error ${error.message}`); - next( + return next( new HttpError( "Internal Server Error", httpStatus.INTERNAL_SERVER_ERROR, diff --git a/src/auth-service/utils/email.msgs.js b/src/auth-service/utils/email.msgs.js index 69a46ef95c..ae3c9939e6 100644 --- a/src/auth-service/utils/email.msgs.js +++ b/src/auth-service/utils/email.msgs.js @@ -173,6 +173,85 @@ module.exports = { `; return constants.EMAIL_BODY({ email, content, name }); }, + yearEndSummary: ({ + username = "", + email = "", + engagementTier = "", + activityDuration = {}, + topServiceDescription = "", + topServices = [], + mostUsedEndpoints = [], + } = {}) => { + const content = ` + + +

Congratulations on an amazing year with AirQo Analytics! 🎉

+ + ${ + engagementTier || + activityDuration.description || + topServiceDescription + ? ` +
+

+ 🌟 Your 2024 Highlights 🌟 +

+ + ${ + engagementTier + ? `

Engagement Level: ${engagementTier}

` + : "" + } + ${ + activityDuration.description + ? `

Activity Duration: ${activityDuration.description}

` + : "" + } + + ${ + topServiceDescription + ? `

Top Service: ${topServiceDescription}

` + : "" + } +
` + : "" + } + + ${ + topServices && topServices.length > 0 + ? ` +
+

+ Most Used Services: +

+ ${topServices + .slice(0, 3) + .map( + (service, index) => ` +

+ ${index + 1}. ${service.service} (Used ${ + service.count + } times) +

+ ` + ) + .join("")} +
` + : "" + } + +

Thank you for being an incredible part of our community!

+ +

Best wishes,
The AirQo Analytics Team

+ +

+ Please visit our website to learn more about us. AirQo +

+ + `; + + return constants.EMAIL_BODY({ email, content, name: username }); + }, afterClientActivation: ({ name = "", email, client_id = "" } = {}) => { const content = ` diff --git a/src/auth-service/utils/generate-filter.js b/src/auth-service/utils/generate-filter.js index 549f1e7df4..1e2f668898 100644 --- a/src/auth-service/utils/generate-filter.js +++ b/src/auth-service/utils/generate-filter.js @@ -13,6 +13,7 @@ const { HttpError } = require("@utils/errors"); const { addMonthsToProvideDateTime, monthsInfront, + addMonthsToProvidedDate, isTimeEmpty, getDifferenceInMonths, addDays, @@ -788,6 +789,7 @@ const filter = { const { service, startTime, endTime, email } = req.query; const today = monthsInfront(0, next); const oneWeekBack = addDays(-7, next); + logObject("the req.query in the logs filter", req.query); let filter = { timestamp: { @@ -810,11 +812,13 @@ const filter = { next ); } else { - delete filter["timestamp"]; + filter["timestamp"]["$lte"] = addMonthsToProvidedDate( + startTime, + 1, + next + ); } - } - - if (endTime && isEmpty(startTime)) { + } else if (endTime && isEmpty(startTime)) { logText("startTime absent and endTime is present"); if (isTimeEmpty(endTime) === false) { filter["timestamp"]["$lte"] = addMonthsToProvideDateTime( @@ -823,25 +827,16 @@ const filter = { next ); } else { - delete filter["timestamp"]; + filter["timestamp"]["$lte"] = addMonthsToProvidedDate( + endTime, + -1, + next + ); } - } - - if (endTime && startTime) { + } else if (endTime && startTime) { logText("startTime present and endTime is also present"); - let months = getDifferenceInMonths(startTime, endTime); - logElement("the number of months", months); - if (months > 1) { - if (isTimeEmpty(endTime) === false) { - filter["timestamp"]["$lte"] = addMonthsToProvideDateTime( - endTime, - -1, - next - ); - } else { - delete filter["timestamp"]; - } - } + filter["timestamp"]["$lte"] = new Date(endTime); + filter["timestamp"]["$gte"] = new Date(startTime); } if (email) { @@ -861,6 +856,102 @@ const filter = { ); } }, + activities: (req, next) => { + try { + const { service, startTime, endTime, email, tenant } = req.query; + const today = monthsInfront(0, next); + const oneWeekBack = addDays(-7, next); + + logObject("the req.query in the activity filter", req.query); + + // Base filter + let filter = {}; + + // Handle required tenant field + if (!tenant) { + throw new Error("Tenant is required"); + } + filter.tenant = tenant; + + // Handle email filtering + if (email) { + logText("email present"); + filter.email = email; + } + + // Date range handling for dailyStats + let dateFilter = {}; + if (startTime && !endTime) { + logText("startTime present and endTime is missing"); + if (!isTimeEmpty(startTime)) { + dateFilter.$gte = addMonthsToProvideDateTime(startTime, 1, next); + } else { + dateFilter.$gte = addMonthsToProvidedDate(startTime, 1, next); + } + } else if (endTime && !startTime) { + logText("startTime absent and endTime is present"); + if (!isTimeEmpty(endTime)) { + dateFilter.$lte = addMonthsToProvideDateTime(endTime, -1, next); + } else { + dateFilter.$lte = addMonthsToProvidedDate(endTime, -1, next); + } + } else if (endTime && startTime) { + logText("startTime present and endTime is also present"); + dateFilter.$lte = new Date(endTime); + dateFilter.$gte = new Date(startTime); + } else { + // Default to one week range + dateFilter.$gte = oneWeekBack; + dateFilter.$lte = today; + } + + // Apply date filter to dailyStats + if (Object.keys(dateFilter).length > 0) { + filter["dailyStats"] = { + $elemMatch: { + date: dateFilter, + }, + }; + } + + // Service filtering + if (service) { + logText("service present"); + // Add service filter to the dailyStats elemMatch + if (filter.dailyStats && filter.dailyStats.$elemMatch) { + filter.dailyStats.$elemMatch["services"] = { + $elemMatch: { name: service }, + }; + } else { + filter.dailyStats = { + $elemMatch: { + services: { + $elemMatch: { name: service }, + }, + }, + }; + } + + // Also check monthly stats for the service + filter["monthlyStats"] = { + $elemMatch: { + uniqueServices: service, + }, + }; + } + + return filter; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, favorites: (req, next) => { try { const { query, params } = req; diff --git a/src/auth-service/utils/mailer.js b/src/auth-service/utils/mailer.js index 8c9041efc2..ef5e2f5691 100644 --- a/src/auth-service/utils/mailer.js +++ b/src/auth-service/utils/mailer.js @@ -1,5 +1,5 @@ const transporter = require("@config/mailer"); -const { logObject } = require("@utils/log"); +const { logObject, logText } = require("@utils/log"); const isEmpty = require("is-empty"); const SubscriptionModel = require("@models/Subscription"); const constants = require("@config/constants"); @@ -284,6 +284,61 @@ const mailer = { ); } }, + yearEndEmail: async ({ + email = "", + firstName = "", + lastName = "", + tenant = "airqo", + userStat = {}, + } = {}) => { + try { + const checkResult = await SubscriptionModel( + tenant + ).checkNotificationStatus({ email, type: "email" }); + + if (!checkResult.success) { + return checkResult; + } + + const mailOptions = { + to: email, + from: { + name: constants.EMAIL_NAME, + address: constants.EMAIL, + }, + subject: "Your AirQo Analytics 2024 Year in Review 🌍", + html: msgs.yearEndSummary(userStat), + attachments: attachments, + }; + + let response = transporter.sendMail(mailOptions); + let data = await response; + + if (isEmpty(data.rejected) && !isEmpty(data.accepted)) { + return { + success: true, + message: "Year-end email successfully sent", + data, + status: httpStatus.OK, + }; + } else { + return { + success: false, + message: "Internal Server Error", + status: httpStatus.INTERNAL_SERVER_ERROR, + errors: { message: "Email not sent", emailResults: data }, + }; + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + return { + success: false, + message: "Internal Server Error", + status: httpStatus.INTERNAL_SERVER_ERROR, + errors: { message: error.message }, + }; + } + }, requestToJoinGroupByEmail: async ( { email, diff --git a/src/auth-service/validators/analytics.validators.js b/src/auth-service/validators/analytics.validators.js new file mode 100644 index 0000000000..9123e36965 --- /dev/null +++ b/src/auth-service/validators/analytics.validators.js @@ -0,0 +1,169 @@ +const { query, body, param, oneOf } = require("express-validator"); +const constants = require("@config/constants"); + +const validateTimeframe = query("timeframe") + .optional() + .isIn(["last7days", "last30days", "last90days", "custom"]) + .withMessage("Invalid timeframe"); + +const validateDateRange = [ + query("startDate") + .if(query("timeframe").equals("custom")) + .isISO8601() + .withMessage("Start date required for custom timeframe"), + query("endDate") + .if(query("timeframe").equals("custom")) + .isISO8601() + .withMessage("End date required for custom timeframe"), +]; + +const validateUserEmail = param("email") + .isEmail() + .withMessage("Valid email is required"); + +const validatePeriod = query("period") + .isIn(["daily", "weekly", "monthly", "quarterly", "yearly"]) + .withMessage("Invalid period"); + +const validateCohortPeriod = query("period") + .isIn(["daily", "weekly", "monthly"]) + .withMessage("Invalid cohort period"); + +const validateMetricType = query("metric") + .isIn(["engagement", "retention", "activity", "growth"]) + .withMessage("Invalid metric type"); + +const validateYearMonth = [ + query("year").isInt().withMessage("Invalid year"), + query("month").isInt({ min: 0, max: 11 }).withMessage("Invalid month (0-11)"), +]; + +const validateGroupBy = query("groupBy") + .isIn(["service", "endpoint", "user"]) + .withMessage("Invalid grouping parameter"); + +const validateEmailsArray = body("emails") + .exists() + .withMessage("the emails array field must be provided in the request body") + .bail() + .isArray() + .withMessage("emails must be provided as an array") + .bail() + .notEmpty() + .withMessage("the provided emails array cannot be empty"); + +// Grouped validation rules for different endpoints +const userEngagementValidators = { + getEngagement: [validateUserEmail, validateTimeframe, ...validateDateRange], + getMetrics: [validateUserEmail, ...validateYearMonth], +}; + +const activityValidators = { + getReport: [validateUserEmail, validateTimeframe, ...validateDateRange], +}; + +const cohortValidators = { + getAnalysis: [ + validateCohortPeriod, + query("metrics") + .optional() + .isArray() + .withMessage("Metrics must be an array"), + query("minCohortSize") + .optional() + .isInt({ min: 1 }) + .withMessage("Invalid minimum cohort size"), + ], +}; + +const predictiveValidators = { + getPredictions: [ + validateUserEmail, + query("includeChurnRisk") + .optional() + .isBoolean() + .withMessage("Invalid churn risk inclusion parameter"), + ], +}; + +const serviceAdoptionValidators = { + getAdoption: [ + validateUserEmail, + query("includeRecommendations") + .optional() + .isBoolean() + .withMessage("Invalid recommendations parameter"), + ], +}; + +const benchmarkValidators = { + getBenchmarks: [validateMetricType, validatePeriod], +}; + +const topUsersValidators = { + getTopUsers: [ + ...validateYearMonth, + query("limit") + .optional() + .isInt({ min: 1, max: 100 }) + .withMessage("Invalid limit"), + validateMetricType.optional(), + ], +}; + +const aggregateValidators = { + getAggregated: [ + query("startDate").isISO8601().withMessage("Invalid start date"), + query("endDate").isISO8601().withMessage("Invalid end date"), + validateGroupBy, + ], +}; + +const retentionValidators = { + getRetention: [ + validateCohortPeriod.optional(), + query("period") + .isInt({ min: 1, max: 12 }) + .withMessage("Invalid period range"), + ], +}; + +const healthScoreValidators = { + getHealthScore: [ + validateUserEmail, + query("includeFactors") + .optional() + .isBoolean() + .withMessage("Invalid factors inclusion parameter"), + ], +}; + +const behaviorValidators = { + getBehavior: [ + validateUserEmail, + query("includeSegmentation") + .optional() + .isBoolean() + .withMessage("Invalid segmentation parameter"), + ], +}; + +const emailValidators = { + sendEmails: [validateEmailsArray], + retrieveStats: [validateEmailsArray], +}; + +module.exports = { + userEngagementValidators, + activityValidators, + cohortValidators, + predictiveValidators, + serviceAdoptionValidators, + benchmarkValidators, + topUsersValidators, + aggregateValidators, + retentionValidators, + healthScoreValidators, + behaviorValidators, + emailValidators, +}; diff --git a/src/device-registry/bin/jobs/kafka-consumer.js b/src/device-registry/bin/jobs/kafka-consumer.js index b131908cf0..66022f64f4 100644 --- a/src/device-registry/bin/jobs/kafka-consumer.js +++ b/src/device-registry/bin/jobs/kafka-consumer.js @@ -5,13 +5,15 @@ const logger = log4js.getLogger( `${constants.ENVIRONMENT} -- bin/jobs/kafka-consumer` ); const { logObject } = require("@utils/log"); -const createEvent = require("@utils/create-event"); +const createEventUtil = require("@utils/create-event"); +const createForecastUtil = require("@utils/create-forecast"); const Joi = require("joi"); const { jsonrepair } = require("jsonrepair"); const cleanDeep = require("clean-deep"); const isEmpty = require("is-empty"); const stringify = require("@utils/stringify"); +// Existing measurement schema const eventSchema = Joi.object({ s2_pm2_5: Joi.number().optional(), s2_pm10: Joi.number().optional(), @@ -62,6 +64,39 @@ const eventSchema = Joi.object({ intakehumidity: Joi.number().optional(), }).unknown(true); +// Forecast validation schemas +const forecastMeasurementsSchema = Joi.object({ + timestamp: Joi.date() + .iso() + .required(), + horizon: Joi.number().required(), + measurements: Joi.object({ + pm2_5: Joi.number().required(), + pm10: Joi.number().required(), + confidence: Joi.object({ + pm2_5_lower: Joi.number().required(), + pm2_5_upper: Joi.number().required(), + pm10_lower: Joi.number().required(), + pm10_upper: Joi.number().required(), + }).required(), + }).required(), +}); + +const forecastSchema = Joi.object({ + type: Joi.string() + .valid("forecast") + .required(), + created_at: Joi.date() + .iso() + .required(), + model_version: Joi.string().required(), + device_id: Joi.string().required(), + site_id: Joi.string().required(), + forecasts: Joi.array() + .items(forecastMeasurementsSchema) + .required(), +}); + const eventsSchema = Joi.array().items(eventSchema); const consumeHourlyMeasurements = async (messageData) => { @@ -103,13 +138,16 @@ const consumeHourlyMeasurements = async (messageData) => { timestamp, }; }); + logger.error(`Validation errors: ${stringify(errorDetails)}`); } const request = { body: cleanedMeasurements, }; - const responseFromInsertMeasurements = await createEvent.create(request); + const responseFromInsertMeasurements = await createEventUtil.create( + request + ); if (responseFromInsertMeasurements.success === false) { console.log("KAFKA: failed to store the measurements"); @@ -122,6 +160,106 @@ const consumeHourlyMeasurements = async (messageData) => { } }; +// Transform the incoming forecast message to match the expected format +const transformForecastData = (forecastData) => { + const { + forecasts, + created_at, + model_version, + device_id, + site_id, + } = forecastData; + + // Transform the forecasts into the expected format + const values = forecasts.map((forecast) => ({ + time: forecast.timestamp, + forecast_horizon: forecast.horizon, + pm2_5: { + value: forecast.measurements.pm2_5, + confidence_lower: forecast.measurements.confidence.pm2_5_lower, + confidence_upper: forecast.measurements.confidence.pm2_5_upper, + }, + pm10: { + value: forecast.measurements.pm10, + confidence_lower: forecast.measurements.confidence.pm10_lower, + confidence_upper: forecast.measurements.confidence.pm10_upper, + }, + })); + + return { + device_id, + site_id, + forecast_created_at: created_at, + model_version, + values, + }; +}; + +// New function to handle forecast messages +const consumeForecasts = async (messageData) => { + try { + if (isEmpty(messageData)) { + logger.error( + `KAFKA: forecast message is undefined --- ${stringify(messageData)}` + ); + return; + } + + const repairedJSONString = jsonrepair(messageData); + const forecastData = JSON.parse(repairedJSONString); + + // Validate the incoming forecast data against our schema + const options = { + abortEarly: false, + }; + + const { error, value } = forecastSchema.validate(forecastData, options); + + if (error) { + const errorDetails = error.details.map((detail) => ({ + message: detail.message, + key: detail.context.key, + path: detail.path.join("."), + })); + logger.error(`Forecast validation errors: ${stringify(errorDetails)}`); + return; + } + + // Transform the validated data to match the expected format for createForecastUtil + const transformedData = transformForecastData(forecastData); + + // Create the request object expected by createForecastUtil.create + const request = { + body: transformedData, + query: { + tenant: constants.DEFAULT_TENANT, + }, + }; + + // Store the forecast using createForecastUtil + const response = await createForecastUtil.create(request, (error) => { + if (error) { + logger.error(`KAFKA: forecast creation error --- ${stringify(error)}`); + throw error; + } + }); + + if (response.success === false) { + logger.error(`KAFKA: failed to store forecasts --- ${response.message}`); + } else if (response.success === true) { + logger.info( + `KAFKA: successfully stored forecasts --- ${response.message}` + ); + logger.info(`KAFKA: stored ${response.data.length} forecast days`); + } + } catch (error) { + logger.error(`🐛🐛 KAFKA: forecast error message --- ${error.message}`); + logger.error( + `🐛🐛 KAFKA: forecast full error object --- ${stringify(error)}` + ); + } +}; + const kafkaConsumer = async () => { try { const kafka = new Kafka({ @@ -137,19 +275,20 @@ const kafkaConsumer = async () => { // Define topic-to-operation function mapping const topicOperations = { - ["hourly-measurements-topic"]: consumeHourlyMeasurements, + "hourly-measurements-topic": consumeHourlyMeasurements, + "airqo.forecasts": consumeForecasts, }; await consumer.connect(); - // First, subscribe to all topics + // Subscribe to all topics await Promise.all( Object.keys(topicOperations).map((topic) => consumer.subscribe({ topic, fromBeginning: false }) ) ); - // Then, start consuming messages + // Start consuming messages await consumer.run({ eachMessage: async ({ topic, partition, message }) => { try { diff --git a/src/device-registry/controllers/create-event.js b/src/device-registry/controllers/create-event.js index 5c1fb30951..0d8302504f 100644 --- a/src/device-registry/controllers/create-event.js +++ b/src/device-registry/controllers/create-event.js @@ -1267,6 +1267,8 @@ const createEvent = { request.query.metadata = "site_id"; request.query.averages = "events"; request.query.brief = "yes"; + request.query.quality_checks = "no"; + const { cohort_id, grid_id } = { ...req.query, ...req.params }; let locationErrors = 0; @@ -1331,6 +1333,92 @@ const createEvent = { return; } }, + listAveragesV2: async (req, res, next) => { + try { + const errors = extractErrorsFromRequest(req); + if (errors) { + next( + new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors) + ); + return; + } + + const request = req; + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + request.query.tenant = isEmpty(req.query.tenant) + ? defaultTenant + : req.query.tenant; + + request.query.recent = "no"; + request.query.metadata = "site_id"; + request.query.averages = "events"; + request.query.brief = "yes"; + request.query.quality_checks = "yes"; + + const { cohort_id, grid_id } = { ...req.query, ...req.params }; + + let locationErrors = 0; + + if (cohort_id) { + await processCohortIds(cohort_id, request); + if (isEmpty(request.query.device_id)) { + locationErrors++; + } + } else if (grid_id) { + await processGridIds(grid_id, request); + if (isEmpty(request.query.site_id)) { + locationErrors++; + } + } + + if (locationErrors === 0) { + const result = await createEventUtil.listAveragesV2(request, next); + + if (isEmpty(result) || res.headersSent) { + return; + } + + if (result.success === true) { + const status = result.status ? result.status : httpStatus.OK; + + res.status(status).json({ + success: true, + isCache: result.isCache, + message: result.message, + measurements: result.data, + }); + } else if (result.success === false) { + const status = result.status + ? result.status + : httpStatus.INTERNAL_SERVER_ERROR; + const errors = result.errors ? result.errors : { message: "" }; + res.status(status).json({ + success: false, + errors, + message: result.message, + }); + } + } else { + res.status(httpStatus.BAD_REQUEST).json({ + success: false, + errors: { + message: `Unable to process measurements for the provided measurement IDs`, + }, + message: "Bad Request", + }); + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + return; + } + }, listHistorical: async (req, res, next) => { try { const errors = extractErrorsFromRequest(req); diff --git a/src/device-registry/controllers/create-forecast.js b/src/device-registry/controllers/create-forecast.js new file mode 100644 index 0000000000..2b9bf502e5 --- /dev/null +++ b/src/device-registry/controllers/create-forecast.js @@ -0,0 +1,108 @@ +const httpStatus = require("http-status"); +const { extractErrorsFromRequest, HttpError } = require("@utils/errors"); +const constants = require("@config/constants"); +const log4js = require("log4js"); +const logger = log4js.getLogger( + `${constants.ENVIRONMENT} -- forecast-controller` +); +const forecastUtil = require("@utils/create-forecast"); +const isEmpty = require("is-empty"); + +const handleError = (error, next) => { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError("Internal Server Error", httpStatus.INTERNAL_SERVER_ERROR, { + message: error.message, + }) + ); +}; + +const validateRequest = (req, next) => { + const errors = extractErrorsFromRequest(req); + if (errors) { + next(new HttpError("bad request errors", httpStatus.BAD_REQUEST, errors)); + return false; + } + return true; +}; + +const getTenantFromRequest = (req) => { + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + return isEmpty(req.query.tenant) ? defaultTenant : req.query.tenant; +}; + +const sendResponse = (res, result) => { + if (result.success === true) { + const status = result.status || httpStatus.OK; + res.status(status).json({ + success: true, + message: result.message, + [result.dataKey || "forecast"]: result.data || [], + }); + } else { + const status = result.status || httpStatus.INTERNAL_SERVER_ERROR; + res.status(status).json({ + success: false, + message: result.message, + errors: result.errors || { message: "" }, + }); + } +}; + +const handleControllerAction = async ( + req, + res, + next, + utilFunction, + dataKey +) => { + try { + if (!validateRequest(req, next)) return; + + const request = req; + request.query.tenant = getTenantFromRequest(req); + + const result = await utilFunction(request, next); + + if (isEmpty(result) || res.headersSent) return; + + result.dataKey = dataKey; + sendResponse(res, result); + } catch (error) { + handleError(error, next); + } +}; + +const forecastController = { + create: async (req, res, next) => { + await handleControllerAction( + req, + res, + next, + forecastUtil.create, + "forecast" + ); + }, + + listByDevice: async (req, res, next) => { + await handleControllerAction( + req, + res, + next, + forecastUtil.listByDevice, + "forecasts" + ); + }, + + listBySite: async (req, res, next) => { + await handleControllerAction( + req, + res, + next, + forecastUtil.listBySite, + "forecasts" + ); + }, +}; + +module.exports = forecastController; diff --git a/src/device-registry/models/Event.js b/src/device-registry/models/Event.js index 5e05cc2a3e..ddd63d39aa 100644 --- a/src/device-registry/models/Event.js +++ b/src/device-registry/models/Event.js @@ -2947,6 +2947,292 @@ eventSchema.statics.getAirQualityAverages = async function(siteId, next) { } }; +eventSchema.statics.v2_getAirQualityAverages = async function(siteId, next) { + try { + const TIMEZONE = "Africa/Kampala"; + const MIN_READINGS_PER_DAY = 12; // Minimum readings per day for validity + + const now = moment() + .tz(TIMEZONE) + .toDate(); + const today = moment() + .tz(TIMEZONE) + .startOf("day") + .toDate(); + const twoWeeksAgo = moment() + .tz(TIMEZONE) + .startOf("day") + .subtract(14, "days") + .toDate(); + + logText("Debug Info:"); + logObject("TIMEZONE", TIMEZONE); + logObject("now", now); + logObject("today", today); + logObject("twoWeeksAgo", twoWeeksAgo); + + const result = await this.aggregate([ + // Initial match + { + $match: { + "values.site_id": mongoose.Types.ObjectId(siteId), + "values.time": { $gte: twoWeeksAgo, $lte: now }, + }, + }, + + // Unwind values + { + $unwind: { + path: "$values", + preserveNullAndEmptyArrays: false, + }, + }, + + // Data quality filtering + { + $match: { + "values.time": { $gte: twoWeeksAgo, $lte: now }, + "values.pm2_5.value": { + $exists: true, + $ne: null, + $gte: 0, + $lte: 1000, + }, + }, + }, + + // Project fields + { + $project: { + _id: 0, + time: "$values.time", + pm2_5: "$values.pm2_5.value", + yearWeek: { + $let: { + vars: { + dateParts: { + $dateToParts: { + date: "$values.time", + timezone: TIMEZONE, + iso8601: true, + }, + }, + }, + in: { + $concat: [ + { $toString: "$$dateParts.isoWeekYear" }, + "-", + { + $cond: [ + { $lt: ["$$dateParts.isoWeek", 10] }, + { $concat: ["0", { $toString: "$$dateParts.isoWeek" }] }, + { $toString: "$$dateParts.isoWeek" }, + ], + }, + ], + }, + }, + }, + dayOfYear: { + $dateToString: { + format: "%Y-%m-%d", + date: "$values.time", + timezone: TIMEZONE, + }, + }, + hourOfDay: { + $hour: { + date: "$values.time", + timezone: TIMEZONE, + }, + }, + }, + }, + + // Group by day with data quality metrics + { + $group: { + _id: "$dayOfYear", + dailyAverage: { $avg: "$pm2_5" }, + readingCount: { $sum: 1 }, + uniqueHours: { $addToSet: "$hourOfDay" }, + yearWeek: { $first: "$yearWeek" }, + minReading: { $min: "$pm2_5" }, + maxReading: { $max: "$pm2_5" }, + }, + }, + + // Add data quality indicators + { + $addFields: { + dataQuality: { + hasMinReadings: { $gte: ["$readingCount", MIN_READINGS_PER_DAY] }, + hoursCovered: { $size: "$uniqueHours" }, + readingSpread: { $subtract: ["$maxReading", "$minReading"] }, + }, + }, + }, + + // Group by week with quality metrics + { + $group: { + _id: "$yearWeek", + weeklyAverage: { $avg: "$dailyAverage" }, + daysWithData: { $sum: 1 }, + daysWithMinReadings: { + $sum: { $cond: ["$dataQuality.hasMinReadings", 1, 0] }, + }, + avgHoursCovered: { $avg: "$dataQuality.hoursCovered" }, + days: { + $push: { + date: "$_id", + average: "$dailyAverage", + readingCount: "$readingCount", + hoursCovered: "$dataQuality.hoursCovered", + readingSpread: "$dataQuality.readingSpread", + }, + }, + }, + }, + + // Sort and limit to 2 weeks + { $sort: { _id: -1 } }, + { $limit: 2 }, + ]).allowDiskUse(true); + + if (result.length < 2) { + return { + success: false, + message: "Insufficient data for comparison", + status: httpStatus.NOT_FOUND, + }; + } + + const [currentWeek, previousWeek] = result; + const todayStr = moment(today) + .tz(TIMEZONE) + .format("YYYY-MM-DD"); + const todayData = currentWeek.days.find((day) => day.date === todayStr); + + // Calculate percentage difference without capping + const percentageDifference = + previousWeek.weeklyAverage !== 0 + ? ((currentWeek.weeklyAverage - previousWeek.weeklyAverage) / + previousWeek.weeklyAverage) * + 100 + : 0; + + // Calculate data quality score + const dataQualityScore = calculateWeeklyDataQuality( + currentWeek, + previousWeek + ); + + return { + success: true, + data: { + dailyAverage: todayData + ? { + value: parseFloat(todayData.average.toFixed(2)), + readingCount: todayData.readingCount, + hoursCovered: todayData.hoursCovered, + } + : null, + percentageDifference: parseFloat(percentageDifference.toFixed(2)), + weeklyAverages: { + currentWeek: parseFloat(currentWeek.weeklyAverage.toFixed(2)), + previousWeek: parseFloat(previousWeek.weeklyAverage.toFixed(2)), + }, + dataQuality: { + score: dataQualityScore, + currentWeek: { + daysWithData: currentWeek.daysWithData, + daysWithMinReadings: currentWeek.daysWithMinReadings, + averageHoursCovered: parseFloat( + currentWeek.avgHoursCovered.toFixed(1) + ), + }, + previousWeek: { + daysWithData: previousWeek.daysWithData, + daysWithMinReadings: previousWeek.daysWithMinReadings, + averageHoursCovered: parseFloat( + previousWeek.avgHoursCovered.toFixed(1) + ), + }, + warning: + dataQualityScore < 0.7 + ? "Low data quality may affect accuracy of comparison" + : null, + }, + }, + message: "Successfully retrieved air quality averages", + status: httpStatus.OK, + }; + } catch (error) { + logger.error( + `Internal Server Error --- getAirQualityAverages --- ${error.message}` + ); + logObject("error", error); + next( + new HttpError("Internal Server Error", httpStatus.INTERNAL_SERVER_ERROR, { + message: error.message, + }) + ); + } +}; + +function calculateWeeklyDataQuality(currentWeek, previousWeek) { + const IDEAL_DAYS = 7; + const IDEAL_HOURS = 24; + + // Calculate scores for each week + const currentScore = + (currentWeek.daysWithMinReadings / IDEAL_DAYS) * + (currentWeek.avgHoursCovered / IDEAL_HOURS); + + const previousScore = + (previousWeek.daysWithMinReadings / IDEAL_DAYS) * + (previousWeek.avgHoursCovered / IDEAL_HOURS); + + // Return average score (0-1 range) + return (currentScore + previousScore) / 2; +} + +// Helper function to calculate confidence score +function calculateConfidenceScore( + currentWeek, + baselineWeeks, + minDataPointsPerDay +) { + const maxPossibleReadings = 24; // Assuming hourly readings + const idealDaysPerWeek = 7; + + // Score current week data completeness + const currentWeekScore = + (currentWeek.daysWithData / idealDaysPerWeek) * + (currentWeek.days.reduce( + (acc, day) => acc + day.readingCount / maxPossibleReadings, + 0 + ) / + currentWeek.daysWithData); + + // Score baseline weeks data completeness + const baselineScore = + baselineWeeks.reduce((acc, week) => { + const weekScore = + (week.daysWithData / idealDaysPerWeek) * + (week.days.reduce( + (acc, day) => acc + day.readingCount / maxPossibleReadings, + 0 + ) / + week.daysWithData); + return acc + weekScore; + }, 0) / baselineWeeks.length; + + // Combine scores (giving more weight to current week) + return (currentWeekScore * 0.6 + baselineScore * 0.4) * 100; +} + const eventsModel = (tenant) => { const defaultTenant = constants.DEFAULT_TENANT || "airqo"; const dbTenant = isEmpty(tenant) ? defaultTenant : tenant; diff --git a/src/device-registry/models/Forecast.js b/src/device-registry/models/Forecast.js new file mode 100644 index 0000000000..44565f6e9d --- /dev/null +++ b/src/device-registry/models/Forecast.js @@ -0,0 +1,276 @@ +const mongoose = require("mongoose"); +const { Schema, model } = require("mongoose"); +const ObjectId = Schema.Types.ObjectId; +const { logObject } = require("@utils/log"); +const httpStatus = require("http-status"); +const { HttpError } = require("@utils/errors"); +const isEmpty = require("is-empty"); +const constants = require("@config/constants"); +const { getModelByTenant } = require("@config/database"); +const log4js = require("log4js"); +const logger = log4js.getLogger(`${constants.ENVIRONMENT} -- forecast-model`); + +const forecastValueSchema = new Schema({ + time: { + type: Date, + required: true, + }, + forecast_created_at: { + type: Date, + required: true, + }, + forecast_horizon: { + type: Number, + required: true, + }, + device_id: { + type: ObjectId, + required: true, + }, + site_id: { + type: ObjectId, + required: true, + }, + pm2_5: { + value: { type: Number, default: null }, + confidence_lower: { type: Number, default: null }, + confidence_upper: { type: Number, default: null }, + }, + pm10: { + value: { type: Number, default: null }, + confidence_lower: { type: Number, default: null }, + confidence_upper: { type: Number, default: null }, + }, + model_version: { + type: String, + required: true, + }, +}); + +const forecastSchema = new Schema( + { + day: { + type: String, + required: true, + }, + device_id: { + type: ObjectId, + required: true, + }, + site_id: { + type: ObjectId, + required: true, + }, + first: { + type: Date, + required: true, + }, + last: { + type: Date, + required: true, + }, + values: [forecastValueSchema], + }, + { + timestamps: true, + } +); + +forecastSchema.methods = { + toJSON() { + return { + _id: this._id, + day: this.day, + device_id: this.device_id, + site_id: this.site_id, + first: this.first, + last: this.last, + values: this.values, + createdAt: this.createdAt, + updatedAt: this.updatedAt, + }; + }, +}; + +forecastSchema.index({ device_id: 1 }); +forecastSchema.index({ day: 1 }); + +forecastSchema.statics = { + async register(args, next) { + try { + let createdForecast = await this.create(args); + + if (!isEmpty(createdForecast)) { + let data = createdForecast._doc; + data.__v = undefined; + data.updatedAt = undefined; + return { + success: true, + data, + message: "Forecast created", + status: httpStatus.CREATED, + }; + } else if (isEmpty(createdForecast)) { + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: "Forecast not created despite successful operation" } + ) + ); + } + } catch (error) { + logObject("the error", error); + let response = {}; + let message = "validation errors for some of the provided fields"; + let status = httpStatus.CONFLICT; + Object.entries(error.errors || {}).forEach(([key, value]) => { + response.message = value.message; + response[key] = value.message; + return response; + }); + + next(new HttpError(message, status, response)); + } + }, + + async list({ skip = 0, limit = 100, filter = {} } = {}, next) { + try { + const response = await this.find(filter) + .sort({ createdAt: -1 }) + .skip(skip) + .limit(limit) + .lean(); + + if (!isEmpty(response)) { + return { + success: true, + message: "successfully retrieved the forecasts", + data: response, + status: httpStatus.OK, + }; + } else if (isEmpty(response)) { + return { + success: true, + message: "no forecasts exist, please crosscheck", + status: httpStatus.OK, + data: [], + }; + } + } catch (error) { + logObject("the error", error); + const stringifiedMessage = JSON.stringify(error || ""); + logger.error(`🐛🐛 Internal Server Error -- ${stringifiedMessage}`); + + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + async modify({ filter = {}, update = {} } = {}, next) { + try { + let options = { new: true, useFindAndModify: false, upsert: false }; + + if (update._id) { + delete update._id; + } + + const updatedForecast = await this.findOneAndUpdate( + filter, + update, + options + ); + + if (!isEmpty(updatedForecast)) { + return { + success: true, + message: "successfully modified the forecast", + data: updatedForecast._doc, + status: httpStatus.OK, + }; + } else if (isEmpty(updatedForecast)) { + next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + ...filter, + message: "forecast does not exist, please crosscheck", + }) + ); + } + } catch (error) { + logObject("the error", error); + const stringifiedMessage = JSON.stringify(error || ""); + logger.error(`🐛🐛 Internal Server Error -- ${stringifiedMessage}`); + + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + async remove({ filter = {} } = {}, next) { + try { + let options = { + projection: { + _id: 1, + day: 1, + device_id: 1, + site_id: 1, + first: 1, + last: 1, + values: 1, + }, + }; + let removedForecast = await this.findOneAndRemove(filter, options).exec(); + + if (!isEmpty(removedForecast)) { + return { + success: true, + message: "successfully removed the forecast", + data: removedForecast._doc, + status: httpStatus.OK, + }; + } else if (isEmpty(removedForecast)) { + next( + new HttpError("Bad Request Error", httpStatus.BAD_REQUEST, { + ...filter, + message: "forecast does not exist, please crosscheck", + }) + ); + } + } catch (error) { + logObject("the error", error); + const stringifiedMessage = JSON.stringify(error || ""); + logger.error(`🐛🐛 Internal Server Error -- ${stringifiedMessage}`); + + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, +}; + +const ForecastModel = (tenant) => { + const defaultTenant = constants.DEFAULT_TENANT || "airqo"; + const dbTenant = isEmpty(tenant) ? defaultTenant : tenant; + try { + const forecasts = mongoose.model("forecasts"); + return forecasts; + } catch (errors) { + return getModelByTenant(dbTenant.toLowerCase(), "forecast", forecastSchema); + } +}; + +module.exports = ForecastModel; diff --git a/src/device-registry/models/HealthTips.js b/src/device-registry/models/HealthTips.js index 8466262b10..025b1b09dd 100644 --- a/src/device-registry/models/HealthTips.js +++ b/src/device-registry/models/HealthTips.js @@ -197,54 +197,7 @@ tipsSchema.statics = { }, async modify({ filter = {}, update = {}, opts = { new: true } } = {}, next) { try { - logObject("the filter in the model", filter); - logObject("the update in the model", update); - logObject("the opts in the model", opts); - let modifiedUpdateBody = Object.assign({}, update); - if (modifiedUpdateBody._id) { - delete modifiedUpdateBody._id; - } - - delete modifiedUpdateBody.aqi_category; - - switch (update.aqi_category) { - case "good": - modifiedUpdateBody.aqi_category = { min: 0, max: 12.09 }; - break; - case "moderate": - modifiedUpdateBody.aqi_category = { min: 12.1, max: 35.49 }; - break; - case "u4sg": - modifiedUpdateBody.aqi_category = { min: 35.5, max: 55.49 }; - break; - case "unhealthy": - modifiedUpdateBody.aqi_category = { min: 55.5, max: 150.49 }; - break; - case "very_unhealthy": - modifiedUpdateBody.aqi_category = { min: 150.5, max: 250.49 }; - break; - case "hazardous": - modifiedUpdateBody.aqi_category = { min: 250.5, max: 500 }; - break; - default: - } - - let options = opts; - let keys = {}; - const setProjection = (object) => { - Object.keys(object).forEach((element) => { - keys[element] = 1; - }); - return keys; - }; - logObject("the new modifiedUpdateBody", modifiedUpdateBody); - - const updatedTip = await this.findOneAndUpdate( - filter, - modifiedUpdateBody, - options - ); - logObject("updatedTip", updatedTip); + const updatedTip = await this.findOneAndUpdate(filter, update, opts); if (!isEmpty(updatedTip)) { return { success: true, diff --git a/src/device-registry/routes/v2/forecasts.js b/src/device-registry/routes/v2/forecasts.js new file mode 100644 index 0000000000..e363e12599 --- /dev/null +++ b/src/device-registry/routes/v2/forecasts.js @@ -0,0 +1,50 @@ +const express = require("express"); +const router = express.Router(); +const forecastController = require("@controllers/create-forecast"); +const forecastValidations = require("@validators/forecast.validators"); +const { oneOf } = require("express-validator"); + +const validatePagination = (req, res, next) => { + let limit = parseInt(req.query.limit, 10); + const skip = parseInt(req.query.skip, 10); + if (Number.isNaN(limit) || limit < 1) { + limit = 1000; + } + if (limit > 2000) { + limit = 2000; + } + if (Number.isNaN(skip) || skip < 0) { + req.query.skip = 0; + } + req.query.limit = limit; + + next(); +}; + +const headers = (req, res, next) => { + res.setHeader("Access-Control-Allow-Origin", "*"); + res.header( + "Access-Control-Allow-Headers", + "Origin, X-Requested-With, Content-Type, Accept, Authorization" + ); + res.header("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE"); + next(); +}; + +router.use(headers); +router.use(validatePagination); + +/******************* forecasts use-case ***************/ +router.post("/", oneOf(forecastValidations.create), forecastController.create); +router.get( + "/devices/:deviceId", + oneOf(forecastValidations.listByDevice), + forecastController.listByDevice +); +router.get( + "/sites/:siteId", + oneOf(forecastValidations.listBySite), + forecastController.listBySite +); + +module.exports = router; diff --git a/src/device-registry/routes/v2/index.js b/src/device-registry/routes/v2/index.js index ef1ec99c7a..f61d15a923 100644 --- a/src/device-registry/routes/v2/index.js +++ b/src/device-registry/routes/v2/index.js @@ -11,6 +11,7 @@ router.use("/measurements", require("@routes/v2/measurements")); router.use("/signals", require("@routes/v2/signals")); router.use("/locations", require("@routes/v2/locations")); router.use("/photos", require("@routes/v2/photos")); +router.use("/forecasts", require("@routes/v2/forecasts")); router.use("/tips", require("@routes/v2/tips")); router.use("/kya", require("@routes/v2/kya")); router.use("/cohorts", require("@routes/v2/cohorts")); diff --git a/src/device-registry/routes/v2/measurements.js b/src/device-registry/routes/v2/measurements.js index cdab50de56..4506681999 100644 --- a/src/device-registry/routes/v2/measurements.js +++ b/src/device-registry/routes/v2/measurements.js @@ -1328,6 +1328,136 @@ router.get( averagesLimiter, eventController.listAverages ); + +/** + * @route GET /sites/:site_id/averages + * @description Get average measurements for a specific site + * @param {string} site_id - MongoDB ObjectId of the site + * @query {string} [tenant] - Optional tenant identifier + * @query {string} [startTime] - ISO8601 start time + * @query {string} [endTime] - ISO8601 end time + * @query {string} [frequency] - Data frequency (hourly|daily|raw|minute) + * @returns {object} Average measurements data + */ +router.get( + "/sites/:site_id/v2-averages", + oneOf([ + query("tenant") + .optional() + .notEmpty() + .withMessage("tenant should not be empty if provided") + .bail() + .trim() + .toLowerCase() + .isIn(constants.NETWORKS) + .withMessage("the tenant value is not among the expected ones"), + ]), + oneOf([ + [ + param("site_id") + .exists() + .withMessage("the site_id should be provided") + .bail() + .notEmpty() + .withMessage("the provided site_id cannot be empty") + .bail() + .trim() + .isMongoId() + .withMessage("the site_id must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + ], + ]), + oneOf([ + [ + query("startTime") + .optional() + .notEmpty() + .withMessage("startTime cannot be empty IF provided") + .bail() + .trim() + .isISO8601({ strict: true, strictSeparator: true }) + .withMessage("startTime must be a valid datetime."), + query("endTime") + .optional() + .notEmpty() + .withMessage("endTime cannot be empty IF provided") + .bail() + .trim() + .isISO8601({ strict: true, strictSeparator: true }) + .withMessage("endTime must be a valid datetime."), + query("frequency") + .optional() + .notEmpty() + .withMessage("the frequency cannot be empty if provided") + .bail() + .trim() + .toLowerCase() + .isIn(["hourly", "daily", "raw", "minute"]) + .withMessage( + "the frequency value is not among the expected ones which include: hourly, daily, minute and raw" + ), + query("format") + .optional() + .notEmpty() + .withMessage("the format cannot be empty if provided") + .bail() + .trim() + .toLowerCase() + .isIn(["json", "csv"]) + .withMessage( + "the format value is not among the expected ones which include: csv and json" + ), + query("external") + .optional() + .notEmpty() + .withMessage("external cannot be empty IF provided") + .bail() + .trim() + .toLowerCase() + .isIn(["yes", "no"]) + .withMessage( + "the external value is not among the expected ones which include: no and yes" + ), + query("recent") + .optional() + .notEmpty() + .withMessage("recent cannot be empty IF provided") + .bail() + .trim() + .toLowerCase() + .isIn(["yes", "no"]) + .withMessage( + "the recent value is not among the expected ones which include: no and yes" + ), + query("metadata") + .optional() + .notEmpty() + .withMessage("metadata cannot be empty IF provided") + .bail() + .trim() + .toLowerCase() + .isIn(["site", "site_id", "device", "device_id"]) + .withMessage( + "valid values include: site, site_id, device and device_id" + ), + query("test") + .optional() + .notEmpty() + .withMessage("test cannot be empty IF provided") + .bail() + .trim() + .toLowerCase() + .isIn(["yes", "no"]) + .withMessage("valid values include: YES and NO"), + ], + ]), + averagesLimiter, + eventController.listAveragesV2 +); + router.get( "/sites/:site_id", oneOf([ diff --git a/src/device-registry/routes/v2/tips.js b/src/device-registry/routes/v2/tips.js index df56513a4d..5f8cc34890 100644 --- a/src/device-registry/routes/v2/tips.js +++ b/src/device-registry/routes/v2/tips.js @@ -1,24 +1,8 @@ const express = require("express"); const router = express.Router(); const healthTipController = require("@controllers/create-health-tips"); -const { check, oneOf, query, body, param } = require("express-validator"); -const constants = require("@config/constants"); -const mongoose = require("mongoose"); -const ObjectId = mongoose.Types.ObjectId; -const { logElement, logText, logObject } = require("@utils/log"); -const isEmpty = require("is-empty"); -const NetworkModel = require("@models/Network"); -const validNetworks = async () => { - const networks = await NetworkModel("airqo").distinct("name"); - return networks.map((network) => network.toLowerCase()); -}; - -const validateNetwork = async (value) => { - const networks = await validNetworks(); - if (!networks.includes(value.toLowerCase())) { - throw new Error("Invalid network"); - } -}; +const healthTipValidations = require("@validators/tips.validators"); +const { oneOf } = require("express-validator"); const validatePagination = (req, res, next) => { let limit = parseInt(req.query.limit, 10); @@ -46,202 +30,21 @@ const headers = (req, res, next) => { res.header("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE"); next(); }; + router.use(headers); router.use(validatePagination); /******************* create-health-tip use-case ***************/ -router.get( - "/", - oneOf([ - [ - query("tenant") - .optional() - .notEmpty() - .withMessage("the tenant cannot be empty, if provided") - .bail() - .trim() - .toLowerCase() - .isIn(constants.NETWORKS) - .withMessage("the tenant value is not among the expected ones"), - ], - ]), - oneOf([ - [ - query("id") - .optional() - .notEmpty() - .withMessage("this tip identifier cannot be empty") - .bail() - .trim() - .isMongoId() - .withMessage("id must be an object ID") - .bail() - .customSanitizer((value) => { - return ObjectId(value); - }), - ], - ]), - oneOf([ - [ - query("language") - .optional() - .notEmpty() - .withMessage("the language cannot be empty when provided") - .bail() - .trim(), - ], - ]), - healthTipController.list -); +router.get("/", oneOf(healthTipValidations.list), healthTipController.list); router.post( "/", - oneOf([ - [ - query("tenant") - .optional() - .notEmpty() - .withMessage("the tenant cannot be empty, if provided") - .bail() - .trim() - .toLowerCase() - .isIn(constants.NETWORKS) - .withMessage("the tenant value is not among the expected ones"), - ], - ]), - oneOf([ - [ - body("description") - .exists() - .withMessage("the description is missing in request") - .bail() - .trim(), - body("title") - .exists() - .withMessage("the title is missing in request") - .bail() - .trim(), - body("image") - .exists() - .withMessage("the image is missing in request") - .bail() - .trim(), - body("aqi_category") - .exists() - .withMessage("the aqi_category is missing in request") - .bail() - .trim() - .toLowerCase() - .isIn(constants.AQI_CATEGORIES) - .withMessage( - "the aqi_category is not among the expected ones: good,moderate,u4sg,unhealthy,very_unhealthy,hazardous" - ), - ], - ]), + oneOf(healthTipValidations.create), healthTipController.create ); -router.put( - "/", - oneOf([ - [ - query("tenant") - .optional() - .notEmpty() - .withMessage("the tenant cannot be empty, if provided") - .bail() - .trim() - .toLowerCase() - .isIn(constants.NETWORKS) - .withMessage("the tenant value is not among the expected ones"), - ], - ]), - oneOf([ - query("id") - .exists() - .withMessage( - "the tip unique identifier is missing in request, consider using the id" - ) - .bail() - .trim() - .isMongoId() - .withMessage("id must be an object ID") - .bail() - .customSanitizer((value) => { - return ObjectId(value); - }), - ]), - oneOf([ - body() - .notEmpty() - .custom((value) => { - return !isEmpty(value); - }) - .withMessage("the request body should not be empty"), - ]), - oneOf([ - [ - body("description") - .optional() - .notEmpty() - .withMessage("the description is missing in request") - .bail() - .trim(), - body("title") - .optional() - .notEmpty() - .withMessage("the title is missing in request") - .bail() - .trim(), - body("image") - .optional() - .notEmpty() - .withMessage("the image is missing in request") - .bail() - .trim(), - body("aqi_category") - .optional() - .notEmpty() - .withMessage("the aqi_category is missing in request") - .bail() - .trim() - .toLowerCase() - .isIn(constants.AQI_CATEGORIES) - .withMessage( - "the aqi_category is not among the expected ones: good,moderate,u4sg,unhealthy,very_unhealthy,hazardous" - ), - ], - ]), - healthTipController.update -); +router.put("/", oneOf(healthTipValidations.update), healthTipController.update); router.delete( "/", - oneOf([ - [ - query("tenant") - .optional() - .notEmpty() - .withMessage("tenant should not be empty if provided") - .bail() - .trim() - .toLowerCase() - .isIn(constants.NETWORKS) - .withMessage("the tenant value is not among the expected ones"), - ], - ]), - oneOf([ - query("id") - .exists() - .withMessage( - "the tip identifier is missing in request, consider using the id" - ) - .bail() - .trim() - .isMongoId() - .withMessage("the id must be an object ID") - .bail() - .customSanitizer((value) => { - return ObjectId(value); - }), - ]), + oneOf(healthTipValidations.delete), healthTipController.delete ); diff --git a/src/device-registry/utils/create-airqloud.js b/src/device-registry/utils/create-airqloud.js index 41788f079b..6f69c1f394 100644 --- a/src/device-registry/utils/create-airqloud.js +++ b/src/device-registry/utils/create-airqloud.js @@ -203,6 +203,14 @@ const createAirqloud = { }, retrieveCoordinates: async (request, entity, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; let entityInstance = {}; if (entity === "location") { entityInstance = createLocationUtil; @@ -264,6 +272,14 @@ const createAirqloud = { }, create: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; const { body, query } = request; const { tenant } = query; const { location_id } = body; @@ -358,6 +374,14 @@ const createAirqloud = { }, update: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; let { query, body } = request; let { tenant } = query; @@ -385,6 +409,14 @@ const createAirqloud = { }, delete: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; let { query } = request; let { tenant } = query; let filter = generateFilter.airqlouds(request, next); @@ -409,6 +441,14 @@ const createAirqloud = { }, refresh: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; const { query, body } = request; const { tenant, id, name, admin_level } = query; @@ -490,6 +530,14 @@ const createAirqloud = { }, calculateGeographicalCenter: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; const { body, query } = request; const { coordinates } = body; const { id } = query; @@ -547,6 +595,14 @@ const createAirqloud = { }, findSites: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; const { query, body } = request; const { id, tenant, name, admin_level } = query; let filter = {}; @@ -658,6 +714,14 @@ const createAirqloud = { }, list: async (request, next) => { try { + return { + success: false, + message: "Deprecated Functionality", + status: httpStatus.GONE, + errors: { + message: "Please use Grids or Cohorts, AirQlouds are deprecated", + }, + }; const { tenant, limit, skip, path } = request.query; const filter = generateFilter.airqlouds(request, next); if (!isEmpty(path)) { diff --git a/src/device-registry/utils/create-event.js b/src/device-registry/utils/create-event.js index e566c253e8..88068e5dcb 100644 --- a/src/device-registry/utils/create-event.js +++ b/src/device-registry/utils/create-event.js @@ -320,6 +320,135 @@ async function processEvents(events, next) { return determineResponse(nAdded, eventsAdded, eventsRejected, errors); } +class AirQualityService { + constructor(tenant) { + this.EventModel = EventModel(tenant); + } + + async getAirQualityData(request, next, version = "v1") { + const { language, site_id } = { ...request.query, ...request.params }; + let responseFromListEvents; + + try { + // Try to get from cache first + const cacheResult = await this.tryGetCache(request, next); + if (cacheResult.success) { + return cacheResult.data; + } + + // Get data based on version + responseFromListEvents = + version === "v2" + ? await this.EventModel.v2_getAirQualityAverages(site_id, next) + : await this.EventModel.getAirQualityAverages(site_id, next); + + // Handle translation if needed (only for v1 as v2 doesn't include health tips) + if (version === "v1") { + await this.translateHealthTips(responseFromListEvents, language, next); + } + + // Set cache if successful + if (responseFromListEvents.success) { + await this.trySetCache(responseFromListEvents.data, request, next); + + return { + success: true, + message: isEmpty(responseFromListEvents.data) + ? "no measurements for this search" + : responseFromListEvents.message, + data: responseFromListEvents.data, + status: responseFromListEvents.status || "", + isCache: false, + }; + } + + logger.error( + `Unable to retrieve events --- ${JSON.stringify( + responseFromListEvents.errors + )}` + ); + return { + success: false, + message: responseFromListEvents.message, + errors: responseFromListEvents.errors || { message: "" }, + status: responseFromListEvents.status || "", + isCache: false, + }; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + } + + async translateHealthTips(response, language, next) { + if ( + language !== undefined && + !isEmpty(response) && + response.success === true && + !isEmpty(response.data) + ) { + const data = response.data; + for (const event of data) { + const translatedHealthTips = await translateUtil.translateTips( + { healthTips: event.health_tips, targetLanguage: language }, + next + ); + if (translatedHealthTips.success === true) { + event.health_tips = translatedHealthTips.data; + } + } + } + } + + async tryGetCache(request, next) { + try { + return await Promise.race([ + createEvent.getCache(request, next), + this.getCacheTimeout(), + ]); + } catch (error) { + logger.error(`🐛🐛 Cache Get Error -- ${JSON.stringify(error)}`); + return { success: false }; + } + } + + async trySetCache(data, request, next) { + try { + const result = await Promise.race([ + createEvent.setCache(data, request, next), + this.getCacheTimeout(), + ]); + + if (!result.success) { + logger.error( + `🐛🐛 Cache Set Error -- ${JSON.stringify( + result.errors || "Unknown error" + )}` + ); + } + } catch (error) { + logger.error(`🐛🐛 Cache Set Error -- ${JSON.stringify(error)}`); + } + } + + getCacheTimeout() { + return new Promise((resolve) => + setTimeout(resolve, 60000, { + success: false, + message: "Internal Server Error", + status: httpStatus.INTERNAL_SERVER_ERROR, + errors: { message: "Cache timeout" }, + }) + ); + } +} + const createEvent = { getMeasurementsFromBigQuery: async (req, next) => { try { @@ -807,7 +936,7 @@ const createEvent = { ); } }, - listAverages: async (request, next) => { + listAveragesV1: async (request, next) => { try { let missingDataMessage = ""; const { language, site_id, tenant } = { @@ -929,6 +1058,16 @@ const createEvent = { ); } }, + + listAverages: async (request, next) => { + const service = new AirQualityService(request.query.tenant); + return service.getAirQualityData(request, next); + }, + + listAveragesV2: async (request, next) => { + const service = new AirQualityService(request.query.tenant); + return service.getAirQualityData(request, next, "v2"); + }, view: async (request, next) => { try { let missingDataMessage = ""; @@ -2248,6 +2387,7 @@ const createEvent = { averages, threshold, pollutant, + quality_checks, } = { ...request.query, ...request.params }; const currentTime = new Date().toISOString(); const day = generateDateFormatWithoutHrs(currentTime); @@ -2281,6 +2421,7 @@ const createEvent = { _${averages ? averages : "noAverages"} _${threshold ? threshold : "noThreshold"} _${pollutant ? pollutant : "noPollutant"} + _${quality_checks ? quality_checks : "noQualityChecks"} `; } catch (error) { logger.error(`🐛🐛 Internal Server Error ${error.message}`); diff --git a/src/device-registry/utils/create-forecast.js b/src/device-registry/utils/create-forecast.js new file mode 100644 index 0000000000..6cebac38fe --- /dev/null +++ b/src/device-registry/utils/create-forecast.js @@ -0,0 +1,202 @@ +const httpStatus = require("http-status"); +const ForecastModel = require("@models/Forecast"); +const constants = require("@config/constants"); +const { logObject } = require("./log"); +const generateFilter = require("./generate-filter"); +const log4js = require("log4js"); +const logger = log4js.getLogger(`${constants.ENVIRONMENT} -- forecast-util`); +const { HttpError } = require("@utils/errors"); +const { Kafka } = require("kafkajs"); +const isEmpty = require("is-empty"); + +const kafka = new Kafka({ + clientId: constants.KAFKA_CLIENT_ID, + brokers: constants.KAFKA_BOOTSTRAP_SERVERS, +}); + +const processValues = (values, forecast_created_at) => { + const processedValues = values.map((value) => ({ + ...value, + forecast_created_at, + })); + + // Group values by day and create day-specific records + const dayGroups = processedValues.reduce((acc, value) => { + const day = value.time.split("T")[0]; + if (!acc[day]) { + acc[day] = []; + } + acc[day].push(value); + return acc; + }, {}); + + return Object.entries(dayGroups).map(([day, dayValues]) => ({ + day, + values: dayValues, + first: dayValues[0].time, + last: dayValues[dayValues.length - 1].time, + })); +}; + +const forecastUtils = { + create: async (request, next) => { + try { + const { body, query } = request; + const { tenant } = query; + const { + device_id, + site_id, + forecast_created_at, + model_version, + values, + } = body; + + // Process values into day-grouped format + const forecastDays = processValues(values, forecast_created_at); + + // Create multiple forecast records, one per day + const createPromises = forecastDays.map(async (dayForecast) => { + const forecastData = { + device_id, + site_id, + day: dayForecast.day, + first: dayForecast.first, + last: dayForecast.last, + values: dayForecast.values.map((value) => ({ + ...value, + device_id, + site_id, + model_version, + })), + }; + + return ForecastModel(tenant).register(forecastData, next); + }); + + const results = await Promise.all(createPromises); + const successfulResults = results.filter((result) => result.success); + + if (successfulResults.length > 0) { + try { + const kafkaProducer = kafka.producer({ + groupId: constants.UNIQUE_PRODUCER_GROUP, + }); + await kafkaProducer.connect(); + await kafkaProducer.send({ + topic: constants.FORECASTS_TOPIC, + messages: [ + { + action: "create", + value: JSON.stringify(successfulResults.map((r) => r.data)), + }, + ], + }); + await kafkaProducer.disconnect(); + } catch (error) { + logger.error(`internal server error -- ${error.message}`); + } + + return { + success: true, + message: "Successfully created forecasts", + data: successfulResults.map((r) => r.data), + status: httpStatus.CREATED, + }; + } else { + return { + success: false, + message: "Failed to create forecasts", + errors: { message: "No forecasts were created successfully" }, + status: httpStatus.INTERNAL_SERVER_ERROR, + }; + } + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + listByDevice: async (request, next) => { + try { + const { params, query } = request; + const { tenant, limit, skip, start_date, end_date } = query; + const { deviceId } = params; + + const filter = { + device_id: deviceId, + }; + + if (start_date || end_date) { + filter.day = {}; + if (start_date) filter.day.$gte = start_date.split("T")[0]; + if (end_date) filter.day.$lte = end_date.split("T")[0]; + } + + const response = await ForecastModel(tenant).list( + { + filter, + limit, + skip, + }, + next + ); + + return response; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, + + listBySite: async (request, next) => { + try { + const { params, query } = request; + const { tenant, limit, skip, start_date, end_date } = query; + const { siteId } = params; + + const filter = { + site_id: siteId, + }; + + if (start_date || end_date) { + filter.day = {}; + if (start_date) filter.day.$gte = start_date.split("T")[0]; + if (end_date) filter.day.$lte = end_date.split("T")[0]; + } + + const response = await ForecastModel(tenant).list( + { + filter, + limit, + skip, + }, + next + ); + + return response; + } catch (error) { + logger.error(`🐛🐛 Internal Server Error ${error.message}`); + next( + new HttpError( + "Internal Server Error", + httpStatus.INTERNAL_SERVER_ERROR, + { message: error.message } + ) + ); + } + }, +}; + +module.exports = forecastUtils; diff --git a/src/device-registry/validators/forecast.validators.js b/src/device-registry/validators/forecast.validators.js new file mode 100644 index 0000000000..a282d9fa55 --- /dev/null +++ b/src/device-registry/validators/forecast.validators.js @@ -0,0 +1,206 @@ +const { check, oneOf, query, body, param } = require("express-validator"); +const constants = require("@config/constants"); +const mongoose = require("mongoose"); +const ObjectId = mongoose.Types.ObjectId; +const isEmpty = require("is-empty"); + +const commonValidations = { + tenant: [ + query("tenant") + .optional() + .notEmpty() + .withMessage("the tenant cannot be empty, if provided") + .bail() + .trim() + .toLowerCase() + .isIn(constants.NETWORKS) + .withMessage("the tenant value is not among the expected ones"), + ], + deviceId: [ + param("deviceId") + .exists() + .withMessage("the device identifier is missing in request") + .bail() + .trim() + .isMongoId() + .withMessage("deviceId must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + ], + siteId: [ + param("siteId") + .exists() + .withMessage("the site identifier is missing in request") + .bail() + .trim() + .isMongoId() + .withMessage("siteId must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + ], +}; + +const validateForecastValue = { + time: [ + body("values.*.time") + .exists() + .withMessage("time is required for each forecast value") + .bail() + .isISO8601() + .withMessage("time must be a valid ISO 8601 date"), + ], + forecast_horizon: [ + body("values.*.forecast_horizon") + .exists() + .withMessage("forecast_horizon is required for each forecast value") + .bail() + .isInt({ min: 0 }) + .withMessage("forecast_horizon must be a non-negative integer"), + ], + pm2_5: [ + body("values.*.pm2_5.value") + .optional() + .isFloat({ min: 0 }) + .withMessage("pm2_5 value must be a non-negative number"), + body("values.*.pm2_5.confidence_lower") + .optional() + .isFloat({ min: 0 }) + .withMessage("pm2_5 confidence_lower must be a non-negative number"), + body("values.*.pm2_5.confidence_upper") + .optional() + .isFloat({ min: 0 }) + .withMessage("pm2_5 confidence_upper must be a non-negative number") + .custom((value, { req }) => { + const valueIndex = req.body.values.findIndex( + (v) => v.pm2_5.confidence_upper === value + ); + const lower = req.body.values[valueIndex].pm2_5.confidence_lower; + if (lower && value <= lower) { + throw new Error( + "confidence_upper must be greater than confidence_lower" + ); + } + return true; + }), + ], + pm10: [ + body("values.*.pm10.value") + .optional() + .isFloat({ min: 0 }) + .withMessage("pm10 value must be a non-negative number"), + body("values.*.pm10.confidence_lower") + .optional() + .isFloat({ min: 0 }) + .withMessage("pm10 confidence_lower must be a non-negative number"), + body("values.*.pm10.confidence_upper") + .optional() + .isFloat({ min: 0 }) + .withMessage("pm10 confidence_upper must be a non-negative number") + .custom((value, { req }) => { + const valueIndex = req.body.values.findIndex( + (v) => v.pm10.confidence_upper === value + ); + const lower = req.body.values[valueIndex].pm10.confidence_lower; + if (lower && value <= lower) { + throw new Error( + "confidence_upper must be greater than confidence_lower" + ); + } + return true; + }), + ], +}; + +const forecastValidations = { + create: [ + ...commonValidations.tenant, + body("device_id") + .exists() + .withMessage("device_id is required") + .bail() + .isMongoId() + .withMessage("device_id must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + body("site_id") + .exists() + .withMessage("site_id is required") + .bail() + .isMongoId() + .withMessage("site_id must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + body("forecast_created_at") + .exists() + .withMessage("forecast_created_at is required") + .bail() + .isISO8601() + .withMessage("forecast_created_at must be a valid ISO 8601 date"), + body("model_version") + .exists() + .withMessage("model_version is required") + .bail() + .trim() + .matches(/^v\d+\.\d+\.\d+$/) + .withMessage("model_version must be in format vX.Y.Z"), + body("values") + .exists() + .withMessage("values array is required") + .bail() + .isArray() + .withMessage("values must be an array") + .bail() + .notEmpty() + .withMessage("values array cannot be empty"), + ...validateForecastValue.time, + ...validateForecastValue.forecast_horizon, + ...validateForecastValue.pm2_5, + ...validateForecastValue.pm10, + ], + listByDevice: [ + ...commonValidations.tenant, + ...commonValidations.deviceId, + query("start_date") + .optional() + .isISO8601() + .withMessage("start_date must be a valid ISO 8601 date"), + query("end_date") + .optional() + .isISO8601() + .withMessage("end_date must be a valid ISO 8601 date") + .custom((value, { req }) => { + if (req.query.start_date && value <= req.query.start_date) { + throw new Error("end_date must be after start_date"); + } + return true; + }), + ], + listBySite: [ + ...commonValidations.tenant, + ...commonValidations.siteId, + query("start_date") + .optional() + .isISO8601() + .withMessage("start_date must be a valid ISO 8601 date"), + query("end_date") + .optional() + .isISO8601() + .withMessage("end_date must be a valid ISO 8601 date") + .custom((value, { req }) => { + if (req.query.start_date && value <= req.query.start_date) { + throw new Error("end_date must be after start_date"); + } + return true; + }), + ], +}; + +module.exports = forecastValidations; diff --git a/src/device-registry/validators/tips.validators.js b/src/device-registry/validators/tips.validators.js new file mode 100644 index 0000000000..af546831f7 --- /dev/null +++ b/src/device-registry/validators/tips.validators.js @@ -0,0 +1,162 @@ +const { check, oneOf, query, body } = require("express-validator"); +const constants = require("@config/constants"); +const mongoose = require("mongoose"); +const ObjectId = mongoose.Types.ObjectId; +const isEmpty = require("is-empty"); + +const commonValidations = { + tenant: [ + query("tenant") + .optional() + .notEmpty() + .withMessage("the tenant cannot be empty, if provided") + .bail() + .trim() + .toLowerCase() + .isIn(constants.NETWORKS) + .withMessage("the tenant value is not among the expected ones"), + ], + id: [ + query("id") + .exists() + .withMessage("the tip identifier is missing in request") + .bail() + .trim() + .isMongoId() + .withMessage("id must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + ], + optionalId: [ + query("id") + .optional() + .notEmpty() + .withMessage("this tip identifier cannot be empty") + .bail() + .trim() + .isMongoId() + .withMessage("id must be an object ID") + .bail() + .customSanitizer((value) => { + return ObjectId(value); + }), + ], + language: [ + query("language") + .optional() + .notEmpty() + .withMessage("the language cannot be empty when provided") + .bail() + .trim(), + ], +}; + +const healthTipValidations = { + create: [ + ...commonValidations.tenant, + body("description") + .exists() + .withMessage("the description is missing in request") + .bail() + .trim(), + body("title") + .exists() + .withMessage("the title is missing in request") + .bail() + .trim(), + body("image") + .exists() + .withMessage("the image is missing in request") + .bail() + .trim(), + body("aqi_category") + .exists() + .withMessage("the aqi_category is missing in request") + .bail() + .isObject() + .withMessage("aqi_category must be an object") + .bail(), + body("aqi_category.min") + .exists() + .withMessage("aqi_category.min is required") + .bail() + .isNumeric() + .withMessage("aqi_category.min must be a number"), + body("aqi_category.max") + .exists() + .withMessage("aqi_category.max is required") + .bail() + .isNumeric() + .withMessage("aqi_category.max must be a number") + .custom((value, { req }) => { + if (value <= req.body.aqi_category.min) { + throw new Error("max value must be greater than min value"); + } + return true; + }), + ], + list: [ + ...commonValidations.tenant, + ...commonValidations.optionalId, + ...commonValidations.language, + ], + update: [ + ...commonValidations.tenant, + ...commonValidations.id, + body() + .notEmpty() + .custom((value) => { + return !isEmpty(value); + }) + .withMessage("the request body should not be empty"), + body("description") + .optional() + .notEmpty() + .withMessage("the description cannot be empty if provided") + .bail() + .trim(), + body("title") + .optional() + .notEmpty() + .withMessage("the title cannot be empty if provided") + .bail() + .trim(), + body("image") + .optional() + .notEmpty() + .withMessage("the image cannot be empty if provided") + .bail() + .trim(), + body("aqi_category") + .optional() + .notEmpty() + .withMessage("the aqi_category cannot be empty if provided") + .bail() + .isObject() + .withMessage("aqi_category must be an object") + .bail(), + body("aqi_category.min") + .optional() + .isNumeric() + .withMessage("aqi_category.min must be a number"), + body("aqi_category.max") + .optional() + .isNumeric() + .withMessage("aqi_category.max must be a number") + .custom((value, { req }) => { + if ( + req.body.aqi_category && + req.body.aqi_category.min && + value <= req.body.aqi_category.min + ) { + throw new Error("max value must be greater than min value"); + } + return true; + }), + ], + delete: [...commonValidations.tenant, ...commonValidations.id], +}; + +module.exports = healthTipValidations;