mirror of
				https://github.com/louislam/uptime-kuma.git
				synced 2025-10-25 07:39:22 +08:00 
			
		
		
		
	Merge remote-tracking branch 'origin/master' into mariadb
# Conflicts: # docker/alpine-base.dockerfile # docker/dockerfile # package-lock.json
This commit is contained in:
		| @@ -1,6 +1,7 @@ | ||||
| /.idea | ||||
| /node_modules | ||||
| /data | ||||
| /cypress | ||||
| /out | ||||
| /test | ||||
| /kubernetes | ||||
| @@ -30,6 +31,9 @@ tsconfig.json | ||||
| /tmp | ||||
| /babel.config.js | ||||
| /ecosystem.config.js | ||||
| /extra/healthcheck.exe | ||||
| /extra/healthcheck | ||||
|  | ||||
|  | ||||
| ### .gitignore content (commented rules are duplicated) | ||||
|  | ||||
|   | ||||
| @@ -19,3 +19,6 @@ indent_size = 2 | ||||
|  | ||||
| [*.vue] | ||||
| trim_trailing_whitespace = false | ||||
|  | ||||
| [*.go] | ||||
| indent_style = tab | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,8 @@ | ||||
| 👉 Delete this line if you have read and agree our pull request rules and guidelines: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md#can-i-create-a-pull-request-for-uptime-kuma | ||||
| ⚠️⚠️⚠️ Since we do not accept all types of pull requests and do not want to waste your time. Please be sure that you have read pull request rules: | ||||
| https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md#can-i-create-a-pull-request-for-uptime-kuma | ||||
|  | ||||
| Tick the checkbox if you understand [x]:  | ||||
| - [ ] I have read and understand the pull request rules. | ||||
|  | ||||
| # Description | ||||
|  | ||||
|   | ||||
							
								
								
									
										16
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -50,3 +50,19 @@ jobs: | ||||
|         cache: 'npm' | ||||
|     - run: npm install | ||||
|     - run: npm run lint | ||||
|  | ||||
|   e2e-tests: | ||||
|     needs: [ check-linters ] | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - run: git config --global core.autocrlf false  # Mainly for Windows | ||||
|     - uses: actions/checkout@v3 | ||||
|  | ||||
|     - name: Use Node.js 14 | ||||
|       uses: actions/setup-node@v3 | ||||
|       with: | ||||
|         node-version: 14 | ||||
|         cache: 'npm' | ||||
|     - run: npm install | ||||
|     - run: npm run build | ||||
|     - run: npm run cy:test | ||||
|   | ||||
							
								
								
									
										22
									
								
								.github/workflows/stale-bot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/stale-bot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| name: 'Automatically close stale issues and PRs' | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: '0 */6 * * *' | ||||
| #Run every 6 hours  | ||||
|  | ||||
| jobs: | ||||
|   stale: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/stale@v5 | ||||
|         with: | ||||
|           stale-issue-message: 'We are clearing up our old issues and your ticket has been open for 3 months with no activity. Remove stale label or comment or this will be closed in 2 days.' | ||||
|           close-issue-message: 'This issue was closed because it has been stalled for 2 days with no activity.' | ||||
|           days-before-stale: 90 | ||||
|           days-before-close: 2 | ||||
|           days-before-pr-stale: 999999999 | ||||
|           days-before-pr-close: 1 | ||||
|           exempt-issue-labels: 'News,Medium,High,discussion,bug,doc,feature-request' | ||||
|           exempt-issue-assignees: 'louislam' | ||||
|           operations-per-run: 200 | ||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -13,3 +13,10 @@ dist-ssr | ||||
| /out | ||||
| /tmp | ||||
| .env | ||||
|  | ||||
| cypress/videos | ||||
| cypress/screenshots | ||||
|  | ||||
| /extra/healthcheck.exe | ||||
| /extra/healthcheck | ||||
| /extra/healthcheck-armv7 | ||||
|   | ||||
| @@ -27,13 +27,11 @@ The frontend code build into "dist" directory. The server (express.js) exposes t | ||||
|  | ||||
| ## Can I create a pull request for Uptime Kuma? | ||||
|  | ||||
| Yes, you can. However, since I don't want to waste your time, be sure to **create empty draft pull request, so we can discuss first** if it is a large pull request or you don't know it will be merged or not. | ||||
| Yes or no, it depends on what you will try to do. Since I don't want to waste your time, be sure to **create an empty draft pull request or open an issue, so we can discuss first**. Especially for a large pull request or you don't know it will be merged or not. | ||||
|  | ||||
| Also, please don't rush or ask for ETA, because I have to understand the pull request, make sure it is no breaking changes and stick to my vision of this project, especially for large pull requests. | ||||
| Here are some references: | ||||
|  | ||||
| I will mark your pull request in the [milestones](https://github.com/louislam/uptime-kuma/milestones), if I am plan to review and merge it. | ||||
|  | ||||
| ✅ Accept: | ||||
| ✅ Usually Accept: | ||||
| - Bug/Security fix | ||||
| - Translations | ||||
| - Adding notification providers | ||||
| @@ -47,8 +45,14 @@ I will mark your pull request in the [milestones](https://github.com/louislam/up | ||||
| - Any breaking changes | ||||
| - Duplicated pull request | ||||
| - Buggy | ||||
| - UI/UX is not close to Uptime Kuma  | ||||
| - Existing logic is completely modified or deleted for no reason | ||||
| - A function that is completely out of scope | ||||
| - Unnecessary large code changes (Hard to review, causes code conflicts to other pull requests) | ||||
|  | ||||
| I will mark your pull request in the [milestones](https://github.com/louislam/uptime-kuma/milestones), if I am plan to review and merge it. | ||||
|  | ||||
| Also, please don't rush or ask for ETA, because I have to understand the pull request, make sure it is no breaking changes and stick to my vision of this project, especially for large pull requests. | ||||
|  | ||||
|  | ||||
| ### Recommended Pull Request Guideline | ||||
| @@ -177,7 +181,18 @@ npm test | ||||
|  | ||||
| By default, the Chromium window will be shown up during the test. Specifying `HEADLESS_TEST=1` for terminal environments. | ||||
|  | ||||
| ## Update Dependencies | ||||
| ## Dependencies | ||||
|  | ||||
| Both frontend and backend share the same package.json. However, the frontend dependencies are eventually not used in the production environment, because it is usually also baked into dist files. So: | ||||
|  | ||||
| - Frontend dependencies = "devDependencies" | ||||
|   - Examples: vue, chart.js | ||||
| - Backend dependencies = "dependencies" | ||||
|   - Examples: socket.io, sqlite3 | ||||
| - Development dependencies = "devDependencies" | ||||
|   - Examples: eslint, sass | ||||
|  | ||||
| ### Update Dependencies | ||||
|  | ||||
| Install `ncu` | ||||
| https://github.com/raineorshine/npm-check-updates | ||||
|   | ||||
							
								
								
									
										24
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								README.md
									
									
									
									
									
								
							| @@ -15,15 +15,14 @@ It is a self-hosted monitoring tool like "Uptime Robot". | ||||
|  | ||||
| Try it! | ||||
|  | ||||
| https://demo.uptime.kuma.pet | ||||
| - Tokyo Demo Server: https://demo.uptime.kuma.pet (Sponsored by [Uptime Kuma Sponsors](https://github.com/louislam/uptime-kuma#%EF%B8%8F-sponsors)) | ||||
| - Europe Demo Server: https://demo.uptime-kuma.karimi.dev:27000 (Provided by [@mhkarimi1383](https://github.com/mhkarimi1383)) | ||||
|  | ||||
| It is a temporary live demo, all data will be deleted after 10 minutes. The server is located in Tokyo, so if you live far from there, it may affect your experience. I suggest that you should install and try it out for the best demo experience. | ||||
|  | ||||
| VPS is sponsored by Uptime Kuma sponsors on [Open Collective](https://opencollective.com/uptime-kuma)! Thank you so much! | ||||
| It is a temporary live demo, all data will be deleted after 10 minutes. Use the one that is closer to you, but I suggest that you should install and try it out for the best demo experience. | ||||
|  | ||||
| ## ⭐ Features | ||||
|  | ||||
| * Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / Ping / DNS Record / Push / Steam Game Server. | ||||
| * Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / Ping / DNS Record / Push / Steam Game Server / Docker Containers. | ||||
| * Fancy, Reactive, Fast UI/UX. | ||||
| * Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [90+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/tree/master/src/components/notifications). | ||||
| * 20 second intervals. | ||||
| @@ -106,7 +105,7 @@ https://github.com/louislam/uptime-kuma/milestones | ||||
|  | ||||
| Project Plan: | ||||
|  | ||||
| https://github.com/louislam/uptime-kuma/projects/1 | ||||
| https://github.com/users/louislam/projects/4/views/1 | ||||
|  | ||||
| ## ❤️ Sponsors | ||||
|  | ||||
| @@ -157,7 +156,14 @@ You can mention me if you ask a question on Reddit. | ||||
|  | ||||
| ## Contribute | ||||
|  | ||||
| ### Beta Version | ||||
| ### Test Pull Requests | ||||
|  | ||||
| There are a lot of pull requests right now, but I don't have time to test them all. | ||||
|  | ||||
| If you want to help, you can check this: | ||||
| https://github.com/louislam/uptime-kuma/wiki/Test-Pull-Requests | ||||
|  | ||||
| ### Test Beta Version | ||||
|  | ||||
| Check out the latest beta release here: https://github.com/louislam/uptime-kuma/releases | ||||
|  | ||||
| @@ -169,5 +175,5 @@ If you want to translate Uptime Kuma into your language, please read: https://gi | ||||
|  | ||||
| Feel free to correct my grammar in this README, source code, or wiki, as my mother language is not English and my grammar is not that great. | ||||
|  | ||||
| ### Pull Requests | ||||
| If you want to modify Uptime Kuma, this guideline may be useful for you: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md | ||||
| ### Create Pull Requests | ||||
| If you want to modify Uptime Kuma, please read this guide and follow the rules here: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md | ||||
|   | ||||
							
								
								
									
										28
									
								
								config/cypress.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								config/cypress.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| const { defineConfig } = require("cypress"); | ||||
|  | ||||
| module.exports = defineConfig({ | ||||
|     projectId: "vyjuem", | ||||
|     e2e: { | ||||
|         experimentalStudio: true, | ||||
|         setupNodeEvents(on, config) { | ||||
|  | ||||
|         }, | ||||
|         fixturesFolder: "test/cypress/fixtures", | ||||
|         screenshotsFolder: "test/cypress/screenshots", | ||||
|         videosFolder: "test/cypress/videos", | ||||
|         downloadsFolder: "test/cypress/downloads", | ||||
|         supportFile: "test/cypress/support/e2e.js", | ||||
|         baseUrl: "http://localhost:3002", | ||||
|         defaultCommandTimeout: 10000, | ||||
|         pageLoadTimeout: 60000, | ||||
|         viewportWidth: 1920, | ||||
|         viewportHeight: 1080, | ||||
|         specPattern: [ | ||||
|             "test/cypress/e2e/setup.cy.js", | ||||
|             "test/cypress/e2e/**/*.js" | ||||
|         ], | ||||
|     }, | ||||
|     env: { | ||||
|         baseUrl: "http://localhost:3002", | ||||
|     }, | ||||
| }); | ||||
| @@ -1,33 +0,0 @@ | ||||
| const PuppeteerEnvironment = require("jest-environment-puppeteer"); | ||||
| const util = require("util"); | ||||
|  | ||||
| class DebugEnv extends PuppeteerEnvironment { | ||||
|     async handleTestEvent(event, state) { | ||||
|         const ignoredEvents = [ | ||||
|             "setup", | ||||
|             "add_hook", | ||||
|             "start_describe_definition", | ||||
|             "add_test", | ||||
|             "finish_describe_definition", | ||||
|             "run_start", | ||||
|             "run_describe_start", | ||||
|             "test_start", | ||||
|             "hook_start", | ||||
|             "hook_success", | ||||
|             "test_fn_start", | ||||
|             "test_fn_success", | ||||
|             "test_done", | ||||
|             "run_describe_finish", | ||||
|             "run_finish", | ||||
|             "teardown", | ||||
|             "test_fn_failure", | ||||
|         ]; | ||||
|         if (!ignoredEvents.includes(event.name)) { | ||||
|             console.log( | ||||
|                 new Date().toString() + ` Unhandled event [${event.name}] ` + util.inspect(event) | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = DebugEnv; | ||||
| @@ -1,5 +0,0 @@ | ||||
| module.exports = { | ||||
|     "rootDir": "..", | ||||
|     "testRegex": "./test/frontend.spec.js", | ||||
| }; | ||||
|  | ||||
| @@ -1,20 +0,0 @@ | ||||
| module.exports = { | ||||
|     "launch": { | ||||
|         "dumpio": true, | ||||
|         "slowMo": 500, | ||||
|         "headless": process.env.HEADLESS_TEST || false, | ||||
|         "userDataDir": "./data/test-chrome-profile", | ||||
|         args: [ | ||||
|             "--disable-setuid-sandbox", | ||||
|             "--disable-gpu", | ||||
|             "--disable-dev-shm-usage", | ||||
|             "--no-default-browser-check", | ||||
|             "--no-experiments", | ||||
|             "--no-first-run", | ||||
|             "--no-pings", | ||||
|             "--no-sandbox", | ||||
|             "--no-zygote", | ||||
|             "--single-process", | ||||
|         ], | ||||
|     } | ||||
| }; | ||||
| @@ -1,12 +0,0 @@ | ||||
| module.exports = { | ||||
|     "verbose": true, | ||||
|     "preset": "jest-puppeteer", | ||||
|     "globals": { | ||||
|         "__DEV__": true | ||||
|     }, | ||||
|     "testRegex": "./test/e2e.spec.js", | ||||
|     "testEnvironment": "./config/jest-debug-env.js", | ||||
|     "rootDir": "..", | ||||
|     "testTimeout": 30000, | ||||
| }; | ||||
|  | ||||
| @@ -11,6 +11,9 @@ const viteCompressionFilter = /\.(js|mjs|json|css|html|svg)$/i; | ||||
|  | ||||
| // https://vitejs.dev/config/ | ||||
| export default defineConfig({ | ||||
|     server: { | ||||
|         port: 3000, | ||||
|     }, | ||||
|     define: { | ||||
|         "FRONTEND_VERSION": JSON.stringify(process.env.npm_package_version), | ||||
|     }, | ||||
|   | ||||
							
								
								
									
										18
									
								
								db/patch-add-docker-columns.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								db/patch-add-docker-columns.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| CREATE TABLE docker_host ( | ||||
| 	id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||
| 	user_id INT NOT NULL, | ||||
| 	docker_daemon VARCHAR(255), | ||||
| 	docker_type VARCHAR(255), | ||||
| 	name VARCHAR(255) | ||||
| ); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
| 	ADD docker_host INTEGER REFERENCES docker_host(id); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
| 	ADD docker_container VARCHAR(255); | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										18
									
								
								db/patch-add-radius-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								db/patch-add-radius-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_username VARCHAR(255); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_password VARCHAR(255); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_calling_station_id VARCHAR(50); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_called_station_id VARCHAR(50); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_secret VARCHAR(255); | ||||
|  | ||||
| COMMIT | ||||
							
								
								
									
										25
									
								
								db/patch-grpc-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								db/patch-grpc-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_url VARCHAR(255) default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_protobuf TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_body TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_metadata TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_method VARCHAR(255) default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_service_name VARCHAR(255) default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_enable_tls BOOLEAN default 0 not null; | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										83
									
								
								db/patch-maintenance-table2.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								db/patch-maintenance-table2.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| -- Just for someone who tested maintenance before (patch-maintenance-table.sql) | ||||
| DROP TABLE IF EXISTS maintenance_status_page; | ||||
| DROP TABLE IF EXISTS monitor_maintenance; | ||||
| DROP TABLE IF EXISTS maintenance; | ||||
| DROP TABLE IF EXISTS maintenance_timeslot; | ||||
|  | ||||
| -- maintenance | ||||
| CREATE TABLE [maintenance] ( | ||||
|     [id] INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, | ||||
|     [title] VARCHAR(150) NOT NULL, | ||||
|     [description] TEXT NOT NULL, | ||||
|     [user_id] INTEGER REFERENCES [user]([id]) ON DELETE SET NULL ON UPDATE CASCADE, | ||||
|     [active] BOOLEAN NOT NULL DEFAULT 1, | ||||
|     [strategy] VARCHAR(50) NOT NULL DEFAULT 'single', | ||||
|     [start_date] DATETIME, | ||||
|     [end_date] DATETIME, | ||||
|     [start_time] TIME, | ||||
|     [end_time] TIME, | ||||
|     [weekdays] VARCHAR2(250) DEFAULT '[]', | ||||
|     [days_of_month] TEXT DEFAULT '[]', | ||||
|     [interval_day] INTEGER | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [manual_active] ON [maintenance] ( | ||||
|     [strategy], | ||||
|     [active] | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [active] ON [maintenance] ([active]); | ||||
|  | ||||
| CREATE INDEX [maintenance_user_id] ON [maintenance] ([user_id]); | ||||
|  | ||||
| -- maintenance_status_page | ||||
| CREATE TABLE maintenance_status_page ( | ||||
|     id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||
|     status_page_id INTEGER NOT NULL, | ||||
|     maintenance_id INTEGER NOT NULL, | ||||
|     CONSTRAINT FK_maintenance FOREIGN KEY (maintenance_id) REFERENCES maintenance (id) ON DELETE CASCADE ON UPDATE CASCADE, | ||||
|     CONSTRAINT FK_status_page FOREIGN KEY (status_page_id) REFERENCES status_page (id) ON DELETE CASCADE ON UPDATE CASCADE | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [status_page_id_index] | ||||
|     ON [maintenance_status_page]([status_page_id]); | ||||
|  | ||||
| CREATE INDEX [maintenance_id_index] | ||||
|     ON [maintenance_status_page]([maintenance_id]); | ||||
|  | ||||
| -- maintenance_timeslot | ||||
| CREATE TABLE [maintenance_timeslot] ( | ||||
|     [id] INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, | ||||
|     [maintenance_id] INTEGER NOT NULL CONSTRAINT [FK_maintenance] REFERENCES [maintenance]([id]) ON DELETE CASCADE ON UPDATE CASCADE, | ||||
|     [start_date] DATETIME NOT NULL, | ||||
|     [end_date] DATETIME, | ||||
|     [generated_next] BOOLEAN DEFAULT 0 | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [maintenance_id] ON [maintenance_timeslot] ([maintenance_id] DESC); | ||||
|  | ||||
| CREATE INDEX [active_timeslot_index] ON [maintenance_timeslot] ( | ||||
|     [maintenance_id] DESC, | ||||
|     [start_date] DESC, | ||||
|     [end_date] DESC | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [generated_next_index] ON [maintenance_timeslot] ([generated_next]); | ||||
|  | ||||
| -- monitor_maintenance | ||||
| CREATE TABLE monitor_maintenance ( | ||||
|     id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||
|     monitor_id INTEGER NOT NULL, | ||||
|     maintenance_id INTEGER NOT NULL, | ||||
|     CONSTRAINT FK_maintenance FOREIGN KEY (maintenance_id) REFERENCES maintenance (id) ON DELETE CASCADE ON UPDATE CASCADE, | ||||
|     CONSTRAINT FK_monitor FOREIGN KEY (monitor_id) REFERENCES monitor (id) ON DELETE CASCADE ON UPDATE CASCADE | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [maintenance_id_index2] ON [monitor_maintenance]([maintenance_id]); | ||||
|  | ||||
| CREATE INDEX [monitor_id_index] ON [monitor_maintenance]([monitor_id]); | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										10
									
								
								db/patch-monitor-add-resend-interval.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								db/patch-monitor-add-resend-interval.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD resend_interval INTEGER default 0 not null; | ||||
|  | ||||
| ALTER TABLE heartbeat | ||||
|     ADD down_count INTEGER default 0 not null; | ||||
|  | ||||
| COMMIT; | ||||
| @@ -11,7 +11,7 @@ WORKDIR /app | ||||
| RUN apt update && \ | ||||
|     apt --yes --no-install-recommends install python3 python3-pip python3-cryptography python3-six python3-yaml python3-click python3-markdown python3-requests python3-requests-oauthlib \ | ||||
|         sqlite3 iputils-ping util-linux dumb-init && \ | ||||
|     pip3 --no-cache-dir install apprise==0.9.9 && \ | ||||
|     pip3 --no-cache-dir install apprise==1.2.0 && \ | ||||
|     rm -rf /var/lib/apt/lists/* && \ | ||||
|     apt --yes autoremove | ||||
|  | ||||
|   | ||||
| @@ -1,36 +1,98 @@ | ||||
| ############################################ | ||||
| # Build in Golang | ||||
| # Run npm run build-healthcheck-armv7 in the host first, another it will be super slow where it is building the armv7 healthcheck | ||||
| ############################################ | ||||
| FROM golang:1.19.4-buster AS build_healthcheck | ||||
| WORKDIR /app | ||||
| ARG TARGETPLATFORM | ||||
| COPY ./extra/ ./extra/ | ||||
|  | ||||
| # Compile healthcheck.go | ||||
| RUN apt update | ||||
| RUN apt --yes --no-install-recommends install curl | ||||
| RUN curl -sL https://deb.nodesource.com/setup_18.x | bash | ||||
| RUN apt --yes --no-install-recommends install nodejs | ||||
| RUN node -v | ||||
| RUN node ./extra/build-healthcheck.js $TARGETPLATFORM | ||||
|  | ||||
| ############################################ | ||||
| # Build in Node.js | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS build | ||||
| WORKDIR /app | ||||
|  | ||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | ||||
|  | ||||
| COPY . . | ||||
| COPY --from=build_healthcheck /app/extra/healthcheck /app/extra/healthcheck | ||||
| RUN npm ci --production && \ | ||||
|     chmod +x /app/extra/entrypoint.sh | ||||
|  | ||||
|  | ||||
| FROM louislam/uptime-kuma:base-debian AS release | ||||
| ############################################ | ||||
| # ⭐ Main Image (Slim) | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS release-slim | ||||
| WORKDIR /app | ||||
|  | ||||
| # Copy app files from build layer | ||||
| COPY --from=build /app /app | ||||
|  | ||||
|  | ||||
| EXPOSE 3001 | ||||
| VOLUME ["/app/data"] | ||||
| HEALTHCHECK --interval=60s --timeout=30s --start-period=180s --retries=5 CMD node extra/healthcheck.js | ||||
| HEALTHCHECK --interval=60s --timeout=30s --start-period=180s --retries=5 CMD extra/healthcheck | ||||
| ENTRYPOINT ["/usr/bin/dumb-init", "--", "extra/entrypoint.sh"] | ||||
| CMD ["node", "server/server.js"] | ||||
|  | ||||
| FROM release AS mariadb | ||||
| # Install MariaDB | ||||
| ############################################ | ||||
| # ⭐ Main Image (With MariaDB) | ||||
| ############################################ | ||||
| FROM release-slim AS release | ||||
| RUN apt update && \ | ||||
|     apt --yes --no-install-recommends install mariadb-server && \ | ||||
|     rm -rf /var/lib/apt/lists/* && \ | ||||
|     apt --yes autoremove | ||||
|  | ||||
| FROM mariadb AS nightly | ||||
| ############################################ | ||||
| # Mark as Nightly | ||||
| ############################################ | ||||
| FROM release AS nightly | ||||
| RUN npm run mark-as-nightly | ||||
|  | ||||
| ############################################ | ||||
| # Build an image for testing pr | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS pr-test | ||||
|  | ||||
| WORKDIR /app | ||||
|  | ||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | ||||
|  | ||||
| ## Install Git | ||||
| RUN apt update \ | ||||
|     && apt --yes --no-install-recommends install curl \ | ||||
|     && curl -fsSL https://cli.github.com/packages/githubcli-archive-keyring.gpg | dd of=/usr/share/keyrings/githubcli-archive-keyring.gpg \ | ||||
|     && chmod go+r /usr/share/keyrings/githubcli-archive-keyring.gpg \ | ||||
|     && echo "deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/githubcli-archive-keyring.gpg] https://cli.github.com/packages stable main" | tee /etc/apt/sources.list.d/github-cli.list > /dev/null \ | ||||
|     && apt update \ | ||||
|     && apt --yes --no-install-recommends  install git | ||||
|  | ||||
| ## Empty the directory, because we have to clone the Git repo. | ||||
| RUN rm -rf ./* && chown node /app | ||||
|  | ||||
| USER node | ||||
| RUN git config --global user.email "no-reply@no-reply.com" | ||||
| RUN git config --global user.name "PR Tester" | ||||
| RUN git clone https://github.com/louislam/uptime-kuma.git . | ||||
| RUN npm ci | ||||
|  | ||||
| EXPOSE 3000 3001 | ||||
| VOLUME ["/app/data"] | ||||
| HEALTHCHECK --interval=60s --timeout=30s --start-period=180s --retries=5 CMD node extra/healthcheck.js | ||||
| CMD ["npm", "run", "start-pr-test"] | ||||
|  | ||||
| ############################################ | ||||
| # Upload the artifact to Github | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS upload-artifact | ||||
| WORKDIR / | ||||
| RUN apt update && \ | ||||
|   | ||||
							
								
								
									
										27
									
								
								extra/build-healthcheck.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								extra/build-healthcheck.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| const childProcess = require("child_process"); | ||||
| const fs = require("fs"); | ||||
| const platform = process.argv[2]; | ||||
|  | ||||
| if (!platform) { | ||||
|     console.error("No platform??"); | ||||
|     process.exit(1); | ||||
| } | ||||
|  | ||||
| if (platform === "linux/arm/v7") { | ||||
|     console.log("Arch: armv7"); | ||||
|     if (fs.existsSync("./extra/healthcheck-armv7")) { | ||||
|         fs.renameSync("./extra/healthcheck-armv7", "./extra/healthcheck"); | ||||
|         console.log("Already built in the host, skip."); | ||||
|         process.exit(0); | ||||
|     } else { | ||||
|         console.log("prebuilt not found, it will be slow! You should execute `npm run build-healthcheck-armv7` before build."); | ||||
|     } | ||||
| } else { | ||||
|     if (fs.existsSync("./extra/healthcheck-armv7")) { | ||||
|         fs.rmSync("./extra/healthcheck-armv7"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| const output = childProcess.execSync("go build -x -o ./extra/healthcheck ./extra/healthcheck.go").toString("utf8"); | ||||
| console.log(output); | ||||
|  | ||||
							
								
								
									
										33
									
								
								extra/checkout-pr.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								extra/checkout-pr.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| const childProcess = require("child_process"); | ||||
|  | ||||
| if (!process.env.UPTIME_KUMA_GH_REPO) { | ||||
|     console.error("Please set a repo to the environment variable 'UPTIME_KUMA_GH_REPO' (e.g. mhkarimi1383:goalert-notification)"); | ||||
|     process.exit(1); | ||||
| } | ||||
|  | ||||
| let inputArray = process.env.UPTIME_KUMA_GH_REPO.split(":"); | ||||
|  | ||||
| if (inputArray.length !== 2) { | ||||
|     console.error("Invalid format. Please set a repo to the environment variable 'UPTIME_KUMA_GH_REPO' (e.g. mhkarimi1383:goalert-notification)"); | ||||
| } | ||||
|  | ||||
| let name = inputArray[0]; | ||||
| let branch = inputArray[1]; | ||||
|  | ||||
| console.log("Checkout pr"); | ||||
|  | ||||
| // Checkout the pr | ||||
| let result = childProcess.spawnSync("git", [ "remote", "add", name, `https://github.com/${name}/uptime-kuma` ]); | ||||
|  | ||||
| console.log(result.stdout.toString()); | ||||
| console.error(result.stderr.toString()); | ||||
|  | ||||
| result = childProcess.spawnSync("git", [ "fetch", name, branch ]); | ||||
|  | ||||
| console.log(result.stdout.toString()); | ||||
| console.error(result.stderr.toString()); | ||||
|  | ||||
| result = childProcess.spawnSync("git", [ "checkout", `${name}/${branch}`, "--force" ]); | ||||
|  | ||||
| console.log(result.stdout.toString()); | ||||
| console.error(result.stderr.toString()); | ||||
							
								
								
									
										77
									
								
								extra/healthcheck.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								extra/healthcheck.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"crypto/tls" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	isFreeBSD := runtime.GOOS == "freebsd" | ||||
|  | ||||
| 	// process.env.NODE_TLS_REJECT_UNAUTHORIZED = "0"; | ||||
| 	http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{ | ||||
| 		InsecureSkipVerify: true, | ||||
| 	} | ||||
|  | ||||
| 	client := http.Client{ | ||||
| 		Timeout: 28 * time.Second, | ||||
| 	} | ||||
|  | ||||
| 	sslKey := os.Getenv("UPTIME_KUMA_SSL_KEY") | ||||
| 	if len(sslKey) == 0 { | ||||
| 		sslKey = os.Getenv("SSL_KEY") | ||||
| 	} | ||||
|  | ||||
| 	sslCert := os.Getenv("UPTIME_KUMA_SSL_CERT") | ||||
| 	if len(sslCert) == 0 { | ||||
| 		sslCert = os.Getenv("SSL_CERT") | ||||
| 	} | ||||
|  | ||||
| 	hostname := os.Getenv("UPTIME_KUMA_HOST") | ||||
| 	if len(hostname) == 0 && !isFreeBSD { | ||||
| 		hostname = os.Getenv("HOST") | ||||
| 	} | ||||
| 	if len(hostname) == 0 { | ||||
| 		hostname = "127.0.0.1" | ||||
| 	} | ||||
|  | ||||
| 	port := os.Getenv("UPTIME_KUMA_PORT") | ||||
| 	if len(port) == 0 { | ||||
| 		port = os.Getenv("PORT") | ||||
| 	} | ||||
| 	if len(port) == 0 { | ||||
| 		port = "3001" | ||||
| 	} | ||||
|  | ||||
| 	protocol := "" | ||||
| 	if len(sslKey) != 0 && len(sslCert) != 0 { | ||||
| 		protocol = "https" | ||||
| 	} else { | ||||
| 		protocol = "http" | ||||
| 	} | ||||
|  | ||||
| 	url := protocol + "://" + hostname + ":" + port | ||||
|  | ||||
| 	log.Println("Checking " + url) | ||||
| 	resp, err := client.Get(url) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		log.Fatalln(err) | ||||
| 	} | ||||
|  | ||||
| 	defer resp.Body.Close() | ||||
|  | ||||
| 	_, err = ioutil.ReadAll(resp.Body) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		log.Fatalln(err) | ||||
| 	} | ||||
|  | ||||
| 	log.Printf("Health Check OK [Res Code: %d]\n", resp.StatusCode) | ||||
|  | ||||
| } | ||||
| @@ -1,4 +1,5 @@ | ||||
| /* | ||||
|  * ⚠️ Deprecated: Changed to healthcheck.go, it will be deleted in the future. | ||||
|  * This script should be run after a period of time (180s), because the server may need some time to prepare. | ||||
|  */ | ||||
| const { FBSD } = require("../server/util-server"); | ||||
|   | ||||
| @@ -5,7 +5,7 @@ const util = require("../src/util"); | ||||
| util.polyfill(); | ||||
|  | ||||
| const oldVersion = pkg.version; | ||||
| const newVersion = oldVersion + "-nightly"; | ||||
| const newVersion = oldVersion + "-nightly-" + util.genSecret(8); | ||||
|  | ||||
| console.log("Old Version: " + oldVersion); | ||||
| console.log("New Version: " + newVersion); | ||||
|   | ||||
| @@ -1,51 +1,45 @@ | ||||
| // Need to use ES6 to read language files | ||||
|  | ||||
| import fs from "fs"; | ||||
| import path from "path"; | ||||
| import util from "util"; | ||||
| import rmSync from "../fs-rmSync.js"; | ||||
|  | ||||
| // https://stackoverflow.com/questions/13786160/copy-folder-recursively-in-node-js | ||||
| /** | ||||
|  * Look ma, it's cp -R. | ||||
|  * @param {string} src  The path to the thing to copy. | ||||
|  * @param {string} dest The path to the new copy. | ||||
|  * Copy across the required language files | ||||
|  * Creates a local directory (./languages) and copies the required files | ||||
|  * into it. | ||||
|  * @param {string} langCode Code of language to update. A file will be | ||||
|  * created with this code if one does not already exist | ||||
|  * @param {string} baseLang The second base language file to copy. This | ||||
|  * will be ignored if set to "en" as en.js is copied by default | ||||
|  */ | ||||
| const copyRecursiveSync = function (src, dest) { | ||||
|     let exists = fs.existsSync(src); | ||||
|     let stats = exists && fs.statSync(src); | ||||
|     let isDirectory = exists && stats.isDirectory(); | ||||
| function copyFiles(langCode, baseLang) { | ||||
|     if (fs.existsSync("./languages")) { | ||||
|         rmSync("./languages", { recursive: true }); | ||||
|     } | ||||
|     fs.mkdirSync("./languages"); | ||||
|  | ||||
|     if (isDirectory) { | ||||
|         fs.mkdirSync(dest); | ||||
|         fs.readdirSync(src).forEach(function (childItemName) { | ||||
|             copyRecursiveSync(path.join(src, childItemName), | ||||
|                 path.join(dest, childItemName)); | ||||
|         }); | ||||
|     if (!fs.existsSync(`../../src/languages/${langCode}.js`)) { | ||||
|         fs.closeSync(fs.openSync(`./languages/${langCode}.js`, "a")); | ||||
|     } else { | ||||
|         fs.copyFileSync(src, dest); | ||||
|         fs.copyFileSync(`../../src/languages/${langCode}.js`, `./languages/${langCode}.js`); | ||||
|     } | ||||
|     fs.copyFileSync("../../src/languages/en.js", "./languages/en.js"); | ||||
|     if (baseLang !== "en") { | ||||
|         fs.copyFileSync(`../../src/languages/${baseLang}.js`, `./languages/${baseLang}.js`); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| console.log("Arguments:", process.argv); | ||||
| const baseLangCode = process.argv[2] || "en"; | ||||
| console.log("Base Lang: " + baseLangCode); | ||||
| if (fs.existsSync("./languages")) { | ||||
|     rmSync("./languages", { recursive: true }); | ||||
| } | ||||
| copyRecursiveSync("../../src/languages", "./languages"); | ||||
|  | ||||
| const en = (await import("./languages/en.js")).default; | ||||
| const baseLang = (await import(`./languages/${baseLangCode}.js`)).default; | ||||
| const files = fs.readdirSync("./languages"); | ||||
| console.log("Files:", files); | ||||
|  | ||||
| for (const file of files) { | ||||
|     if (! file.endsWith(".js")) { | ||||
|         console.log("Skipping " + file); | ||||
|         continue; | ||||
|     } | ||||
| /** | ||||
|  * Update the specified language file | ||||
|  * @param {string} langCode Language code to update | ||||
|  * @param {string} baseLang Second language to copy keys from | ||||
|  */ | ||||
| async function updateLanguage(langCode, baseLangCode) { | ||||
|     const en = (await import("./languages/en.js")).default; | ||||
|     const baseLang = (await import(`./languages/${baseLangCode}.js`)).default; | ||||
|  | ||||
|     let file = langCode + ".js"; | ||||
|     console.log("Processing " + file); | ||||
|     const lang = await import("./languages/" + file); | ||||
|  | ||||
| @@ -83,5 +77,20 @@ for (const file of files) { | ||||
|     fs.writeFileSync(`../../src/languages/${file}`, code); | ||||
| } | ||||
|  | ||||
| // Get command line arguments | ||||
| const baseLangCode = process.env.npm_config_baselang || "en"; | ||||
| const langCode = process.env.npm_config_language; | ||||
|  | ||||
| // We need the file to edit | ||||
| if (langCode == null) { | ||||
|     throw new Error("Argument --language=<code> must be provided"); | ||||
| } | ||||
|  | ||||
| console.log("Base Lang: " + baseLangCode); | ||||
| console.log("Updating: " + langCode); | ||||
|  | ||||
| copyFiles(langCode, baseLangCode); | ||||
| await updateLanguage(langCode, baseLangCode); | ||||
| rmSync("./languages", { recursive: true }); | ||||
|  | ||||
| console.log("Done. Fixing formatting by ESLint..."); | ||||
|   | ||||
							
								
								
									
										9518
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9518
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										96
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										96
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "uptime-kuma", | ||||
|     "version": "1.17.1", | ||||
|     "version": "1.19.0-beta.2", | ||||
|     "license": "MIT", | ||||
|     "repository": { | ||||
|         "type": "git", | ||||
| @@ -23,11 +23,9 @@ | ||||
|         "start-server": "node server/server.js", | ||||
|         "start-server-dev": "cross-env NODE_ENV=development node server/server.js", | ||||
|         "build": "vite build --config ./config/vite.config.js", | ||||
|         "test": "node test/prepare-test-server.js && node server/server.js --port=3002 --data-dir=./data/test/ --test", | ||||
|         "test": "node test/prepare-test-server.js && npm run jest-backend", | ||||
|         "test-with-build": "npm run build && npm test", | ||||
|         "jest": "node test/prepare-jest.js && npm run jest-frontend && npm run jest-backend", | ||||
|         "jest-frontend": "cross-env TEST_FRONTEND=1 jest --config=./config/jest-frontend.config.js", | ||||
|         "jest-backend": "cross-env TEST_BACKEND=1 jest --config=./config/jest-backend.config.js", | ||||
|         "jest-backend": "cross-env TEST_BACKEND=1 jest --runInBand --detectOpenHandles --forceExit --config=./config/jest-backend.config.js", | ||||
|         "tsc": "tsc", | ||||
|         "vite-preview-dist": "vite preview --host --config ./config/vite.config.js", | ||||
|         "build-docker": "npm run build && npm run build-docker-debian && npm run build-docker-debian-mariadb", | ||||
| @@ -36,8 +34,9 @@ | ||||
|         "build-docker-debian-mariadb": "node ./extra/env2arg.js docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:mariadb -t louislam/uptime-kuma:1-mariadb -t louislam/uptime-kuma:$VERSION-mariadb --target mariadb . --push", | ||||
|         "build-docker-nightly": "npm run build && docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:nightly --target nightly . --push", | ||||
|         "build-docker-nightly-amd64": "docker buildx build -f docker/dockerfile --platform linux/amd64 -t louislam/uptime-kuma:nightly-amd64 --target nightly . --push --progress plain", | ||||
|         "build-docker-pr-test": "docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64 -t louislam/uptime-kuma:pr-test --target pr-test . --push", | ||||
|         "upload-artifacts": "docker buildx build -f docker/dockerfile --platform linux/amd64 -t louislam/uptime-kuma:upload-artifact --build-arg VERSION --build-arg GITHUB_TOKEN --target upload-artifact . --progress plain", | ||||
|         "setup": "git checkout 1.17.1 && npm ci --production && npm run download-dist", | ||||
|         "setup": "git checkout 1.18.5 && npm ci --production && npm run download-dist", | ||||
|         "download-dist": "node extra/download-dist.js", | ||||
|         "mark-as-nightly": "node extra/mark-as-nightly.js", | ||||
|         "reset-password": "node extra/reset-password.js", | ||||
| @@ -49,55 +48,66 @@ | ||||
|         "test-nodejs16": "docker build --progress plain -f test/ubuntu-nodejs16.dockerfile .", | ||||
|         "simple-dns-server": "node extra/simple-dns-server.js", | ||||
|         "simple-mqtt-server": "node extra/simple-mqtt-server.js", | ||||
|         "update-language-files-with-base-lang": "cd extra/update-language-files && node index.js %npm_config_base_lang% && eslint ../../src/languages/**.js --fix", | ||||
|         "update-language-files": "cd extra/update-language-files && node index.js && eslint ../../src/languages/**.js --fix", | ||||
|         "update-language-files": "cd extra/update-language-files && node index.js && cross-env-shell eslint ../../src/languages/$npm_config_language.js --fix", | ||||
|         "ncu-patch": "npm-check-updates -u -t patch", | ||||
|         "release-final": "node extra/update-version.js && npm run build-docker && node ./extra/press-any-key.js && npm run upload-artifacts && node ./extra/update-wiki-version.js", | ||||
|         "release-beta": "node extra/beta/update-version.js && npm run build && node ./extra/env2arg.js docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:$VERSION -t louislam/uptime-kuma:beta .  --target release --push && node ./extra/press-any-key.js && npm run upload-artifacts", | ||||
|         "git-remove-tag": "git tag -d", | ||||
|         "build-dist-and-restart": "npm run build && npm run start-server-dev" | ||||
|         "build-dist-and-restart": "npm run build && npm run start-server-dev", | ||||
|         "start-pr-test": "node extra/checkout-pr.js && npm install && npm run dev", | ||||
|         "cy:test": "node test/prepare-test-server.js && node server/server.js --port=3002 --data-dir=./data/test/ --e2e", | ||||
|         "cy:run": "npx cypress run --browser chrome --headless --config-file ./config/cypress.config.js", | ||||
|         "cypress-open": "concurrently -k -r \"node test/prepare-test-server.js && node server/server.js --port=3002 --data-dir=./data/test/\" \"cypress open --config-file ./config/cypress.config.js\"", | ||||
|         "build-healthcheck-armv7": "cross-env GOOS=linux GOARCH=arm GOARM=7 go build -x -o ./extra/healthcheck-armv7 ./extra/healthcheck.go" | ||||
|     }, | ||||
|     "dependencies": { | ||||
|         "@louislam/sqlite3": "~15.0.6", | ||||
|         "@grpc/grpc-js": "~1.7.3", | ||||
|         "@louislam/sqlite3": "15.1.2", | ||||
|         "args-parser": "~1.3.0", | ||||
|         "axios": "~0.26.1", | ||||
|         "axios-ntlm": "^1.3.0", | ||||
|         "badge-maker": "^3.3.1", | ||||
|         "axios": "~0.27.0", | ||||
|         "axios-ntlm": "1.3.0", | ||||
|         "badge-maker": "~3.3.1", | ||||
|         "bcryptjs": "~2.4.3", | ||||
|         "bree": "~7.1.5", | ||||
|         "cacheable-lookup": "~6.0.4", | ||||
|         "chardet": "^1.3.0", | ||||
|         "chardet": "~1.4.0", | ||||
|         "check-password-strength": "^2.0.5", | ||||
|         "cheerio": "^1.0.0-rc.10", | ||||
|         "chroma-js": "^2.1.2", | ||||
|         "cheerio": "~1.0.0-rc.12", | ||||
|         "chroma-js": "~2.4.2", | ||||
|         "command-exists": "~1.2.9", | ||||
|         "compare-versions": "~3.6.0", | ||||
|         "compression": "^1.7.4", | ||||
|         "dayjs": "^1.11.0", | ||||
|         "compression": "~1.7.4", | ||||
|         "dayjs": "~1.11.5", | ||||
|         "express": "~4.17.3", | ||||
|         "express-basic-auth": "~1.2.1", | ||||
|         "express-static-gzip": "^2.1.7", | ||||
|         "express-static-gzip": "~2.1.7", | ||||
|         "form-data": "~4.0.0", | ||||
|         "http-graceful-shutdown": "~3.1.7", | ||||
|         "http-proxy-agent": "^5.0.0", | ||||
|         "https-proxy-agent": "^5.0.0", | ||||
|         "iconv-lite": "^0.6.3", | ||||
|         "http-proxy-agent": "~5.0.0", | ||||
|         "https-proxy-agent": "~5.0.1", | ||||
|         "iconv-lite": "~0.6.3", | ||||
|         "jsesc": "~3.0.2", | ||||
|         "jsonwebtoken": "~8.5.1", | ||||
|         "jwt-decode": "^3.1.2", | ||||
|         "limiter": "^2.1.0", | ||||
|         "mqtt": "^4.2.8", | ||||
|         "mssql": "^8.1.0", | ||||
|         "jwt-decode": "~3.1.2", | ||||
|         "limiter": "~2.1.0", | ||||
|         "mqtt": "~4.3.7", | ||||
|         "mssql": "~8.1.4", | ||||
|         "mysql2": "~2.3.3", | ||||
|         "node-cloudflared-tunnel": "~1.0.9", | ||||
|         "node-radius-client": "~1.0.0", | ||||
|         "nodemailer": "~6.6.5", | ||||
|         "notp": "~2.0.3", | ||||
|         "password-hash": "~1.2.2", | ||||
|         "pg": "~8.8.0", | ||||
|         "pg-connection-string": "~2.5.0", | ||||
|         "prom-client": "~13.2.0", | ||||
|         "prometheus-api-metrics": "~3.2.1", | ||||
|         "protobufjs": "~7.1.1", | ||||
|         "redbean-node": "0.1.4", | ||||
|         "socket.io": "~4.4.1", | ||||
|         "socket.io-client": "~4.4.1", | ||||
|         "socket.io": "~4.5.3", | ||||
|         "socket.io-client": "~4.5.3", | ||||
|         "socks-proxy-agent": "6.1.1", | ||||
|         "tar": "^6.1.11", | ||||
|         "tar": "~6.1.11", | ||||
|         "tcp-ping": "~0.1.1", | ||||
|         "thirty-two": "~1.0.2" | ||||
|     }, | ||||
| @@ -111,46 +121,48 @@ | ||||
|         "@fortawesome/vue-fontawesome": "~3.0.0-5", | ||||
|         "@popperjs/core": "~2.10.2", | ||||
|         "@types/bootstrap": "~5.1.9", | ||||
|         "@vitejs/plugin-legacy": "~1.8.2", | ||||
|         "@vitejs/plugin-vue": "~2.3.3", | ||||
|         "@vitejs/plugin-legacy": "~2.1.0", | ||||
|         "@vitejs/plugin-vue": "~3.1.0", | ||||
|         "@vue/compiler-sfc": "~3.2.36", | ||||
|         "@vuepic/vue-datepicker": "~3.4.8", | ||||
|         "aedes": "^0.46.3", | ||||
|         "babel-plugin-rewire": "~1.2.0", | ||||
|         "bootstrap": "5.1.3", | ||||
|         "chart.js": "~3.6.2", | ||||
|         "chartjs-adapter-dayjs": "~1.0.0", | ||||
|         "concurrently": "^7.1.0", | ||||
|         "core-js": "~3.18.3", | ||||
|         "core-js": "~3.26.1", | ||||
|         "cross-env": "~7.0.3", | ||||
|         "cypress": "^10.1.0", | ||||
|         "delay": "^5.0.0", | ||||
|         "dns2": "~2.0.1", | ||||
|         "eslint": "~8.14.0", | ||||
|         "eslint-plugin-vue": "~8.7.1", | ||||
|         "favico.js": "^0.3.10", | ||||
|         "favico.js": "~0.3.10", | ||||
|         "jest": "~27.2.5", | ||||
|         "jest-puppeteer": "~6.0.3", | ||||
|         "postcss-html": "^1.3.1", | ||||
|         "postcss-rtlcss": "~3.4.1", | ||||
|         "postcss-scss": "~4.0.3", | ||||
|         "prismjs": "^1.27.0", | ||||
|         "puppeteer": "~13.1.3", | ||||
|         "postcss-html": "~1.5.0", | ||||
|         "postcss-rtlcss": "~3.7.2", | ||||
|         "postcss-scss": "~4.0.4", | ||||
|         "prismjs": "~1.29.0", | ||||
|         "qrcode": "~1.5.0", | ||||
|         "rollup-plugin-visualizer": "^5.6.0", | ||||
|         "sass": "~1.42.1", | ||||
|         "stylelint": "~14.7.1", | ||||
|         "stylelint-config-standard": "~25.0.0", | ||||
|         "terser": "~5.15.0", | ||||
|         "timezones-list": "~3.0.1", | ||||
|         "typescript": "~4.4.4", | ||||
|         "v-pagination-3": "~0.1.7", | ||||
|         "vite": "~2.9.9", | ||||
|         "vite": "~3.1.0", | ||||
|         "vite-plugin-compression": "^0.5.1", | ||||
|         "vue": "next", | ||||
|         "vue-chart-3": "3.0.9", | ||||
|         "vue-confirm-dialog": "~1.0.2", | ||||
|         "vue-contenteditable": "~3.0.4", | ||||
|         "vue-i18n": "~9.1.9", | ||||
|         "vue-i18n": "~9.2.2", | ||||
|         "vue-image-crop-upload": "~3.0.3", | ||||
|         "vue-multiselect": "~3.0.0-alpha.2", | ||||
|         "vue-prism-editor": "^2.0.0-alpha.2", | ||||
|         "vue-prism-editor": "~2.0.0-alpha.2", | ||||
|         "vue-qrcode": "~1.0.0", | ||||
|         "vue-router": "~4.0.14", | ||||
|         "vue-toastification": "~2.0.0-rc.5", | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| const https = require("https"); | ||||
| const http = require("http"); | ||||
| const CacheableLookup = require("cacheable-lookup"); | ||||
| const { Settings } = require("./settings"); | ||||
| const { log } = require("../src/util"); | ||||
|  | ||||
| class CacheableDnsHttpAgent { | ||||
|  | ||||
| @@ -9,12 +11,30 @@ class CacheableDnsHttpAgent { | ||||
|     static httpAgentList = {}; | ||||
|     static httpsAgentList = {}; | ||||
|  | ||||
|     static enable = false; | ||||
|  | ||||
|     /** | ||||
|      * Register cacheable to global agents | ||||
|      * Register/Disable cacheable to global agents | ||||
|      */ | ||||
|     static registerGlobalAgent() { | ||||
|         this.cacheable.install(http.globalAgent); | ||||
|         this.cacheable.install(https.globalAgent); | ||||
|     static async update() { | ||||
|         log.debug("CacheableDnsHttpAgent", "update"); | ||||
|         let isEnable = await Settings.get("dnsCache"); | ||||
|  | ||||
|         if (isEnable !== this.enable) { | ||||
|             log.debug("CacheableDnsHttpAgent", "value changed"); | ||||
|  | ||||
|             if (isEnable) { | ||||
|                 log.debug("CacheableDnsHttpAgent", "enable"); | ||||
|                 this.cacheable.install(http.globalAgent); | ||||
|                 this.cacheable.install(https.globalAgent); | ||||
|             } else { | ||||
|                 log.debug("CacheableDnsHttpAgent", "disable"); | ||||
|                 this.cacheable.uninstall(http.globalAgent); | ||||
|                 this.cacheable.uninstall(https.globalAgent); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this.enable = isEnable; | ||||
|     } | ||||
|  | ||||
|     static install(agent) { | ||||
| @@ -26,6 +46,10 @@ class CacheableDnsHttpAgent { | ||||
|      * @return {https.Agent} | ||||
|      */ | ||||
|     static getHttpsAgent(agentOptions) { | ||||
|         if (!this.enable) { | ||||
|             return new https.Agent(agentOptions); | ||||
|         } | ||||
|  | ||||
|         let key = JSON.stringify(agentOptions); | ||||
|         if (!(key in this.httpsAgentList)) { | ||||
|             this.httpsAgentList[key] = new https.Agent(agentOptions); | ||||
| @@ -39,6 +63,10 @@ class CacheableDnsHttpAgent { | ||||
|      * @return {https.Agents} | ||||
|      */ | ||||
|     static getHttpAgent(agentOptions) { | ||||
|         if (!this.enable) { | ||||
|             return new http.Agent(agentOptions); | ||||
|         } | ||||
|  | ||||
|         let key = JSON.stringify(agentOptions); | ||||
|         if (!(key in this.httpAgentList)) { | ||||
|             this.httpAgentList[key] = new http.Agent(agentOptions); | ||||
|   | ||||
| @@ -25,7 +25,7 @@ exports.startInterval = () => { | ||||
|             let checkBeta = await setting("checkBeta"); | ||||
|  | ||||
|             if (checkBeta && res.data.beta) { | ||||
|                 if (compareVersions.compare(res.data.beta, res.data.beta, ">")) { | ||||
|                 if (compareVersions.compare(res.data.beta, res.data.slow, ">")) { | ||||
|                     exports.latestVersion = res.data.beta; | ||||
|                     return; | ||||
|                 } | ||||
|   | ||||
| @@ -4,7 +4,8 @@ | ||||
| const { TimeLogger } = require("../src/util"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { UptimeKumaServer } = require("./uptime-kuma-server"); | ||||
| const io = UptimeKumaServer.getInstance().io; | ||||
| const server = UptimeKumaServer.getInstance(); | ||||
| const io = server.io; | ||||
| const { setting } = require("./util-server"); | ||||
| const checkVersion = require("./check-version"); | ||||
|  | ||||
| @@ -121,14 +122,41 @@ async function sendInfo(socket) { | ||||
|     socket.emit("info", { | ||||
|         version: checkVersion.version, | ||||
|         latestVersion: checkVersion.latestVersion, | ||||
|         primaryBaseURL: await setting("primaryBaseURL") | ||||
|         primaryBaseURL: await setting("primaryBaseURL"), | ||||
|         serverTimezone: await server.getTimezone(), | ||||
|         serverTimezoneOffset: server.getTimezoneOffset(), | ||||
|     }); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Send list of docker hosts to client | ||||
|  * @param {Socket} socket Socket.io socket instance | ||||
|  * @returns {Promise<Bean[]>} | ||||
|  */ | ||||
| async function sendDockerHostList(socket) { | ||||
|     const timeLogger = new TimeLogger(); | ||||
|  | ||||
|     let result = []; | ||||
|     let list = await R.find("docker_host", " user_id = ? ", [ | ||||
|         socket.userID, | ||||
|     ]); | ||||
|  | ||||
|     for (let bean of list) { | ||||
|         result.push(bean.toJSON()); | ||||
|     } | ||||
|  | ||||
|     io.to(socket.userID).emit("dockerHostList", result); | ||||
|  | ||||
|     timeLogger.print("Send Docker Host List"); | ||||
|  | ||||
|     return list; | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     sendNotificationList, | ||||
|     sendImportantHeartbeatList, | ||||
|     sendHeartbeatList, | ||||
|     sendProxyList, | ||||
|     sendInfo, | ||||
|     sendDockerHostList | ||||
| }; | ||||
|   | ||||
| @@ -53,6 +53,7 @@ class Database { | ||||
|         "patch-2fa-invalidate-used-token.sql": true, | ||||
|         "patch-notification_sent_history.sql": true, | ||||
|         "patch-monitor-basic-auth.sql": true, | ||||
|         "patch-add-docker-columns.sql": true, | ||||
|         "patch-status-page.sql": true, | ||||
|         "patch-proxy.sql": true, | ||||
|         "patch-monitor-expiry-notification.sql": true, | ||||
| @@ -61,6 +62,10 @@ class Database { | ||||
|         "patch-add-clickable-status-page-link.sql": true, | ||||
|         "patch-add-sqlserver-monitor.sql": true, | ||||
|         "patch-add-other-auth.sql": { parents: [ "patch-monitor-basic-auth.sql" ] }, | ||||
|         "patch-grpc-monitor.sql": true, | ||||
|         "patch-add-radius-monitor.sql": true, | ||||
|         "patch-monitor-add-resend-interval.sql": true, | ||||
|         "patch-maintenance-table2.sql": true, | ||||
|     }; | ||||
|  | ||||
|     /** | ||||
|   | ||||
							
								
								
									
										118
									
								
								server/docker.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								server/docker.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| const axios = require("axios"); | ||||
| const { R } = require("redbean-node"); | ||||
| const version = require("../package.json").version; | ||||
| const https = require("https"); | ||||
|  | ||||
| class DockerHost { | ||||
|     /** | ||||
|      * Save a docker host | ||||
|      * @param {Object} dockerHost Docker host to save | ||||
|      * @param {?number} dockerHostID ID of the docker host to update | ||||
|      * @param {number} userID ID of the user who adds the docker host | ||||
|      * @returns {Promise<Bean>} | ||||
|      */ | ||||
|     static async save(dockerHost, dockerHostID, userID) { | ||||
|         let bean; | ||||
|  | ||||
|         if (dockerHostID) { | ||||
|             bean = await R.findOne("docker_host", " id = ? AND user_id = ? ", [ dockerHostID, userID ]); | ||||
|  | ||||
|             if (!bean) { | ||||
|                 throw new Error("docker host not found"); | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
|             bean = R.dispense("docker_host"); | ||||
|         } | ||||
|  | ||||
|         bean.user_id = userID; | ||||
|         bean.docker_daemon = dockerHost.dockerDaemon; | ||||
|         bean.docker_type = dockerHost.dockerType; | ||||
|         bean.name = dockerHost.name; | ||||
|  | ||||
|         await R.store(bean); | ||||
|  | ||||
|         return bean; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Delete a Docker host | ||||
|      * @param {number} dockerHostID ID of the Docker host to delete | ||||
|      * @param {number} userID ID of the user who created the Docker host | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     static async delete(dockerHostID, userID) { | ||||
|         let bean = await R.findOne("docker_host", " id = ? AND user_id = ? ", [ dockerHostID, userID ]); | ||||
|  | ||||
|         if (!bean) { | ||||
|             throw new Error("docker host not found"); | ||||
|         } | ||||
|  | ||||
|         // Delete removed proxy from monitors if exists | ||||
|         await R.exec("UPDATE monitor SET docker_host = null WHERE docker_host = ?", [ dockerHostID ]); | ||||
|  | ||||
|         await R.trash(bean); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Fetches the amount of containers on the Docker host | ||||
|      * @param {Object} dockerHost Docker host to check for | ||||
|      * @returns {number} Total amount of containers on the host | ||||
|      */ | ||||
|     static async testDockerHost(dockerHost) { | ||||
|         const options = { | ||||
|             url: "/containers/json?all=true", | ||||
|             headers: { | ||||
|                 "Accept": "*/*", | ||||
|                 "User-Agent": "Uptime-Kuma/" + version | ||||
|             }, | ||||
|             httpsAgent: new https.Agent({ | ||||
|                 maxCachedSessions: 0,      // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940) | ||||
|                 rejectUnauthorized: false, | ||||
|             }), | ||||
|         }; | ||||
|  | ||||
|         if (dockerHost.dockerType === "socket") { | ||||
|             options.socketPath = dockerHost.dockerDaemon; | ||||
|         } else if (dockerHost.dockerType === "tcp") { | ||||
|             options.baseURL = DockerHost.patchDockerURL(dockerHost.dockerDaemon); | ||||
|         } | ||||
|  | ||||
|         let res = await axios.request(options); | ||||
|  | ||||
|         if (Array.isArray(res.data)) { | ||||
|  | ||||
|             if (res.data.length > 1) { | ||||
|  | ||||
|                 if ("ImageID" in res.data[0]) { | ||||
|                     return res.data.length; | ||||
|                 } else { | ||||
|                     throw new Error("Invalid Docker response, is it Docker really a daemon?"); | ||||
|                 } | ||||
|  | ||||
|             } else { | ||||
|                 return res.data.length; | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
|             throw new Error("Invalid Docker response, is it Docker really a daemon?"); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Since axios 0.27.X, it does not accept `tcp://` protocol. | ||||
|      * Change it to `http://` on the fly in order to fix it. (https://github.com/louislam/uptime-kuma/issues/2165) | ||||
|      */ | ||||
|     static patchDockerURL(url) { | ||||
|         if (typeof url === "string") { | ||||
|             // Replace the first occurrence only with g | ||||
|             return url.replace(/tcp:\/\//g, "http://"); | ||||
|         } | ||||
|         return url; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     DockerHost, | ||||
| }; | ||||
							
								
								
									
										19
									
								
								server/model/docker_host.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								server/model/docker_host.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
|  | ||||
| class DockerHost extends BeanModel { | ||||
|     /** | ||||
|      * Returns an object that ready to parse to JSON | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     toJSON() { | ||||
|         return { | ||||
|             id: this.id, | ||||
|             userID: this.user_id, | ||||
|             dockerDaemon: this.docker_daemon, | ||||
|             dockerType: this.docker_type, | ||||
|             name: this.name, | ||||
|         }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = DockerHost; | ||||
| @@ -1,8 +1,3 @@ | ||||
| const dayjs = require("dayjs"); | ||||
| const utc = require("dayjs/plugin/utc"); | ||||
| let timezone = require("dayjs/plugin/timezone"); | ||||
| dayjs.extend(utc); | ||||
| dayjs.extend(timezone); | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
|  | ||||
| /** | ||||
| @@ -10,6 +5,7 @@ const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
|  *      0 = DOWN | ||||
|  *      1 = UP | ||||
|  *      2 = PENDING | ||||
|  *      3 = MAINTENANCE | ||||
|  */ | ||||
| class Heartbeat extends BeanModel { | ||||
|  | ||||
|   | ||||
							
								
								
									
										217
									
								
								server/model/maintenance.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										217
									
								
								server/model/maintenance.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,217 @@ | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { parseTimeObject, parseTimeFromTimeObject, utcToLocal, localToUTC, log } = require("../../src/util"); | ||||
| const { timeObjectToUTC, timeObjectToLocal } = require("../util-server"); | ||||
| const { R } = require("redbean-node"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| class Maintenance extends BeanModel { | ||||
|  | ||||
|     /** | ||||
|      * Return an object that ready to parse to JSON for public | ||||
|      * Only show necessary data to public | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     async toPublicJSON() { | ||||
|  | ||||
|         let dateRange = []; | ||||
|         if (this.start_date) { | ||||
|             dateRange.push(utcToLocal(this.start_date)); | ||||
|             if (this.end_date) { | ||||
|                 dateRange.push(utcToLocal(this.end_date)); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let timeRange = []; | ||||
|         let startTime = timeObjectToLocal(parseTimeObject(this.start_time)); | ||||
|         timeRange.push(startTime); | ||||
|         let endTime = timeObjectToLocal(parseTimeObject(this.end_time)); | ||||
|         timeRange.push(endTime); | ||||
|  | ||||
|         let obj = { | ||||
|             id: this.id, | ||||
|             title: this.title, | ||||
|             description: this.description, | ||||
|             strategy: this.strategy, | ||||
|             intervalDay: this.interval_day, | ||||
|             active: !!this.active, | ||||
|             dateRange: dateRange, | ||||
|             timeRange: timeRange, | ||||
|             weekdays: (this.weekdays) ? JSON.parse(this.weekdays) : [], | ||||
|             daysOfMonth: (this.days_of_month) ? JSON.parse(this.days_of_month) : [], | ||||
|             timeslotList: [], | ||||
|         }; | ||||
|  | ||||
|         const timeslotList = await this.getTimeslotList(); | ||||
|  | ||||
|         for (let timeslot of timeslotList) { | ||||
|             obj.timeslotList.push(await timeslot.toPublicJSON()); | ||||
|         } | ||||
|  | ||||
|         if (!Array.isArray(obj.weekdays)) { | ||||
|             obj.weekdays = []; | ||||
|         } | ||||
|  | ||||
|         if (!Array.isArray(obj.daysOfMonth)) { | ||||
|             obj.daysOfMonth = []; | ||||
|         } | ||||
|  | ||||
|         // Maintenance Status | ||||
|         if (!obj.active) { | ||||
|             obj.status = "inactive"; | ||||
|         } else if (obj.strategy === "manual") { | ||||
|             obj.status = "under-maintenance"; | ||||
|         } else if (obj.timeslotList.length > 0) { | ||||
|             let currentTimestamp = dayjs().unix(); | ||||
|  | ||||
|             for (let timeslot of obj.timeslotList) { | ||||
|                 if (dayjs.utc(timeslot.startDate).unix() <= currentTimestamp && dayjs.utc(timeslot.endDate).unix() >= currentTimestamp) { | ||||
|                     log.debug("timeslot", "Timeslot ID: " + timeslot.id); | ||||
|                     log.debug("timeslot", "currentTimestamp:" + currentTimestamp); | ||||
|                     log.debug("timeslot", "timeslot.start_date:" + dayjs.utc(timeslot.startDate).unix()); | ||||
|                     log.debug("timeslot", "timeslot.end_date:" + dayjs.utc(timeslot.endDate).unix()); | ||||
|  | ||||
|                     obj.status = "under-maintenance"; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (!obj.status) { | ||||
|                 obj.status = "scheduled"; | ||||
|             } | ||||
|         } else if (obj.timeslotList.length === 0) { | ||||
|             obj.status = "ended"; | ||||
|         } else { | ||||
|             obj.status = "unknown"; | ||||
|         } | ||||
|  | ||||
|         return obj; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Only get future or current timeslots only | ||||
|      * @returns {Promise<[]>} | ||||
|      */ | ||||
|     async getTimeslotList() { | ||||
|         return R.convertToBeans("maintenance_timeslot", await R.getAll(` | ||||
|             SELECT maintenance_timeslot.* | ||||
|             FROM maintenance_timeslot, maintenance | ||||
|             WHERE maintenance_timeslot.maintenance_id = maintenance.id | ||||
|             AND maintenance.id = ? | ||||
|             AND ${Maintenance.getActiveAndFutureMaintenanceSQLCondition()} | ||||
|         `, [ | ||||
|             this.id | ||||
|         ])); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Return an object that ready to parse to JSON | ||||
|      * @param {string} timezone If not specified, the timeRange will be in UTC | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     async toJSON(timezone = null) { | ||||
|         return this.toPublicJSON(timezone); | ||||
|     } | ||||
|  | ||||
|     getDayOfWeekList() { | ||||
|         log.debug("timeslot", "List: " + this.weekdays); | ||||
|         return JSON.parse(this.weekdays).sort(function (a, b) { | ||||
|             return a - b; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     getDayOfMonthList() { | ||||
|         return JSON.parse(this.days_of_month).sort(function (a, b) { | ||||
|             return a - b; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     getStartDateTime() { | ||||
|         let startOfTheDay = dayjs.utc(this.start_date).format("HH:mm"); | ||||
|         log.debug("timeslot", "startOfTheDay: " + startOfTheDay); | ||||
|  | ||||
|         // Start Time | ||||
|         let startTimeSecond = dayjs.utc(this.start_time, "HH:mm").diff(dayjs.utc(startOfTheDay, "HH:mm"), "second"); | ||||
|         log.debug("timeslot", "startTime: " + startTimeSecond); | ||||
|  | ||||
|         // Bake StartDate + StartTime = Start DateTime | ||||
|         return dayjs.utc(this.start_date).add(startTimeSecond, "second"); | ||||
|     } | ||||
|  | ||||
|     getDuration() { | ||||
|         let duration = dayjs.utc(this.end_time, "HH:mm").diff(dayjs.utc(this.start_time, "HH:mm"), "second"); | ||||
|         // Add 24hours if it is across day | ||||
|         if (duration < 0) { | ||||
|             duration += 24 * 3600; | ||||
|         } | ||||
|         return duration; | ||||
|     } | ||||
|  | ||||
|     static jsonToBean(bean, obj) { | ||||
|         if (obj.id) { | ||||
|             bean.id = obj.id; | ||||
|         } | ||||
|  | ||||
|         // Apply timezone offset to timeRange, as it cannot apply automatically. | ||||
|         if (obj.timeRange[0]) { | ||||
|             timeObjectToUTC(obj.timeRange[0]); | ||||
|             if (obj.timeRange[1]) { | ||||
|                 timeObjectToUTC(obj.timeRange[1]); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         bean.title = obj.title; | ||||
|         bean.description = obj.description; | ||||
|         bean.strategy = obj.strategy; | ||||
|         bean.interval_day = obj.intervalDay; | ||||
|         bean.active = obj.active; | ||||
|  | ||||
|         if (obj.dateRange[0]) { | ||||
|             bean.start_date = localToUTC(obj.dateRange[0]); | ||||
|  | ||||
|             if (obj.dateRange[1]) { | ||||
|                 bean.end_date = localToUTC(obj.dateRange[1]); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         bean.start_time = parseTimeFromTimeObject(obj.timeRange[0]); | ||||
|         bean.end_time = parseTimeFromTimeObject(obj.timeRange[1]); | ||||
|  | ||||
|         bean.weekdays = JSON.stringify(obj.weekdays); | ||||
|         bean.days_of_month = JSON.stringify(obj.daysOfMonth); | ||||
|  | ||||
|         return bean; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SQL conditions for active maintenance | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     static getActiveMaintenanceSQLCondition() { | ||||
|         return ` | ||||
|             ( | ||||
|                 (maintenance_timeslot.start_date <= DATETIME('now') | ||||
|                 AND maintenance_timeslot.end_date >= DATETIME('now') | ||||
|                 AND maintenance.active = 1) | ||||
|                 OR | ||||
|                 (maintenance.strategy = 'manual' AND active = 1) | ||||
|             ) | ||||
|         `; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SQL conditions for active and future maintenance | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     static getActiveAndFutureMaintenanceSQLCondition() { | ||||
|         return ` | ||||
|             ( | ||||
|                 ((maintenance_timeslot.end_date >= DATETIME('now') | ||||
|                 AND maintenance.active = 1) | ||||
|                 OR | ||||
|                 (maintenance.strategy = 'manual' AND active = 1)) | ||||
|             ) | ||||
|         `; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Maintenance; | ||||
							
								
								
									
										189
									
								
								server/model/maintenance_timeslot.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								server/model/maintenance_timeslot.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,189 @@ | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { R } = require("redbean-node"); | ||||
| const dayjs = require("dayjs"); | ||||
| const { log, utcToLocal, SQL_DATETIME_FORMAT_WITHOUT_SECOND, localToUTC } = require("../../src/util"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
|  | ||||
| class MaintenanceTimeslot extends BeanModel { | ||||
|  | ||||
|     async toPublicJSON() { | ||||
|         const serverTimezoneOffset = UptimeKumaServer.getInstance().getTimezoneOffset(); | ||||
|  | ||||
|         const obj = { | ||||
|             id: this.id, | ||||
|             startDate: this.start_date, | ||||
|             endDate: this.end_date, | ||||
|             startDateServerTimezone: utcToLocal(this.start_date, SQL_DATETIME_FORMAT_WITHOUT_SECOND), | ||||
|             endDateServerTimezone: utcToLocal(this.end_date, SQL_DATETIME_FORMAT_WITHOUT_SECOND), | ||||
|             serverTimezoneOffset, | ||||
|         }; | ||||
|  | ||||
|         return obj; | ||||
|     } | ||||
|  | ||||
|     async toJSON() { | ||||
|         return await this.toPublicJSON(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * @param {Maintenance} maintenance | ||||
|      * @param {dayjs} minDate (For recurring type only) Generate a next timeslot from this date. | ||||
|      * @param {boolean} removeExist Remove existing timeslot before create | ||||
|      * @returns {Promise<MaintenanceTimeslot>} | ||||
|      */ | ||||
|     static async generateTimeslot(maintenance, minDate = null, removeExist = false) { | ||||
|         if (removeExist) { | ||||
|             await R.exec("DELETE FROM maintenance_timeslot WHERE maintenance_id = ? ", [ | ||||
|                 maintenance.id | ||||
|             ]); | ||||
|         } | ||||
|  | ||||
|         if (maintenance.strategy === "manual") { | ||||
|             log.debug("maintenance", "No need to generate timeslot for manual type"); | ||||
|  | ||||
|         } else if (maintenance.strategy === "single") { | ||||
|             let bean = R.dispense("maintenance_timeslot"); | ||||
|             bean.maintenance_id = maintenance.id; | ||||
|             bean.start_date = maintenance.start_date; | ||||
|             bean.end_date = maintenance.end_date; | ||||
|             bean.generated_next = true; | ||||
|             return await R.store(bean); | ||||
|  | ||||
|         } else if (maintenance.strategy === "recurring-interval") { | ||||
|             // Prevent dead loop, in case interval_day is not set | ||||
|             if (!maintenance.interval_day || maintenance.interval_day <= 0) { | ||||
|                 maintenance.interval_day = 1; | ||||
|             } | ||||
|  | ||||
|             return await this.handleRecurringType(maintenance, minDate, (startDateTime) => { | ||||
|                 return startDateTime.add(maintenance.interval_day, "day"); | ||||
|             }, () => { | ||||
|                 return true; | ||||
|             }); | ||||
|  | ||||
|         } else if (maintenance.strategy === "recurring-weekday") { | ||||
|             let dayOfWeekList = maintenance.getDayOfWeekList(); | ||||
|             log.debug("timeslot", dayOfWeekList); | ||||
|  | ||||
|             if (dayOfWeekList.length <= 0) { | ||||
|                 log.debug("timeslot", "No weekdays selected?"); | ||||
|                 return null; | ||||
|             } | ||||
|  | ||||
|             const isValid = (startDateTime) => { | ||||
|                 log.debug("timeslot", "nextDateTime: " + startDateTime); | ||||
|  | ||||
|                 let day = startDateTime.local().day(); | ||||
|                 log.debug("timeslot", "nextDateTime.day(): " + day); | ||||
|  | ||||
|                 return dayOfWeekList.includes(day); | ||||
|             }; | ||||
|  | ||||
|             return await this.handleRecurringType(maintenance, minDate, (startDateTime) => { | ||||
|                 while (true) { | ||||
|                     startDateTime = startDateTime.add(1, "day"); | ||||
|  | ||||
|                     if (isValid(startDateTime)) { | ||||
|                         return startDateTime; | ||||
|                     } | ||||
|                 } | ||||
|             }, isValid); | ||||
|  | ||||
|         } else if (maintenance.strategy === "recurring-day-of-month") { | ||||
|             let dayOfMonthList = maintenance.getDayOfMonthList(); | ||||
|             if (dayOfMonthList.length <= 0) { | ||||
|                 log.debug("timeslot", "No day selected?"); | ||||
|                 return null; | ||||
|             } | ||||
|  | ||||
|             const isValid = (startDateTime) => { | ||||
|                 let day = parseInt(startDateTime.local().format("D")); | ||||
|  | ||||
|                 log.debug("timeslot", "day: " + day); | ||||
|  | ||||
|                 // Check 1-31 | ||||
|                 if (dayOfMonthList.includes(day)) { | ||||
|                     return startDateTime; | ||||
|                 } | ||||
|  | ||||
|                 // Check "lastDay1","lastDay2"... | ||||
|                 let daysInMonth = startDateTime.daysInMonth(); | ||||
|                 let lastDayList = []; | ||||
|  | ||||
|                 // Small first, e.g. 28 > 29 > 30 > 31 | ||||
|                 for (let i = 4; i >= 1; i--) { | ||||
|                     if (dayOfMonthList.includes("lastDay" + i)) { | ||||
|                         lastDayList.push(daysInMonth - i + 1); | ||||
|                     } | ||||
|                 } | ||||
|                 log.debug("timeslot", lastDayList); | ||||
|                 return lastDayList.includes(day); | ||||
|             }; | ||||
|  | ||||
|             return await this.handleRecurringType(maintenance, minDate, (startDateTime) => { | ||||
|                 while (true) { | ||||
|                     startDateTime = startDateTime.add(1, "day"); | ||||
|                     if (isValid(startDateTime)) { | ||||
|                         return startDateTime; | ||||
|                     } | ||||
|                 } | ||||
|             }, isValid); | ||||
|         } else { | ||||
|             throw new Error("Unknown maintenance strategy"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Generate a next timeslot for all recurring types | ||||
|      * @param maintenance | ||||
|      * @param minDate | ||||
|      * @param {function} nextDayCallback The logic how to get the next possible day | ||||
|      * @param {function} isValidCallback Check the day whether is matched the current strategy | ||||
|      * @returns {Promise<null|MaintenanceTimeslot>} | ||||
|      */ | ||||
|     static async handleRecurringType(maintenance, minDate, nextDayCallback, isValidCallback) { | ||||
|         let bean = R.dispense("maintenance_timeslot"); | ||||
|  | ||||
|         let duration = maintenance.getDuration(); | ||||
|         let startDateTime = maintenance.getStartDateTime(); | ||||
|         let endDateTime; | ||||
|  | ||||
|         // Keep generating from the first possible date, until it is ok | ||||
|         while (true) { | ||||
|             log.debug("timeslot", "startDateTime: " + startDateTime.format()); | ||||
|  | ||||
|             // Handling out of effective date range | ||||
|             if (startDateTime.diff(dayjs.utc(maintenance.end_date)) > 0) { | ||||
|                 log.debug("timeslot", "Out of effective date range"); | ||||
|                 return null; | ||||
|             } | ||||
|  | ||||
|             endDateTime = startDateTime.add(duration, "second"); | ||||
|  | ||||
|             // If endDateTime is out of effective date range, use the end datetime from effective date range | ||||
|             if (endDateTime.diff(dayjs.utc(maintenance.end_date)) > 0) { | ||||
|                 endDateTime = dayjs.utc(maintenance.end_date); | ||||
|             } | ||||
|  | ||||
|             // If minDate is set, the endDateTime must be bigger than it. | ||||
|             // And the endDateTime must be bigger current time | ||||
|             // Is valid under current recurring strategy | ||||
|             if ( | ||||
|                 (!minDate || endDateTime.diff(minDate) > 0) && | ||||
|                 endDateTime.diff(dayjs()) > 0 && | ||||
|                 isValidCallback(startDateTime) | ||||
|             ) { | ||||
|                 break; | ||||
|             } | ||||
|             startDateTime = nextDayCallback(startDateTime); | ||||
|         } | ||||
|  | ||||
|         bean.maintenance_id = maintenance.id; | ||||
|         bean.start_date = localToUTC(startDateTime); | ||||
|         bean.end_date = localToUTC(endDateTime); | ||||
|         bean.generated_next = false; | ||||
|         return await R.store(bean); | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = MaintenanceTimeslot; | ||||
| @@ -1,13 +1,9 @@ | ||||
| const https = require("https"); | ||||
| const dayjs = require("dayjs"); | ||||
| const utc = require("dayjs/plugin/utc"); | ||||
| let timezone = require("dayjs/plugin/timezone"); | ||||
| dayjs.extend(utc); | ||||
| dayjs.extend(timezone); | ||||
| const axios = require("axios"); | ||||
| const { Prometheus } = require("../prometheus"); | ||||
| const { log, UP, DOWN, PENDING, flipStatus, TimeLogger } = require("../../src/util"); | ||||
| const { tcping, ping, dnsResolve, checkCertificate, checkStatusCode, getTotalClientInRoom, setting, mssqlQuery, mqttAsync, setSetting, httpNtlm } = require("../util-server"); | ||||
| const { log, UP, DOWN, PENDING, MAINTENANCE, flipStatus, TimeLogger, MAX_INTERVAL_SECOND, MIN_INTERVAL_SECOND } = require("../../src/util"); | ||||
| const { tcping, ping, dnsResolve, checkCertificate, checkStatusCode, getTotalClientInRoom, setting, mssqlQuery, postgresQuery, mysqlQuery, mqttAsync, setSetting, httpNtlm, radius, grpcQuery } = require("../util-server"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { Notification } = require("../notification"); | ||||
| @@ -17,12 +13,16 @@ const version = require("../../package.json").version; | ||||
| const apicache = require("../modules/apicache"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const { CacheableDnsHttpAgent } = require("../cacheable-dns-http-agent"); | ||||
| const { DockerHost } = require("../docker"); | ||||
| const Maintenance = require("./maintenance"); | ||||
| const { UptimeCacheList } = require("../uptime-cache-list"); | ||||
|  | ||||
| /** | ||||
|  * status: | ||||
|  *      0 = DOWN | ||||
|  *      1 = UP | ||||
|  *      2 = PENDING | ||||
|  *      3 = MAINTENANCE | ||||
|  */ | ||||
| class Monitor extends BeanModel { | ||||
|  | ||||
| @@ -36,6 +36,7 @@ class Monitor extends BeanModel { | ||||
|             id: this.id, | ||||
|             name: this.name, | ||||
|             sendUrl: this.sendUrl, | ||||
|             maintenance: await Monitor.isUnderMaintenance(this.id), | ||||
|         }; | ||||
|  | ||||
|         if (this.sendUrl) { | ||||
| @@ -79,6 +80,7 @@ class Monitor extends BeanModel { | ||||
|             type: this.type, | ||||
|             interval: this.interval, | ||||
|             retryInterval: this.retryInterval, | ||||
|             resendInterval: this.resendInterval, | ||||
|             keyword: this.keyword, | ||||
|             expiryNotification: this.isEnabledExpiryNotification(), | ||||
|             ignoreTls: this.getIgnoreTls(), | ||||
| @@ -88,18 +90,23 @@ class Monitor extends BeanModel { | ||||
|             dns_resolve_type: this.dns_resolve_type, | ||||
|             dns_resolve_server: this.dns_resolve_server, | ||||
|             dns_last_result: this.dns_last_result, | ||||
|             docker_container: this.docker_container, | ||||
|             docker_host: this.docker_host, | ||||
|             proxyId: this.proxy_id, | ||||
|             notificationIDList, | ||||
|             tags: tags, | ||||
|             mqttUsername: this.mqttUsername, | ||||
|             mqttPassword: this.mqttPassword, | ||||
|             maintenance: await Monitor.isUnderMaintenance(this.id), | ||||
|             mqttTopic: this.mqttTopic, | ||||
|             mqttSuccessMessage: this.mqttSuccessMessage, | ||||
|             databaseConnectionString: this.databaseConnectionString, | ||||
|             databaseQuery: this.databaseQuery, | ||||
|             authMethod: this.authMethod, | ||||
|             authWorkstation: this.authWorkstation, | ||||
|             authDomain: this.authDomain, | ||||
|             grpcUrl: this.grpcUrl, | ||||
|             grpcProtobuf: this.grpcProtobuf, | ||||
|             grpcMethod: this.grpcMethod, | ||||
|             grpcServiceName: this.grpcServiceName, | ||||
|             grpcEnableTls: this.getGrpcEnableTls(), | ||||
|             radiusCalledStationId: this.radiusCalledStationId, | ||||
|             radiusCallingStationId: this.radiusCallingStationId, | ||||
|         }; | ||||
|  | ||||
|         if (includeSensitiveData) { | ||||
| @@ -107,12 +114,23 @@ class Monitor extends BeanModel { | ||||
|                 ...data, | ||||
|                 headers: this.headers, | ||||
|                 body: this.body, | ||||
|                 grpcBody: this.grpcBody, | ||||
|                 grpcMetadata: this.grpcMetadata, | ||||
|                 basic_auth_user: this.basic_auth_user, | ||||
|                 basic_auth_pass: this.basic_auth_pass, | ||||
|                 pushToken: this.pushToken, | ||||
|                 databaseConnectionString: this.databaseConnectionString, | ||||
|                 radiusUsername: this.radiusUsername, | ||||
|                 radiusPassword: this.radiusPassword, | ||||
|                 radiusSecret: this.radiusSecret, | ||||
|                 mqttUsername: this.mqttUsername, | ||||
|                 mqttPassword: this.mqttPassword, | ||||
|                 authWorkstation: this.authWorkstation, | ||||
|                 authDomain: this.authDomain, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         data.includeSensitiveData = includeSensitiveData; | ||||
|         return data; | ||||
|     } | ||||
|  | ||||
| @@ -157,6 +175,14 @@ class Monitor extends BeanModel { | ||||
|         return Boolean(this.upsideDown); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Parse to boolean | ||||
|      * @returns {boolean} | ||||
|      */ | ||||
|     getGrpcEnableTls() { | ||||
|         return Boolean(this.grpcEnableTls); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get accepted status codes | ||||
|      * @returns {Object} | ||||
| @@ -206,6 +232,7 @@ class Monitor extends BeanModel { | ||||
|             bean.monitor_id = this.id; | ||||
|             bean.time = R.isoDateTimeMillis(dayjs.utc()); | ||||
|             bean.status = DOWN; | ||||
|             bean.downCount = previousBeat?.downCount || 0; | ||||
|  | ||||
|             if (this.isUpsideDown()) { | ||||
|                 bean.status = flipStatus(bean.status); | ||||
| @@ -219,7 +246,10 @@ class Monitor extends BeanModel { | ||||
|             } | ||||
|  | ||||
|             try { | ||||
|                 if (this.type === "http" || this.type === "keyword") { | ||||
|                 if (await Monitor.isUnderMaintenance(this.id)) { | ||||
|                     bean.msg = "Monitor under maintenance"; | ||||
|                     bean.status = MAINTENANCE; | ||||
|                 } else if (this.type === "http" || this.type === "keyword") { | ||||
|                     // Do not do any queries/high loading things before the "bean.ping" | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
| @@ -238,12 +268,16 @@ class Monitor extends BeanModel { | ||||
|  | ||||
|                     log.debug("monitor", `[${this.name}] Prepare Options for axios`); | ||||
|  | ||||
|                     // Axios Options | ||||
|                     const options = { | ||||
|                         url: this.url, | ||||
|                         method: (this.method || "get").toLowerCase(), | ||||
|                         ...(this.body ? { data: JSON.parse(this.body) } : {}), | ||||
|                         timeout: this.interval * 1000 * 0.8, | ||||
|                         headers: { | ||||
|                             // Fix #2253 | ||||
|                             // Read more: https://stackoverflow.com/questions/1759956/curl-error-18-transfer-closed-with-outstanding-read-data-remaining | ||||
|                             "Accept-Encoding": "gzip, deflate", | ||||
|                             "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9", | ||||
|                             "User-Agent": "Uptime-Kuma/" + version, | ||||
|                             ...(this.headers ? JSON.parse(this.headers) : {}), | ||||
| @@ -468,6 +502,35 @@ class Monitor extends BeanModel { | ||||
|                     } else { | ||||
|                         throw new Error("Server not found on Steam"); | ||||
|                     } | ||||
|                 } else if (this.type === "docker") { | ||||
|                     log.debug(`[${this.name}] Prepare Options for Axios`); | ||||
|  | ||||
|                     const dockerHost = await R.load("docker_host", this.docker_host); | ||||
|  | ||||
|                     const options = { | ||||
|                         url: `/containers/${this.docker_container}/json`, | ||||
|                         headers: { | ||||
|                             "Accept": "*/*", | ||||
|                             "User-Agent": "Uptime-Kuma/" + version, | ||||
|                         }, | ||||
|                         httpsAgent: new https.Agent({ | ||||
|                             maxCachedSessions: 0,      // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940) | ||||
|                             rejectUnauthorized: ! this.getIgnoreTls(), | ||||
|                         }), | ||||
|                     }; | ||||
|  | ||||
|                     if (dockerHost._dockerType === "socket") { | ||||
|                         options.socketPath = dockerHost._dockerDaemon; | ||||
|                     } else if (dockerHost._dockerType === "tcp") { | ||||
|                         options.baseURL = DockerHost.patchDockerURL(dockerHost._dockerDaemon); | ||||
|                     } | ||||
|  | ||||
|                     log.debug(`[${this.name}] Axios Request`); | ||||
|                     let res = await axios.request(options); | ||||
|                     if (res.data.State.Running) { | ||||
|                         bean.status = UP; | ||||
|                         bean.msg = ""; | ||||
|                     } | ||||
|                 } else if (this.type === "mqtt") { | ||||
|                     bean.msg = await mqttAsync(this.hostname, this.mqttTopic, this.mqttSuccessMessage, { | ||||
|                         port: this.port, | ||||
| @@ -484,6 +547,89 @@ class Monitor extends BeanModel { | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "grpc-keyword") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|                     const options = { | ||||
|                         grpcUrl: this.grpcUrl, | ||||
|                         grpcProtobufData: this.grpcProtobuf, | ||||
|                         grpcServiceName: this.grpcServiceName, | ||||
|                         grpcEnableTls: this.grpcEnableTls, | ||||
|                         grpcMethod: this.grpcMethod, | ||||
|                         grpcBody: this.grpcBody, | ||||
|                         keyword: this.keyword | ||||
|                     }; | ||||
|                     const response = await grpcQuery(options); | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                     log.debug("monitor:", `gRPC response: ${JSON.stringify(response)}`); | ||||
|                     let responseData = response.data; | ||||
|                     if (responseData.length > 50) { | ||||
|                         responseData = response.substring(0, 47) + "..."; | ||||
|                     } | ||||
|                     if (response.code !== 1) { | ||||
|                         bean.status = DOWN; | ||||
|                         bean.msg = `Error in send gRPC ${response.code} ${response.errorMessage}`; | ||||
|                     } else { | ||||
|                         if (response.data.toString().includes(this.keyword)) { | ||||
|                             bean.status = UP; | ||||
|                             bean.msg = `${responseData}, keyword [${this.keyword}] is found`; | ||||
|                         } else { | ||||
|                             log.debug("monitor:", `GRPC response [${response.data}] + ", but keyword [${this.keyword}] is not in [" + ${response.data} + "]"`); | ||||
|                             bean.status = DOWN; | ||||
|                             bean.msg = `, but keyword [${this.keyword}] is not in [" + ${responseData} + "]`; | ||||
|                         } | ||||
|                     } | ||||
|                 } else if (this.type === "postgres") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     await postgresQuery(this.databaseConnectionString, this.databaseQuery); | ||||
|  | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "mysql") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     await mysqlQuery(this.databaseConnectionString, this.databaseQuery); | ||||
|  | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "radius") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     // Handle monitors that were created before the | ||||
|                     // update and as such don't have a value for | ||||
|                     // this.port. | ||||
|                     let port; | ||||
|                     if (this.port == null) { | ||||
|                         port = 1812; | ||||
|                     } else { | ||||
|                         port = this.port; | ||||
|                     } | ||||
|  | ||||
|                     try { | ||||
|                         const resp = await radius( | ||||
|                             this.hostname, | ||||
|                             this.radiusUsername, | ||||
|                             this.radiusPassword, | ||||
|                             this.radiusCalledStationId, | ||||
|                             this.radiusCallingStationId, | ||||
|                             this.radiusSecret, | ||||
|                             port | ||||
|                         ); | ||||
|                         if (resp.code) { | ||||
|                             bean.msg = resp.code; | ||||
|                         } | ||||
|                         bean.status = UP; | ||||
|                     } catch (error) { | ||||
|                         bean.status = DOWN; | ||||
|                         if (error.response?.code) { | ||||
|                             bean.msg = error.response.code; | ||||
|                         } else { | ||||
|                             bean.msg = error.message; | ||||
|                         } | ||||
|                     } | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else { | ||||
|                     bean.msg = "Unknown Monitor Type"; | ||||
|                     bean.status = PENDING; | ||||
| @@ -522,15 +668,36 @@ class Monitor extends BeanModel { | ||||
|             if (isImportant) { | ||||
|                 bean.important = true; | ||||
|  | ||||
|                 log.debug("monitor", `[${this.name}] sendNotification`); | ||||
|                 await Monitor.sendNotification(isFirstBeat, this, bean); | ||||
|                 if (Monitor.isImportantForNotification(isFirstBeat, previousBeat?.status, bean.status)) { | ||||
|                     log.debug("monitor", `[${this.name}] sendNotification`); | ||||
|                     await Monitor.sendNotification(isFirstBeat, this, bean); | ||||
|                 } else { | ||||
|                     log.debug("monitor", `[${this.name}] will not sendNotification because it is (or was) under maintenance`); | ||||
|                 } | ||||
|  | ||||
|                 // Reset down count | ||||
|                 bean.downCount = 0; | ||||
|  | ||||
|                 // Clear Status Page Cache | ||||
|                 log.debug("monitor", `[${this.name}] apicache clear`); | ||||
|                 apicache.clear(); | ||||
|  | ||||
|                 UptimeKumaServer.getInstance().sendMaintenanceListByUserID(this.user_id); | ||||
|  | ||||
|             } else { | ||||
|                 bean.important = false; | ||||
|  | ||||
|                 if (bean.status === DOWN && this.resendInterval > 0) { | ||||
|                     ++bean.downCount; | ||||
|                     if (bean.downCount >= this.resendInterval) { | ||||
|                         // Send notification again, because we are still DOWN | ||||
|                         log.debug("monitor", `[${this.name}] sendNotification again: Down Count: ${bean.downCount} | Resend Interval: ${this.resendInterval}`); | ||||
|                         await Monitor.sendNotification(isFirstBeat, this, bean); | ||||
|  | ||||
|                         // Reset down count | ||||
|                         bean.downCount = 0; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (bean.status === UP) { | ||||
| @@ -540,11 +707,14 @@ class Monitor extends BeanModel { | ||||
|                     beatInterval = this.retryInterval; | ||||
|                 } | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Pending: ${bean.msg} | Max retries: ${this.maxretries} | Retry: ${retries} | Retry Interval: ${beatInterval} seconds | Type: ${this.type}`); | ||||
|             } else if (bean.status === MAINTENANCE) { | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Under Maintenance | Type: ${this.type}`); | ||||
|             } else { | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Failing: ${bean.msg} | Interval: ${beatInterval} seconds | Type: ${this.type}`); | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Failing: ${bean.msg} | Interval: ${beatInterval} seconds | Type: ${this.type} | Down Count: ${bean.downCount} | Resend Interval: ${this.resendInterval}`); | ||||
|             } | ||||
|  | ||||
|             log.debug("monitor", `[${this.name}] Send to socket`); | ||||
|             UptimeCacheList.clearCache(this.id); | ||||
|             io.to(this.user_id).emit("heartbeat", bean.toJSON()); | ||||
|             Monitor.sendStats(io, this.id, this.user_id); | ||||
|  | ||||
| @@ -729,7 +899,15 @@ class Monitor extends BeanModel { | ||||
|      * @param {number} duration Hours | ||||
|      * @param {number} monitorID ID of monitor to calculate | ||||
|      */ | ||||
|     static async calcUptime(duration, monitorID) { | ||||
|     static async calcUptime(duration, monitorID, forceNoCache = false) { | ||||
|  | ||||
|         if (!forceNoCache) { | ||||
|             let cachedUptime = UptimeCacheList.getUptime(monitorID, duration); | ||||
|             if (cachedUptime != null) { | ||||
|                 return cachedUptime; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         const timeLogger = new TimeLogger(); | ||||
|  | ||||
|         const startTime = R.isoDateTime(dayjs.utc().subtract(duration, "hour")); | ||||
| @@ -750,7 +928,7 @@ class Monitor extends BeanModel { | ||||
|                -- SUM all uptime duration, also trim off the beat out of time window | ||||
|                 SUM( | ||||
|                     CASE | ||||
|                         WHEN (status = 1) | ||||
|                         WHEN (status = 1 OR status = 3) | ||||
|                         THEN | ||||
|                             CASE | ||||
|                                 WHEN (JULIANDAY(\`time\`) - JULIANDAY(?)) * 86400 < duration | ||||
| @@ -788,6 +966,9 @@ class Monitor extends BeanModel { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Cache | ||||
|         UptimeCacheList.addUptime(monitorID, duration, uptime); | ||||
|  | ||||
|         return uptime; | ||||
|     } | ||||
|  | ||||
| @@ -821,11 +1002,49 @@ class Monitor extends BeanModel { | ||||
|         // DOWN -> PENDING = this case not exists | ||||
|         // DOWN -> DOWN = not important | ||||
|         // * DOWN -> UP = important | ||||
|         let isImportant = isFirstBeat || | ||||
|         // MAINTENANCE -> MAINTENANCE = not important | ||||
|         // * MAINTENANCE -> UP = important | ||||
|         // * MAINTENANCE -> DOWN = important | ||||
|         // * DOWN -> MAINTENANCE = important | ||||
|         // * UP -> MAINTENANCE = important | ||||
|         return isFirstBeat || | ||||
|             (previousBeatStatus === DOWN && currentBeatStatus === MAINTENANCE) || | ||||
|             (previousBeatStatus === UP && currentBeatStatus === MAINTENANCE) || | ||||
|             (previousBeatStatus === MAINTENANCE && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === MAINTENANCE && currentBeatStatus === UP) || | ||||
|             (previousBeatStatus === UP && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === DOWN && currentBeatStatus === UP) || | ||||
|             (previousBeatStatus === PENDING && currentBeatStatus === DOWN); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Is this beat important for notifications? | ||||
|      * @param {boolean} isFirstBeat Is this the first beat of this monitor? | ||||
|      * @param {const} previousBeatStatus Status of the previous beat | ||||
|      * @param {const} currentBeatStatus Status of the current beat | ||||
|      * @returns {boolean} True if is an important beat else false | ||||
|      */ | ||||
|     static isImportantForNotification(isFirstBeat, previousBeatStatus, currentBeatStatus) { | ||||
|         // * ? -> ANY STATUS = important [isFirstBeat] | ||||
|         // UP -> PENDING = not important | ||||
|         // * UP -> DOWN = important | ||||
|         // UP -> UP = not important | ||||
|         // PENDING -> PENDING = not important | ||||
|         // * PENDING -> DOWN = important | ||||
|         // PENDING -> UP = not important | ||||
|         // DOWN -> PENDING = this case not exists | ||||
|         // DOWN -> DOWN = not important | ||||
|         // * DOWN -> UP = important | ||||
|         // MAINTENANCE -> MAINTENANCE = not important | ||||
|         // MAINTENANCE -> UP = not important | ||||
|         // * MAINTENANCE -> DOWN = important | ||||
|         // DOWN -> MAINTENANCE = not important | ||||
|         // UP -> MAINTENANCE = not important | ||||
|         return isFirstBeat || | ||||
|             (previousBeatStatus === MAINTENANCE && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === UP && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === DOWN && currentBeatStatus === UP) || | ||||
|             (previousBeatStatus === PENDING && currentBeatStatus === DOWN); | ||||
|         return isImportant; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -962,6 +1181,35 @@ class Monitor extends BeanModel { | ||||
|             monitorID | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if monitor is under maintenance | ||||
|      * @param {number} monitorID ID of monitor to check | ||||
|      * @returns {Promise<boolean>} | ||||
|      */ | ||||
|     static async isUnderMaintenance(monitorID) { | ||||
|         let activeCondition = Maintenance.getActiveMaintenanceSQLCondition(); | ||||
|         const maintenance = await R.getRow(` | ||||
|             SELECT COUNT(*) AS count | ||||
|             FROM monitor_maintenance mm | ||||
|             JOIN maintenance | ||||
|                 ON mm.maintenance_id = maintenance.id | ||||
|                 AND mm.monitor_id = ? | ||||
|             LEFT JOIN maintenance_timeslot | ||||
|                 ON maintenance_timeslot.maintenance_id = maintenance.id | ||||
|             WHERE ${activeCondition} | ||||
|             LIMIT 1`, [ monitorID ]); | ||||
|         return maintenance.count !== 0; | ||||
|     } | ||||
|  | ||||
|     validate() { | ||||
|         if (this.interval > MAX_INTERVAL_SECOND) { | ||||
|             throw new Error(`Interval cannot be more than ${MAX_INTERVAL_SECOND} seconds`); | ||||
|         } | ||||
|         if (this.interval < MIN_INTERVAL_SECOND) { | ||||
|             throw new Error(`Interval cannot be less than ${MIN_INTERVAL_SECOND} seconds`); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Monitor; | ||||
|   | ||||
| @@ -2,6 +2,8 @@ const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { R } = require("redbean-node"); | ||||
| const cheerio = require("cheerio"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const jsesc = require("jsesc"); | ||||
| const Maintenance = require("./maintenance"); | ||||
|  | ||||
| class StatusPage extends BeanModel { | ||||
|  | ||||
| @@ -36,7 +38,7 @@ class StatusPage extends BeanModel { | ||||
|      */ | ||||
|     static async renderHTML(indexHTML, statusPage) { | ||||
|         const $ = cheerio.load(indexHTML); | ||||
|         const description155 = statusPage.description?.substring(0, 155); | ||||
|         const description155 = statusPage.description?.substring(0, 155) ?? ""; | ||||
|  | ||||
|         $("title").text(statusPage.title); | ||||
|         $("meta[name=description]").attr("content", description155); | ||||
| @@ -56,13 +58,19 @@ class StatusPage extends BeanModel { | ||||
|         head.append(`<meta property="og:description" content="${description155}" />`); | ||||
|  | ||||
|         // Preload data | ||||
|         const json = JSON.stringify(await StatusPage.getStatusPageData(statusPage)); | ||||
|         head.append(` | ||||
|             <script> | ||||
|                 window.preloadData = ${json} | ||||
|         // Add jsesc, fix https://github.com/louislam/uptime-kuma/issues/2186 | ||||
|         const escapedJSONObject = jsesc(await StatusPage.getStatusPageData(statusPage), { | ||||
|             "isScriptContext": true | ||||
|         }); | ||||
|  | ||||
|         const script = $(` | ||||
|             <script id="preload-data" data-json="{}"> | ||||
|                 window.preloadData = ${escapedJSONObject}; | ||||
|             </script> | ||||
|         `); | ||||
|  | ||||
|         head.append(script); | ||||
|  | ||||
|         // manifest.json | ||||
|         $("link[rel=manifest]").attr("href", `/api/status-page/${statusPage.slug}/manifest.json`); | ||||
|  | ||||
| @@ -83,6 +91,8 @@ class StatusPage extends BeanModel { | ||||
|             incident = incident.toPublicJSON(); | ||||
|         } | ||||
|  | ||||
|         let maintenanceList = await StatusPage.getMaintenanceList(statusPage.id); | ||||
|  | ||||
|         // Public Group List | ||||
|         const publicGroupList = []; | ||||
|         const showTags = !!statusPage.show_tags; | ||||
| @@ -100,7 +110,8 @@ class StatusPage extends BeanModel { | ||||
|         return { | ||||
|             config: await statusPage.toPublicJSON(), | ||||
|             incident, | ||||
|             publicGroupList | ||||
|             publicGroupList, | ||||
|             maintenanceList, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
| @@ -259,6 +270,38 @@ class StatusPage extends BeanModel { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get list of maintenances | ||||
|      * @param {number} statusPageId ID of status page to get maintenance for | ||||
|      * @returns {Object} Object representing maintenances sanitized for public | ||||
|      */ | ||||
|     static async getMaintenanceList(statusPageId) { | ||||
|         try { | ||||
|             const publicMaintenanceList = []; | ||||
|  | ||||
|             let activeCondition = Maintenance.getActiveMaintenanceSQLCondition(); | ||||
|             let maintenanceBeanList = R.convertToBeans("maintenance", await R.getAll(` | ||||
|                 SELECT maintenance.* | ||||
|                 FROM maintenance | ||||
|                 JOIN maintenance_status_page | ||||
|                     ON maintenance_status_page.maintenance_id = maintenance.id | ||||
|                     AND maintenance_status_page.status_page_id = ? | ||||
|                 LEFT JOIN maintenance_timeslot | ||||
|                     ON maintenance_timeslot.maintenance_id = maintenance.id | ||||
|                 WHERE ${activeCondition} | ||||
|                 ORDER BY maintenance.end_date | ||||
|             `, [ statusPageId ])); | ||||
|  | ||||
|             for (const bean of maintenanceBeanList) { | ||||
|                 publicMaintenanceList.push(await bean.toPublicJSON()); | ||||
|             } | ||||
|  | ||||
|             return publicMaintenanceList; | ||||
|  | ||||
|         } catch (error) { | ||||
|             return []; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = StatusPage; | ||||
|   | ||||
							
								
								
									
										20
									
								
								server/modules/dayjs/plugin/timezone.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								server/modules/dayjs/plugin/timezone.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| import { PluginFunc, ConfigType } from 'dayjs' | ||||
|  | ||||
| declare const plugin: PluginFunc | ||||
| export = plugin | ||||
|  | ||||
| declare module 'dayjs' { | ||||
|   interface Dayjs { | ||||
|     tz(timezone?: string, keepLocalTime?: boolean): Dayjs | ||||
|     offsetName(type?: 'short' | 'long'): string | undefined | ||||
|   } | ||||
|  | ||||
|   interface DayjsTimezone { | ||||
|     (date: ConfigType, timezone?: string): Dayjs | ||||
|     (date: ConfigType, format: string, timezone?: string): Dayjs | ||||
|     guess(): string | ||||
|     setDefault(timezone?: string): void | ||||
|   } | ||||
|  | ||||
|   const tz: DayjsTimezone | ||||
| } | ||||
							
								
								
									
										115
									
								
								server/modules/dayjs/plugin/timezone.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								server/modules/dayjs/plugin/timezone.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,115 @@ | ||||
| /** | ||||
|  * Copy from node_modules/dayjs/plugin/timezone.js | ||||
|  * Try to fix https://github.com/louislam/uptime-kuma/issues/2318 | ||||
|  * Source: https://github.com/iamkun/dayjs/tree/dev/src/plugin/utc | ||||
|  * License: MIT | ||||
|  */ | ||||
| !function (t, e) { | ||||
|     // eslint-disable-next-line no-undef | ||||
|     typeof exports == "object" && typeof module != "undefined" ? module.exports = e() : typeof define == "function" && define.amd ? define(e) : (t = typeof globalThis != "undefined" ? globalThis : t || self).dayjs_plugin_timezone = e(); | ||||
| }(this, (function () { | ||||
|     "use strict"; | ||||
|     let t = { | ||||
|         year: 0, | ||||
|         month: 1, | ||||
|         day: 2, | ||||
|         hour: 3, | ||||
|         minute: 4, | ||||
|         second: 5 | ||||
|     }; | ||||
|     let e = {}; | ||||
|     return function (n, i, o) { | ||||
|         let r; | ||||
|         let a = function (t, n, i) { | ||||
|             void 0 === i && (i = {}); | ||||
|             let o = new Date(t); | ||||
|             let r = function (t, n) { | ||||
|                 void 0 === n && (n = {}); | ||||
|                 let i = n.timeZoneName || "short"; | ||||
|                 let o = t + "|" + i; | ||||
|                 let r = e[o]; | ||||
|                 return r || (r = new Intl.DateTimeFormat("en-US", { | ||||
|                     hour12: !1, | ||||
|                     timeZone: t, | ||||
|                     year: "numeric", | ||||
|                     month: "2-digit", | ||||
|                     day: "2-digit", | ||||
|                     hour: "2-digit", | ||||
|                     minute: "2-digit", | ||||
|                     second: "2-digit", | ||||
|                     timeZoneName: i | ||||
|                 }), e[o] = r), r; | ||||
|             }(n, i); | ||||
|             return r.formatToParts(o); | ||||
|         }; | ||||
|         let u = function (e, n) { | ||||
|             let i = a(e, n); | ||||
|             let r = []; | ||||
|             let u = 0; | ||||
|             for (; u < i.length; u += 1) { | ||||
|                 let f = i[u]; | ||||
|                 let s = f.type; | ||||
|                 let m = f.value; | ||||
|                 let c = t[s]; | ||||
|                 c >= 0 && (r[c] = parseInt(m, 10)); | ||||
|             } | ||||
|             let d = r[3]; | ||||
|             let l = d === 24 ? 0 : d; | ||||
|             let v = r[0] + "-" + r[1] + "-" + r[2] + " " + l + ":" + r[4] + ":" + r[5] + ":000"; | ||||
|             let h = +e; | ||||
|             return (o.utc(v).valueOf() - (h -= h % 1e3)) / 6e4; | ||||
|         }; | ||||
|         let f = i.prototype; | ||||
|         f.tz = function (t, e) { | ||||
|             void 0 === t && (t = r); | ||||
|             let n = this.utcOffset(); | ||||
|             let i = this.toDate(); | ||||
|             let a = i.toLocaleString("en-US", { timeZone: t }).replace("\u202f", " "); | ||||
|             let u = Math.round((i - new Date(a)) / 1e3 / 60); | ||||
|             let f = o(a).$set("millisecond", this.$ms).utcOffset(15 * -Math.round(i.getTimezoneOffset() / 15) - u, !0); | ||||
|             if (e) { | ||||
|                 let s = f.utcOffset(); | ||||
|                 f = f.add(n - s, "minute"); | ||||
|             } | ||||
|             return f.$x.$timezone = t, f; | ||||
|         }, f.offsetName = function (t) { | ||||
|             let e = this.$x.$timezone || o.tz.guess(); | ||||
|             let n = a(this.valueOf(), e, { timeZoneName: t }).find((function (t) { | ||||
|                 return t.type.toLowerCase() === "timezonename"; | ||||
|             })); | ||||
|             return n && n.value; | ||||
|         }; | ||||
|         let s = f.startOf; | ||||
|         f.startOf = function (t, e) { | ||||
|             if (!this.$x || !this.$x.$timezone) { | ||||
|                 return s.call(this, t, e); | ||||
|             } | ||||
|             let n = o(this.format("YYYY-MM-DD HH:mm:ss:SSS")); | ||||
|             return s.call(n, t, e).tz(this.$x.$timezone, !0); | ||||
|         }, o.tz = function (t, e, n) { | ||||
|             let i = n && e; | ||||
|             let a = n || e || r; | ||||
|             let f = u(+o(), a); | ||||
|             if (typeof t != "string") { | ||||
|                 return o(t).tz(a); | ||||
|             } | ||||
|             let s = function (t, e, n) { | ||||
|                 let i = t - 60 * e * 1e3; | ||||
|                 let o = u(i, n); | ||||
|                 if (e === o) { | ||||
|                     return [ i, e ]; | ||||
|                 } | ||||
|                 let r = u(i -= 60 * (o - e) * 1e3, n); | ||||
|                 return o === r ? [ i, o ] : [ t - 60 * Math.min(o, r) * 1e3, Math.max(o, r) ]; | ||||
|             }(o.utc(t, i).valueOf(), f, a); | ||||
|             let m = s[0]; | ||||
|             let c = s[1]; | ||||
|             let d = o(m).utcOffset(c); | ||||
|             return d.$x.$timezone = a, d; | ||||
|         }, o.tz.guess = function () { | ||||
|             return Intl.DateTimeFormat().resolvedOptions().timeZone; | ||||
|         }, o.tz.setDefault = function (t) { | ||||
|             r = t; | ||||
|         }; | ||||
|     }; | ||||
| })); | ||||
| @@ -12,9 +12,7 @@ const { default: axios } = require("axios"); | ||||
|  | ||||
| // bark is an APN bridge that sends notifications to Apple devices. | ||||
|  | ||||
| const barkNotificationGroup = "UptimeKuma"; | ||||
| const barkNotificationAvatar = "https://github.com/louislam/uptime-kuma/raw/master/public/icon.png"; | ||||
| const barkNotificationSound = "telegraph"; | ||||
| const successMessage = "Successes!"; | ||||
|  | ||||
| class Bark extends NotificationProvider { | ||||
| @@ -30,17 +28,17 @@ class Bark extends NotificationProvider { | ||||
|  | ||||
|         if (msg != null && heartbeatJSON != null && heartbeatJSON["status"] === UP) { | ||||
|             let title = "UptimeKuma Monitor Up"; | ||||
|             return await this.postNotification(title, msg, barkEndpoint); | ||||
|             return await this.postNotification(notification, title, msg, barkEndpoint); | ||||
|         } | ||||
|  | ||||
|         if (msg != null && heartbeatJSON != null && heartbeatJSON["status"] === DOWN) { | ||||
|             let title = "UptimeKuma Monitor Down"; | ||||
|             return await this.postNotification(title, msg, barkEndpoint); | ||||
|             return await this.postNotification(notification, title, msg, barkEndpoint); | ||||
|         } | ||||
|  | ||||
|         if (msg != null) { | ||||
|             let title = "UptimeKuma Message"; | ||||
|             return await this.postNotification(title, msg, barkEndpoint); | ||||
|             return await this.postNotification(notification, title, msg, barkEndpoint); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -50,13 +48,23 @@ class Bark extends NotificationProvider { | ||||
|      * @param {string} postUrl URL to append parameters to | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     appendAdditionalParameters(postUrl) { | ||||
|         // grouping all our notifications | ||||
|         postUrl += "?group=" + barkNotificationGroup; | ||||
|     appendAdditionalParameters(notification, postUrl) { | ||||
|         // set icon to uptime kuma icon, 11kb should be fine | ||||
|         postUrl += "&icon=" + barkNotificationAvatar; | ||||
|         postUrl += "?icon=" + barkNotificationAvatar; | ||||
|         // grouping all our notifications | ||||
|         if (notification.barkGroup != null) { | ||||
|             postUrl += "&group=" + notification.barkGroup; | ||||
|         } else { | ||||
|             // default name | ||||
|             postUrl += "&group=" + "UptimeKuma"; | ||||
|         } | ||||
|         // picked a sound, this should follow system's mute status when arrival | ||||
|         postUrl += "&sound=" + barkNotificationSound; | ||||
|         if (notification.barkSound != null) { | ||||
|             postUrl += "&sound=" + notification.barkSound; | ||||
|         } else { | ||||
|             // default sound | ||||
|             postUrl += "&sound=" + "telegraph"; | ||||
|         } | ||||
|         return postUrl; | ||||
|     } | ||||
|  | ||||
| @@ -81,12 +89,12 @@ class Bark extends NotificationProvider { | ||||
|      * @param {string} endpoint Endpoint to send request to | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     async postNotification(title, subtitle, endpoint) { | ||||
|     async postNotification(notification, title, subtitle, endpoint) { | ||||
|         // url encode title and subtitle | ||||
|         title = encodeURIComponent(title); | ||||
|         subtitle = encodeURIComponent(subtitle); | ||||
|         let postUrl = endpoint + "/" + title + "/" + subtitle; | ||||
|         postUrl = this.appendAdditionalParameters(postUrl); | ||||
|         postUrl = this.appendAdditionalParameters(notification, postUrl); | ||||
|         let result = await axios.get(postUrl); | ||||
|         this.checkResult(result); | ||||
|         if (result.statusText != null) { | ||||
|   | ||||
							
								
								
									
										24
									
								
								server/notification-providers/freemobile.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								server/notification-providers/freemobile.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class FreeMobile extends NotificationProvider { | ||||
|  | ||||
|     name = "FreeMobile"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             await axios.post(`https://smsapi.free-mobile.fr/sendmsg?msg=${encodeURIComponent(msg.replace("🔴", "⛔️"))}`, { | ||||
|                 "user": notification.freemobileUser, | ||||
|                 "pass": notification.freemobilePass, | ||||
|             }); | ||||
|  | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = FreeMobile; | ||||
							
								
								
									
										35
									
								
								server/notification-providers/goalert.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								server/notification-providers/goalert.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { UP } = require("../../src/util"); | ||||
|  | ||||
| class GoAlert extends NotificationProvider { | ||||
|  | ||||
|     name = "GoAlert"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             let closeAction = "close"; | ||||
|             let data = { | ||||
|                 summary: msg, | ||||
|             }; | ||||
|             if (heartbeatJSON != null && heartbeatJSON["status"] === UP) { | ||||
|                 data["action"] = closeAction; | ||||
|             } | ||||
|             let headers = { | ||||
|                 "Content-Type": "multipart/form-data", | ||||
|             }; | ||||
|             let config = { | ||||
|                 headers: headers | ||||
|             }; | ||||
|             await axios.post(`${notification.goAlertBaseURL}/api/v2/generic/incoming?token=${notification.goAlertToken}`, data, config); | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             let msg = (error.response.data) ? error.response.data : "Error without response"; | ||||
|             throw new Error(msg); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = GoAlert; | ||||
							
								
								
									
										38
									
								
								server/notification-providers/home-assistant.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								server/notification-providers/home-assistant.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| const defaultNotificationService = "notify"; | ||||
|  | ||||
| class HomeAssistant extends NotificationProvider { | ||||
|     name = "HomeAssistant"; | ||||
|  | ||||
|     async send(notification, message, monitor = null, heartbeat = null) { | ||||
|         const notificationService = notification?.notificationService || defaultNotificationService; | ||||
|  | ||||
|         try { | ||||
|             await axios.post( | ||||
|                 `${notification.homeAssistantUrl}/api/services/notify/${notificationService}`, | ||||
|                 { | ||||
|                     title: "Uptime Kuma", | ||||
|                     message, | ||||
|                     ...(notificationService !== "persistent_notification" && { data: { | ||||
|                         name: monitor?.name, | ||||
|                         status: heartbeat?.status, | ||||
|                     } }), | ||||
|                 }, | ||||
|                 { | ||||
|                     headers: { | ||||
|                         Authorization: `Bearer ${notification.longLivedAccessToken}`, | ||||
|                         "Content-Type": "application/json", | ||||
|                     }, | ||||
|                 } | ||||
|             ); | ||||
|  | ||||
|             return "Sent Successfully."; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = HomeAssistant; | ||||
							
								
								
									
										43
									
								
								server/notification-providers/linenotify.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								server/notification-providers/linenotify.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const qs = require("qs"); | ||||
| const { DOWN, UP } = require("../../src/util"); | ||||
|  | ||||
| class LineNotify extends NotificationProvider { | ||||
|  | ||||
|     name = "LineNotify"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             let lineAPIUrl = "https://notify-api.line.me/api/notify"; | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "Content-Type": "application/x-www-form-urlencoded", | ||||
|                     "Authorization": "Bearer " + notification.lineNotifyAccessToken | ||||
|                 } | ||||
|             }; | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let testMessage = { | ||||
|                     "message": msg, | ||||
|                 }; | ||||
|                 await axios.post(lineAPIUrl, qs.stringify(testMessage), config); | ||||
|             } else if (heartbeatJSON["status"] === DOWN) { | ||||
|                 let downMessage = { | ||||
|                     "message": "\n[🔴 Down]\n" + "Name: " + monitorJSON["name"] + " \n" + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"] | ||||
|                 }; | ||||
|                 await axios.post(lineAPIUrl, qs.stringify(downMessage), config); | ||||
|             } else if (heartbeatJSON["status"] === UP) { | ||||
|                 let upMessage = { | ||||
|                     "message": "\n[✅ Up]\n" + "Name: " + monitorJSON["name"] + " \n" + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"] | ||||
|                 }; | ||||
|                 await axios.post(lineAPIUrl, qs.stringify(upMessage), config); | ||||
|             } | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = LineNotify; | ||||
| @@ -8,12 +8,24 @@ class Ntfy extends NotificationProvider { | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             await axios.post(`${notification.ntfyserverurl}`, { | ||||
|             let headers = {}; | ||||
|             if (notification.ntfyusername) { | ||||
|                 headers = { | ||||
|                     "Authorization": "Basic " + Buffer.from(notification.ntfyusername + ":" + notification.ntfypassword).toString("base64"), | ||||
|                 }; | ||||
|             } | ||||
|             let data = { | ||||
|                 "topic": notification.ntfytopic, | ||||
|                 "message": msg, | ||||
|                 "priority": notification.ntfyPriority || 4, | ||||
|                 "title": "Uptime-Kuma", | ||||
|             }); | ||||
|             }; | ||||
|  | ||||
|             if (notification.ntfyIcon) { | ||||
|                 data.icon = notification.ntfyIcon; | ||||
|             } | ||||
|  | ||||
|             await axios.post(`${notification.ntfyserverurl}`, data, { headers: headers }); | ||||
|  | ||||
|             return okMsg; | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,7 @@ class Octopush extends NotificationProvider { | ||||
|  | ||||
|         try { | ||||
|         // Default - V2 | ||||
|             if (notification.octopushVersion === 2 || !notification.octopushVersion) { | ||||
|             if (notification.octopushVersion === "2" || !notification.octopushVersion) { | ||||
|                 let config = { | ||||
|                     headers: { | ||||
|                         "api-key": notification.octopushAPIKey, | ||||
| @@ -31,7 +31,7 @@ class Octopush extends NotificationProvider { | ||||
|                     "sender": notification.octopushSenderName | ||||
|                 }; | ||||
|                 await axios.post("https://api.octopush.com/v1/public/sms-campaign/send", data, config); | ||||
|             } else if (notification.octopushVersion === 1) { | ||||
|             } else if (notification.octopushVersion === "1") { | ||||
|                 let data = { | ||||
|                     "user_login": notification.octopushDMLogin, | ||||
|                     "api_key": notification.octopushDMAPIKey, | ||||
| @@ -49,7 +49,15 @@ class Octopush extends NotificationProvider { | ||||
|                     }, | ||||
|                     params: data | ||||
|                 }; | ||||
|                 await axios.post("https://www.octopush-dm.com/api/sms/json", {}, config); | ||||
|  | ||||
|                 // V1 API returns 200 even on error so we must check | ||||
|                 // response data | ||||
|                 let response = await axios.post("https://www.octopush-dm.com/api/sms/json", {}, config); | ||||
|                 if ("error_code" in response.data) { | ||||
|                     if (response.data.error_code !== "000") { | ||||
|                         this.throwGeneralAxiosError(`Octopush error ${JSON.stringify(response.data)}`); | ||||
|                     } | ||||
|                 } | ||||
|             } else { | ||||
|                 throw new Error("Unknown Octopush version!"); | ||||
|             } | ||||
|   | ||||
| @@ -19,26 +19,26 @@ class Pushbullet extends NotificationProvider { | ||||
|                 } | ||||
|             }; | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let testdata = { | ||||
|                 let data = { | ||||
|                     "type": "note", | ||||
|                     "title": "Uptime Kuma Alert", | ||||
|                     "body": "Testing Successful.", | ||||
|                     "body": msg, | ||||
|                 }; | ||||
|                 await axios.post(pushbulletUrl, testdata, config); | ||||
|                 await axios.post(pushbulletUrl, data, config); | ||||
|             } else if (heartbeatJSON["status"] === DOWN) { | ||||
|                 let downdata = { | ||||
|                 let downData = { | ||||
|                     "type": "note", | ||||
|                     "title": "UptimeKuma Alert: " + monitorJSON["name"], | ||||
|                     "body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"], | ||||
|                 }; | ||||
|                 await axios.post(pushbulletUrl, downdata, config); | ||||
|                 await axios.post(pushbulletUrl, downData, config); | ||||
|             } else if (heartbeatJSON["status"] === UP) { | ||||
|                 let updata = { | ||||
|                 let upData = { | ||||
|                     "type": "note", | ||||
|                     "title": "UptimeKuma Alert: " + monitorJSON["name"], | ||||
|                     "body": "[✅ Up] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"], | ||||
|                 }; | ||||
|                 await axios.post(pushbulletUrl, updata, config); | ||||
|                 await axios.post(pushbulletUrl, upData, config); | ||||
|             } | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|   | ||||
							
								
								
									
										36
									
								
								server/notification-providers/serverchan.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								server/notification-providers/serverchan.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { DOWN, UP } = require("../../src/util"); | ||||
|  | ||||
| class ServerChan extends NotificationProvider { | ||||
|  | ||||
|     name = "ServerChan"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             await axios.post(`https://sctapi.ftqq.com/${notification.serverChanSendKey}.send`, { | ||||
|                 "title": this.checkStatus(heartbeatJSON, monitorJSON), | ||||
|                 "desp": msg, | ||||
|             }); | ||||
|  | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     checkStatus(heartbeatJSON, monitorJSON) { | ||||
|         let title = "UptimeKuma Message"; | ||||
|         if (heartbeatJSON != null && heartbeatJSON["status"] === UP) { | ||||
|             title = "UptimeKuma Monitor Up " + monitorJSON["name"]; | ||||
|         } | ||||
|         if (heartbeatJSON != null && heartbeatJSON["status"] === DOWN) { | ||||
|             title = "UptimeKuma Monitor Down " + monitorJSON["name"]; | ||||
|         } | ||||
|         return title; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = ServerChan; | ||||
							
								
								
									
										71
									
								
								server/notification-providers/smseagle.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								server/notification-providers/smseagle.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class SMSEagle extends NotificationProvider { | ||||
|  | ||||
|     name = "SMSEagle"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "Content-Type": "application/json", | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             let postData; | ||||
|             let sendMethod; | ||||
|             let recipientType; | ||||
|  | ||||
|             let encoding = (notification.smseagleEncoding) ? "1" : "0"; | ||||
|             let priority = (notification.smseaglePriority) ? notification.smseaglePriority : "0"; | ||||
|  | ||||
|             if (notification.smseagleRecipientType === "smseagle-contact") { | ||||
|                 recipientType = "contactname"; | ||||
|                 sendMethod = "sms.send_tocontact"; | ||||
|             } | ||||
|             if (notification.smseagleRecipientType === "smseagle-group") { | ||||
|                 recipientType = "groupname"; | ||||
|                 sendMethod = "sms.send_togroup"; | ||||
|             } | ||||
|             if (notification.smseagleRecipientType === "smseagle-to") { | ||||
|                 recipientType = "to"; | ||||
|                 sendMethod = "sms.send_sms"; | ||||
|             } | ||||
|  | ||||
|             let params = { | ||||
|                 access_token: notification.smseagleToken, | ||||
|                 [recipientType]: notification.smseagleRecipient, | ||||
|                 message: msg, | ||||
|                 responsetype: "extended", | ||||
|                 unicode: encoding, | ||||
|                 highpriority: priority | ||||
|             }; | ||||
|  | ||||
|             postData = { | ||||
|                 method: sendMethod, | ||||
|                 params: params | ||||
|             }; | ||||
|  | ||||
|             let resp = await axios.post(notification.smseagleUrl + "/jsonrpc/sms", postData, config); | ||||
|  | ||||
|             if ((JSON.stringify(resp.data)).indexOf("message_id") === -1) { | ||||
|                 let error = ""; | ||||
|                 if (resp.data.result && resp.data.result.error_text) { | ||||
|                     error = `SMSEagle API returned error: ${JSON.stringify(resp.data.result.error_text)}`; | ||||
|                 } else { | ||||
|                     error = "SMSEagle API returned an unexpected response"; | ||||
|                 } | ||||
|                 throw new Error(error); | ||||
|             } | ||||
|  | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = SMSEagle; | ||||
							
								
								
									
										25
									
								
								server/notification-providers/smsmanager.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								server/notification-providers/smsmanager.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class SMSManager extends NotificationProvider { | ||||
|  | ||||
|     name = "SMSManager"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         try { | ||||
|             let data = { | ||||
|                 apikey: notification.smsmanagerApiKey, | ||||
|                 endpoint: "https://http-api.smsmanager.cz/Send", | ||||
|                 message: msg.replace(/[^\x00-\x7F]/g, ""), | ||||
|                 to: notification.numbers, | ||||
|                 messageType: notification.messageType, | ||||
|             }; | ||||
|             await axios.get(`${data.endpoint}?apikey=${data.apikey}&message=${data.message}&number=${data.to}&gateway=${data.messageType}`); | ||||
|             return "SMS sent sucessfully."; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = SMSManager; | ||||
							
								
								
									
										76
									
								
								server/notification-providers/squadcast.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								server/notification-providers/squadcast.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { DOWN } = require("../../src/util"); | ||||
|  | ||||
| class Squadcast extends NotificationProvider { | ||||
|  | ||||
|     name = "squadcast"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|  | ||||
|             let config = {}; | ||||
|             let data = { | ||||
|                 message: msg, | ||||
|                 description: "", | ||||
|                 tags: {}, | ||||
|                 heartbeat: heartbeatJSON, | ||||
|                 source: "uptime-kuma" | ||||
|             }; | ||||
|  | ||||
|             if (heartbeatJSON !== null) { | ||||
|                 data.description = heartbeatJSON["msg"]; | ||||
|                 data.event_id = heartbeatJSON["monitorID"]; | ||||
|  | ||||
|                 if (heartbeatJSON["status"] === DOWN) { | ||||
|                     data.message = `${monitorJSON["name"]} is DOWN`; | ||||
|                     data.status = "trigger"; | ||||
|                 } else { | ||||
|                     data.message = `${monitorJSON["name"]} is UP`; | ||||
|                     data.status = "resolve"; | ||||
|                 } | ||||
|  | ||||
|                 let address; | ||||
|                 switch (monitorJSON["type"]) { | ||||
|                     case "ping": | ||||
|                         address = monitorJSON["hostname"]; | ||||
|                         break; | ||||
|                     case "port": | ||||
|                     case "dns": | ||||
|                     case "steam": | ||||
|                         address = monitorJSON["hostname"]; | ||||
|                         if (monitorJSON["port"]) { | ||||
|                             address += ":" + monitorJSON["port"]; | ||||
|                         } | ||||
|                         break; | ||||
|                     default: | ||||
|                         address = monitorJSON["url"]; | ||||
|                         break; | ||||
|                 } | ||||
|  | ||||
|                 data.tags["AlertAddress"] = address; | ||||
|  | ||||
|                 monitorJSON["tags"].forEach(tag => { | ||||
|                     data.tags[tag["name"]] = { | ||||
|                         value: tag["value"] | ||||
|                     }; | ||||
|                     if (tag["color"] !== null) { | ||||
|                         data.tags[tag["name"]]["color"] = tag["color"]; | ||||
|                     } | ||||
|                 }); | ||||
|             } | ||||
|  | ||||
|             await axios.post(notification.squadcastWebhookURL, data, config); | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| module.exports = Squadcast; | ||||
| @@ -63,7 +63,7 @@ class Teams extends NotificationProvider { | ||||
|             }); | ||||
|         } | ||||
|  | ||||
|         if (monitorUrl) { | ||||
|         if (monitorUrl && monitorUrl !== "https://") { | ||||
|             facts.push({ | ||||
|                 name: "URL", | ||||
|                 value: monitorUrl, | ||||
| @@ -127,13 +127,17 @@ class Teams extends NotificationProvider { | ||||
|  | ||||
|             let url; | ||||
|  | ||||
|             if (monitorJSON["type"] === "port") { | ||||
|                 url = monitorJSON["hostname"]; | ||||
|                 if (monitorJSON["port"]) { | ||||
|                     url += ":" + monitorJSON["port"]; | ||||
|                 } | ||||
|             } else { | ||||
|                 url = monitorJSON["url"]; | ||||
|             switch (monitorJSON["type"]) { | ||||
|                 case "http": | ||||
|                 case "keywork": | ||||
|                     url = monitorJSON["url"]; | ||||
|                     break; | ||||
|                 case "docker": | ||||
|                     url = monitorJSON["docker_host"]; | ||||
|                     break; | ||||
|                 default: | ||||
|                     url = monitorJSON["hostname"]; | ||||
|                     break; | ||||
|             } | ||||
|  | ||||
|             const payload = this._notificationPayloadFactory({ | ||||
|   | ||||
| @@ -16,20 +16,29 @@ class Webhook extends NotificationProvider { | ||||
|                 msg, | ||||
|             }; | ||||
|             let finalData; | ||||
|             let config = {}; | ||||
|             let config = { | ||||
|                 headers: {} | ||||
|             }; | ||||
|  | ||||
|             if (notification.webhookContentType === "form-data") { | ||||
|                 finalData = new FormData(); | ||||
|                 finalData.append("data", JSON.stringify(data)); | ||||
|  | ||||
|                 config = { | ||||
|                     headers: finalData.getHeaders(), | ||||
|                 }; | ||||
|  | ||||
|                 config.headers = finalData.getHeaders(); | ||||
|             } else { | ||||
|                 finalData = data; | ||||
|             } | ||||
|  | ||||
|             if (notification.webhookAdditionalHeaders) { | ||||
|                 try { | ||||
|                     config.headers = { | ||||
|                         ...config.headers, | ||||
|                         ...JSON.parse(notification.webhookAdditionalHeaders) | ||||
|                     }; | ||||
|                 } catch (err) { | ||||
|                     throw "Additional Headers is not a valid JSON"; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             await axios.post(notification.webhookURL, finalData, config); | ||||
|             return okMsg; | ||||
|  | ||||
|   | ||||
| @@ -9,10 +9,13 @@ const ClickSendSMS = require("./notification-providers/clicksendsms"); | ||||
| const DingDing = require("./notification-providers/dingding"); | ||||
| const Discord = require("./notification-providers/discord"); | ||||
| const Feishu = require("./notification-providers/feishu"); | ||||
| const FreeMobile = require("./notification-providers/freemobile"); | ||||
| const GoogleChat = require("./notification-providers/google-chat"); | ||||
| const Gorush = require("./notification-providers/gorush"); | ||||
| const Gotify = require("./notification-providers/gotify"); | ||||
| const HomeAssistant = require("./notification-providers/home-assistant"); | ||||
| const Line = require("./notification-providers/line"); | ||||
| const LineNotify = require("./notification-providers/linenotify"); | ||||
| const LunaSea = require("./notification-providers/lunasea"); | ||||
| const Matrix = require("./notification-providers/matrix"); | ||||
| const Mattermost = require("./notification-providers/mattermost"); | ||||
| @@ -29,13 +32,18 @@ const RocketChat = require("./notification-providers/rocket-chat"); | ||||
| const SerwerSMS = require("./notification-providers/serwersms"); | ||||
| const Signal = require("./notification-providers/signal"); | ||||
| const Slack = require("./notification-providers/slack"); | ||||
| const SMSEagle = require("./notification-providers/smseagle"); | ||||
| const SMTP = require("./notification-providers/smtp"); | ||||
| const Squadcast = require("./notification-providers/squadcast"); | ||||
| const Stackfield = require("./notification-providers/stackfield"); | ||||
| const Teams = require("./notification-providers/teams"); | ||||
| const TechulusPush = require("./notification-providers/techulus-push"); | ||||
| const Telegram = require("./notification-providers/telegram"); | ||||
| const Webhook = require("./notification-providers/webhook"); | ||||
| const WeCom = require("./notification-providers/wecom"); | ||||
| const GoAlert = require("./notification-providers/goalert"); | ||||
| const SMSManager = require("./notification-providers/smsmanager"); | ||||
| const ServerChan = require("./notification-providers/serverchan"); | ||||
|  | ||||
| class Notification { | ||||
|  | ||||
| @@ -57,10 +65,13 @@ class Notification { | ||||
|             new DingDing(), | ||||
|             new Discord(), | ||||
|             new Feishu(), | ||||
|             new FreeMobile(), | ||||
|             new GoogleChat(), | ||||
|             new Gorush(), | ||||
|             new Gotify(), | ||||
|             new HomeAssistant(), | ||||
|             new Line(), | ||||
|             new LineNotify(), | ||||
|             new LunaSea(), | ||||
|             new Matrix(), | ||||
|             new Mattermost(), | ||||
| @@ -74,16 +85,21 @@ class Notification { | ||||
|             new Pushover(), | ||||
|             new Pushy(), | ||||
|             new RocketChat(), | ||||
|             new ServerChan(), | ||||
|             new SerwerSMS(), | ||||
|             new Signal(), | ||||
|             new SMSManager(), | ||||
|             new Slack(), | ||||
|             new SMSEagle(), | ||||
|             new SMTP(), | ||||
|             new Squadcast(), | ||||
|             new Stackfield(), | ||||
|             new Teams(), | ||||
|             new TechulusPush(), | ||||
|             new Telegram(), | ||||
|             new Webhook(), | ||||
|             new WeCom(), | ||||
|             new GoAlert(), | ||||
|         ]; | ||||
|  | ||||
|         for (let item of list) { | ||||
|   | ||||
| @@ -105,7 +105,7 @@ Ping.prototype.send = function (callback) { | ||||
|     let _exited; | ||||
|     let _errored; | ||||
|  | ||||
|     this._ping = spawn(this._bin, this._args); // spawn the binary | ||||
|     this._ping = spawn(this._bin, this._args, { windowsHide: true }); // spawn the binary | ||||
|  | ||||
|     this._ping.on("error", function (err) { // handle binary errors | ||||
|         _errored = true; | ||||
|   | ||||
| @@ -7,7 +7,7 @@ const { UptimeKumaServer } = require("./uptime-kuma-server"); | ||||
|  | ||||
| class Proxy { | ||||
|  | ||||
|     static SUPPORTED_PROXY_PROTOCOLS = [ "http", "https", "socks", "socks5", "socks4" ]; | ||||
|     static SUPPORTED_PROXY_PROTOCOLS = [ "http", "https", "socks", "socks5", "socks5h", "socks4" ]; | ||||
|  | ||||
|     /** | ||||
|      * Saves and updates given proxy entity | ||||
| @@ -126,6 +126,7 @@ class Proxy { | ||||
|                 break; | ||||
|             case "socks": | ||||
|             case "socks5": | ||||
|             case "socks5h": | ||||
|             case "socks4": | ||||
|                 agent = new SocksProxyAgent({ | ||||
|                     ...httpAgentOptions, | ||||
|   | ||||
| @@ -4,7 +4,7 @@ const { R } = require("redbean-node"); | ||||
| const apicache = require("../modules/apicache"); | ||||
| const Monitor = require("../model/monitor"); | ||||
| const dayjs = require("dayjs"); | ||||
| const { UP, DOWN, flipStatus, log } = require("../../src/util"); | ||||
| const { UP, MAINTENANCE, DOWN, flipStatus, log } = require("../../src/util"); | ||||
| const StatusPage = require("../model/status_page"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const { makeBadge } = require("badge-maker"); | ||||
| @@ -67,6 +67,11 @@ router.get("/api/push/:pushToken", async (request, response) => { | ||||
|             duration = dayjs(bean.time).diff(dayjs(previousHeartbeat.time), "second"); | ||||
|         } | ||||
|  | ||||
|         if (await Monitor.isUnderMaintenance(monitor.id)) { | ||||
|             msg = "Monitor under maintenance"; | ||||
|             status = MAINTENANCE; | ||||
|         } | ||||
|  | ||||
|         log.debug("router", `/api/push/ called at ${dayjs().format("YYYY-MM-DD HH:mm:ss.SSS")}`); | ||||
|         log.debug("router", "PreviousStatus: " + previousStatus); | ||||
|         log.debug("router", "Current Status: " + status); | ||||
| @@ -87,7 +92,7 @@ router.get("/api/push/:pushToken", async (request, response) => { | ||||
|             ok: true, | ||||
|         }); | ||||
|  | ||||
|         if (bean.important) { | ||||
|         if (Monitor.isImportantForNotification(isFirstBeat, previousStatus, status)) { | ||||
|             await Monitor.sendNotification(isFirstBeat, monitor, bean); | ||||
|         } | ||||
|  | ||||
| @@ -136,6 +141,7 @@ router.get("/api/badge/:id/status", cache("5 minutes"), async (request, response | ||||
|             const heartbeat = await Monitor.getPreviousHeartbeat(requestedMonitorId); | ||||
|             const state = overrideValue !== undefined ? overrideValue : heartbeat.status === 1; | ||||
|  | ||||
|             badgeValues.label = label ? label : ""; | ||||
|             badgeValues.color = state ? upColor : downColor; | ||||
|             badgeValues.message = label ?? state ? upLabel : downLabel; | ||||
|         } | ||||
|   | ||||
							
								
								
									
										140
									
								
								server/server.js
									
									
									
									
									
								
							
							
						
						
									
										140
									
								
								server/server.js
									
									
									
									
									
								
							| @@ -5,6 +5,12 @@ | ||||
|  */ | ||||
| console.log("Welcome to Uptime Kuma"); | ||||
|  | ||||
| // As the log function need to use dayjs, it should be very top | ||||
| const dayjs = require("dayjs"); | ||||
| dayjs.extend(require("dayjs/plugin/utc")); | ||||
| dayjs.extend(require("./modules/dayjs/plugin/timezone")); | ||||
| dayjs.extend(require("dayjs/plugin/customParseFormat")); | ||||
|  | ||||
| // Check Node.js Version | ||||
| const nodeVersion = parseInt(process.versions.node.split(".")[0]); | ||||
| const requiredVersion = 14; | ||||
| @@ -33,6 +39,7 @@ log.info("server", "Importing Node libraries"); | ||||
| const fs = require("fs"); | ||||
|  | ||||
| log.info("server", "Importing 3rd-party libraries"); | ||||
|  | ||||
| log.debug("server", "Importing express"); | ||||
| const express = require("express"); | ||||
| const expressStaticGzip = require("express-static-gzip"); | ||||
| @@ -61,7 +68,7 @@ log.info("server", "Importing this project modules"); | ||||
| log.debug("server", "Importing Monitor"); | ||||
| const Monitor = require("./model/monitor"); | ||||
| log.debug("server", "Importing Settings"); | ||||
| const { getSettings, setSettings, setting, initJWTSecret, checkLogin, startUnitTest, FBSD, doubleCheckPassword } = require("./util-server"); | ||||
| const { getSettings, setSettings, setting, initJWTSecret, checkLogin, startUnitTest, FBSD, doubleCheckPassword, startE2eTests } = require("./util-server"); | ||||
|  | ||||
| log.debug("server", "Importing Notification"); | ||||
| const { Notification } = require("./notification"); | ||||
| @@ -112,19 +119,25 @@ const twoFAVerifyOptions = { | ||||
|  * @type {boolean} | ||||
|  */ | ||||
| const testMode = !!args["test"] || false; | ||||
| const e2eTestMode = !!args["e2e"] || false; | ||||
|  | ||||
| if (config.demoMode) { | ||||
|     log.info("server", "==== Demo Mode ===="); | ||||
| } | ||||
|  | ||||
| // Must be after io instantiation | ||||
| const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList, sendInfo, sendProxyList } = require("./client"); | ||||
| const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList, sendInfo, sendProxyList, sendDockerHostList } = require("./client"); | ||||
| const { statusPageSocketHandler } = require("./socket-handlers/status-page-socket-handler"); | ||||
| const databaseSocketHandler = require("./socket-handlers/database-socket-handler"); | ||||
| const TwoFA = require("./2fa"); | ||||
| const StatusPage = require("./model/status_page"); | ||||
| const { cloudflaredSocketHandler, autoStart: cloudflaredAutoStart, stop: cloudflaredStop } = require("./socket-handlers/cloudflared-socket-handler"); | ||||
| const { proxySocketHandler } = require("./socket-handlers/proxy-socket-handler"); | ||||
| const { dockerSocketHandler } = require("./socket-handlers/docker-socket-handler"); | ||||
| const { maintenanceSocketHandler } = require("./socket-handlers/maintenance-socket-handler"); | ||||
| const { generalSocketHandler } = require("./socket-handlers/general-socket-handler"); | ||||
| const { Settings } = require("./settings"); | ||||
| const { CacheableDnsHttpAgent } = require("./cacheable-dns-http-agent"); | ||||
|  | ||||
| app.use(express.json()); | ||||
|  | ||||
| @@ -152,8 +165,9 @@ let needSetup = false; | ||||
| (async () => { | ||||
|     Database.init(args); | ||||
|     await initDatabase(testMode); | ||||
|     await server.initAfterDatabaseReady(); | ||||
|  | ||||
|     exports.entryPage = await setting("entryPage"); | ||||
|     server.entryPage = await Settings.get("entryPage"); | ||||
|     await StatusPage.loadDomainMappingList(); | ||||
|  | ||||
|     log.info("server", "Adding route"); | ||||
| @@ -164,16 +178,25 @@ let needSetup = false; | ||||
|  | ||||
|     // Entry Page | ||||
|     app.get("/", async (request, response) => { | ||||
|         log.debug("entry", `Request Domain: ${request.hostname}`); | ||||
|         let hostname = request.hostname; | ||||
|         if (await setting("trustProxy")) { | ||||
|             const proxy = request.headers["x-forwarded-host"]; | ||||
|             if (proxy) { | ||||
|                 hostname = proxy; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (request.hostname in StatusPage.domainMappingList) { | ||||
|         log.debug("entry", `Request Domain: ${hostname}`); | ||||
|  | ||||
|         const uptimeKumaEntryPage = server.entryPage; | ||||
|         if (hostname in StatusPage.domainMappingList) { | ||||
|             log.debug("entry", "This is a status page domain"); | ||||
|  | ||||
|             let slug = StatusPage.domainMappingList[request.hostname]; | ||||
|             let slug = StatusPage.domainMappingList[hostname]; | ||||
|             await StatusPage.handleStatusPageResponse(response, server.indexHTML, slug); | ||||
|  | ||||
|         } else if (exports.entryPage && exports.entryPage.startsWith("statusPage-")) { | ||||
|             response.redirect("/status/" + exports.entryPage.replace("statusPage-", "")); | ||||
|         } else if (uptimeKumaEntryPage && uptimeKumaEntryPage.startsWith("statusPage-")) { | ||||
|             response.redirect("/status/" + uptimeKumaEntryPage.replace("statusPage-", "")); | ||||
|  | ||||
|         } else { | ||||
|             response.redirect("/dashboard"); | ||||
| @@ -182,6 +205,7 @@ let needSetup = false; | ||||
|  | ||||
|     if (isDev) { | ||||
|         app.post("/test-webhook", async (request, response) => { | ||||
|             log.debug("test", request.headers); | ||||
|             log.debug("test", request.body); | ||||
|             response.send("OK"); | ||||
|         }); | ||||
| @@ -190,7 +214,7 @@ let needSetup = false; | ||||
|     // Robots.txt | ||||
|     app.get("/robots.txt", async (_request, response) => { | ||||
|         let txt = "User-agent: *\nDisallow:"; | ||||
|         if (! await setting("searchEngineIndex")) { | ||||
|         if (!await setting("searchEngineIndex")) { | ||||
|             txt += " /"; | ||||
|         } | ||||
|         response.setHeader("Content-Type", "text/plain"); | ||||
| @@ -246,7 +270,9 @@ let needSetup = false; | ||||
|         // *************************** | ||||
|  | ||||
|         socket.on("loginByToken", async (token, callback) => { | ||||
|             log.info("auth", `Login by token. IP=${getClientIp(socket)}`); | ||||
|             const clientIP = await server.getClientIP(socket); | ||||
|  | ||||
|             log.info("auth", `Login by token. IP=${clientIP}`); | ||||
|  | ||||
|             try { | ||||
|                 let decoded = jwt.verify(token, jwtSecret); | ||||
| @@ -262,14 +288,14 @@ let needSetup = false; | ||||
|                     afterLogin(socket, user); | ||||
|                     log.debug("auth", "afterLogin ok"); | ||||
|  | ||||
|                     log.info("auth", `Successfully logged in user ${decoded.username}. IP=${getClientIp(socket)}`); | ||||
|                     log.info("auth", `Successfully logged in user ${decoded.username}. IP=${clientIP}`); | ||||
|  | ||||
|                     callback({ | ||||
|                         ok: true, | ||||
|                     }); | ||||
|                 } else { | ||||
|  | ||||
|                     log.info("auth", `Inactive or deleted user ${decoded.username}. IP=${getClientIp(socket)}`); | ||||
|                     log.info("auth", `Inactive or deleted user ${decoded.username}. IP=${clientIP}`); | ||||
|  | ||||
|                     callback({ | ||||
|                         ok: false, | ||||
| @@ -278,7 +304,7 @@ let needSetup = false; | ||||
|                 } | ||||
|             } catch (error) { | ||||
|  | ||||
|                 log.error("auth", `Invalid token. IP=${getClientIp(socket)}`); | ||||
|                 log.error("auth", `Invalid token. IP=${clientIP}`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: false, | ||||
| @@ -289,7 +315,9 @@ let needSetup = false; | ||||
|         }); | ||||
|  | ||||
|         socket.on("login", async (data, callback) => { | ||||
|             log.info("auth", `Login by username + password. IP=${getClientIp(socket)}`); | ||||
|             const clientIP = await server.getClientIP(socket); | ||||
|  | ||||
|             log.info("auth", `Login by username + password. IP=${clientIP}`); | ||||
|  | ||||
|             // Checking | ||||
|             if (typeof callback !== "function") { | ||||
| @@ -302,7 +330,7 @@ let needSetup = false; | ||||
|  | ||||
|             // Login Rate Limit | ||||
|             if (! await loginRateLimiter.pass(callback)) { | ||||
|                 log.info("auth", `Too many failed requests for user ${data.username}. IP=${getClientIp(socket)}`); | ||||
|                 log.info("auth", `Too many failed requests for user ${data.username}. IP=${clientIP}`); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
| @@ -312,7 +340,7 @@ let needSetup = false; | ||||
|                 if (user.twofa_status === 0) { | ||||
|                     afterLogin(socket, user); | ||||
|  | ||||
|                     log.info("auth", `Successfully logged in user ${data.username}. IP=${getClientIp(socket)}`); | ||||
|                     log.info("auth", `Successfully logged in user ${data.username}. IP=${clientIP}`); | ||||
|  | ||||
|                     callback({ | ||||
|                         ok: true, | ||||
| @@ -324,7 +352,7 @@ let needSetup = false; | ||||
|  | ||||
|                 if (user.twofa_status === 1 && !data.token) { | ||||
|  | ||||
|                     log.info("auth", `2FA token required for user ${data.username}. IP=${getClientIp(socket)}`); | ||||
|                     log.info("auth", `2FA token required for user ${data.username}. IP=${clientIP}`); | ||||
|  | ||||
|                     callback({ | ||||
|                         tokenRequired: true, | ||||
| @@ -342,7 +370,7 @@ let needSetup = false; | ||||
|                             socket.userID, | ||||
|                         ]); | ||||
|  | ||||
|                         log.info("auth", `Successfully logged in user ${data.username}. IP=${getClientIp(socket)}`); | ||||
|                         log.info("auth", `Successfully logged in user ${data.username}. IP=${clientIP}`); | ||||
|  | ||||
|                         callback({ | ||||
|                             ok: true, | ||||
| @@ -352,7 +380,7 @@ let needSetup = false; | ||||
|                         }); | ||||
|                     } else { | ||||
|  | ||||
|                         log.warn("auth", `Invalid token provided for user ${data.username}. IP=${getClientIp(socket)}`); | ||||
|                         log.warn("auth", `Invalid token provided for user ${data.username}. IP=${clientIP}`); | ||||
|  | ||||
|                         callback({ | ||||
|                             ok: false, | ||||
| @@ -362,7 +390,7 @@ let needSetup = false; | ||||
|                 } | ||||
|             } else { | ||||
|  | ||||
|                 log.warn("auth", `Incorrect username or password for user ${data.username}. IP=${getClientIp(socket)}`); | ||||
|                 log.warn("auth", `Incorrect username or password for user ${data.username}. IP=${clientIP}`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: false, | ||||
| @@ -434,6 +462,8 @@ let needSetup = false; | ||||
|         }); | ||||
|  | ||||
|         socket.on("save2FA", async (currentPassword, callback) => { | ||||
|             const clientIP = await server.getClientIP(socket); | ||||
|  | ||||
|             try { | ||||
|                 if (! await twoFaRateLimiter.pass(callback)) { | ||||
|                     return; | ||||
| @@ -446,7 +476,7 @@ let needSetup = false; | ||||
|                     socket.userID, | ||||
|                 ]); | ||||
|  | ||||
|                 log.info("auth", `Saved 2FA token. IP=${getClientIp(socket)}`); | ||||
|                 log.info("auth", `Saved 2FA token. IP=${clientIP}`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: true, | ||||
| @@ -454,7 +484,7 @@ let needSetup = false; | ||||
|                 }); | ||||
|             } catch (error) { | ||||
|  | ||||
|                 log.error("auth", `Error changing 2FA token. IP=${getClientIp(socket)}`); | ||||
|                 log.error("auth", `Error changing 2FA token. IP=${clientIP}`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: false, | ||||
| @@ -464,6 +494,8 @@ let needSetup = false; | ||||
|         }); | ||||
|  | ||||
|         socket.on("disable2FA", async (currentPassword, callback) => { | ||||
|             const clientIP = await server.getClientIP(socket); | ||||
|  | ||||
|             try { | ||||
|                 if (! await twoFaRateLimiter.pass(callback)) { | ||||
|                     return; | ||||
| @@ -473,7 +505,7 @@ let needSetup = false; | ||||
|                 await doubleCheckPassword(socket, currentPassword); | ||||
|                 await TwoFA.disable2FA(socket.userID); | ||||
|  | ||||
|                 log.info("auth", `Disabled 2FA token. IP=${getClientIp(socket)}`); | ||||
|                 log.info("auth", `Disabled 2FA token. IP=${clientIP}`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: true, | ||||
| @@ -481,7 +513,7 @@ let needSetup = false; | ||||
|                 }); | ||||
|             } catch (error) { | ||||
|  | ||||
|                 log.error("auth", `Error disabling 2FA token. IP=${getClientIp(socket)}`); | ||||
|                 log.error("auth", `Error disabling 2FA token. IP=${clientIP}`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: false, | ||||
| @@ -602,6 +634,9 @@ let needSetup = false; | ||||
|  | ||||
|                 bean.import(monitor); | ||||
|                 bean.user_id = socket.userID; | ||||
|  | ||||
|                 bean.validate(); | ||||
|  | ||||
|                 await R.store(bean); | ||||
|  | ||||
|                 await updateMonitorNotification(bean.id, notificationIDList); | ||||
| @@ -652,6 +687,7 @@ let needSetup = false; | ||||
|                 bean.basic_auth_pass = monitor.basic_auth_pass; | ||||
|                 bean.interval = monitor.interval; | ||||
|                 bean.retryInterval = monitor.retryInterval; | ||||
|                 bean.resendInterval = monitor.resendInterval; | ||||
|                 bean.hostname = monitor.hostname; | ||||
|                 bean.maxretries = monitor.maxretries; | ||||
|                 bean.port = parseInt(monitor.port); | ||||
| @@ -664,6 +700,8 @@ let needSetup = false; | ||||
|                 bean.dns_resolve_type = monitor.dns_resolve_type; | ||||
|                 bean.dns_resolve_server = monitor.dns_resolve_server; | ||||
|                 bean.pushToken = monitor.pushToken; | ||||
|                 bean.docker_container = monitor.docker_container; | ||||
|                 bean.docker_host = monitor.docker_host; | ||||
|                 bean.proxyId = Number.isInteger(monitor.proxyId) ? monitor.proxyId : null; | ||||
|                 bean.mqttUsername = monitor.mqttUsername; | ||||
|                 bean.mqttPassword = monitor.mqttPassword; | ||||
| @@ -674,6 +712,19 @@ let needSetup = false; | ||||
|                 bean.authMethod = monitor.authMethod; | ||||
|                 bean.authWorkstation = monitor.authWorkstation; | ||||
|                 bean.authDomain = monitor.authDomain; | ||||
|                 bean.grpcUrl = monitor.grpcUrl; | ||||
|                 bean.grpcProtobuf = monitor.grpcProtobuf; | ||||
|                 bean.grpcMethod = monitor.grpcMethod; | ||||
|                 bean.grpcBody = monitor.grpcBody; | ||||
|                 bean.grpcMetadata = monitor.grpcMetadata; | ||||
|                 bean.grpcEnableTls = monitor.grpcEnableTls; | ||||
|                 bean.radiusUsername = monitor.radiusUsername; | ||||
|                 bean.radiusPassword = monitor.radiusPassword; | ||||
|                 bean.radiusCalledStationId = monitor.radiusCalledStationId; | ||||
|                 bean.radiusCallingStationId = monitor.radiusCallingStationId; | ||||
|                 bean.radiusSecret = monitor.radiusSecret; | ||||
|  | ||||
|                 bean.validate(); | ||||
|  | ||||
|                 await R.store(bean); | ||||
|  | ||||
| @@ -1029,10 +1080,15 @@ let needSetup = false; | ||||
|         socket.on("getSettings", async (callback) => { | ||||
|             try { | ||||
|                 checkLogin(socket); | ||||
|                 const data = await getSettings("general"); | ||||
|  | ||||
|                 if (!data.serverTimezone) { | ||||
|                     data.serverTimezone = await server.getTimezone(); | ||||
|                 } | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: true, | ||||
|                     data: await getSettings("general"), | ||||
|                     data: data, | ||||
|                 }); | ||||
|  | ||||
|             } catch (e) { | ||||
| @@ -1058,7 +1114,14 @@ let needSetup = false; | ||||
|                 } | ||||
|  | ||||
|                 await setSettings("general", data); | ||||
|                 exports.entryPage = data.entryPage; | ||||
|                 server.entryPage = data.entryPage; | ||||
|  | ||||
|                 await CacheableDnsHttpAgent.update(); | ||||
|  | ||||
|                 // Also need to apply timezone globally | ||||
|                 if (data.serverTimezone) { | ||||
|                     await server.setTimezone(data.serverTimezone); | ||||
|                 } | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: true, | ||||
| @@ -1066,6 +1129,7 @@ let needSetup = false; | ||||
|                 }); | ||||
|  | ||||
|                 sendInfo(socket); | ||||
|                 server.sendMaintenanceList(socket); | ||||
|  | ||||
|             } catch (e) { | ||||
|                 callback({ | ||||
| @@ -1254,6 +1318,7 @@ let needSetup = false; | ||||
|                                 authDomain: monitorListData[i].authDomain, | ||||
|                                 interval: monitorListData[i].interval, | ||||
|                                 retryInterval: retryInterval, | ||||
|                                 resendInterval: monitorListData[i].resendInterval || 0, | ||||
|                                 hostname: monitorListData[i].hostname, | ||||
|                                 maxretries: monitorListData[i].maxretries, | ||||
|                                 port: monitorListData[i].port, | ||||
| @@ -1422,6 +1487,9 @@ let needSetup = false; | ||||
|         cloudflaredSocketHandler(socket); | ||||
|         databaseSocketHandler(socket); | ||||
|         proxySocketHandler(socket); | ||||
|         dockerSocketHandler(socket); | ||||
|         maintenanceSocketHandler(socket); | ||||
|         generalSocketHandler(socket, server); | ||||
|  | ||||
|         log.debug("server", "added all socket handlers"); | ||||
|  | ||||
| @@ -1459,6 +1527,10 @@ let needSetup = false; | ||||
|         if (testMode) { | ||||
|             startUnitTest(); | ||||
|         } | ||||
|  | ||||
|         if (e2eTestMode) { | ||||
|             startE2eTests(); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     initBackgroundJobs(args); | ||||
| @@ -1520,8 +1592,10 @@ async function afterLogin(socket, user) { | ||||
|     socket.join(user.id); | ||||
|  | ||||
|     let monitorList = await server.sendMonitorList(socket); | ||||
|     server.sendMaintenanceList(socket); | ||||
|     sendNotificationList(socket); | ||||
|     sendProxyList(socket); | ||||
|     sendDockerHostList(socket); | ||||
|  | ||||
|     await sleep(500); | ||||
|  | ||||
| @@ -1538,6 +1612,13 @@ async function afterLogin(socket, user) { | ||||
|     for (let monitorID in monitorList) { | ||||
|         await Monitor.sendStats(io, monitorID, user.id); | ||||
|     } | ||||
|  | ||||
|     // Set server timezone from client browser if not set | ||||
|     // It should be run once only | ||||
|     if (! await Settings.get("initServerTimezone")) { | ||||
|         log.debug("server", "emit initServerTimezone"); | ||||
|         socket.emit("initServerTimezone"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -1664,6 +1745,8 @@ async function shutdownFunction(signal) { | ||||
|     log.info("server", "Shutdown requested"); | ||||
|     log.info("server", "Called signal: " + signal); | ||||
|  | ||||
|     await server.stop(); | ||||
|  | ||||
|     log.info("server", "Stopping all monitors"); | ||||
|     for (let id in server.monitorList) { | ||||
|         let monitor = server.monitorList[id]; | ||||
| @@ -1674,10 +1757,7 @@ async function shutdownFunction(signal) { | ||||
|  | ||||
|     stopBackgroundJobs(); | ||||
|     await cloudflaredStop(); | ||||
| } | ||||
|  | ||||
| function getClientIp(socket) { | ||||
|     return socket.client.conn.remoteAddress.replace(/^.*:/, ""); | ||||
|     Settings.stopCacheCleaner(); | ||||
| } | ||||
|  | ||||
| /** Final function called before application exits */ | ||||
|   | ||||
| @@ -158,6 +158,13 @@ class Settings { | ||||
|             delete Settings.cacheList[key]; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     static stopCacheCleaner() { | ||||
|         if (Settings.cacheCleaner) { | ||||
|             clearInterval(Settings.cacheCleaner); | ||||
|             Settings.cacheCleaner = null; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| const { checkLogin, setSetting, setting, doubleCheckPassword } = require("../util-server"); | ||||
| const { CloudflaredTunnel } = require("node-cloudflared-tunnel"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const { log } = require("../../src/util"); | ||||
| const io = UptimeKumaServer.getInstance().io; | ||||
|  | ||||
| const prefix = "cloudflared_"; | ||||
| @@ -107,7 +108,7 @@ module.exports.autoStart = async (token) => { | ||||
|  | ||||
| /** Stop cloudflared */ | ||||
| module.exports.stop = async () => { | ||||
|     console.log("Stop cloudflared"); | ||||
|     log.info("cloudflared", "Stop cloudflared"); | ||||
|     if (cloudflared) { | ||||
|         cloudflared.stop(); | ||||
|     } | ||||
|   | ||||
							
								
								
									
										79
									
								
								server/socket-handlers/docker-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								server/socket-handlers/docker-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | ||||
| const { sendDockerHostList } = require("../client"); | ||||
| const { checkLogin } = require("../util-server"); | ||||
| const { DockerHost } = require("../docker"); | ||||
| const { log } = require("../../src/util"); | ||||
|  | ||||
| /** | ||||
|  * Handlers for docker hosts | ||||
|  * @param {Socket} socket Socket.io instance | ||||
|  */ | ||||
| module.exports.dockerSocketHandler = (socket) => { | ||||
|     socket.on("addDockerHost", async (dockerHost, dockerHostID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             let dockerHostBean = await DockerHost.save(dockerHost, dockerHostID, socket.userID); | ||||
|             await sendDockerHostList(socket); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Saved", | ||||
|                 id: dockerHostBean.id, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("deleteDockerHost", async (dockerHostID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             await DockerHost.delete(dockerHostID, socket.userID); | ||||
|             await sendDockerHostList(socket); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Deleted", | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("testDockerHost", async (dockerHost, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             let amount = await DockerHost.testDockerHost(dockerHost); | ||||
|             let msg; | ||||
|  | ||||
|             if (amount >= 1) { | ||||
|                 msg = "Connected Successfully. Amount of containers: " + amount; | ||||
|             } else { | ||||
|                 msg = "Connected Successfully, but there are no containers?"; | ||||
|             } | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             log.error("docker", e); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
| }; | ||||
							
								
								
									
										20
									
								
								server/socket-handlers/general-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								server/socket-handlers/general-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| const { log } = require("../../src/util"); | ||||
| const { Settings } = require("../settings"); | ||||
| const { sendInfo } = require("../client"); | ||||
| const { checkLogin } = require("../util-server"); | ||||
|  | ||||
| module.exports.generalSocketHandler = (socket, server) => { | ||||
|  | ||||
|     socket.on("initServerTimezone", async (timezone) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|             log.debug("generalSocketHandler", "Timezone: " + timezone); | ||||
|             await Settings.set("initServerTimezone", true); | ||||
|             await server.setTimezone(timezone); | ||||
|             await sendInfo(socket); | ||||
|         } catch (e) { | ||||
|             log.warn("initServerTimezone", e.message); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| }; | ||||
							
								
								
									
										311
									
								
								server/socket-handlers/maintenance-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										311
									
								
								server/socket-handlers/maintenance-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,311 @@ | ||||
| const { checkLogin } = require("../util-server"); | ||||
| const { log } = require("../../src/util"); | ||||
| const { R } = require("redbean-node"); | ||||
| const apicache = require("../modules/apicache"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const Maintenance = require("../model/maintenance"); | ||||
| const server = UptimeKumaServer.getInstance(); | ||||
| const MaintenanceTimeslot = require("../model/maintenance_timeslot"); | ||||
|  | ||||
| /** | ||||
|  * Handlers for Maintenance | ||||
|  * @param {Socket} socket Socket.io instance | ||||
|  */ | ||||
| module.exports.maintenanceSocketHandler = (socket) => { | ||||
|     // Add a new maintenance | ||||
|     socket.on("addMaintenance", async (maintenance, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", maintenance); | ||||
|  | ||||
|             let bean = Maintenance.jsonToBean(R.dispense("maintenance"), maintenance); | ||||
|             bean.user_id = socket.userID; | ||||
|             let maintenanceID = await R.store(bean); | ||||
|             await MaintenanceTimeslot.generateTimeslot(bean); | ||||
|  | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Added Successfully.", | ||||
|                 maintenanceID, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     // Edit a maintenance | ||||
|     socket.on("editMaintenance", async (maintenance, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             let bean = await R.findOne("maintenance", " id = ? ", [ maintenance.id ]); | ||||
|  | ||||
|             if (bean.user_id !== socket.userID) { | ||||
|                 throw new Error("Permission denied."); | ||||
|             } | ||||
|  | ||||
|             Maintenance.jsonToBean(bean, maintenance); | ||||
|  | ||||
|             await R.store(bean); | ||||
|             await MaintenanceTimeslot.generateTimeslot(bean, null, true); | ||||
|  | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Saved.", | ||||
|                 maintenanceID: bean.id, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             console.error(e); | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     // Add a new monitor_maintenance | ||||
|     socket.on("addMonitorMaintenance", async (maintenanceID, monitors, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             await R.exec("DELETE FROM monitor_maintenance WHERE maintenance_id = ?", [ | ||||
|                 maintenanceID | ||||
|             ]); | ||||
|  | ||||
|             for await (const monitor of monitors) { | ||||
|                 let bean = R.dispense("monitor_maintenance"); | ||||
|  | ||||
|                 bean.import({ | ||||
|                     monitor_id: monitor.id, | ||||
|                     maintenance_id: maintenanceID | ||||
|                 }); | ||||
|                 await R.store(bean); | ||||
|             } | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Added Successfully.", | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     // Add a new monitor_maintenance | ||||
|     socket.on("addMaintenanceStatusPage", async (maintenanceID, statusPages, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             await R.exec("DELETE FROM maintenance_status_page WHERE maintenance_id = ?", [ | ||||
|                 maintenanceID | ||||
|             ]); | ||||
|  | ||||
|             for await (const statusPage of statusPages) { | ||||
|                 let bean = R.dispense("maintenance_status_page"); | ||||
|  | ||||
|                 bean.import({ | ||||
|                     status_page_id: statusPage.id, | ||||
|                     maintenance_id: maintenanceID | ||||
|                 }); | ||||
|                 await R.store(bean); | ||||
|             } | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Added Successfully.", | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("getMaintenance", async (maintenanceID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", `Get Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             let bean = await R.findOne("maintenance", " id = ? AND user_id = ? ", [ | ||||
|                 maintenanceID, | ||||
|                 socket.userID, | ||||
|             ]); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 maintenance: await bean.toJSON(), | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("getMaintenanceList", async (callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|             await server.sendMaintenanceList(socket); | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|             }); | ||||
|         } catch (e) { | ||||
|             console.error(e); | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("getMonitorMaintenance", async (maintenanceID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", `Get Monitors for Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             let monitors = await R.getAll("SELECT monitor.id, monitor.name FROM monitor_maintenance mm JOIN monitor ON mm.monitor_id = monitor.id WHERE mm.maintenance_id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 monitors, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             console.error(e); | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("getMaintenanceStatusPage", async (maintenanceID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", `Get Status Pages for Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             let statusPages = await R.getAll("SELECT status_page.id, status_page.title FROM maintenance_status_page msp JOIN status_page ON msp.status_page_id = status_page.id WHERE msp.maintenance_id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 statusPages, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             console.error(e); | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("deleteMaintenance", async (maintenanceID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", `Delete Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             if (maintenanceID in server.maintenanceList) { | ||||
|                 delete server.maintenanceList[maintenanceID]; | ||||
|             } | ||||
|  | ||||
|             await R.exec("DELETE FROM maintenance WHERE id = ? AND user_id = ? ", [ | ||||
|                 maintenanceID, | ||||
|                 socket.userID, | ||||
|             ]); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Deleted Successfully.", | ||||
|             }); | ||||
|  | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("pauseMaintenance", async (maintenanceID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", `Pause Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("UPDATE maintenance SET active = 0 WHERE id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Paused Successfully.", | ||||
|             }); | ||||
|  | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("resumeMaintenance", async (maintenanceID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("maintenance", `Resume Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("UPDATE maintenance SET active = 1 WHERE id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Resume Successfully", | ||||
|             }); | ||||
|  | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
| }; | ||||
| @@ -202,7 +202,11 @@ module.exports.statusPageSocketHandler = (socket) => { | ||||
|                     relationBean.weight = monitorOrder++; | ||||
|                     relationBean.group_id = groupBean.id; | ||||
|                     relationBean.monitor_id = monitor.id; | ||||
|                     relationBean.send_url = monitor.sendUrl; | ||||
|  | ||||
|                     if (monitor.sendUrl !== undefined) { | ||||
|                         relationBean.send_url = monitor.sendUrl; | ||||
|                     } | ||||
|  | ||||
|                     await R.store(relationBean); | ||||
|                 } | ||||
|  | ||||
|   | ||||
							
								
								
									
										39
									
								
								server/uptime-cache-list.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								server/uptime-cache-list.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| const { log } = require("../src/util"); | ||||
| class UptimeCacheList { | ||||
|     /** | ||||
|      * list[monitorID][duration] | ||||
|      */ | ||||
|     static list = {}; | ||||
|  | ||||
|     /** | ||||
|      * | ||||
|      * @param monitorID | ||||
|      * @param duration | ||||
|      * @return number | ||||
|      */ | ||||
|     static getUptime(monitorID, duration) { | ||||
|         if (UptimeCacheList.list[monitorID] && UptimeCacheList.list[monitorID][duration]) { | ||||
|             log.debug("UptimeCacheList", "getUptime: " + monitorID + " " + duration); | ||||
|             return UptimeCacheList.list[monitorID][duration]; | ||||
|         } else { | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     static addUptime(monitorID, duration, uptime) { | ||||
|         log.debug("UptimeCacheList", "addUptime: " + monitorID + " " + duration); | ||||
|         if (!UptimeCacheList.list[monitorID]) { | ||||
|             UptimeCacheList.list[monitorID] = {}; | ||||
|         } | ||||
|         UptimeCacheList.list[monitorID][duration] = uptime; | ||||
|     } | ||||
|  | ||||
|     static clearCache(monitorID) { | ||||
|         log.debug("UptimeCacheList", "clearCache: " + monitorID); | ||||
|         delete UptimeCacheList.list[monitorID]; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     UptimeCacheList, | ||||
| }; | ||||
| @@ -8,6 +8,9 @@ const { log } = require("../src/util"); | ||||
| const Database = require("./database"); | ||||
| const util = require("util"); | ||||
| const { CacheableDnsHttpAgent } = require("./cacheable-dns-http-agent"); | ||||
| const { Settings } = require("./settings"); | ||||
| const dayjs = require("dayjs"); | ||||
| // DO NOT IMPORT HERE IF THE MODULES USED `UptimeKumaServer.getInstance()` | ||||
|  | ||||
| /** | ||||
|  * `module.exports` (alias: `server`) should be inside this class, in order to avoid circular dependency issue. | ||||
| @@ -25,6 +28,13 @@ class UptimeKumaServer { | ||||
|      * @type {{}} | ||||
|      */ | ||||
|     monitorList = {}; | ||||
|  | ||||
|     /** | ||||
|      * Main maintenance list | ||||
|      * @type {{}} | ||||
|      */ | ||||
|     maintenanceList = {}; | ||||
|  | ||||
|     entryPage = "dashboard"; | ||||
|     app = undefined; | ||||
|     httpServer = undefined; | ||||
| @@ -36,6 +46,8 @@ class UptimeKumaServer { | ||||
|      */ | ||||
|     indexHTML = ""; | ||||
|  | ||||
|     generateMaintenanceTimeslotsInterval = undefined; | ||||
|  | ||||
|     static getInstance(args) { | ||||
|         if (UptimeKumaServer.instance == null) { | ||||
|             UptimeKumaServer.instance = new UptimeKumaServer(args); | ||||
| @@ -50,7 +62,6 @@ class UptimeKumaServer { | ||||
|  | ||||
|         log.info("server", "Creating express and socket.io instance"); | ||||
|         this.app = express(); | ||||
|  | ||||
|         if (sslKey && sslCert) { | ||||
|             log.info("server", "Server Type: HTTPS"); | ||||
|             this.httpServer = https.createServer({ | ||||
| @@ -72,11 +83,21 @@ class UptimeKumaServer { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         CacheableDnsHttpAgent.registerGlobalAgent(); | ||||
|  | ||||
|         this.io = new Server(this.httpServer); | ||||
|     } | ||||
|  | ||||
|     async initAfterDatabaseReady() { | ||||
|         await CacheableDnsHttpAgent.update(); | ||||
|  | ||||
|         process.env.TZ = await this.getTimezone(); | ||||
|         dayjs.tz.setDefault(process.env.TZ); | ||||
|         log.debug("DEBUG", "Timezone: " + process.env.TZ); | ||||
|         log.debug("DEBUG", "Current Time: " + dayjs.tz().format()); | ||||
|  | ||||
|         await this.generateMaintenanceTimeslots(); | ||||
|         this.generateMaintenanceTimeslotsInterval = setInterval(this.generateMaintenanceTimeslots, 60 * 1000); | ||||
|     } | ||||
|  | ||||
|     async sendMonitorList(socket) { | ||||
|         let list = await this.getMonitorJSONList(socket.userID); | ||||
|         this.io.to(socket.userID).emit("monitorList", list); | ||||
| @@ -104,6 +125,40 @@ class UptimeKumaServer { | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Send maintenance list to client | ||||
|      * @param {Socket} socket Socket.io instance to send to | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     async sendMaintenanceList(socket) { | ||||
|         return await this.sendMaintenanceListByUserID(socket.userID); | ||||
|     } | ||||
|  | ||||
|     async sendMaintenanceListByUserID(userID) { | ||||
|         let list = await this.getMaintenanceJSONList(userID); | ||||
|         this.io.to(userID).emit("maintenanceList", list); | ||||
|         return list; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a list of maintenances for the given user. | ||||
|      * @param {string} userID - The ID of the user to get maintenances for. | ||||
|      * @returns {Promise<Object>} A promise that resolves to an object with maintenance IDs as keys and maintenances objects as values. | ||||
|      */ | ||||
|     async getMaintenanceJSONList(userID) { | ||||
|         let result = {}; | ||||
|  | ||||
|         let maintenanceList = await R.find("maintenance", " user_id = ? ORDER BY end_date DESC, title", [ | ||||
|             userID, | ||||
|         ]); | ||||
|  | ||||
|         for (let maintenance of maintenanceList) { | ||||
|             result[maintenance.id] = await maintenance.toJSON(); | ||||
|         } | ||||
|  | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Write error to log file | ||||
|      * @param {any} error The error to write | ||||
| @@ -129,8 +184,67 @@ class UptimeKumaServer { | ||||
|  | ||||
|         errorLogStream.end(); | ||||
|     } | ||||
|  | ||||
|     async getClientIP(socket) { | ||||
|         let clientIP = socket.client.conn.remoteAddress; | ||||
|  | ||||
|         if (clientIP === undefined) { | ||||
|             clientIP = ""; | ||||
|         } | ||||
|  | ||||
|         if (await Settings.get("trustProxy")) { | ||||
|             const forwardedFor = socket.client.conn.request.headers["x-forwarded-for"]; | ||||
|  | ||||
|             return (typeof forwardedFor === "string" ? forwardedFor.split(",")[0].trim() : null) | ||||
|                 || socket.client.conn.request.headers["x-real-ip"] | ||||
|                 || clientIP.replace(/^.*:/, ""); | ||||
|         } else { | ||||
|             return clientIP.replace(/^.*:/, ""); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async getTimezone() { | ||||
|         let timezone = await Settings.get("serverTimezone"); | ||||
|         if (timezone) { | ||||
|             return timezone; | ||||
|         } else if (process.env.TZ) { | ||||
|             return process.env.TZ; | ||||
|         } else { | ||||
|             return dayjs.tz.guess(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     getTimezoneOffset() { | ||||
|         return dayjs().format("Z"); | ||||
|     } | ||||
|  | ||||
|     async setTimezone(timezone) { | ||||
|         await Settings.set("serverTimezone", timezone, "general"); | ||||
|         process.env.TZ = timezone; | ||||
|         dayjs.tz.setDefault(timezone); | ||||
|     } | ||||
|  | ||||
|     async generateMaintenanceTimeslots() { | ||||
|  | ||||
|         let list = await R.find("maintenance_timeslot", " generated_next = 0 AND start_date <= DATETIME('now') "); | ||||
|  | ||||
|         for (let maintenanceTimeslot of list) { | ||||
|             let maintenance = await maintenanceTimeslot.maintenance; | ||||
|             await MaintenanceTimeslot.generateTimeslot(maintenance, maintenanceTimeslot.end_date, false); | ||||
|             maintenanceTimeslot.generated_next = true; | ||||
|             await R.store(maintenanceTimeslot); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     async stop() { | ||||
|         clearTimeout(this.generateMaintenanceTimeslotsInterval); | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     UptimeKumaServer | ||||
| }; | ||||
|  | ||||
| // Must be at the end | ||||
| const MaintenanceTimeslot = require("./model/maintenance_timeslot"); | ||||
|   | ||||
| @@ -11,8 +11,20 @@ const mqtt = require("mqtt"); | ||||
| const chroma = require("chroma-js"); | ||||
| const { badgeConstants } = require("./config"); | ||||
| const mssql = require("mssql"); | ||||
| const { Client } = require("pg"); | ||||
| const postgresConParse = require("pg-connection-string").parse; | ||||
| const mysql = require("mysql2"); | ||||
| const { NtlmClient } = require("axios-ntlm"); | ||||
| const { Settings } = require("./settings"); | ||||
| const grpc = require("@grpc/grpc-js"); | ||||
| const protojs = require("protobufjs"); | ||||
| const radiusClient = require("node-radius-client"); | ||||
| const { | ||||
|     dictionaries: { | ||||
|         rfc2865: { file, attributes }, | ||||
|     }, | ||||
| } = require("node-radius-utils"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| // From ping-lite | ||||
| exports.WIN = /^win/.test(process.platform); | ||||
| @@ -251,10 +263,102 @@ exports.mssqlQuery = function (connectionString, query) { | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Run a query on Postgres | ||||
|  * @param {string} connectionString The database connection string | ||||
|  * @param {string} query The query to validate the database with | ||||
|  * @returns {Promise<(string[]|Object[]|Object)>} | ||||
|  */ | ||||
| exports.postgresQuery = function (connectionString, query) { | ||||
|     return new Promise((resolve, reject) => { | ||||
|         const config = postgresConParse(connectionString); | ||||
|  | ||||
|         if (config.password === "") { | ||||
|             // See https://github.com/brianc/node-postgres/issues/1927 | ||||
|             return reject(new Error("Password is undefined.")); | ||||
|         } | ||||
|  | ||||
|         const client = new Client({ connectionString }); | ||||
|  | ||||
|         client.connect(); | ||||
|  | ||||
|         return client.query(query) | ||||
|             .then(res => { | ||||
|                 resolve(res); | ||||
|             }) | ||||
|             .catch(err => { | ||||
|                 reject(err); | ||||
|             }) | ||||
|             .finally(() => { | ||||
|                 client.end(); | ||||
|             }); | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Run a query on MySQL/MariaDB | ||||
|  * @param {string} connectionString The database connection string | ||||
|  * @param {string} query The query to validate the database with | ||||
|  * @returns {Promise<(string[]|Object[]|Object)>} | ||||
|  */ | ||||
| exports.mysqlQuery = function (connectionString, query) { | ||||
|     return new Promise((resolve, reject) => { | ||||
|         const connection = mysql.createConnection(connectionString); | ||||
|         connection.promise().query(query) | ||||
|             .then(res => { | ||||
|                 resolve(res); | ||||
|             }) | ||||
|             .catch(err => { | ||||
|                 reject(err); | ||||
|             }) | ||||
|             .finally(() => { | ||||
|                 connection.end(); | ||||
|             }); | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Query radius server | ||||
|  * @param {string} hostname Hostname of radius server | ||||
|  * @param {string} username Username to use | ||||
|  * @param {string} password Password to use | ||||
|  * @param {string} calledStationId ID of called station | ||||
|  * @param {string} callingStationId ID of calling station | ||||
|  * @param {string} secret Secret to use | ||||
|  * @param {number} [port=1812] Port to contact radius server on | ||||
|  * @returns {Promise<any>} | ||||
|  */ | ||||
| exports.radius = function ( | ||||
|     hostname, | ||||
|     username, | ||||
|     password, | ||||
|     calledStationId, | ||||
|     callingStationId, | ||||
|     secret, | ||||
|     port = 1812, | ||||
| ) { | ||||
|     const client = new radiusClient({ | ||||
|         host: hostname, | ||||
|         hostPort: port, | ||||
|         dictionaries: [ file ], | ||||
|     }); | ||||
|  | ||||
|     return client.accessRequest({ | ||||
|         secret: secret, | ||||
|         attributes: [ | ||||
|             [ attributes.USER_NAME, username ], | ||||
|             [ attributes.USER_PASSWORD, password ], | ||||
|             [ attributes.CALLING_STATION_ID, callingStationId ], | ||||
|             [ attributes.CALLED_STATION_ID, calledStationId ], | ||||
|         ], | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Retrieve value of setting based on key | ||||
|  * @param {string} key Key of setting to retrieve | ||||
|  * @returns {Promise<any>} Value | ||||
|  * @deprecated Use await Settings.get(key) | ||||
|  */ | ||||
| exports.setting = async function (key) { | ||||
|     return await Settings.get(key); | ||||
| @@ -366,6 +470,10 @@ const parseCertificateInfo = function (info) { | ||||
|  * @returns {Object} Object containing certificate information | ||||
|  */ | ||||
| exports.checkCertificate = function (res) { | ||||
|     if (!res.request.res.socket) { | ||||
|         throw new Error("No socket found"); | ||||
|     } | ||||
|  | ||||
|     const info = res.request.res.socket.getPeerCertificate(true); | ||||
|     const valid = res.request.res.socket.authorized || false; | ||||
|  | ||||
| @@ -492,7 +600,27 @@ exports.doubleCheckPassword = async (socket, currentPassword) => { | ||||
| exports.startUnitTest = async () => { | ||||
|     console.log("Starting unit test..."); | ||||
|     const npm = /^win/.test(process.platform) ? "npm.cmd" : "npm"; | ||||
|     const child = childProcess.spawn(npm, [ "run", "jest" ]); | ||||
|     const child = childProcess.spawn(npm, [ "run", "jest-backend" ]); | ||||
|  | ||||
|     child.stdout.on("data", (data) => { | ||||
|         console.log(data.toString()); | ||||
|     }); | ||||
|  | ||||
|     child.stderr.on("data", (data) => { | ||||
|         console.log(data.toString()); | ||||
|     }); | ||||
|  | ||||
|     child.on("close", function (code) { | ||||
|         console.log("Jest exit code: " + code); | ||||
|         process.exit(code); | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| /** Start end-to-end tests */ | ||||
| exports.startE2eTests = async () => { | ||||
|     console.log("Starting unit test..."); | ||||
|     const npm = /^win/.test(process.platform) ? "npm.cmd" : "npm"; | ||||
|     const child = childProcess.spawn(npm, [ "run", "cy:run" ]); | ||||
|  | ||||
|     child.stdout.on("data", (data) => { | ||||
|         console.log(data.toString()); | ||||
| @@ -560,3 +688,112 @@ module.exports.send403 = (res, msg = "") => { | ||||
|         "msg": msg, | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| function timeObjectConvertTimezone(obj, timezone, timeObjectToUTC = true) { | ||||
|     let offsetString; | ||||
|  | ||||
|     if (timezone) { | ||||
|         offsetString = dayjs().tz(timezone).format("Z"); | ||||
|     } else { | ||||
|         offsetString = dayjs().format("Z"); | ||||
|     } | ||||
|  | ||||
|     let hours = parseInt(offsetString.substring(1, 3)); | ||||
|     let minutes = parseInt(offsetString.substring(4, 6)); | ||||
|  | ||||
|     if ( | ||||
|         (timeObjectToUTC && offsetString.startsWith("+")) || | ||||
|         (!timeObjectToUTC && offsetString.startsWith("-")) | ||||
|     ) { | ||||
|         hours *= -1; | ||||
|         minutes *= -1; | ||||
|     } | ||||
|  | ||||
|     obj.hours += hours; | ||||
|     obj.minutes += minutes; | ||||
|  | ||||
|     // Handle out of bound | ||||
|     if (obj.minutes < 0) { | ||||
|         obj.minutes += 60; | ||||
|         obj.hours--; | ||||
|     } else if (obj.minutes > 60) { | ||||
|         obj.minutes -= 60; | ||||
|         obj.hours++; | ||||
|     } | ||||
|  | ||||
|     if (obj.hours < 0) { | ||||
|         obj.hours += 24; | ||||
|     } else if (obj.hours > 24) { | ||||
|         obj.hours -= 24; | ||||
|     } | ||||
|  | ||||
|     return obj; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * | ||||
|  * @param {object} obj | ||||
|  * @param {string} timezone | ||||
|  * @returns {object} | ||||
|  */ | ||||
| module.exports.timeObjectToUTC = (obj, timezone = undefined) => { | ||||
|     return timeObjectConvertTimezone(obj, timezone, true); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * | ||||
|  * @param {object} obj | ||||
|  * @param {string} timezone | ||||
|  * @returns {object} | ||||
|  */ | ||||
| module.exports.timeObjectToLocal = (obj, timezone = undefined) => { | ||||
|     return timeObjectConvertTimezone(obj, timezone, false); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Create gRPC client stib | ||||
|  * @param {Object} options from gRPC client | ||||
|  */ | ||||
| module.exports.grpcQuery = async (options) => { | ||||
|     const { grpcUrl, grpcProtobufData, grpcServiceName, grpcEnableTls, grpcMethod, grpcBody } = options; | ||||
|     const protocObject = protojs.parse(grpcProtobufData); | ||||
|     const protoServiceObject = protocObject.root.lookupService(grpcServiceName); | ||||
|     const Client = grpc.makeGenericClientConstructor({}); | ||||
|     const credentials = grpcEnableTls ? grpc.credentials.createSsl() : grpc.credentials.createInsecure(); | ||||
|     const client = new Client( | ||||
|         grpcUrl, | ||||
|         credentials | ||||
|     ); | ||||
|     const grpcService = protoServiceObject.create(function (method, requestData, cb) { | ||||
|         const fullServiceName = method.fullName; | ||||
|         const serviceFQDN = fullServiceName.split("."); | ||||
|         const serviceMethod = serviceFQDN.pop(); | ||||
|         const serviceMethodClientImpl = `/${serviceFQDN.slice(1).join(".")}/${serviceMethod}`; | ||||
|         log.debug("monitor", `gRPC method ${serviceMethodClientImpl}`); | ||||
|         client.makeUnaryRequest( | ||||
|             serviceMethodClientImpl, | ||||
|             arg => arg, | ||||
|             arg => arg, | ||||
|             requestData, | ||||
|             cb); | ||||
|     }, false, false); | ||||
|     return new Promise((resolve, _) => { | ||||
|         return grpcService[`${grpcMethod}`](JSON.parse(grpcBody), function (err, response) { | ||||
|             const responseData = JSON.stringify(response); | ||||
|             if (err) { | ||||
|                 return resolve({ | ||||
|                     code: err.code, | ||||
|                     errorMessage: err.details, | ||||
|                     data: "" | ||||
|                 }); | ||||
|             } else { | ||||
|                 log.debug("monitor:", `gRPC response: ${response}`); | ||||
|                 return resolve({ | ||||
|                     code: 1, | ||||
|                     errorMessage: "", | ||||
|                     data: responseData | ||||
|                 }); | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
| }; | ||||
|   | ||||
| @@ -22,6 +22,19 @@ textarea.form-control { | ||||
|     width: 10px; | ||||
| } | ||||
|  | ||||
| .bg-maintenance { | ||||
|     color: white !important; | ||||
|     background-color: $maintenance !important; | ||||
| } | ||||
|  | ||||
| .bg-dark { | ||||
|     color: white; | ||||
| } | ||||
|  | ||||
| .text-maintenance { | ||||
|     color: $maintenance !important; | ||||
| } | ||||
|  | ||||
| .list-group { | ||||
|     border-radius: 0.75rem; | ||||
|  | ||||
| @@ -107,6 +120,19 @@ optgroup { | ||||
|     } | ||||
| } | ||||
|  | ||||
| .btn-normal { | ||||
|     $bg-color: #F5F5F5; | ||||
|  | ||||
|     background-color: $bg-color; | ||||
|     border-color: $bg-color; | ||||
|  | ||||
|     &:hover { | ||||
|         $hover-color: darken($bg-color, 3%); | ||||
|         background-color: $hover-color; | ||||
|         border-color: $hover-color; | ||||
|     } | ||||
| } | ||||
|  | ||||
| .btn-warning { | ||||
|     color: white; | ||||
|  | ||||
| @@ -256,6 +282,20 @@ optgroup { | ||||
|         color: white; | ||||
|     } | ||||
|  | ||||
|     .btn-normal { | ||||
|         $bg-color: $dark-header-bg; | ||||
|  | ||||
|         color: $dark-font-color; | ||||
|         background-color: $bg-color; | ||||
|         border-color: $bg-color; | ||||
|  | ||||
|         &:hover { | ||||
|             $hover-color: darken($bg-color, 3%); | ||||
|             background-color: $hover-color; | ||||
|             border-color: $hover-color; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     .btn-warning { | ||||
|         color: $dark-font-color2; | ||||
|  | ||||
| @@ -323,6 +363,7 @@ optgroup { | ||||
|         &.bg-info, | ||||
|         &.bg-warning, | ||||
|         &.bg-danger, | ||||
|         &.bg-maintenance, | ||||
|         &.bg-light { | ||||
|             color: $dark-font-color2; | ||||
|         } | ||||
| @@ -382,7 +423,7 @@ optgroup { | ||||
|         overflow-y: auto; | ||||
|         height: calc(100% - 65px); | ||||
|     } | ||||
|      | ||||
|  | ||||
|     @media (max-width: 770px) { | ||||
|         &.scrollbar { | ||||
|             height: calc(100% - 40px); | ||||
| @@ -403,7 +444,6 @@ optgroup { | ||||
|         .info { | ||||
|             white-space: nowrap; | ||||
|             overflow: hidden; | ||||
|             text-overflow: ellipsis; | ||||
|         } | ||||
|  | ||||
|         &:hover { | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| $primary: #5cdd8b; | ||||
| $danger: #dc3545; | ||||
| $warning: #f8a306; | ||||
| $maintenance: #1747f5; | ||||
| $link-color: #111; | ||||
| $border-radius: 50rem; | ||||
|  | ||||
|   | ||||
							
								
								
									
										39
									
								
								src/assets/vue-datepicker.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/assets/vue-datepicker.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| @import "@vuepic/vue-datepicker/dist/main.css"; | ||||
| @import "vars.scss"; | ||||
|  | ||||
| // Must use #{ } | ||||
| // Remark: https://stackoverflow.com/questions/50202991/unable-to-set-scss-variable-to-css-variable | ||||
| .dp__theme_dark { | ||||
|     --dp-background-color: #{$dark-bg2}; | ||||
|     --dp-text-color: #{$dark-font-color}; | ||||
|     --dp-hover-color: #484848; | ||||
|     --dp-hover-text-color: #ffffff; | ||||
|     --dp-hover-icon-color: #959595; | ||||
|     --dp-primary-color: #{#5cdd8b}; | ||||
|     --dp-primary-text-color: #ffffff; | ||||
|     --dp-secondary-color: #494949; | ||||
|     --dp-border-color: #{$dark-border-color}; | ||||
|     --dp-menu-border-color: #2d2d2d; | ||||
|     --dp-border-color-hover: #{$dark-border-color}; | ||||
|     --dp-disabled-color: #212121; | ||||
|     --dp-scroll-bar-background: #212121; | ||||
|     --dp-scroll-bar-color: #484848; | ||||
|     --dp-success-color: #{$primary}; | ||||
|     --dp-success-color-disabled: #428f59; | ||||
|     --dp-icon-color: #959595; | ||||
|     --dp-danger-color: #e53935; | ||||
|     --dp-highlight-color: rgba(0, 92, 178, 0.2); | ||||
| } | ||||
|  | ||||
| .dp__input { | ||||
|     border-radius: $border-radius; | ||||
| } | ||||
|  | ||||
| // Fix: Full width of text input when using "inline textInput inlineWithInput" mode | ||||
| .dp__main > div[aria-label="Datepicker input"] { | ||||
|     width: 100%; | ||||
| } | ||||
|  | ||||
| .dp__main > div[aria-label="Datepicker menu"]:nth-child(2) { | ||||
|     margin-top: 20px; | ||||
| } | ||||
| @@ -3,14 +3,6 @@ | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import dayjs from "dayjs"; | ||||
| import relativeTime from "dayjs/plugin/relativeTime"; | ||||
| import timezone from "dayjs/plugin/timezone"; // dependent on utc plugin | ||||
| import utc from "dayjs/plugin/utc"; | ||||
| dayjs.extend(utc); | ||||
| dayjs.extend(timezone); | ||||
| dayjs.extend(relativeTime); | ||||
|  | ||||
| export default { | ||||
|     props: { | ||||
|         /** Value of date time */ | ||||
|   | ||||
							
								
								
									
										178
									
								
								src/components/DockerHostDialog.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								src/components/DockerHostDialog.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | ||||
| <template> | ||||
|     <form @submit.prevent="submit"> | ||||
|         <div ref="modal" class="modal fade" tabindex="-1" data-bs-backdrop="static"> | ||||
|             <div class="modal-dialog"> | ||||
|                 <div class="modal-content"> | ||||
|                     <div class="modal-header"> | ||||
|                         <h5 id="exampleModalLabel" class="modal-title"> | ||||
|                             {{ $t("Setup Docker Host") }} | ||||
|                         </h5> | ||||
|                         <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close" /> | ||||
|                     </div> | ||||
|                     <div class="modal-body"> | ||||
|                         <div class="mb-3"> | ||||
|                             <label for="docker-name" class="form-label">{{ $t("Friendly Name") }}</label> | ||||
|                             <input id="docker-name" v-model="dockerHost.name" type="text" class="form-control" required> | ||||
|                         </div> | ||||
|  | ||||
|                         <div class="mb-3"> | ||||
|                             <label for="docker-type" class="form-label">{{ $t("Connection Type") }}</label> | ||||
|                             <select id="docker-type" v-model="dockerHost.dockerType" class="form-select"> | ||||
|                                 <option v-for="type in connectionTypes" :key="type" :value="type">{{ $t(type) }}</option> | ||||
|                             </select> | ||||
|                         </div> | ||||
|  | ||||
|                         <div class="mb-3"> | ||||
|                             <label for="docker-daemon" class="form-label">{{ $t("Docker Daemon") }}</label> | ||||
|                             <input id="docker-daemon" v-model="dockerHost.dockerDaemon" type="text" class="form-control" required> | ||||
|  | ||||
|                             <div class="form-text"> | ||||
|                                 {{ $t("Examples") }}: | ||||
|                                 <ul> | ||||
|                                     <li>/var/run/docker.sock</li> | ||||
|                                     <li>http://localhost:2375</li> | ||||
|                                     <li>https://localhost:2376 (TLS)</li> | ||||
|                                 </ul> | ||||
|                             </div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|  | ||||
|                     <div class="modal-footer"> | ||||
|                         <button v-if="id" type="button" class="btn btn-danger" :disabled="processing" @click="deleteConfirm"> | ||||
|                             {{ $t("Delete") }} | ||||
|                         </button> | ||||
|                         <button type="button" class="btn btn-warning" :disabled="processing" @click="test"> | ||||
|                             {{ $t("Test") }} | ||||
|                         </button> | ||||
|                         <button type="submit" class="btn btn-primary" :disabled="processing"> | ||||
|                             <div v-if="processing" class="spinner-border spinner-border-sm me-1"></div> | ||||
|                             {{ $t("Save") }} | ||||
|                         </button> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     </form> | ||||
|  | ||||
|     <Confirm ref="confirmDelete" btn-style="btn-danger" :yes-text="$t('Yes')" :no-text="$t('No')" @yes="deleteDockerHost"> | ||||
|         {{ $t("deleteDockerHostMsg") }} | ||||
|     </Confirm> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { Modal } from "bootstrap"; | ||||
| import Confirm from "./Confirm.vue"; | ||||
| import { useToast } from "vue-toastification"; | ||||
| const toast = useToast(); | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         Confirm, | ||||
|     }, | ||||
|     props: {}, | ||||
|     emits: [ "added" ], | ||||
|     data() { | ||||
|         return { | ||||
|             model: null, | ||||
|             processing: false, | ||||
|             id: null, | ||||
|             connectionTypes: [ "socket", "tcp" ], | ||||
|             dockerHost: { | ||||
|                 name: "", | ||||
|                 dockerDaemon: "", | ||||
|                 dockerType: "", | ||||
|                 // Do not set default value here, please scroll to show() | ||||
|             } | ||||
|         }; | ||||
|     }, | ||||
|  | ||||
|     mounted() { | ||||
|         this.modal = new Modal(this.$refs.modal); | ||||
|     }, | ||||
|     methods: { | ||||
|  | ||||
|         deleteConfirm() { | ||||
|             this.modal.hide(); | ||||
|             this.$refs.confirmDelete.show(); | ||||
|         }, | ||||
|  | ||||
|         show(dockerHostID) { | ||||
|             if (dockerHostID) { | ||||
|                 let found = false; | ||||
|  | ||||
|                 this.id = dockerHostID; | ||||
|  | ||||
|                 for (let n of this.$root.dockerHostList) { | ||||
|                     if (n.id === dockerHostID) { | ||||
|                         this.dockerHost = n; | ||||
|                         found = true; | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 if (!found) { | ||||
|                     toast.error("Docker Host not found!"); | ||||
|                 } | ||||
|  | ||||
|             } else { | ||||
|                 this.id = null; | ||||
|                 this.dockerHost = { | ||||
|                     name: "", | ||||
|                     dockerType: "socket", | ||||
|                     dockerDaemon: "/var/run/docker.sock", | ||||
|                 }; | ||||
|             } | ||||
|  | ||||
|             this.modal.show(); | ||||
|         }, | ||||
|  | ||||
|         submit() { | ||||
|             this.processing = true; | ||||
|             this.$root.getSocket().emit("addDockerHost", this.dockerHost, this.id, (res) => { | ||||
|                 this.$root.toastRes(res); | ||||
|                 this.processing = false; | ||||
|  | ||||
|                 if (res.ok) { | ||||
|                     this.modal.hide(); | ||||
|  | ||||
|                     // Emit added event, doesn't emit edit. | ||||
|                     if (! this.id) { | ||||
|                         this.$emit("added", res.id); | ||||
|                     } | ||||
|  | ||||
|                 } | ||||
|             }); | ||||
|         }, | ||||
|  | ||||
|         test() { | ||||
|             this.processing = true; | ||||
|             this.$root.getSocket().emit("testDockerHost", this.dockerHost, (res) => { | ||||
|                 this.$root.toastRes(res); | ||||
|                 this.processing = false; | ||||
|             }); | ||||
|         }, | ||||
|  | ||||
|         deleteDockerHost() { | ||||
|             this.processing = true; | ||||
|             this.$root.getSocket().emit("deleteDockerHost", this.id, (res) => { | ||||
|                 this.$root.toastRes(res); | ||||
|                 this.processing = false; | ||||
|  | ||||
|                 if (res.ok) { | ||||
|                     this.modal.hide(); | ||||
|                 } | ||||
|             }); | ||||
|         }, | ||||
|     }, | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| @import "../assets/vars.scss"; | ||||
|  | ||||
| .dark { | ||||
|     .modal-dialog .form-text, .modal-dialog p { | ||||
|         color: $dark-font-color; | ||||
|     } | ||||
| } | ||||
| </style> | ||||
| @@ -5,7 +5,7 @@ | ||||
|                 v-for="(beat, index) in shortBeatList" | ||||
|                 :key="index" | ||||
|                 class="beat" | ||||
|                 :class="{ 'empty' : (beat === 0), 'down' : (beat.status === 0), 'pending' : (beat.status === 2) }" | ||||
|                 :class="{ 'empty' : (beat === 0), 'down' : (beat.status === 0), 'pending' : (beat.status === 2), 'maintenance' : (beat.status === 3) }" | ||||
|                 :style="beatStyle" | ||||
|                 :title="getBeatTitle(beat)" | ||||
|             /> | ||||
| @@ -211,6 +211,10 @@ export default { | ||||
|             background-color: $warning; | ||||
|         } | ||||
|  | ||||
|         &.maintenance { | ||||
|             background-color: $maintenance; | ||||
|         } | ||||
|  | ||||
|         &:not(.empty):hover { | ||||
|             transition: all ease-in-out 0.15s; | ||||
|             opacity: 0.8; | ||||
|   | ||||
| @@ -42,7 +42,7 @@ export default { | ||||
|         /** Should the field auto complete */ | ||||
|         autocomplete: { | ||||
|             type: String, | ||||
|             default: undefined, | ||||
|             default: "new-password", | ||||
|         }, | ||||
|         /** Is the input required? */ | ||||
|         required: { | ||||
|   | ||||
| @@ -54,6 +54,15 @@ export default { | ||||
|             tokenRequired: false, | ||||
|         }; | ||||
|     }, | ||||
|  | ||||
|     mounted() { | ||||
|         document.title += " - Login"; | ||||
|     }, | ||||
|  | ||||
|     unmounted() { | ||||
|         document.title = document.title.replace(" - Login", ""); | ||||
|     }, | ||||
|  | ||||
|     methods: { | ||||
|         /** Submit the user details and attempt to log in */ | ||||
|         submit() { | ||||
|   | ||||
							
								
								
									
										44
									
								
								src/components/MaintenanceTime.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								src/components/MaintenanceTime.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| <template> | ||||
|     <div> | ||||
|         <div v-if="maintenance.strategy === 'manual'" class="timeslot"> | ||||
|             {{ $t("Manual") }} | ||||
|         </div> | ||||
|         <div v-else-if="maintenance.timeslotList.length > 0" class="timeslot"> | ||||
|             {{ maintenance.timeslotList[0].startDateServerTimezone }} | ||||
|             <span class="to">-</span> | ||||
|             {{ maintenance.timeslotList[0].endDateServerTimezone }} | ||||
|             (UTC{{ maintenance.timeslotList[0].serverTimezoneOffset }}) | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| export default { | ||||
|     props: { | ||||
|         maintenance: { | ||||
|             type: Object, | ||||
|             required: true | ||||
|         }, | ||||
|     }, | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss"> | ||||
| .timeslot { | ||||
|     margin-top: 5px; | ||||
|     display: inline-block; | ||||
|     font-size: 14px; | ||||
|     background-color: rgba(255, 255, 255, 0.5); | ||||
|     border-radius: 20px; | ||||
|     padding: 0 10px; | ||||
|  | ||||
|     .to { | ||||
|         margin: 0 6px; | ||||
|     } | ||||
|  | ||||
|     .dark & { | ||||
|         color: white; | ||||
|         background-color: rgba(255, 255, 255, 0.1); | ||||
|     } | ||||
| } | ||||
| </style> | ||||
| @@ -206,6 +206,16 @@ export default { | ||||
| .search-icon { | ||||
|     padding: 10px; | ||||
|     color: #c0c0c0; | ||||
|  | ||||
|     // Clear filter button (X) | ||||
|     svg[data-icon="times"] { | ||||
|         cursor: pointer; | ||||
|         transition: all ease-in-out 0.1s; | ||||
|  | ||||
|         &:hover { | ||||
|             opacity: 0.5; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| .search-input { | ||||
|   | ||||
| @@ -16,18 +16,14 @@ | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| <script lang="js"> | ||||
| import { BarController, BarElement, Chart, Filler, LinearScale, LineController, LineElement, PointElement, TimeScale, Tooltip } from "chart.js"; | ||||
| import "chartjs-adapter-dayjs"; | ||||
| import dayjs from "dayjs"; | ||||
| import timezone from "dayjs/plugin/timezone"; | ||||
| import utc from "dayjs/plugin/utc"; | ||||
| import { LineChart } from "vue-chart-3"; | ||||
| import { useToast } from "vue-toastification"; | ||||
| import { DOWN, log } from "../util.ts"; | ||||
| import { DOWN, PENDING, MAINTENANCE, log } from "../util.ts"; | ||||
|  | ||||
| dayjs.extend(utc); | ||||
| dayjs.extend(timezone); | ||||
| const toast = useToast(); | ||||
|  | ||||
| Chart.register(LineController, BarController, LineElement, PointElement, TimeScale, BarElement, LinearScale, Tooltip, Filler); | ||||
| @@ -163,7 +159,8 @@ export default { | ||||
|         }, | ||||
|         chartData() { | ||||
|             let pingData = [];  // Ping Data for Line Chart, y-axis contains ping time | ||||
|             let downData = [];  // Down Data for Bar Chart, y-axis is 1 if target is down, 0 if target is up | ||||
|             let downData = [];  // Down Data for Bar Chart, y-axis is 1 if target is down (red color), under maintenance (blue color) or pending (orange color), 0 if target is up | ||||
|             let colorData = []; // Color Data for Bar Chart | ||||
|  | ||||
|             let heartbeatList = this.heartbeatList || | ||||
|              (this.monitorId in this.$root.heartbeatList && this.$root.heartbeatList[this.monitorId]) || | ||||
| @@ -185,8 +182,9 @@ export default { | ||||
|                     }); | ||||
|                     downData.push({ | ||||
|                         x, | ||||
|                         y: beat.status === DOWN ? 1 : 0, | ||||
|                         y: (beat.status === DOWN || beat.status === MAINTENANCE || beat.status === PENDING) ? 1 : 0, | ||||
|                     }); | ||||
|                     colorData.push((beat.status === MAINTENANCE) ? "rgba(23,71,245,0.41)" : ((beat.status === PENDING) ? "rgba(245,182,23,0.41)" : "#DC354568")); | ||||
|                 }); | ||||
|  | ||||
|             return { | ||||
| @@ -205,7 +203,7 @@ export default { | ||||
|                         type: "bar", | ||||
|                         data: downData, | ||||
|                         borderColor: "#00000000", | ||||
|                         backgroundColor: "#DC354568", | ||||
|                         backgroundColor: colorData, | ||||
|                         yAxisID: "y1", | ||||
|                         barThickness: "flex", | ||||
|                         barPercentage: 1, | ||||
|   | ||||
| @@ -17,6 +17,7 @@ | ||||
|                                 <option value="http">HTTP</option> | ||||
|                                 <option value="socks">SOCKS</option> | ||||
|                                 <option value="socks5">SOCKS v5</option> | ||||
|                                 <option value="socks5h">SOCKS v5 (+DNS)</option> | ||||
|                                 <option value="socks4">SOCKS v4</option> | ||||
|                             </select> | ||||
|                         </div> | ||||
|   | ||||
| @@ -44,6 +44,7 @@ | ||||
|                                                 :href="monitor.element.url" | ||||
|                                                 class="item-name" | ||||
|                                                 target="_blank" | ||||
|                                                 rel="noopener noreferrer" | ||||
|                                             > | ||||
|                                                 {{ monitor.element.name }} | ||||
|                                             </a> | ||||
| @@ -224,4 +225,8 @@ export default { | ||||
|     } | ||||
| } | ||||
|  | ||||
| .bg-maintenance { | ||||
|     background-color: $maintenance; | ||||
| } | ||||
|  | ||||
| </style> | ||||
|   | ||||
| @@ -26,6 +26,10 @@ export default { | ||||
|                 return "warning"; | ||||
|             } | ||||
|  | ||||
|             if (this.status === 3) { | ||||
|                 return "maintenance"; | ||||
|             } | ||||
|  | ||||
|             return "secondary"; | ||||
|         }, | ||||
|  | ||||
| @@ -42,6 +46,10 @@ export default { | ||||
|                 return this.$t("Pending"); | ||||
|             } | ||||
|  | ||||
|             if (this.status === 3) { | ||||
|                 return this.$t("statusMaintenance"); | ||||
|             } | ||||
|  | ||||
|             return this.$t("Unknown"); | ||||
|         }, | ||||
|     }, | ||||
|   | ||||
| @@ -25,6 +25,10 @@ export default { | ||||
|     computed: { | ||||
|         uptime() { | ||||
|  | ||||
|             if (this.type === "maintenance") { | ||||
|                 return this.$t("statusMaintenance"); | ||||
|             } | ||||
|  | ||||
|             let key = this.monitor.id + "_" + this.type; | ||||
|  | ||||
|             if (this.$root.uptimeList[key] !== undefined) { | ||||
| @@ -35,6 +39,10 @@ export default { | ||||
|         }, | ||||
|  | ||||
|         color() { | ||||
|             if (this.type === "maintenance" || this.monitor.maintenance) { | ||||
|                 return "maintenance"; | ||||
|             } | ||||
|  | ||||
|             if (this.lastHeartBeat.status === 0) { | ||||
|                 return "danger"; | ||||
|             } | ||||
|   | ||||
| @@ -2,9 +2,6 @@ | ||||
|     <div class="mb-3"> | ||||
|         <label for="Bark Endpoint" class="form-label">{{ $t("Bark Endpoint") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <input id="Bark Endpoint" v-model="$parent.notification.barkEndpoint" type="text" class="form-control" required> | ||||
|         <div class="form-text"> | ||||
|             <p><span style="color: red;"><sup>*</sup></span>{{ $t("Required") }}</p> | ||||
|         </div> | ||||
|         <i18n-t tag="div" keypath="wayToGetTeamsURL" class="form-text"> | ||||
|             <a | ||||
|                 href="https://github.com/Finb/Bark" | ||||
| @@ -12,4 +9,45 @@ | ||||
|             >{{ $t("here") }}</a> | ||||
|         </i18n-t> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="Bark Group" class="form-label">{{ $t("Bark Group") }}</label> | ||||
|         <input id="Bark Group" v-model="$parent.notification.barkGroup" type="text" class="form-control" required> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="Bark Sound" class="form-label">{{ $t("Bark Sound") }}</label> | ||||
|         <select id="Bark Sound" v-model="$parent.notification.barkSound" class="form-select" required> | ||||
|             <option value="alarm">alarm</option> | ||||
|             <option value="anticipate">anticipate</option> | ||||
|             <option value="bell">bell</option> | ||||
|             <option value="birdsong">birdsong</option> | ||||
|             <option value="bloom">bloom</option> | ||||
|             <option value="calypso">calypso</option> | ||||
|             <option value="chime">chime</option> | ||||
|             <option value="choo">choo</option> | ||||
|             <option value="descent">descent</option> | ||||
|             <option value="electronic">electronic</option> | ||||
|             <option value="fanfare">fanfare</option> | ||||
|             <option value="glass">glass</option> | ||||
|             <option value="gotosleep">gotosleep</option> | ||||
|             <option value="healthnotification">healthnotification</option> | ||||
|             <option value="horn">horn</option> | ||||
|             <option value="ladder">ladder</option> | ||||
|             <option value="mailsent">mailsent</option> | ||||
|             <option value="minuet">minuet</option> | ||||
|             <option value="multiwayinvitation">multiwayinvitation</option> | ||||
|             <option value="newmail">newmail</option> | ||||
|             <option value="newsflash">newsflash</option> | ||||
|             <option value="noir">noir</option> | ||||
|             <option value="paymentsuccess">paymentsuccess</option> | ||||
|             <option value="shake">shake</option> | ||||
|             <option value="sherwoodforest">sherwoodforest</option> | ||||
|             <option value="silence">silence</option> | ||||
|             <option value="spell">spell</option> | ||||
|             <option value="suspense">suspense</option> | ||||
|             <option value="telegraph">telegraph</option> | ||||
|             <option value="tiptoes">tiptoes</option> | ||||
|             <option value="typewriters">typewriters</option> | ||||
|             <option value="update">update</option> | ||||
|         </select> | ||||
|     </div> | ||||
| </template> | ||||
|   | ||||
| @@ -6,7 +6,7 @@ | ||||
|         </i18n-t> | ||||
|         <input id="clicksendsms-login" v-model="$parent.notification.clicksendsmsLogin" type="text" class="form-control" required> | ||||
|         <label for="clicksendsms-key" class="form-label">{{ $t("API Key") }}</label> | ||||
|         <HiddenInput id="clicksendsms-key" v-model="$parent.notification.clicksendsmsPassword" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="clicksendsms-key" v-model="$parent.notification.clicksendsmsPassword" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <div class="form-text"> | ||||
|   | ||||
							
								
								
									
										12
									
								
								src/components/notifications/FreeMobile.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								src/components/notifications/FreeMobile.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="freemobileUser" class="form-label">{{ $t("Free Mobile User Identifier") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <input id="freemobileUser" v-model="$parent.notification.freemobileUser" type="text" class="form-control" required> | ||||
|     </div> | ||||
|  | ||||
|     <div class="mb-3"> | ||||
|         <label for="freemobilePass" class="form-label">{{ $t("Free Mobile API Key") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <input id="freemobilePass" v-model="$parent.notification.freemobilePass" type="text" class="form-control" required> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
							
								
								
									
										30
									
								
								src/components/notifications/GoAlert.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/components/notifications/GoAlert.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="goalert-base-url" class="form-label">{{ $t("Base URL") }}</label> | ||||
|         <div class="input-group mb-3"> | ||||
|             <input id="goalert-base-url" v-model="$parent.notification.goAlertBaseURL" type="text" class="form-control" required> | ||||
|         </div> | ||||
|         <i18n-t tag="div" keypath="goAlertInfo" class="form-text"> | ||||
|             <a href="https://goalert.me" target="_blank">https://goalert.me</a> | ||||
|         </i18n-t> | ||||
|     </div> | ||||
|  | ||||
|     <div class="mb-3"> | ||||
|         <label for="goalert-token" class="form-label">{{ $t("Token") }}</label> | ||||
|         <HiddenInput id="goalert-token" v-model="$parent.notification.goAlertToken" autocomplete="new-password" :required="true"></HiddenInput> | ||||
|  | ||||
|         <div class="form-text"> | ||||
|             {{ $t("goAlertIntegrationKeyInfo") }} | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import HiddenInput from "../HiddenInput.vue"; | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         HiddenInput, | ||||
|     }, | ||||
| }; | ||||
| </script> | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="gotify-application-token" class="form-label">{{ $t("Application Token") }}</label> | ||||
|         <HiddenInput id="gotify-application-token" v-model="$parent.notification.gotifyapplicationToken" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="gotify-application-token" v-model="$parent.notification.gotifyapplicationToken" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="gotify-server-url" class="form-label">{{ $t("Server URL") }}</label> | ||||
|   | ||||
							
								
								
									
										40
									
								
								src/components/notifications/HomeAssistant.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/components/notifications/HomeAssistant.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="homeAssistantUrl" class="form-label">{{ $t("Home Assistant URL") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <input id="homeAssistantUrl" v-model="$parent.notification.homeAssistantUrl" type="url" class="form-control" required> | ||||
|     </div> | ||||
|  | ||||
|     <div class="mb-3"> | ||||
|         <label for="longLivedAccessToken" class="form-label">{{ $t("Long-Lived Access Token") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <input id="longLivedAccessToken" v-model="$parent.notification.longLivedAccessToken" type="text" class="form-control" required> | ||||
|  | ||||
|         <div class="form-text"> | ||||
|             <p>{{ $t("Long-Lived Access Token can be created by clicking on your profile name (bottom left) and scrolling to the bottom then click Create Token. ") }}</p> | ||||
|         </div> | ||||
|     </div> | ||||
|  | ||||
|     <div class="mb-3"> | ||||
|         <label for="notificationService" class="form-label">{{ $t("Notification Service") }}</label> | ||||
|         <input id="notificationService" v-model="$parent.notification.notificationService" type="text" :placeholder="$t('default: notify all devices')" class="form-control"> | ||||
|  | ||||
|         <div class="form-text"> | ||||
|             <p>{{ $t('A list of Notification Services can be found in Home Assistant under "Developer Tools > Services" search for "notification" to find your device/phone name.') }}</p> | ||||
|             <p>{{ $t("Automations can optionally be triggered in Home Assistant:") }}</p> | ||||
|             <p> | ||||
|                 {{ $t("Trigger type:") }} <code>Event</code><br /> | ||||
|                 {{ $t("Event type:") }} <code>call_service</code><br /> | ||||
|                 {{ $t("Event data:") }} | ||||
|             </p> | ||||
|             <pre>domain: notify | ||||
| service: mobile_app_my_phone # change to your device name | ||||
| service_data: | ||||
|   title: Uptime Kuma | ||||
|   data: | ||||
|     status: 0 # 0=down 1=up | ||||
|     # name: Optional Uptime Kuma Monitor Name to filter by</pre> | ||||
|             <p> | ||||
|                 {{ $t("Then choose an action, for example switch the scene to where an RGB light is red.") }} | ||||
|             </p> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="line-channel-access-token" class="form-label">{{ $t("Channel access token") }}</label> | ||||
|         <HiddenInput id="line-channel-access-token" v-model="$parent.notification.lineChannelAccessToken" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="line-channel-access-token" v-model="$parent.notification.lineChannelAccessToken" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|     </div> | ||||
|     <i18n-t tag="div" keypath="lineDevConsoleTo" class="form-text"> | ||||
|         <b>{{ $t("Basic Settings") }}</b> | ||||
|   | ||||
							
								
								
									
										9
									
								
								src/components/notifications/LineNotify.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/components/notifications/LineNotify.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="line-notify-access-token" class="form-label">{{ $t("Access Token") }}</label> | ||||
|         <input id="line-notify-access-token" v-model="$parent.notification.lineNotifyAccessToken" type="text" class="form-control" :required="true"> | ||||
|     </div> | ||||
|     <i18n-t tag="div" keypath="wayToGetLineNotifyToken" class="form-text" style="margin-top: 8px;"> | ||||
|         <a href="https://notify-bot.line.me/" target="_blank">https://notify-bot.line.me/</a> | ||||
|     </i18n-t> | ||||
| </template> | ||||
| @@ -9,7 +9,7 @@ | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="access-token" class="form-label">{{ $t("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> | ||||
|         <HiddenInput id="access-token" v-model="$parent.notification.accessToken" :required="true" autocomplete="new-password" :maxlength="500"></HiddenInput> | ||||
|     </div> | ||||
|  | ||||
|     <div class="form-text"> | ||||
|   | ||||
| @@ -11,15 +11,35 @@ | ||||
|             <input id="ntfy-server-url" v-model="$parent.notification.ntfyserverurl" type="text" class="form-control" required> | ||||
|         </div> | ||||
|     </div> | ||||
|  | ||||
|     <div class="mb-3"> | ||||
|         <label for="ntfy-priority" class="form-label">{{ $t("Priority") }}</label> | ||||
|         <input id="ntfy-priority" v-model="$parent.notification.ntfyPriority" type="number" class="form-control" required min="1" max="5" step="1"> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="ntfy-username" class="form-label">{{ $t("Username") }} ({{ $t("Optional") }})</label> | ||||
|         <div class="input-group mb-3"> | ||||
|             <input id="ntfy-username" v-model="$parent.notification.ntfyusername" type="text" class="form-control"> | ||||
|         </div> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="ntfy-password" class="form-label">{{ $t("Password") }} ({{ $t("Optional") }})</label> | ||||
|         <div class="input-group mb-3"> | ||||
|             <HiddenInput id="ntfy-password" v-model="$parent.notification.ntfypassword" autocomplete="new-password"></HiddenInput> | ||||
|         </div> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="ntfy-icon" class="form-label">{{ $t("IconUrl") }}</label> | ||||
|         <input id="ntfy-icon" v-model="$parent.notification.ntfyIcon" type="text" class="form-control"> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| import HiddenInput from "../HiddenInput.vue"; | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         HiddenInput, | ||||
|     }, | ||||
|     mounted() { | ||||
|         if (typeof this.$parent.notification.ntfyPriority === "undefined") { | ||||
|             this.$parent.notification.ntfyserverurl = "https://ntfy.sh"; | ||||
|   | ||||
| @@ -11,7 +11,7 @@ | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="octopush-key" class="form-label">{{ $t("octopushAPIKey") }}</label> | ||||
|         <HiddenInput id="octopush-key" v-model="$parent.notification.octopushAPIKey" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="octopush-key" v-model="$parent.notification.octopushAPIKey" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|         <label for="octopush-login" class="form-label">{{ $t("octopushLogin") }}</label> | ||||
|         <input id="octopush-login" v-model="$parent.notification.octopushLogin" type="text" class="form-control" required> | ||||
|     </div> | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
|         <label for="promosms-login" class="form-label">{{ $t("promosmsLogin") }}</label> | ||||
|         <input id="promosms-login" v-model="$parent.notification.promosmsLogin" type="text" class="form-control" required> | ||||
|         <label for="promosms-key" class="form-label">{{ $t("promosmsPassword") }}</label> | ||||
|         <HiddenInput id="promosms-key" v-model="$parent.notification.promosmsPassword" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="promosms-key" v-model="$parent.notification.promosmsPassword" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|     </div> | ||||
|     <div class="mb-3"> | ||||
|         <label for="promosms-type-sms" class="form-label">{{ $t("SMS Type") }}</label> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="pushdeer-key" class="form-label">{{ $t("PushDeer Key") }}</label> | ||||
|         <HiddenInput id="pushdeer-key" v-model="$parent.notification.pushdeerKey" :required="true" autocomplete="one-time-code" placeholder="PDUxxxx"></HiddenInput> | ||||
|         <HiddenInput id="pushdeer-key" v-model="$parent.notification.pushdeerKey" :required="true" autocomplete="new-password" placeholder="PDUxxxx"></HiddenInput> | ||||
|     </div> | ||||
|  | ||||
|     <i18n-t tag="p" keypath="More info on:" style="margin-top: 8px;"> | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="pushbullet-access-token" class="form-label">{{ $t("Access Token") }}</label> | ||||
|         <HiddenInput id="pushbullet-access-token" v-model="$parent.notification.pushbulletAccessToken" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="pushbullet-access-token" v-model="$parent.notification.pushbulletAccessToken" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|     </div> | ||||
|  | ||||
|     <i18n-t tag="p" keypath="More info on:" style="margin-top: 8px;"> | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| <template> | ||||
|     <div class="mb-3"> | ||||
|         <label for="pushover-user" class="form-label">{{ $t("User Key") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <HiddenInput id="pushover-user" v-model="$parent.notification.pushoveruserkey" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="pushover-user" v-model="$parent.notification.pushoveruserkey" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|         <label for="pushover-app-token" class="form-label">{{ $t("Application Token") }}<span style="color: red;"><sup>*</sup></span></label> | ||||
|         <HiddenInput id="pushover-app-token" v-model="$parent.notification.pushoverapptoken" :required="true" autocomplete="one-time-code"></HiddenInput> | ||||
|         <HiddenInput id="pushover-app-token" v-model="$parent.notification.pushoverapptoken" :required="true" autocomplete="new-password"></HiddenInput> | ||||
|         <label for="pushover-device" class="form-label">{{ $t("Device") }}</label> | ||||
|         <input id="pushover-device" v-model="$parent.notification.pushoverdevice" type="text" class="form-control"> | ||||
|         <label for="pushover-device" class="form-label">{{ $t("Message Title") }}</label> | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user