Compare commits

...

33 Commits

Author SHA1 Message Date
6885443bc2 repair resend notif wa 2026-01-08 09:32:20 +07:00
f5494fb4a1 Merge pull request 'repair: resend wa' (#29) from wisdom into main
Reviewed-on: #29
2026-01-07 05:18:12 +00:00
ed77576958 Merge pull request 'repair: resend notif wa' (#28) from wisdom into main
Reviewed-on: #28
2026-01-07 05:01:44 +00:00
9e862d1a48 Merge pull request 'repair: service & controller notif and notif user' (#27) from wisdom into main
Reviewed-on: #27
2026-01-07 03:10:33 +00:00
9913724d08 Merge pull request 'wisdom' (#26) from wisdom into main
Reviewed-on: #26
2026-01-06 11:41:36 +00:00
63a646fce3 Merge pull request 'add: self-signed certificate' (#25) from wisdom into main
Reviewed-on: #25
2026-01-06 05:38:46 +00:00
8d947a818b Merge pull request 'repair: change message wa notification' (#24) from wisdom into main
Reviewed-on: #24
2026-01-06 03:05:33 +00:00
81e07ed927 Merge pull request 'wisdom' (#23) from wisdom into main
Reviewed-on: #23
2026-01-05 07:06:16 +00:00
019c79d5bc Merge pull request 'wisdom' (#22) from wisdom into main
Reviewed-on: #22
2026-01-05 04:25:09 +00:00
fae6bb7a43 Merge pull request 'repair: notification error log created_by' (#21) from wisdom into main
Reviewed-on: #21
2025-12-31 03:20:14 +00:00
8ecb00a4d3 Merge pull request 'wisdom' (#20) from wisdom into main
Reviewed-on: #20
2025-12-23 05:19:45 +00:00
b62ca35185 Merge pull request 'wisdom' (#19) from wisdom into main
Reviewed-on: #19
2025-12-22 09:18:17 +00:00
5e74122b9e Merge pull request 'wisdom' (#18) from wisdom into main
Reviewed-on: #18
2025-11-28 05:09:58 +00:00
050543dbbf Merge pull request 'wisdom' (#17) from wisdom into main
Reviewed-on: #17
2025-11-25 03:50:08 +00:00
e1b397e1d3 Merge pull request 'add brand id' (#16) from wisdom into main
Reviewed-on: #16
2025-11-20 03:42:49 +00:00
34db6b8d89 Merge pull request 'wisdom' (#15) from wisdom into main
Reviewed-on: #15
2025-11-19 01:04:37 +00:00
5d1b6daef6 Merge pull request 'repair brand device: add brand sparepart in 2 step' (#14) from wisdom into main
Reviewed-on: #14
2025-11-18 00:25:14 +00:00
30431be379 Merge pull request 'wisdom' (#13) from wisdom into main
Reviewed-on: #13
2025-11-17 08:27:07 +00:00
361f750330 Update ecosystem.config.js 2025-11-04 06:14:58 +00:00
31f50d05ab Update .gitignore 2025-11-04 06:12:17 +00:00
961f0d6314 Update ecosystem.config.js 2025-11-04 05:50:27 +00:00
d87fc07a8e Update ecosystem.config.js 2025-11-04 05:46:55 +00:00
95e0c90a16 Merge pull request 'wisdom' (#11) from wisdom into main
Reviewed-on: #11
2025-10-28 09:46:55 +00:00
55e8a6d9ca Merge pull request 'repair: params user_schedule' (#10) from wisdom into main
Reviewed-on: #10
2025-10-28 04:48:25 +00:00
253d83357f Merge pull request 'add: tag_description in tags schema' (#9) from wisdom into main
Reviewed-on: #9
2025-10-27 04:07:05 +00:00
88a0404af0 Merge pull request 'wisdom' (#8) from wisdom into main
Reviewed-on: #8
2025-10-27 03:48:36 +00:00
d11207aedb Merge pull request 'wisdom' (#7) from wisdom into main
Reviewed-on: #7
2025-10-25 09:19:08 +00:00
d7044521bd Merge pull request 'wisdom' (#6) from wisdom into main
Reviewed-on: #6
2025-10-24 05:42:48 +00:00
e2a008c2e1 Merge pull request 'wisdom' (#5) from wisdom into main
Reviewed-on: #5
2025-10-24 03:35:12 +00:00
6d575f649a Merge pull request 'wisdom' (#4) from wisdom into main
Reviewed-on: #4
2025-10-23 04:51:40 +00:00
e8fd307a05 Merge pull request 'wisdom' (#3) from wisdom into main
Reviewed-on: #3
2025-10-22 05:27:56 +00:00
00239db472 Merge pull request 'wisdom' (#2) from wisdom into main
Reviewed-on: #2
2025-10-20 03:26:32 +00:00
251f7148b6 Merge pull request 'wisdom' (#1) from wisdom into main
Reviewed-on: #1
2025-09-17 08:40:15 +00:00
4 changed files with 82 additions and 45 deletions

1
.gitignore vendored
View File

@@ -4,3 +4,4 @@ node_modules
request.http request.http
*.rest *.rest
package-lock.json package-lock.json
*.log

View File

@@ -1,10 +1,10 @@
const { default: axios } = require('axios'); const { default: axios } = require('axios');
const CryptoJS = require('crypto-js'); const CryptoJS = require('crypto-js');
const https = require('https'); // const https = require('https');
const httpsAgent = new https.Agent({ // const httpsAgent = new https.Agent({
rejectUnauthorized: false, // rejectUnauthorized: false,
}); // });
const generateTokenRedirect = async (userPhone, userName, id) => { const generateTokenRedirect = async (userPhone, userName, id) => {
@@ -23,7 +23,7 @@ const shortUrltiny = async (encodedToken) => {
const encodedUrl = encodeURIComponent(url); // ⬅️ Encode dulu! const encodedUrl = encodeURIComponent(url); // ⬅️ Encode dulu!
const response = await axios.get(`https://tinyurl.com/api-create.php?url=${encodedUrl}`,{ httpsAgent }); const response = await axios.get(`https://tinyurl.com/api-create.php?url=${encodedUrl}`);
let shortUrl = response.data; let shortUrl = response.data;
if (!shortUrl.startsWith('http')) { if (!shortUrl.startsWith('http')) {
@@ -44,7 +44,7 @@ const sendNotifikasi = async (phone, message) => {
const endPointWhatsapp = process.env.ENDPOINT_WHATSAPP; const endPointWhatsapp = process.env.ENDPOINT_WHATSAPP;
try { try {
const response = await axios.post(endPointWhatsapp, payload, { httpsAgent }); const response = await axios.post(endPointWhatsapp, payload);
// console.log(response.data); // console.log(response.data);
return response?.data return response?.data
} catch (error) { } catch (error) {

View File

@@ -1,7 +1,7 @@
module.exports = { module.exports = {
apps: [ apps: [
{ {
name: "bengkel-api", name: "cod-api",
script: "./index.js", // Path to your entry file script: "./index.js", // Path to your entry file
env: { env: {
NODE_ENV: "development", NODE_ENV: "development",
@@ -9,6 +9,14 @@ module.exports = {
env_production: { env_production: {
NODE_ENV: "production", NODE_ENV: "production",
}, },
// Logging configuration
// error_file: "C:\IDETAMA\pm2-log\cod-api\cod-api-error.log",
// out_file: "C:\IDETAMA\pm2-log\cod-api\cod-api-out.log",
// log_file: "C:\IDETAMA\pm2-log\cod-api\cod-api-combined.log", // optional combined file
error_file: "cod-api-error.log",
out_file: "cod-api-out.log",
log_file: "cod-api-combined.log", // optional combined file
time: true, // adds timestamps to logs
}, },
], ],
}; };

View File

@@ -172,57 +172,85 @@ class NotificationService {
static async resendNotification(id) { static async resendNotification(id) {
const deviceNotification = await getNotificationByIdDb(id); const deviceNotification = await getNotificationByIdDb(id);
if (!deviceNotification) throw new ErrorHandler(404, "Data not found"); if (!deviceNotification)
throw new ErrorHandler(404, "Notification Data not found");
const errorCode = await getErrorCodeByIdDb(deviceNotification.error_code_id); const errorCode = await getErrorCodeByIdDb(
deviceNotification.error_code_id
);
const dataExist = await getUsersNotificationErrorDb(id); const dataExist = await getUsersNotificationErrorDb(id);
const activeUsers = dataExist?.filter((user) => user.is_active === true) || [];
if (activeUsers.length < 1) throw new ErrorHandler(404, "No active contacts"); const activeUsers =
dataExist?.filter((user) => user.is_active === true) || [];
this._executeResendInBackground(id, activeUsers, deviceNotification, errorCode) if (activeUsers.length < 1)
.catch(err => console.log("error:", err)); throw new ErrorHandler(404, "No active contacts");
this._executeResendWA(id, activeUsers, deviceNotification, errorCode).catch(
(err) => console.error("process error:", err)
);
return { return {
status: "success", count: activeUsers.length,
message: "Pesan sedang diproses di background",
count: activeUsers.length
}; };
} }
static async _executeResendInBackground(id, activeUsers, deviceNotification, errorCode) { static async _executeResendWA(
console.log(`Background process untuk ID: ${id}`); id,
activeUsers,
deviceNotification,
errorCode
) {
console.log(`process id: ${id}`);
for (const user of activeUsers) { const sendPromises = activeUsers.map(async (user) => {
try { try {
console.log(`Mengirim ke: ${user.contact_phone}`); const tokenRedirect = await generateTokenRedirect(
user.contact_phone,
const tokenRedirect = await generateTokenRedirect(user.contact_phone, user.contact_name, id); user.contact_name,
id
);
const encodedToken = encodeURIComponent(tokenRedirect); const encodedToken = encodeURIComponent(tokenRedirect);
const shortUrl = await shortUrltiny(encodedToken); const shortUrl = await shortUrltiny(encodedToken);
console.log(`Link: ${shortUrl}`);
const bodyWithUrl = `Hai ${user.contact_name || "-"}\n` + const bodyWithUrl =
`Terjadi peringatan dengan kode ${errorCode?.error_code || "-"} - ${errorCode?.error_code_name} pada device ${deviceNotification.device_name || "-"}.\n` + `Hai ${user.contact_name || "-"}\n` +
`silahkan cek detail pada link berikut:\n ${shortUrl}`; `Terjadi peringatan dengan kode ${errorCode?.error_code || "-"} - ${
errorCode?.error_code_name
} pada device ${deviceNotification.device_name || "-"}.\n` +
`Silahkan cek detail pada link berikut:\n ${shortUrl}`;
const resultSend = await sendNotifikasi(user.contact_phone, bodyWithUrl); const resultSend = await sendNotifikasi(
user.contact_phone,
bodyWithUrl
);
console.log(`WHATSAPP API Respon:`, JSON.stringify(resultSend)); const isSuccess = resultSend?.error ? false : true;
const isSuccess = !resultSend?.error;
await updateNotificationErrorDb(user.notification_error_id, { await updateNotificationErrorDb(user.notification_error_id, {
is_send: isSuccess, is_send: isSuccess,
is_delivered: isSuccess, is_delivered: isSuccess,
}); });
return { phone: user.contact_phone, status: true };
} catch (err) { } catch (err) {
console.log(`error pada ${user.contact_phone}:`, err.message);
console.log(`Gagal mengirim ke ${user.contact_phone}:`, err.message);
return {
phone: user.contact_phone,
status: "failed",
error: err.message,
};
} }
} });
console.log(` pesan untuk ID: ${id} selesai diproses.`);
const results = await Promise.all(sendPromises);
console.log(
`Notification resend has been succesfully: ${
results.filter((r) => r.status === "success").length
}, Error: ${results.filter((r) => r.status === "failed").length}`
);
} }
} }