From edee1e0cdcea2dad39d95b85f808d0aa2a24c963 Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Fri, 22 Nov 2024 15:50:02 +0530 Subject: [PATCH 1/7] savepoint --- module/observations/helper.js | 24 +++++++++++++++++++++++- 1 file changed, 23 insertions(+), 1 deletion(-) diff --git a/module/observations/helper.js b/module/observations/helper.js index 543e09a4..709b06c9 100644 --- a/module/observations/helper.js +++ b/module/observations/helper.js @@ -217,7 +217,8 @@ module.exports = class ObservationsHelper { static createObservation(data,userId,solution,userRoleInformation="",userProfileInformation = {}) { return new Promise(async (resolve, reject) => { try { - + + let overWriteUserInfoIfReferenceFromProjectsIsFound = false; if (data.entities) { let entitiesToAdd = await entitiesHelper.validateEntities(data.entities, solution.entityType); @@ -227,6 +228,7 @@ module.exports = class ObservationsHelper { if( data.project ) { data.project._id = ObjectId(data.project._id); data.referenceFrom = messageConstants.common.PROJECT; + overWriteUserInfoIfReferenceFromProjectsIsFound = true; } //compare & update userProfile with userRoleInformation @@ -246,6 +248,26 @@ module.exports = class ObservationsHelper { userProfileInformation = updatedUserProfile.data; } } + + if(overWriteUserInfoIfReferenceFromProjectsIsFound){ + + try{ + + let projectDocument = await database.models.projects.findOne({ + "_id":ObjectId(data.project._id), + }) + + if(projectDocument){ + + userRoleInformation = projectDocument.userRoleInformation; + userProfileInformation = projectDocument.userProfile + } + } + catch(err){ + + } + + } let observationData = await database.models.observations.create( From 2f880b0f3a8082a06f81a38273fea826bd98f366 Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Tue, 26 Nov 2024 11:06:17 +0530 Subject: [PATCH 2/7] bugfix completed --- controllers/v1/projectsController.js | 27 ++++ models/projects.js | 183 +++++++++++++++++++++++++++ 2 files changed, 210 insertions(+) create mode 100644 controllers/v1/projectsController.js create mode 100644 models/projects.js diff --git a/controllers/v1/projectsController.js b/controllers/v1/projectsController.js new file mode 100644 index 00000000..5f6c8047 --- /dev/null +++ b/controllers/v1/projectsController.js @@ -0,0 +1,27 @@ +/** + * name : projectsController.js + * author : Saish Borkar + * created-date : 26-nov-2024 + * Description : Projects controller + */ + +// Dependencies +const pollSubmissionsHelper = require(MODULES_BASE_PATH + "/pollSubmissions/helper"); + + +/** + * PollSubmissions + * @class +*/ +module.exports = class Projects extends Abstract { + + constructor() { + super(projectsSchema); + } + + static get name() { + return "projects"; + } + + +} diff --git a/models/projects.js b/models/projects.js new file mode 100644 index 00000000..21df83c7 --- /dev/null +++ b/models/projects.js @@ -0,0 +1,183 @@ +/** + * name : projects.js. + * author : Saish Borkar + * created-date : 26-NOV-2024. + * Description : Schema for projects. + */ + +module.exports = { + name: "projects", + schema: { + title : { + type : String, + index: true + }, + description : { + type : String, + index: true + }, + taskReport : { + type : Object, + default : {} + }, + metaInformation : { + type : Object, + default : {} + }, + userId : { + type : String, + default : "SYSTEM", + index: true + }, + userRole : { + type : String, + default : "", + index: true + }, + status : { + type : String, + default : "started", + index: true + }, + lastDownloadedAt : Date, + syncedAt : Date, + isDeleted : { + type : Boolean, + default : false, + index: true + }, + categories : { + type : Array, + default : [] + }, + createdBy : { + type : String, + default : "SYSTEM", + index: true + }, + tasks : { + type : Array, + default : [] + }, + entityInformation : { + type : Object, + default : {} + }, + programInformation : { + type : Object, + default : {} + }, + solutionInformation : { + type : Object, + default : {} + }, + updatedBy : { + type : String, + default : "SYSTEM" + }, + projectTemplateId : { + type : "ObjectId", + index: true + }, + projectTemplateExternalId : { + type : String, + index: true + }, + startDate: Date, + endDate: Date, + learningResources : { + type : Array, + default : [] + }, + entityId : { + type : String, + index : true + }, + programId : { + type : "ObjectId", + index : true + }, + programExternalId : { + type : String, + index : true + }, + solutionId : { + type : "ObjectId", + index : true + }, + solutionExternalId : { + type : String, + index : true + }, + isAPrivateProgram : { + type : Boolean, + index : true + }, + appInformation : Object, + userRoleInformation : Object, + hasAcceptedTAndC : { + type : Boolean, + default : false + }, + referenceFrom : { + type : String, + index : true + }, + submissions : Object, + link : { + type : String, + index : true + }, + taskSequence : { + type : Array, + default : [] + }, + completedDate: Date, + recommendedFor : { + type : Array, + default : [] + }, + attachments : { + type : Array, + default : [] + }, + remarks : String, + userProfile : Object, + certificate : { + templateId : "ObjectId", + osid : { + type : String, + index : true, + unique : true + }, + transactionId : { + type : String, + index : true, + unique : true + }, + templateUrl : String, + status : String, + eligible : Boolean, + message : String, + issuedOn : Date, + criteria : Object, + reIssuedAt : Date, + transactionIdCreatedAt : Date, + originalTransactionInformation :{ + transactionId : String, + osid : String + } + + } + }, + compoundIndex: [ + { + "name" :{ userId: 1, solutionId: 1 }, + "indexType" : { unique: true, partialFilterExpression: { solutionId: { $exists: true }}} + } + ] + + + +}; + From 01306e0fb3bb76788084ea039631180d335eeec3 Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Tue, 26 Nov 2024 11:07:56 +0530 Subject: [PATCH 3/7] self review 1 --- controllers/v1/projectsController.js | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/controllers/v1/projectsController.js b/controllers/v1/projectsController.js index 5f6c8047..a976ebae 100644 --- a/controllers/v1/projectsController.js +++ b/controllers/v1/projectsController.js @@ -5,12 +5,10 @@ * Description : Projects controller */ -// Dependencies -const pollSubmissionsHelper = require(MODULES_BASE_PATH + "/pollSubmissions/helper"); /** - * PollSubmissions + * Projects * @class */ module.exports = class Projects extends Abstract { From a1e7caf1db5cb97e3d318f42ea545fbb85333425 Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Tue, 26 Nov 2024 16:05:33 +0530 Subject: [PATCH 4/7] savepoint --- .../copyUserDataFromProjectsToObservations.js | 100 ++++++++++++++++++ module/observations/helper.js | 28 ++--- 2 files changed, 111 insertions(+), 17 deletions(-) create mode 100644 migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js diff --git a/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js b/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js new file mode 100644 index 00000000..ed87d380 --- /dev/null +++ b/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js @@ -0,0 +1,100 @@ +/** + * name : copyUserDataFromProjectsToObservations.js + * author : Saish Borkar + * created-date : 26-nov-2024 + * Description : Migration script for updating userProfile in observation & observationSubmission based on project information + */ + +const path = require("path"); +const fs = require("fs"); +const _ = require("lodash"); +const { MongoClient } = require("mongodb"); + +const rootPath = path.join(__dirname, "../../"); +require("dotenv").config({ path: rootPath + "/.env" }); + +const mongoUrl = process.env.MONGODB_URL; +const dbName = mongoUrl.split("/").pop(); +const url = mongoUrl.split(dbName)[0]; + +(async () => { + const connection = await MongoClient.connect(url, { + useNewUrlParser: true, + useUnifiedTopology: true, + }); + const db = connection.db(dbName); + + try { + console.log('Migration started...') + let observationIdsUpdated = []; + // Get all observations with referenceFrom "project" + const observationDocuments = await db + .collection("observations") + .find({ referenceFrom: "project" }) + .toArray(); + + if (!observationDocuments.length) { + console.log("No observations found to process."); + connection.close(); + return; + } + + const chunkedObservations = _.chunk(observationDocuments, 10); + const totalChunkLength = chunkedObservations.length; + let iteration = 1; + for (const chunk of chunkedObservations) { + console.log(`processing chunk of ${iteration++}/${totalChunkLength}`) + const projectIds = chunk.map((obs) => obs.project._id); + + // Fetch relevant projects + const projectRecords = await db + .collection("projects") + .find({ _id: { $in: projectIds } }) + .toArray(); + + for (const project of projectRecords) { + + const targetObservationIds = chunk + .filter((obs) => { + return obs.project._id.equals(project._id); + }) + .map((obs) => obs._id); + + + observationIdsUpdated.push(...targetObservationIds) + // Update observations + const updatedObservations = await db + .collection("observations") + .updateMany( + { _id: { $in: targetObservationIds } }, + { + $set: { + userRoleInformation: project.userRoleInformation, + userProfile: project.userProfile, + }, + } + ); + + // Update observationSubmissions + const updatedObservationSubmissions = await db + .collection("observationSubmissions") + .updateMany( + { observationId: { $in: targetObservationIds } }, + { + $set: { + userRoleInformation: project.userRoleInformation, + userProfile: project.userProfile, + }, + } + ); + } + } + + require('fs').writeFileSync('observationIdsUpdated'+Date.now()+'.json',JSON.stringify(observationIdsUpdated)) + console.log("Migration completed successfully."); + } catch (error) { + console.error("Migration failed:", error); + } finally { + connection.close(); + } +})(); diff --git a/module/observations/helper.js b/module/observations/helper.js index 709b06c9..f75c3a99 100644 --- a/module/observations/helper.js +++ b/module/observations/helper.js @@ -218,7 +218,6 @@ module.exports = class ObservationsHelper { return new Promise(async (resolve, reject) => { try { - let overWriteUserInfoIfReferenceFromProjectsIsFound = false; if (data.entities) { let entitiesToAdd = await entitiesHelper.validateEntities(data.entities, solution.entityType); @@ -249,25 +248,20 @@ module.exports = class ObservationsHelper { } } - if(overWriteUserInfoIfReferenceFromProjectsIsFound){ + try { + let projectDocument = await database.models.projects.findOne({ + _id: ObjectId(data.project._id), + }); - try{ - - let projectDocument = await database.models.projects.findOne({ - "_id":ObjectId(data.project._id), - }) - - if(projectDocument){ - - userRoleInformation = projectDocument.userRoleInformation; - userProfileInformation = projectDocument.userProfile - } - } - catch(err){ - - } + if (projectDocument) { + userRoleInformation = projectDocument.userRoleInformation; + userProfileInformation = projectDocument.userProfile; + } + } catch (err) { } + + let observationData = await database.models.observations.create( From d2975b6d490462a33110d9cf535cf1572d87e682 Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Tue, 26 Nov 2024 16:09:07 +0530 Subject: [PATCH 5/7] self review 2 --- module/observations/helper.js | 18 ++++++++++-------- 1 file changed, 10 insertions(+), 8 deletions(-) diff --git a/module/observations/helper.js b/module/observations/helper.js index f75c3a99..1baed668 100644 --- a/module/observations/helper.js +++ b/module/observations/helper.js @@ -227,7 +227,6 @@ module.exports = class ObservationsHelper { if( data.project ) { data.project._id = ObjectId(data.project._id); data.referenceFrom = messageConstants.common.PROJECT; - overWriteUserInfoIfReferenceFromProjectsIsFound = true; } //compare & update userProfile with userRoleInformation @@ -249,14 +248,17 @@ module.exports = class ObservationsHelper { } try { - let projectDocument = await database.models.projects.findOne({ - _id: ObjectId(data.project._id), - }); + if( data.project ){ + let projectDocument = await database.models.projects.findOne({ + _id: ObjectId(data.project._id), + }); + + if (projectDocument) { + userRoleInformation = projectDocument.userRoleInformation; + userProfileInformation = projectDocument.userProfile; + } + } - if (projectDocument) { - userRoleInformation = projectDocument.userRoleInformation; - userProfileInformation = projectDocument.userProfile; - } } catch (err) { } From 01707fd18f6c6560307856c621850e617d4fab19 Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Tue, 26 Nov 2024 18:42:44 +0530 Subject: [PATCH 6/7] validating set object --- .../copyUserDataFromProjectsToObservations.js | 35 ++++++++++++++----- 1 file changed, 26 insertions(+), 9 deletions(-) diff --git a/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js b/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js index ed87d380..d48bdf86 100644 --- a/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js +++ b/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js @@ -60,18 +60,35 @@ const url = mongoUrl.split(dbName)[0]; }) .map((obs) => obs._id); + let setObject = {}; + + if ( + project.userRoleInformation && + project.userRoleInformation.role && + project.userRoleInformation.state + ) { + setObject.userRoleInformation = project.userRoleInformation; + } + + if ( + project.userProfile && + project.userProfile.id + ) { + setObject.userProfileInformation = project.userProfile; + } + + + if (Object.keys(setObject).length === 0) { + continue; + } - observationIdsUpdated.push(...targetObservationIds) // Update observations const updatedObservations = await db .collection("observations") .updateMany( { _id: { $in: targetObservationIds } }, { - $set: { - userRoleInformation: project.userRoleInformation, - userProfile: project.userProfile, - }, + $set: setObject } ); @@ -81,12 +98,12 @@ const url = mongoUrl.split(dbName)[0]; .updateMany( { observationId: { $in: targetObservationIds } }, { - $set: { - userRoleInformation: project.userRoleInformation, - userProfile: project.userProfile, - }, + $set: setObject } ); + + observationIdsUpdated.push(...targetObservationIds) + } } From 4dfcfa2d6ac798821aea8bdab314899448e7670a Mon Sep 17 00:00:00 2001 From: "saish.borkar" Date: Tue, 26 Nov 2024 23:28:32 +0530 Subject: [PATCH 7/7] addressed comments and retested --- .../copyUserDataFromProjectsToObservations.js | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js b/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js index d48bdf86..a0befbc6 100644 --- a/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js +++ b/migrations/updateUserProfilesFromProjects/copyUserDataFromProjectsToObservations.js @@ -31,6 +31,7 @@ const url = mongoUrl.split(dbName)[0]; const observationDocuments = await db .collection("observations") .find({ referenceFrom: "project" }) + .project({ "_id": 1,"project":1}) .toArray(); if (!observationDocuments.length) { @@ -50,6 +51,7 @@ const url = mongoUrl.split(dbName)[0]; const projectRecords = await db .collection("projects") .find({ _id: { $in: projectIds } }) + .project({"_id":1,"userRoleInformation":1,"userProfile":1}) .toArray(); for (const project of projectRecords) { @@ -74,7 +76,7 @@ const url = mongoUrl.split(dbName)[0]; project.userProfile && project.userProfile.id ) { - setObject.userProfileInformation = project.userProfile; + setObject.userProfile = project.userProfile; }