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 2.0.X
# Conflicts: # docker/alpine-base.dockerfile # docker/debian-base.dockerfile # docker/dockerfile # package.json # server/database.js # server/jobs/util-worker.js # server/model/maintenance.js # server/model/monitor.js # server/routers/api-router.js # server/server.js # server/uptime-kuma-server.js
This commit is contained in:
		| @@ -33,7 +33,7 @@ tsconfig.json | ||||
| /ecosystem.config.js | ||||
| /extra/healthcheck.exe | ||||
| /extra/healthcheck | ||||
|  | ||||
| extra/exe-builder | ||||
|  | ||||
| ### .gitignore content (commented rules are duplicated) | ||||
|  | ||||
| @@ -48,6 +48,4 @@ dist-ssr | ||||
| #!/data/.gitkeep | ||||
| #.vscode | ||||
|  | ||||
|  | ||||
|  | ||||
| ### End of .gitignore content | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/ISSUE_TEMPLATE/ask-for-help.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/ISSUE_TEMPLATE/ask-for-help.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -26,6 +26,12 @@ body: | ||||
|       label: "📝 Describe your problem" | ||||
|       description: "Please walk us through it step by step." | ||||
|       placeholder: "Describe what are you asking for..." | ||||
|   - type: textarea | ||||
|     id: error-msg | ||||
|     validations: | ||||
|       required: false | ||||
|     attributes: | ||||
|       label: "📝 Error Message(s) or Log" | ||||
|   - type: input | ||||
|     id: uptime-kuma-version | ||||
|     attributes: | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -61,8 +61,8 @@ body: | ||||
|     id: operating-system | ||||
|     attributes: | ||||
|       label: "💻 Operating System and Arch" | ||||
|       description: "Which OS is your server/device running on?" | ||||
|       placeholder: "Ex. Ubuntu 20.04 x86" | ||||
|       description: "Which OS is your server/device running on? (For Replit, please do not report this bug)" | ||||
|       placeholder: "Ex. Ubuntu 20.04 x64 " | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: input | ||||
|   | ||||
							
								
								
									
										19
									
								
								.github/ISSUE_TEMPLATE/security.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.github/ISSUE_TEMPLATE/security.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| --- | ||||
|  | ||||
| name: "Security Issue" | ||||
| about: "Just for alerting @louislam, do not provide any details here" | ||||
| title: "Security Issue" | ||||
| ref: "main" | ||||
| labels: | ||||
|  | ||||
| - security | ||||
|  | ||||
| --- | ||||
|  | ||||
| DO NOT PROVIDE ANY DETAILS HERE. Please privately report to https://github.com/louislam/uptime-kuma/security/advisories/new. | ||||
|  | ||||
|  | ||||
| Why need this issue? It is because GitHub Advisory do not send a notification to @louislam, it is a workaround to do so. | ||||
|  | ||||
| Your GitHub Advisory URL: | ||||
|  | ||||
							
								
								
									
										1
									
								
								.github/config/exclude.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.github/config/exclude.txt
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| # This is a .gitignore style file for 'GrantBirki/json-yaml-validate' Action workflow | ||||
							
								
								
									
										29
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,8 +21,8 @@ jobs: | ||||
|  | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [macos-latest, ubuntu-latest, windows-latest] | ||||
|         node: [ 14, 16, 18, 19 ] | ||||
|         os: [macos-latest, ubuntu-latest, windows-latest, ARM64] | ||||
|         node: [ 14, 18 ] | ||||
|         # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ | ||||
|  | ||||
|     steps: | ||||
| @@ -34,6 +34,7 @@ jobs: | ||||
|       with: | ||||
|         node-version: ${{ matrix.node }} | ||||
|         cache: 'npm' | ||||
|     - run: npm install npm@latest -g | ||||
|     - run: npm install | ||||
|     - run: npm run build | ||||
|     - run: npm test | ||||
| @@ -41,6 +42,30 @@ jobs: | ||||
|         HEADLESS_TEST: 1 | ||||
|         JUST_FOR_TEST: ${{ secrets.JUST_FOR_TEST }} | ||||
|  | ||||
|   # As a lot of dev dependencies are not supported on ARMv7, we have to test it separately and just test if `npm ci --production` works | ||||
|   armv7-simple-test: | ||||
|     needs: [ check-linters ] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     timeout-minutes: 15 | ||||
|  | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ ARMv7 ] | ||||
|         node: [ 14, 18 ] | ||||
|         # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ | ||||
|  | ||||
|     steps: | ||||
|       - run: git config --global core.autocrlf false  # Mainly for Windows | ||||
|       - uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Use Node.js ${{ matrix.node }} | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ matrix.node }} | ||||
|           cache: 'npm' | ||||
|       - run: npm install npm@latest -g | ||||
|       - run: npm ci --production | ||||
|  | ||||
|   check-linters: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|   | ||||
							
								
								
									
										26
									
								
								.github/workflows/json-yaml-validate.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.github/workflows/json-yaml-validate.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| name: json-yaml-validate  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|   pull_request: | ||||
|     branches: | ||||
|       - master | ||||
|   workflow_dispatch: | ||||
|  | ||||
| permissions: | ||||
|   contents: read | ||||
|   pull-requests: write # enable write permissions for pull request comments | ||||
|  | ||||
| jobs: | ||||
|   json-yaml-validate: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: json-yaml-validate | ||||
|         id: json-yaml-validate | ||||
|         uses: GrantBirki/json-yaml-validate@v1.3.0 | ||||
|         with: | ||||
|           comment: "true" # enable comment mode | ||||
|           exclude_file: ".github/config/exclude.txt" # gitignore style file for exclusions | ||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -21,3 +21,9 @@ cypress/screenshots | ||||
| /extra/healthcheck.exe | ||||
| /extra/healthcheck | ||||
| /extra/healthcheck-armv7 | ||||
|  | ||||
| extra/exe-builder/bin | ||||
| extra/exe-builder/obj | ||||
|  | ||||
| .vs | ||||
| .vscode | ||||
|   | ||||
| @@ -10,5 +10,6 @@ | ||||
|         "color-function-notation": "legacy", | ||||
|         "shorthand-property-no-redundant-values": null, | ||||
|         "color-hex-length": null, | ||||
|         "declaration-block-no-redundant-longhand-properties": null | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -47,17 +47,17 @@ Here are some references: | ||||
|  | ||||
| ❌ Won't Merge | ||||
| - A dedicated pr for translating existing languages (You can now translate on https://weblate.kuma.pet)  | ||||
| - Do not pass auto test | ||||
| - Do not pass the auto test | ||||
| - Any breaking changes | ||||
| - Duplicated pull request | ||||
| - Duplicated pull requests | ||||
| - 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 | ||||
| - Convert existing code into other programming languages | ||||
| - Unnecessary large code changes (Hard to review, causes code conflicts to other pull requests) | ||||
| - Modifications or deletions of existing logic without a valid reason. | ||||
| - Adding functions that is completely out of scope | ||||
| - Converting existing code into other programming languages | ||||
| - Unnecessarily large code changes that are hard to review and cause conflicts with other PRs. | ||||
|  | ||||
| The above cases cannot cover all situations. | ||||
| The above cases may not cover all possible situations. | ||||
|  | ||||
| I (@louislam) have the final say. If your pull request does not meet my expectations, I will reject it, no matter how much time you spend on it. Therefore, it is essential to have a discussion beforehand. | ||||
|  | ||||
| @@ -235,12 +235,13 @@ https://github.com/louislam/uptime-kuma/issues?q=sort%3Aupdated-desc | ||||
|  | ||||
| 1. Draft a release note | ||||
| 2. Make sure the repo is cleared | ||||
| 3. If the healthcheck is updated, remember to re-compile it: `npm run build-docker-builder-go` | ||||
| 3. `npm run release-final with env vars: `VERSION` and `GITHUB_TOKEN` | ||||
| 4. Wait until the `Press any key to continue` | ||||
| 5. `git push` | ||||
| 6. Publish the release note as 1.X.X  | ||||
| 7. Press any key to continue | ||||
| 8. SSH to demo site server and update to 1.X.X | ||||
| 8. Deploy to the demo server: `npm run deploy-demo-server` | ||||
|  | ||||
| Checking: | ||||
|  | ||||
|   | ||||
							
								
								
									
										24
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								README.md
									
									
									
									
									
								
							| @@ -49,8 +49,13 @@ Uptime Kuma is now running on http://localhost:3001 | ||||
|  | ||||
| ### 💪🏻 Non-Docker | ||||
|  | ||||
| Required Tools:  | ||||
| - [Node.js](https://nodejs.org/en/download/) >= 14 | ||||
| Requirements:  | ||||
| - Platform | ||||
|   - ✅ Major Linux distros such as Debian, Ubuntu, CentOS, Fedora and ArchLinux etc.  | ||||
|   - ✅ Windows 10 (x64), Windows Server 2012 R2 (x64) or higher | ||||
|   - ❌ Replit / Heroku | ||||
| - [Node.js](https://nodejs.org/en/download/) 14 / 16 / 18 (20 is not supported) | ||||
| - [npm](https://docs.npmjs.com/cli/) >= 7 | ||||
| - [Git](https://git-scm.com/downloads)  | ||||
| - [pm2](https://pm2.keymetrics.io/) - For running Uptime Kuma in the background | ||||
|  | ||||
| @@ -86,6 +91,10 @@ pm2 monit | ||||
| pm2 save && pm2 startup | ||||
| ``` | ||||
|  | ||||
| ### Windows Portable (x64) | ||||
|  | ||||
| https://github.com/louislam/uptime-kuma/files/11886108/uptime-kuma-win64-portable-1.0.1.zip | ||||
|  | ||||
| ### Advanced Installation | ||||
|  | ||||
| If you need more options or need to browse via a reverse proxy, please read: | ||||
| @@ -143,17 +152,18 @@ Telegram Notification Sample: | ||||
|  | ||||
| If you love this project, please consider giving me a ⭐. | ||||
|  | ||||
| ## 🗣️ Discussion | ||||
| ## 🗣️ Discussion / Ask for Help | ||||
|  | ||||
| ### Issues Page | ||||
| ⚠️ For any general or technical questions, please don't send me an email, as I am unable to provide support in that manner. I will not response if you asked such questions. | ||||
|  | ||||
| You can discuss or ask for help in [issues](https://github.com/louislam/uptime-kuma/issues). | ||||
| I recommend using Google, GitHub Issues, or Uptime Kuma's Subreddit for finding answers to your question. If you cannot find the information you need, feel free to ask: | ||||
|  | ||||
| ### Subreddit | ||||
| - [GitHub Issues](https://github.com/louislam/uptime-kuma/issues) | ||||
| - [Subreddit r/Uptime kuma](https://www.reddit.com/r/UptimeKuma/) | ||||
|  | ||||
| My Reddit account: [u/louislamlam](https://reddit.com/u/louislamlam).   | ||||
| You can mention me if you ask a question on Reddit. | ||||
| [r/Uptime kuma](https://www.reddit.com/r/UptimeKuma/) | ||||
|  | ||||
|  | ||||
| ## Contribute | ||||
|  | ||||
|   | ||||
| @@ -2,7 +2,8 @@ | ||||
|  | ||||
| ## Reporting a Vulnerability | ||||
|  | ||||
| Please report security issues to https://github.com/louislam/uptime-kuma/security/advisories/new. | ||||
| 1. Please report security issues to https://github.com/louislam/uptime-kuma/security/advisories/new. | ||||
| 1. Please also create a empty security issues for alerting me, as GitHub Advisory do not send a notification, I probably will miss without this. https://github.com/louislam/uptime-kuma/issues/new?assignees=&labels=help&template=security.md | ||||
|  | ||||
| Do not use the public issue tracker or discuss it in the public as it will cause more damage. | ||||
|  | ||||
|   | ||||
							
								
								
									
										7
									
								
								db/patch-add-description-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								db/patch-add-description-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD description TEXT default null; | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										6
									
								
								db/patch-add-parent-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								db/patch-add-parent-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD parent INTEGER REFERENCES [monitor] ([id]) ON DELETE SET NULL ON UPDATE CASCADE; | ||||
|  | ||||
| COMMIT | ||||
							
								
								
									
										13
									
								
								db/patch-api-key-table.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								db/patch-api-key-table.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
| CREATE TABLE [api_key] ( | ||||
|     [id] INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||
|     [key] VARCHAR(255) NOT NULL, | ||||
|     [name] VARCHAR(255) NOT NULL, | ||||
|     [user_id] INTEGER NOT NULL, | ||||
|     [created_date] DATETIME DEFAULT (DATETIME('now')) NOT NULL, | ||||
|     [active] BOOLEAN DEFAULT 1 NOT NULL, | ||||
|     [expires] DATETIME DEFAULT NULL, | ||||
|     CONSTRAINT FK_user FOREIGN KEY ([user_id]) REFERENCES [user]([id]) ON DELETE CASCADE ON UPDATE CASCADE | ||||
| ); | ||||
| COMMIT; | ||||
							
								
								
									
										12
									
								
								db/patch-http-body-encoding.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								db/patch-http-body-encoding.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor ADD http_body_encoding VARCHAR(25); | ||||
|  | ||||
| COMMIT; | ||||
|  | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| UPDATE monitor SET http_body_encoding = 'json' WHERE (type = 'http' or type = 'keyword') AND http_body_encoding IS NULL; | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										11
									
								
								db/patch-maintenance-cron.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								db/patch-maintenance-cron.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| DROP TABLE maintenance_timeslot; | ||||
|  | ||||
| -- 999 characters. https://stackoverflow.com/questions/46134830/maximum-length-for-cron-job | ||||
| ALTER TABLE maintenance ADD cron TEXT; | ||||
| ALTER TABLE maintenance ADD timezone VARCHAR(255); | ||||
| ALTER TABLE maintenance ADD duration INTEGER; | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										13
									
								
								db/patch-monitor-tls.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								db/patch-monitor-tls.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD tls_ca TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD tls_cert TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD tls_key TEXT default null; | ||||
|  | ||||
| COMMIT; | ||||
| @@ -5,23 +5,23 @@ ARG TARGETPLATFORM | ||||
|  | ||||
| # Install Curl | ||||
| # Install Apprise, add sqlite3 cli for debugging in the future, iputils-ping for ping, util-linux for setpriv | ||||
| # Stupid python3 and python3-pip actually install a lot of useless things into Debian, specify --no-install-recommends to skip them. | ||||
| 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 git && \ | ||||
|     pip3 --no-cache-dir install apprise==1.2.1 && \ | ||||
| # Stupid python3 and python3-pip actually install a lot of useless things into Debian, specify --no-install-recommends to skip them, make the base even smaller than alpine! | ||||
| RUN apt-get update && \ | ||||
|     apt-get --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 git curl ca-certificates && \ | ||||
|     pip3 --no-cache-dir install apprise==1.4.0 && \ | ||||
|     rm -rf /var/lib/apt/lists/* && \ | ||||
|     apt --yes autoremove | ||||
|  | ||||
| # Install cloudflared | ||||
| # dpkg --add-architecture arm: cloudflared do not provide armhf, this is workaround. Read more: https://github.com/cloudflare/cloudflared/issues/583 | ||||
| COPY extra/download-cloudflared.js ./extra/download-cloudflared.js | ||||
| RUN node ./extra/download-cloudflared.js $TARGETPLATFORM && \ | ||||
|     dpkg --add-architecture arm && \ | ||||
|     apt update && \ | ||||
|     apt --yes --no-install-recommends install ./cloudflared.deb && \ | ||||
| RUN set -eux && \ | ||||
|     mkdir -p --mode=0755 /usr/share/keyrings && \ | ||||
|     curl --fail --show-error --silent --location --insecure https://pkg.cloudflare.com/cloudflare-main.gpg --output /usr/share/keyrings/cloudflare-main.gpg && \ | ||||
|     echo 'deb [signed-by=/usr/share/keyrings/cloudflare-main.gpg] https://pkg.cloudflare.com/cloudflared buster main' | tee /etc/apt/sources.list.d/cloudflared.list && \ | ||||
|     apt-get update && \ | ||||
|     apt-get install --yes --no-install-recommends cloudflared && \ | ||||
|     cloudflared version && \ | ||||
|     rm -rf /var/lib/apt/lists/* && \ | ||||
|     rm -f cloudflared.deb && \ | ||||
|     apt --yes autoremove | ||||
|  | ||||
| FROM base2-slim AS base2 | ||||
|   | ||||
| @@ -29,6 +29,8 @@ FROM $BASE_IMAGE AS release | ||||
| USER node | ||||
| WORKDIR /app | ||||
|  | ||||
| ENV UPTIME_KUMA_IS_CONTAINER=1 | ||||
|  | ||||
| # Copy app files from build layer | ||||
| COPY --chown=node:node --from=build /app /app | ||||
|  | ||||
| @@ -70,7 +72,7 @@ RUN git clone https://github.com/louislam/uptime-kuma.git . | ||||
| RUN npm ci | ||||
|  | ||||
| EXPOSE 3000 3001 | ||||
| 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 | ||||
| CMD ["npm", "run", "start-pr-test"] | ||||
|  | ||||
| ############################################ | ||||
|   | ||||
| @@ -22,7 +22,8 @@ if (! exists) { | ||||
|     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n"); | ||||
|  | ||||
|     // Also update package-lock.json | ||||
|     childProcess.spawnSync("npm", [ "install" ]); | ||||
|     const npm = /^win/.test(process.platform) ? "npm.cmd" : "npm"; | ||||
|     childProcess.spawnSync(npm, [ "install" ]); | ||||
|  | ||||
|     commit(version); | ||||
|     tag(version); | ||||
|   | ||||
							
								
								
									
										60
									
								
								extra/deploy-demo-server.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								extra/deploy-demo-server.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| require("dotenv").config(); | ||||
| const { NodeSSH } = require("node-ssh"); | ||||
| const readline = require("readline"); | ||||
| const rl = readline.createInterface({ input: process.stdin, | ||||
|     output: process.stdout }); | ||||
| const prompt = (query) => new Promise((resolve) => rl.question(query, resolve)); | ||||
|  | ||||
| (async () => { | ||||
|     try { | ||||
|         console.log("SSH to demo server"); | ||||
|         const ssh = new NodeSSH(); | ||||
|         await ssh.connect({ | ||||
|             host: process.env.UPTIME_KUMA_DEMO_HOST, | ||||
|             port: process.env.UPTIME_KUMA_DEMO_PORT, | ||||
|             username: process.env.UPTIME_KUMA_DEMO_USERNAME, | ||||
|             privateKeyPath: process.env.UPTIME_KUMA_DEMO_PRIVATE_KEY_PATH | ||||
|         }); | ||||
|  | ||||
|         let cwd = process.env.UPTIME_KUMA_DEMO_CWD; | ||||
|         let result; | ||||
|  | ||||
|         const version = await prompt("Enter Version: "); | ||||
|  | ||||
|         result = await ssh.execCommand("git fetch --all", { | ||||
|             cwd, | ||||
|         }); | ||||
|         console.log(result.stdout + result.stderr); | ||||
|  | ||||
|         await prompt("Press any key to continue..."); | ||||
|  | ||||
|         result = await ssh.execCommand(`git checkout ${version} --force`, { | ||||
|             cwd, | ||||
|         }); | ||||
|         console.log(result.stdout + result.stderr); | ||||
|  | ||||
|         result = await ssh.execCommand("npm run download-dist", { | ||||
|             cwd, | ||||
|         }); | ||||
|         console.log(result.stdout + result.stderr); | ||||
|  | ||||
|         result = await ssh.execCommand("npm install --production", { | ||||
|             cwd, | ||||
|         }); | ||||
|         console.log(result.stdout + result.stderr); | ||||
|  | ||||
|         /* | ||||
|         result = await ssh.execCommand("pm2 restart 1", { | ||||
|             cwd, | ||||
|         }); | ||||
|         console.log(result.stdout + result.stderr);*/ | ||||
|  | ||||
|     } catch (e) { | ||||
|         console.log(e); | ||||
|     } finally { | ||||
|         rl.close(); | ||||
|     } | ||||
| })(); | ||||
|  | ||||
| // When done reading prompt, exit program | ||||
| rl.on("close", () => process.exit(0)); | ||||
| @@ -1,48 +0,0 @@ | ||||
| // | ||||
|  | ||||
| const http = require("https"); // or 'https' for https:// URLs | ||||
| const fs = require("fs"); | ||||
|  | ||||
| const platform = process.argv[2]; | ||||
|  | ||||
| if (!platform) { | ||||
|     console.error("No platform??"); | ||||
|     process.exit(1); | ||||
| } | ||||
|  | ||||
| let arch = null; | ||||
|  | ||||
| if (platform === "linux/amd64") { | ||||
|     arch = "amd64"; | ||||
| } else if (platform === "linux/arm64") { | ||||
|     arch = "arm64"; | ||||
| } else if (platform === "linux/arm/v7") { | ||||
|     arch = "arm"; | ||||
| } else { | ||||
|     console.error("Invalid platform?? " + platform); | ||||
| } | ||||
|  | ||||
| const file = fs.createWriteStream("cloudflared.deb"); | ||||
| get("https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-linux-" + arch + ".deb"); | ||||
|  | ||||
| /** | ||||
|  * Download specified file | ||||
|  * @param {string} url URL to request | ||||
|  */ | ||||
| function get(url) { | ||||
|     http.get(url, function (res) { | ||||
|         if (res.statusCode >= 300 && res.statusCode < 400 && res.headers.location) { | ||||
|             console.log("Redirect to " + res.headers.location); | ||||
|             get(res.headers.location); | ||||
|         } else if (res.statusCode >= 200 && res.statusCode < 300) { | ||||
|             res.pipe(file); | ||||
|  | ||||
|             res.on("end", function () { | ||||
|                 console.log("Downloaded"); | ||||
|             }); | ||||
|         } else { | ||||
|             console.error(res.statusCode); | ||||
|             process.exit(1); | ||||
|         } | ||||
|     }); | ||||
| } | ||||
| @@ -47,6 +47,7 @@ function download(url) { | ||||
|                     }); | ||||
|                 } | ||||
|                 console.log("Done"); | ||||
|                 process.exit(0); | ||||
|             }); | ||||
|  | ||||
|             tarStream.on("error", () => { | ||||
|   | ||||
							
								
								
									
										1
									
								
								extra/exe-builder/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								extra/exe-builder/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| packages/ | ||||
							
								
								
									
										35
									
								
								extra/exe-builder/App.config
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								extra/exe-builder/App.config
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <configuration> | ||||
|     <startup> | ||||
|         <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.7.2" /> | ||||
|     </startup> | ||||
|  | ||||
|   <runtime> | ||||
|     <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1"> | ||||
|       <dependentAssembly> | ||||
|         <assemblyIdentity name="System.Diagnostics.DiagnosticSource" publicKeyToken="cc7b13ffcd2ddd51" culture="neutral" /> | ||||
|         <bindingRedirect oldVersion="0.0.0.0-4.0.1.0" newVersion="4.0.1.0" /> | ||||
|       </dependentAssembly> | ||||
|       <dependentAssembly> | ||||
|         <assemblyIdentity name="System.Diagnostics.Tracing" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" /> | ||||
|         <bindingRedirect oldVersion="0.0.0.0-4.1.1.0" newVersion="4.1.1.0" /> | ||||
|       </dependentAssembly> | ||||
|       <dependentAssembly> | ||||
|         <assemblyIdentity name="System.Reflection" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" /> | ||||
|         <bindingRedirect oldVersion="0.0.0.0-4.1.1.0" newVersion="4.1.1.0" /> | ||||
|       </dependentAssembly> | ||||
|       <dependentAssembly> | ||||
|         <assemblyIdentity name="System.Runtime" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" /> | ||||
|         <bindingRedirect oldVersion="0.0.0.0-4.1.1.1" newVersion="4.1.1.1" /> | ||||
|       </dependentAssembly> | ||||
|       <dependentAssembly> | ||||
|         <assemblyIdentity name="System.Runtime.InteropServices" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" /> | ||||
|         <bindingRedirect oldVersion="0.0.0.0-4.1.1.0" newVersion="4.1.1.0" /> | ||||
|       </dependentAssembly> | ||||
|       <dependentAssembly> | ||||
|         <assemblyIdentity name="System.Runtime.CompilerServices.Unsafe" publicKeyToken="b03f5f7f11d50a3a" culture="neutral" /> | ||||
|         <bindingRedirect oldVersion="0.0.0.0-6.0.0.0" newVersion="6.0.0.0" /> | ||||
|       </dependentAssembly> | ||||
|     </assemblyBinding> | ||||
|   </runtime> | ||||
| </configuration> | ||||
							
								
								
									
										84
									
								
								extra/exe-builder/DownloadForm.Designer.cs
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								extra/exe-builder/DownloadForm.Designer.cs
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| using System.ComponentModel; | ||||
|  | ||||
| namespace UptimeKuma { | ||||
|     partial class DownloadForm { | ||||
|         /// <summary> | ||||
|         /// Required designer variable. | ||||
|         /// </summary> | ||||
|         private IContainer components = null; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Clean up any resources being used. | ||||
|         /// </summary> | ||||
|         /// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param> | ||||
|         protected override void Dispose(bool disposing) { | ||||
|             if (disposing && (components != null)) { | ||||
|                 components.Dispose(); | ||||
|             } | ||||
|  | ||||
|             base.Dispose(disposing); | ||||
|         } | ||||
|  | ||||
|         #region Windows Form Designer generated code | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Required method for Designer support - do not modify | ||||
|         /// the contents of this method with the code editor. | ||||
|         /// </summary> | ||||
|         private void InitializeComponent() { | ||||
|             System.ComponentModel.ComponentResourceManager resources = new System.ComponentModel.ComponentResourceManager(typeof(DownloadForm)); | ||||
|             this.progressBar = new System.Windows.Forms.ProgressBar(); | ||||
|             this.label = new System.Windows.Forms.Label(); | ||||
|             this.labelData = new System.Windows.Forms.Label(); | ||||
|             this.SuspendLayout(); | ||||
|             //  | ||||
|             // progressBar | ||||
|             //  | ||||
|             this.progressBar.Location = new System.Drawing.Point(12, 12); | ||||
|             this.progressBar.Name = "progressBar"; | ||||
|             this.progressBar.Size = new System.Drawing.Size(472, 41); | ||||
|             this.progressBar.TabIndex = 0; | ||||
|             //  | ||||
|             // label | ||||
|             //  | ||||
|             this.label.Location = new System.Drawing.Point(12, 59); | ||||
|             this.label.Name = "label"; | ||||
|             this.label.Size = new System.Drawing.Size(472, 23); | ||||
|             this.label.TabIndex = 1; | ||||
|             this.label.Text = "Preparing..."; | ||||
|             //  | ||||
|             // labelData | ||||
|             //  | ||||
|             this.labelData.Location = new System.Drawing.Point(12, 82); | ||||
|             this.labelData.Name = "labelData"; | ||||
|             this.labelData.Size = new System.Drawing.Size(472, 23); | ||||
|             this.labelData.TabIndex = 2; | ||||
|             //  | ||||
|             // DownloadForm | ||||
|             //  | ||||
|             this.AutoScaleDimensions = new System.Drawing.SizeF(8F, 16F); | ||||
|             this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font; | ||||
|             this.ClientSize = new System.Drawing.Size(496, 117); | ||||
|             this.Controls.Add(this.labelData); | ||||
|             this.Controls.Add(this.label); | ||||
|             this.Controls.Add(this.progressBar); | ||||
|             this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedDialog; | ||||
|             this.Icon = ((System.Drawing.Icon)(resources.GetObject("$this.Icon"))); | ||||
|             this.MaximizeBox = false; | ||||
|             this.Name = "DownloadForm"; | ||||
|             this.StartPosition = System.Windows.Forms.FormStartPosition.CenterScreen; | ||||
|             this.Text = "Uptime Kuma"; | ||||
|             this.Load += new System.EventHandler(this.DownloadForm_Load); | ||||
|             this.ResumeLayout(false); | ||||
|         } | ||||
|  | ||||
|         private System.Windows.Forms.Label labelData; | ||||
|  | ||||
|         private System.Windows.Forms.Label label; | ||||
|  | ||||
|         private System.Windows.Forms.ProgressBar progressBar; | ||||
|  | ||||
|         #endregion | ||||
|     } | ||||
| } | ||||
|  | ||||
							
								
								
									
										204
									
								
								extra/exe-builder/DownloadForm.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								extra/exe-builder/DownloadForm.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,204 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.ComponentModel; | ||||
| using System.Diagnostics; | ||||
| using System.IO; | ||||
| using System.IO.Compression; | ||||
| using System.Net; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using System.Windows.Forms; | ||||
| using Newtonsoft.Json; | ||||
|  | ||||
| namespace UptimeKuma { | ||||
|     public partial class DownloadForm : Form { | ||||
|         private readonly Queue<DownloadItem> downloadQueue = new(); | ||||
|         private readonly WebClient webClient = new(); | ||||
|         private DownloadItem currentDownloadItem; | ||||
|  | ||||
|         public DownloadForm() { | ||||
|             InitializeComponent(); | ||||
|         } | ||||
|  | ||||
|         private void DownloadForm_Load(object sender, EventArgs e) { | ||||
|             webClient.DownloadProgressChanged += DownloadProgressChanged; | ||||
|             webClient.DownloadFileCompleted += DownloadFileCompleted; | ||||
|  | ||||
|             label.Text = "Reading latest version..."; | ||||
|  | ||||
|             // Read json from https://uptime.kuma.pet/version | ||||
|             var versionJson = new WebClient().DownloadString("https://uptime.kuma.pet/version"); | ||||
|             var versionObj = JsonConvert.DeserializeObject<Version>(versionJson); | ||||
|  | ||||
|             var nodeVersion = versionObj.nodejs; | ||||
|             var uptimeKumaVersion = versionObj.latest; | ||||
|             var hasUpdateFile = File.Exists("update"); | ||||
|  | ||||
|             if (!Directory.Exists("node")) { | ||||
|                 downloadQueue.Enqueue(new DownloadItem { | ||||
|                     URL = $"https://nodejs.org/dist/v{nodeVersion}/node-v{nodeVersion}-win-x64.zip", | ||||
|                     Filename = "node.zip", | ||||
|                     TargetFolder = "node" | ||||
|                 }); | ||||
|             } | ||||
|  | ||||
|             if (!Directory.Exists("core") || hasUpdateFile) { | ||||
|  | ||||
|                 // It is update, rename the core folder to core.old | ||||
|                 if (Directory.Exists("core")) { | ||||
|                     // Remove the old core.old folder | ||||
|                     if (Directory.Exists("core.old")) { | ||||
|                         Directory.Delete("core.old", true); | ||||
|                     } | ||||
|  | ||||
|                     Directory.Move("core", "core.old"); | ||||
|                 } | ||||
|  | ||||
|                 downloadQueue.Enqueue(new DownloadItem { | ||||
|                     URL = $"https://github.com/louislam/uptime-kuma/archive/refs/tags/{uptimeKumaVersion}.zip", | ||||
|                     Filename = "core.zip", | ||||
|                     TargetFolder = "core" | ||||
|                 }); | ||||
|  | ||||
|                 File.WriteAllText("version.json", versionJson); | ||||
|  | ||||
|                 // Delete the update file | ||||
|                 if (hasUpdateFile) { | ||||
|                     File.Delete("update"); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             DownloadNextFile(); | ||||
|         } | ||||
|  | ||||
|         void DownloadNextFile() { | ||||
|             if (downloadQueue.Count > 0) { | ||||
|                 var item = downloadQueue.Dequeue(); | ||||
|  | ||||
|                 currentDownloadItem = item; | ||||
|  | ||||
|                 // Download if the zip file is not existing | ||||
|                 if (!File.Exists(item.Filename)) { | ||||
|                     label.Text = item.URL; | ||||
|                     webClient.DownloadFileAsync(new Uri(item.URL), item.Filename); | ||||
|                 } else { | ||||
|                     progressBar.Value = 100; | ||||
|                     label.Text = "Use local " + item.Filename; | ||||
|                     DownloadFileCompleted(null, null); | ||||
|                 } | ||||
|             } else { | ||||
|                 npmSetup(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         void npmSetup() { | ||||
|             labelData.Text = ""; | ||||
|  | ||||
|             var npm = "..\\node\\npm.cmd"; | ||||
|             var cmd = $"{npm} ci --production & {npm} run download-dist & exit"; | ||||
|  | ||||
|             var startInfo = new ProcessStartInfo { | ||||
|                 FileName = "cmd.exe", | ||||
|                 Arguments = $"/k \"{cmd}\"", | ||||
|                 RedirectStandardOutput = false, | ||||
|                 RedirectStandardError = false, | ||||
|                 RedirectStandardInput = true, | ||||
|                 UseShellExecute = false, | ||||
|                 CreateNoWindow = false, | ||||
|                 WorkingDirectory = "core" | ||||
|             }; | ||||
|  | ||||
|             var process = new Process(); | ||||
|             process.StartInfo = startInfo; | ||||
|             process.EnableRaisingEvents = true; | ||||
|             process.Exited += (_, e) => { | ||||
|                 progressBar.Value = 100; | ||||
|  | ||||
|                if (process.ExitCode == 0) { | ||||
|                    Task.Delay(2000).ContinueWith(_ => { | ||||
|                        Application.Restart(); | ||||
|                    }); | ||||
|                    label.Text = "Done"; | ||||
|                } else { | ||||
|                    label.Text = "Failed, exit code: " + process.ExitCode; | ||||
|                } | ||||
|  | ||||
|             }; | ||||
|             process.Start(); | ||||
|             label.Text = "Installing dependencies and download dist files"; | ||||
|             progressBar.Value = 50; | ||||
|             process.WaitForExit(); | ||||
|         } | ||||
|  | ||||
|         void DownloadProgressChanged(object sender, DownloadProgressChangedEventArgs e) { | ||||
|             progressBar.Value = e.ProgressPercentage; | ||||
|             var total = e.TotalBytesToReceive / 1024; | ||||
|             var current = e.BytesReceived / 1024; | ||||
|  | ||||
|             if (total > 0) { | ||||
|                 labelData.Text = $"{current}KB/{total}KB"; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         void DownloadFileCompleted(object sender, AsyncCompletedEventArgs e) { | ||||
|             Extract(currentDownloadItem); | ||||
|             DownloadNextFile(); | ||||
|         } | ||||
|  | ||||
|         void Extract(DownloadItem item) { | ||||
|             if (Directory.Exists(item.TargetFolder)) { | ||||
|                 var dir = new DirectoryInfo(item.TargetFolder); | ||||
|                 dir.Delete(true); | ||||
|             } | ||||
|  | ||||
|             if (Directory.Exists("temp")) { | ||||
|                 var dir = new DirectoryInfo("temp"); | ||||
|                 dir.Delete(true); | ||||
|             } | ||||
|  | ||||
|             labelData.Text = $"Extracting {item.Filename}..."; | ||||
|  | ||||
|             ZipFile.ExtractToDirectory(item.Filename, "temp"); | ||||
|  | ||||
|             string[] dirList; | ||||
|  | ||||
|             // Move to the correct level | ||||
|             dirList = Directory.GetDirectories("temp"); | ||||
|  | ||||
|  | ||||
|  | ||||
|             if (dirList.Length > 0) { | ||||
|                 var dir = dirList[0]; | ||||
|  | ||||
|                 // As sometime ExtractToDirectory is still locking the directory, loop until ok | ||||
|                 while (true) { | ||||
|                     try { | ||||
|                         Directory.Move(dir, item.TargetFolder); | ||||
|                         break; | ||||
|                     } catch (Exception exception) { | ||||
|                         Thread.Sleep(1000); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|             } else { | ||||
|                 MessageBox.Show("Unexcepted Error: Cannot move extracted files, folder not found."); | ||||
|             } | ||||
|  | ||||
|             labelData.Text = $"Extracted"; | ||||
|  | ||||
|             if (Directory.Exists("temp")) { | ||||
|                 var dir = new DirectoryInfo("temp"); | ||||
|                 dir.Delete(true); | ||||
|             } | ||||
|  | ||||
|             File.Delete(item.Filename); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public class DownloadItem { | ||||
|         public string URL { get; set; } | ||||
|         public string Filename { get; set; } | ||||
|         public string TargetFolder { get; set; } | ||||
|     } | ||||
| } | ||||
|  | ||||
							
								
								
									
										377
									
								
								extra/exe-builder/DownloadForm.resx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										377
									
								
								extra/exe-builder/DownloadForm.resx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,377 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <root> | ||||
|   <!--  | ||||
|     Microsoft ResX Schema  | ||||
|      | ||||
|     Version 2.0 | ||||
|      | ||||
|     The primary goals of this format is to allow a simple XML format  | ||||
|     that is mostly human readable. The generation and parsing of the  | ||||
|     various data types are done through the TypeConverter classes  | ||||
|     associated with the data types. | ||||
|      | ||||
|     Example: | ||||
|      | ||||
|     ... ado.net/XML headers & schema ... | ||||
|     <resheader name="resmimetype">text/microsoft-resx</resheader> | ||||
|     <resheader name="version">2.0</resheader> | ||||
|     <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> | ||||
|     <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> | ||||
|     <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> | ||||
|     <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> | ||||
|     <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> | ||||
|         <value>[base64 mime encoded serialized .NET Framework object]</value> | ||||
|     </data> | ||||
|     <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | ||||
|         <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> | ||||
|         <comment>This is a comment</comment> | ||||
|     </data> | ||||
|                  | ||||
|     There are any number of "resheader" rows that contain simple  | ||||
|     name/value pairs. | ||||
|      | ||||
|     Each data row contains a name, and value. The row also contains a  | ||||
|     type or mimetype. Type corresponds to a .NET class that support  | ||||
|     text/value conversion through the TypeConverter architecture.  | ||||
|     Classes that don't support this are serialized and stored with the  | ||||
|     mimetype set. | ||||
|      | ||||
|     The mimetype is used for serialized objects, and tells the  | ||||
|     ResXResourceReader how to depersist the object. This is currently not  | ||||
|     extensible. For a given mimetype the value must be set accordingly: | ||||
|      | ||||
|     Note - application/x-microsoft.net.object.binary.base64 is the format  | ||||
|     that the ResXResourceWriter will generate, however the reader can  | ||||
|     read any of the formats listed below. | ||||
|      | ||||
|     mimetype: application/x-microsoft.net.object.binary.base64 | ||||
|     value   : The object must be serialized with  | ||||
|             : System.Runtime.Serialization.Formatters.Binary.BinaryFormatter | ||||
|             : and then encoded with base64 encoding. | ||||
|      | ||||
|     mimetype: application/x-microsoft.net.object.soap.base64 | ||||
|     value   : The object must be serialized with  | ||||
|             : System.Runtime.Serialization.Formatters.Soap.SoapFormatter | ||||
|             : and then encoded with base64 encoding. | ||||
|  | ||||
|     mimetype: application/x-microsoft.net.object.bytearray.base64 | ||||
|     value   : The object must be serialized into a byte array  | ||||
|             : using a System.ComponentModel.TypeConverter | ||||
|             : and then encoded with base64 encoding. | ||||
|     --> | ||||
|   <xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> | ||||
|     <xsd:import namespace="http://www.w3.org/XML/1998/namespace" /> | ||||
|     <xsd:element name="root" msdata:IsDataSet="true"> | ||||
|       <xsd:complexType> | ||||
|         <xsd:choice maxOccurs="unbounded"> | ||||
|           <xsd:element name="metadata"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:sequence> | ||||
|                 <xsd:element name="value" type="xsd:string" minOccurs="0" /> | ||||
|               </xsd:sequence> | ||||
|               <xsd:attribute name="name" use="required" type="xsd:string" /> | ||||
|               <xsd:attribute name="type" type="xsd:string" /> | ||||
|               <xsd:attribute name="mimetype" type="xsd:string" /> | ||||
|               <xsd:attribute ref="xml:space" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|           <xsd:element name="assembly"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:attribute name="alias" type="xsd:string" /> | ||||
|               <xsd:attribute name="name" type="xsd:string" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|           <xsd:element name="data"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:sequence> | ||||
|                 <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||||
|                 <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> | ||||
|               </xsd:sequence> | ||||
|               <xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" /> | ||||
|               <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> | ||||
|               <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> | ||||
|               <xsd:attribute ref="xml:space" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|           <xsd:element name="resheader"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:sequence> | ||||
|                 <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||||
|               </xsd:sequence> | ||||
|               <xsd:attribute name="name" type="xsd:string" use="required" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|         </xsd:choice> | ||||
|       </xsd:complexType> | ||||
|     </xsd:element> | ||||
|   </xsd:schema> | ||||
|   <resheader name="resmimetype"> | ||||
|     <value>text/microsoft-resx</value> | ||||
|   </resheader> | ||||
|   <resheader name="version"> | ||||
|     <value>2.0</value> | ||||
|   </resheader> | ||||
|   <resheader name="reader"> | ||||
|     <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||||
|   </resheader> | ||||
|   <resheader name="writer"> | ||||
|     <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||||
|   </resheader> | ||||
|   <assembly alias="System.Drawing" name="System.Drawing, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a" /> | ||||
|   <data name="$this.Icon" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | ||||
|     <value> | ||||
|         AAABAAMAMDAAAAEAIACoJQAANgAAACAgAAABACAAqBAAAN4lAAAQEAAAAQAgAGgEAACGNgAAKAAAADAA | ||||
|         AABgAAAAAQAgAAAAAAAAJAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAA////BPT09Bfu7u4e8fHxJPPz8yv19fUy9fX1M/Pz8yvx8fEk9vb2HPPz8xXMzMwFAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP// | ||||
|         /wHv7+8f7u7uPPPz81Tx8fFs8fHxgPHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGB8fHxcfHx8V3x8fFI9PT0MOvr6w0AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AADy8vIU8fHxS/Dw8Hbx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fFr9PT0R/Dw8CIAAAABAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAA8vLyFPHx8Vnx8fGB8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fFs9fX1Mb+/vwQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAICAgALy8vI88fHxfvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvLy8nby8vI8gICAAgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAzMzMBfHx8Vrx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8vLyYf///wwAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADMzMwF8vLyYPHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8W/z8/MWAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADv7+9R8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLw8PB26urqDAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAPLy8ijx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgu7w7Ifj79ud2u7PtNLrw83P677dzeu85c3r | ||||
|         u+rM67rwzOu68c7rverQ68Dj0uvD3NbuyM3b7c+64u7apujv5ZPx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxXgAAAAEAAAAAAAAAAAAAAAAAAAAA4+PjCfDw | ||||
|         8Hfx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLd7tSmzeu92MbqsvvG6bH/xumy/8fq | ||||
|         s//H6rP/yOq0/8jqtf/J6rb/yeq2/8rrt//K67j/y+u4/8vruf/M67r/zOu7/83ru//Q7MDx1u7Kz9/t | ||||
|         163s8OuJ8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgu/v7y8AAAAAAAAAAAAA | ||||
|         AAAAAAAA7u7uPfHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC5PDdl8jqtuTE6a7/xOmv/8Xp | ||||
|         sP/G6bH/xumx/8bpsv/H6rP/x+qz/8jqtP/I6rX/yeq2/8nqtv/K67f/yuu4/8vruP/L67n/zOu6/8zr | ||||
|         u//N67v/zey8/87svf/P67742e3Mx+jv5ZLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvDw | ||||
|         8HWAgIACAAAAAAAAAACqqqoD8vLyc/Hx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLf7degxOiu+cPo | ||||
|         rf/D6a7/xOmu/8Xpr//F6bD/xumx/8bpsf/G6bL/x+qz/8fqs//I6rT/yOq1/8nqtv/J6rb/yuu3/8rr | ||||
|         uP/L67j/y+u5/8zruv/M67v/zeu7/83svP/O7L3/zuy9/87svfzc7tK28fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fEkAAAAAAAAAADz8/Mq8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgunv | ||||
|         5o3D6a/0wuis/8Lorf/D6K3/xOmu/8Tprv/F6a//xemw/8bpsf/G6bH/xumy/8fqs//H6rP/yOq0/8jq | ||||
|         tf/J6rb/yeq2/8rrt//K67j/y+u4/8vruf/M67r/zOu7/83ru//N7Lz/zuy9/87svf/O7L3/3e/TtPHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLy8vJNAAAAAAAAAADy8vJM8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgszqutDB6Kv/weir/8LorP/D6K3/w+it/8Tprv/E6a7/xemv/8XpsP/G6bH/xumx/8bp | ||||
|         sv/H6rP/x+qz/8jqtP/I6rX/yeq2/8nqtv/K67f/yuu4/8vruP/L67n/zOu6/8zru//N67v/zey8/87s | ||||
|         vf/O7L3/zuy++u3w6Yzx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLy8vJ1AAAAAAAAAADx8fFr8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC6O/kjsDoqvzA6Kr/weir/8Loq//C6Kz/w+it/8Porf/E6a7/xOmu/8Xp | ||||
|         r//F6bD/xumx/8bpsf/G6bL/x+qz/8fqtP/I6rT/yOq1/8nqtv/J6rb/yuu3/8rruP/L67n/y+u5/8zr | ||||
|         uv/M67v/zeu7/83svP/O7L3/zuy9/93u07Xx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC////Bv// | ||||
|         /wfx8fGB8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC1ezJsr/nqf/A56n/weiq/8Hoq//C6Kv/wuis/8Po | ||||
|         rf/D6K3/xOmu/8Pprv+856T/uOed/7bmmv+05Zf/teWZ/7jnnf+86KP/wOio/8fqs//J6rb/yeq2/8rr | ||||
|         t//K67j/y+u5/8vruf/M67r/zOu7/83ru//N7Lz/zuy9/9buyNLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8vLyE/Ly8hPx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGCy+q6zr/nqP/A56n/wOep/8Ho | ||||
|         qv/B6Kv/wuir/8LorP+u5Y//neF2/5bgav+V4Gr/luBr/5fhbP+Y4W7/meFv/5rhcf+b4nL/nOJ0/53i | ||||
|         dv+j5H//reaM/7nnnf/E6q//y+y4/8vruf/L67n/zOu6/8zru//N67v/zey8/9Lsxd/x8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC7+/vIPb29hzx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGCx+m03L/n | ||||
|         qP+/56j/wOep/8Dnqf/B6Kr/weir/7nmn/+R32T/kt9l/5PfZ/+U4Gj/leBq/5bga/+X4W3/mOFu/5nh | ||||
|         b/+a4XH/m+Jy/5zidP+d4nX/nuN3/5/jeP+f4nn/weqq/8rruP/L67n/y+u5/8zruv/M67v/zeu7/9Ls | ||||
|         w+Lx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8PDwI/Hx8SXx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGCxeix5L/nqP+/56j/v+eo/8Dnqf/A56n/weiq/7Pllv+Q3mP/kd9k/5LfZf+T32f/lOBo/5Xg | ||||
|         av+W4Gv/l+Ft/5jhbv+Z4W//muFx/5vicv+c4nT/neJ1/57jd/+f43j/xOmu/8rrt//K67j/y+u5/8vr | ||||
|         uf/M67r/zOu7/9Tsxtfx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC9PT0GO/v7yDx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGCx+m037/nqP+/56j/v+eo/7/nqP/A56n/wOip/7TmmP+P3mH/kN5j/5Hf | ||||
|         ZP+S32b/k99n/5TgaP+V4Gr/luBr/5fhbf+Y4W7/meFw/5rhcf+b4nL/nOJ0/53idf+h5Hz/yuu2/8nq | ||||
|         t//K67f/yuu4/8vruf/L67n/zOu6/9ftysrx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC7e3tDvT0 | ||||
|         9Bfx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGCyOq117/nqP+/56j/v+eo/7/nqP+/56j/wOep/7vn | ||||
|         of+O3mD/j95h/5DeY/+R32T/kt9m/5PfZ/+U4Gj/leBq/5bga/+X4W3/mOFu/5nhcP+a4nH/m+Jy/5zi | ||||
|         dP+r5Yr/yOq1/8nqtv/J6rf/yuu3/8rruP/L67n/y+u5/9zu1LHx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLz8/OA////A+7u7g/x8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGCz+q+xb/nqP+/56j/v+eo/7/n | ||||
|         qP+/56j/v+eo/8Dnqf+S4Gb/jt5g/4/eYf+Q3mP/kd9k/5LfZv+T32f/lOBo/5Xgav+W4Gv/l+Ft/5jh | ||||
|         bv+Z4XD/muJx/5vic/+4553/yOq0/8jqtf/J6rb/yeq3/8rrt//K67j/y+u5/+bw4Zfx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fFrAAAAAP///wHz8/N88fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC1+zMrr/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+f4Xn/jd5f/47eYP+P3mH/kN5j/5HfZP+S32b/k99n/5Tg | ||||
|         af+V4Gr/luBr/5fhbf+Y4W7/meFw/5vic//F6rD/x+q0/8jqtP/I6rX/yeq2/8nqt//K67f/zOu88u/x | ||||
|         74Px8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLv7+9QAAAAAAAAAADw8PBm8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC5e7gk7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+u5I//jN1d/43eX/+O3mD/j95h/5De | ||||
|         Y/+R32T/kt9m/5PfZ/+U4Gn/leBq/5bga/+X4W3/mOFu/6rliP/G6rL/x+qz/8fqtP/I6rT/yOq1/8nq | ||||
|         tv/J6rf/1OzGy/Hx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YL19fUzAAAAAAAAAADy8vJO8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgsPoru2/56j/v+eo/7/nqP+/56j/v+eo/7/nqP++6Kf/j95i/4zd | ||||
|         Xf+N3l//jt5g/4/eYv+Q3mP/kd9k/5LfZv+T32f/lOBp/5Xgav+W4Gz/l+Ft/7voov/G6bL/xuqy/8fq | ||||
|         s//H6rT/yOq1/8jqtf/J6rb/4e/Zo/Hx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLw8PARAAAAAAAA | ||||
|         AADu7u4u8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgszpvMm/56j/v+eo/7/nqP+/56j/v+eo/7/n | ||||
|         qP+/56j/q+SL/4vdXP+M3V3/jd5f/47eYP+P3mL/kN9j/5HfZP+S32b/k99n/5Tgaf+V4Gr/qOOH/8Xp | ||||
|         sP/G6bH/xumy/8bqsv/H6rP/x+q0/8jqtf/K67jy8PHwhPHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8WoAAAAAAAAAAAAAAADo6OgL8fHxgfHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxguDv2J2/56j/v+eo/7/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/v+eo/6Xjgv+L3Vz/jN1d/43eX/+O3mD/j95i/5DfY/+R32T/kt9m/5Pf | ||||
|         Z/+k44D/xOmu/8XpsP/F6bD/xumx/8bpsv/G6rL/x+qz/8fqtP/W7cnB8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvPz80AAAAAAAAAAAAAAAAAAAAAA8PDwZ/Hx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLD6K/rv+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+u5I//kt5n/4zdXf+N3l//jt5g/4/e | ||||
|         Yv+Q32P/luFs/67kj//D6K3/xOmu/8Tpr//F6bD/xemw/8bpsf/G6bL/xuqy/8fqtP7o7+WR8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvPz8xYAAAAAAAAAAAAAAAAAAAAA8vLyPPHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLV7ci0v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/wOio/7Xl | ||||
|         mv+u5I7/rOSM/67kj/+35pz/wumr/8Lorf/D6K3/w+it/8Tprv/E6a//xemw/8XpsP/G6bH/xumy/9Ds | ||||
|         wNPx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8vLyZQAAAAAAAAAAAAAAAAAAAAAAAAAA////DPHx | ||||
|         8YDx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGCx+m03L/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/wOep/8Doqv/B6Kr/weir/8LorP/C6K3/w+it/8Porv/E6a7/xOmv/8Xp | ||||
|         sP/F6bD/yOq18uvw6Yvx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC7+/vMQAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAPHx8Vzx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC6O/ij8LorPG/56j/v+eo/7/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/8Dnqf/A6Kr/weiq/8Hoq//C6Kz/wuit/8Po | ||||
|         rf/D6K7/xOmu/8Tpr//F6bH74u/anvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLw8PB6////BQAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAPPz8yrx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxguHu | ||||
|         2pnB56v2v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP/A56n/wOiq/8Ho | ||||
|         q//B6Kv/wuis/8Lorf/D6K3/w+mu/8Tprv3b7dKq8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fFJAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAHy8vJf8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLi7tyXwumt8L/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/n | ||||
|         qP+/56j/wOep/8Doqv/B6Kv/weir/8LorP/C6K3/xOiv+d7u1aTx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvLy8nb///8KAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADv7+8Q8/Pze/Hx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC6/Dpiszqu82/56j/v+eo/7/nqP+/56j/v+eo/7/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/v+eo/8Dnqf/A6Kr/weir/8Hoq//H6bTj5e7elfHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvPz8yoAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAA9fX1MvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLe7tShx+mz3r/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/v+eo/7/nqP/A56n/xumy5drtz6rv8e+D8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8vLyTgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAPHx8Unx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgubv45DU68e2y+q6z8XoseTD6a7uweir9MPpru7F6bHly+q50tLsxLrl796U8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLy8vJh////AwAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP///wHx8fFZ8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8Wzf398IAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAD///8D8/PzVfHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8PDwZujo | ||||
|         6AsAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAA////AfHx8Ujx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fFa////BQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADz8/Mp8vLydvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8/PzfPHx8TcAAAABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA////CvLy8lDz8/N/8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvPz84Hx8fFa8PDwEQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AADw8PAR8vLyTvHx8X3x8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fF/8/PzVvT09BgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAP///wXz8/Mq8/PzU/Hx8XDx8fGB8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLy8vJz8fHxWO/v7y////8IAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAD///8G7e3tHfLy | ||||
|         8ifu7u4u8PDwNPT09C/y8vIo7+/vH+Pj4wkAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAP///////wAA////////AAD///////8AAP//gAf//wAA//gAAD//AAD/wAAAB/8AAP+A | ||||
|         AAAB/wAA/gAAAAB/AAD8AAAAAD8AAPgAAAAAHwAA8AAAAAAPAADwAAAAAAcAAOAAAAAABwAA4AAAAAAD | ||||
|         AADAAAAAAAMAAMAAAAAAAwAAwAAAAAABAACAAAAAAAEAAIAAAAAAAQAAgAAAAAABAACAAAAAAAEAAIAA | ||||
|         AAAAAQAAgAAAAAABAACAAAAAAAMAAMAAAAAAAwAAwAAAAAADAADAAAAAAAMAAMAAAAAABwAAwAAAAAAH | ||||
|         AADgAAAAAAcAAOAAAAAADwAA4AAAAAAPAADwAAAAAB8AAPAAAAAAHwAA+AAAAAA/AAD8AAAAAD8AAPwA | ||||
|         AAAAfwAA/gAAAAD/AAD/AAAAAf8AAP+AAAAD/wAA/8AAAAf/AAD/8AAAH/8AAP/8AAA//wAA//8AAf// | ||||
|         AAD//+AP//8AAP///////wAA////////AAD///////8AACgAAAAgAAAAQAAAAAEAIAAAAAAAABAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAgICAAu/v7xD09PQX7u7uHvDw8CP29vYb8vLyFOrq6gwAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAICA | ||||
|         gALy8vIm7+/vT/Pz82fz8/N98fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvDw8Hrw8PBm7+/vUPT0 | ||||
|         9C3o6OgLAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAOPj | ||||
|         4wnz8/NC8vLydPHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YHy8vJj8/PzKoCAgAIAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AADx8fEl8vLydfHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxcfHx8SUAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAA9PT0LfHx8YDx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8/PzgPLy8j0AAAABAAAAAAAA | ||||
|         AAAAAAAAAAAAAO3t7Rzx8fGA8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLr8OmM5O7emeTv | ||||
|         3Z7h79mj5fDem+nv45Tu8u6H8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvLy | ||||
|         8joAAAAAAAAAAAAAAAD///8E8fHxbvHx8YLx8fGC8fHxgvHx8YLx8fGC7vDshtns0K7N67zayeq288fq | ||||
|         s//I6rT/yOq1/8nqtv/K67f/y+u4/8vruf/P7L7w0+zF29vv0Lrn8OKX8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8/PzfvPz8xUAAAAAAAAAAPX19TLx8fGC8fHxgvHx8YLx8fGC8fHxgt3u1KXF6rHzxOmv/8Xp | ||||
|         sP/G6bH/xumy/8fqs//I6rT/yOq1/8nqtv/K67f/y+u4/8vruf/M67v/zey8/87svf/S7MPj4u7Zp/Hx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8/PzVQAAAAAAAAAA8fHxavHx8YLx8fGC8fHxgvHx8YLf7defwuis/cPo | ||||
|         rf/E6a7/xOmv/8XpsP/G6bH/xumy/8fqs//I6rT/yOq1/8nqtv/K67f/y+u4/8vruv/M67v/zey8/87s | ||||
|         vf/N67z/3e7SufHx8YLx8fGC8fHxgvHx8YLz8/N8////Bf///w3x8fGC8fHxgvHx8YLx8fGC8fHxgsXp | ||||
|         sOnB6Kv/wuis/8Porf/E6a7/xOmv/8XpsP/G6bH/xumy/8fqs//I6rT/yOq1/8nqtv/K67f/y+u4/8vr | ||||
|         uv/M67v/zey8/87svf/O67z96/Hoj/Hx8YLx8fGC8fHxgvHx8YLy8vIm8/PzK/Hx8YLx8fGC8fHxgvHx | ||||
|         8YLg79icwOep/8Hoqv/B6Kv/wuis/8Porf/E6a7/wuit/73opP+76KL/u+eh/77opv/D6a3/yeu1/8nq | ||||
|         tv/K67f/y+u5/8zruv/M67v/zey8/87svf/d7tSz8fHxgvHx8YLx8fGC8fHxgvHx8Tby8vI68fHxgvHx | ||||
|         8YLx8fGC8fHxgtTrxre/56j/wOep/8Hoqv/B6Kv/uOad/53idv+V4Gn/leBq/5fhbP+Y4W//muFx/5vi | ||||
|         c/+e4Xb/puWD/7PmlP/D6a3/y+u5/8zruv/M67v/zey8/9rtzsHx8fGC8fHxgvHx8YLx8fGC8/PzQfPz | ||||
|         80Lx8fGC8fHxgvHx8YLx8fGC0OvAwr/nqP+/56j/wOep/8Hoqv+o44b/kd9k/5LfZv+U4Gj/leBq/5fh | ||||
|         bf+Y4W//muFx/5vic/+d4nX/n+N3/7fnm//K67j/y+u5/8zruv/M67v/2u3QvPHx8YLx8fGC8fHxgvHx | ||||
|         8YLy8vI98/PzP/Hx8YLx8fGC8fHxgvHx8YLQ6sK/v+eo/7/nqP+/56j/wOep/6jjhv+P3mL/kd9k/5Lf | ||||
|         Zv+U4Gj/leBr/5fhbf+Y4W//muFx/5zic/+d4nX/v+mm/8nqt//K67j/y+u5/8zruv/f79au8fHxgvHx | ||||
|         8YLx8fGC8fHxgvX19TLx8fE38fHxgvHx8YLx8fGC8fHxgtTrybO/56j/v+eo/7/nqP+/56j/sOSS/47e | ||||
|         YP+P3mL/kd9k/5LfZv+U4Gj/leBr/5fhbf+Z4W//muJx/5/jd//H6bP/yeq2/8nqt//K67j/y+u5/+nv | ||||
|         45Tx8fGC8fHxgvHx8YLx8fGC7+/vIPHx8SXx8fGC8fHxgvHx8YLx8fGC4e/Zm7/nqP+/56j/v+eo/7/n | ||||
|         qP+956X/jt5h/47eYP+P3mL/kd9k/5LfZv+U4Gn/luBr/5fhbf+Z4W//q+aK/8fqs//I6rT/yeq2/8nq | ||||
|         t//N7Lvw8fHxgvHx8YLx8fGC8fHxgvPz84D///8G6+vrDfHx8YLx8fGC8fHxgvHx8YLv8e+Dweis87/n | ||||
|         qP+/56j/v+eo/7/nqP+d4XX/jN1e/47eYP+P3mL/kd9k/5PfZ/+U4Gn/luBr/5fhbf+86KP/xuqy/8fq | ||||
|         s//I6rX/yeq2/9Tsx8nx8fGC8fHxgvHx8YLx8fGC8PDwaAAAAAAAAAAA8fHxbPHx8YLx8fGC8fHxgvHx | ||||
|         8YLM6rrMv+eo/7/nqP+/56j/v+eo/7blmv+N3V//jN1e/47eYP+Q3mL/kd9k/5PfZ/+U4Gn/qeSH/8Xp | ||||
|         sP/G6bH/xuqy/8fqs//I6rX/5fDem/Hx8YLx8fGC8fHxgvHx8YLz8/M/AAAAAAAAAADz8/NB8fHxgvHx | ||||
|         8YLx8fGC8fHxgt3s06O/56j/v+eo/7/nqP+/56j/v+eo/7Xmmf+U32n/jN1e/47eYP+Q3mL/k99o/6zk | ||||
|         i//D6a7/xemv/8XpsP/G6bH/xuqy/8vqu+jx8fGC8fHxgvHx8YLx8fGC8fHxgvPz8xUAAAAAAAAAAPT0 | ||||
|         9Bfx8fGC8fHxgvHx8YLx8fGC8fHvg8Tpsee/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+35pz/suWV/7Xm | ||||
|         mf/A6Kj/wuit/8Porf/E6a7/xemv/8XpsP/G6bH/3e3UqvHx8YLx8fGC8fHxgvHx8YLw8PBmAAAAAAAA | ||||
|         AAAAAAAAAAAAAPHx8W7x8fGC8fHxgvHx8YLx8fGC4u7cmMHnqvm/56j/v+eo/7/nqP+/56j/v+eo/7/n | ||||
|         qP+/56j/wOep/8Hoqv/C6Kz/wuit/8Porf/E6a7/xemv/9Hrwszx8fGC8fHxgvHx8YLx8fGC8fHxgvX1 | ||||
|         9TEAAAAAAAAAAAAAAAAAAAAA7u7uO/Hx8YLx8fGC8fHxgvHx8YLx8fGC3e7SpMHoqfq/56j/v+eo/7/n | ||||
|         qP+/56j/v+eo/7/nqP+/56j/wOip/8Hoq//C6Kz/wuit/8Porf/O67zV8PHwhPHx8YLx8fGC8fHxgvHx | ||||
|         8YLy8vJ2////BQAAAAAAAAAAAAAAAAAAAACqqqoD8PDwafHx8YLx8fGC8fHxgvHx8YLx8fGC4O/YnMTo | ||||
|         ruy/56j/v+eo/7/nqP+/56j/v+eo/7/nqP+/56j/wOip/8Hoq//C6Kz90uvEwe/x74Px8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvPz8ykAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADz8/MW8fHxfPHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8PLuhdXtyLXF6bHlv+eo/7/nqP+/56j/v+eo/7/nqP/B6Kv0zeq8zOXv4JTx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLy8vJNAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADy8vIm8fHxgPHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLs8OmJ4e/Zm93u06Pf7def5+/hkvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxXf///wIAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AADy8vIo8/PzffHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8VnMzMwFAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAD29vYb8fHxbvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvPz83/v7+9BgICAAgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAADMzMwF8/PzQPLy8nnx8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgvPz84Hx8fFc9PT0GAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA////B/X19TLx8fFc8PDwevHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8fHxgPHx8Wv09PRE9PT0FwAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAA7+/vEPb29hvw8PAj7+/vH/T09Be/v78EAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA////////////8B///wAA//wAAD/wAAAP4AAAB+AA | ||||
|         AAfAAAADwAAAA4AAAAGAAAABgAAAAYAAAAGAAAABgAAAAYAAAAGAAAADwAAAA8AAAAPAAAAH4AAAB+AA | ||||
|         AA/wAAAP+AAAH/gAAD/+AAB//wAB///AA///+B////////////8oAAAAEAAAACAAAAABACAAAAAAAAAE | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA////CfDw8BH///8GAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAgICAAu7u7i7x8fFe8PDwevHx8YLx8fGC8fHxgvDw | ||||
|         8Hvx8fFs7+/vT/Dw8CMAAAABAAAAAAAAAAAAAAAA5ubmCvLy8l/x8fGC8fHxgvHx8YLx8fGC8fHxgvHx | ||||
|         8YLx8fGC8fHxgvHx8YLx8fGC8/PzZu7u7g8AAAAAAAAAAPHx8V3x8fGC8fHxgunv5o7Z7c200+vFytTs | ||||
|         xc7W7cnH2+7QueLu2qbu8OyH8fHxgvHx8YLx8fFu////BfHx8STx8fGC8fHxgtrtzq3D6a/8xemw/8bp | ||||
|         sv/I6rT/yeq2/8vruP/M67v/z+u++Nzu0bjx8fGC8fHxgu/v7zDx8fFI8fHxguzw6ojC56z3wuis/8Tp | ||||
|         rv/E6q3/weiq/8fqsv/J6rb/y+u5/8zru//N67z/6/HpjfHx8YLy8vJN8fHxXPHx8YLg79icv+eo/8Ho | ||||
|         qv+k4n//lOBo/5fhbf+a4XH/n+J5/7Pmlv/L67n/zOu7/+Xw353x8fGC8fHxXvHx8Vrx8fGC4O3Zm7/n | ||||
|         qP+/56j/nuF3/5HfZP+U4Gj/l+Ft/5ricf+x5pL/yeq3/8vruf/r8emN8fHxgu/v70/x8fFK8fHxguzw | ||||
|         6ojA6Kn8v+eo/6njiP+O3mD/kd9k/5Tgaf+X4W3/vuim/8jqtP/N67zr8fHxgvHx8YLy8vI68/PzK/Hx | ||||
|         8YLx8fGCx+m03L/nqP++6Kb/meBw/47eYP+S32X/q+SL/8XpsP/G6rL/1+zLvvHx8YLz8/OB8PDwEdXV | ||||
|         1Qbx8fF98fHxgt/t1Z/A56j9v+eo/7/nqP+656H/vuim/8Lorf/E6a7/yOq18Ovw6Yvx8fGC8vLyYwAA | ||||
|         AAAAAAAA8fHxR/Hx8YLx8fGC2O3NrMDnqfq/56j/v+eo/7/nqP/B6Kv/xumy7OTu3Zfx8fGC8/PzgfLy | ||||
|         8icAAAAAAAAAAP///wPz8/Nm8fHxgvHx8YLo7+SO0+zFuczquszM6bzJ1+zMru7w7Ibx8fGC8fHxgvHx | ||||
|         8UcAAAAAAAAAAAAAAAAAAAAA4+PjCfHx8Vzx8fGC8fHxgvHx8YLx8fGC8fHxgvHx8YLx8fGC8fHxgfPz | ||||
|         80D///8BAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAB8/PzK/Ly8mDz8/N+8fHxgvHx8YLy8vJ68vLyUezs | ||||
|         7BsAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAevr6w3j4+MJAAAAAAAA | ||||
|         AAAAAAAAAAAAAAAAAAAAAAAAAAAAAP//AAD8fwAA4AcAAMADAACAAQAAgAEAAIABAACAAQAAgAEAAIAB | ||||
|         AADAAwAAwAMAAOAHAADwDwAA/n8AAP//AAA= | ||||
| </value> | ||||
|   </data> | ||||
| </root> | ||||
							
								
								
									
										3
									
								
								extra/exe-builder/FodyWeavers.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								extra/exe-builder/FodyWeavers.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| <Weavers xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="FodyWeavers.xsd"> | ||||
|   <Costura DisableCompression='true' IncludeDebugSymbols='false' /> | ||||
| </Weavers> | ||||
							
								
								
									
										141
									
								
								extra/exe-builder/FodyWeavers.xsd
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								extra/exe-builder/FodyWeavers.xsd
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"> | ||||
|   <!-- This file was generated by Fody. Manual changes to this file will be lost when your project is rebuilt. --> | ||||
|   <xs:element name="Weavers"> | ||||
|     <xs:complexType> | ||||
|       <xs:all> | ||||
|         <xs:element name="Costura" minOccurs="0" maxOccurs="1"> | ||||
|           <xs:complexType> | ||||
|             <xs:all> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="ExcludeAssemblies" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>A list of assembly names to exclude from the default action of "embed all Copy Local references", delimited with line breaks</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="IncludeAssemblies" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>A list of assembly names to include from the default action of "embed all Copy Local references", delimited with line breaks.</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="ExcludeRuntimeAssemblies" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>A list of runtime assembly names to exclude from the default action of "embed all Copy Local references", delimited with line breaks</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="IncludeRuntimeAssemblies" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>A list of runtime assembly names to include from the default action of "embed all Copy Local references", delimited with line breaks.</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="Unmanaged32Assemblies" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>A list of unmanaged 32 bit assembly names to include, delimited with line breaks.</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="Unmanaged64Assemblies" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>A list of unmanaged 64 bit assembly names to include, delimited with line breaks.</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|               <xs:element minOccurs="0" maxOccurs="1" name="PreloadOrder" type="xs:string"> | ||||
|                 <xs:annotation> | ||||
|                   <xs:documentation>The order of preloaded assemblies, delimited with line breaks.</xs:documentation> | ||||
|                 </xs:annotation> | ||||
|               </xs:element> | ||||
|             </xs:all> | ||||
|             <xs:attribute name="CreateTemporaryAssemblies" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>This will copy embedded files to disk before loading them into memory. This is helpful for some scenarios that expected an assembly to be loaded from a physical file.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="IncludeDebugSymbols" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>Controls if .pdbs for reference assemblies are also embedded.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="IncludeRuntimeReferences" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>Controls if runtime assemblies are also embedded.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="UseRuntimeReferencePaths" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>Controls whether the runtime assemblies are embedded with their full path or only with their assembly name.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="DisableCompression" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>Embedded assemblies are compressed by default, and uncompressed when they are loaded. You can turn compression off with this option.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="DisableCleanup" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>As part of Costura, embedded assemblies are no longer included as part of the build. This cleanup can be turned off.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="LoadAtModuleInit" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>Costura by default will load as part of the module initialization. This flag disables that behavior. Make sure you call CosturaUtility.Initialize() somewhere in your code.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="IgnoreSatelliteAssemblies" type="xs:boolean"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>Costura will by default use assemblies with a name like 'resources.dll' as a satellite resource and prepend the output path. This flag disables that behavior.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="ExcludeAssemblies" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>A list of assembly names to exclude from the default action of "embed all Copy Local references", delimited with |</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="IncludeAssemblies" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>A list of assembly names to include from the default action of "embed all Copy Local references", delimited with |.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="ExcludeRuntimeAssemblies" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>A list of runtime assembly names to exclude from the default action of "embed all Copy Local references", delimited with |</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="IncludeRuntimeAssemblies" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>A list of runtime assembly names to include from the default action of "embed all Copy Local references", delimited with |.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="Unmanaged32Assemblies" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>A list of unmanaged 32 bit assembly names to include, delimited with |.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="Unmanaged64Assemblies" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>A list of unmanaged 64 bit assembly names to include, delimited with |.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|             <xs:attribute name="PreloadOrder" type="xs:string"> | ||||
|               <xs:annotation> | ||||
|                 <xs:documentation>The order of preloaded assemblies, delimited with |.</xs:documentation> | ||||
|               </xs:annotation> | ||||
|             </xs:attribute> | ||||
|           </xs:complexType> | ||||
|         </xs:element> | ||||
|       </xs:all> | ||||
|       <xs:attribute name="VerifyAssembly" type="xs:boolean"> | ||||
|         <xs:annotation> | ||||
|           <xs:documentation>'true' to run assembly verification (PEVerify) on the target assembly after all weavers have been executed.</xs:documentation> | ||||
|         </xs:annotation> | ||||
|       </xs:attribute> | ||||
|       <xs:attribute name="VerifyIgnoreCodes" type="xs:string"> | ||||
|         <xs:annotation> | ||||
|           <xs:documentation>A comma-separated list of error codes that can be safely ignored in assembly verification.</xs:documentation> | ||||
|         </xs:annotation> | ||||
|       </xs:attribute> | ||||
|       <xs:attribute name="GenerateXsd" type="xs:boolean"> | ||||
|         <xs:annotation> | ||||
|           <xs:documentation>'false' to turn off automatic generation of the XML Schema file.</xs:documentation> | ||||
|         </xs:annotation> | ||||
|       </xs:attribute> | ||||
|     </xs:complexType> | ||||
|   </xs:element> | ||||
| </xs:schema> | ||||
							
								
								
									
										243
									
								
								extra/exe-builder/Program.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										243
									
								
								extra/exe-builder/Program.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,243 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Diagnostics; | ||||
| using System.Drawing; | ||||
| using System.IO; | ||||
| using System.Linq; | ||||
| using System.Net; | ||||
| using System.Net.Sockets; | ||||
| using System.Reflection; | ||||
| using System.Runtime.InteropServices; | ||||
| using System.Threading; | ||||
| using System.Threading.Tasks; | ||||
| using System.Windows.Forms; | ||||
| using Microsoft.Win32; | ||||
| using Newtonsoft.Json; | ||||
| using UptimeKuma.Properties; | ||||
|  | ||||
| namespace UptimeKuma { | ||||
|     static class Program { | ||||
|         /// <summary> | ||||
|         /// The main entry point for the application. | ||||
|         /// </summary> | ||||
|         [STAThread] | ||||
|         static void Main(string[] args) { | ||||
|             var cwd = Path.GetDirectoryName(Application.ExecutablePath); | ||||
|  | ||||
|             if (cwd != null) { | ||||
|                 Environment.CurrentDirectory = cwd; | ||||
|             } | ||||
|  | ||||
|             Application.EnableVisualStyles(); | ||||
|             Application.SetCompatibleTextRenderingDefault(false); | ||||
|             Application.Run(new UptimeKumaApplicationContext()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public class UptimeKumaApplicationContext : ApplicationContext | ||||
|     { | ||||
|         private static Mutex mutex = null; | ||||
|  | ||||
|         const string appName = "Uptime Kuma"; | ||||
|  | ||||
|         private NotifyIcon trayIcon; | ||||
|         private Process process; | ||||
|  | ||||
|         private MenuItem statusMenuItem; | ||||
|         private MenuItem runWhenStarts; | ||||
|         private MenuItem openMenuItem; | ||||
|  | ||||
|         private RegistryKey registryKey = Registry.CurrentUser.OpenSubKey("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run", true); | ||||
|  | ||||
|  | ||||
|         public UptimeKumaApplicationContext() { | ||||
|  | ||||
|             // Single instance only | ||||
|             bool createdNew; | ||||
|             mutex = new Mutex(true, appName, out createdNew); | ||||
|             if (!createdNew) { | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             var startingText = "Starting server..."; | ||||
|             trayIcon = new NotifyIcon(); | ||||
|             trayIcon.Text = startingText; | ||||
|  | ||||
|             runWhenStarts = new MenuItem("Run when system starts", RunWhenStarts); | ||||
|             runWhenStarts.Checked = registryKey.GetValue(appName) != null; | ||||
|  | ||||
|             statusMenuItem = new MenuItem(startingText); | ||||
|             statusMenuItem.Enabled = false; | ||||
|  | ||||
|             openMenuItem = new MenuItem("Open", Open); | ||||
|             openMenuItem.Enabled = false; | ||||
|  | ||||
|             trayIcon.Icon = Icon.ExtractAssociatedIcon(Assembly.GetExecutingAssembly().Location); | ||||
|             trayIcon.ContextMenu = new ContextMenu(new MenuItem[] { | ||||
|                 statusMenuItem, | ||||
|                 openMenuItem, | ||||
|                 //new("Debug Console", DebugConsole), | ||||
|                 runWhenStarts, | ||||
|                 new("Check for Update...", CheckForUpdate), | ||||
|                 new("Visit GitHub...", VisitGitHub), | ||||
|                 new("About", About), | ||||
|                 new("Exit", Exit), | ||||
|             }); | ||||
|  | ||||
|             trayIcon.MouseDoubleClick += new MouseEventHandler(Open); | ||||
|             trayIcon.Visible = true; | ||||
|  | ||||
|             var hasUpdateFile = File.Exists("update"); | ||||
|  | ||||
|             if (!hasUpdateFile && Directory.Exists("core") && Directory.Exists("node") && Directory.Exists("core/node_modules") && Directory.Exists("core/dist")) { | ||||
|                 // Go go go | ||||
|                 StartProcess(); | ||||
|             } else { | ||||
|                 DownloadFiles(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         void DownloadFiles() { | ||||
|             var form = new DownloadForm(); | ||||
|             form.Closed += Exit; | ||||
|             form.Show(); | ||||
|         } | ||||
|  | ||||
|         private void RunWhenStarts(object sender, EventArgs e) { | ||||
|             if (registryKey == null) { | ||||
|                 MessageBox.Show("Error: Unable to set startup registry key."); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if (runWhenStarts.Checked) { | ||||
|                 registryKey.DeleteValue(appName, false); | ||||
|                 runWhenStarts.Checked = false; | ||||
|             } else { | ||||
|                 registryKey.SetValue(appName, Application.ExecutablePath); | ||||
|                 runWhenStarts.Checked = true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         void StartProcess() { | ||||
|             var startInfo = new ProcessStartInfo { | ||||
|                 FileName = "node/node.exe", | ||||
|                 Arguments = "server/server.js --data-dir=\"../data/\"", | ||||
|                 RedirectStandardOutput = false, | ||||
|                 RedirectStandardError = false, | ||||
|                 UseShellExecute = false, | ||||
|                 CreateNoWindow = true, | ||||
|                 WorkingDirectory = "core" | ||||
|             }; | ||||
|  | ||||
|             process = new Process(); | ||||
|             process.StartInfo = startInfo; | ||||
|             process.EnableRaisingEvents = true; | ||||
|             process.Exited += ProcessExited; | ||||
|  | ||||
|             try { | ||||
|                 process.Start(); | ||||
|                 //Open(null, null); | ||||
|  | ||||
|                 // Async task to check if the server is ready | ||||
|                 Task.Run(() => { | ||||
|                     var runningText = "Server is running"; | ||||
|                     using TcpClient tcpClient = new TcpClient(); | ||||
|                     while (true) { | ||||
|                         try { | ||||
|                             tcpClient.Connect("127.0.0.1", 3001); | ||||
|                             statusMenuItem.Text = runningText; | ||||
|                             openMenuItem.Enabled = true; | ||||
|                             trayIcon.Text = runningText; | ||||
|                             break; | ||||
|                         } catch (Exception) { | ||||
|                             System.Threading.Thread.Sleep(2000); | ||||
|                         } | ||||
|                     } | ||||
|                 }); | ||||
|  | ||||
|             } catch (Exception e) { | ||||
|                 MessageBox.Show("Startup failed: " + e.Message, "Uptime Kuma Error"); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         void StopProcess() { | ||||
|             process?.Kill(); | ||||
|         } | ||||
|  | ||||
|         void Open(object sender, EventArgs e) { | ||||
|             Process.Start("http://localhost:3001"); | ||||
|         } | ||||
|  | ||||
|         void DebugConsole(object sender, EventArgs e) { | ||||
|  | ||||
|         } | ||||
|  | ||||
|         void CheckForUpdate(object sender, EventArgs e) { | ||||
|             var needUpdate = false; | ||||
|  | ||||
|             // Check version.json exists | ||||
|             if (File.Exists("version.json")) { | ||||
|                 // Load version.json and compare with the latest version from GitHub | ||||
|                 var currentVersionObj = JsonConvert.DeserializeObject<Version>(File.ReadAllText("version.json")); | ||||
|  | ||||
|                 var versionJson = new WebClient().DownloadString("https://uptime.kuma.pet/version"); | ||||
|                 var latestVersionObj = JsonConvert.DeserializeObject<Version>(versionJson); | ||||
|  | ||||
|                 // Compare version, if the latest version is newer, then update | ||||
|                 if (new System.Version(latestVersionObj.latest).CompareTo(new System.Version(currentVersionObj.latest)) > 0) { | ||||
|                     var result = MessageBox.Show("A new version is available. Do you want to update?", "Update", MessageBoxButtons.YesNo); | ||||
|                     if (result == DialogResult.Yes) { | ||||
|                         // Create a empty file `update`, so the app will download the core files again at startup | ||||
|                         File.Create("update").Close(); | ||||
|  | ||||
|                         trayIcon.Visible = false; | ||||
|                         process?.Kill(); | ||||
|  | ||||
|                         // Restart the app, it will download the core files again at startup | ||||
|                         Application.Restart(); | ||||
|                     } | ||||
|                 } else { | ||||
|                     MessageBox.Show("You are using the latest version."); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|  | ||||
|         } | ||||
|  | ||||
|         void VisitGitHub(object sender, EventArgs e) | ||||
|         { | ||||
|             Process.Start("https://github.com/louislam/uptime-kuma"); | ||||
|         } | ||||
|  | ||||
|         void About(object sender, EventArgs e) | ||||
|         { | ||||
|             MessageBox.Show("Uptime Kuma Windows Runtime v1.0.0" + Environment.NewLine + "© 2023 Louis Lam", "Info"); | ||||
|         } | ||||
|  | ||||
|         void Exit(object sender, EventArgs e) | ||||
|         { | ||||
|             // Hide tray icon, otherwise it will remain shown until user mouses over it | ||||
|             trayIcon.Visible = false; | ||||
|             process?.Kill(); | ||||
|             Application.Exit(); | ||||
|         } | ||||
|  | ||||
|         void ProcessExited(object sender, EventArgs e) { | ||||
|  | ||||
|             if (process.ExitCode != 0) { | ||||
|                 var line = ""; | ||||
|                 while (!process.StandardOutput.EndOfStream) | ||||
|                 { | ||||
|                     line += process.StandardOutput.ReadLine(); | ||||
|                 } | ||||
|  | ||||
|                 MessageBox.Show("Uptime Kuma exited unexpectedly. Exit code: " + process.ExitCode + " " + line); | ||||
|             } | ||||
|  | ||||
|             trayIcon.Visible = false; | ||||
|             Application.Exit(); | ||||
|         } | ||||
|  | ||||
|     } | ||||
| } | ||||
|  | ||||
							
								
								
									
										36
									
								
								extra/exe-builder/Properties/AssemblyInfo.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								extra/exe-builder/Properties/AssemblyInfo.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| using System.Reflection; | ||||
| using System.Runtime.CompilerServices; | ||||
| using System.Runtime.InteropServices; | ||||
|  | ||||
| // General Information about an assembly is controlled through the following | ||||
| // set of attributes. Change these attribute values to modify the information | ||||
| // associated with an assembly. | ||||
| [assembly: AssemblyTitle("Uptime Kuma")] | ||||
| [assembly: AssemblyDescription("A portable executable for running Uptime Kuma")] | ||||
| [assembly: AssemblyConfiguration("")] | ||||
| [assembly: AssemblyCompany("Uptime Kuma")] | ||||
| [assembly: AssemblyProduct("Uptime Kuma")] | ||||
| [assembly: AssemblyCopyright("Copyright © 2023 Louis Lam")] | ||||
| [assembly: AssemblyTrademark("")] | ||||
| [assembly: AssemblyCulture("")] | ||||
|  | ||||
| // Setting ComVisible to false makes the types in this assembly not visible | ||||
| // to COM components.  If you need to access a type in this assembly from | ||||
| // COM, set the ComVisible attribute to true on that type. | ||||
| [assembly: ComVisible(false)] | ||||
|  | ||||
| // The following GUID is for the ID of the typelib if this project is exposed to COM | ||||
| [assembly: Guid("86B40AFB-61FC-433D-8C31-650B0F32EA8F")] | ||||
|  | ||||
| // Version information for an assembly consists of the following four values: | ||||
| // | ||||
| //      Major Version | ||||
| //      Minor Version | ||||
| //      Build Number | ||||
| //      Revision | ||||
| // | ||||
| // You can specify all the values or you can default the Build and Revision Numbers | ||||
| // by using the '*' as shown below: | ||||
| // [assembly: AssemblyVersion("1.0.*")] | ||||
| [assembly: AssemblyVersion("1.0.1.0")] | ||||
| [assembly: AssemblyFileVersion("1.0.1.0")] | ||||
							
								
								
									
										62
									
								
								extra/exe-builder/Properties/Resources.Designer.cs
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								extra/exe-builder/Properties/Resources.Designer.cs
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | ||||
| //------------------------------------------------------------------------------ | ||||
| // <auto-generated> | ||||
| //     This code was generated by a tool. | ||||
| //     Runtime Version:4.0.30319.42000 | ||||
| // | ||||
| //     Changes to this file may cause incorrect behavior and will be lost if | ||||
| //     the code is regenerated. | ||||
| // </auto-generated> | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
| namespace UptimeKuma.Properties { | ||||
|     /// <summary> | ||||
|     ///   A strongly-typed resource class, for looking up localized strings, etc. | ||||
|     /// </summary> | ||||
|     // This class was auto-generated by the StronglyTypedResourceBuilder | ||||
|     // class via a tool like ResGen or Visual Studio. | ||||
|     // To add or remove a member, edit your .ResX file then rerun ResGen | ||||
|     // with the /str option, or rebuild your VS project. | ||||
|     [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", | ||||
|         "4.0.0.0")] | ||||
|     [global::System.Diagnostics.DebuggerNonUserCodeAttribute()] | ||||
|     [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] | ||||
|     internal class Resources { | ||||
|         private static global::System.Resources.ResourceManager resourceMan; | ||||
|  | ||||
|         private static global::System.Globalization.CultureInfo resourceCulture; | ||||
|  | ||||
|         [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", | ||||
|             "CA1811:AvoidUncalledPrivateCode")] | ||||
|         internal Resources() { | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         ///   Returns the cached ResourceManager instance used by this class. | ||||
|         /// </summary> | ||||
|         [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState | ||||
|             .Advanced)] | ||||
|         internal static global::System.Resources.ResourceManager ResourceManager { | ||||
|             get { | ||||
|                 if ((resourceMan == null)) { | ||||
|                     global::System.Resources.ResourceManager temp = | ||||
|                         new global::System.Resources.ResourceManager("UptimeKuma.Properties.Resources", | ||||
|                             typeof(Resources).Assembly); | ||||
|                     resourceMan = temp; | ||||
|                 } | ||||
|  | ||||
|                 return resourceMan; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         ///   Overrides the current thread's CurrentUICulture property for all | ||||
|         ///   resource lookups using this strongly typed resource class. | ||||
|         /// </summary> | ||||
|         [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState | ||||
|             .Advanced)] | ||||
|         internal static global::System.Globalization.CultureInfo Culture { | ||||
|             get { return resourceCulture; } | ||||
|             set { resourceCulture = value; } | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										117
									
								
								extra/exe-builder/Properties/Resources.resx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								extra/exe-builder/Properties/Resources.resx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <root> | ||||
|   <!--  | ||||
|     Microsoft ResX Schema  | ||||
|      | ||||
|     Version 2.0 | ||||
|      | ||||
|     The primary goals of this format is to allow a simple XML format  | ||||
|     that is mostly human readable. The generation and parsing of the  | ||||
|     various data types are done through the TypeConverter classes  | ||||
|     associated with the data types. | ||||
|      | ||||
|     Example: | ||||
|      | ||||
|     ... ado.net/XML headers & schema ... | ||||
|     <resheader name="resmimetype">text/microsoft-resx</resheader> | ||||
|     <resheader name="version">2.0</resheader> | ||||
|     <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> | ||||
|     <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> | ||||
|     <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> | ||||
|     <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> | ||||
|     <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> | ||||
|         <value>[base64 mime encoded serialized .NET Framework object]</value> | ||||
|     </data> | ||||
|     <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | ||||
|         <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> | ||||
|         <comment>This is a comment</comment> | ||||
|     </data> | ||||
|                  | ||||
|     There are any number of "resheader" rows that contain simple  | ||||
|     name/value pairs. | ||||
|      | ||||
|     Each data row contains a name, and value. The row also contains a  | ||||
|     type or mimetype. Type corresponds to a .NET class that support  | ||||
|     text/value conversion through the TypeConverter architecture.  | ||||
|     Classes that don't support this are serialized and stored with the  | ||||
|     mimetype set. | ||||
|      | ||||
|     The mimetype is used for serialized objects, and tells the  | ||||
|     ResXResourceReader how to depersist the object. This is currently not  | ||||
|     extensible. For a given mimetype the value must be set accordingly: | ||||
|      | ||||
|     Note - application/x-microsoft.net.object.binary.base64 is the format  | ||||
|     that the ResXResourceWriter will generate, however the reader can  | ||||
|     read any of the formats listed below. | ||||
|      | ||||
|     mimetype: application/x-microsoft.net.object.binary.base64 | ||||
|     value   : The object must be serialized with  | ||||
|             : System.Serialization.Formatters.Binary.BinaryFormatter | ||||
|             : and then encoded with base64 encoding. | ||||
|      | ||||
|     mimetype: application/x-microsoft.net.object.soap.base64 | ||||
|     value   : The object must be serialized with  | ||||
|             : System.Runtime.Serialization.Formatters.Soap.SoapFormatter | ||||
|             : and then encoded with base64 encoding. | ||||
|  | ||||
|     mimetype: application/x-microsoft.net.object.bytearray.base64 | ||||
|     value   : The object must be serialized into a byte array  | ||||
|             : using a System.ComponentModel.TypeConverter | ||||
|             : and then encoded with base64 encoding. | ||||
|     --> | ||||
|   <xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> | ||||
|     <xsd:element name="root" msdata:IsDataSet="true"> | ||||
|       <xsd:complexType> | ||||
|         <xsd:choice maxOccurs="unbounded"> | ||||
|           <xsd:element name="metadata"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:sequence> | ||||
|                 <xsd:element name="value" type="xsd:string" minOccurs="0" /> | ||||
|               </xsd:sequence> | ||||
|               <xsd:attribute name="name" type="xsd:string" /> | ||||
|               <xsd:attribute name="type" type="xsd:string" /> | ||||
|               <xsd:attribute name="mimetype" type="xsd:string" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|           <xsd:element name="assembly"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:attribute name="alias" type="xsd:string" /> | ||||
|               <xsd:attribute name="name" type="xsd:string" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|           <xsd:element name="data"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:sequence> | ||||
|                 <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||||
|                 <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> | ||||
|               </xsd:sequence> | ||||
|               <xsd:attribute name="name" type="xsd:string" msdata:Ordinal="1" /> | ||||
|               <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> | ||||
|               <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|           <xsd:element name="resheader"> | ||||
|             <xsd:complexType> | ||||
|               <xsd:sequence> | ||||
|                 <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | ||||
|               </xsd:sequence> | ||||
|               <xsd:attribute name="name" type="xsd:string" use="required" /> | ||||
|             </xsd:complexType> | ||||
|           </xsd:element> | ||||
|         </xsd:choice> | ||||
|       </xsd:complexType> | ||||
|     </xsd:element> | ||||
|   </xsd:schema> | ||||
|   <resheader name="resmimetype"> | ||||
|     <value>text/microsoft-resx</value> | ||||
|   </resheader> | ||||
|   <resheader name="version"> | ||||
|     <value>2.0</value> | ||||
|   </resheader> | ||||
|   <resheader name="reader"> | ||||
|     <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||||
|   </resheader> | ||||
|   <resheader name="writer"> | ||||
|     <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | ||||
|   </resheader> | ||||
| </root> | ||||
							
								
								
									
										23
									
								
								extra/exe-builder/Properties/Settings.Designer.cs
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								extra/exe-builder/Properties/Settings.Designer.cs
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| //------------------------------------------------------------------------------ | ||||
| // <auto-generated> | ||||
| //     This code was generated by a tool. | ||||
| //     Runtime Version:4.0.30319.42000 | ||||
| // | ||||
| //     Changes to this file may cause incorrect behavior and will be lost if | ||||
| //     the code is regenerated. | ||||
| // </auto-generated> | ||||
| //------------------------------------------------------------------------------ | ||||
|  | ||||
| namespace UptimeKuma.Properties { | ||||
|     [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()] | ||||
|     [global::System.CodeDom.Compiler.GeneratedCodeAttribute( | ||||
|         "Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "11.0.0.0")] | ||||
|     internal sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase { | ||||
|         private static Settings defaultInstance = | ||||
|             ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings()))); | ||||
|  | ||||
|         public static Settings Default { | ||||
|             get { return defaultInstance; } | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										7
									
								
								extra/exe-builder/Properties/Settings.settings
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								extra/exe-builder/Properties/Settings.settings
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| <?xml version='1.0' encoding='utf-8'?> | ||||
| <SettingsFile xmlns="http://schemas.microsoft.com/VisualStudio/2004/01/settings" CurrentProfile="(Default)"> | ||||
|   <Profiles> | ||||
|     <Profile Name="(Default)" /> | ||||
|   </Profiles> | ||||
|   <Settings /> | ||||
| </SettingsFile> | ||||
							
								
								
									
										212
									
								
								extra/exe-builder/UptimeKuma.csproj
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										212
									
								
								extra/exe-builder/UptimeKuma.csproj
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,212 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <Project ToolsVersion="15.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> | ||||
|     <Import Project="packages\Costura.Fody.5.7.0\build\Costura.Fody.props" Condition="Exists('packages\Costura.Fody.5.7.0\build\Costura.Fody.props')" /> | ||||
|     <Import Project="$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props" Condition="Exists('$(MSBuildExtensionsPath)\$(MSBuildToolsVersion)\Microsoft.Common.props')" /> | ||||
|     <PropertyGroup> | ||||
|         <Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration> | ||||
|         <Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform> | ||||
|         <ProjectGuid>{2DB53988-1D93-4AC0-90C4-96ADEAAC5C04}</ProjectGuid> | ||||
|         <OutputType>WinExe</OutputType> | ||||
|         <RootNamespace>UptimeKuma</RootNamespace> | ||||
|         <AssemblyName>uptime-kuma</AssemblyName> | ||||
|         <TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion> | ||||
|         <FileAlignment>512</FileAlignment> | ||||
|         <AutoGenerateBindingRedirects>true</AutoGenerateBindingRedirects> | ||||
|         <Deterministic>true</Deterministic> | ||||
|         <ApplicationIcon>..\..\public\favicon.ico</ApplicationIcon> | ||||
|         <LangVersion>9</LangVersion> | ||||
|     </PropertyGroup> | ||||
|     <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' "> | ||||
|         <PlatformTarget>AnyCPU</PlatformTarget> | ||||
|         <DebugSymbols>true</DebugSymbols> | ||||
|         <DebugType>full</DebugType> | ||||
|         <Optimize>false</Optimize> | ||||
|         <OutputPath>bin\Debug\</OutputPath> | ||||
|         <DefineConstants>DEBUG;TRACE</DefineConstants> | ||||
|         <ErrorReport>prompt</ErrorReport> | ||||
|         <WarningLevel>4</WarningLevel> | ||||
|     </PropertyGroup> | ||||
|     <PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' "> | ||||
|         <PlatformTarget>AnyCPU</PlatformTarget> | ||||
|         <DebugType>pdbonly</DebugType> | ||||
|         <Optimize>true</Optimize> | ||||
|         <OutputPath>bin\Release\</OutputPath> | ||||
|         <DefineConstants>TRACE</DefineConstants> | ||||
|         <ErrorReport>prompt</ErrorReport> | ||||
|         <WarningLevel>4</WarningLevel> | ||||
|     </PropertyGroup> | ||||
|     <PropertyGroup> | ||||
|         <ApplicationManifest>app.manifest</ApplicationManifest> | ||||
|     </PropertyGroup> | ||||
|     <PropertyGroup> | ||||
|       <PostBuildEvent>COPY "$(SolutionDir)bin\Debug\uptime-kuma.exe" "%UserProfile%\Desktop\uptime-kuma-win64\"</PostBuildEvent> | ||||
|     </PropertyGroup> | ||||
|     <ItemGroup> | ||||
|         <Reference Include="Costura, Version=5.7.0.0, Culture=neutral, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\Costura.Fody.5.7.0\lib\netstandard1.0\Costura.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="Microsoft.Win32.Primitives, Version=4.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\Microsoft.Win32.Primitives.4.3.0\lib\net46\Microsoft.Win32.Primitives.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="mscorlib" /> | ||||
|         <Reference Include="Newtonsoft.Json, Version=13.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\Newtonsoft.Json.13.0.2\lib\net45\Newtonsoft.Json.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System" /> | ||||
|         <Reference Include="System.AppContext, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.AppContext.4.3.0\lib\net463\System.AppContext.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Buffers, Version=4.0.3.0, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Buffers.4.5.1\lib\net461\System.Buffers.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.ComponentModel.Composition" /> | ||||
|         <Reference Include="System.Console, Version=4.0.1.1, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Console.4.3.1\lib\net46\System.Console.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Core" /> | ||||
|         <Reference Include="System.Diagnostics.DiagnosticSource, Version=7.0.0.1, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Diagnostics.DiagnosticSource.7.0.1\lib\net462\System.Diagnostics.DiagnosticSource.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Diagnostics.Tracing, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Diagnostics.Tracing.4.3.0\lib\net462\System.Diagnostics.Tracing.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Globalization.Calendars, Version=4.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Globalization.Calendars.4.3.0\lib\net46\System.Globalization.Calendars.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.IO, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.IO.4.3.0\lib\net462\System.IO.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.IO.Compression, Version=4.1.2.0, Culture=neutral, PublicKeyToken=b77a5c561934e089, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.IO.Compression.4.3.0\lib\net46\System.IO.Compression.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.IO.Compression.FileSystem" /> | ||||
|         <Reference Include="System.IO.Compression.ZipFile, Version=4.0.2.0, Culture=neutral, PublicKeyToken=b77a5c561934e089, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.IO.Compression.ZipFile.4.3.0\lib\net46\System.IO.Compression.ZipFile.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.IO.FileSystem, Version=4.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.IO.FileSystem.4.3.0\lib\net46\System.IO.FileSystem.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.IO.FileSystem.Primitives, Version=4.0.2.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.IO.FileSystem.Primitives.4.3.0\lib\net46\System.IO.FileSystem.Primitives.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Linq, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Linq.4.3.0\lib\net463\System.Linq.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Linq.Expressions, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Linq.Expressions.4.3.0\lib\net463\System.Linq.Expressions.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Memory, Version=4.0.1.2, Culture=neutral, PublicKeyToken=cc7b13ffcd2ddd51, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Memory.4.5.5\lib\net461\System.Memory.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Net.Http, Version=4.1.1.3, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Net.Http.4.3.4\lib\net46\System.Net.Http.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Net.Sockets, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Net.Sockets.4.3.0\lib\net46\System.Net.Sockets.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Numerics" /> | ||||
|         <Reference Include="System.Numerics.Vectors, Version=4.1.4.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Numerics.Vectors.4.5.0\lib\net46\System.Numerics.Vectors.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Reflection, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Reflection.4.3.0\lib\net462\System.Reflection.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Runtime, Version=4.1.1.1, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Runtime.4.3.1\lib\net462\System.Runtime.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Runtime.CompilerServices.Unsafe, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Runtime.CompilerServices.Unsafe.6.0.0\lib\net461\System.Runtime.CompilerServices.Unsafe.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Runtime.Extensions, Version=4.1.1.1, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Runtime.Extensions.4.3.1\lib\net462\System.Runtime.Extensions.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Runtime.InteropServices, Version=4.1.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Runtime.InteropServices.4.3.0\lib\net463\System.Runtime.InteropServices.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Runtime.InteropServices.RuntimeInformation, Version=4.0.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Runtime.InteropServices.RuntimeInformation.4.3.0\lib\net45\System.Runtime.InteropServices.RuntimeInformation.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Security.Cryptography.Algorithms, Version=4.2.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Security.Cryptography.Algorithms.4.3.1\lib\net463\System.Security.Cryptography.Algorithms.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Security.Cryptography.Encoding, Version=4.0.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Security.Cryptography.Encoding.4.3.0\lib\net46\System.Security.Cryptography.Encoding.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Security.Cryptography.Primitives, Version=4.0.1.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Security.Cryptography.Primitives.4.3.0\lib\net46\System.Security.Cryptography.Primitives.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Security.Cryptography.X509Certificates, Version=4.1.1.2, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Security.Cryptography.X509Certificates.4.3.2\lib\net461\System.Security.Cryptography.X509Certificates.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Text.RegularExpressions, Version=4.1.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Text.RegularExpressions.4.3.1\lib\net463\System.Text.RegularExpressions.dll</HintPath> | ||||
|         </Reference> | ||||
|         <Reference Include="System.Xml.Linq" /> | ||||
|         <Reference Include="System.Data.DataSetExtensions" /> | ||||
|         <Reference Include="Microsoft.CSharp" /> | ||||
|         <Reference Include="System.Data" /> | ||||
|         <Reference Include="System.Deployment" /> | ||||
|         <Reference Include="System.Drawing" /> | ||||
|         <Reference Include="System.Windows.Forms" /> | ||||
|         <Reference Include="System.Xml" /> | ||||
|         <Reference Include="System.Xml.ReaderWriter, Version=4.1.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a, processorArchitecture=MSIL"> | ||||
|           <HintPath>packages\System.Xml.ReaderWriter.4.3.1\lib\net46\System.Xml.ReaderWriter.dll</HintPath> | ||||
|         </Reference> | ||||
|     </ItemGroup> | ||||
|     <ItemGroup> | ||||
|         <Compile Include="DownloadForm.cs"> | ||||
|           <SubType>Form</SubType> | ||||
|         </Compile> | ||||
|         <Compile Include="DownloadForm.Designer.cs"> | ||||
|           <DependentUpon>DownloadForm.cs</DependentUpon> | ||||
|         </Compile> | ||||
|         <Compile Include="Program.cs" /> | ||||
|         <Compile Include="Properties\AssemblyInfo.cs" /> | ||||
|         <Compile Include="Version.cs" /> | ||||
|         <EmbeddedResource Include="DownloadForm.resx"> | ||||
|           <DependentUpon>DownloadForm.cs</DependentUpon> | ||||
|         </EmbeddedResource> | ||||
|         <EmbeddedResource Include="Properties\Resources.resx"> | ||||
|             <Generator>ResXFileCodeGenerator</Generator> | ||||
|             <LastGenOutput>Resources.Designer.cs</LastGenOutput> | ||||
|             <SubType>Designer</SubType> | ||||
|         </EmbeddedResource> | ||||
|         <Compile Include="Properties\Resources.Designer.cs"> | ||||
|             <AutoGen>True</AutoGen> | ||||
|             <DependentUpon>Resources.resx</DependentUpon> | ||||
|         </Compile> | ||||
|         <None Include="..\..\public\favicon.ico"> | ||||
|           <Link>favicon.ico</Link> | ||||
|         </None> | ||||
|         <None Include="packages.config" /> | ||||
|         <None Include="Properties\Settings.settings"> | ||||
|             <Generator>SettingsSingleFileGenerator</Generator> | ||||
|             <LastGenOutput>Settings.Designer.cs</LastGenOutput> | ||||
|         </None> | ||||
|         <Compile Include="Properties\Settings.Designer.cs"> | ||||
|             <AutoGen>True</AutoGen> | ||||
|             <DependentUpon>Settings.settings</DependentUpon> | ||||
|             <DesignTimeSharedInput>True</DesignTimeSharedInput> | ||||
|         </Compile> | ||||
|     </ItemGroup> | ||||
|     <ItemGroup> | ||||
|         <None Include="App.config" /> | ||||
|     </ItemGroup> | ||||
|     <ItemGroup> | ||||
|       <Content Include=".gitignore" /> | ||||
|       <Content Include="app.manifest" /> | ||||
|     </ItemGroup> | ||||
|     <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" /> | ||||
|     <Target Name="EnsureNuGetPackageBuildImports" BeforeTargets="PrepareForBuild"> | ||||
|       <PropertyGroup> | ||||
|         <ErrorText>This project references NuGet package(s) that are missing on this computer. Enable NuGet Package Restore to download them.  For more information, see http://go.microsoft.com/fwlink/?LinkID=322105.The missing file is {0}.</ErrorText> | ||||
|       </PropertyGroup> | ||||
|       <Error Condition="!Exists('packages\Costura.Fody.5.7.0\build\Costura.Fody.props')" Text="$([System.String]::Format('$(ErrorText)', 'packages\Costura.Fody.5.7.0\build\Costura.Fody.props'))" /> | ||||
|       <Error Condition="!Exists('packages\Costura.Fody.5.7.0\build\Costura.Fody.targets')" Text="$([System.String]::Format('$(ErrorText)', 'packages\Costura.Fody.5.7.0\build\Costura.Fody.targets'))" /> | ||||
|       <Error Condition="!Exists('packages\Fody.6.6.4\build\Fody.targets')" Text="$([System.String]::Format('$(ErrorText)', 'packages\Fody.6.6.4\build\Fody.targets'))" /> | ||||
|       <Error Condition="!Exists('packages\NETStandard.Library.2.0.3\build\netstandard2.0\NETStandard.Library.targets')" Text="$([System.String]::Format('$(ErrorText)', 'packages\NETStandard.Library.2.0.3\build\netstandard2.0\NETStandard.Library.targets'))" /> | ||||
|     </Target> | ||||
|     <Import Project="packages\Costura.Fody.5.7.0\build\Costura.Fody.targets" Condition="Exists('packages\Costura.Fody.5.7.0\build\Costura.Fody.targets')" /> | ||||
|     <Import Project="packages\Fody.6.6.4\build\Fody.targets" Condition="Exists('packages\Fody.6.6.4\build\Fody.targets')" /> | ||||
|     <Import Project="packages\NETStandard.Library.2.0.3\build\netstandard2.0\NETStandard.Library.targets" Condition="Exists('packages\NETStandard.Library.2.0.3\build\netstandard2.0\NETStandard.Library.targets')" /> | ||||
| </Project> | ||||
							
								
								
									
										16
									
								
								extra/exe-builder/UptimeKuma.sln
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								extra/exe-builder/UptimeKuma.sln
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
|  | ||||
| Microsoft Visual Studio Solution File, Format Version 12.00 | ||||
| Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UptimeKuma", "UptimeKuma.csproj", "{2DB53988-1D93-4AC0-90C4-96ADEAAC5C04}" | ||||
| EndProject | ||||
| Global | ||||
| 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | ||||
| 		Debug|Any CPU = Debug|Any CPU | ||||
| 		Release|Any CPU = Release|Any CPU | ||||
| 	EndGlobalSection | ||||
| 	GlobalSection(ProjectConfigurationPlatforms) = postSolution | ||||
| 		{2DB53988-1D93-4AC0-90C4-96ADEAAC5C04}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||
| 		{2DB53988-1D93-4AC0-90C4-96ADEAAC5C04}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||
| 		{2DB53988-1D93-4AC0-90C4-96ADEAAC5C04}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||
| 		{2DB53988-1D93-4AC0-90C4-96ADEAAC5C04}.Release|Any CPU.Build.0 = Release|Any CPU | ||||
| 	EndGlobalSection | ||||
| EndGlobal | ||||
							
								
								
									
										3
									
								
								extra/exe-builder/UptimeKuma.sln.DotSettings.user
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								extra/exe-builder/UptimeKuma.sln.DotSettings.user
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| <wpf:ResourceDictionary xml:space="preserve" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:s="clr-namespace:System;assembly=mscorlib" xmlns:ss="urn:shemas-jetbrains-com:settings-storage-xaml" xmlns:wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation"> | ||||
| 	<s:Boolean x:Key="/Default/ResxEditorPersonal/CheckedGroups/=UptimeKuma_002FProperties_002FResources/@EntryIndexedValue">True</s:Boolean> | ||||
| 	<s:Boolean x:Key="/Default/ResxEditorPersonal/Initialized/@EntryValue">True</s:Boolean></wpf:ResourceDictionary> | ||||
							
								
								
									
										9
									
								
								extra/exe-builder/Version.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								extra/exe-builder/Version.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| namespace UptimeKuma { | ||||
|     public class Version { | ||||
|         public string latest { get; set; } | ||||
|         public string slow { get; set; } | ||||
|         public string beta { get; set; } | ||||
|         public string nodejs { get; set; } | ||||
|         public string exe { get; set; } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										28
									
								
								extra/exe-builder/app.manifest
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								extra/exe-builder/app.manifest
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="yes"?> | ||||
| <assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0" xmlns:asmv3="urn:schemas-microsoft-com:asm.v3"> | ||||
|     <asmv3:application> | ||||
|         <asmv3:windowsSettings> | ||||
|             <dpiAware xmlns="http://schemas.microsoft.com/SMI/2005/WindowsSettings">true</dpiAware> | ||||
|             <dpiAwareness xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">PerMonitorV2</dpiAwareness> | ||||
|         </asmv3:windowsSettings> | ||||
|     </asmv3:application> | ||||
|     <trustInfo xmlns="urn:schemas-microsoft-com:asm.v2"> | ||||
|         <security> | ||||
|             <requestedPrivileges xmlns="urn:schemas-microsoft-com:asm.v3"> | ||||
|                 <!-- UAC Manifest Options | ||||
|                      If you want to change the Windows User Account Control level replace the | ||||
|                      requestedExecutionLevel node with one of the following. | ||||
|  | ||||
|                 <requestedExecutionLevel  level="asInvoker" uiAccess="false" /> | ||||
|                 <requestedExecutionLevel  level="requireAdministrator" uiAccess="false" /> | ||||
|                 <requestedExecutionLevel  level="highestAvailable" uiAccess="false" /> | ||||
|  | ||||
|                     Specifying requestedExecutionLevel element will disable file and registry virtualization. | ||||
|                     Remove this element if your application requires this virtualization for backwards | ||||
|                     compatibility. | ||||
|                 --> | ||||
|                 <requestedExecutionLevel level="asInvoker" uiAccess="false" /> | ||||
|             </requestedPrivileges> | ||||
|         </security> | ||||
|     </trustInfo> | ||||
| </assembly> | ||||
							
								
								
									
										56
									
								
								extra/exe-builder/packages.config
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								extra/exe-builder/packages.config
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| <?xml version="1.0" encoding="utf-8"?> | ||||
| <packages> | ||||
|   <package id="Costura.Fody" version="5.7.0" targetFramework="net472" developmentDependency="true" /> | ||||
|   <package id="Fody" version="6.6.4" targetFramework="net472" developmentDependency="true" /> | ||||
|   <package id="Microsoft.NETCore.Platforms" version="7.0.0" targetFramework="net472" /> | ||||
|   <package id="Microsoft.Win32.Primitives" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="NETStandard.Library" version="2.0.3" targetFramework="net472" /> | ||||
|   <package id="Newtonsoft.Json" version="13.0.2" targetFramework="net472" /> | ||||
|   <package id="System.AppContext" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Console" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Diagnostics.DiagnosticSource" version="7.0.1" targetFramework="net472" /> | ||||
|   <package id="System.Net.Http" version="4.3.4" targetFramework="net472" /> | ||||
|   <package id="System.Runtime.Extensions" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Security.Cryptography.Algorithms" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Security.Cryptography.X509Certificates" version="4.3.2" targetFramework="net472" /> | ||||
|   <package id="System.Text.RegularExpressions" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Xml.ReaderWriter" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Memory" version="4.5.5" targetFramework="net472" /> | ||||
|   <package id="System.Net.Primitives" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Runtime" version="4.3.1" targetFramework="net472" /> | ||||
|   <package id="System.Buffers" version="4.5.1" targetFramework="net472" /> | ||||
|   <package id="System.Collections" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Collections.Concurrent" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Diagnostics.Debug" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Diagnostics.Tools" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Diagnostics.Tracing" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Globalization" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Globalization.Calendars" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.IO" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.IO.Compression" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.IO.Compression.ZipFile" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.IO.FileSystem" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.IO.FileSystem.Primitives" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Linq" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Linq.Expressions" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Net.Sockets" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Numerics.Vectors" version="4.5.0" targetFramework="net472" /> | ||||
|   <package id="System.ObjectModel" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Reflection" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Reflection.Extensions" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Reflection.Primitives" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Resources.ResourceManager" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Runtime.CompilerServices.Unsafe" version="6.0.0" targetFramework="net472" /> | ||||
|   <package id="System.Runtime.Handles" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Runtime.InteropServices" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Runtime.InteropServices.RuntimeInformation" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Runtime.Numerics" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Security.Cryptography.Encoding" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Security.Cryptography.Primitives" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Text.Encoding" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Text.Encoding.Extensions" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Threading" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Threading.Tasks" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Threading.Timer" version="4.3.0" targetFramework="net472" /> | ||||
|   <package id="System.Xml.XDocument" version="4.3.0" targetFramework="net472" /> | ||||
| </packages> | ||||
| @@ -1,4 +1,8 @@ | ||||
| /* | ||||
|  * ⚠️ ⚠️ ⚠️ ⚠️ Due to the weird issue in Portainer that the healthcheck script is still pointing to this script for unknown reason. | ||||
|  * IT CANNOT BE DROPPED, even though it looks like it is not used. | ||||
|  * See more: https://github.com/louislam/uptime-kuma/issues/2774#issuecomment-1429092359 | ||||
|  * | ||||
|  * ⚠️ 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. | ||||
|  */ | ||||
| @@ -19,17 +23,17 @@ if (sslKey && sslCert) { | ||||
|  | ||||
| // If host is omitted, the server will accept connections on the unspecified IPv6 address (::) when IPv6 is available and the unspecified IPv4 address (0.0.0.0) otherwise. | ||||
| // Dual-stack support for (::) | ||||
| let hostname = process.env.UPTIME_KUMA_SERVICE_HOST || process.env.UPTIME_KUMA_HOST || "::"; | ||||
| let hostname = process.env.UPTIME_KUMA_HOST; | ||||
|  | ||||
| // Also read HOST if not *BSD, as HOST is a system environment variable in FreeBSD | ||||
| if (!hostname && !FBSD) { | ||||
|     hostname = process.env.HOST; | ||||
| } | ||||
|  | ||||
| const port = parseInt(process.env.UPTIME_KUMA_SERVICE_PORT || process.env.UPTIME_KUMA_PORT || process.env.PORT || 3001); | ||||
| const port = parseInt(process.env.UPTIME_KUMA_PORT || process.env.PORT || 3001); | ||||
|  | ||||
| let options = { | ||||
|     host: hostname, | ||||
|     host: hostname || "127.0.0.1", | ||||
|     port: port, | ||||
|     timeout: 28 * 1000, | ||||
| }; | ||||
|   | ||||
							
								
								
									
										22
									
								
								extra/sort-contributors.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								extra/sort-contributors.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| const fs = require("fs"); | ||||
|  | ||||
| // Read the file from private/sort-contributors.txt | ||||
| const file = fs.readFileSync("private/sort-contributors.txt", "utf8"); | ||||
|  | ||||
| // Convert to an array of lines | ||||
| let lines = file.split("\n"); | ||||
|  | ||||
| // Remove empty lines | ||||
| lines = lines.filter((line) => line !== ""); | ||||
|  | ||||
| // Remove duplicates | ||||
| lines = [ ...new Set(lines) ]; | ||||
|  | ||||
| // Remove @weblate and @UptimeKumaBot | ||||
| lines = lines.filter((line) => line !== "@weblate" && line !== "@UptimeKumaBot" && line !== "@louislam"); | ||||
|  | ||||
| // Sort the lines | ||||
| lines = lines.sort(); | ||||
|  | ||||
| // Output the lines, concat with " " | ||||
| console.log(lines.join(" ")); | ||||
| @@ -26,7 +26,8 @@ if (! exists) { | ||||
|     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n"); | ||||
|  | ||||
|     // Also update package-lock.json | ||||
|     childProcess.spawnSync("npm", [ "install" ]); | ||||
|     const npm = /^win/.test(process.platform) ? "npm.cmd" : "npm"; | ||||
|     childProcess.spawnSync(npm, [ "install" ]); | ||||
|  | ||||
|     commit(newVersion); | ||||
|     tag(newVersion); | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| <html lang="en"> | ||||
| <head> | ||||
|     <meta charset="UTF-8" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0" /> | ||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0, viewport-fit=cover" /> | ||||
|     <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png"> | ||||
|     <link rel="icon" type="image/svg+xml" href="/icon.svg" /> | ||||
|     <link rel="manifest" href="/manifest.json" /> | ||||
|   | ||||
							
								
								
									
										20080
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										20080
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										44
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,13 +1,13 @@ | ||||
| { | ||||
|     "name": "uptime-kuma", | ||||
|     "version": "1.20.0-beta.0", | ||||
|     "version": "1.22.0", | ||||
|     "license": "MIT", | ||||
|     "repository": { | ||||
|         "type": "git", | ||||
|         "url": "https://github.com/louislam/uptime-kuma.git" | ||||
|     }, | ||||
|     "engines": { | ||||
|         "node": "14.* || >=16.*" | ||||
|         "node": "14.* || 16.* || 18.*" | ||||
|     }, | ||||
|     "scripts": { | ||||
|         "install-legacy": "npm install", | ||||
| @@ -38,7 +38,7 @@ | ||||
|         "build-docker-nightly-local": "docker build -f docker/dockerfile -t louislam/uptime-kuma:nightly2 --target nightly .", | ||||
|         "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.19.6 && npm ci --production && npm run download-dist", | ||||
|         "setup": "git checkout 1.22.0 && 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", | ||||
| @@ -60,19 +60,20 @@ | ||||
|         "cy:run:unit": "npx cypress run --browser chrome --headless --config-file ./config/cypress.frontend.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", | ||||
|         "deploy-demo-server": "node extra/deploy-demo-server.js", | ||||
|         "sort-contributors": "node extra/sort-contributors.js", | ||||
|         "quick-run-nightly": "docker run --rm --env NODE_ENV=development -p 3001:3001 louislam/uptime-kuma:nightly2", | ||||
|         "start-dev-container": "cd docker && docker-compose -f docker-compose-dev.yml up" | ||||
|     }, | ||||
|     "dependencies": { | ||||
|         "@grpc/grpc-js": "~1.7.3", | ||||
|         "@louislam/ping": "~0.4.2-mod.1", | ||||
|         "@louislam/sqlite3": "15.1.2", | ||||
|         "@louislam/ping": "~0.4.4-mod.0", | ||||
|         "@louislam/sqlite3": "15.1.6", | ||||
|         "args-parser": "~1.3.0", | ||||
|         "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.4.0", | ||||
|         "check-password-strength": "^2.0.5", | ||||
| @@ -81,13 +82,14 @@ | ||||
|         "command-exists": "~1.2.9", | ||||
|         "compare-versions": "~3.6.0", | ||||
|         "compression": "~1.7.4", | ||||
|         "croner": "~6.0.5", | ||||
|         "dayjs": "~1.11.5", | ||||
|         "dotenv": "~16.0.3", | ||||
|         "express": "~4.17.3", | ||||
|         "express-basic-auth": "~1.2.1", | ||||
|         "express-static-gzip": "~2.1.7", | ||||
|         "form-data": "~4.0.0", | ||||
|         "gamedig": "^4.0.5", | ||||
|         "gamedig": "~4.0.5", | ||||
|         "http-graceful-shutdown": "~3.1.7", | ||||
|         "http-proxy-agent": "~5.0.0", | ||||
|         "https-proxy-agent": "~5.0.1", | ||||
| @@ -97,10 +99,11 @@ | ||||
|         "jwt-decode": "~3.1.2", | ||||
|         "knex": "^2.4.2", | ||||
|         "limiter": "~2.1.0", | ||||
|         "mongodb": "~4.13.0", | ||||
|         "mongodb": "~4.14.0", | ||||
|         "mqtt": "~4.3.7", | ||||
|         "mssql": "~8.1.4", | ||||
|         "mysql2": "~2.3.3", | ||||
|         "nanoid": "~3.3.4", | ||||
|         "node-cloudflared-tunnel": "~1.0.9", | ||||
|         "node-radius-client": "~1.0.0", | ||||
|         "nodemailer": "~6.6.5", | ||||
| @@ -108,13 +111,15 @@ | ||||
|         "password-hash": "~1.2.2", | ||||
|         "pg": "~8.8.0", | ||||
|         "pg-connection-string": "~2.5.0", | ||||
|         "playwright-core": "~1.35.1", | ||||
|         "prom-client": "~13.2.0", | ||||
|         "prometheus-api-metrics": "~3.2.1", | ||||
|         "protobufjs": "~7.1.1", | ||||
|         "redbean-node": "~0.2.0", | ||||
|         "qs": "~6.10.4", | ||||
|         "redbean-node": "~0.3.0", | ||||
|         "redis": "~4.5.1", | ||||
|         "socket.io": "~4.5.3", | ||||
|         "socket.io-client": "~4.5.3", | ||||
|         "socket.io": "~4.6.1", | ||||
|         "socket.io-client": "~4.6.1", | ||||
|         "socks-proxy-agent": "6.1.1", | ||||
|         "tar": "~6.1.11", | ||||
|         "tcp-ping": "~0.1.1", | ||||
| @@ -137,10 +142,11 @@ | ||||
|         "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", | ||||
|         "chart.js": "~4.2.1", | ||||
|         "chartjs-adapter-dayjs-4": "~1.0.4", | ||||
|         "concurrently": "^7.1.0", | ||||
|         "core-js": "~3.26.1", | ||||
|         "cronstrue": "~2.24.0", | ||||
|         "cross-env": "~7.0.3", | ||||
|         "cypress": "^10.1.0", | ||||
|         "delay": "^5.0.0", | ||||
| @@ -151,6 +157,7 @@ | ||||
|         "favico.js": "~0.3.10", | ||||
|         "jest": "~27.2.5", | ||||
|         "marked": "~4.2.5", | ||||
|         "node-ssh": "~13.0.1", | ||||
|         "postcss-html": "~1.5.0", | ||||
|         "postcss-rtlcss": "~3.7.2", | ||||
|         "postcss-scss": "~4.0.4", | ||||
| @@ -158,16 +165,16 @@ | ||||
|         "qrcode": "~1.5.0", | ||||
|         "rollup-plugin-visualizer": "^5.6.0", | ||||
|         "sass": "~1.42.1", | ||||
|         "stylelint": "~14.7.1", | ||||
|         "stylelint": "~15.9.0", | ||||
|         "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": "~3.1.0", | ||||
|         "vite": "~3.2.7", | ||||
|         "vite-plugin-compression": "^0.5.1", | ||||
|         "vue": "next", | ||||
|         "vue-chart-3": "3.0.9", | ||||
|         "vue": "~3.2.47", | ||||
|         "vue-chartjs": "~5.2.0", | ||||
|         "vue-confirm-dialog": "~1.0.2", | ||||
|         "vue-contenteditable": "~3.0.4", | ||||
|         "vue-i18n": "~9.2.2", | ||||
| @@ -178,6 +185,7 @@ | ||||
|         "vue-router": "~4.0.14", | ||||
|         "vue-toastification": "~2.0.0-rc.5", | ||||
|         "vuedraggable": "~4.1.0", | ||||
|         "wait-on": "^6.0.1" | ||||
|         "wait-on": "^6.0.1", | ||||
|         "whatwg-url": "~12.0.1" | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										100
									
								
								server/auth.js
									
									
									
									
									
								
							
							
						
						
									
										100
									
								
								server/auth.js
									
									
									
									
									
								
							| @@ -2,7 +2,10 @@ const basicAuth = require("express-basic-auth"); | ||||
| const passwordHash = require("./password-hash"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { setting } = require("./util-server"); | ||||
| const { loginRateLimiter } = require("./rate-limiter"); | ||||
| const { log } = require("../src/util"); | ||||
| const { loginRateLimiter, apiRateLimiter } = require("./rate-limiter"); | ||||
| const { Settings } = require("./settings"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| /** | ||||
|  * Login to web app | ||||
| @@ -34,8 +37,36 @@ exports.login = async function (username, password) { | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Callback for myAuthorizer | ||||
|  * @callback myAuthorizerCB | ||||
|  * Validate a provided API key | ||||
|  * @param {string} key API key to verify | ||||
|  */ | ||||
| async function verifyAPIKey(key) { | ||||
|     if (typeof key !== "string") { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     // uk prefix + key ID is before _ | ||||
|     let index = key.substring(2, key.indexOf("_")); | ||||
|     let clear = key.substring(key.indexOf("_") + 1, key.length); | ||||
|  | ||||
|     let hash = await R.findOne("api_key", " id=? ", [ index ]); | ||||
|  | ||||
|     if (hash === null) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     let current = dayjs(); | ||||
|     let expiry = dayjs(hash.expires); | ||||
|     if (expiry.diff(current) < 0 || !hash.active) { | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     return hash && passwordHash.verify(clear, hash.key); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Callback for basic auth authorizers | ||||
|  * @callback authCallback | ||||
|  * @param {any} err Any error encountered | ||||
|  * @param {boolean} authorized Is the client authorized? | ||||
|  */ | ||||
| @@ -44,9 +75,35 @@ exports.login = async function (username, password) { | ||||
|  * Custom authorizer for express-basic-auth | ||||
|  * @param {string} username | ||||
|  * @param {string} password | ||||
|  * @param {myAuthorizerCB} callback | ||||
|  * @param {authCallback} callback | ||||
|  */ | ||||
| function myAuthorizer(username, password, callback) { | ||||
| function apiAuthorizer(username, password, callback) { | ||||
|     // API Rate Limit | ||||
|     apiRateLimiter.pass(null, 0).then((pass) => { | ||||
|         if (pass) { | ||||
|             verifyAPIKey(password).then((valid) => { | ||||
|                 if (!valid) { | ||||
|                     log.warn("api-auth", "Failed API auth attempt: invalid API Key"); | ||||
|                 } | ||||
|                 callback(null, valid); | ||||
|                 // Only allow a set number of api requests per minute | ||||
|                 // (currently set to 60) | ||||
|                 apiRateLimiter.removeTokens(1); | ||||
|             }); | ||||
|         } else { | ||||
|             log.warn("api-auth", "Failed API auth attempt: rate limit exceeded"); | ||||
|             callback(null, false); | ||||
|         } | ||||
|     }); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Custom authorizer for express-basic-auth | ||||
|  * @param {string} username | ||||
|  * @param {string} password | ||||
|  * @param {authCallback} callback | ||||
|  */ | ||||
| function userAuthorizer(username, password, callback) { | ||||
|     // Login Rate Limit | ||||
|     loginRateLimiter.pass(null, 0).then((pass) => { | ||||
|         if (pass) { | ||||
| @@ -54,10 +111,12 @@ function myAuthorizer(username, password, callback) { | ||||
|                 callback(null, user != null); | ||||
|  | ||||
|                 if (user == null) { | ||||
|                     log.warn("basic-auth", "Failed basic auth attempt: invalid username/password"); | ||||
|                     loginRateLimiter.removeTokens(1); | ||||
|                 } | ||||
|             }); | ||||
|         } else { | ||||
|             log.warn("basic-auth", "Failed basic auth attempt: rate limit exceeded"); | ||||
|             callback(null, false); | ||||
|         } | ||||
|     }); | ||||
| @@ -71,7 +130,7 @@ function myAuthorizer(username, password, callback) { | ||||
|  */ | ||||
| exports.basicAuth = async function (req, res, next) { | ||||
|     const middleware = basicAuth({ | ||||
|         authorizer: myAuthorizer, | ||||
|         authorizer: userAuthorizer, | ||||
|         authorizeAsync: true, | ||||
|         challenge: true, | ||||
|     }); | ||||
| @@ -84,3 +143,32 @@ exports.basicAuth = async function (req, res, next) { | ||||
|         next(); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Use use API Key if API keys enabled, else use basic auth | ||||
|  * @param {express.Request} req Express request object | ||||
|  * @param {express.Response} res Express response object | ||||
|  * @param {express.NextFunction} next | ||||
|  */ | ||||
| exports.apiAuth = async function (req, res, next) { | ||||
|     if (!await Settings.get("disableAuth")) { | ||||
|         let usingAPIKeys = await Settings.get("apiKeysEnabled"); | ||||
|         let middleware; | ||||
|         if (usingAPIKeys) { | ||||
|             middleware = basicAuth({ | ||||
|                 authorizer: apiAuthorizer, | ||||
|                 authorizeAsync: true, | ||||
|                 challenge: true, | ||||
|             }); | ||||
|         } else { | ||||
|             middleware = basicAuth({ | ||||
|                 authorizer: userAuthorizer, | ||||
|                 authorizeAsync: true, | ||||
|                 challenge: true, | ||||
|             }); | ||||
|         } | ||||
|         middleware(req, res, next); | ||||
|     } else { | ||||
|         next(); | ||||
|     } | ||||
| }; | ||||
|   | ||||
| @@ -113,6 +113,31 @@ async function sendProxyList(socket) { | ||||
|     return list; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Emit API key list to client | ||||
|  * @param {Socket} socket Socket.io socket instance | ||||
|  * @returns {Promise<void>} | ||||
|  */ | ||||
| async function sendAPIKeyList(socket) { | ||||
|     const timeLogger = new TimeLogger(); | ||||
|  | ||||
|     let result = []; | ||||
|     const list = await R.find( | ||||
|         "api_key", | ||||
|         "user_id=?", | ||||
|         [ socket.userID ], | ||||
|     ); | ||||
|  | ||||
|     for (let bean of list) { | ||||
|         result.push(bean.toPublicJSON()); | ||||
|     } | ||||
|  | ||||
|     io.to(socket.userID).emit("apiKeyList", result); | ||||
|     timeLogger.print("Sent API Key List"); | ||||
|  | ||||
|     return list; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Emits the version information to the client. | ||||
|  * @param {Socket} socket Socket.io socket instance | ||||
| @@ -157,6 +182,7 @@ module.exports = { | ||||
|     sendImportantHeartbeatList, | ||||
|     sendHeartbeatList, | ||||
|     sendProxyList, | ||||
|     sendAPIKeyList, | ||||
|     sendInfo, | ||||
|     sendDockerHostList | ||||
| }; | ||||
|   | ||||
| @@ -2,7 +2,6 @@ const fs = require("fs"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { setSetting, setting } = require("./util-server"); | ||||
| const { log, sleep } = require("../src/util"); | ||||
| const dayjs = require("dayjs"); | ||||
| const knex = require("knex"); | ||||
| const { PluginsManager } = require("./plugins-manager"); | ||||
| const path = require("path"); | ||||
| @@ -26,6 +25,8 @@ class Database { | ||||
|      */ | ||||
|     static uploadDir; | ||||
|  | ||||
|     static screenshotDir; | ||||
|  | ||||
|     static sqlitePath; | ||||
|  | ||||
|     /** | ||||
| @@ -33,11 +34,6 @@ class Database { | ||||
|      */ | ||||
|     static patched = false; | ||||
|  | ||||
|     /** | ||||
|      * For Backup only | ||||
|      */ | ||||
|     static backupPath = null; | ||||
|  | ||||
|     /** | ||||
|      * SQLite only | ||||
|      * Add patch filename in key | ||||
| @@ -75,6 +71,12 @@ class Database { | ||||
|         "patch-maintenance-table2.sql": true, | ||||
|         "patch-add-gamedig-monitor.sql": true, | ||||
|         "patch-add-google-analytics-status-page-tag.sql": true, | ||||
|         "patch-http-body-encoding.sql": true, | ||||
|         "patch-add-description-monitor.sql": true, | ||||
|         "patch-api-key-table.sql": true, | ||||
|         "patch-monitor-tls.sql": true, | ||||
|         "patch-maintenance-cron.sql": true, | ||||
|         "patch-add-parent-monitor.sql": true, | ||||
|     }; | ||||
|  | ||||
|     /** | ||||
| @@ -114,6 +116,12 @@ class Database { | ||||
|             fs.mkdirSync(Database.uploadDir, { recursive: true }); | ||||
|         } | ||||
|  | ||||
|         // Create screenshot dir | ||||
|         Database.screenshotDir = Database.dataDir + "screenshots/"; | ||||
|         if (! fs.existsSync(Database.screenshotDir)) { | ||||
|             fs.mkdirSync(Database.screenshotDir, { recursive: true }); | ||||
|         } | ||||
|  | ||||
|         log.info("db", `Data Dir: ${Database.dataDir}`); | ||||
|     } | ||||
|  | ||||
| @@ -269,12 +277,12 @@ class Database { | ||||
|             await R.exec("PRAGMA journal_mode = WAL"); | ||||
|         } | ||||
|         await R.exec("PRAGMA cache_size = -12000"); | ||||
|         await R.exec("PRAGMA auto_vacuum = FULL"); | ||||
|         await R.exec("PRAGMA auto_vacuum = INCREMENTAL"); | ||||
|  | ||||
|         // This ensures that an operating system crash or power failure will not corrupt the database. | ||||
|         // FULL synchronous is very safe, but it is also slower. | ||||
|         // Read more: https://sqlite.org/pragma.html#pragma_synchronous | ||||
|         await R.exec("PRAGMA synchronous = FULL"); | ||||
|         await R.exec("PRAGMA synchronous = NORMAL"); | ||||
|  | ||||
|         if (!noLog) { | ||||
|             log.info("db", "SQLite config:"); | ||||
| @@ -330,15 +338,7 @@ class Database { | ||||
|         } else { | ||||
|             log.info("db", "Database patch is needed"); | ||||
|  | ||||
|             try { | ||||
|                 this.backup(version); | ||||
|             } catch (e) { | ||||
|                 log.error("db", e); | ||||
|                 log.error("db", "Unable to create a backup before patching the database. Please make sure you have enough space and permission."); | ||||
|                 process.exit(1); | ||||
|             } | ||||
|  | ||||
|             // Try catch anything here, if gone wrong, restore the backup | ||||
|             // Try catch anything here | ||||
|             try { | ||||
|                 for (let i = version + 1; i <= this.latestVersion; i++) { | ||||
|                     const sqlFile = `./db/patch${i}.sql`; | ||||
| @@ -354,7 +354,6 @@ class Database { | ||||
|                 log.error("db", "Start Uptime-Kuma failed due to issue patching the database"); | ||||
|                 log.error("db", "Please submit a bug report if you still encounter the problem after restart: https://github.com/louislam/uptime-kuma/issues"); | ||||
|  | ||||
|                 this.restore(); | ||||
|                 process.exit(1); | ||||
|             } | ||||
|         } | ||||
| @@ -397,8 +396,6 @@ class Database { | ||||
|             log.error("db", "Start Uptime-Kuma failed due to issue patching the database"); | ||||
|             log.error("db", "Please submit the bug report if you still encounter the problem after restart: https://github.com/louislam/uptime-kuma/issues"); | ||||
|  | ||||
|             this.restore(); | ||||
|  | ||||
|             process.exit(1); | ||||
|         } | ||||
|  | ||||
| @@ -501,8 +498,6 @@ class Database { | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             this.backup(dayjs().format("YYYYMMDDHHmmss")); | ||||
|  | ||||
|             log.info("db", sqlFilename + " is patching"); | ||||
|             this.patched = true; | ||||
|             await this.importSQLFile("./db/" + sqlFilename); | ||||
| @@ -568,6 +563,9 @@ class Database { | ||||
|  | ||||
|         log.info("db", "Closing the database"); | ||||
|  | ||||
|         // Flush WAL to main database | ||||
|         await R.exec("PRAGMA wal_checkpoint(TRUNCATE)"); | ||||
|  | ||||
|         while (true) { | ||||
|             Database.noReject = true; | ||||
|             await R.close(); | ||||
| @@ -584,90 +582,6 @@ class Database { | ||||
|         process.removeListener("unhandledRejection", listener); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * One backup one time in this process. | ||||
|      * Reset this.backupPath if you want to backup again | ||||
|      * @param {string} version Version code of backup | ||||
|      */ | ||||
|     static backup(version) { | ||||
|         if (! this.backupPath) { | ||||
|             log.info("db", "Backing up the database"); | ||||
|             this.backupPath = this.dataDir + "kuma.db.bak" + version; | ||||
|             fs.copyFileSync(Database.sqlitePath, this.backupPath); | ||||
|  | ||||
|             const shmPath = Database.sqlitePath + "-shm"; | ||||
|             if (fs.existsSync(shmPath)) { | ||||
|                 this.backupShmPath = shmPath + ".bak" + version; | ||||
|                 fs.copyFileSync(shmPath, this.backupShmPath); | ||||
|             } | ||||
|  | ||||
|             const walPath = Database.sqlitePath + "-wal"; | ||||
|             if (fs.existsSync(walPath)) { | ||||
|                 this.backupWalPath = walPath + ".bak" + version; | ||||
|                 fs.copyFileSync(walPath, this.backupWalPath); | ||||
|             } | ||||
|  | ||||
|             // Double confirm if all files actually backup | ||||
|             if (!fs.existsSync(this.backupPath)) { | ||||
|                 throw new Error("Backup failed! " + this.backupPath); | ||||
|             } | ||||
|  | ||||
|             if (fs.existsSync(shmPath)) { | ||||
|                 if (!fs.existsSync(this.backupShmPath)) { | ||||
|                     throw new Error("Backup failed! " + this.backupShmPath); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (fs.existsSync(walPath)) { | ||||
|                 if (!fs.existsSync(this.backupWalPath)) { | ||||
|                     throw new Error("Backup failed! " + this.backupWalPath); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** Restore from most recent backup */ | ||||
|     static restore() { | ||||
|         if (this.backupPath) { | ||||
|             log.error("db", "Patching the database failed!!! Restoring the backup"); | ||||
|  | ||||
|             const shmPath = Database.sqlitePath + "-shm"; | ||||
|             const walPath = Database.sqlitePath + "-wal"; | ||||
|  | ||||
|             // Delete patch failed db | ||||
|             try { | ||||
|                 if (fs.existsSync(Database.sqlitePath)) { | ||||
|                     fs.unlinkSync(Database.sqlitePath); | ||||
|                 } | ||||
|  | ||||
|                 if (fs.existsSync(shmPath)) { | ||||
|                     fs.unlinkSync(shmPath); | ||||
|                 } | ||||
|  | ||||
|                 if (fs.existsSync(walPath)) { | ||||
|                     fs.unlinkSync(walPath); | ||||
|                 } | ||||
|             } catch (e) { | ||||
|                 log.error("db", "Restore failed; you may need to restore the backup manually"); | ||||
|                 process.exit(1); | ||||
|             } | ||||
|  | ||||
|             // Restore backup | ||||
|             fs.copyFileSync(this.backupPath, Database.sqlitePath); | ||||
|  | ||||
|             if (this.backupShmPath) { | ||||
|                 fs.copyFileSync(this.backupShmPath, shmPath); | ||||
|             } | ||||
|  | ||||
|             if (this.backupWalPath) { | ||||
|                 fs.copyFileSync(this.backupWalPath, walPath); | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
|             log.info("db", "Nothing to restore"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** Get the size of the database */ | ||||
|     static getSize() { | ||||
|         log.debug("db", "Database.getSize()"); | ||||
|   | ||||
| @@ -1,41 +1,44 @@ | ||||
| const path = require("path"); | ||||
| const Bree = require("bree"); | ||||
| const { SHARE_ENV } = require("worker_threads"); | ||||
| const { log } = require("../src/util"); | ||||
| let bree; | ||||
| const { UptimeKumaServer } = require("./uptime-kuma-server"); | ||||
| const { clearOldData } = require("./jobs/clear-old-data"); | ||||
| const Cron = require("croner"); | ||||
|  | ||||
| const jobs = [ | ||||
|     { | ||||
|         name: "clear-old-data", | ||||
|         interval: "at 03:14", | ||||
|         interval: "14 03 * * *", | ||||
|         jobFunc: clearOldData, | ||||
|         croner: null, | ||||
|     }, | ||||
| ]; | ||||
|  | ||||
| /** | ||||
|  * Initialize background jobs | ||||
|  * @param {Object} args Arguments to pass to workers | ||||
|  * @returns {Bree} | ||||
|  * @returns {Promise<void>} | ||||
|  */ | ||||
| const initBackgroundJobs = function (args) { | ||||
|     bree = new Bree({ | ||||
|         root: path.resolve("server", "jobs"), | ||||
|         jobs, | ||||
|         worker: { | ||||
|             env: SHARE_ENV, | ||||
|             workerData: args, | ||||
|         }, | ||||
|         workerMessageHandler: (message) => { | ||||
|             log.info("jobs", message); | ||||
|         } | ||||
|     }); | ||||
| const initBackgroundJobs = async function () { | ||||
|     const timezone = await UptimeKumaServer.getInstance().getTimezone(); | ||||
|  | ||||
|     for (const job of jobs) { | ||||
|         const cornerJob = new Cron( | ||||
|             job.interval, | ||||
|             { | ||||
|                 name: job.name, | ||||
|                 timezone, | ||||
|             }, | ||||
|             job.jobFunc, | ||||
|         ); | ||||
|         job.croner = cornerJob; | ||||
|     } | ||||
|  | ||||
|     bree.start(); | ||||
|     return bree; | ||||
| }; | ||||
|  | ||||
| /** Stop all background jobs if running */ | ||||
| const stopBackgroundJobs = function () { | ||||
|     if (bree) { | ||||
|         bree.stop(); | ||||
|     for (const job of jobs) { | ||||
|         if (job.croner) { | ||||
|             job.croner.stop(); | ||||
|             job.croner = null; | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,13 +1,16 @@ | ||||
| const { log, exit, connectDb } = require("./util-worker"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { log } = require("../../src/util"); | ||||
| const { setSetting, setting } = require("../util-server"); | ||||
| const Database = require("../database"); | ||||
|  | ||||
| const DEFAULT_KEEP_PERIOD = 180; | ||||
|  | ||||
| (async () => { | ||||
|     await connectDb(); | ||||
| /** | ||||
|  * Clears old data from the heartbeat table of the database. | ||||
|  * @return {Promise<void>} A promise that resolves when the data has been cleared. | ||||
|  */ | ||||
|  | ||||
| const clearOldData = async () => { | ||||
|     let period = await setting("keepDataPeriodDays"); | ||||
|  | ||||
|     // Set Default Period | ||||
| @@ -21,16 +24,16 @@ const DEFAULT_KEEP_PERIOD = 180; | ||||
|     try { | ||||
|         parsedPeriod = parseInt(period); | ||||
|     } catch (_) { | ||||
|         log("Failed to parse setting, resetting to default.."); | ||||
|         log.warn("clearOldData", "Failed to parse setting, resetting to default.."); | ||||
|         await setSetting("keepDataPeriodDays", DEFAULT_KEEP_PERIOD, "general"); | ||||
|         parsedPeriod = DEFAULT_KEEP_PERIOD; | ||||
|     } | ||||
|  | ||||
|     if (parsedPeriod < 1) { | ||||
|         log(`Data deletion has been disabled as period is less than 1. Period is ${parsedPeriod} days.`); | ||||
|         log.info("clearOldData", `Data deletion has been disabled as period is less than 1. Period is ${parsedPeriod} days.`); | ||||
|     } else { | ||||
|  | ||||
|         log(`Clearing Data older than ${parsedPeriod} days...`); | ||||
|         log.debug("clearOldData", `Clearing Data older than ${parsedPeriod} days...`); | ||||
|  | ||||
|         const sqlHourOffset = Database.sqlHourOffset(); | ||||
|  | ||||
| @@ -40,9 +43,11 @@ const DEFAULT_KEEP_PERIOD = 180; | ||||
|                 [ parsedPeriod * -24 ] | ||||
|             ); | ||||
|         } catch (e) { | ||||
|             log(`Failed to clear old data: ${e.message}`); | ||||
|             log.error("clearOldData", `Failed to clear old data: ${e.message}`); | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
|     exit(); | ||||
| })(); | ||||
| module.exports = { | ||||
|     clearOldData, | ||||
| }; | ||||
|   | ||||
| @@ -1,50 +0,0 @@ | ||||
| const { parentPort, workerData } = require("worker_threads"); | ||||
| const Database = require("../database"); | ||||
| const path = require("path"); | ||||
|  | ||||
| /** | ||||
|  * Send message to parent process for logging | ||||
|  * since worker_thread does not have access to stdout, this is used | ||||
|  * instead of console.log() | ||||
|  * @param {any} any The message to log | ||||
|  */ | ||||
| const log = function (any) { | ||||
|     if (parentPort) { | ||||
|         parentPort.postMessage(any); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Exit the worker process | ||||
|  * @param {number} error The status code to exit | ||||
|  */ | ||||
| const exit = function (error) { | ||||
|     if (error && error !== 0) { | ||||
|         process.exit(error); | ||||
|     } else { | ||||
|         if (parentPort) { | ||||
|             parentPort.postMessage("done"); | ||||
|         } else { | ||||
|             process.exit(0); | ||||
|         } | ||||
|     } | ||||
| }; | ||||
|  | ||||
| /** Connects to the database */ | ||||
| const connectDb = async function () { | ||||
|     const dbPath = path.join( | ||||
|         process.env.DATA_DIR || workerData["data-dir"] || "./data/" | ||||
|     ); | ||||
|  | ||||
|     Database.initDataDir({ | ||||
|         "data-dir": dbPath, | ||||
|     }); | ||||
|  | ||||
|     await Database.connect(); | ||||
| }; | ||||
|  | ||||
| module.exports = { | ||||
|     log, | ||||
|     exit, | ||||
|     connectDb, | ||||
| }; | ||||
							
								
								
									
										76
									
								
								server/model/api_key.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								server/model/api_key.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { R } = require("redbean-node"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| class APIKey extends BeanModel { | ||||
|     /** | ||||
|      * Get the current status of this API key | ||||
|      * @returns {string} active, inactive or expired | ||||
|      */ | ||||
|     getStatus() { | ||||
|         let current = dayjs(); | ||||
|         let expiry = dayjs(this.expires); | ||||
|         if (expiry.diff(current) < 0) { | ||||
|             return "expired"; | ||||
|         } | ||||
|  | ||||
|         return this.active ? "active" : "inactive"; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns an object that ready to parse to JSON | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     toJSON() { | ||||
|         return { | ||||
|             id: this.id, | ||||
|             key: this.key, | ||||
|             name: this.name, | ||||
|             userID: this.user_id, | ||||
|             createdDate: this.created_date, | ||||
|             active: this.active, | ||||
|             expires: this.expires, | ||||
|             status: this.getStatus(), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Returns an object that ready to parse to JSON with sensitive fields | ||||
|      * removed | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     toPublicJSON() { | ||||
|         return { | ||||
|             id: this.id, | ||||
|             name: this.name, | ||||
|             userID: this.user_id, | ||||
|             createdDate: this.created_date, | ||||
|             active: this.active, | ||||
|             expires: this.expires, | ||||
|             status: this.getStatus(), | ||||
|         }; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Create a new API Key and store it in the database | ||||
|      * @param {Object} key Object sent by client | ||||
|      * @param {int} userID ID of socket user | ||||
|      * @returns {Promise<bean>} | ||||
|      */ | ||||
|     static async save(key, userID) { | ||||
|         let bean; | ||||
|         bean = R.dispense("api_key"); | ||||
|  | ||||
|         bean.key = key.key; | ||||
|         bean.name = key.name; | ||||
|         bean.user_id = userID; | ||||
|         bean.active = key.active; | ||||
|         bean.expires = key.expires; | ||||
|  | ||||
|         await R.store(bean); | ||||
|  | ||||
|         return bean; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = APIKey; | ||||
| @@ -1,8 +1,10 @@ | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { parseTimeObject, parseTimeFromTimeObject, utcToLocal, localToUTC, log } = require("../../src/util"); | ||||
| const { timeObjectToUTC, timeObjectToLocal } = require("../util-server"); | ||||
| const { parseTimeObject, parseTimeFromTimeObject, log } = require("../../src/util"); | ||||
| const { R } = require("redbean-node"); | ||||
| const dayjs = require("dayjs"); | ||||
| const Cron = require("croner"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const apicache = require("../modules/apicache"); | ||||
|  | ||||
| class Maintenance extends BeanModel { | ||||
|  | ||||
| @@ -15,16 +17,19 @@ class Maintenance extends BeanModel { | ||||
|  | ||||
|         let dateRange = []; | ||||
|         if (this.start_date) { | ||||
|             dateRange.push(utcToLocal(this.start_date)); | ||||
|             if (this.end_date) { | ||||
|                 dateRange.push(utcToLocal(this.end_date)); | ||||
|             } | ||||
|             dateRange.push(this.start_date); | ||||
|         } else { | ||||
|             dateRange.push(null); | ||||
|         } | ||||
|  | ||||
|         if (this.end_date) { | ||||
|             dateRange.push(this.end_date); | ||||
|         } | ||||
|  | ||||
|         let timeRange = []; | ||||
|         let startTime = timeObjectToLocal(parseTimeObject(this.start_time)); | ||||
|         let startTime = parseTimeObject(this.start_time); | ||||
|         timeRange.push(startTime); | ||||
|         let endTime = timeObjectToLocal(parseTimeObject(this.end_time)); | ||||
|         let endTime = parseTimeObject(this.end_time); | ||||
|         timeRange.push(endTime); | ||||
|  | ||||
|         let obj = { | ||||
| @@ -39,12 +44,44 @@ class Maintenance extends BeanModel { | ||||
|             weekdays: (this.weekdays) ? JSON.parse(this.weekdays) : [], | ||||
|             daysOfMonth: (this.days_of_month) ? JSON.parse(this.days_of_month) : [], | ||||
|             timeslotList: [], | ||||
|             cron: this.cron, | ||||
|             duration: this.duration, | ||||
|             durationMinutes: parseInt(this.duration / 60), | ||||
|             timezone: await this.getTimezone(),         // Only valid timezone | ||||
|             timezoneOption: this.timezone,               // Mainly for dropdown menu, because there is a option "SAME_AS_SERVER" | ||||
|             timezoneOffset: await this.getTimezoneOffset(), | ||||
|             status: await this.getStatus(), | ||||
|         }; | ||||
|  | ||||
|         const timeslotList = await this.getTimeslotList(); | ||||
|         if (this.strategy === "manual") { | ||||
|             // Do nothing, no timeslots | ||||
|         } else if (this.strategy === "single") { | ||||
|             obj.timeslotList.push({ | ||||
|                 startDate: this.start_date, | ||||
|                 endDate: this.end_date, | ||||
|             }); | ||||
|         } else { | ||||
|             // Should be cron or recurring here | ||||
|             if (this.beanMeta.job) { | ||||
|                 let runningTimeslot = this.getRunningTimeslot(); | ||||
|  | ||||
|         for (let timeslot of timeslotList) { | ||||
|             obj.timeslotList.push(await timeslot.toPublicJSON()); | ||||
|                 if (runningTimeslot) { | ||||
|                     obj.timeslotList.push(runningTimeslot); | ||||
|                 } | ||||
|  | ||||
|                 let nextRunDate = this.beanMeta.job.nextRun(); | ||||
|                 if (nextRunDate) { | ||||
|                     let startDateDayjs = dayjs(nextRunDate); | ||||
|  | ||||
|                     let startDate = startDateDayjs.toISOString(); | ||||
|                     let endDate = startDateDayjs.add(this.duration, "second").toISOString(); | ||||
|  | ||||
|                     obj.timeslotList.push({ | ||||
|                         startDate, | ||||
|                         endDate, | ||||
|                     }); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (!Array.isArray(obj.weekdays)) { | ||||
| @@ -55,54 +92,9 @@ class Maintenance extends BeanModel { | ||||
|             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 | ||||
| @@ -126,7 +118,7 @@ class Maintenance extends BeanModel { | ||||
|  | ||||
|     /** | ||||
|      * Get a list of days in month that maintenance is active for | ||||
|      * @returns {number[]} Array of active days in month | ||||
|      * @returns {number[]|string[]} Array of active days in month | ||||
|      */ | ||||
|     getDayOfMonthList() { | ||||
|         return JSON.parse(this.days_of_month).sort(function (a, b) { | ||||
| @@ -135,26 +127,10 @@ class Maintenance extends BeanModel { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the start date and time for maintenance | ||||
|      * @returns {dayjs.Dayjs} Start date and time | ||||
|      */ | ||||
|     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"); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get the duraction of maintenance in seconds | ||||
|      * Get the duration of maintenance in seconds | ||||
|      * @returns {number} Duration of maintenance | ||||
|      */ | ||||
|     getDuration() { | ||||
|     calcDuration() { | ||||
|         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) { | ||||
| @@ -169,71 +145,270 @@ class Maintenance extends BeanModel { | ||||
|      * @param {Object} obj Data to fill bean with | ||||
|      * @returns {Bean} Filled bean | ||||
|      */ | ||||
|     static jsonToBean(bean, obj) { | ||||
|     static async 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.timezone = obj.timezoneOption; | ||||
|         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_date = obj.dateRange[0]; | ||||
|         } else { | ||||
|             bean.start_date = null; | ||||
|         } | ||||
|  | ||||
|         bean.start_time = parseTimeFromTimeObject(obj.timeRange[0]); | ||||
|         bean.end_time = parseTimeFromTimeObject(obj.timeRange[1]); | ||||
|         if (obj.dateRange[1]) { | ||||
|             bean.end_date = obj.dateRange[1]; | ||||
|         } else { | ||||
|             bean.end_date = null; | ||||
|         } | ||||
|  | ||||
|         bean.weekdays = JSON.stringify(obj.weekdays); | ||||
|         bean.days_of_month = JSON.stringify(obj.daysOfMonth); | ||||
|         if (bean.strategy === "cron") { | ||||
|             bean.duration = obj.durationMinutes * 60; | ||||
|             bean.cron = obj.cron; | ||||
|             this.validateCron(bean.cron); | ||||
|         } | ||||
|  | ||||
|         if (bean.strategy.startsWith("recurring-")) { | ||||
|             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); | ||||
|             await bean.generateCron(); | ||||
|             this.validateCron(bean.cron); | ||||
|         } | ||||
|         return bean; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SQL conditions for active maintenance | ||||
|      * @returns {string} | ||||
|      * Throw error if cron is invalid | ||||
|      * @param cron | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     static getActiveMaintenanceSQLCondition() { | ||||
|         return ` | ||||
|             ( | ||||
|                 (maintenance_timeslot.start_date <= CURRENT_TIMESTAMP | ||||
|                 AND maintenance_timeslot.end_date >= CURRENT_TIMESTAMP | ||||
|                 AND maintenance.active = 1) | ||||
|                 OR | ||||
|                 (maintenance.strategy = 'manual' AND active = 1) | ||||
|             ) | ||||
|         `; | ||||
|     static async validateCron(cron) { | ||||
|         let job = new Cron(cron, () => {}); | ||||
|         job.stop(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SQL conditions for active and future maintenance | ||||
|      * @returns {string} | ||||
|      * Run the cron | ||||
|      */ | ||||
|     static getActiveAndFutureMaintenanceSQLCondition() { | ||||
|         return ` | ||||
|             ( | ||||
|                 ((maintenance_timeslot.end_date >= CURRENT_TIMESTAMP | ||||
|                 AND maintenance.active = 1) | ||||
|                 OR | ||||
|                 (maintenance.strategy = 'manual' AND active = 1)) | ||||
|             ) | ||||
|         `; | ||||
|     async run(throwError = false) { | ||||
|         if (this.beanMeta.job) { | ||||
|             log.debug("maintenance", "Maintenance is already running, stop it first. id: " + this.id); | ||||
|             this.stop(); | ||||
|         } | ||||
|  | ||||
|         log.debug("maintenance", "Run maintenance id: " + this.id); | ||||
|  | ||||
|         // 1.21.2 migration | ||||
|         if (!this.cron) { | ||||
|             await this.generateCron(); | ||||
|             if (!this.timezone) { | ||||
|                 this.timezone = "UTC"; | ||||
|             } | ||||
|             if (this.cron) { | ||||
|                 await R.store(this); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (this.strategy === "manual") { | ||||
|             // Do nothing, because it is controlled by the user | ||||
|         } else if (this.strategy === "single") { | ||||
|             this.beanMeta.job = new Cron(this.start_date, { timezone: await this.getTimezone() }, () => { | ||||
|                 log.info("maintenance", "Maintenance id: " + this.id + " is under maintenance now"); | ||||
|                 UptimeKumaServer.getInstance().sendMaintenanceListByUserID(this.user_id); | ||||
|                 apicache.clear(); | ||||
|             }); | ||||
|         } else if (this.cron != null) { | ||||
|             // Here should be cron or recurring | ||||
|             try { | ||||
|                 this.beanMeta.status = "scheduled"; | ||||
|  | ||||
|                 let startEvent = (customDuration = 0) => { | ||||
|                     log.info("maintenance", "Maintenance id: " + this.id + " is under maintenance now"); | ||||
|  | ||||
|                     this.beanMeta.status = "under-maintenance"; | ||||
|                     clearTimeout(this.beanMeta.durationTimeout); | ||||
|  | ||||
|                     // Check if duration is still in the window. If not, use the duration from the current time to the end of the window | ||||
|                     let duration; | ||||
|  | ||||
|                     if (customDuration > 0) { | ||||
|                         duration = customDuration; | ||||
|                     } else if (this.end_date) { | ||||
|                         let d = dayjs(this.end_date).diff(dayjs(), "second"); | ||||
|                         if (d < this.duration) { | ||||
|                             duration = d * 1000; | ||||
|                         } | ||||
|                     } else { | ||||
|                         duration = this.duration * 1000; | ||||
|                     } | ||||
|  | ||||
|                     UptimeKumaServer.getInstance().sendMaintenanceListByUserID(this.user_id); | ||||
|  | ||||
|                     this.beanMeta.durationTimeout = setTimeout(() => { | ||||
|                         // End of maintenance for this timeslot | ||||
|                         this.beanMeta.status = "scheduled"; | ||||
|                         UptimeKumaServer.getInstance().sendMaintenanceListByUserID(this.user_id); | ||||
|                     }, duration); | ||||
|                 }; | ||||
|  | ||||
|                 // Create Cron | ||||
|                 this.beanMeta.job = new Cron(this.cron, { | ||||
|                     timezone: await this.getTimezone(), | ||||
|                 }, startEvent); | ||||
|  | ||||
|                 // Continue if the maintenance is still in the window | ||||
|                 let runningTimeslot = this.getRunningTimeslot(); | ||||
|                 let current = dayjs(); | ||||
|  | ||||
|                 if (runningTimeslot) { | ||||
|                     let duration = dayjs(runningTimeslot.endDate).diff(current, "second") * 1000; | ||||
|                     log.debug("maintenance", "Maintenance id: " + this.id + " Remaining duration: " + duration + "ms"); | ||||
|                     startEvent(duration); | ||||
|                 } | ||||
|  | ||||
|             } catch (e) { | ||||
|                 log.error("maintenance", "Error in maintenance id: " + this.id); | ||||
|                 log.error("maintenance", "Cron: " + this.cron); | ||||
|                 log.error("maintenance", e); | ||||
|  | ||||
|                 if (throwError) { | ||||
|                     throw e; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|         } else { | ||||
|             log.error("maintenance", "Maintenance id: " + this.id + " has no cron"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     getRunningTimeslot() { | ||||
|         let start = dayjs(this.beanMeta.job.nextRun(dayjs().add(-this.duration, "second").toDate())); | ||||
|         let end = start.add(this.duration, "second"); | ||||
|         let current = dayjs(); | ||||
|  | ||||
|         if (current.isAfter(start) && current.isBefore(end)) { | ||||
|             return { | ||||
|                 startDate: start.toISOString(), | ||||
|                 endDate: end.toISOString(), | ||||
|             }; | ||||
|         } else { | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     stop() { | ||||
|         if (this.beanMeta.job) { | ||||
|             this.beanMeta.job.stop(); | ||||
|             delete this.beanMeta.job; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async isUnderMaintenance() { | ||||
|         return (await this.getStatus()) === "under-maintenance"; | ||||
|     } | ||||
|  | ||||
|     async getTimezone() { | ||||
|         if (!this.timezone || this.timezone === "SAME_AS_SERVER") { | ||||
|             return await UptimeKumaServer.getInstance().getTimezone(); | ||||
|         } | ||||
|         return this.timezone; | ||||
|     } | ||||
|  | ||||
|     async getTimezoneOffset() { | ||||
|         return dayjs.tz(dayjs(), await this.getTimezone()).format("Z"); | ||||
|     } | ||||
|  | ||||
|     async getStatus() { | ||||
|         if (!this.active) { | ||||
|             return "inactive"; | ||||
|         } | ||||
|  | ||||
|         if (this.strategy === "manual") { | ||||
|             return "under-maintenance"; | ||||
|         } | ||||
|  | ||||
|         // Check if the maintenance is started | ||||
|         if (this.start_date && dayjs().isBefore(dayjs.tz(this.start_date, await this.getTimezone()))) { | ||||
|             return "scheduled"; | ||||
|         } | ||||
|  | ||||
|         // Check if the maintenance is ended | ||||
|         if (this.end_date && dayjs().isAfter(dayjs.tz(this.end_date, await this.getTimezone()))) { | ||||
|             return "ended"; | ||||
|         } | ||||
|  | ||||
|         if (this.strategy === "single") { | ||||
|             return "under-maintenance"; | ||||
|         } | ||||
|  | ||||
|         if (!this.beanMeta.status) { | ||||
|             return "unknown"; | ||||
|         } | ||||
|  | ||||
|         return this.beanMeta.status; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Generate Cron for recurring maintenance | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     async generateCron() { | ||||
|         log.info("maintenance", "Generate cron for maintenance id: " + this.id); | ||||
|  | ||||
|         if (this.strategy === "cron") { | ||||
|             // Do nothing for cron | ||||
|         } else if (!this.strategy.startsWith("recurring-")) { | ||||
|             this.cron = ""; | ||||
|         } else if (this.strategy === "recurring-interval") { | ||||
|             let array = this.start_time.split(":"); | ||||
|             let hour = parseInt(array[0]); | ||||
|             let minute = parseInt(array[1]); | ||||
|             this.cron = minute + " " + hour + " */" + this.interval_day + " * *"; | ||||
|             this.duration = this.calcDuration(); | ||||
|             log.debug("maintenance", "Cron: " + this.cron); | ||||
|             log.debug("maintenance", "Duration: " + this.duration); | ||||
|         } else if (this.strategy === "recurring-weekday") { | ||||
|             let list = this.getDayOfWeekList(); | ||||
|             let array = this.start_time.split(":"); | ||||
|             let hour = parseInt(array[0]); | ||||
|             let minute = parseInt(array[1]); | ||||
|             this.cron = minute + " " + hour + " * * " + list.join(","); | ||||
|             this.duration = this.calcDuration(); | ||||
|         } else if (this.strategy === "recurring-day-of-month") { | ||||
|             let list = this.getDayOfMonthList(); | ||||
|             let array = this.start_time.split(":"); | ||||
|             let hour = parseInt(array[0]); | ||||
|             let minute = parseInt(array[1]); | ||||
|  | ||||
|             let dayList = []; | ||||
|  | ||||
|             for (let day of list) { | ||||
|                 if (typeof day === "string" && day.startsWith("lastDay")) { | ||||
|                     if (day === "lastDay1") { | ||||
|                         dayList.push("L"); | ||||
|                     } | ||||
|                     // Unfortunately, lastDay2-4 is not supported by cron | ||||
|                 } else { | ||||
|                     dayList.push(day); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             // Remove duplicate | ||||
|             dayList = [ ...new Set(dayList) ]; | ||||
|  | ||||
|             this.cron = minute + " " + hour + " " + dayList.join(",") + " * *"; | ||||
|             this.duration = this.calcDuration(); | ||||
|         } | ||||
|  | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,198 +0,0 @@ | ||||
| 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 { | ||||
|  | ||||
|     /** | ||||
|      * Return an object that ready to parse to JSON for public | ||||
|      * Only show necessary data to public | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     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; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Return an object that ready to parse to JSON | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     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; | ||||
| @@ -2,7 +2,9 @@ const https = require("https"); | ||||
| const dayjs = require("dayjs"); | ||||
| const axios = require("axios"); | ||||
| const { Prometheus } = require("../prometheus"); | ||||
| const { log, UP, DOWN, PENDING, MAINTENANCE, flipStatus, TimeLogger, MAX_INTERVAL_SECOND, MIN_INTERVAL_SECOND } = require("../../src/util"); | ||||
| const { log, UP, DOWN, PENDING, MAINTENANCE, flipStatus, TimeLogger, MAX_INTERVAL_SECOND, MIN_INTERVAL_SECOND, | ||||
|     SQL_DATETIME_FORMAT | ||||
| } = require("../../src/util"); | ||||
| const { tcping, ping, dnsResolve, checkCertificate, checkStatusCode, getTotalClientInRoom, setting, mssqlQuery, postgresQuery, mysqlQuery, mqttAsync, setSetting, httpNtlm, radius, grpcQuery, | ||||
|     redisPingAsync, mongodbPing, | ||||
| } = require("../util-server"); | ||||
| @@ -16,9 +18,9 @@ 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"); | ||||
| const Gamedig = require("gamedig"); | ||||
| const jwt = require("jsonwebtoken"); | ||||
| const Database = require("../database"); | ||||
|  | ||||
| /** | ||||
| @@ -70,16 +72,27 @@ class Monitor extends BeanModel { | ||||
|  | ||||
|         const tags = await this.getTags(); | ||||
|  | ||||
|         let screenshot = null; | ||||
|  | ||||
|         if (this.type === "real-browser") { | ||||
|             screenshot = "/screenshots/" + jwt.sign(this.id, UptimeKumaServer.getInstance().jwtSecret) + ".png"; | ||||
|         } | ||||
|  | ||||
|         let data = { | ||||
|             id: this.id, | ||||
|             name: this.name, | ||||
|             description: this.description, | ||||
|             pathName: await this.getPathName(), | ||||
|             parent: this.parent, | ||||
|             childrenIDs: await Monitor.getAllChildrenIDs(this.id), | ||||
|             url: this.url, | ||||
|             method: this.method, | ||||
|             hostname: this.hostname, | ||||
|             port: this.port, | ||||
|             maxretries: this.maxretries, | ||||
|             weight: this.weight, | ||||
|             active: this.active, | ||||
|             active: await this.isActive(), | ||||
|             forceInactive: !await Monitor.isParentActive(this.id), | ||||
|             type: this.type, | ||||
|             interval: this.interval, | ||||
|             retryInterval: this.retryInterval, | ||||
| @@ -112,6 +125,8 @@ class Monitor extends BeanModel { | ||||
|             radiusCalledStationId: this.radiusCalledStationId, | ||||
|             radiusCallingStationId: this.radiusCallingStationId, | ||||
|             game: this.game, | ||||
|             httpBodyEncoding: this.httpBodyEncoding, | ||||
|             screenshot, | ||||
|         }; | ||||
|  | ||||
|         if (includeSensitiveData) { | ||||
| @@ -132,6 +147,9 @@ class Monitor extends BeanModel { | ||||
|                 mqttPassword: this.mqttPassword, | ||||
|                 authWorkstation: this.authWorkstation, | ||||
|                 authDomain: this.authDomain, | ||||
|                 tlsCa: this.tlsCa, | ||||
|                 tlsCert: this.tlsCert, | ||||
|                 tlsKey: this.tlsKey, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
| @@ -139,12 +157,22 @@ class Monitor extends BeanModel { | ||||
|         return data; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| 	 * Checks if the monitor is active based on itself and its parents | ||||
| 	 * @returns {Promise<Boolean>} | ||||
| 	 */ | ||||
|     async isActive() { | ||||
|         const parentActive = await Monitor.isParentActive(this.id); | ||||
|  | ||||
|         return this.active && parentActive; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get all tags applied to this monitor | ||||
|      * @returns {Promise<LooseObject<any>[]>} | ||||
|      */ | ||||
|     async getTags() { | ||||
|         return await R.getAll("SELECT mt.*, tag.name, tag.color FROM monitor_tag mt JOIN tag ON mt.tag_id = tag.id WHERE mt.monitor_id = ?", [ this.id ]); | ||||
|         return await R.getAll("SELECT mt.*, tag.name, tag.color FROM monitor_tag mt JOIN tag ON mt.tag_id = tag.id WHERE mt.monitor_id = ? ORDER BY tag.name", [ this.id ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -204,7 +232,7 @@ class Monitor extends BeanModel { | ||||
|         let previousBeat = null; | ||||
|         let retries = 0; | ||||
|  | ||||
|         let prometheus = new Prometheus(this); | ||||
|         this.prometheus = new Prometheus(this); | ||||
|  | ||||
|         const beat = async () => { | ||||
|  | ||||
| @@ -254,6 +282,36 @@ class Monitor extends BeanModel { | ||||
|                 if (await Monitor.isUnderMaintenance(this.id)) { | ||||
|                     bean.msg = "Monitor under maintenance"; | ||||
|                     bean.status = MAINTENANCE; | ||||
|                 } else if (this.type === "group") { | ||||
|                     const children = await Monitor.getChildren(this.id); | ||||
|  | ||||
|                     if (children.length > 0) { | ||||
|                         bean.status = UP; | ||||
|                         bean.msg = "All children up and running"; | ||||
|                         for (const child of children) { | ||||
|                             if (!child.active) { | ||||
|                                 // Ignore inactive childs | ||||
|                                 continue; | ||||
|                             } | ||||
|                             const lastBeat = await Monitor.getPreviousHeartbeat(child.id); | ||||
|  | ||||
|                             // Only change state if the monitor is in worse conditions then the ones before | ||||
|                             if (bean.status === UP && (lastBeat.status === PENDING || lastBeat.status === DOWN)) { | ||||
|                                 bean.status = lastBeat.status; | ||||
|                             } else if (bean.status === PENDING && lastBeat.status === DOWN) { | ||||
|                                 bean.status = lastBeat.status; | ||||
|                             } | ||||
|                         } | ||||
|  | ||||
|                         if (bean.status !== UP) { | ||||
|                             bean.msg = "Child inaccessible"; | ||||
|                         } | ||||
|                     } else { | ||||
|                         // Set status pending if group is empty | ||||
|                         bean.status = PENDING; | ||||
|                         bean.msg = "Group empty"; | ||||
|                     } | ||||
|  | ||||
|                 } else if (this.type === "http" || this.type === "keyword") { | ||||
|                     // Do not do any queries/high loading things before the "bean.ping" | ||||
|                     let startTime = dayjs().valueOf(); | ||||
| @@ -273,17 +331,34 @@ class Monitor extends BeanModel { | ||||
|  | ||||
|                     log.debug("monitor", `[${this.name}] Prepare Options for axios`); | ||||
|  | ||||
|                     let contentType = null; | ||||
|                     let bodyValue = null; | ||||
|  | ||||
|                     if (this.body && (typeof this.body === "string" && this.body.trim().length > 0)) { | ||||
|                         if (!this.httpBodyEncoding || this.httpBodyEncoding === "json") { | ||||
|                             try { | ||||
|                                 bodyValue = JSON.parse(this.body); | ||||
|                                 contentType = "application/json"; | ||||
|                             } catch (e) { | ||||
|                                 throw new Error("Your JSON body is invalid. " + e.message); | ||||
|                             } | ||||
|                         } else if (this.httpBodyEncoding === "xml") { | ||||
|                             bodyValue = this.body; | ||||
|                             contentType = "text/xml; charset=utf-8"; | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     // 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: { | ||||
|                             "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) : {}), | ||||
|                             ...(contentType ? { "Content-Type": contentType } : {}), | ||||
|                             ...(basicAuthHeader), | ||||
|                             ...(this.headers ? JSON.parse(this.headers) : {}) | ||||
|                         }, | ||||
|                         maxRedirects: this.maxredirects, | ||||
|                         validateStatus: (status) => { | ||||
| @@ -291,6 +366,10 @@ class Monitor extends BeanModel { | ||||
|                         }, | ||||
|                     }; | ||||
|  | ||||
|                     if (bodyValue) { | ||||
|                         options.data = bodyValue; | ||||
|                     } | ||||
|  | ||||
|                     if (this.proxy_id) { | ||||
|                         const proxy = await R.load("proxy", this.proxy_id); | ||||
|  | ||||
| @@ -309,6 +388,18 @@ class Monitor extends BeanModel { | ||||
|                         options.httpsAgent = new https.Agent(httpsAgentOptions); | ||||
|                     } | ||||
|  | ||||
|                     if (this.auth_method === "mtls") { | ||||
|                         if (this.tlsCert !== null && this.tlsCert !== "") { | ||||
|                             options.httpsAgent.options.cert = Buffer.from(this.tlsCert); | ||||
|                         } | ||||
|                         if (this.tlsCa !== null && this.tlsCa !== "") { | ||||
|                             options.httpsAgent.options.ca = Buffer.from(this.tlsCa); | ||||
|                         } | ||||
|                         if (this.tlsKey !== null && this.tlsKey !== "") { | ||||
|                             options.httpsAgent.options.key = Buffer.from(this.tlsKey); | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     log.debug("monitor", `[${this.name}] Axios Options: ${JSON.stringify(options)}`); | ||||
|                     log.debug("monitor", `[${this.name}] Axios Request`); | ||||
|  | ||||
| @@ -327,8 +418,8 @@ class Monitor extends BeanModel { | ||||
|                             tlsInfo = await this.updateTlsInfo(tlsInfoObject); | ||||
|  | ||||
|                             if (!this.getIgnoreTls() && this.isEnabledExpiryNotification()) { | ||||
|                                 log.debug("monitor", `[${this.name}] call sendCertNotification`); | ||||
|                                 await this.sendCertNotification(tlsInfoObject); | ||||
|                                 log.debug("monitor", `[${this.name}] call checkCertExpiryNotifications`); | ||||
|                                 await this.checkCertExpiryNotifications(tlsInfoObject); | ||||
|                             } | ||||
|  | ||||
|                         } catch (e) { | ||||
| @@ -362,7 +453,7 @@ class Monitor extends BeanModel { | ||||
|                             bean.msg += ", keyword is found"; | ||||
|                             bean.status = UP; | ||||
|                         } else { | ||||
|                             data = data.replace(/<[^>]*>?|[\n\r]|\s+/gm, " "); | ||||
|                             data = data.replace(/<[^>]*>?|[\n\r]|\s+/gm, " ").trim(); | ||||
|                             if (data.length > 50) { | ||||
|                                 data = data.substring(0, 47) + "..."; | ||||
|                             } | ||||
| @@ -600,9 +691,7 @@ class Monitor extends BeanModel { | ||||
|                 } else if (this.type === "mysql") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     await mysqlQuery(this.databaseConnectionString, this.databaseQuery); | ||||
|  | ||||
|                     bean.msg = ""; | ||||
|                     bean.msg = await mysqlQuery(this.databaseConnectionString, this.databaseQuery); | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "mongodb") { | ||||
| @@ -660,7 +749,7 @@ class Monitor extends BeanModel { | ||||
|                 } else if (this.type in UptimeKumaServer.monitorTypeList) { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|                     const monitorType = UptimeKumaServer.monitorTypeList[this.type]; | ||||
|                     await monitorType.check(this, bean); | ||||
|                     await monitorType.check(this, bean, UptimeKumaServer.getInstance()); | ||||
|                     if (!bean.ping) { | ||||
|                         bean.ping = dayjs().valueOf() - startTime; | ||||
|                     } | ||||
| @@ -756,7 +845,7 @@ class Monitor extends BeanModel { | ||||
|             await R.store(bean); | ||||
|  | ||||
|             log.debug("monitor", `[${this.name}] prometheus.update`); | ||||
|             prometheus.update(bean, tlsInfo); | ||||
|             this.prometheus?.update(bean, tlsInfo); | ||||
|  | ||||
|             previousBeat = bean; | ||||
|  | ||||
| @@ -814,7 +903,6 @@ class Monitor extends BeanModel { | ||||
|                     domain: this.authDomain, | ||||
|                     workstation: this.authWorkstation ? this.authWorkstation : undefined | ||||
|                 }); | ||||
|  | ||||
|             } else { | ||||
|                 res = await axios.request(options); | ||||
|             } | ||||
| @@ -841,15 +929,15 @@ class Monitor extends BeanModel { | ||||
|         clearTimeout(this.heartbeatInterval); | ||||
|         this.isStop = true; | ||||
|  | ||||
|         this.prometheus().remove(); | ||||
|         this.prometheus?.remove(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a new prometheus instance | ||||
|      * @returns {Prometheus} | ||||
|      * Get prometheus instance | ||||
|      * @returns {Prometheus|undefined} | ||||
|      */ | ||||
|     prometheus() { | ||||
|         return new Prometheus(this); | ||||
|     getPrometheus() { | ||||
|         return this.prometheus; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -1144,12 +1232,18 @@ class Monitor extends BeanModel { | ||||
|  | ||||
|             for (let notification of notificationList) { | ||||
|                 try { | ||||
|                     // Prevent if the msg is undefined, notifications such as Discord cannot send out. | ||||
|                     const heartbeatJSON = bean.toJSON(); | ||||
|  | ||||
|                     // Prevent if the msg is undefined, notifications such as Discord cannot send out. | ||||
|                     if (!heartbeatJSON["msg"]) { | ||||
|                         heartbeatJSON["msg"] = "N/A"; | ||||
|                     } | ||||
|  | ||||
|                     // Also provide the time in server timezone | ||||
|                     heartbeatJSON["timezone"] = await UptimeKumaServer.getInstance().getTimezone(); | ||||
|                     heartbeatJSON["timezoneOffset"] = UptimeKumaServer.getInstance().getTimezoneOffset(); | ||||
|                     heartbeatJSON["localDateTime"] = dayjs.utc(heartbeatJSON["time"]).tz(heartbeatJSON["timezone"]).format(SQL_DATETIME_FORMAT); | ||||
|  | ||||
|                     await Notification.send(JSON.parse(notification.config), msg, await monitor.toJSON(false), heartbeatJSON); | ||||
|                 } catch (e) { | ||||
|                     log.error("monitor", "Cannot send notification to " + notification.name); | ||||
| @@ -1172,13 +1266,19 @@ class Monitor extends BeanModel { | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Send notification about a certificate | ||||
|      * checks certificate chain for expiring certificates | ||||
|      * @param {Object} tlsInfoObject Information about certificate | ||||
|      */ | ||||
|     async sendCertNotification(tlsInfoObject) { | ||||
|     async checkCertExpiryNotifications(tlsInfoObject) { | ||||
|         if (tlsInfoObject && tlsInfoObject.certInfo && tlsInfoObject.certInfo.daysRemaining) { | ||||
|             const notificationList = await Monitor.getNotificationList(this); | ||||
|  | ||||
|             if (! notificationList.length > 0) { | ||||
|                 // fail fast. If no notification is set, all the following checks can be skipped. | ||||
|                 log.debug("monitor", "No notification, no need to send cert notification"); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             let notifyDays = await setting("tlsExpiryNotifyDays"); | ||||
|             if (notifyDays == null || !Array.isArray(notifyDays)) { | ||||
|                 // Reset Default | ||||
| @@ -1186,10 +1286,19 @@ class Monitor extends BeanModel { | ||||
|                 notifyDays = [ 7, 14, 21 ]; | ||||
|             } | ||||
|  | ||||
|             if (notifyDays != null && Array.isArray(notifyDays)) { | ||||
|                 for (const day of notifyDays) { | ||||
|                     log.debug("monitor", "call sendCertNotificationByTargetDays", day); | ||||
|                     await this.sendCertNotificationByTargetDays(tlsInfoObject.certInfo.daysRemaining, day, notificationList); | ||||
|             if (Array.isArray(notifyDays)) { | ||||
|                 for (const targetDays of notifyDays) { | ||||
|                     let certInfo = tlsInfoObject.certInfo; | ||||
|                     while (certInfo) { | ||||
|                         let subjectCN = certInfo.subject["CN"]; | ||||
|                         if (certInfo.daysRemaining > targetDays) { | ||||
|                             log.debug("monitor", `No need to send cert notification for ${certInfo.certType} certificate "${subjectCN}" (${certInfo.daysRemaining} days valid) on ${targetDays} deadline.`); | ||||
|                         } else { | ||||
|                             log.debug("monitor", `call sendCertNotificationByTargetDays for ${targetDays} deadline on certificate ${subjectCN}.`); | ||||
|                             await this.sendCertNotificationByTargetDays(subjectCN, certInfo.certType, certInfo.daysRemaining, targetDays, notificationList); | ||||
|                         } | ||||
|                         certInfo = certInfo.issuerCertificate; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1198,55 +1307,47 @@ class Monitor extends BeanModel { | ||||
|     /** | ||||
|      * Send a certificate notification when certificate expires in less | ||||
|      * than target days | ||||
|      * @param {number} daysRemaining Number of days remaining on certifcate | ||||
|      * @param {string} certCN  Common Name attribute from the certificate subject | ||||
|      * @param {string} certType  certificate type | ||||
|      * @param {number} daysRemaining Number of days remaining on certificate | ||||
|      * @param {number} targetDays Number of days to alert after | ||||
|      * @param {LooseObject<any>[]} notificationList List of notification providers | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     async sendCertNotificationByTargetDays(daysRemaining, targetDays, notificationList) { | ||||
|     async sendCertNotificationByTargetDays(certCN, certType, daysRemaining, targetDays, notificationList) { | ||||
|  | ||||
|         if (daysRemaining > targetDays) { | ||||
|             log.debug("monitor", `No need to send cert notification. ${daysRemaining} > ${targetDays}`); | ||||
|         let row = await R.getRow("SELECT * FROM notification_sent_history WHERE type = ? AND monitor_id = ? AND days <= ?", [ | ||||
|             "certificate", | ||||
|             this.id, | ||||
|             targetDays, | ||||
|         ]); | ||||
|  | ||||
|         // Sent already, no need to send again | ||||
|         if (row) { | ||||
|             log.debug("monitor", "Sent already, no need to send again"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         if (notificationList.length > 0) { | ||||
|         let sent = false; | ||||
|         log.debug("monitor", "Send certificate notification"); | ||||
|  | ||||
|             let row = await R.getRow("SELECT * FROM notification_sent_history WHERE type = ? AND monitor_id = ? AND days = ?", [ | ||||
|         for (let notification of notificationList) { | ||||
|             try { | ||||
|                 log.debug("monitor", "Sending to " + notification.name); | ||||
|                 await Notification.send(JSON.parse(notification.config), `[${this.name}][${this.url}] ${certType} certificate ${certCN} will be expired in ${daysRemaining} days`); | ||||
|                 sent = true; | ||||
|             } catch (e) { | ||||
|                 log.error("monitor", "Cannot send cert notification to " + notification.name); | ||||
|                 log.error("monitor", e); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if (sent) { | ||||
|             await R.exec("INSERT INTO notification_sent_history (type, monitor_id, days) VALUES(?, ?, ?)", [ | ||||
|                 "certificate", | ||||
|                 this.id, | ||||
|                 targetDays, | ||||
|             ]); | ||||
|  | ||||
|             // Sent already, no need to send again | ||||
|             if (row) { | ||||
|                 log.debug("monitor", "Sent already, no need to send again"); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             let sent = false; | ||||
|             log.debug("monitor", "Send certificate notification"); | ||||
|  | ||||
|             for (let notification of notificationList) { | ||||
|                 try { | ||||
|                     log.debug("monitor", "Sending to " + notification.name); | ||||
|                     await Notification.send(JSON.parse(notification.config), `[${this.name}][${this.url}] Certificate will be expired in ${daysRemaining} days`); | ||||
|                     sent = true; | ||||
|                 } catch (e) { | ||||
|                     log.error("monitor", "Cannot send cert notification to " + notification.name); | ||||
|                     log.error("monitor", e); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (sent) { | ||||
|                 await R.exec("INSERT INTO notification_sent_history (type, monitor_id, days) VALUES(?, ?, ?)", [ | ||||
|                     "certificate", | ||||
|                     this.id, | ||||
|                     targetDays, | ||||
|                 ]); | ||||
|             } | ||||
|         } else { | ||||
|             log.debug("monitor", "No notification, no need to send cert notification"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -1270,18 +1371,24 @@ class Monitor extends BeanModel { | ||||
|      * @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; | ||||
|         const maintenanceIDList = await R.getCol(` | ||||
|             SELECT maintenance_id FROM monitor_maintenance | ||||
|             WHERE monitor_id = ? | ||||
|         `, [ monitorID ]); | ||||
|  | ||||
|         for (const maintenanceID of maintenanceIDList) { | ||||
|             const maintenance = await UptimeKumaServer.getInstance().getMaintenance(maintenanceID); | ||||
|             if (maintenance && await maintenance.isUnderMaintenance()) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         const parent = await Monitor.getParent(monitorID); | ||||
|         if (parent != null) { | ||||
|             return await Monitor.isUnderMaintenance(parent.id); | ||||
|         } | ||||
|  | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     /** Make sure monitor interval is between bounds */ | ||||
| @@ -1293,6 +1400,105 @@ class Monitor extends BeanModel { | ||||
|             throw new Error(`Interval cannot be less than ${MIN_INTERVAL_SECOND} seconds`); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets Parent of the monitor | ||||
|      * @param {number} monitorID ID of monitor to get | ||||
|      * @returns {Promise<LooseObject<any>>} | ||||
|      */ | ||||
|     static async getParent(monitorID) { | ||||
|         return await R.getRow(` | ||||
|             SELECT parent.* FROM monitor parent | ||||
|     		LEFT JOIN monitor child | ||||
|     			ON child.parent = parent.id | ||||
|             WHERE child.id = ? | ||||
|         `, [ | ||||
|             monitorID, | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets all Children of the monitor | ||||
|      * @param {number} monitorID ID of monitor to get | ||||
|      * @returns {Promise<LooseObject<any>>} | ||||
|      */ | ||||
|     static async getChildren(monitorID) { | ||||
|         return await R.getAll(` | ||||
|             SELECT * FROM monitor | ||||
|             WHERE parent = ? | ||||
|         `, [ | ||||
|             monitorID, | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets Full Path-Name (Groups and Name) | ||||
|      * @returns {Promise<String>} | ||||
|      */ | ||||
|     async getPathName() { | ||||
|         let path = this.name; | ||||
|  | ||||
|         if (this.parent === null) { | ||||
|             return path; | ||||
|         } | ||||
|  | ||||
|         let parent = await Monitor.getParent(this.id); | ||||
|         while (parent !== null) { | ||||
|             path = `${parent.name} / ${path}`; | ||||
|             parent = await Monitor.getParent(parent.id); | ||||
|         } | ||||
|  | ||||
|         return path; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Gets recursive all child ids | ||||
| 	 * @param {number} monitorID ID of the monitor to get | ||||
|      * @returns {Promise<Array>} | ||||
|      */ | ||||
|     static async getAllChildrenIDs(monitorID) { | ||||
|         const childs = await Monitor.getChildren(monitorID); | ||||
|  | ||||
|         if (childs === null) { | ||||
|             return []; | ||||
|         } | ||||
|  | ||||
|         let childrenIDs = []; | ||||
|  | ||||
|         for (const child of childs) { | ||||
|             childrenIDs.push(child.id); | ||||
|             childrenIDs = childrenIDs.concat(await Monitor.getAllChildrenIDs(child.id)); | ||||
|         } | ||||
|  | ||||
|         return childrenIDs; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Unlinks all children of the the group monitor | ||||
|      * @param {number} groupID ID of group to remove children of | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     static async unlinkAllChildren(groupID) { | ||||
|         return await R.exec("UPDATE `monitor` SET parent = ? WHERE parent = ? ", [ | ||||
|             null, groupID | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| 	 * Checks recursive if parent (ancestors) are active | ||||
| 	 * @param {number} monitorID ID of the monitor to get | ||||
| 	 * @returns {Promise<Boolean>} | ||||
| 	 */ | ||||
|     static async isParentActive(monitorID) { | ||||
|         const parent = await Monitor.getParent(monitorID); | ||||
|  | ||||
|         if (parent === null) { | ||||
|             return true; | ||||
|         } | ||||
|  | ||||
|         const parentActive = await Monitor.isParentActive(parent.id); | ||||
|         return parent.active && parentActive; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Monitor; | ||||
|   | ||||
| @@ -3,7 +3,6 @@ const { R } = require("redbean-node"); | ||||
| const cheerio = require("cheerio"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const jsesc = require("jsesc"); | ||||
| const Maintenance = require("./maintenance"); | ||||
| const googleAnalytics = require("../google-analytics"); | ||||
|  | ||||
| class StatusPage extends BeanModel { | ||||
| @@ -290,21 +289,17 @@ class StatusPage extends BeanModel { | ||||
|         try { | ||||
|             const publicMaintenanceList = []; | ||||
|  | ||||
|             let activeCondition = Maintenance.getActiveMaintenanceSQLCondition(); | ||||
|             let maintenanceBeanList = R.convertToBeans("maintenance", await R.getAll(` | ||||
|                 SELECT DISTINCT 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 ])); | ||||
|             let maintenanceIDList = await R.getCol(` | ||||
|                 SELECT DISTINCT maintenance_id | ||||
|                 FROM maintenance_status_page | ||||
|                 WHERE status_page_id = ? | ||||
|             `, [ statusPageId ]); | ||||
|  | ||||
|             for (const bean of maintenanceBeanList) { | ||||
|                 publicMaintenanceList.push(await bean.toPublicJSON()); | ||||
|             for (const maintenanceID of maintenanceIDList) { | ||||
|                 let maintenance = UptimeKumaServer.getInstance().getMaintenance(maintenanceID); | ||||
|                 if (maintenance && await maintenance.isUnderMaintenance()) { | ||||
|                     publicMaintenanceList.push(await maintenance.toPublicJSON()); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             return publicMaintenanceList; | ||||
|   | ||||
| @@ -6,9 +6,10 @@ class MonitorType { | ||||
|      * | ||||
|      * @param {Monitor} monitor | ||||
|      * @param {Heartbeat} heartbeat | ||||
|      * @param {UptimeKumaServer} server | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     async check(monitor, heartbeat) { | ||||
|     async check(monitor, heartbeat, server) { | ||||
|         throw new Error("You need to override check()"); | ||||
|     } | ||||
|  | ||||
|   | ||||
							
								
								
									
										164
									
								
								server/monitor-types/real-browser-monitor-type.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										164
									
								
								server/monitor-types/real-browser-monitor-type.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,164 @@ | ||||
| const { MonitorType } = require("./monitor-type"); | ||||
| const { chromium, Browser } = require("playwright-core"); | ||||
| const { UP, log } = require("../../src/util"); | ||||
| const { Settings } = require("../settings"); | ||||
| const commandExistsSync = require("command-exists").sync; | ||||
| const childProcess = require("child_process"); | ||||
| const path = require("path"); | ||||
| const Database = require("../database"); | ||||
| const jwt = require("jsonwebtoken"); | ||||
|  | ||||
| /** | ||||
|  * | ||||
|  * @type {Browser} | ||||
|  */ | ||||
| let browser = null; | ||||
|  | ||||
| async function getBrowser() { | ||||
|     if (!browser) { | ||||
|         let executablePath = await Settings.get("chromeExecutable"); | ||||
|  | ||||
|         executablePath = await prepareChromeExecutable(executablePath); | ||||
|  | ||||
|         browser = await chromium.launch({ | ||||
|             //headless: false, | ||||
|             executablePath, | ||||
|         }); | ||||
|     } | ||||
|     return browser; | ||||
| } | ||||
|  | ||||
| async function prepareChromeExecutable(executablePath) { | ||||
|     // Special code for using the playwright_chromium | ||||
|     if (typeof executablePath === "string" && executablePath.toLocaleLowerCase() === "#playwright_chromium") { | ||||
|         executablePath = undefined; | ||||
|     } else if (!executablePath) { | ||||
|         if (process.env.UPTIME_KUMA_IS_CONTAINER) { | ||||
|             executablePath = "/usr/bin/chromium"; | ||||
|  | ||||
|             // Install chromium in container via apt install | ||||
|             if ( !commandExistsSync(executablePath)) { | ||||
|                 await new Promise((resolve, reject) => { | ||||
|                     log.info("Chromium", "Installing Chromium..."); | ||||
|                     let child = childProcess.exec("apt update && apt --yes --no-install-recommends install chromium fonts-indic fonts-noto fonts-noto-cjk"); | ||||
|  | ||||
|                     // On exit | ||||
|                     child.on("exit", (code) => { | ||||
|                         log.info("Chromium", "apt install chromium exited with code " + code); | ||||
|  | ||||
|                         if (code === 0) { | ||||
|                             log.info("Chromium", "Installed Chromium"); | ||||
|                             let version = childProcess.execSync(executablePath + " --version").toString("utf8"); | ||||
|                             log.info("Chromium", "Chromium version: " + version); | ||||
|                             resolve(); | ||||
|                         } else if (code === 100) { | ||||
|                             reject(new Error("Installing Chromium, please wait...")); | ||||
|                         } else { | ||||
|                             reject(new Error("apt install chromium failed with code " + code)); | ||||
|                         } | ||||
|                     }); | ||||
|                 }); | ||||
|             } | ||||
|  | ||||
|         } else if (process.platform === "win32") { | ||||
|             executablePath = findChrome([ | ||||
|                 "C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe", | ||||
|                 "C:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe", | ||||
|                 "D:\\Program Files\\Google\\Chrome\\Application\\chrome.exe", | ||||
|                 "D:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe", | ||||
|                 "E:\\Program Files\\Google\\Chrome\\Application\\chrome.exe", | ||||
|                 "E:\\Program Files (x86)\\Google\\Chrome\\Application\\chrome.exe", | ||||
|             ]); | ||||
|         } else if (process.platform === "linux") { | ||||
|             executablePath = findChrome([ | ||||
|                 "chromium-browser", | ||||
|                 "chromium", | ||||
|                 "google-chrome", | ||||
|             ]); | ||||
|         } | ||||
|         // TODO: Mac?? | ||||
|     } | ||||
|     return executablePath; | ||||
| } | ||||
|  | ||||
| function findChrome(executables) { | ||||
|     for (let executable of executables) { | ||||
|         if (commandExistsSync(executable)) { | ||||
|             return executable; | ||||
|         } | ||||
|     } | ||||
|     throw new Error("Chromium not found, please specify Chromium executable path in the settings page."); | ||||
| } | ||||
|  | ||||
| async function resetChrome() { | ||||
|     if (browser) { | ||||
|         await browser.close(); | ||||
|         browser = null; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Test if the chrome executable is valid and return the version | ||||
|  * @param executablePath | ||||
|  * @returns {Promise<string>} | ||||
|  */ | ||||
| async function testChrome(executablePath) { | ||||
|     try { | ||||
|         executablePath = await prepareChromeExecutable(executablePath); | ||||
|  | ||||
|         log.info("Chromium", "Testing Chromium executable: " + executablePath); | ||||
|  | ||||
|         const browser = await chromium.launch({ | ||||
|             executablePath, | ||||
|         }); | ||||
|         const version = browser.version(); | ||||
|         await browser.close(); | ||||
|         return version; | ||||
|     } catch (e) { | ||||
|         throw new Error(e.message); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * TODO: connect remote browser? https://playwright.dev/docs/api/class-browsertype#browser-type-connect | ||||
|  * | ||||
|  */ | ||||
| class RealBrowserMonitorType extends MonitorType { | ||||
|  | ||||
|     name = "real-browser"; | ||||
|  | ||||
|     async check(monitor, heartbeat, server) { | ||||
|         const browser = await getBrowser(); | ||||
|         const context = await browser.newContext(); | ||||
|         const page = await context.newPage(); | ||||
|  | ||||
|         const res = await page.goto(monitor.url, { | ||||
|             waitUntil: "networkidle", | ||||
|             timeout: monitor.interval * 1000 * 0.8, | ||||
|         }); | ||||
|  | ||||
|         let filename = jwt.sign(monitor.id, server.jwtSecret) + ".png"; | ||||
|  | ||||
|         await page.screenshot({ | ||||
|             path: path.join(Database.screenshotDir, filename), | ||||
|         }); | ||||
|  | ||||
|         await context.close(); | ||||
|  | ||||
|         if (res.status() >= 200 && res.status() < 400) { | ||||
|             heartbeat.status = UP; | ||||
|             heartbeat.msg = res.status(); | ||||
|  | ||||
|             const timing = res.request().timing(); | ||||
|             heartbeat.ping = timing.responseEnd; | ||||
|         } else { | ||||
|             throw new Error(res.status() + ""); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     RealBrowserMonitorType, | ||||
|     testChrome, | ||||
|     resetChrome, | ||||
| }; | ||||
| @@ -15,7 +15,7 @@ class DingDing extends NotificationProvider { | ||||
|                     msgtype: "markdown", | ||||
|                     markdown: { | ||||
|                         title: `[${this.statusToString(heartbeatJSON["status"])}] ${monitorJSON["name"]}`, | ||||
|                         text: `## [${this.statusToString(heartbeatJSON["status"])}] ${monitorJSON["name"]} \n > ${heartbeatJSON["msg"]}  \n > Time(UTC):${heartbeatJSON["time"]}`, | ||||
|                         text: `## [${this.statusToString(heartbeatJSON["status"])}] ${monitorJSON["name"]} \n> ${heartbeatJSON["msg"]}\n> Time (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}`, | ||||
|                     } | ||||
|                 }; | ||||
|                 if (this.sendToDingDing(notification, params)) { | ||||
|   | ||||
| @@ -59,8 +59,8 @@ class Discord extends NotificationProvider { | ||||
|                                 value: monitorJSON["type"] === "push" ? "Heartbeat" : address, | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: "Time (UTC)", | ||||
|                                 value: heartbeatJSON["time"], | ||||
|                                 name: `Time (${heartbeatJSON["timezone"]})`, | ||||
|                                 value: heartbeatJSON["localDateTime"], | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: "Error", | ||||
| @@ -94,8 +94,8 @@ class Discord extends NotificationProvider { | ||||
|                                 value: monitorJSON["type"] === "push" ? "Heartbeat" : address, | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: "Time (UTC)", | ||||
|                                 value: heartbeatJSON["time"], | ||||
|                                 name: `Time (${heartbeatJSON["timezone"]})`, | ||||
|                                 value: heartbeatJSON["localDateTime"], | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: "Ping", | ||||
|   | ||||
| @@ -35,8 +35,7 @@ class Feishu extends NotificationProvider { | ||||
|                                             text: | ||||
|                                                 "[Down] " + | ||||
|                                                 heartbeatJSON["msg"] + | ||||
|                                                 "\nTime (UTC): " + | ||||
|                                                 heartbeatJSON["time"], | ||||
|                                                 `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                                         }, | ||||
|                                     ], | ||||
|                                 ], | ||||
| @@ -62,8 +61,7 @@ class Feishu extends NotificationProvider { | ||||
|                                             text: | ||||
|                                                 "[Up] " + | ||||
|                                                 heartbeatJSON["msg"] + | ||||
|                                                 "\nTime (UTC): " + | ||||
|                                                 heartbeatJSON["time"], | ||||
|                                                 `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}`, | ||||
|                                         }, | ||||
|                                     ], | ||||
|                                 ], | ||||
|   | ||||
| @@ -11,7 +11,7 @@ class HomeAssistant extends NotificationProvider { | ||||
|  | ||||
|         try { | ||||
|             await axios.post( | ||||
|                 `${notification.homeAssistantUrl}/api/services/notify/${notificationService}`, | ||||
|                 `${notification.homeAssistantUrl.trim().replace(/\/*$/, "")}/api/services/notify/${notificationService}`, | ||||
|                 { | ||||
|                     title: "Uptime Kuma", | ||||
|                     message, | ||||
|   | ||||
| @@ -33,7 +33,10 @@ class Line extends NotificationProvider { | ||||
|                     "messages": [ | ||||
|                         { | ||||
|                             "type": "text", | ||||
|                             "text": "UptimeKuma Alert: [🔴 Down]\n" + "Name: " + monitorJSON["name"] + " \n" + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"] | ||||
|                             "text": "UptimeKuma Alert: [🔴 Down]\n" + | ||||
|                                 "Name: " + monitorJSON["name"] + " \n" + | ||||
|                                 heartbeatJSON["msg"] + | ||||
|                                 `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                         } | ||||
|                     ] | ||||
|                 }; | ||||
| @@ -44,7 +47,10 @@ class Line extends NotificationProvider { | ||||
|                     "messages": [ | ||||
|                         { | ||||
|                             "type": "text", | ||||
|                             "text": "UptimeKuma Alert: [✅ Up]\n" + "Name: " + monitorJSON["name"] + " \n" + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"] | ||||
|                             "text": "UptimeKuma Alert: [✅ Up]\n" + | ||||
|                                 "Name: " + monitorJSON["name"] + " \n" + | ||||
|                                 heartbeatJSON["msg"] + | ||||
|                                 `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                         } | ||||
|                     ] | ||||
|                 }; | ||||
|   | ||||
| @@ -24,12 +24,18 @@ class LineNotify extends NotificationProvider { | ||||
|                 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"] | ||||
|                     "message": "\n[🔴 Down]\n" + | ||||
|                         "Name: " + monitorJSON["name"] + " \n" + | ||||
|                         heartbeatJSON["msg"] + "\n" + | ||||
|                         `Time (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                 }; | ||||
|                 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"] | ||||
|                     "message": "\n[✅ Up]\n" + | ||||
|                         "Name: " + monitorJSON["name"] + " \n" + | ||||
|                         heartbeatJSON["msg"] + "\n" + | ||||
|                         `Time (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                 }; | ||||
|                 await axios.post(lineAPIUrl, qs.stringify(upMessage), config); | ||||
|             } | ||||
|   | ||||
| @@ -8,7 +8,12 @@ class LunaSea extends NotificationProvider { | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         let lunaseadevice = "https://notify.lunasea.app/v1/custom/device/" + notification.lunaseaDevice; | ||||
|         let lunaseaurl = ""; | ||||
|         if (notification.lunaseaTarget === "user") { | ||||
|             lunaseaurl = "https://notify.lunasea.app/v1/custom/user/" + notification.lunaseaUserID; | ||||
|         } else { | ||||
|             lunaseaurl = "https://notify.lunasea.app/v1/custom/device/" + notification.lunaseaDevice; | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             if (heartbeatJSON == null) { | ||||
| @@ -16,25 +21,29 @@ class LunaSea extends NotificationProvider { | ||||
|                     "title": "Uptime Kuma Alert", | ||||
|                     "body": msg, | ||||
|                 }; | ||||
|                 await axios.post(lunaseadevice, testdata); | ||||
|                 await axios.post(lunaseaurl, testdata); | ||||
|                 return okMsg; | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON["status"] === DOWN) { | ||||
|                 let downdata = { | ||||
|                     "title": "UptimeKuma Alert: " + monitorJSON["name"], | ||||
|                     "body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"], | ||||
|                     "body": "[🔴 Down] " + | ||||
|                         heartbeatJSON["msg"] + | ||||
|                         `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                 }; | ||||
|                 await axios.post(lunaseadevice, downdata); | ||||
|                 await axios.post(lunaseaurl, downdata); | ||||
|                 return okMsg; | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON["status"] === UP) { | ||||
|                 let updata = { | ||||
|                     "title": "UptimeKuma Alert: " + monitorJSON["name"], | ||||
|                     "body": "[✅ Up] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"], | ||||
|                     "body": "[✅ Up] " + | ||||
|                         heartbeatJSON["msg"] + | ||||
|                         `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}` | ||||
|                 }; | ||||
|                 await axios.post(lunaseadevice, updata); | ||||
|                 await axios.post(lunaseaurl, updata); | ||||
|                 return okMsg; | ||||
|             } | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,7 @@ class Mattermost extends NotificationProvider { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             const mattermostUserName = notification.mattermostusername || "Uptime Kuma"; | ||||
|             // If heartbeatJSON is null, assume we're testing. | ||||
|             // If heartbeatJSON is null, assume non monitoring notification (Certificate warning) or testing. | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let mattermostTestData = { | ||||
|                     username: mattermostUserName, | ||||
| @@ -27,97 +27,79 @@ class Mattermost extends NotificationProvider { | ||||
|             } | ||||
|  | ||||
|             const mattermostIconEmoji = notification.mattermosticonemo; | ||||
|             const mattermostIconUrl = notification.mattermosticonurl; | ||||
|             let mattermostIconEmojiOnline = ""; | ||||
|             let mattermostIconEmojiOffline = ""; | ||||
|  | ||||
|             if (heartbeatJSON["status"] === DOWN) { | ||||
|                 let mattermostdowndata = { | ||||
|                     username: mattermostUserName, | ||||
|                     text: "Uptime Kuma Alert", | ||||
|                     channel: mattermostChannel, | ||||
|                     icon_emoji: mattermostIconEmoji, | ||||
|                     icon_url: mattermostIconUrl, | ||||
|                     attachments: [ | ||||
|                         { | ||||
|                             fallback: | ||||
|                                 "Your " + | ||||
|                                 monitorJSON["name"] + | ||||
|                                 " service went down.", | ||||
|                             color: "#FF0000", | ||||
|                             title: | ||||
|                                 "❌ " + | ||||
|                                 monitorJSON["name"] + | ||||
|                                 " service went down. ❌", | ||||
|                             title_link: monitorJSON["url"], | ||||
|                             fields: [ | ||||
|                                 { | ||||
|                                     short: true, | ||||
|                                     title: "Service Name", | ||||
|                                     value: monitorJSON["name"], | ||||
|                                 }, | ||||
|                                 { | ||||
|                                     short: true, | ||||
|                                     title: "Time (UTC)", | ||||
|                                     value: heartbeatJSON["time"], | ||||
|                                 }, | ||||
|                                 { | ||||
|                                     short: false, | ||||
|                                     title: "Error", | ||||
|                                     value: heartbeatJSON["msg"], | ||||
|                                 }, | ||||
|                             ], | ||||
|                         }, | ||||
|                     ], | ||||
|                 }; | ||||
|                 await axios.post( | ||||
|                     notification.mattermostWebhookUrl, | ||||
|                     mattermostdowndata | ||||
|                 ); | ||||
|                 return okMsg; | ||||
|             } else if (heartbeatJSON["status"] === UP) { | ||||
|                 let mattermostupdata = { | ||||
|                     username: mattermostUserName, | ||||
|                     text: "Uptime Kuma Alert", | ||||
|                     channel: mattermostChannel, | ||||
|                     icon_emoji: mattermostIconEmoji, | ||||
|                     icon_url: mattermostIconUrl, | ||||
|                     attachments: [ | ||||
|                         { | ||||
|                             fallback: | ||||
|                                 "Your " + | ||||
|                                 monitorJSON["name"] + | ||||
|                                 " service went up!", | ||||
|                             color: "#32CD32", | ||||
|                             title: | ||||
|                                 "✅ " + | ||||
|                                 monitorJSON["name"] + | ||||
|                                 " service went up! ✅", | ||||
|                             title_link: monitorJSON["url"], | ||||
|                             fields: [ | ||||
|                                 { | ||||
|                                     short: true, | ||||
|                                     title: "Service Name", | ||||
|                                     value: monitorJSON["name"], | ||||
|                                 }, | ||||
|                                 { | ||||
|                                     short: true, | ||||
|                                     title: "Time (UTC)", | ||||
|                                     value: heartbeatJSON["time"], | ||||
|                                 }, | ||||
|                                 { | ||||
|                                     short: false, | ||||
|                                     title: "Ping", | ||||
|                                     value: heartbeatJSON["ping"] + "ms", | ||||
|                                 }, | ||||
|                             ], | ||||
|                         }, | ||||
|                     ], | ||||
|                 }; | ||||
|                 await axios.post( | ||||
|                     notification.mattermostWebhookUrl, | ||||
|                     mattermostupdata | ||||
|                 ); | ||||
|                 return okMsg; | ||||
|             if (mattermostIconEmoji && typeof mattermostIconEmoji === "string") { | ||||
|                 const emojiArray = mattermostIconEmoji.split(" "); | ||||
|                 if (emojiArray.length >= 2) { | ||||
|                     mattermostIconEmojiOnline = emojiArray[0]; | ||||
|                     mattermostIconEmojiOffline = emojiArray[1]; | ||||
|                 } | ||||
|             } | ||||
|             const mattermostIconUrl = notification.mattermosticonurl; | ||||
|             let iconEmoji = mattermostIconEmoji; | ||||
|             let statusField = { | ||||
|                 short: false, | ||||
|                 title: "Error", | ||||
|                 value: heartbeatJSON.msg, | ||||
|             }; | ||||
|             let statusText = "unknown"; | ||||
|             let color = "#000000"; | ||||
|             if (heartbeatJSON.status === DOWN) { | ||||
|                 iconEmoji = mattermostIconEmojiOffline || mattermostIconEmoji; | ||||
|                 statusField = { | ||||
|                     short: false, | ||||
|                     title: "Error", | ||||
|                     value: heartbeatJSON.msg, | ||||
|                 }; | ||||
|                 statusText = "down."; | ||||
|                 color = "#FF0000"; | ||||
|             } else if (heartbeatJSON.status === UP) { | ||||
|                 iconEmoji = mattermostIconEmojiOnline || mattermostIconEmoji; | ||||
|                 statusField = { | ||||
|                     short: false, | ||||
|                     title: "Ping", | ||||
|                     value: heartbeatJSON.ping + "ms", | ||||
|                 }; | ||||
|                 statusText = "up!"; | ||||
|                 color = "#32CD32"; | ||||
|             } | ||||
|  | ||||
|             let mattermostdata = { | ||||
|                 username: monitorJSON.name + " " + mattermostUserName, | ||||
|                 channel: mattermostChannel, | ||||
|                 icon_emoji: iconEmoji, | ||||
|                 icon_url: mattermostIconUrl, | ||||
|                 attachments: [ | ||||
|                     { | ||||
|                         fallback: | ||||
|                             "Your " + | ||||
|                             monitorJSON.name + | ||||
|                             " service went " + | ||||
|                             statusText, | ||||
|                         color: color, | ||||
|                         title: | ||||
|                             monitorJSON.name + | ||||
|                             " service went " + | ||||
|                             statusText, | ||||
|                         title_link: monitorJSON.url, | ||||
|                         fields: [ | ||||
|                             statusField, | ||||
|                             { | ||||
|                                 short: true, | ||||
|                                 title: `Time (${heartbeatJSON["timezone"]})`, | ||||
|                                 value: heartbeatJSON.localDateTime, | ||||
|                             }, | ||||
|                         ], | ||||
|                     }, | ||||
|                 ], | ||||
|             }; | ||||
|             await axios.post( | ||||
|                 notification.mattermostWebhookUrl, | ||||
|                 mattermostdata | ||||
|             ); | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { DOWN, UP } = require("../../src/util"); | ||||
|  | ||||
| class Ntfy extends NotificationProvider { | ||||
|  | ||||
| @@ -9,16 +10,54 @@ class Ntfy extends NotificationProvider { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             let headers = {}; | ||||
|             if (notification.ntfyusername) { | ||||
|             if (notification.ntfyAuthenticationMethod === "usernamePassword") { | ||||
|                 headers = { | ||||
|                     "Authorization": "Basic " + Buffer.from(notification.ntfyusername + ":" + notification.ntfypassword).toString("base64"), | ||||
|                 }; | ||||
|             } else if (notification.ntfyAuthenticationMethod === "accessToken") { | ||||
|                 headers = { | ||||
|                     "Authorization": "Bearer " + notification.ntfyaccesstoken, | ||||
|                 }; | ||||
|             } | ||||
|             // If heartbeatJSON is null, assume non monitoring notification (Certificate warning) or testing. | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let ntfyTestData = { | ||||
|                     "topic": notification.ntfytopic, | ||||
|                     "title": (monitorJSON?.name || notification.ntfytopic) + " [Uptime-Kuma]", | ||||
|                     "message": msg, | ||||
|                     "priority": notification.ntfyPriority, | ||||
|                     "tags": [ "test_tube" ], | ||||
|                 }; | ||||
|                 await axios.post(`${notification.ntfyserverurl}`, ntfyTestData, { headers: headers }); | ||||
|                 return okMsg; | ||||
|             } | ||||
|             let tags = []; | ||||
|             let status = "unknown"; | ||||
|             let priority = notification.ntfyPriority || 4; | ||||
|             if ("status" in heartbeatJSON) { | ||||
|                 if (heartbeatJSON.status === DOWN) { | ||||
|                     tags = [ "red_circle" ]; | ||||
|                     status = "Down"; | ||||
|                     // if priority is not 5, increase priority for down alerts | ||||
|                     priority = priority === 5 ? priority : priority + 1; | ||||
|                 } else if (heartbeatJSON["status"] === UP) { | ||||
|                     tags = [ "green_circle" ]; | ||||
|                     status = "Up"; | ||||
|                 } | ||||
|             } | ||||
|             let data = { | ||||
|                 "topic": notification.ntfytopic, | ||||
|                 "message": msg, | ||||
|                 "priority": notification.ntfyPriority || 4, | ||||
|                 "title": "Uptime-Kuma", | ||||
|                 "message": heartbeatJSON.msg, | ||||
|                 "priority": priority, | ||||
|                 "title": monitorJSON.name + " " + status + " [Uptime-Kuma]", | ||||
|                 "tags": tags, | ||||
|                 "actions": [ | ||||
|                     { | ||||
|                         "action": "view", | ||||
|                         "label": "Open " + monitorJSON.name, | ||||
|                         "url": monitorJSON.url, | ||||
|                     } | ||||
|                 ] | ||||
|             }; | ||||
|  | ||||
|             if (notification.ntfyIcon) { | ||||
|   | ||||
							
								
								
									
										97
									
								
								server/notification-providers/opsgenie.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								server/notification-providers/opsgenie.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { UP, DOWN } = require("../../src/util"); | ||||
|  | ||||
| const opsgenieAlertsUrlEU = "https://api.eu.opsgenie.com/v2/alerts"; | ||||
| const opsgenieAlertsUrlUS = "https://api.opsgenie.com/v2/alerts"; | ||||
| let okMsg = "Sent Successfully."; | ||||
|  | ||||
| class Opsgenie extends NotificationProvider { | ||||
|  | ||||
|     name = "Opsgenie"; | ||||
|  | ||||
|     /** | ||||
|      * @inheritdoc | ||||
|      */ | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let opsgenieAlertsUrl; | ||||
|         let priority = (!notification.opsgeniePriority) ? 3 : notification.opsgeniePriority; | ||||
|         const textMsg = "Uptime Kuma Alert"; | ||||
|  | ||||
|         try { | ||||
|             switch (notification.opsgenieRegion) { | ||||
|                 case "US": | ||||
|                     opsgenieAlertsUrl = opsgenieAlertsUrlUS; | ||||
|                     break; | ||||
|                 case "EU": | ||||
|                     opsgenieAlertsUrl = opsgenieAlertsUrlEU; | ||||
|                     break; | ||||
|                 default: | ||||
|                     opsgenieAlertsUrl = opsgenieAlertsUrlUS; | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON == null) { | ||||
|                 let notificationTestAlias = "uptime-kuma-notification-test"; | ||||
|                 let data = { | ||||
|                     "message": msg, | ||||
|                     "alias": notificationTestAlias, | ||||
|                     "source": "Uptime Kuma", | ||||
|                     "priority": "P5" | ||||
|                 }; | ||||
|  | ||||
|                 return this.post(notification, opsgenieAlertsUrl, data); | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON.status === DOWN) { | ||||
|                 let data = { | ||||
|                     "message": monitorJSON ? textMsg + `: ${monitorJSON.name}` : textMsg, | ||||
|                     "alias": monitorJSON.name, | ||||
|                     "description": msg, | ||||
|                     "source": "Uptime Kuma", | ||||
|                     "priority": `P${priority}` | ||||
|                 }; | ||||
|  | ||||
|                 return this.post(notification, opsgenieAlertsUrl, data); | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON.status === UP) { | ||||
|                 let opsgenieAlertsCloseUrl = `${opsgenieAlertsUrl}/${encodeURIComponent(monitorJSON.name)}/close?identifierType=alias`; | ||||
|                 let data = { | ||||
|                     "source": "Uptime Kuma", | ||||
|                 }; | ||||
|  | ||||
|                 return this.post(notification, opsgenieAlertsCloseUrl, data); | ||||
|             } | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * | ||||
|      * @param {BeanModel} notification | ||||
|      * @param {string} url Request url | ||||
|      * @param {Object} data Request body | ||||
|      * @returns {Promise<string>} | ||||
|      */ | ||||
|     async post(notification, url, data) { | ||||
|         let config = { | ||||
|             headers: { | ||||
|                 "Content-Type": "application/json", | ||||
|                 "Authorization": `GenieKey ${notification.opsgenieApiKey}`, | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         let res = await axios.post(url, data, config); | ||||
|         if (res.status == null) { | ||||
|             return "Opsgenie notification failed with invalid response!"; | ||||
|         } | ||||
|         if (res.status < 200 || res.status >= 300) { | ||||
|             return `Opsgenie notification failed with status code ${res.status}`; | ||||
|         } | ||||
|  | ||||
|         return okMsg; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Opsgenie; | ||||
							
								
								
									
										91
									
								
								server/notification-providers/pagertree.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								server/notification-providers/pagertree.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { UP, DOWN, getMonitorRelativeURL } = require("../../src/util"); | ||||
| const { setting } = require("../util-server"); | ||||
| let successMessage = "Sent Successfully."; | ||||
|  | ||||
| class PagerTree extends NotificationProvider { | ||||
|     name = "PagerTree"; | ||||
|  | ||||
|     /** | ||||
|      * @inheritdoc | ||||
|      */ | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         try { | ||||
|             if (heartbeatJSON == null) { | ||||
|                 // general messages | ||||
|                 return this.postNotification(notification, msg, monitorJSON, heartbeatJSON); | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON.status === UP && notification.pagertreeAutoResolve === "resolve") { | ||||
|                 return this.postNotification(notification, null, monitorJSON, heartbeatJSON, notification.pagertreeAutoResolve); | ||||
|             } | ||||
|  | ||||
|             if (heartbeatJSON.status === DOWN) { | ||||
|                 const title = `Uptime Kuma Monitor "${monitorJSON.name}" is DOWN`; | ||||
|                 return this.postNotification(notification, title, monitorJSON, heartbeatJSON); | ||||
|             } | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if result is successful, result code should be in range 2xx | ||||
|      * @param {Object} result Axios response object | ||||
|      * @throws {Error} The status code is not in range 2xx | ||||
|      */ | ||||
|     checkResult(result) { | ||||
|         if (result.status == null) { | ||||
|             throw new Error("PagerTree notification failed with invalid response!"); | ||||
|         } | ||||
|         if (result.status < 200 || result.status >= 300) { | ||||
|             throw new Error("PagerTree notification failed with status code " + result.status); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Send the message | ||||
|      * @param {BeanModel} notification Message title | ||||
|      * @param {string} title Message title | ||||
|      * @param {Object} monitorJSON Monitor details (For Up/Down only) | ||||
|      * @param {?string} eventAction Action event for PagerTree (create, resolve) | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     async postNotification(notification, title, monitorJSON, heartbeatJSON, eventAction = "create") { | ||||
|  | ||||
|         if (eventAction == null) { | ||||
|             return "No action required"; | ||||
|         } | ||||
|  | ||||
|         const options = { | ||||
|             method: "POST", | ||||
|             url: notification.pagertreeIntegrationUrl, | ||||
|             headers: { "Content-Type": "application/json" }, | ||||
|             data: { | ||||
|                 event_type: eventAction, | ||||
|                 id: heartbeatJSON?.monitorID || "uptime-kuma", | ||||
|                 title: title, | ||||
|                 urgency: notification.pagertreeUrgency, | ||||
|                 heartbeat: heartbeatJSON, | ||||
|                 monitor: monitorJSON | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         const baseURL = await setting("primaryBaseURL"); | ||||
|         if (baseURL && monitorJSON) { | ||||
|             options.client = "Uptime Kuma"; | ||||
|             options.client_url = baseURL + getMonitorRelativeURL(monitorJSON.id); | ||||
|         } | ||||
|  | ||||
|         let result = await axios.request(options); | ||||
|         this.checkResult(result); | ||||
|         if (result.statusText != null) { | ||||
|             return "PagerTree notification succeed: " + result.statusText; | ||||
|         } | ||||
|  | ||||
|         return successMessage; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = PagerTree; | ||||
| @@ -29,14 +29,18 @@ class Pushbullet extends NotificationProvider { | ||||
|                 let downData = { | ||||
|                     "type": "note", | ||||
|                     "title": "UptimeKuma Alert: " + monitorJSON["name"], | ||||
|                     "body": "[🔴 Down] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"], | ||||
|                     "body": "[🔴 Down] " + | ||||
|                         heartbeatJSON["msg"] + | ||||
|                         `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}`, | ||||
|                 }; | ||||
|                 await axios.post(pushbulletUrl, downData, config); | ||||
|             } else if (heartbeatJSON["status"] === UP) { | ||||
|                 let upData = { | ||||
|                     "type": "note", | ||||
|                     "title": "UptimeKuma Alert: " + monitorJSON["name"], | ||||
|                     "body": "[✅ Up] " + heartbeatJSON["msg"] + "\nTime (UTC): " + heartbeatJSON["time"], | ||||
|                     "body": "[✅ Up] " + | ||||
|                         heartbeatJSON["msg"] + | ||||
|                         `\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}`, | ||||
|                 }; | ||||
|                 await axios.post(pushbulletUrl, upData, config); | ||||
|             } | ||||
|   | ||||
| @@ -24,13 +24,16 @@ class Pushover extends NotificationProvider { | ||||
|         if (notification.pushoverdevice) { | ||||
|             data.device = notification.pushoverdevice; | ||||
|         } | ||||
|         if (notification.pushoverttl) { | ||||
|             data.ttl = notification.pushoverttl; | ||||
|         } | ||||
|  | ||||
|         try { | ||||
|             if (heartbeatJSON == null) { | ||||
|                 await axios.post(pushoverlink, data); | ||||
|                 return okMsg; | ||||
|             } else { | ||||
|                 data.message += "\n<b>Time (UTC)</b>:" + heartbeatJSON["time"]; | ||||
|                 data.message += `\n<b>Time (${heartbeatJSON["timezone"]})</b>:${heartbeatJSON["localDateTime"]}`; | ||||
|                 await axios.post(pushoverlink, data); | ||||
|                 return okMsg; | ||||
|             } | ||||
|   | ||||
| @@ -22,8 +22,6 @@ class RocketChat extends NotificationProvider { | ||||
|                 return okMsg; | ||||
|             } | ||||
|  | ||||
|             const time = heartbeatJSON["time"]; | ||||
|  | ||||
|             let data = { | ||||
|                 "text": "Uptime Kuma Alert", | ||||
|                 "channel": notification.rocketchannel, | ||||
| @@ -31,7 +29,7 @@ class RocketChat extends NotificationProvider { | ||||
|                 "icon_emoji": notification.rocketiconemo, | ||||
|                 "attachments": [ | ||||
|                     { | ||||
|                         "title": "Uptime Kuma Alert *Time (UTC)*\n" + time, | ||||
|                         "title": `Uptime Kuma Alert *Time (${heartbeatJSON["timezone"]})*\n${heartbeatJSON["localDateTime"]}`, | ||||
|                         "text": "*Message*\n" + msg, | ||||
|                     } | ||||
|                 ] | ||||
|   | ||||
| @@ -39,10 +39,9 @@ class Slack extends NotificationProvider { | ||||
|                 return okMsg; | ||||
|             } | ||||
|  | ||||
|             const time = heartbeatJSON["time"]; | ||||
|             const textMsg = "Uptime Kuma Alert"; | ||||
|             let data = { | ||||
|                 "text": monitorJSON ? textMsg + `: ${monitorJSON.name}` : textMsg, | ||||
|                 "text": `${textMsg}\n${msg}`, | ||||
|                 "channel": notification.slackchannel, | ||||
|                 "username": notification.slackusername, | ||||
|                 "icon_emoji": notification.slackiconemo, | ||||
| @@ -65,7 +64,7 @@ class Slack extends NotificationProvider { | ||||
|                                 }, | ||||
|                                 { | ||||
|                                     "type": "mrkdwn", | ||||
|                                     "text": "*Time (UTC)*\n" + time, | ||||
|                                     "text": `*Time (${heartbeatJSON["timezone"]})*\n${heartbeatJSON["localDateTime"]}`, | ||||
|                                 }], | ||||
|                             } | ||||
|                         ], | ||||
|   | ||||
| @@ -91,7 +91,7 @@ class SMTP extends NotificationProvider { | ||||
|  | ||||
|         let bodyTextContent = msg; | ||||
|         if (heartbeatJSON) { | ||||
|             bodyTextContent = `${msg}\nTime (UTC): ${heartbeatJSON["time"]}`; | ||||
|             bodyTextContent = `${msg}\nTime (${heartbeatJSON["timezone"]}): ${heartbeatJSON["localDateTime"]}`; | ||||
|         } | ||||
|  | ||||
|         // send mail with defined transport object | ||||
|   | ||||
| @@ -9,17 +9,27 @@ class Telegram extends NotificationProvider { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         try { | ||||
|             let params = { | ||||
|                 chat_id: notification.telegramChatID, | ||||
|                 text: msg, | ||||
|                 disable_notification: notification.telegramSendSilently ?? false, | ||||
|                 protect_content: notification.telegramProtectContent ?? false, | ||||
|             }; | ||||
|             if (notification.telegramMessageThreadID) { | ||||
|                 params.message_thread_id = notification.telegramMessageThreadID; | ||||
|             } | ||||
|  | ||||
|             await axios.get(`https://api.telegram.org/bot${notification.telegramBotToken}/sendMessage`, { | ||||
|                 params: { | ||||
|                     chat_id: notification.telegramChatID, | ||||
|                     text: msg, | ||||
|                 }, | ||||
|                 params: params, | ||||
|             }); | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             let msg = (error.response.data.description) ? error.response.data.description : "Error without description"; | ||||
|             throw new Error(msg); | ||||
|             if (error.response && error.response.data && error.response.data.description) { | ||||
|                 throw new Error(error.response.data.description); | ||||
|             } else { | ||||
|                 throw new Error(error.message); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										41
									
								
								server/notification-providers/twilio.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								server/notification-providers/twilio.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class Twilio extends NotificationProvider { | ||||
|  | ||||
|     name = "twilio"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|  | ||||
|         let okMsg = "Sent Successfully."; | ||||
|  | ||||
|         let accountSID = notification.twilioAccountSID; | ||||
|         let authToken = notification.twilioAuthToken; | ||||
|  | ||||
|         try { | ||||
|  | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "Content-Type": "application/x-www-form-urlencoded;charset=utf-8", | ||||
|                     "Authorization": "Basic " + Buffer.from(accountSID + ":" + authToken).toString("base64"), | ||||
|                 } | ||||
|             }; | ||||
|  | ||||
|             let data = new URLSearchParams(); | ||||
|             data.append("To", notification.twilioToNumber); | ||||
|             data.append("From", notification.twilioFromNumber); | ||||
|             data.append("Body", msg); | ||||
|  | ||||
|             let url = "https://api.twilio.com/2010-04-01/Accounts/" + accountSID + "/Messages.json"; | ||||
|  | ||||
|             await axios.post(url, data, config); | ||||
|  | ||||
|             return okMsg; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| module.exports = Twilio; | ||||
| @@ -23,7 +23,9 @@ const Mattermost = require("./notification-providers/mattermost"); | ||||
| const Ntfy = require("./notification-providers/ntfy"); | ||||
| const Octopush = require("./notification-providers/octopush"); | ||||
| const OneBot = require("./notification-providers/onebot"); | ||||
| const Opsgenie = require("./notification-providers/opsgenie"); | ||||
| const PagerDuty = require("./notification-providers/pagerduty"); | ||||
| const PagerTree = require("./notification-providers/pagertree"); | ||||
| const PromoSMS = require("./notification-providers/promosms"); | ||||
| const Pushbullet = require("./notification-providers/pushbullet"); | ||||
| const PushDeer = require("./notification-providers/pushdeer"); | ||||
| @@ -40,6 +42,7 @@ 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 Twilio = require("./notification-providers/twilio"); | ||||
| const Splunk = require("./notification-providers/splunk"); | ||||
| const Webhook = require("./notification-providers/webhook"); | ||||
| const WeCom = require("./notification-providers/wecom"); | ||||
| @@ -82,7 +85,9 @@ class Notification { | ||||
|             new Ntfy(), | ||||
|             new Octopush(), | ||||
|             new OneBot(), | ||||
|             new Opsgenie(), | ||||
|             new PagerDuty(), | ||||
|             new PagerTree(), | ||||
|             new PromoSMS(), | ||||
|             new Pushbullet(), | ||||
|             new PushDeer(), | ||||
| @@ -101,6 +106,7 @@ class Notification { | ||||
|             new Teams(), | ||||
|             new TechulusPush(), | ||||
|             new Telegram(), | ||||
|             new Twilio(), | ||||
|             new Splunk(), | ||||
|             new Webhook(), | ||||
|             new WeCom(), | ||||
|   | ||||
| @@ -28,7 +28,7 @@ const monitorResponseTime = new PrometheusClient.Gauge({ | ||||
|  | ||||
| const monitorStatus = new PrometheusClient.Gauge({ | ||||
|     name: "monitor_status", | ||||
|     help: "Monitor Status (1 = UP, 0= DOWN)", | ||||
|     help: "Monitor Status (1 = UP, 0= DOWN, 2= PENDING, 3= MAINTENANCE)", | ||||
|     labelNames: commonLabels | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -132,6 +132,9 @@ class Proxy { | ||||
|                     ...httpAgentOptions, | ||||
|                     ...httpsAgentOptions, | ||||
|                     ...proxyOptions, | ||||
|                     tls: { | ||||
|                         rejectUnauthorized: httpsAgentOptions.rejectUnauthorized, | ||||
|                     }, | ||||
|                 }); | ||||
|  | ||||
|                 httpAgent = agent; | ||||
|   | ||||
| @@ -54,6 +54,13 @@ const loginRateLimiter = new KumaRateLimiter({ | ||||
|     errorMessage: "Too frequently, try again later." | ||||
| }); | ||||
|  | ||||
| const apiRateLimiter = new KumaRateLimiter({ | ||||
|     tokensPerInterval: 60, | ||||
|     interval: "minute", | ||||
|     fireImmediately: true, | ||||
|     errorMessage: "Too frequently, try again later." | ||||
| }); | ||||
|  | ||||
| const twoFaRateLimiter = new KumaRateLimiter({ | ||||
|     tokensPerInterval: 30, | ||||
|     interval: "minute", | ||||
| @@ -63,5 +70,6 @@ const twoFaRateLimiter = new KumaRateLimiter({ | ||||
|  | ||||
| module.exports = { | ||||
|     loginRateLimiter, | ||||
|     apiRateLimiter, | ||||
|     twoFaRateLimiter, | ||||
| }; | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| let express = require("express"); | ||||
| const { allowDevAllOrigin, allowAllOrigin, percentageToColor, filterAndJoin, send403 } = require("../util-server"); | ||||
| const { allowDevAllOrigin, allowAllOrigin, percentageToColor, filterAndJoin, sendHttpError } = require("../util-server"); | ||||
| const { R } = require("redbean-node"); | ||||
| const apicache = require("../modules/apicache"); | ||||
| const Monitor = require("../model/monitor"); | ||||
| @@ -7,8 +7,10 @@ const dayjs = require("dayjs"); | ||||
| const { UP, MAINTENANCE, DOWN, PENDING, flipStatus, log } = require("../../src/util"); | ||||
| const StatusPage = require("../model/status_page"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const { UptimeCacheList } = require("../uptime-cache-list"); | ||||
| const { makeBadge } = require("badge-maker"); | ||||
| const { badgeConstants } = require("../config"); | ||||
| const { Prometheus } = require("../prometheus"); | ||||
| const Database = require("../database"); | ||||
|  | ||||
| let router = express.Router(); | ||||
| @@ -37,7 +39,7 @@ router.get("/api/push/:pushToken", async (request, response) => { | ||||
|  | ||||
|         let pushToken = request.params.pushToken; | ||||
|         let msg = request.query.msg || "OK"; | ||||
|         let ping = request.query.ping || null; | ||||
|         let ping = parseInt(request.query.ping) || null; | ||||
|         let statusString = request.query.status || "up"; | ||||
|         let status = (statusString === "up") ? UP : DOWN; | ||||
|  | ||||
| @@ -87,7 +89,9 @@ router.get("/api/push/:pushToken", async (request, response) => { | ||||
|         await R.store(bean); | ||||
|  | ||||
|         io.to(monitor.user_id).emit("heartbeat", bean.toJSON()); | ||||
|         UptimeCacheList.clearCache(monitor.id); | ||||
|         Monitor.sendStats(io, monitor.id, monitor.user_id); | ||||
|         new Prometheus(monitor).update(bean, undefined); | ||||
|  | ||||
|         response.json({ | ||||
|             ok: true, | ||||
| @@ -146,7 +150,11 @@ 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; | ||||
|  | ||||
|             badgeValues.label = label ?? "Status"; | ||||
|             if (label === undefined) { | ||||
|                 badgeValues.label = "Status"; | ||||
|             } else { | ||||
|                 badgeValues.label = label; | ||||
|             } | ||||
|             switch (state) { | ||||
|                 case DOWN: | ||||
|                     badgeValues.color = downColor; | ||||
| @@ -176,7 +184,7 @@ router.get("/api/badge/:id/status", cache("5 minutes"), async (request, response | ||||
|         response.type("image/svg+xml"); | ||||
|         response.send(svg); | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -223,7 +231,7 @@ router.get("/api/badge/:id/uptime/:duration?", cache("5 minutes"), async (reques | ||||
|             ); | ||||
|  | ||||
|             // limit the displayed uptime percentage to four (two, when displayed as percent) decimal digits | ||||
|             const cleanUptime = parseFloat(uptime.toPrecision(4)); | ||||
|             const cleanUptime = (uptime * 100).toPrecision(4); | ||||
|  | ||||
|             // use a given, custom color or calculate one based on the uptime value | ||||
|             badgeValues.color = color ?? percentageToColor(uptime); | ||||
| @@ -234,7 +242,7 @@ router.get("/api/badge/:id/uptime/:duration?", cache("5 minutes"), async (reques | ||||
|                 labelPrefix, | ||||
|                 label ?? `Uptime (${requestedDuration}${labelSuffix})`, | ||||
|             ]); | ||||
|             badgeValues.message = filterAndJoin([ prefix, `${cleanUptime * 100}`, suffix ]); | ||||
|             badgeValues.message = filterAndJoin([ prefix, cleanUptime, suffix ]); | ||||
|         } | ||||
|  | ||||
|         // build the SVG based on given values | ||||
| @@ -243,7 +251,7 @@ router.get("/api/badge/:id/uptime/:duration?", cache("5 minutes"), async (reques | ||||
|         response.type("image/svg+xml"); | ||||
|         response.send(svg); | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -306,7 +314,7 @@ router.get("/api/badge/:id/ping/:duration?", cache("5 minutes"), async (request, | ||||
|         response.type("image/svg+xml"); | ||||
|         response.send(svg); | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -378,7 +386,7 @@ router.get("/api/badge/:id/avg-response/:duration?", cache("5 minutes"), async ( | ||||
|         response.type("image/svg+xml"); | ||||
|         response.send(svg); | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -469,7 +477,7 @@ router.get("/api/badge/:id/cert-exp", cache("5 minutes"), async (request, respon | ||||
|         response.type("image/svg+xml"); | ||||
|         response.send(svg); | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -541,7 +549,7 @@ router.get("/api/badge/:id/response", cache("5 minutes"), async (request, respon | ||||
|         response.type("image/svg+xml"); | ||||
|         response.send(svg); | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -2,7 +2,7 @@ let express = require("express"); | ||||
| const apicache = require("../modules/apicache"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const StatusPage = require("../model/status_page"); | ||||
| const { allowDevAllOrigin, send403 } = require("../util-server"); | ||||
| const { allowDevAllOrigin, sendHttpError } = require("../util-server"); | ||||
| const { R } = require("redbean-node"); | ||||
| const Monitor = require("../model/monitor"); | ||||
|  | ||||
| @@ -44,10 +44,7 @@ router.get("/api/status-page/:slug", cache("5 minutes"), async (request, respons | ||||
|         let statusPageData = await StatusPage.getStatusPageData(statusPage); | ||||
|  | ||||
|         if (!statusPageData) { | ||||
|             response.statusCode = 404; | ||||
|             response.json({ | ||||
|                 msg: "Not Found" | ||||
|             }); | ||||
|             sendHttpError(response, "Not Found"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -55,7 +52,7 @@ router.get("/api/status-page/:slug", cache("5 minutes"), async (request, respons | ||||
|         response.json(statusPageData); | ||||
|  | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -103,7 +100,7 @@ router.get("/api/status-page/heartbeat/:slug", cache("1 minutes"), async (reques | ||||
|         }); | ||||
|  | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
| @@ -119,10 +116,7 @@ router.get("/api/status-page/:slug/manifest.json", cache("1440 minutes"), async | ||||
|         ]); | ||||
|  | ||||
|         if (!statusPage) { | ||||
|             response.statusCode = 404; | ||||
|             response.json({ | ||||
|                 msg: "Not Found" | ||||
|             }); | ||||
|             sendHttpError(response, "Not Found"); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @@ -141,7 +135,7 @@ router.get("/api/status-page/:slug/manifest.json", cache("1440 minutes"), async | ||||
|         }); | ||||
|  | ||||
|     } catch (error) { | ||||
|         send403(response, error.message); | ||||
|         sendHttpError(response, error.message); | ||||
|     } | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -19,6 +19,11 @@ const nodeVersion = parseInt(process.versions.node.split(".")[0]); | ||||
| const requiredVersion = 14; | ||||
| console.log(`Your Node.js version: ${nodeVersion}`); | ||||
|  | ||||
| // See more: https://github.com/louislam/uptime-kuma/issues/3138 | ||||
| if (nodeVersion >= 20) { | ||||
|     console.warn("\x1b[31m%s\x1b[0m", "Warning: Uptime Kuma is currently not stable on Node.js >= 20, please use Node.js 18."); | ||||
| } | ||||
|  | ||||
| if (nodeVersion < requiredVersion) { | ||||
|     console.error(`Error: Your Node.js version is not supported, please upgrade to Node.js >= ${requiredVersion}.`); | ||||
|     process.exit(-1); | ||||
| @@ -87,7 +92,7 @@ log.debug("server", "Importing Background Jobs"); | ||||
| const { initBackgroundJobs, stopBackgroundJobs } = require("./jobs"); | ||||
| const { loginRateLimiter, twoFaRateLimiter } = require("./rate-limiter"); | ||||
|  | ||||
| const { basicAuth } = require("./auth"); | ||||
| const { apiAuth } = require("./auth"); | ||||
| const { login } = require("./auth"); | ||||
| const passwordHash = require("./password-hash"); | ||||
|  | ||||
| @@ -129,7 +134,7 @@ if (config.demoMode) { | ||||
| } | ||||
|  | ||||
| // Must be after io instantiation | ||||
| const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList, sendInfo, sendProxyList, sendDockerHostList } = require("./client"); | ||||
| const { sendNotificationList, sendHeartbeatList, sendImportantHeartbeatList, sendInfo, sendProxyList, sendDockerHostList, sendAPIKeyList } = require("./client"); | ||||
| const { statusPageSocketHandler } = require("./socket-handlers/status-page-socket-handler"); | ||||
| const databaseSocketHandler = require("./socket-handlers/database-socket-handler"); | ||||
| const TwoFA = require("./2fa"); | ||||
| @@ -138,10 +143,13 @@ const { cloudflaredSocketHandler, autoStart: cloudflaredAutoStart, stop: cloudfl | ||||
| 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 { apiKeySocketHandler } = require("./socket-handlers/api-key-socket-handler"); | ||||
| const { generalSocketHandler } = require("./socket-handlers/general-socket-handler"); | ||||
| const { Settings } = require("./settings"); | ||||
| const { CacheableDnsHttpAgent } = require("./cacheable-dns-http-agent"); | ||||
| const { pluginsHandler } = require("./socket-handlers/plugins-handler"); | ||||
| const apicache = require("./modules/apicache"); | ||||
| const { resetChrome } = require("./monitor-types/real-browser-monitor-type"); | ||||
| const { EmbeddedMariaDB } = require("./embedded-mariadb"); | ||||
| const { SetupDatabase } = require("./setup-database"); | ||||
|  | ||||
| @@ -156,12 +164,6 @@ app.use(function (req, res, next) { | ||||
|     next(); | ||||
| }); | ||||
|  | ||||
| /** | ||||
|  * Use for decode the auth object | ||||
|  * @type {null} | ||||
|  */ | ||||
| let jwtSecret = null; | ||||
|  | ||||
| /** | ||||
|  * Show Setup Page | ||||
|  * @type {boolean} | ||||
| @@ -248,7 +250,7 @@ let needSetup = false; | ||||
|  | ||||
|     // Prometheus API metrics  /metrics | ||||
|     // With Basic Auth using the first user's username/password | ||||
|     app.get("/metrics", basicAuth, prometheusAPIMetrics()); | ||||
|     app.get("/metrics", apiAuth, prometheusAPIMetrics()); | ||||
|  | ||||
|     app.use("/", expressStaticGzip("dist", { | ||||
|         enableBrotli: true, | ||||
| @@ -298,7 +300,7 @@ let needSetup = false; | ||||
|             log.info("auth", `Login by token. IP=${clientIP}`); | ||||
|  | ||||
|             try { | ||||
|                 let decoded = jwt.verify(token, jwtSecret); | ||||
|                 let decoded = jwt.verify(token, server.jwtSecret); | ||||
|  | ||||
|                 log.info("auth", "Username from JWT: " + decoded.username); | ||||
|  | ||||
| @@ -369,7 +371,7 @@ let needSetup = false; | ||||
|                         ok: true, | ||||
|                         token: jwt.sign({ | ||||
|                             username: data.username, | ||||
|                         }, jwtSecret), | ||||
|                         }, server.jwtSecret), | ||||
|                     }); | ||||
|                 } | ||||
|  | ||||
| @@ -399,7 +401,7 @@ let needSetup = false; | ||||
|                             ok: true, | ||||
|                             token: jwt.sign({ | ||||
|                                 username: data.username, | ||||
|                             }, jwtSecret), | ||||
|                             }, server.jwtSecret), | ||||
|                         }); | ||||
|                     } else { | ||||
|  | ||||
| @@ -688,6 +690,7 @@ let needSetup = false; | ||||
|         // Edit a monitor | ||||
|         socket.on("editMonitor", async (monitor, callback) => { | ||||
|             try { | ||||
|                 let removeGroupChildren = false; | ||||
|                 checkLogin(socket); | ||||
|  | ||||
|                 let bean = await R.findOne("monitor", " id = ? ", [ monitor.id ]); | ||||
| @@ -696,10 +699,22 @@ let needSetup = false; | ||||
|                     throw new Error("Permission denied."); | ||||
|                 } | ||||
|  | ||||
|                 // Reset Prometheus labels | ||||
|                 server.monitorList[monitor.id]?.prometheus()?.remove(); | ||||
|                 // Check if Parent is Descendant (would cause endless loop) | ||||
|                 if (monitor.parent !== null) { | ||||
|                     const childIDs = await Monitor.getAllChildrenIDs(monitor.id); | ||||
|                     if (childIDs.includes(monitor.parent)) { | ||||
|                         throw new Error("Invalid Monitor Group"); | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 // Remove children if monitor type has changed (from group to non-group) | ||||
|                 if (bean.type === "group" && monitor.type !== bean.type) { | ||||
|                     removeGroupChildren = true; | ||||
|                 } | ||||
|  | ||||
|                 bean.name = monitor.name; | ||||
|                 bean.description = monitor.description; | ||||
|                 bean.parent = monitor.parent; | ||||
|                 bean.type = monitor.type; | ||||
|                 bean.url = monitor.url; | ||||
|                 bean.method = monitor.method; | ||||
| @@ -707,6 +722,9 @@ let needSetup = false; | ||||
|                 bean.headers = monitor.headers; | ||||
|                 bean.basic_auth_user = monitor.basic_auth_user; | ||||
|                 bean.basic_auth_pass = monitor.basic_auth_pass; | ||||
|                 bean.tlsCa = monitor.tlsCa; | ||||
|                 bean.tlsCert = monitor.tlsCert; | ||||
|                 bean.tlsKey = monitor.tlsKey; | ||||
|                 bean.interval = monitor.interval; | ||||
|                 bean.retryInterval = monitor.retryInterval; | ||||
|                 bean.resendInterval = monitor.resendInterval; | ||||
| @@ -753,14 +771,19 @@ let needSetup = false; | ||||
|                 bean.radiusCalledStationId = monitor.radiusCalledStationId; | ||||
|                 bean.radiusCallingStationId = monitor.radiusCallingStationId; | ||||
|                 bean.radiusSecret = monitor.radiusSecret; | ||||
|                 bean.httpBodyEncoding = monitor.httpBodyEncoding; | ||||
|  | ||||
|                 bean.validate(); | ||||
|  | ||||
|                 await R.store(bean); | ||||
|  | ||||
|                 if (removeGroupChildren) { | ||||
|                     await Monitor.unlinkAllChildren(monitor.id); | ||||
|                 } | ||||
|  | ||||
|                 await updateMonitorNotification(bean.id, monitor.notificationIDList); | ||||
|  | ||||
|                 if (bean.active) { | ||||
|                 if (bean.isActive()) { | ||||
|                     await restartMonitor(socket.userID, bean.id); | ||||
|                 } | ||||
|  | ||||
| @@ -906,11 +929,20 @@ let needSetup = false; | ||||
|                     delete server.monitorList[monitorID]; | ||||
|                 } | ||||
|  | ||||
|                 const startTime = Date.now(); | ||||
|  | ||||
|                 await R.exec("DELETE FROM monitor WHERE id = ? AND user_id = ? ", [ | ||||
|                     monitorID, | ||||
|                     socket.userID, | ||||
|                 ]); | ||||
|  | ||||
|                 // Fix #2880 | ||||
|                 apicache.clear(); | ||||
|  | ||||
|                 const endTime = Date.now(); | ||||
|  | ||||
|                 log.info("DB", `Delete Monitor completed in : ${endTime - startTime} ms`); | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: true, | ||||
|                     msg: "Deleted Successfully.", | ||||
| @@ -1154,6 +1186,8 @@ let needSetup = false; | ||||
|                     await doubleCheckPassword(socket, currentPassword); | ||||
|                 } | ||||
|  | ||||
|                 const previousChromeExecutable = await Settings.get("chromeExecutable"); | ||||
|  | ||||
|                 await setSettings("general", data); | ||||
|                 server.entryPage = data.entryPage; | ||||
|  | ||||
| @@ -1164,6 +1198,12 @@ let needSetup = false; | ||||
|                     await server.setTimezone(data.serverTimezone); | ||||
|                 } | ||||
|  | ||||
|                 // If Chrome Executable is changed, need to reset the browser | ||||
|                 if (previousChromeExecutable !== data.chromeExecutable) { | ||||
|                     log.info("settings", "Chrome executable is changed. Resetting Chrome..."); | ||||
|                     await resetChrome(); | ||||
|                 } | ||||
|  | ||||
|                 callback({ | ||||
|                     ok: true, | ||||
|                     msg: "Saved" | ||||
| @@ -1347,6 +1387,7 @@ let needSetup = false; | ||||
|                             let monitor = { | ||||
|                                 // Define the new variable from earlier here | ||||
|                                 name: monitorListData[i].name, | ||||
|                                 description: monitorListData[i].description, | ||||
|                                 type: monitorListData[i].type, | ||||
|                                 url: monitorListData[i].url, | ||||
|                                 method: monitorListData[i].method || "GET", | ||||
| @@ -1370,7 +1411,7 @@ let needSetup = false; | ||||
|                                 accepted_statuscodes: monitorListData[i].accepted_statuscodes, | ||||
|                                 dns_resolve_type: monitorListData[i].dns_resolve_type, | ||||
|                                 dns_resolve_server: monitorListData[i].dns_resolve_server, | ||||
|                                 notificationIDList: {}, | ||||
|                                 notificationIDList: monitorListData[i].notificationIDList, | ||||
|                                 proxy_id: monitorListData[i].proxy_id || null, | ||||
|                             }; | ||||
|  | ||||
| @@ -1530,6 +1571,7 @@ let needSetup = false; | ||||
|         proxySocketHandler(socket); | ||||
|         dockerSocketHandler(socket); | ||||
|         maintenanceSocketHandler(socket); | ||||
|         apiKeySocketHandler(socket); | ||||
|         generalSocketHandler(socket, server); | ||||
|         pluginsHandler(socket, server); | ||||
|  | ||||
| @@ -1575,7 +1617,7 @@ let needSetup = false; | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     initBackgroundJobs(args); | ||||
|     await initBackgroundJobs(); | ||||
|  | ||||
|     // Start cloudflared at the end if configured | ||||
|     await cloudflaredAutoStart(cloudflaredToken); | ||||
| @@ -1638,6 +1680,7 @@ async function afterLogin(socket, user) { | ||||
|     sendNotificationList(socket); | ||||
|     sendProxyList(socket); | ||||
|     sendDockerHostList(socket); | ||||
|     sendAPIKeyList(socket); | ||||
|  | ||||
|     await sleep(500); | ||||
|  | ||||
| @@ -1695,7 +1738,7 @@ async function initDatabase(testMode = false) { | ||||
|         needSetup = true; | ||||
|     } | ||||
|  | ||||
|     jwtSecret = jwtSecretBean.value; | ||||
|     server.jwtSecret = jwtSecretBean.value; | ||||
| } | ||||
|  | ||||
| /** | ||||
|   | ||||
							
								
								
									
										150
									
								
								server/socket-handlers/api-key-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								server/socket-handlers/api-key-socket-handler.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,150 @@ | ||||
| const { checkLogin } = require("../util-server"); | ||||
| const { log } = require("../../src/util"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { nanoid } = require("nanoid"); | ||||
| const passwordHash = require("../password-hash"); | ||||
| const apicache = require("../modules/apicache"); | ||||
| const APIKey = require("../model/api_key"); | ||||
| const { Settings } = require("../settings"); | ||||
| const { sendAPIKeyList } = require("../client"); | ||||
|  | ||||
| /** | ||||
|  * Handlers for Maintenance | ||||
|  * @param {Socket} socket Socket.io instance | ||||
|  */ | ||||
| module.exports.apiKeySocketHandler = (socket) => { | ||||
|     // Add a new api key | ||||
|     socket.on("addAPIKey", async (key, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             let clearKey = nanoid(40); | ||||
|             let hashedKey = passwordHash.generate(clearKey); | ||||
|             key["key"] = hashedKey; | ||||
|             let bean = await APIKey.save(key, socket.userID); | ||||
|  | ||||
|             log.debug("apikeys", "Added API Key"); | ||||
|             log.debug("apikeys", key); | ||||
|  | ||||
|             // Append key ID and prefix to start of key seperated by _, used to get | ||||
|             // correct hash when validating key. | ||||
|             let formattedKey = "uk" + bean.id + "_" + clearKey; | ||||
|             await sendAPIKeyList(socket); | ||||
|  | ||||
|             // Enable API auth if the user creates a key, otherwise only basic | ||||
|             // auth will be used for API. | ||||
|             await Settings.set("apiKeysEnabled", true); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Added Successfully.", | ||||
|                 key: formattedKey, | ||||
|                 keyID: bean.id, | ||||
|             }); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("getAPIKeyList", async (callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|             await sendAPIKeyList(socket); | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|             }); | ||||
|         } catch (e) { | ||||
|             console.error(e); | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("deleteAPIKey", async (keyID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("apikeys", `Deleted API Key: ${keyID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("DELETE FROM api_key WHERE id = ? AND user_id = ? ", [ | ||||
|                 keyID, | ||||
|                 socket.userID, | ||||
|             ]); | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Deleted Successfully.", | ||||
|             }); | ||||
|  | ||||
|             await sendAPIKeyList(socket); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("disableAPIKey", async (keyID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("apikeys", `Disabled Key: ${keyID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("UPDATE api_key SET active = 0 WHERE id = ? ", [ | ||||
|                 keyID, | ||||
|             ]); | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Disabled Successfully.", | ||||
|             }); | ||||
|  | ||||
|             await sendAPIKeyList(socket); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|     socket.on("enableAPIKey", async (keyID, callback) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             log.debug("apikeys", `Enabled Key: ${keyID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("UPDATE api_key SET active = 1 WHERE id = ? ", [ | ||||
|                 keyID, | ||||
|             ]); | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Enabled Successfully", | ||||
|             }); | ||||
|  | ||||
|             await sendAPIKeyList(socket); | ||||
|  | ||||
|         } catch (e) { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         } | ||||
|     }); | ||||
| }; | ||||
| @@ -3,16 +3,17 @@ const { Settings } = require("../settings"); | ||||
| const { sendInfo } = require("../client"); | ||||
| const { checkLogin } = require("../util-server"); | ||||
| const GameResolver = require("gamedig/lib/GameResolver"); | ||||
| const { testChrome } = require("../monitor-types/real-browser-monitor-type"); | ||||
|  | ||||
| let gameResolver = new GameResolver(); | ||||
| let gameList = null; | ||||
|  | ||||
| /** | ||||
|  * Get a game list via GameDig | ||||
|  * @returns {any[]} | ||||
|  * @returns {Object[]} list of games supported by GameDig | ||||
|  */ | ||||
| function getGameList() { | ||||
|     if (!gameList) { | ||||
|     if (gameList == null) { | ||||
|         gameList = gameResolver._readGames().games.sort((a, b) => { | ||||
|             if ( a.pretty < b.pretty ) { | ||||
|                 return -1; | ||||
| @@ -22,9 +23,8 @@ function getGameList() { | ||||
|             } | ||||
|             return 0; | ||||
|         }); | ||||
|     } else { | ||||
|         return gameList; | ||||
|     } | ||||
|     return gameList; | ||||
| } | ||||
|  | ||||
| module.exports.generalSocketHandler = (socket, server) => { | ||||
| @@ -48,4 +48,18 @@ module.exports.generalSocketHandler = (socket, server) => { | ||||
|         }); | ||||
|     }); | ||||
|  | ||||
|     socket.on("testChrome", (executable, callback) => { | ||||
|         // Just noticed that await call could block the whole socket.io server!!! Use pure promise instead. | ||||
|         testChrome(executable).then((version) => { | ||||
|             callback({ | ||||
|                 ok: true, | ||||
|                 msg: "Found Chromium/Chrome. Version: " + version, | ||||
|             }); | ||||
|         }).catch((e) => { | ||||
|             callback({ | ||||
|                 ok: false, | ||||
|                 msg: e.message, | ||||
|             }); | ||||
|         }); | ||||
|     }); | ||||
| }; | ||||
|   | ||||
| @@ -5,7 +5,6 @@ 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 | ||||
| @@ -19,10 +18,12 @@ module.exports.maintenanceSocketHandler = (socket) => { | ||||
|  | ||||
|             log.debug("maintenance", maintenance); | ||||
|  | ||||
|             let bean = Maintenance.jsonToBean(R.dispense("maintenance"), maintenance); | ||||
|             let bean = await Maintenance.jsonToBean(R.dispense("maintenance"), maintenance); | ||||
|             bean.user_id = socket.userID; | ||||
|             let maintenanceID = await R.store(bean); | ||||
|             await MaintenanceTimeslot.generateTimeslot(bean); | ||||
|  | ||||
|             server.maintenanceList[maintenanceID] = bean; | ||||
|             await bean.run(true); | ||||
|  | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
| @@ -45,17 +46,15 @@ module.exports.maintenanceSocketHandler = (socket) => { | ||||
|         try { | ||||
|             checkLogin(socket); | ||||
|  | ||||
|             let bean = await R.findOne("maintenance", " id = ? ", [ maintenance.id ]); | ||||
|             let bean = server.getMaintenance(maintenance.id); | ||||
|  | ||||
|             if (bean.user_id !== socket.userID) { | ||||
|                 throw new Error("Permission denied."); | ||||
|             } | ||||
|  | ||||
|             Maintenance.jsonToBean(bean, maintenance); | ||||
|  | ||||
|             await Maintenance.jsonToBean(bean, maintenance); | ||||
|             await R.store(bean); | ||||
|             await MaintenanceTimeslot.generateTimeslot(bean, null, true); | ||||
|  | ||||
|             await bean.run(true); | ||||
|             await server.sendMaintenanceList(socket); | ||||
|  | ||||
|             callback({ | ||||
| @@ -187,7 +186,7 @@ module.exports.maintenanceSocketHandler = (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 = ? ", [ | ||||
|             let monitors = await R.getAll("SELECT monitor.id FROM monitor_maintenance mm JOIN monitor ON mm.monitor_id = monitor.id WHERE mm.maintenance_id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|  | ||||
| @@ -236,6 +235,7 @@ module.exports.maintenanceSocketHandler = (socket) => { | ||||
|             log.debug("maintenance", `Delete Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             if (maintenanceID in server.maintenanceList) { | ||||
|                 server.maintenanceList[maintenanceID].stop(); | ||||
|                 delete server.maintenanceList[maintenanceID]; | ||||
|             } | ||||
|  | ||||
| @@ -267,9 +267,15 @@ module.exports.maintenanceSocketHandler = (socket) => { | ||||
|  | ||||
|             log.debug("maintenance", `Pause Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("UPDATE maintenance SET active = 0 WHERE id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|             let maintenance = server.getMaintenance(maintenanceID); | ||||
|  | ||||
|             if (!maintenance) { | ||||
|                 throw new Error("Maintenance not found"); | ||||
|             } | ||||
|  | ||||
|             maintenance.active = false; | ||||
|             await R.store(maintenance); | ||||
|             maintenance.stop(); | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
| @@ -294,9 +300,15 @@ module.exports.maintenanceSocketHandler = (socket) => { | ||||
|  | ||||
|             log.debug("maintenance", `Resume Maintenance: ${maintenanceID} User ID: ${socket.userID}`); | ||||
|  | ||||
|             await R.exec("UPDATE maintenance SET active = 1 WHERE id = ? ", [ | ||||
|                 maintenanceID, | ||||
|             ]); | ||||
|             let maintenance = server.getMaintenance(maintenanceID); | ||||
|  | ||||
|             if (!maintenance) { | ||||
|                 throw new Error("Maintenance not found"); | ||||
|             } | ||||
|  | ||||
|             maintenance.active = true; | ||||
|             await R.store(maintenance); | ||||
|             await maintenance.run(); | ||||
|  | ||||
|             apicache.clear(); | ||||
|  | ||||
|   | ||||
| @@ -276,7 +276,7 @@ module.exports.statusPageSocketHandler = (socket) => { | ||||
|             let statusPage = R.dispense("status_page"); | ||||
|             statusPage.slug = slug; | ||||
|             statusPage.title = title; | ||||
|             statusPage.theme = "light"; | ||||
|             statusPage.theme = "auto"; | ||||
|             statusPage.icon = ""; | ||||
|             await R.store(statusPage); | ||||
|  | ||||
|   | ||||
| @@ -47,8 +47,6 @@ class UptimeKumaServer { | ||||
|      */ | ||||
|     indexHTML = ""; | ||||
|  | ||||
|     generateMaintenanceTimeslotsInterval = undefined; | ||||
|  | ||||
|     /** | ||||
|      * Plugins Manager | ||||
|      * @type {PluginsManager} | ||||
| @@ -63,6 +61,12 @@ class UptimeKumaServer { | ||||
|  | ||||
|     }; | ||||
|  | ||||
|     /** | ||||
|      * Use for decode the auth object | ||||
|      * @type {null} | ||||
|      */ | ||||
|     jwtSecret = null; | ||||
|  | ||||
|     static getInstance(args) { | ||||
|         if (UptimeKumaServer.instance == null) { | ||||
|             UptimeKumaServer.instance = new UptimeKumaServer(args); | ||||
| @@ -74,6 +78,7 @@ class UptimeKumaServer { | ||||
|         // SSL | ||||
|         const sslKey = args["ssl-key"] || process.env.UPTIME_KUMA_SSL_KEY || process.env.SSL_KEY || undefined; | ||||
|         const sslCert = args["ssl-cert"] || process.env.UPTIME_KUMA_SSL_CERT || process.env.SSL_CERT || undefined; | ||||
|         const sslKeyPassphrase = args["ssl-key-passphrase"] || process.env.UPTIME_KUMA_SSL_KEY_PASSPHRASE || process.env.SSL_KEY_PASSPHRASE || undefined; | ||||
|  | ||||
|         log.info("server", "Creating express and socket.io instance"); | ||||
|         this.app = express(); | ||||
| @@ -81,7 +86,8 @@ class UptimeKumaServer { | ||||
|             log.info("server", "Server Type: HTTPS"); | ||||
|             this.httpServer = https.createServer({ | ||||
|                 key: fs.readFileSync(sslKey), | ||||
|                 cert: fs.readFileSync(sslCert) | ||||
|                 cert: fs.readFileSync(sslCert), | ||||
|                 passphrase: sslKeyPassphrase, | ||||
|             }, this.app); | ||||
|         } else { | ||||
|             log.info("server", "Server Type: HTTP"); | ||||
| @@ -98,11 +104,17 @@ class UptimeKumaServer { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Set Monitor Types | ||||
|         UptimeKumaServer.monitorTypeList["real-browser"] = new RealBrowserMonitorType(); | ||||
|  | ||||
|         this.io = new Server(this.httpServer); | ||||
|     } | ||||
|  | ||||
|     /** Initialise app after the database has been set up */ | ||||
|     async initAfterDatabaseReady() { | ||||
|         // Static | ||||
|         this.app.use("/screenshots", express.static(Database.screenshotDir)); | ||||
|  | ||||
|         await CacheableDnsHttpAgent.update(); | ||||
|  | ||||
|         process.env.TZ = await this.getTimezone(); | ||||
| @@ -110,8 +122,7 @@ class UptimeKumaServer { | ||||
|         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); | ||||
|         await this.loadMaintenanceList(); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -173,16 +184,33 @@ class UptimeKumaServer { | ||||
|      */ | ||||
|     async getMaintenanceJSONList(userID) { | ||||
|         let result = {}; | ||||
|         for (let maintenanceID in this.maintenanceList) { | ||||
|             result[maintenanceID] = await this.maintenanceList[maintenanceID].toJSON(); | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Load maintenance list and run | ||||
|      * @param userID | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     async loadMaintenanceList(userID) { | ||||
|         let maintenanceList = await R.findAll("maintenance", " ORDER BY end_date DESC, title", [ | ||||
|  | ||||
|         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(); | ||||
|             this.maintenanceList[maintenance.id] = maintenance; | ||||
|             maintenance.run(this); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|         return result; | ||||
|     getMaintenance(maintenanceID) { | ||||
|         if (this.maintenanceList[maintenanceID]) { | ||||
|             return this.maintenanceList[maintenanceID]; | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -238,7 +266,7 @@ class UptimeKumaServer { | ||||
|      * Attempt to get the current server timezone | ||||
|      * If this fails, fall back to environment variables and then make a | ||||
|      * guess. | ||||
|      * @returns {string} | ||||
|      * @returns {Promise<string>} | ||||
|      */ | ||||
|     async getTimezone() { | ||||
|         let timezone = await Settings.get("serverTimezone"); | ||||
| @@ -269,23 +297,9 @@ class UptimeKumaServer { | ||||
|         dayjs.tz.setDefault(timezone); | ||||
|     } | ||||
|  | ||||
|     /** Load the timeslots for maintenance */ | ||||
|     async generateMaintenanceTimeslots() { | ||||
|  | ||||
|         let list = await R.find("maintenance_timeslot", " generated_next = 0 AND start_date <= CURRENT_TIMESTAMP "); | ||||
|  | ||||
|         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); | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     /** Stop the server */ | ||||
|     async stop() { | ||||
|         clearTimeout(this.generateMaintenanceTimeslotsInterval); | ||||
|  | ||||
|     } | ||||
|  | ||||
|     loadPlugins() { | ||||
| @@ -334,5 +348,5 @@ module.exports = { | ||||
| }; | ||||
|  | ||||
| // Must be at the end | ||||
| const MaintenanceTimeslot = require("./model/maintenance_timeslot"); | ||||
| const { MonitorType } = require("./monitor-types/monitor-type"); | ||||
| const { RealBrowserMonitorType } = require("./monitor-types/real-browser-monitor-type"); | ||||
|   | ||||
| @@ -90,7 +90,10 @@ exports.ping = async (hostname, size = 56) => { | ||||
|         return await exports.pingAsync(hostname, false, size); | ||||
|     } catch (e) { | ||||
|         // If the host cannot be resolved, try again with ipv6 | ||||
|         if (e.message.includes("service not known")) { | ||||
|         console.debug("ping", "IPv6 error message: " + e.message); | ||||
|  | ||||
|         // As node-ping does not report a specific error for this, try again if it is an empty message with ipv6 no matter what. | ||||
|         if (!e.message) { | ||||
|             return await exports.pingAsync(hostname, true, size); | ||||
|         } else { | ||||
|             throw e; | ||||
| @@ -295,14 +298,23 @@ exports.postgresQuery = function (connectionString, query) { | ||||
|                 client.end(); | ||||
|             } else { | ||||
|                 // Connected here | ||||
|                 client.query(query, (err, res) => { | ||||
|                     if (err) { | ||||
|                         reject(err); | ||||
|                     } else { | ||||
|                         resolve(res); | ||||
|                 try { | ||||
|                     // No query provided by user, use SELECT 1 | ||||
|                     if (!query || (typeof query === "string" && query.trim() === "")) { | ||||
|                         query = "SELECT 1"; | ||||
|                     } | ||||
|                     client.end(); | ||||
|                 }); | ||||
|  | ||||
|                     client.query(query, (err, res) => { | ||||
|                         if (err) { | ||||
|                             reject(err); | ||||
|                         } else { | ||||
|                             resolve(res); | ||||
|                         } | ||||
|                         client.end(); | ||||
|                     }); | ||||
|                 } catch (e) { | ||||
|                     reject(e); | ||||
|                 } | ||||
|             } | ||||
|         }); | ||||
|  | ||||
| @@ -313,21 +325,33 @@ exports.postgresQuery = function (connectionString, query) { | ||||
|  * 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)>} | ||||
|  * @returns {Promise<(string)>} | ||||
|  */ | ||||
| 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 => { | ||||
|  | ||||
|         connection.on("error", (err) => { | ||||
|             reject(err); | ||||
|         }); | ||||
|  | ||||
|         connection.query(query, (err, res) => { | ||||
|             if (err) { | ||||
|                 reject(err); | ||||
|             }) | ||||
|             .finally(() => { | ||||
|             } else { | ||||
|                 if (Array.isArray(res)) { | ||||
|                     resolve("Rows: " + res.length); | ||||
|                 } else { | ||||
|                     resolve("No Error, but the result is not an array. Type: " + typeof res); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             try { | ||||
|                 connection.end(); | ||||
|             } catch (_) { | ||||
|                 connection.destroy(); | ||||
|             }); | ||||
|             } | ||||
|         }); | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| @@ -392,19 +416,28 @@ exports.radius = function ( | ||||
| exports.redisPingAsync = function (dsn) { | ||||
|     return new Promise((resolve, reject) => { | ||||
|         const client = redis.createClient({ | ||||
|             url: dsn, | ||||
|             url: dsn | ||||
|         }); | ||||
|         client.on("error", (err) => { | ||||
|             if (client.isOpen) { | ||||
|                 client.disconnect(); | ||||
|             } | ||||
|             reject(err); | ||||
|         }); | ||||
|         client.connect().then(() => { | ||||
|             if (!client.isOpen) { | ||||
|                 client.emit("error", new Error("connection isn't open")); | ||||
|             } | ||||
|             client.ping().then((res, err) => { | ||||
|                 if (client.isOpen) { | ||||
|                     client.disconnect(); | ||||
|                 } | ||||
|                 if (err) { | ||||
|                     reject(err); | ||||
|                 } else { | ||||
|                     resolve(res); | ||||
|                 } | ||||
|             }); | ||||
|             }).catch(error => reject(error)); | ||||
|         }); | ||||
|     }); | ||||
| }; | ||||
| @@ -500,12 +533,16 @@ const parseCertificateInfo = function (info) { | ||||
|  | ||||
|         // Move up the chain until loop is encountered | ||||
|         if (link.issuerCertificate == null) { | ||||
|             link.certType = (i === 0) ? "self-signed" : "root CA"; | ||||
|             break; | ||||
|         } else if (link.issuerCertificate.fingerprint in existingList) { | ||||
|             // a root CA certificate is typically "signed by itself"  (=> "self signed certificate") and thus the "issuerCertificate" is a reference to itself. | ||||
|             log.debug("cert", `[Last] ${link.issuerCertificate.fingerprint}`); | ||||
|             link.certType = (i === 0) ? "self-signed" : "root CA"; | ||||
|             link.issuerCertificate = null; | ||||
|             break; | ||||
|         } else { | ||||
|             link.certType = (i === 0) ? "server" : "intermediate CA"; | ||||
|             link = link.issuerCertificate; | ||||
|         } | ||||
|  | ||||
| @@ -734,15 +771,27 @@ exports.filterAndJoin = (parts, connector = "") => { | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Send a 403 response | ||||
|  * Send an Error response | ||||
|  * @param {Object} res Express response object | ||||
|  * @param {string} [msg=""] Message to send | ||||
|  */ | ||||
| module.exports.send403 = (res, msg = "") => { | ||||
|     res.status(403).json({ | ||||
|         "status": "fail", | ||||
|         "msg": msg, | ||||
|     }); | ||||
| module.exports.sendHttpError = (res, msg = "") => { | ||||
|     if (msg.includes("SQLITE_BUSY") || msg.includes("SQLITE_LOCKED")) { | ||||
|         res.status(503).json({ | ||||
|             "status": "fail", | ||||
|             "msg": msg, | ||||
|         }); | ||||
|     } else if (msg.toLowerCase().includes("not found")) { | ||||
|         res.status(404).json({ | ||||
|             "status": "fail", | ||||
|             "msg": msg, | ||||
|         }); | ||||
|     } else { | ||||
|         res.status(403).json({ | ||||
|             "status": "fail", | ||||
|             "msg": msg, | ||||
|         }); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| function timeObjectConvertTimezone(obj, timezone, timeObjectToUTC = true) { | ||||
|   | ||||
| @@ -266,6 +266,11 @@ optgroup { | ||||
|         background-color: $dark-bg2; | ||||
|     } | ||||
|  | ||||
|     .form-select:disabled { | ||||
|         color: rgba($dark-font-color, 0.7); | ||||
|         background-color: $dark-bg; | ||||
|     } | ||||
|  | ||||
|     .form-control, .form-select { | ||||
|         border-color: $dark-border-color; | ||||
|     } | ||||
| @@ -556,6 +561,31 @@ h5.settings-subheading::after { | ||||
|     border-bottom: 1px solid $dark-border-color; | ||||
| } | ||||
|  | ||||
|  | ||||
| $shadow-box-padding: 20px; | ||||
|  | ||||
| .shadow-box-with-fixed-bottom-bar { | ||||
|     padding-top: $shadow-box-padding; | ||||
|     padding-bottom: 0; | ||||
|     padding-right: $shadow-box-padding; | ||||
|     padding-left: $shadow-box-padding; | ||||
| } | ||||
|  | ||||
| .fixed-bottom-bar { | ||||
|     position: sticky; | ||||
|     bottom: 0; | ||||
|     margin-left: -$shadow-box-padding; | ||||
|     margin-right: -$shadow-box-padding; | ||||
|     z-index: 100; | ||||
|     background-color: rgba(white, 0.2); | ||||
|     backdrop-filter: blur(2px); | ||||
|     border-radius: 0 0 10px 10px; | ||||
|  | ||||
|     .dark & { | ||||
|         background-color: rgba($dark-header-bg, 0.9); | ||||
|     } | ||||
| } | ||||
|  | ||||
| // Localization | ||||
|  | ||||
| @import "localization.scss"; | ||||
|   | ||||
| @@ -1,6 +1,12 @@ | ||||
| @import "vars.scss"; | ||||
| @import "node_modules/vue-multiselect/dist/vue-multiselect"; | ||||
|  | ||||
| .multiselect { | ||||
|     .dark & { | ||||
|         color: $dark-font-color; | ||||
|     } | ||||
| } | ||||
|  | ||||
| .multiselect__tags { | ||||
|     border-radius: 1.5rem; | ||||
|     border: 1px solid #ced4da; | ||||
| @@ -14,10 +20,12 @@ | ||||
|  | ||||
| .multiselect__option--highlight { | ||||
|     background: $primary !important; | ||||
|     color: $dark-font-color2 !important; | ||||
| } | ||||
|  | ||||
| .multiselect__option--highlight::after { | ||||
|     background: $primary !important; | ||||
|     color: $dark-font-color2 !important; | ||||
| } | ||||
|  | ||||
| .multiselect__tag { | ||||
|   | ||||
							
								
								
									
										228
									
								
								src/components/APIKeyDialog.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										228
									
								
								src/components/APIKeyDialog.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,228 @@ | ||||
| <template> | ||||
|     <form @submit.prevent="submit"> | ||||
|         <div ref="keyaddmodal" class="modal fade" tabindex="-1" data-bs-backdrop="static"> | ||||
|             <div class="modal-dialog"> | ||||
|                 <div class="modal-content"> | ||||
|                     <div class="modal-header"> | ||||
|                         <h5 class="modal-title"> | ||||
|                             {{ $t("Add API Key") }} | ||||
|                         </h5> | ||||
|                         <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close" /> | ||||
|                     </div> | ||||
|                     <div class="modal-body"> | ||||
|                         <!-- Name --> | ||||
|                         <div class="mb-3"> | ||||
|                             <label for="name" class="form-label">{{ $t("Name") }}</label> | ||||
|                             <input | ||||
|                                 id="name" v-model="key.name" type="text" class="form-control" | ||||
|                                 required | ||||
|                             > | ||||
|                         </div> | ||||
|  | ||||
|                         <!-- Expiry --> | ||||
|                         <div class="my-3"> | ||||
|                             <label class="form-label">{{ $t("Expiry date") }}</label> | ||||
|                             <div class="d-flex flex-row align-items-center"> | ||||
|                                 <div class="col-6"> | ||||
|                                     <Datepicker | ||||
|                                         v-model="key.expires" | ||||
|                                         :dark="$root.isDark" | ||||
|                                         :monthChangeOnScroll="false" | ||||
|                                         :minDate="minDate" | ||||
|                                         format="yyyy-MM-dd HH:mm" | ||||
|                                         modelType="yyyy-MM-dd HH:mm:ss" | ||||
|                                         :required="!noExpire" | ||||
|                                         :disabled="noExpire" | ||||
|                                     /> | ||||
|                                 </div> | ||||
|                                 <div class="col-6 ms-3"> | ||||
|                                     <div class="form-check mb-0"> | ||||
|                                         <input | ||||
|                                             id="no-expire" v-model="noExpire" class="form-check-input" | ||||
|                                             type="checkbox" | ||||
|                                         > | ||||
|                                         <label class="form-check-label" for="no-expire">{{ | ||||
|                                             $t("Don't expire") | ||||
|                                         }}</label> | ||||
|                                     </div> | ||||
|                                 </div> | ||||
|                             </div> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                     <div class="modal-footer"> | ||||
|                         <button | ||||
|                             id="monitor-submit-btn" class="btn btn-primary" type="submit" | ||||
|                             :disabled="processing" | ||||
|                         > | ||||
|                             {{ $t("Generate") }} | ||||
|                         </button> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|         <div ref="keymodal" class="modal fade" tabindex="-1" data-bs-backdrop="static"> | ||||
|             <div class="modal-dialog"> | ||||
|                 <div class="modal-content"> | ||||
|                     <div class="modal-header"> | ||||
|                         <h5 class="modal-title"> | ||||
|                             {{ $t("Key Added") }} | ||||
|                         </h5> | ||||
|                         <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close" /> | ||||
|                     </div> | ||||
|  | ||||
|                     <div class="modal-body"> | ||||
|                         <div class="mb-3"> | ||||
|                             {{ $t("apiKeyAddedMsg") }} | ||||
|                         </div> | ||||
|                         <div class="mb-3"> | ||||
|                             <CopyableInput v-model="clearKey" disabled="disabled" /> | ||||
|                         </div> | ||||
|                     </div> | ||||
|  | ||||
|                     <div class="modal-footer"> | ||||
|                         <button type="button" class="btn btn-primary" data-bs-dismiss="modal"> | ||||
|                             {{ $t('Continue') }} | ||||
|                         </button> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     </form> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { Modal } from "bootstrap"; | ||||
| import { useToast } from "vue-toastification"; | ||||
| import dayjs from "dayjs"; | ||||
| import Datepicker from "@vuepic/vue-datepicker"; | ||||
| import CopyableInput from "./CopyableInput.vue"; | ||||
| const toast = useToast(); | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         CopyableInput, | ||||
|         Datepicker | ||||
|     }, | ||||
|     props: {}, | ||||
|     // emits: [ "added" ], | ||||
|     data() { | ||||
|         return { | ||||
|             keyaddmodal: null, | ||||
|             keymodal: null, | ||||
|             processing: false, | ||||
|             key: {}, | ||||
|             dark: (this.$root.theme === "dark"), | ||||
|             minDate: this.$root.date(dayjs()) + " 00:00", | ||||
|             clearKey: null, | ||||
|             noExpire: false, | ||||
|         }; | ||||
|     }, | ||||
|  | ||||
|     mounted() { | ||||
|         this.keyaddmodal = new Modal(this.$refs.keyaddmodal); | ||||
|         this.keymodal = new Modal(this.$refs.keymodal); | ||||
|     }, | ||||
|  | ||||
|     methods: { | ||||
|         /** | ||||
|          * Show modal | ||||
|          */ | ||||
|         show() { | ||||
|             this.id = null; | ||||
|             this.key = { | ||||
|                 name: "", | ||||
|                 expires: this.minDate, | ||||
|                 active: 1, | ||||
|             }; | ||||
|  | ||||
|             this.keyaddmodal.show(); | ||||
|         }, | ||||
|  | ||||
|         /** Submit data to server */ | ||||
|         async submit() { | ||||
|             this.processing = true; | ||||
|  | ||||
|             if (this.noExpire) { | ||||
|                 this.key.expires = null; | ||||
|             } | ||||
|  | ||||
|             this.$root.addAPIKey(this.key, async (res) => { | ||||
|                 this.keyaddmodal.hide(); | ||||
|                 this.processing = false; | ||||
|                 if (res.ok) { | ||||
|                     this.clearKey = res.key; | ||||
|                     this.keymodal.show(); | ||||
|                     this.clearForm(); | ||||
|                 } else { | ||||
|                     toast.error(res.msg); | ||||
|                 } | ||||
|             }); | ||||
|         }, | ||||
|  | ||||
|         /** Clear Form inputs */ | ||||
|         clearForm() { | ||||
|             this.key = { | ||||
|                 name: "", | ||||
|                 expires: this.minDate, | ||||
|                 active: 1, | ||||
|             }; | ||||
|             this.noExpire = false; | ||||
|         }, | ||||
|     } | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| @import "../assets/vars.scss"; | ||||
|  | ||||
| .dark { | ||||
|     .modal-dialog .form-text, .modal-dialog p { | ||||
|         color: $dark-font-color; | ||||
|     } | ||||
| } | ||||
|  | ||||
| .shadow-box { | ||||
|     padding: 20px; | ||||
| } | ||||
|  | ||||
| textarea { | ||||
|     min-height: 150px; | ||||
| } | ||||
|  | ||||
| .dark-calendar::-webkit-calendar-picker-indicator { | ||||
|     filter: invert(1); | ||||
| } | ||||
|  | ||||
| .weekday-picker { | ||||
|     display: flex; | ||||
|     gap: 10px; | ||||
|  | ||||
|     & > div { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|         width: 40px; | ||||
|  | ||||
|         .form-check-inline { | ||||
|             margin-right: 0; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| .day-picker { | ||||
|     display: flex; | ||||
|     gap: 10px; | ||||
|     flex-wrap: wrap; | ||||
|  | ||||
|     & > div { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|         width: 40px; | ||||
|  | ||||
|         .form-check-inline { | ||||
|             margin-right: 0; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										299
									
								
								src/components/BadgeGeneratorDialog.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										299
									
								
								src/components/BadgeGeneratorDialog.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,299 @@ | ||||
| <template> | ||||
|     <div ref="BadgeGeneratorModal" class="modal fade" tabindex="-1" data-bs-backdrop="static"> | ||||
|         <div class="modal-dialog"> | ||||
|             <div class="modal-content"> | ||||
|                 <div class="modal-header"> | ||||
|                     <h5 class="modal-title"> | ||||
|                         {{ $t("Badge Generator", [monitor.name]) }} | ||||
|                     </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="type" class="form-label">{{ $t("Badge Type") }}</label> | ||||
|                         <select id="type" v-model="badge.type" class="form-select"> | ||||
|                             <option value="status">status</option> | ||||
|                             <option value="uptime">uptime</option> | ||||
|                             <option value="ping">ping</option> | ||||
|                             <option value="avg-response">avg-response</option> | ||||
|                             <option value="cert-exp">cert-exp</option> | ||||
|                             <option value="response">response</option> | ||||
|                         </select> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('duration') " class="mb-3"> | ||||
|                         <label for="duration" class="form-label">{{ $t("Badge Duration") }}</label> | ||||
|                         <input id="duration" v-model="badge.duration" type="number" min="0" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('label') " class="mb-3"> | ||||
|                         <label for="label" class="form-label">{{ $t("Badge Label") }}</label> | ||||
|                         <input id="label" v-model="badge.label" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('prefix') " class="mb-3"> | ||||
|                         <label for="prefix" class="form-label">{{ $t("Badge Prefix") }}</label> | ||||
|                         <input id="prefix" v-model="badge.prefix" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('suffix') " class="mb-3"> | ||||
|                         <label for="suffix" class="form-label">{{ $t("Badge Suffix") }}</label> | ||||
|                         <input id="suffix" v-model="badge.suffix" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('labelColor') " class="mb-3"> | ||||
|                         <label for="labelColor" class="form-label">{{ $t("Badge Label Color") }}</label> | ||||
|                         <input id="labelColor" v-model="badge.labelColor" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('color') " class="mb-3"> | ||||
|                         <label for="color" class="form-label">{{ $t("Badge Color") }}</label> | ||||
|                         <input id="color" v-model="badge.color" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('labelPrefix') " class="mb-3"> | ||||
|                         <label for="labelPrefix" class="form-label">{{ $t("Badge Label Prefix") }}</label> | ||||
|                         <input id="labelPrefix" v-model="badge.labelPrefix" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('labelSuffix') " class="mb-3"> | ||||
|                         <label for="labelSuffix" class="form-label">{{ $t("Badge Label Suffix") }}</label> | ||||
|                         <input id="labelSuffix" v-model="badge.labelSuffix" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('upColor') " class="mb-3"> | ||||
|                         <label for="upColor" class="form-label">{{ $t("Badge Up Color") }}</label> | ||||
|                         <input id="upColor" v-model="badge.upColor" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('downColor') " class="mb-3"> | ||||
|                         <label for="downColor" class="form-label">{{ $t("Badge Down Color") }}</label> | ||||
|                         <input id="downColor" v-model="badge.downColor" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('pendingColor') " class="mb-3"> | ||||
|                         <label for="pendingColor" class="form-label">{{ $t("Badge Pending Color") }}</label> | ||||
|                         <input id="pendingColor" v-model="badge.pendingColor" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('maintenanceColor') " class="mb-3"> | ||||
|                         <label for="maintenanceColor" class="form-label">{{ $t("Badge Maintenance Color") }}</label> | ||||
|                         <input id="maintenanceColor" v-model="badge.maintenanceColor" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('warnColor') " class="mb-3"> | ||||
|                         <label for="warnColor" class="form-label">{{ $t("Badge Warn Color") }}</label> | ||||
|                         <input id="warnColor" v-model="badge.warnColor" type="number" min="0" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('warnDays') " class="mb-3"> | ||||
|                         <label for="warnDays" class="form-label">{{ $t("Badge Warn Days") }}</label> | ||||
|                         <input id="warnDays" v-model="badge.warnDays" type="number" min="0" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div v-if=" (parameters[badge.type || 'null'] || [] ).includes('downDays') " class="mb-3"> | ||||
|                         <label for="downDays" class="form-label">{{ $t("Badge Down Days") }}</label> | ||||
|                         <input id="downDays" v-model="badge.downDays" type="number" min="0" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div class="mb-3"> | ||||
|                         <label for="style" class="form-label">{{ $t("Badge Style") }}</label> | ||||
|                         <select id="style" v-model="badge.style" class="form-select"> | ||||
|                             <option value="plastic">plastic</option> | ||||
|                             <option value="flat">flat</option> | ||||
|                             <option value="flat-square">flat-square</option> | ||||
|                             <option value="for-the-badge">for-the-badge</option> | ||||
|                             <option value="social">social</option> | ||||
|                         </select> | ||||
|                     </div> | ||||
|  | ||||
|                     <div class="mb-3"> | ||||
|                         <label for="value" class="form-label">{{ $t("Badge value (For Testing only.)") }}</label> | ||||
|                         <input id="value" v-model="badge.value" type="text" class="form-control" required> | ||||
|                     </div> | ||||
|  | ||||
|                     <div class="my-3"> | ||||
|                         <label for="push-url" class="form-label">{{ $t("Badge URL") }}</label> | ||||
|                         <CopyableInput id="push-url" v-model="badgeURL" type="url" disabled="disabled" /> | ||||
|                     </div> | ||||
|                 </div> | ||||
|  | ||||
|                 <div class="modal-footer"> | ||||
|                     <button type="submit" class="btn btn-danger" data-bs-dismiss="modal"> | ||||
|                         {{ $t("Close") }} | ||||
|                     </button> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <script lang="ts"> | ||||
| import { Modal } from "bootstrap"; | ||||
| import CopyableInput from "./CopyableInput.vue"; | ||||
|  | ||||
| export default { | ||||
|     components: { | ||||
|         CopyableInput | ||||
|     }, | ||||
|     props: {}, | ||||
|     emits: [], | ||||
|     data() { | ||||
|         return { | ||||
|             model: null, | ||||
|             processing: false, | ||||
|             monitor: { | ||||
|                 id: null, | ||||
|                 name: null, | ||||
|             }, | ||||
|             badge: { | ||||
|                 type: "status", | ||||
|                 duration: null, | ||||
|                 label: null, | ||||
|                 prefix: null, | ||||
|                 suffix: null, | ||||
|                 labelColor: null, | ||||
|                 color: null, | ||||
|                 labelPrefix: null, | ||||
|                 labelSuffix: null, | ||||
|                 upColor: null, | ||||
|                 downColor: null, | ||||
|                 pendingColor: null, | ||||
|                 maintenanceColor: null, | ||||
|                 warnColor: null, | ||||
|                 warnDays: null, | ||||
|                 downDays: null, | ||||
|                 style: "flat", | ||||
|                 value: null, | ||||
|             }, | ||||
|             parameters: { | ||||
|                 status: [ | ||||
|                     "upLabel", | ||||
|                     "downLabel", | ||||
|                     "pendingLabel", | ||||
|                     "maintenanceLabel", | ||||
|                     "upColor", | ||||
|                     "downColor", | ||||
|                     "pendingColor", | ||||
|                     "maintenanceColor", | ||||
|                 ], | ||||
|                 uptime: [ | ||||
|                     "duration", | ||||
|                     "labelPrefix", | ||||
|                     "labelSuffix", | ||||
|                     "prefix", | ||||
|                     "suffix", | ||||
|                     "color", | ||||
|                     "labelColor", | ||||
|                 ], | ||||
|                 ping: [ | ||||
|                     "duration", | ||||
|                     "labelPrefix", | ||||
|                     "labelSuffix", | ||||
|                     "prefix", | ||||
|                     "suffix", | ||||
|                     "color", | ||||
|                     "labelColor", | ||||
|                 ], | ||||
|                 "avg-response": [ | ||||
|                     "duration", | ||||
|                     "labelPrefix", | ||||
|                     "labelSuffix", | ||||
|                     "prefix", | ||||
|                     "suffix", | ||||
|                     "color", | ||||
|                     "labelColor", | ||||
|                 ], | ||||
|                 "cert-exp": [ | ||||
|                     "labelPrefix", | ||||
|                     "labelSuffix", | ||||
|                     "prefix", | ||||
|                     "suffix", | ||||
|                     "upColor", | ||||
|                     "warnColor", | ||||
|                     "downColor", | ||||
|                     "warnDays", | ||||
|                     "downDays", | ||||
|                     "labelColor", | ||||
|                 ], | ||||
|                 response: [ | ||||
|                     "labelPrefix", | ||||
|                     "labelSuffix", | ||||
|                     "prefix", | ||||
|                     "suffix", | ||||
|                     "color", | ||||
|                     "labelColor", | ||||
|                 ], | ||||
|             } | ||||
|         }; | ||||
|     }, | ||||
|  | ||||
|     computed: { | ||||
|         badgeURL() { | ||||
|             if (!this.monitor.id || !this.badge.type) { | ||||
|                 return; | ||||
|             } | ||||
|             let badgeURL = this.$root.baseURL + "/api/badge/" + this.monitor.id + "/" + this.badge.type; | ||||
|  | ||||
|             let parameterList = {}; | ||||
|  | ||||
|             for (let parameter of this.parameters[this.badge.type] || []) { | ||||
|                 if (parameter === "duration" && this.badge.duration) { | ||||
|                     badgeURL += "/" + this.badge.duration; | ||||
|                     continue; | ||||
|                 } | ||||
|  | ||||
|                 if (this.badge[parameter]) { | ||||
|                     parameterList[parameter] = this.badge[parameter]; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             for (let parameter of [ "label", "style", "value" ]) { | ||||
|                 if (parameter === "style" && this.badge.style === "flat") { | ||||
|                     continue; | ||||
|                 } | ||||
|  | ||||
|                 if (this.badge[parameter]) { | ||||
|                     parameterList[parameter] = this.badge[parameter]; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (Object.keys(parameterList).length > 0) { | ||||
|                 return badgeURL + "?" + new URLSearchParams(parameterList); | ||||
|             } | ||||
|  | ||||
|             return badgeURL; | ||||
|         }, | ||||
|     }, | ||||
|  | ||||
|     mounted() { | ||||
|         this.BadgeGeneratorModal = new Modal(this.$refs.BadgeGeneratorModal); | ||||
|     }, | ||||
|  | ||||
|     methods: { | ||||
|         /** | ||||
|          * Setting monitor | ||||
|          * @param {number} monitorId    ID of monitor | ||||
|          * @param {string} monitorName  Name of monitor | ||||
|          */ | ||||
|         show(monitorId, monitorName) { | ||||
|             this.monitor = { | ||||
|                 id: monitorId, | ||||
|                 name: monitorName, | ||||
|             }; | ||||
|  | ||||
|             this.BadgeGeneratorModal.show(); | ||||
|         }, | ||||
|     }, | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| @import "../assets/vars.scss"; | ||||
|  | ||||
| .dark { | ||||
|     .modal-dialog .form-text, .modal-dialog p { | ||||
|         color: $dark-font-color; | ||||
|     } | ||||
| } | ||||
| </style> | ||||
| @@ -4,7 +4,7 @@ | ||||
|             <div class="modal-content"> | ||||
|                 <div class="modal-header"> | ||||
|                     <h5 id="exampleModalLabel" class="modal-title"> | ||||
|                         {{ $t("Confirm") }} | ||||
|                         {{ title || $t("Confirm") }} | ||||
|                     </h5> | ||||
|                     <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close" /> | ||||
|                 </div> | ||||
| @@ -15,7 +15,7 @@ | ||||
|                     <button type="button" class="btn" :class="btnStyle" data-bs-dismiss="modal" @click="yes"> | ||||
|                         {{ yesText }} | ||||
|                     </button> | ||||
|                     <button type="button" class="btn btn-secondary" data-bs-dismiss="modal"> | ||||
|                     <button type="button" class="btn btn-secondary" data-bs-dismiss="modal" @click="no"> | ||||
|                         {{ noText }} | ||||
|                     </button> | ||||
|                 </div> | ||||
| @@ -44,8 +44,13 @@ export default { | ||||
|             type: String, | ||||
|             default: "No", | ||||
|         }, | ||||
|         /** Title to show on modal. Defaults to translated version of "Config" */ | ||||
|         title: { | ||||
|             type: String, | ||||
|             default: null, | ||||
|         } | ||||
|     }, | ||||
|     emits: [ "yes" ], | ||||
|     emits: [ "yes", "no" ], | ||||
|     data: () => ({ | ||||
|         modal: null, | ||||
|     }), | ||||
| @@ -63,6 +68,12 @@ export default { | ||||
|         yes() { | ||||
|             this.$emit("yes"); | ||||
|         }, | ||||
|         /** | ||||
|          * @emits string "no" Notify the parent when No is pressed | ||||
|          */ | ||||
|         no() { | ||||
|             this.$emit("no"); | ||||
|         } | ||||
|     }, | ||||
| }; | ||||
| </script> | ||||
|   | ||||
| @@ -13,6 +13,9 @@ | ||||
|             :disabled="disabled" | ||||
|         > | ||||
|  | ||||
|         <!-- A hidden textarea for copying text on non-https --> | ||||
|         <textarea ref="hiddenTextarea" style="position: fixed; left: -999999px; top: -999999px;"></textarea> | ||||
|  | ||||
|         <a class="btn btn-outline-primary" @click="copyToClipboard(model)"> | ||||
|             <font-awesome-icon :icon="icon" /> | ||||
|         </a> | ||||
| @@ -111,24 +114,19 @@ export default { | ||||
|             }, 3000); | ||||
|  | ||||
|             // navigator clipboard api needs a secure context (https) | ||||
|             // For http, use the text area method (else part) | ||||
|             if (navigator.clipboard && window.isSecureContext) { | ||||
|                 // navigator clipboard api method' | ||||
|                 return navigator.clipboard.writeText(textToCopy); | ||||
|             } else { | ||||
|                 // text area method | ||||
|                 let textArea = document.createElement("textarea"); | ||||
|                 let textArea = this.$refs.hiddenTextarea; | ||||
|                 textArea.value = textToCopy; | ||||
|                 // make the textarea out of viewport | ||||
|                 textArea.style.position = "fixed"; | ||||
|                 textArea.style.left = "-999999px"; | ||||
|                 textArea.style.top = "-999999px"; | ||||
|                 document.body.appendChild(textArea); | ||||
|                 textArea.focus(); | ||||
|                 textArea.select(); | ||||
|                 return new Promise((res, rej) => { | ||||
|                     // here the magic happens | ||||
|                     document.execCommand("copy") ? res() : rej(); | ||||
|                     textArea.remove(); | ||||
|                 }); | ||||
|             } | ||||
|         } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user