mirror of
				https://github.com/louislam/uptime-kuma.git
				synced 2025-10-25 15:59:20 +08:00 
			
		
		
		
	Merge remote-tracking branch 'louislam/master' into feature/add-support-for-method-body-and-headers
This commit is contained in:
		
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/ask-for-help.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/ask-for-help.md
									
									
									
									
										vendored
									
									
								
							| @@ -9,6 +9,10 @@ assignees: '' | ||||
| **Is it a duplicate question?** | ||||
| Please search in Issues without filters: https://github.com/louislam/uptime-kuma/issues?q= | ||||
|  | ||||
|  | ||||
| **Describe your problem** | ||||
|  | ||||
|  | ||||
| **Info** | ||||
| Uptime Kuma Version: | ||||
| Using Docker?: Yes/No | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,7 @@ jobs: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [macos-latest, ubuntu-latest, windows-latest] | ||||
|         node-version: [14.x, 15.x, 16.x] | ||||
|         node-version: [14.x, 16.x] | ||||
|         # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ | ||||
|  | ||||
|     steps: | ||||
| @@ -27,7 +27,7 @@ jobs: | ||||
|       with: | ||||
|         node-version: ${{ matrix.node-version }} | ||||
|         cache: 'npm' | ||||
|     - run: npm ci | ||||
|     - run: npm run install-legacy | ||||
|     - run: npm run build | ||||
|     - run: npm test | ||||
|       env: | ||||
|   | ||||
| @@ -92,7 +92,7 @@ Light Mode: | ||||
|  | ||||
| Status Page: | ||||
|  | ||||
| <img src="https://user-images.githubusercontent.com/1336778/133384019-962e1120-6c3a-481f-9d07-d7df765e9ba4.png" width="512" alt="" /> | ||||
| <img src="https://user-images.githubusercontent.com/1336778/134628766-a3fe0981-0926-4285-ab46-891a21c3e4cb.png" width="512" alt="" /> | ||||
|  | ||||
| Settings Page: | ||||
|  | ||||
|   | ||||
							
								
								
									
										7
									
								
								babel.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								babel.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| const config = {}; | ||||
|  | ||||
| if (process.env.TEST_FRONTEND) { | ||||
|     config.presets = ["@babel/preset-env"]; | ||||
| } | ||||
|  | ||||
| module.exports = config; | ||||
							
								
								
									
										5
									
								
								jest-frontend.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								jest-frontend.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| module.exports = { | ||||
|     "rootDir": ".", | ||||
|     "testRegex": "./test/frontend.spec.js", | ||||
| }; | ||||
|  | ||||
							
								
								
									
										11
									
								
								jest.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								jest.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| module.exports = { | ||||
|     "verbose": true, | ||||
|     "preset": "jest-puppeteer", | ||||
|     "globals": { | ||||
|         "__DEV__": true | ||||
|     }, | ||||
|     "testRegex": "./test/e2e.spec.js", | ||||
|     "rootDir": ".", | ||||
|     "testTimeout": 30000, | ||||
| }; | ||||
|  | ||||
							
								
								
									
										2281
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2281
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										14
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								package.json
									
									
									
									
									
								
							| @@ -22,7 +22,8 @@ | ||||
|         "build": "vite build", | ||||
|         "test": "node test/prepare-test-server.js && node server/server.js --port=3002 --data-dir=./data/test/ --test", | ||||
|         "test-with-build": "npm run build && npm test", | ||||
|         "jest": "node test/prepare-jest.js && jest", | ||||
|         "jest": "node test/prepare-jest.js && npm run jest-frontend && jest ", | ||||
|         "jest-frontend": "cross-env TEST_FRONTEND=1 jest --config=./jest-frontend.config.js", | ||||
|         "tsc": "tsc", | ||||
|         "vite-preview-dist": "vite preview --host", | ||||
|         "build-docker": "npm run build-docker-debian && npm run build-docker-alpine", | ||||
| @@ -98,6 +99,7 @@ | ||||
|     }, | ||||
|     "devDependencies": { | ||||
|         "@babel/eslint-parser": "~7.15.7", | ||||
|         "@babel/preset-env": "^7.15.8", | ||||
|         "@types/bootstrap": "~5.1.6", | ||||
|         "@vitejs/plugin-legacy": "~1.6.1", | ||||
|         "@vitejs/plugin-vue": "~1.9.2", | ||||
| @@ -115,15 +117,5 @@ | ||||
|         "stylelint-config-standard": "~22.0.0", | ||||
|         "typescript": "~4.4.3", | ||||
|         "vite": "~2.6.4" | ||||
|     }, | ||||
|     "jest": { | ||||
|         "verbose": true, | ||||
|         "preset": "jest-puppeteer", | ||||
|         "globals": { | ||||
|             "__DEV__": true | ||||
|         }, | ||||
|         "testRegex": "./test/*.spec.js", | ||||
|         "rootDir": ".", | ||||
|         "testTimeout": 30000 | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| const { setSetting } = require("./util-server"); | ||||
| const axios = require("axios"); | ||||
| const { isDev } = require("../src/util"); | ||||
|  | ||||
| exports.version = require("../package.json").version; | ||||
| exports.latestVersion = null; | ||||
|   | ||||
| @@ -4,6 +4,8 @@ | ||||
| const { TimeLogger } = require("../src/util"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { io } = require("./server"); | ||||
| const { setting } = require("./util-server"); | ||||
| const checkVersion = require("./check-version"); | ||||
|  | ||||
| async function sendNotificationList(socket) { | ||||
|     const timeLogger = new TimeLogger(); | ||||
| @@ -14,10 +16,10 @@ async function sendNotificationList(socket) { | ||||
|     ]); | ||||
|  | ||||
|     for (let bean of list) { | ||||
|         result.push(bean.export()) | ||||
|         result.push(bean.export()); | ||||
|     } | ||||
|  | ||||
|     io.to(socket.userID).emit("notificationList", result) | ||||
|     io.to(socket.userID).emit("notificationList", result); | ||||
|  | ||||
|     timeLogger.print("Send Notification List"); | ||||
|  | ||||
| @@ -39,7 +41,7 @@ async function sendHeartbeatList(socket, monitorID, toUser = false, overwrite = | ||||
|         LIMIT 100 | ||||
|     `, [ | ||||
|         monitorID, | ||||
|     ]) | ||||
|     ]); | ||||
|  | ||||
|     let result = list.reverse(); | ||||
|  | ||||
| @@ -69,7 +71,7 @@ async function sendImportantHeartbeatList(socket, monitorID, toUser = false, ove | ||||
|         LIMIT 500 | ||||
|     `, [ | ||||
|         monitorID, | ||||
|     ]) | ||||
|     ]); | ||||
|  | ||||
|     timeLogger.print(`[Monitor: ${monitorID}] sendImportantHeartbeatList`); | ||||
|  | ||||
| @@ -81,8 +83,18 @@ async function sendImportantHeartbeatList(socket, monitorID, toUser = false, ove | ||||
|  | ||||
| } | ||||
|  | ||||
| async function sendInfo(socket) { | ||||
|     socket.emit("info", { | ||||
|         version: checkVersion.version, | ||||
|         latestVersion: checkVersion.latestVersion, | ||||
|         primaryBaseURL: await setting("primaryBaseURL") | ||||
|     }); | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     sendNotificationList, | ||||
|     sendImportantHeartbeatList, | ||||
|     sendHeartbeatList, | ||||
| } | ||||
|     sendInfo | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ class Discord extends NotificationProvider { | ||||
|     name = "discord"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             const discordDisplayName = notification.discordUsername || "Uptime Kuma"; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ class Gotify extends NotificationProvider { | ||||
|     name = "gotify"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             if (notification.gotifyserverurl && notification.gotifyserverurl.endsWith("/")) { | ||||
|                 notification.gotifyserverurl = notification.gotifyserverurl.slice(0, -1); | ||||
|   | ||||
| @@ -7,7 +7,7 @@ class Line extends NotificationProvider { | ||||
|     name = "line"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             let lineAPIUrl = "https://api.line.me/v2/bot/message/push"; | ||||
|             let config = { | ||||
|   | ||||
| @@ -7,7 +7,7 @@ class LunaSea extends NotificationProvider { | ||||
|     name = "lunasea"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         let lunaseadevice = "https://notify.lunasea.app/v1/custom/device/" + notification.lunaseaDevice | ||||
|  | ||||
|         try { | ||||
|   | ||||
							
								
								
									
										45
									
								
								server/notification-providers/matrix.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								server/notification-providers/matrix.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const Crypto = require("crypto"); | ||||
| const { debug } = require("../../src/util"); | ||||
|  | ||||
| class Matrix extends NotificationProvider { | ||||
|     name = "matrix"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         const size = 20; | ||||
|         const randomString = encodeURIComponent( | ||||
|             Crypto | ||||
|                 .randomBytes(size) | ||||
|                 .toString("base64") | ||||
|                 .slice(0, size) | ||||
|         ); | ||||
|  | ||||
|         debug("Random String: " + randomString); | ||||
|  | ||||
|         const roomId = encodeURIComponent(notification.internalRoomId); | ||||
|  | ||||
|         debug("Matrix Room ID: " + roomId); | ||||
|  | ||||
|         try { | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "Authorization": `Bearer ${notification.accessToken}`, | ||||
|                 } | ||||
|             }; | ||||
|             let data = { | ||||
|                 "msgtype": "m.text", | ||||
|                 "body": msg | ||||
|             }; | ||||
|  | ||||
|             await axios.put(`${notification.homeserverUrl}/_matrix/client/r0/rooms/${roomId}/send/m.room.message/${randomString}`, data, config); | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Matrix; | ||||
| @@ -7,7 +7,7 @@ class Mattermost extends NotificationProvider { | ||||
|     name = "mattermost"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             const mattermostUserName = notification.mattermostusername || "Uptime Kuma"; | ||||
|             // If heartbeatJSON is null, assume we're testing. | ||||
|   | ||||
| @@ -6,30 +6,54 @@ class Octopush extends NotificationProvider { | ||||
|     name = "octopush"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "api-key": notification.octopushAPIKey, | ||||
|                     "api-login": notification.octopushLogin, | ||||
|                     "cache-control": "no-cache" | ||||
|                 } | ||||
|             }; | ||||
|             let data = { | ||||
|                 "recipients": [ | ||||
|                     { | ||||
|                         "phone_number": notification.octopushPhoneNumber | ||||
|         // Default - V2 | ||||
|             if (notification.octopushVersion == 2 || !notification.octopushVersion) { | ||||
|                 let config = { | ||||
|                     headers: { | ||||
|                         "api-key": notification.octopushAPIKey, | ||||
|                         "api-login": notification.octopushLogin, | ||||
|                         "cache-control": "no-cache" | ||||
|                     } | ||||
|                 ], | ||||
|                 //octopush not supporting non ascii char | ||||
|                 "text": msg.replace(/[^\x00-\x7F]/g, ""), | ||||
|                 "type": notification.octopushSMSType, | ||||
|                 "purpose": "alert", | ||||
|                 "sender": notification.octopushSenderName | ||||
|             }; | ||||
|                 }; | ||||
|                 let data = { | ||||
|                     "recipients": [ | ||||
|                         { | ||||
|                             "phone_number": notification.octopushPhoneNumber | ||||
|                         } | ||||
|                     ], | ||||
|                     //octopush not supporting non ascii char | ||||
|                     "text": msg.replace(/[^\x00-\x7F]/g, ""), | ||||
|                     "type": notification.octopushSMSType, | ||||
|                     "purpose": "alert", | ||||
|                     "sender": notification.octopushSenderName | ||||
|                 }; | ||||
|                 await axios.post("https://api.octopush.com/v1/public/sms-campaign/send", data, config) | ||||
|             } else if (notification.octopushVersion == 1) { | ||||
|                 let data = { | ||||
|                     "user_login": notification.octopushDMLogin, | ||||
|                     "api_key": notification.octopushDMAPIKey, | ||||
|                     "sms_recipients": notification.octopushDMPhoneNumber, | ||||
|                     "sms_sender": notification.octopushDMSenderName, | ||||
|                     "sms_type": (notification.octopushDMSMSType == "sms_premium") ? "FR" : "XXX", | ||||
|                     "transactional": "1", | ||||
|                     //octopush not supporting non ascii char | ||||
|                     "sms_text": msg.replace(/[^\x00-\x7F]/g, ""), | ||||
|                 }; | ||||
|  | ||||
|                 let config = { | ||||
|                     headers: { | ||||
|                         "cache-control": "no-cache" | ||||
|                     }, | ||||
|                     params: data | ||||
|                 }; | ||||
|                 await axios.post("https://www.octopush-dm.com/api/sms/json", {}, config) | ||||
|             } else { | ||||
|                 throw new Error("Unknown Octopush version!"); | ||||
|             } | ||||
|  | ||||
|             await axios.post("https://api.octopush.com/v1/public/sms-campaign/send", data, config) | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|   | ||||
							
								
								
									
										41
									
								
								server/notification-providers/promosms.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								server/notification-providers/promosms.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class PromoSMS extends NotificationProvider { | ||||
|  | ||||
|     name = "promosms"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "Content-Type": "application/json", | ||||
|                     "Authorization": "Basic " + Buffer.from(notification.promosmsLogin + ":" + notification.promosmsPassword).toString('base64'), | ||||
|                     "Accept": "text/json", | ||||
|                 } | ||||
|             }; | ||||
|             let data = { | ||||
|                 "recipients": [ notification.promosmsPhoneNumber ], | ||||
|                 //Lets remove non ascii char | ||||
|                 "text": msg.replace(/[^\x00-\x7F]/g, ""), | ||||
|                 "type": Number(notification.promosmsSMSType), | ||||
|                 "sender": notification.promosmsSenderName | ||||
|             }; | ||||
|  | ||||
|             let resp = await axios.post("https://promosms.com/api/rest/v3_2/sms", data, config); | ||||
|  | ||||
|             if (resp.data.response.status !== 0) { | ||||
|                 let error = "Something gone wrong. Api returned " + resp.data.response.status + "."; | ||||
|                 this.throwGeneralAxiosError(error); | ||||
|             } | ||||
|              | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = PromoSMS; | ||||
| @@ -8,7 +8,7 @@ class Pushbullet extends NotificationProvider { | ||||
|     name = "pushbullet"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let pushbulletUrl = "https://api.pushbullet.com/v2/pushes"; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ class Pushover extends NotificationProvider { | ||||
|     name = "pushover"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         let pushoverlink = "https://api.pushover.net/1/messages.json" | ||||
|  | ||||
|         try { | ||||
|   | ||||
| @@ -6,7 +6,7 @@ class Pushy extends NotificationProvider { | ||||
|     name = "pushy"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             await axios.post(`https://api.pushy.me/push?api_key=${notification.pushyAPIKey}`, { | ||||
|   | ||||
| @@ -9,7 +9,7 @@ class RocketChat extends NotificationProvider { | ||||
|     name = "rocket.chat"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let data = { | ||||
|   | ||||
| @@ -6,7 +6,7 @@ class Signal extends NotificationProvider { | ||||
|     name = "signal"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let data = { | ||||
|   | ||||
| @@ -25,7 +25,7 @@ class Slack extends NotificationProvider { | ||||
|     } | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let data = { | ||||
|   | ||||
| @@ -87,7 +87,7 @@ class Teams extends NotificationProvider { | ||||
|     }; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             if (heartbeatJSON == null) { | ||||
|   | ||||
| @@ -6,7 +6,7 @@ class Telegram extends NotificationProvider { | ||||
|     name = "telegram"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             await axios.get(`https://api.telegram.org/bot${notification.telegramBotToken}/sendMessage`, { | ||||
|   | ||||
| @@ -7,7 +7,7 @@ class Webhook extends NotificationProvider { | ||||
|     name = "webhook"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully. "; | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let data = { | ||||
|   | ||||
| @@ -5,7 +5,9 @@ const Gotify = require("./notification-providers/gotify"); | ||||
| const Line = require("./notification-providers/line"); | ||||
| const LunaSea = require("./notification-providers/lunasea"); | ||||
| const Mattermost = require("./notification-providers/mattermost"); | ||||
| const Matrix = require("./notification-providers/matrix"); | ||||
| const Octopush = require("./notification-providers/octopush"); | ||||
| const PromoSMS = require("./notification-providers/promosms"); | ||||
| const Pushbullet = require("./notification-providers/pushbullet"); | ||||
| const Pushover = require("./notification-providers/pushover"); | ||||
| const Pushy = require("./notification-providers/pushy"); | ||||
| @@ -34,7 +36,9 @@ class Notification { | ||||
|             new Line(), | ||||
|             new LunaSea(), | ||||
|             new Mattermost(), | ||||
|             new Matrix(), | ||||
|             new Octopush(), | ||||
|             new PromoSMS(), | ||||
|             new Pushbullet(), | ||||
|             new Pushover(), | ||||
|             new Pushy(), | ||||
|   | ||||
| @@ -90,7 +90,7 @@ const io = new Server(server); | ||||
| module.exports.io = io; | ||||
|  | ||||
| // Must be after io instantiation | ||||
| const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList } = require("./client"); | ||||
| const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList, sendInfo } = require("./client"); | ||||
| const { statusPageSocketHandler } = require("./socket-handlers/status-page-socket-handler"); | ||||
|  | ||||
| app.use(express.json()); | ||||
| @@ -180,10 +180,7 @@ exports.entryPage = "dashboard"; | ||||
|     console.log("Adding socket handler"); | ||||
|     io.on("connection", async (socket) => { | ||||
|  | ||||
|         socket.emit("info", { | ||||
|             version: checkVersion.version, | ||||
|             latestVersion: checkVersion.latestVersion, | ||||
|         }); | ||||
|         sendInfo(socket); | ||||
|  | ||||
|         totalClient++; | ||||
|  | ||||
| @@ -873,6 +870,8 @@ exports.entryPage = "dashboard"; | ||||
|                     msg: "Saved" | ||||
|                 }); | ||||
|  | ||||
|                 sendInfo(socket); | ||||
|  | ||||
|             } catch (e) { | ||||
|                 callback({ | ||||
|                     ok: false, | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
| @import "node_modules/bootstrap/scss/bootstrap"; | ||||
|  | ||||
| #app { | ||||
|     font-family: ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, segoe ui, Roboto, helvetica neue, Arial, noto sans, sans-serif, apple color emoji, segoe ui emoji, segoe ui symbol, noto color emoji; | ||||
|     font-family: BlinkMacSystemFont, segoe ui, Roboto, helvetica neue, Arial, noto sans, sans-serif, apple color emoji, segoe ui emoji, segoe ui symbol, noto color emoji; | ||||
| } | ||||
|  | ||||
| h1 { | ||||
|   | ||||
| @@ -11,18 +11,18 @@ export default { | ||||
|     computed: { | ||||
|         color() { | ||||
|             if (this.status === 0) { | ||||
|                 return "danger" | ||||
|                 return "danger"; | ||||
|             } | ||||
|  | ||||
|             if (this.status === 1) { | ||||
|                 return "primary" | ||||
|                 return "primary"; | ||||
|             } | ||||
|  | ||||
|             if (this.status === 2) { | ||||
|                 return "warning" | ||||
|                 return "warning"; | ||||
|             } | ||||
|  | ||||
|             return "secondary" | ||||
|             return "secondary"; | ||||
|         }, | ||||
|  | ||||
|         text() { | ||||
| @@ -41,11 +41,11 @@ export default { | ||||
|             return this.$t("Unknown"); | ||||
|         }, | ||||
|     }, | ||||
| } | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style scoped> | ||||
|     span { | ||||
|         width: 64px; | ||||
|         min-width: 64px; | ||||
|     } | ||||
| </style> | ||||
|   | ||||
							
								
								
									
										34
									
								
								src/components/notifications/Matrix.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								src/components/notifications/Matrix.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="homeserver-url" class="form-label">Homeserver URL (with http(s):// and optionally port)</label><span style="color: red;"><sup>*</sup></span> | ||||
|         <input id="homeserver-url" v-model="$parent.notification.homeserverUrl" type="text" class="form-control" :required="true"> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="internal-room-id" class="form-label">Internal Room Id</label><span style="color: red;"><sup>*</sup></span> | ||||
|         <input id="internal-room-id" v-model="$parent.notification.internalRoomId" type="text" class="form-control" required="true"> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="access-token" class="form-label">Access Token</label><span style="color: red;"><sup>*</sup></span> | ||||
|         <HiddenInput id="access-token" v-model="$parent.notification.accessToken" :required="true" autocomplete="one-time-code" :maxlength="500"></HiddenInput> | ||||
|     </div> | ||||
|  | ||||
|     <div class="form-text"> | ||||
|         <span style="color: red;"><sup>*</sup></span>Required | ||||
|         <p style="margin-top: 8px;"> | ||||
|             You can find the internal room ID by looking in the advanced section of the room settings in your Matrix client. It should look like !QMdRCpUIfLwsfjxye6:home.server. | ||||
|         </p> | ||||
|         <p style="margin-top: 8px;"> | ||||
|             It is highly recommended you create a new user and do not use your own Matrix user's access token as it will allow full access to your account and all the rooms you joined. Instead, create a new user and only invite it to the room that you want to receive the notification in. You can get the access token by running <code>curl -XPOST -d '{"type": "m.login.password", "identifier": {"user": "botusername", "type": "m.id.user"}, "password": "passwordforuser"}' "https://home.server/_matrix/client/r0/login"</code>. | ||||
|         </p> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import HiddenInput from "../HiddenInput.vue"; | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         HiddenInput, | ||||
|     }, | ||||
| } | ||||
| </script> | ||||
| @@ -1,4 +1,14 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="octopush-version" class="form-label">Octopush API Version</label> | ||||
|         <select id="octopush-version" v-model="$parent.notification.octopushVersion" class="form-select"> | ||||
|             <option value="2">Octopush (endpoint: api.octopush.com)</option> | ||||
|             <option value="1">Legacy Octopush-DM (endpoint: www.octopush-dm.com)</option> | ||||
|         </select> | ||||
|         <div class="form-text"> | ||||
|             Do you use the legacy version of Octopush (2011-2020) or the new version? | ||||
|         </div> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="octopush-key" class="form-label">API KEY</label> | ||||
|         <HiddenInput id="octopush-key" v-model="$parent.notification.octopushAPIKey" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|   | ||||
							
								
								
									
										39
									
								
								src/components/notifications/PromoSMS.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/components/notifications/PromoSMS.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="promosms-login" class="form-label">API LOGIN</label> | ||||
|         <input id="promosms-login" v-model="$parent.notification.promosmsLogin" type="text" class="form-control" required> | ||||
|         <label for="promosms-key" class="form-label">API PASSWORD</label> | ||||
|         <HiddenInput id="promosms-key" v-model="$parent.notification.promosmsPassword" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="promosms-type-sms" class="form-label">{{ $t("SMS Type") }}</label> | ||||
|         <select id="promosms-type-sms" v-model="$parent.notification.promosmsSMSType" class="form-select"> | ||||
|             <option value="0">{{ $t("promosmsTypeFlash") }}</option> | ||||
|             <option value="1">{{ $t("promosmsTypeEco") }}</option> | ||||
|             <option value="2">{{ $t("promosmsTypeFull") }}</option> | ||||
|             <option value="3">{{ $t("promosmsTypeSpeed") }}</option> | ||||
|         </select> | ||||
|         <i18n-t tag="div" keypath="Check PromoSMS prices" class="form-text"> | ||||
|             <a href="https://promosms.com/cennik/" target="_blank">https://promosms.com/cennik/</a> | ||||
|         </i18n-t> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="promosms-phone-number" class="form-label">{{ $t("promosmsPhoneNumber") }}</label> | ||||
|         <input id="promosms-phone-number" v-model="$parent.notification.promosmsPhoneNumber" type="text" class="form-control" required> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="promosms-sender-name" class="form-label">{{ $t("promosmsSMSSender") }}</label> | ||||
|         <input id="promosms-sender-name" v-model="$parent.notification.promosmsSenderName" type="text" minlength="3" maxlength="11" class="form-control"> | ||||
|     </div> | ||||
|  | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import HiddenInput from "../HiddenInput.vue"; | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         HiddenInput, | ||||
|     }, | ||||
| }; | ||||
| </script> | ||||
| @@ -10,11 +10,13 @@ import Teams from "./Teams.vue"; | ||||
| import Pushover from "./Pushover.vue"; | ||||
| import Pushy from "./Pushy.vue"; | ||||
| import Octopush from "./Octopush.vue"; | ||||
| import PromoSMS from "./PromoSMS.vue"; | ||||
| import LunaSea from "./LunaSea.vue"; | ||||
| import Apprise from "./Apprise.vue"; | ||||
| import Pushbullet from "./Pushbullet.vue"; | ||||
| import Line from "./Line.vue"; | ||||
| import Mattermost from "./Mattermost.vue"; | ||||
| import Matrix from "./Matrix.vue"; | ||||
|  | ||||
| /** | ||||
|  * Manage all notification form. | ||||
| @@ -34,11 +36,13 @@ const NotificationFormList = { | ||||
|     "pushover": Pushover, | ||||
|     "pushy": Pushy, | ||||
|     "octopush": Octopush, | ||||
|     "promosms": PromoSMS, | ||||
|     "lunasea": LunaSea, | ||||
|     "apprise": Apprise, | ||||
|     "pushbullet": Pushbullet, | ||||
|     "line": Line, | ||||
|     "mattermost": Mattermost | ||||
|     "mattermost": Mattermost, | ||||
|     "matrix": Matrix, | ||||
| } | ||||
|  | ||||
| export default NotificationFormList | ||||
|   | ||||
| @@ -49,7 +49,10 @@ const languageList = { | ||||
|  | ||||
| const rtlLangs = ["fa"]; | ||||
|  | ||||
| export const currentLocale = () => localStorage.locale || "en"; | ||||
| export const currentLocale = () => localStorage.locale | ||||
|     || languageList[navigator.language] && navigator.language | ||||
|     || languageList[navigator.language.substring(0, 2)] && navigator.language.substring(0, 2) | ||||
|     || "en"; | ||||
|  | ||||
| export const localeDirection = () => { | ||||
|     return rtlLangs.includes(currentLocale()) ? "rtl" : "ltr"; | ||||
|   | ||||
| @@ -190,6 +190,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Поддържа 50+ услуги за инвестяване)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -191,6 +191,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -235,11 +235,51 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|     line: "Line Messenger", | ||||
|     mattermost: "Mattermost", | ||||
|     "User Key": "User Key", | ||||
|     "Device": "Device", | ||||
|     "Message Title": "Message Title", | ||||
|     "Notification Sound": "Notification Sound", | ||||
|     "More info on:": "More info on: {0}", | ||||
|     pushoverDesc1: "Emergency priority (2) has default 30 second timeout between retries and will expire after 1 hour.", | ||||
|     pushoverDesc2: "If you want to send notifications to different devices, fill out Device field.", | ||||
|     "SMS Type": "SMS Type", | ||||
|     octopushTypePremium: "Premium (Fast - recommended for alerting)", | ||||
|     octopushTypeLowCost: "Low Cost (Slow, sometimes blocked by operator)", | ||||
|     "Check octopush prices": "Check octopush prices {0}.", | ||||
|     octopushPhoneNumber: "Phone number (intl format, eg : +33612345678) ", | ||||
|     octopushSMSSender: "SMS Sender Name : 3-11 alphanumeric characters and space (a-zA-Z0-9)", | ||||
|     "LunaSea Device ID": "LunaSea Device ID", | ||||
|     "Apprise URL": "Apprise URL", | ||||
|     "Example:": "Example: {0}", | ||||
|     "Read more:": "Read more: {0}", | ||||
|     "Status:": "Status: {0}", | ||||
|     "Read more": "Read more", | ||||
|     appriseInstalled: "Apprise is installed.", | ||||
|     appriseNotInstalled: "Apprise is not installed. {0}", | ||||
|     "Access Token": "Access Token", | ||||
|     "Channel access token": "Channel access token", | ||||
|     "Line Developers Console": "Line Developers Console", | ||||
|     lineDevConsoleTo: "Line Developers Console - {0}", | ||||
|     "Basic Settings": "Basic Settings", | ||||
|     "User ID": "User ID", | ||||
|     "Messaging API": "Messaging API", | ||||
|     wayToGetLineChannelToken: "First access the {0}, create a provider and channel (Messaging API), then you can get the channel access token and user id from the above mentioned menu items.", | ||||
|     "Icon URL": "Icon URL", | ||||
|     aboutIconURL: "You can provide a link to a picture in \"Icon URL\" to override the default profile picture. Will not be used if Icon Emoji is set.", | ||||
|     aboutMattermostChannelName: "You can override the default channel that webhook posts to by entering the channel name into \"Channel Name\" field. This needs to be enabled in Mattermost webhook settings. Ex: #other-channel", | ||||
|     "matrix": "Matrix", | ||||
|     promosmsTypeEco: "SMS ECO - cheap but slow and often overloaded. Limited only to Polish recipients.", | ||||
|     promosmsTypeFlash: "SMS FLASH - Message will automatically show on recipient device. Limited only to Polish recipients.", | ||||
|     promosmsTypeFull: "SMS FULL - Premium tier of SMS, You can use Your Sender Name (You need to register name first). Reliable for alerts.", | ||||
|     promosmsTypeFull: "SMS SPEED - Highest priority in system. Very quick and reliable but costly (about twice of SMS FULL price).", | ||||
|     promosmsPhoneNumber: "Phone number (for Polish recipient You can skip area codes)", | ||||
|     promosmsSMSSender: "SMS Sender Name : Pre-registred name or one of defaults: InfoSMS, SMS Info, MaxSMS, INFO, SMS", | ||||
|     // End notification form | ||||
|     "Status Page": "Status Page", | ||||
|     Method: "Method", | ||||
|   | ||||
| @@ -17,7 +17,7 @@ export default { | ||||
|     pauseMonitorMsg: "¿Seguro que quieres pausar?", | ||||
|     Settings: "Ajustes", | ||||
|     Dashboard: "Panel", | ||||
|     "New Update": "Vueva actualización", | ||||
|     "New Update": "Nueva actualización", | ||||
|     Language: "Idioma", | ||||
|     Appearance: "Apariencia", | ||||
|     Theme: "Tema", | ||||
| @@ -61,11 +61,11 @@ export default { | ||||
|     Retries: "Reintentos", | ||||
|     Advanced: "Avanzado", | ||||
|     "Upside Down Mode": "Modo invertido", | ||||
|     "Max. Redirects": "Máx. redirecciones", | ||||
|     "Max. Redirects": "Redirecciones Máximas", | ||||
|     "Accepted Status Codes": "Códigos de estado aceptados", | ||||
|     Save: "Guardar", | ||||
|     Notifications: "Notificaciones", | ||||
|     "Not available, please setup.": "No disponible, por favor configurar.", | ||||
|     "Not available, please setup.": "No disponible, por favor configúrelo.", | ||||
|     "Setup Notification": "Configurar notificación", | ||||
|     Light: "Claro", | ||||
|     Dark: "Oscuro", | ||||
| @@ -83,11 +83,11 @@ export default { | ||||
|     "New Password": "Nueva contraseña", | ||||
|     "Repeat New Password": "Repetir nueva contraseña", | ||||
|     "Update Password": "Actualizar contraseña", | ||||
|     "Disable Auth": "Deshabilitar Autenticación ", | ||||
|     "Enable Auth": "Habilitar Autenticación ", | ||||
|     "Disable Auth": "Deshabilitar Autenticación", | ||||
|     "Enable Auth": "Habilitar Autenticación", | ||||
|     Logout: "Cerrar sesión", | ||||
|     Leave: "Salir", | ||||
|     "I understand, please disable": "Lo comprendo, por favor deshabilitar", | ||||
|     "I understand, please disable": "Entiendo, por favor deshabilitar", | ||||
|     Confirm: "Confirmar", | ||||
|     Yes: "Sí", | ||||
|     No: "No", | ||||
| @@ -100,7 +100,7 @@ export default { | ||||
|     "Notification Type": "Tipo de notificación", | ||||
|     Email: "Email", | ||||
|     Test: "Test", | ||||
|     "Certificate Info": "Información del certificado ", | ||||
|     "Certificate Info": "Información del certificado", | ||||
|     "Resolver Server": "Servidor de resolución", | ||||
|     "Resource Record Type": "Tipo de Registro", | ||||
|     "Last Result": "Último resultado", | ||||
| @@ -108,78 +108,78 @@ export default { | ||||
|     "Repeat Password": "Repetir contraseña", | ||||
|     respTime: "Tiempo de resp. (ms)", | ||||
|     notAvailableShort: "N/A", | ||||
|     Create: "Create", | ||||
|     clearEventsMsg: "Are you sure want to delete all events for this monitor?", | ||||
|     clearHeartbeatsMsg: "Are you sure want to delete all heartbeats for this monitor?", | ||||
|     confirmClearStatisticsMsg: "Are you sure want to delete ALL statistics?", | ||||
|     "Clear Data": "Clear Data", | ||||
|     Events: "Events", | ||||
|     Heartbeats: "Heartbeats", | ||||
|     "Auto Get": "Auto Get", | ||||
|     enableDefaultNotificationDescription: "For every new monitor this notification will be enabled by default. You can still disable the notification separately for each monitor.", | ||||
|     "Default enabled": "Default enabled", | ||||
|     "Also apply to existing monitors": "Also apply to existing monitors", | ||||
|     Export: "Export", | ||||
|     Import: "Import", | ||||
|     backupDescription: "You can backup all monitors and all notifications into a JSON file.", | ||||
|     backupDescription2: "PS: History and event data is not included.", | ||||
|     backupDescription3: "Sensitive data such as notification tokens is included in the export file, please keep it carefully.", | ||||
|     alertNoFile: "Please select a file to import.", | ||||
|     alertWrongFileType: "Please select a JSON file.", | ||||
|     twoFAVerifyLabel: "Please type in your token to verify that 2FA is working", | ||||
|     tokenValidSettingsMsg: "Token is valid! You can now save the 2FA settings.", | ||||
|     confirmEnableTwoFAMsg: "Are you sure you want to enable 2FA?", | ||||
|     confirmDisableTwoFAMsg: "Are you sure you want to disable 2FA?", | ||||
|     "Apply on all existing monitors": "Apply on all existing monitors", | ||||
|     "Verify Token": "Verify Token", | ||||
|     "Setup 2FA": "Setup 2FA", | ||||
|     "Enable 2FA": "Enable 2FA", | ||||
|     "Disable 2FA": "Disable 2FA", | ||||
|     "2FA Settings": "2FA Settings", | ||||
|     "Two Factor Authentication": "Two Factor Authentication", | ||||
|     Active: "Active", | ||||
|     Inactive: "Inactive", | ||||
|     Create: "Crear", | ||||
|     clearEventsMsg: "¿Está seguro de que desea eliminar todos los eventos de este monitor?", | ||||
|     clearHeartbeatsMsg: "¿Está seguro de que desea eliminar todos los latidos de este monitor?", | ||||
|     confirmClearStatisticsMsg: "¿Está seguro de que desea eliminar TODAS las estadísticas?", | ||||
|     "Clear Data": "Borrar Datos", | ||||
|     Events: "Eventos", | ||||
|     Heartbeats: "Latidos", | ||||
|     "Auto Get": "Obtener automáticamente", | ||||
|     enableDefaultNotificationDescription: "Para cada nuevo monitor, esta notificación estará habilitada de forma predeterminada. Aún puede deshabilitar la notificación por separado para cada monitor.", | ||||
|     "Default enabled": "Habilitado por defecto", | ||||
|     "Also apply to existing monitors": "También se aplica a monitores existentes", | ||||
|     Export: "Exportar", | ||||
|     Import: "Importar", | ||||
|     backupDescription: "Puede hacer una copia de seguridad de todos los monitores y todas las notificaciones en un archivo JSON.", | ||||
|     backupDescription2: "PD: el historial y los datos de eventos no están incluidos.", | ||||
|     backupDescription3: "Los datos confidenciales, como los tokens de notificación, se incluyen en el archivo de exportación. Guárdelo con cuidado.", | ||||
|     alertNoFile: "Seleccione un archivo para importar.", | ||||
|     alertWrongFileType: "Seleccione un archivo JSON.", | ||||
|     twoFAVerifyLabel: "Ingrese su token para verificar que 2FA está funcionando", | ||||
|     tokenValidSettingsMsg: "¡El token es válido! Ahora puede guardar la configuración de 2FA.", | ||||
|     confirmEnableTwoFAMsg: "¿Estás seguro de que quieres habilitar 2FA?", | ||||
|     confirmDisableTwoFAMsg: "¿Estás seguro de que quieres desactivar 2FA?", | ||||
|     "Apply on all existing monitors": "Aplicar en todos los monitores existentes", | ||||
|     "Verify Token": "Verificar token", | ||||
|     "Setup 2FA": "Configurar 2FA", | ||||
|     "Enable 2FA": "Habilitar 2FA", | ||||
|     "Disable 2FA": "Desactivar 2FA", | ||||
|     "2FA Settings": "Ajustes 2FA", | ||||
|     "Two Factor Authentication": "Autenticación de dos factores", | ||||
|     Active: "Activo", | ||||
|     Inactive: "Inactivo", | ||||
|     Token: "Token", | ||||
|     "Show URI": "Show URI", | ||||
|     "Clear all statistics": "Clear all Statistics", | ||||
|     retryCheckEverySecond: "Retry every {0} seconds.", | ||||
|     importHandleDescription: "Choose 'Skip existing' if you want to skip every monitor or notification with the same name. 'Overwrite' will delete every existing monitor and notification.", | ||||
|     confirmImportMsg: "Are you sure to import the backup? Please make sure you've selected the right import option.", | ||||
|     "Heartbeat Retry Interval": "Heartbeat Retry Interval", | ||||
|     "Import Backup": "Import Backup", | ||||
|     "Export Backup": "Export Backup", | ||||
|     "Skip existing": "Skip existing", | ||||
|     Overwrite: "Overwrite", | ||||
|     Options: "Options", | ||||
|     "Keep both": "Keep both", | ||||
|     Tags: "Tags", | ||||
|     "Add New below or Select...": "Add New below or Select...", | ||||
|     "Tag with this name already exist.": "Tag with this name already exist.", | ||||
|     "Tag with this value already exist.": "Tag with this value already exist.", | ||||
|     "Show URI": "Mostrar URI", | ||||
|     "Clear all statistics": "Borrar todas las estadísticas", | ||||
|     retryCheckEverySecond: "Reintentar cada {0} segundo.", | ||||
|     importHandleDescription: "Elija 'Omitir existente' si desea omitir todos los monitores o notificaciones con el mismo nombre. 'Sobrescribir' eliminará todos los monitores y notificaciones existentes.", | ||||
|     confirmImportMsg: "¿Estás seguro de importar la copia de seguridad? Asegúrese de haber seleccionado la opción de importación correcta.", | ||||
|     "Heartbeat Retry Interval": "Intervalo de reintento de latido", | ||||
|     "Import Backup": "Importar copia de seguridad", | ||||
|     "Export Backup": "Exportar copia de seguridad", | ||||
|     "Skip existing": "Omitir existente", | ||||
|     Overwrite: "Sobrescribir", | ||||
|     Options: "Opciones", | ||||
|     "Keep both": "Mantén ambos", | ||||
|     Tags: "Etiquetas", | ||||
|     "Add New below or Select...": "Agregar nuevo a continuación o Seleccionar...", | ||||
|     "Tag with this name already exist.": "La etiqueta con este nombre ya existe.", | ||||
|     "Tag with this value already exist.": "La etiqueta con este valor ya existe.", | ||||
|     color: "color", | ||||
|     "value (optional)": "value (optional)", | ||||
|     Gray: "Gray", | ||||
|     Red: "Red", | ||||
|     Orange: "Orange", | ||||
|     Green: "Green", | ||||
|     Blue: "Blue", | ||||
|     Indigo: "Indigo", | ||||
|     Purple: "Purple", | ||||
|     Pink: "Pink", | ||||
|     "Search...": "Search...", | ||||
|     "Avg. Ping": "Avg. Ping", | ||||
|     "Avg. Response": "Avg. Response", | ||||
|     "Entry Page": "Entry Page", | ||||
|     statusPageNothing: "Nothing here, please add a group or a monitor.", | ||||
|     "No Services": "No Services", | ||||
|     "All Systems Operational": "All Systems Operational", | ||||
|     "Partially Degraded Service": "Partially Degraded Service", | ||||
|     "Degraded Service": "Degraded Service", | ||||
|     "Add Group": "Add Group", | ||||
|     "Add a monitor": "Add a monitor", | ||||
|     "Edit Status Page": "Edit Status Page", | ||||
|     "Go to Dashboard": "Go to Dashboard", | ||||
|     "Status Page": "Status Page", | ||||
|     "value (optional)": "valor (opcional)", | ||||
|     Gray: "Gris", | ||||
|     Red: "Rojo", | ||||
|     Orange: "Naranja", | ||||
|     Green: "Verde", | ||||
|     Blue: "Azul", | ||||
|     Indigo: "Índigo", | ||||
|     Purple: "Morado", | ||||
|     Pink: "Rosa", | ||||
|     "Search...": "Buscar...", | ||||
|     "Avg. Ping": "Ping promedio", | ||||
|     "Avg. Response": "Respuesta promedio", | ||||
|     "Entry Page": "Página de entrada", | ||||
|     statusPageNothing: "No hay nada aquí, agregue un grupo o un monitor.", | ||||
|     "No Services": "Sin servicio", | ||||
|     "All Systems Operational": "Todos los sistemas están operativos", | ||||
|     "Partially Degraded Service": "Servicio parcialmente degradado", | ||||
|     "Degraded Service": "Servicio degradado", | ||||
|     "Add Group": "Agregar Grupo", | ||||
|     "Add a monitor": "Agregar un monitor", | ||||
|     "Edit Status Page": "Editar página de estado", | ||||
|     "Go to Dashboard": "Ir al panel de control", | ||||
|     "Status Page": "Página de estado", | ||||
|     telegram: "Telegram", | ||||
|     webhook: "Webhook", | ||||
|     smtp: "Email (SMTP)", | ||||
| @@ -192,8 +192,9 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     apprise: "Apprise (Admite más de 50 servicios de notificación)", | ||||
|     pushbullet: "Pushbullet", | ||||
|     line: "Line Messenger", | ||||
|     mattermost: "Mattermost", | ||||
|   | ||||
| @@ -191,6 +191,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (vahendab üle 65 teavitusteenust)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -199,6 +199,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -190,6 +190,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -191,6 +191,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (obsługuje 50+ usług powiadamiania)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -191,6 +191,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -196,6 +196,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -191,6 +191,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (Support 50+ Notification services)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -192,6 +192,7 @@ export default { | ||||
|     pushover: "Pushover", | ||||
|     pushy: "Pushy", | ||||
|     octopush: "Octopush", | ||||
|     promosms: "PromoSMS", | ||||
|     lunasea: "LunaSea", | ||||
|     apprise: "Apprise (支援 50 多種通知)", | ||||
|     pushbullet: "Pushbullet", | ||||
|   | ||||
| @@ -43,3 +43,9 @@ app.component("Editable", contenteditable); | ||||
| app.component("FontAwesomeIcon", FontAwesomeIcon); | ||||
|  | ||||
| app.mount("#app"); | ||||
|  | ||||
| // Expose the vue instance for development | ||||
| if (process.env.NODE_ENV === "development") { | ||||
|     console.log("Dev Only: window.app is the vue instance"); | ||||
|     window.app = app._instance; | ||||
| } | ||||
|   | ||||
| @@ -38,11 +38,15 @@ export default { | ||||
|         }, | ||||
|  | ||||
|         baseURL() { | ||||
|             if (this.$root.info.primaryBaseURL) { | ||||
|                 return this.$root.info.primaryBaseURL; | ||||
|             } | ||||
|  | ||||
|             if (env === "development" || localStorage.dev === "dev") { | ||||
|                 return axios.defaults.baseURL; | ||||
|             } else { | ||||
|                 return location.protocol + "//" + location.host; | ||||
|             } | ||||
|         } | ||||
|         }, | ||||
|     } | ||||
| }; | ||||
|   | ||||
| @@ -171,7 +171,7 @@ | ||||
|                                 <input id="retry-interval" v-model="monitor.retryInterval" type="number" class="form-control" required min="20" step="1"> | ||||
|                             </div> | ||||
|  | ||||
|                             <h2 class="mt-5 mb-2">{{ $t("Advanced") }}</h2> | ||||
|                             <h2 v-if="monitor.type !== 'push'" class="mt-5 mb-2">{{ $t("Advanced") }}</h2> | ||||
|  | ||||
|                             <div v-if="monitor.type === 'http' || monitor.type === 'keyword' " class="my-3 form-check"> | ||||
|                                 <input id="ignore-tls" v-model="monitor.ignoreTls" class="form-check-input" type="checkbox" value=""> | ||||
| @@ -180,7 +180,7 @@ | ||||
|                                 </label> | ||||
|                             </div> | ||||
|  | ||||
|                             <div class="my-3 form-check"> | ||||
|                             <div v-if="monitor.type !== 'push'" class="my-3 form-check"> | ||||
|                                 <input id="upside-down" v-model="monitor.upsideDown" class="form-check-input" type="checkbox"> | ||||
|                                 <label class="form-check-label" for="upside-down"> | ||||
|                                     {{ $t("Upside Down Mode") }} | ||||
|   | ||||
| @@ -74,7 +74,7 @@ class TimeLogger { | ||||
|         this.startTime = dayjs().valueOf(); | ||||
|     } | ||||
|     print(name) { | ||||
|         if (exports.isDev && process && process.env.TIMELOGGER === "1") { | ||||
|         if (exports.isDev && process.env.TIMELOGGER === "1") { | ||||
|             console.log(name + ": " + (dayjs().valueOf() - this.startTime) + "ms"); | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -86,7 +86,7 @@ export class TimeLogger { | ||||
|     } | ||||
|  | ||||
|     print(name: string) { | ||||
|         if (isDev && process && process.env.TIMELOGGER === "1") { | ||||
|         if (isDev && process.env.TIMELOGGER === "1") { | ||||
|             console.log(name + ": " + (dayjs().valueOf() - this.startTime) + "ms") | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -36,7 +36,7 @@ describe("Init", () => { | ||||
|     }); | ||||
| 
 | ||||
|     it(`should be titled "${title}"`, async () => { | ||||
|         await expect(page.title()).resolves.toMatch(title); | ||||
|         await expect(page.title()).resolves.toEqual(title); | ||||
|     }); | ||||
| 
 | ||||
|     // Setup Page
 | ||||
| @@ -75,11 +75,11 @@ describe("Init", () => { | ||||
| 
 | ||||
|             await page.select("#language", "zh-HK"); | ||||
|             let languageTitle = await page.evaluate(() => document.querySelector("[for=language]").innerText); | ||||
|             expect(languageTitle).toMatch("語言"); | ||||
|             expect(languageTitle).toEqual("語言"); | ||||
| 
 | ||||
|             await page.select("#language", "en"); | ||||
|             languageTitle = await page.evaluate(() => document.querySelector("[for=language]").innerText); | ||||
|             expect(languageTitle).toMatch("Language"); | ||||
|             expect(languageTitle).toEqual("Language"); | ||||
|         }); | ||||
| 
 | ||||
|         it("Change Theme", async () => { | ||||
| @@ -103,21 +103,21 @@ describe("Init", () => { | ||||
|         it("Search Engine Visibility", async () => { | ||||
|             // Default
 | ||||
|             let res = await axios.get(baseURL + "/robots.txt"); | ||||
|             expect(res.data).toMatch("Disallow: /"); | ||||
|             expect(res.data).toContain("Disallow: /"); | ||||
| 
 | ||||
|             // Yes
 | ||||
|             await click(page, "#searchEngineIndexYes"); | ||||
|             await click(page, "form > div > .btn[type=submit]"); | ||||
|             await sleep(2000); | ||||
|             res = await axios.get(baseURL + "/robots.txt"); | ||||
|             expect(res.data).not.toMatch("Disallow: /"); | ||||
|             expect(res.data).not.toContain("Disallow: /"); | ||||
| 
 | ||||
|             // No
 | ||||
|             await click(page, "#searchEngineIndexNo"); | ||||
|             await click(page, "form > div > .btn[type=submit]"); | ||||
|             await sleep(2000); | ||||
|             res = await axios.get(baseURL + "/robots.txt"); | ||||
|             expect(res.data).toMatch("Disallow: /"); | ||||
|             expect(res.data).toContain("Disallow: /"); | ||||
|         }); | ||||
| 
 | ||||
|         it("Entry Page", async () => { | ||||
| @@ -218,7 +218,7 @@ describe("Init", () => { | ||||
|             await page.goto(baseURL + "/status"); | ||||
|         }); | ||||
|         it(`should be titled "${title}"`, async () => { | ||||
|             await expect(page.title()).resolves.toMatch(title); | ||||
|             await expect(page.title()).resolves.toEqual(title); | ||||
|         }); | ||||
|     }); | ||||
| }); | ||||
							
								
								
									
										42
									
								
								test/frontend.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								test/frontend.spec.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| // eslint-disable-next-line no-global-assign | ||||
| global.localStorage = {}; | ||||
| global.navigator = { | ||||
|     language: "en" | ||||
| }; | ||||
|  | ||||
| const { currentLocale } = require("../src/i18n"); | ||||
|  | ||||
| describe("Test i18n.js", () => { | ||||
|  | ||||
|     it("currentLocale()", () => { | ||||
|         expect(currentLocale()).toEqual("en"); | ||||
|  | ||||
|         navigator.language = "zh-HK"; | ||||
|         expect(currentLocale()).toEqual("zh-HK"); | ||||
|  | ||||
|         // Note that in Safari on iOS prior to 10.2, the country code returned is lowercase: "en-us", "fr-fr" etc. | ||||
|         // https://developer.mozilla.org/en-US/docs/Web/API/Navigator/language | ||||
|         navigator.language = "zh-hk"; | ||||
|         expect(currentLocale()).toEqual("en"); | ||||
|  | ||||
|         navigator.language = "en-US"; | ||||
|         expect(currentLocale()).toEqual("en"); | ||||
|  | ||||
|         navigator.language = "ja-ZZ"; | ||||
|         expect(currentLocale()).toEqual("ja"); | ||||
|  | ||||
|         navigator.language = "zz"; | ||||
|         expect(currentLocale()).toEqual("en"); | ||||
|  | ||||
|         navigator.language = "zz-ZZ"; | ||||
|         expect(currentLocale()).toEqual("en"); | ||||
|  | ||||
|         localStorage.locale = "en"; | ||||
|         expect(currentLocale()).toEqual("en"); | ||||
|  | ||||
|         localStorage.locale = "zh-HK"; | ||||
|         expect(currentLocale()).toEqual("zh-HK"); | ||||
|     }); | ||||
|  | ||||
| }); | ||||
|  | ||||
| @@ -6,5 +6,4 @@ FROM ubuntu | ||||
| # RUN ln -s /usr/bin/nodejs /usr/bin/node | ||||
| # RUN node -v | ||||
|  | ||||
| COPY ./install.sh . | ||||
| RUN bash install.sh local /opt/uptime-kuma 3000 0.0.0.0 | ||||
| RUN curl -o kuma_install.sh http://git.kuma.pet/install.sh && bash kuma_install.sh local /opt/uptime-kuma 3000 0.0.0.0 | ||||
|   | ||||
		Reference in New Issue
	
	Block a user