Merged in feature/IO-3096-GlobalNotifications (pull request #2191)

Feature/IO-3096 GlobalNotifications
This commit is contained in:
Dave Richer
2025-03-12 16:07:09 +00:00
4 changed files with 390 additions and 657 deletions

View File

@@ -1,10 +1,13 @@
/** Notification Scenarios
* @description This file contains the scenarios for job notifications.
* @type {string[]}
*/
const notificationScenarios = [ const notificationScenarios = [
"job-assigned-to-me", "job-assigned-to-me",
"bill-posted", "bill-posted",
"critical-parts-status-changed", "critical-parts-status-changed",
"part-marked-back-ordered", "part-marked-back-ordered",
"new-note-added", "new-note-added",
"supplement-imported",
"schedule-dates-changed", "schedule-dates-changed",
"tasks-updated-created", "tasks-updated-created",
"new-media-added-reassigned", "new-media-added-reassigned",
@@ -14,6 +17,7 @@ const notificationScenarios = [
"job-status-change", "job-status-change",
"payment-collected-completed", "payment-collected-completed",
"alternate-transport-changed" "alternate-transport-changed"
// "supplement-imported", // Disabled for now
]; ];
export { notificationScenarios }; export { notificationScenarios };

File diff suppressed because it is too large Load Diff

View File

@@ -1,6 +1,6 @@
const { const {
jobAssignedToMeBuilder, jobAssignedToMeBuilder,
billPostedHandler, billPostedBuilder,
newNoteAddedBuilder, newNoteAddedBuilder,
scheduledDatesChangedBuilder, scheduledDatesChangedBuilder,
tasksUpdatedCreatedBuilder, tasksUpdatedCreatedBuilder,
@@ -30,10 +30,12 @@ const { isFunction } = require("lodash");
* - builder {Function}: A function to handle the scenario. * - builder {Function}: A function to handle the scenario.
* - onlyTruthyValues {boolean|Array<string>}: Specifies fields that must have truthy values for the scenario to match. * - onlyTruthyValues {boolean|Array<string>}: Specifies fields that must have truthy values for the scenario to match.
* - filterCallback {Function}: Optional callback (sync or async) to further filter the scenario based on event data (returns boolean). * - filterCallback {Function}: Optional callback (sync or async) to further filter the scenario based on event data (returns boolean).
* - enabled {boolean}: If true, the scenario is active; if false or omitted, the scenario is skipped.
*/ */
const notificationScenarios = [ const notificationScenarios = [
{ {
key: "job-assigned-to-me", key: "job-assigned-to-me",
enabled: true,
table: "jobs", table: "jobs",
fields: ["employee_prep", "employee_body", "employee_csr", "employee_refinish"], fields: ["employee_prep", "employee_body", "employee_csr", "employee_refinish"],
matchToUserFields: ["employee_prep", "employee_body", "employee_csr", "employee_refinish"], matchToUserFields: ["employee_prep", "employee_body", "employee_csr", "employee_refinish"],
@@ -41,24 +43,28 @@ const notificationScenarios = [
}, },
{ {
key: "bill-posted", key: "bill-posted",
enabled: true,
table: "bills", table: "bills",
builder: billPostedHandler, builder: billPostedBuilder,
onNew: true onNew: true
}, },
{ {
key: "new-note-added", key: "new-note-added",
enabled: true,
table: "notes", table: "notes",
builder: newNoteAddedBuilder, builder: newNoteAddedBuilder,
onNew: true onNew: true
}, },
{ {
key: "schedule-dates-changed", key: "schedule-dates-changed",
enabled: true,
table: "jobs", table: "jobs",
fields: ["scheduled_in", "scheduled_completion", "scheduled_delivery"], fields: ["scheduled_in", "scheduled_completion", "scheduled_delivery"],
builder: scheduledDatesChangedBuilder builder: scheduledDatesChangedBuilder
}, },
{ {
key: "tasks-updated-created", key: "tasks-updated-created",
enabled: true,
table: "tasks", table: "tasks",
fields: ["updated_at"], fields: ["updated_at"],
// onNew: true, // onNew: true,
@@ -66,12 +72,14 @@ const notificationScenarios = [
}, },
{ {
key: "job-status-change", key: "job-status-change",
enabled: true,
table: "jobs", table: "jobs",
fields: ["status"], fields: ["status"],
builder: jobStatusChangeBuilder builder: jobStatusChangeBuilder
}, },
{ {
key: "job-added-to-production", key: "job-added-to-production",
enabled: true,
table: "jobs", table: "jobs",
fields: ["inproduction"], fields: ["inproduction"],
onlyTruthyValues: ["inproduction"], onlyTruthyValues: ["inproduction"],
@@ -79,36 +87,42 @@ const notificationScenarios = [
}, },
{ {
key: "alternate-transport-changed", key: "alternate-transport-changed",
enabled: true,
table: "jobs", table: "jobs",
fields: ["alt_transport"], fields: ["alt_transport"],
builder: alternateTransportChangedBuilder builder: alternateTransportChangedBuilder
}, },
{ {
key: "new-time-ticket-posted", key: "new-time-ticket-posted",
enabled: true,
table: "timetickets", table: "timetickets",
builder: newTimeTicketPostedBuilder builder: newTimeTicketPostedBuilder
}, },
{ {
key: "intake-delivery-checklist-completed", key: "intake-delivery-checklist-completed",
enabled: true,
table: "jobs", table: "jobs",
fields: ["intakechecklist", "deliverchecklist"], fields: ["intakechecklist", "deliverchecklist"],
builder: intakeDeliveryChecklistCompletedBuilder builder: intakeDeliveryChecklistCompletedBuilder
}, },
{ {
key: "payment-collected-completed", key: "payment-collected-completed",
enabled: true,
table: "payments", table: "payments",
onNew: true, onNew: true,
builder: paymentCollectedCompletedBuilder builder: paymentCollectedCompletedBuilder
}, },
{ {
// MAKE SURE YOU ARE NOT ON A LMS ENVIRONMENT // Only works on a non LMS ENV
key: "new-media-added-reassigned", key: "new-media-added-reassigned",
enabled: true,
table: "documents", table: "documents",
fields: ["jobid"], fields: ["jobid"],
builder: newMediaAddedReassignedBuilder builder: newMediaAddedReassignedBuilder
}, },
{ {
key: "critical-parts-status-changed", key: "critical-parts-status-changed",
enabled: true,
table: "joblines", table: "joblines",
fields: ["status"], fields: ["status"],
onlyTruthyValues: ["status"], onlyTruthyValues: ["status"],
@@ -117,6 +131,7 @@ const notificationScenarios = [
}, },
{ {
key: "part-marked-back-ordered", key: "part-marked-back-ordered",
enabled: true,
table: "joblines", table: "joblines",
fields: ["status"], fields: ["status"],
builder: partMarkedBackOrderedBuilder, builder: partMarkedBackOrderedBuilder,
@@ -133,12 +148,11 @@ const notificationScenarios = [
} }
} }
}, },
// -------------- Difficult --------------- // Holding off on this one for now, spans multiple tables
// Holding off on this one for now
{ {
key: "supplement-imported", key: "supplement-imported",
enabled: false,
builder: supplementImportedBuilder builder: supplementImportedBuilder
// spans multiple tables,
} }
]; ];
@@ -159,6 +173,11 @@ const notificationScenarios = [
const getMatchingScenarios = async (eventData, getBodyshopFromRedis) => { const getMatchingScenarios = async (eventData, getBodyshopFromRedis) => {
const matches = []; const matches = [];
for (const scenario of notificationScenarios) { for (const scenario of notificationScenarios) {
// Check if the scenario is enabled; skip if not explicitly true
if (scenario.enabled !== true) {
continue;
}
// If eventData has a table, then only scenarios with a table property that matches should be considered. // If eventData has a table, then only scenarios with a table property that matches should be considered.
if (eventData.table) { if (eventData.table) {
if (!scenario.table || eventData.table.name !== scenario.table) { if (!scenario.table || eventData.table.name !== scenario.table) {

View File

@@ -35,7 +35,6 @@ const scenarioParser = async (req, jobIdField) => {
} = req; } = req;
// Step 1: Validate we know what user committed the action that fired the parser // Step 1: Validate we know what user committed the action that fired the parser
// console.log("Step 1");
const hasuraUserRole = event?.session_variables?.["x-hasura-role"]; const hasuraUserRole = event?.session_variables?.["x-hasura-role"];
const hasuraUserId = event?.session_variables?.["x-hasura-user-id"]; const hasuraUserId = event?.session_variables?.["x-hasura-user-id"];
@@ -52,7 +51,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 2: Extract just the jobId using the provided jobIdField // Step 2: Extract just the jobId using the provided jobIdField
// console.log("Step 2");
let jobId = null; let jobId = null;
if (jobIdField) { if (jobIdField) {
@@ -70,7 +68,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 3: Query job watchers associated with the job ID using GraphQL // Step 3: Query job watchers associated with the job ID using GraphQL
// console.log("Step 3");
const watcherData = await gqlClient.request(queries.GET_JOB_WATCHERS, { const watcherData = await gqlClient.request(queries.GET_JOB_WATCHERS, {
jobid: jobId jobid: jobId
@@ -96,7 +93,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 5: Perform the full event diff now that we know there are watchers // Step 5: Perform the full event diff now that we know there are watchers
// console.log("Step 5");
const eventData = await eventParser({ const eventData = await eventParser({
newData: event.data.new, newData: event.data.new,
@@ -107,7 +103,6 @@ const scenarioParser = async (req, jobIdField) => {
}); });
// Step 6: Extract body shop information from the job data // Step 6: Extract body shop information from the job data
// console.log("Step 6");
const bodyShopId = watcherData?.job?.bodyshop?.id; const bodyShopId = watcherData?.job?.bodyshop?.id;
const bodyShopName = watcherData?.job?.bodyshop?.shopname; const bodyShopName = watcherData?.job?.bodyshop?.shopname;
@@ -122,7 +117,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 7: Identify scenarios that match the event data and job context // Step 7: Identify scenarios that match the event data and job context
// console.log("Step 7");
const matchingScenarios = await getMatchingScenarios( const matchingScenarios = await getMatchingScenarios(
{ {
@@ -155,7 +149,6 @@ const scenarioParser = async (req, jobIdField) => {
}; };
// Step 8: Query notification settings for the job watchers // Step 8: Query notification settings for the job watchers
// console.log("Step 8");
const associationsData = await gqlClient.request(queries.GET_NOTIFICATION_ASSOCIATIONS, { const associationsData = await gqlClient.request(queries.GET_NOTIFICATION_ASSOCIATIONS, {
emails: jobWatchers.map((x) => x.email), emails: jobWatchers.map((x) => x.email),
@@ -173,7 +166,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 9: Filter scenario watchers based on their enabled notification methods // Step 9: Filter scenario watchers based on their enabled notification methods
// console.log("Step 9");
finalScenarioData.matchingScenarios = finalScenarioData.matchingScenarios.map((scenario) => ({ finalScenarioData.matchingScenarios = finalScenarioData.matchingScenarios.map((scenario) => ({
...scenario, ...scenario,
@@ -213,7 +205,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 10: Build and collect scenarios to dispatch notifications for // Step 10: Build and collect scenarios to dispatch notifications for
// console.log("Step 10");
const scenariosToDispatch = []; const scenariosToDispatch = [];
@@ -240,7 +231,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 11: Filter scenario fields to include only those that changed // Step 11: Filter scenario fields to include only those that changed
// console.log("Step 11");
const filteredScenarioFields = const filteredScenarioFields =
scenario.fields?.filter((field) => eventData.changedFieldNames.includes(field)) || []; scenario.fields?.filter((field) => eventData.changedFieldNames.includes(field)) || [];
@@ -274,7 +264,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 12: Dispatch email notifications to the email queue // Step 12: Dispatch email notifications to the email queue
// console.log("Step 12");
const emailsToDispatch = scenariosToDispatch.map((scenario) => scenario?.email); const emailsToDispatch = scenariosToDispatch.map((scenario) => scenario?.email);
if (!isEmpty(emailsToDispatch)) { if (!isEmpty(emailsToDispatch)) {
@@ -287,7 +276,6 @@ const scenarioParser = async (req, jobIdField) => {
} }
// Step 13: Dispatch app notifications to the app queue // Step 13: Dispatch app notifications to the app queue
// console.log("Step 13");
const appsToDispatch = scenariosToDispatch.map((scenario) => scenario?.app); const appsToDispatch = scenariosToDispatch.map((scenario) => scenario?.app);
if (!isEmpty(appsToDispatch)) { if (!isEmpty(appsToDispatch)) {