mirror of
				https://github.com/louislam/uptime-kuma.git
				synced 2025-10-25 07:39:22 +08:00 
			
		
		
		
	Merge branch 'master' into feature/opsgenie-alerts
This commit is contained in:
		| @@ -1,6 +1,7 @@ | ||||
| /.idea | ||||
| /node_modules | ||||
| /data | ||||
| /cypress | ||||
| /out | ||||
| /test | ||||
| /kubernetes | ||||
| @@ -30,6 +31,9 @@ tsconfig.json | ||||
| /tmp | ||||
| /babel.config.js | ||||
| /ecosystem.config.js | ||||
| /extra/healthcheck.exe | ||||
| /extra/healthcheck | ||||
| extra/exe-builder | ||||
|  | ||||
| ### .gitignore content (commented rules are duplicated) | ||||
|  | ||||
| @@ -44,6 +48,4 @@ dist-ssr | ||||
| #!/data/.gitkeep | ||||
| #.vscode | ||||
|  | ||||
|  | ||||
|  | ||||
| ### End of .gitignore content | ||||
|   | ||||
| @@ -19,3 +19,6 @@ indent_size = 2 | ||||
|  | ||||
| [*.vue] | ||||
| trim_trailing_whitespace = false | ||||
|  | ||||
| [*.go] | ||||
| indent_style = tab | ||||
|   | ||||
							
								
								
									
										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: | ||||
|  | ||||
							
								
								
									
										7
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,8 @@ | ||||
| 👉 Delete this line if you have read and agree our pull request rules and guidelines: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md#can-i-create-a-pull-request-for-uptime-kuma | ||||
| ⚠️⚠️⚠️ Since we do not accept all types of pull requests and do not want to waste your time. Please be sure that you have read pull request rules: | ||||
| https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md#can-i-create-a-pull-request-for-uptime-kuma | ||||
|  | ||||
| Tick the checkbox if you understand [x]:  | ||||
| - [ ] I have read and understand the pull request rules. | ||||
|  | ||||
| # Description | ||||
|  | ||||
| @@ -12,7 +16,6 @@ Please delete any options that are not relevant. | ||||
| - User interface (UI) | ||||
| - New feature (non-breaking change which adds functionality) | ||||
| - Breaking change (fix or feature that would cause existing functionality to not work as expected) | ||||
| - Translation update | ||||
| - Other | ||||
| - This change requires a documentation update | ||||
|  | ||||
|   | ||||
							
								
								
									
										39
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,8 +6,12 @@ name: Auto Test | ||||
| on: | ||||
|   push: | ||||
|     branches: [ master ] | ||||
|     paths-ignore: | ||||
|       - '*.md' | ||||
|   pull_request: | ||||
|     branches: [ master ] | ||||
|     paths-ignore: | ||||
|       - '*.md' | ||||
|  | ||||
| jobs: | ||||
|   auto-test: | ||||
| @@ -18,7 +22,7 @@ jobs: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [macos-latest, ubuntu-latest, windows-latest] | ||||
|         node: [ 14, 16, 17, 18 ] | ||||
|         node: [ 14, 16, 18, 19 ] | ||||
|         # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ | ||||
|  | ||||
|     steps: | ||||
| @@ -36,6 +40,7 @@ jobs: | ||||
|       env: | ||||
|         HEADLESS_TEST: 1 | ||||
|         JUST_FOR_TEST: ${{ secrets.JUST_FOR_TEST }} | ||||
|  | ||||
|   check-linters: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
| @@ -50,3 +55,35 @@ jobs: | ||||
|         cache: 'npm' | ||||
|     - run: npm install | ||||
|     - run: npm run lint | ||||
|  | ||||
|   e2e-tests: | ||||
|     needs: [ check-linters ] | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - run: git config --global core.autocrlf false  # Mainly for Windows | ||||
|     - uses: actions/checkout@v3 | ||||
|  | ||||
|     - name: Use Node.js 14 | ||||
|       uses: actions/setup-node@v3 | ||||
|       with: | ||||
|         node-version: 14 | ||||
|         cache: 'npm' | ||||
|     - run: npm install | ||||
|     - run: npm run build | ||||
|     - run: npm run cy:test | ||||
|  | ||||
|   frontend-unit-tests: | ||||
|     needs: [ check-linters ] | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - run: git config --global core.autocrlf false  # Mainly for Windows | ||||
|     - uses: actions/checkout@v3 | ||||
|  | ||||
|     - name: Use Node.js 14 | ||||
|       uses: actions/setup-node@v3 | ||||
|       with: | ||||
|         node-version: 14 | ||||
|         cache: 'npm' | ||||
|     - run: npm install | ||||
|     - run: npm run build | ||||
|     - run: npm run cy:run:unit | ||||
|   | ||||
							
								
								
									
										7
									
								
								.github/workflows/close-incorrect-issue.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/close-incorrect-issue.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,3 @@ | ||||
|  | ||||
| name: Close Incorrect Issue | ||||
|  | ||||
| on: | ||||
| @@ -12,13 +11,13 @@ jobs: | ||||
|     strategy: | ||||
|       matrix: | ||||
|         os: [ubuntu-latest] | ||||
|         node-version: [16.x] | ||||
|         node-version: [16] | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|     - uses: actions/checkout@v3 | ||||
|  | ||||
|     - name: Use Node.js ${{ matrix.node-version }} | ||||
|       uses: actions/setup-node@v2 | ||||
|       uses: actions/setup-node@v3 | ||||
|       with: | ||||
|         node-version: ${{ matrix.node-version }} | ||||
|         cache: 'npm' | ||||
|   | ||||
							
								
								
									
										22
									
								
								.github/workflows/stale-bot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/stale-bot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| name: 'Automatically close stale issues and PRs' | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: '0 */6 * * *' | ||||
| #Run every 6 hours | ||||
|  | ||||
| jobs: | ||||
|   stale: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/stale@v7 | ||||
|         with: | ||||
|           stale-issue-message: 'We are clearing up our old issues and your ticket has been open for 3 months with no activity. Remove stale label or comment or this will be closed in 2 days.' | ||||
|           close-issue-message: 'This issue was closed because it has been stalled for 2 days with no activity.' | ||||
|           days-before-stale: 90 | ||||
|           days-before-close: 2 | ||||
|           days-before-pr-stale: 999999999 | ||||
|           days-before-pr-close: 1 | ||||
|           exempt-issue-labels: 'News,Medium,High,discussion,bug,doc,feature-request' | ||||
|           exempt-issue-assignees: 'louislam' | ||||
|           operations-per-run: 200 | ||||
							
								
								
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -13,3 +13,13 @@ dist-ssr | ||||
| /out | ||||
| /tmp | ||||
| .env | ||||
|  | ||||
| cypress/videos | ||||
| cypress/screenshots | ||||
|  | ||||
| /extra/healthcheck.exe | ||||
| /extra/healthcheck | ||||
| /extra/healthcheck-armv7 | ||||
|  | ||||
| extra/exe-builder/bin | ||||
| extra/exe-builder/obj | ||||
|   | ||||
| @@ -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 | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| # Project Info | ||||
|  | ||||
| First of all, thank you everyone who made pull requests for Uptime Kuma, I never thought GitHub Community can be that nice! And also because of this, I also never thought other people actually read my code and edit my code. It is not structured and commented so well, lol. Sorry about that. | ||||
| First of all, I want to thank everyone who made pull requests for Uptime Kuma. I never thought the GitHub Community would be so nice! Because of this, I also never thought that other people would actually read and edit my code. It is not very well structured or commented, sorry about that. | ||||
|  | ||||
| The project was created with vite.js (vue3). Then I created a subdirectory called "server" for server part. Both frontend and backend share the same package.json. | ||||
|  | ||||
| @@ -17,8 +17,11 @@ The frontend code build into "dist" directory. The server (express.js) exposes t | ||||
|  | ||||
| ## Directories | ||||
|  | ||||
| - config (dev config files) | ||||
| - data (App data) | ||||
| - db (Base database and migration scripts) | ||||
| - dist (Frontend build) | ||||
| - docker (Dockerfiles) | ||||
| - extra (Extra useful scripts) | ||||
| - public (Frontend resources for dev only) | ||||
| - server (Server source code) | ||||
| @@ -27,28 +30,40 @@ The frontend code build into "dist" directory. The server (express.js) exposes t | ||||
|  | ||||
| ## Can I create a pull request for Uptime Kuma? | ||||
|  | ||||
| Yes, you can. However, since I don't want to waste your time, be sure to **create empty draft pull request, so we can discuss first** if it is a large pull request or you don't know it will be merged or not. | ||||
| Yes or no, it depends on what you will try to do. Since I don't want to waste your time, be sure to **create an empty draft pull request or open an issue, so we can have a discussion first**. Especially for a large pull request or you don't know it will be merged or not. | ||||
|  | ||||
| Also, please don't rush or ask for ETA, because I have to understand the pull request, make sure it is no breaking changes and stick to my vision of this project, especially for large pull requests. | ||||
| Here are some references: | ||||
|  | ||||
| I will mark your pull request in the [milestones](https://github.com/louislam/uptime-kuma/milestones), if I am plan to review and merge it. | ||||
|  | ||||
| ✅ Accept: | ||||
| - Bug/Security fix | ||||
| - Translations | ||||
| ✅ Usually Accept: | ||||
| - Bug fix | ||||
| - Security fix | ||||
| - Adding notification providers | ||||
| - Adding new language files (You should go to https://weblate.kuma.pet for existing languages) | ||||
| - Adding new language keys: `$t("...")` | ||||
|  | ||||
| ⚠️ Discussion First | ||||
| - Large pull requests | ||||
| - New features | ||||
|  | ||||
| ❌ Won't Merge | ||||
| - A dedicated pr for translating existing languages (You can now translate on https://weblate.kuma.pet)  | ||||
| - Do not pass auto test | ||||
| - Any breaking changes | ||||
| - Duplicated pull request | ||||
| - Buggy | ||||
| - UI/UX is not close to Uptime Kuma  | ||||
| - Existing logic is completely modified or deleted for no reason | ||||
| - A function that is completely out of scope | ||||
| - Convert existing code into other programming languages | ||||
| - Unnecessary large code changes (Hard to review, causes code conflicts to other pull requests) | ||||
|  | ||||
| The above cases cannot cover all 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. | ||||
|  | ||||
| I will mark your pull request in the [milestones](https://github.com/louislam/uptime-kuma/milestones), if I am plan to review and merge it. | ||||
|  | ||||
| Also, please don't rush or ask for ETA, because I have to understand the pull request, make sure it is no breaking changes and stick to my vision of this project, especially for large pull requests. | ||||
|  | ||||
|  | ||||
| ### Recommended Pull Request Guideline | ||||
| @@ -68,13 +83,13 @@ Before deep into coding, discussion first is preferred. Creating an empty pull r | ||||
|  | ||||
| ## Project Styles | ||||
|  | ||||
| I personally do not like something need to learn so much and need to config so much before you can finally start the app. | ||||
| I personally do not like something that requires so many configurations before you can finally start the app. I hope Uptime Kuma installation could be as easy as like installing a mobile app. | ||||
|  | ||||
| - Easy to install for non-Docker users, no native build dependency is needed (at least for x86_64), no extra config, no extra effort to get it run | ||||
| - Easy to install for non-Docker users, no native build dependency is needed (for x86_64/armv7/arm64), no extra config, no extra effort required to get it running | ||||
| - Single container for Docker users, no very complex docker-compose file. Just map the volume and expose the port, then good to go | ||||
| - Settings should be configurable in the frontend. Environment variable is not encouraged, unless it is related to startup such as `DATA_DIR`. | ||||
| - Settings should be configurable in the frontend. Environment variable is not encouraged, unless it is related to startup such as `DATA_DIR` | ||||
| - Easy to use | ||||
| - The web UI styling should be consistent and nice. | ||||
| - The web UI styling should be consistent and nice | ||||
|  | ||||
| ## Coding Styles | ||||
|  | ||||
| @@ -83,7 +98,7 @@ I personally do not like something need to learn so much and need to config so m | ||||
| - Follow ESLint | ||||
| - Methods and functions should be documented with JSDoc | ||||
|  | ||||
| ## Name convention | ||||
| ## Name Conventions | ||||
|  | ||||
| - Javascript/Typescript: camelCaseType | ||||
| - SQLite: snake_case (Underscore) | ||||
| @@ -97,7 +112,7 @@ I personally do not like something need to learn so much and need to config so m | ||||
| - IDE that supports ESLint and EditorConfig (I am using IntelliJ IDEA) | ||||
| - A SQLite GUI tool (SQLite Expert Personal is suggested) | ||||
|  | ||||
| ## Install dependencies | ||||
| ## Install Dependencies for Development | ||||
|  | ||||
| ```bash | ||||
| npm ci | ||||
| @@ -115,6 +130,12 @@ Port `3000` and port `3001` will be used. | ||||
| npm run dev | ||||
| ``` | ||||
|  | ||||
| But sometimes, you would like to keep restart the server, but not the frontend, you can run these command in two terminals: | ||||
| ``` | ||||
| npm run start-frontend-dev | ||||
| npm run start-server-dev | ||||
| ``` | ||||
|  | ||||
| ## Backend Server | ||||
|  | ||||
| It binds to `0.0.0.0:3001` by default. | ||||
| @@ -130,12 +151,15 @@ express.js is used for: | ||||
|  | ||||
| ### Structure in /server/ | ||||
|  | ||||
| - jobs/ (Jobs that are running in another process) | ||||
| - model/ (Object model, auto mapping to the database table name) | ||||
| - modules/ (Modified 3rd-party modules) | ||||
| - monitor_types (Monitor Types) | ||||
| - notification-providers/ (individual notification logic) | ||||
| - routers/ (Express Routers) | ||||
| - socket-handler (Socket.io Handlers) | ||||
| - server.js (Server entry point and main logic) | ||||
| - server.js (Server entry point) | ||||
| - uptime-kuma-server.js (UptimeKumaServer class, main logic should be here, but some still in `server.js`) | ||||
|  | ||||
| ## Frontend Dev Server | ||||
|  | ||||
| @@ -168,29 +192,30 @@ The data and socket logic are in `src/mixins/socket.js`. | ||||
|  | ||||
| ## Unit Test | ||||
|  | ||||
| It is an end-to-end testing. It is using Jest and Puppeteer. | ||||
|  | ||||
| ```bash | ||||
| npm run build | ||||
| npm test | ||||
| ``` | ||||
|  | ||||
| By default, the Chromium window will be shown up during the test. Specifying `HEADLESS_TEST=1` for terminal environments. | ||||
| ## Dependencies | ||||
|  | ||||
| ## Update Dependencies | ||||
| Both frontend and backend share the same package.json. However, the frontend dependencies are eventually not used in the production environment, because it is usually also baked into dist files. So: | ||||
|  | ||||
| Install `ncu` | ||||
| https://github.com/raineorshine/npm-check-updates | ||||
| - Frontend dependencies = "devDependencies" | ||||
|   - Examples: vue, chart.js | ||||
| - Backend dependencies = "dependencies" | ||||
|   - Examples: socket.io, sqlite3 | ||||
| - Development dependencies = "devDependencies" | ||||
|   - Examples: eslint, sass | ||||
|  | ||||
| ```bash | ||||
| ncu -u -t patch | ||||
| npm install | ||||
| ``` | ||||
| ### Update Dependencies | ||||
|  | ||||
| Since previously updating Vite 2.5.10 to 2.6.0 broke the application completely, from now on, it should update patch release version only. | ||||
|  | ||||
| Patch release = the third digit ([Semantic Versioning](https://semver.org/)) | ||||
|  | ||||
| If for maybe security reasons, a library must be updated. Then you must need to check if there are any breaking changes. | ||||
|  | ||||
| ## Translations | ||||
|  | ||||
| Please read: https://github.com/louislam/uptime-kuma/tree/master/src/languages | ||||
| @@ -210,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: | ||||
|  | ||||
|   | ||||
							
								
								
									
										60
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										60
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,39 +1,39 @@ | ||||
| # Uptime Kuma | ||||
|  | ||||
| <a target="_blank" href="https://github.com/louislam/uptime-kuma"><img src="https://img.shields.io/github/stars/louislam/uptime-kuma" /></a> <a target="_blank" href="https://hub.docker.com/r/louislam/uptime-kuma"><img src="https://img.shields.io/docker/pulls/louislam/uptime-kuma" /></a> <a target="_blank" href="https://hub.docker.com/r/louislam/uptime-kuma"><img src="https://img.shields.io/docker/v/louislam/uptime-kuma/latest?label=docker%20image%20ver." /></a> <a target="_blank" href="https://github.com/louislam/uptime-kuma"><img src="https://img.shields.io/github/last-commit/louislam/uptime-kuma" /></a>  <a target="_blank" href="https://opencollective.com/uptime-kuma"><img src="https://opencollective.com/uptime-kuma/total/badge.svg?label=Open%20Collective%20Backers&color=brightgreen" /></a> | ||||
| [](https://github.com/sponsors/louislam) | ||||
| [](https://github.com/sponsors/louislam) <a href="https://weblate.kuma.pet/projects/uptime-kuma/uptime-kuma/"> | ||||
| <img src="https://weblate.kuma.pet/widgets/uptime-kuma/-/svg-badge.svg" alt="Translation status" /> | ||||
| </a> | ||||
|  | ||||
| <div align="center" width="100%"> | ||||
|     <img src="./public/icon.svg" width="128" alt="" /> | ||||
| </div> | ||||
|  | ||||
| It is a self-hosted monitoring tool like "Uptime Robot". | ||||
| Uptime Kuma is an easy-to-use self-hosted monitoring tool. | ||||
|  | ||||
| <img src="https://uptime.kuma.pet/img/dark.jpg" width="700" alt="" /> | ||||
| <img src="https://user-images.githubusercontent.com/1336778/212262296-e6205815-ad62-488c-83ec-a5b0d0689f7c.jpg" width="700" alt="" /> | ||||
|  | ||||
| ## 🥔 Live Demo | ||||
|  | ||||
| Try it! | ||||
|  | ||||
| https://demo.uptime.kuma.pet | ||||
| - Tokyo Demo Server: https://demo.uptime.kuma.pet (Sponsored by [Uptime Kuma Sponsors](https://github.com/louislam/uptime-kuma#%EF%B8%8F-sponsors)) | ||||
|  | ||||
| It is a temporary live demo, all data will be deleted after 10 minutes. The server is located in Tokyo, so if you live far from there, it may affect your experience. I suggest that you should install and try it out for the best demo experience. | ||||
|  | ||||
| VPS is sponsored by Uptime Kuma sponsors on [Open Collective](https://opencollective.com/uptime-kuma)! Thank you so much! | ||||
| It is a temporary live demo, all data will be deleted after 10 minutes. Use the one that is closer to you, but I suggest that you should install and try it out for the best demo experience. | ||||
|  | ||||
| ## ⭐ Features | ||||
|  | ||||
| * Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / Ping / DNS Record / Push / Steam Game Server / Docker Containers. | ||||
| * Fancy, Reactive, Fast UI/UX. | ||||
| * Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [90+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/tree/master/src/components/notifications). | ||||
| * 20 second intervals. | ||||
| * [Multi Languages](https://github.com/louislam/uptime-kuma/tree/master/src/languages) | ||||
| * Multiple Status Pages | ||||
| * Map Status Page to Domain | ||||
| * Ping Chart | ||||
| * Certificate Info | ||||
| * Proxy Support | ||||
| * 2FA available | ||||
| * Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / Ping / DNS Record / Push / Steam Game Server / Docker Containers | ||||
| * Fancy, Reactive, Fast UI/UX | ||||
| * Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [90+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/tree/master/src/components/notifications) | ||||
| * 20 second intervals | ||||
| * [Multi Languages](https://github.com/louislam/uptime-kuma/tree/master/src/lang) | ||||
| * Multiple status pages | ||||
| * Map status pages to specific domains | ||||
| * Ping chart | ||||
| * Certificate info | ||||
| * Proxy support | ||||
| * 2FA support | ||||
|  | ||||
| ## 🔧 How to Install | ||||
|  | ||||
| @@ -45,14 +45,15 @@ docker run -d --restart=always -p 3001:3001 -v uptime-kuma:/app/data --name upti | ||||
|  | ||||
| ⚠️ Please use a **local volume** only. Other types such as NFS are not supported. | ||||
|  | ||||
| Browse to http://localhost:3001 after starting. | ||||
| Uptime Kuma is now running on http://localhost:3001 | ||||
|  | ||||
| ### 💪🏻 Non-Docker | ||||
|  | ||||
| Required Tools:  | ||||
| - [Node.js](https://nodejs.org/en/download/) >= 14 | ||||
| - [npm](https://docs.npmjs.com/cli/) >= 7 | ||||
| - [Git](https://git-scm.com/downloads)  | ||||
| - [pm2](https://pm2.keymetrics.io/) - For run in background | ||||
| - [pm2](https://pm2.keymetrics.io/) - For running Uptime Kuma in the background | ||||
|  | ||||
| ```bash | ||||
| # Update your npm to the latest version | ||||
| @@ -74,7 +75,7 @@ pm2 start server/server.js --name uptime-kuma | ||||
|  | ||||
|  | ||||
| ``` | ||||
| Browse to http://localhost:3001 after starting. | ||||
| Uptime Kuma is now running on http://localhost:3001 | ||||
|  | ||||
| More useful PM2 Commands | ||||
|  | ||||
| @@ -106,7 +107,7 @@ https://github.com/louislam/uptime-kuma/milestones | ||||
|  | ||||
| Project Plan: | ||||
|  | ||||
| https://github.com/louislam/uptime-kuma/projects/1 | ||||
| https://github.com/users/louislam/projects/4/views/1 | ||||
|  | ||||
| ## ❤️ Sponsors | ||||
|  | ||||
| @@ -157,7 +158,14 @@ You can mention me if you ask a question on Reddit. | ||||
|  | ||||
| ## Contribute | ||||
|  | ||||
| ### Beta Version | ||||
| ### Test Pull Requests | ||||
|  | ||||
| There are a lot of pull requests right now, but I don't have time to test them all. | ||||
|  | ||||
| If you want to help, you can check this: | ||||
| https://github.com/louislam/uptime-kuma/wiki/Test-Pull-Requests | ||||
|  | ||||
| ### Test Beta Version | ||||
|  | ||||
| Check out the latest beta release here: https://github.com/louislam/uptime-kuma/releases | ||||
|  | ||||
| @@ -165,9 +173,9 @@ Check out the latest beta release here: https://github.com/louislam/uptime-kuma/ | ||||
| If you want to report a bug or request a new feature, feel free to open a [new issue](https://github.com/louislam/uptime-kuma/issues). | ||||
|  | ||||
| ### Translations | ||||
| If you want to translate Uptime Kuma into your language, please read: https://github.com/louislam/uptime-kuma/tree/master/src/languages | ||||
| If you want to translate Uptime Kuma into your language, please visit [Weblate Readme](https://github.com/louislam/uptime-kuma/blob/master/src/lang/README.md). | ||||
|  | ||||
| Feel free to correct my grammar in this README, source code, or wiki, as my mother language is not English and my grammar is not that great. | ||||
|  | ||||
| ### Pull Requests | ||||
| If you want to modify Uptime Kuma, this guideline may be useful for you: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md | ||||
| ### Create Pull Requests | ||||
| If you want to modify Uptime Kuma, please read this guide and follow the rules here: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md | ||||
|   | ||||
| @@ -2,9 +2,14 @@ | ||||
|  | ||||
| ## Reporting a Vulnerability | ||||
|  | ||||
| Please report security issues to uptime@kuma.pet. | ||||
| 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 issue tracker or discuss it in the public as it will cause more damage. | ||||
| Do not use the public issue tracker or discuss it in the public as it will cause more damage. | ||||
|  | ||||
| ## Do you accept other 3rd-party bug bounty platforms? | ||||
|  | ||||
| At this moment, I DO NOT accept other bug bounty platforms, because I am not familiar with these platforms and someone have tried to send a phishing link to me by this already. To minimize my own risk, please report through GitHub Advisories only. I will ignore all 3rd-party bug bounty platforms emails. | ||||
|  | ||||
| ## Supported Versions | ||||
|  | ||||
|   | ||||
							
								
								
									
										28
									
								
								config/cypress.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								config/cypress.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| const { defineConfig } = require("cypress"); | ||||
|  | ||||
| module.exports = defineConfig({ | ||||
|     projectId: "vyjuem", | ||||
|     e2e: { | ||||
|         experimentalStudio: true, | ||||
|         setupNodeEvents(on, config) { | ||||
|  | ||||
|         }, | ||||
|         fixturesFolder: "test/cypress/fixtures", | ||||
|         screenshotsFolder: "test/cypress/screenshots", | ||||
|         videosFolder: "test/cypress/videos", | ||||
|         downloadsFolder: "test/cypress/downloads", | ||||
|         supportFile: "test/cypress/support/e2e.js", | ||||
|         baseUrl: "http://localhost:3002", | ||||
|         defaultCommandTimeout: 10000, | ||||
|         pageLoadTimeout: 60000, | ||||
|         viewportWidth: 1920, | ||||
|         viewportHeight: 1080, | ||||
|         specPattern: [ | ||||
|             "test/cypress/e2e/setup.cy.js", | ||||
|             "test/cypress/e2e/**/*.js" | ||||
|         ], | ||||
|     }, | ||||
|     env: { | ||||
|         baseUrl: "http://localhost:3002", | ||||
|     }, | ||||
| }); | ||||
							
								
								
									
										10
									
								
								config/cypress.frontend.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								config/cypress.frontend.config.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| const { defineConfig } = require("cypress"); | ||||
|  | ||||
| module.exports = defineConfig({ | ||||
|     e2e: { | ||||
|         supportFile: false, | ||||
|         specPattern: [ | ||||
|             "test/cypress/unit/**/*.js" | ||||
|         ], | ||||
|     } | ||||
| }); | ||||
| @@ -1,33 +0,0 @@ | ||||
| const PuppeteerEnvironment = require("jest-environment-puppeteer"); | ||||
| const util = require("util"); | ||||
|  | ||||
| class DebugEnv extends PuppeteerEnvironment { | ||||
|     async handleTestEvent(event, state) { | ||||
|         const ignoredEvents = [ | ||||
|             "setup", | ||||
|             "add_hook", | ||||
|             "start_describe_definition", | ||||
|             "add_test", | ||||
|             "finish_describe_definition", | ||||
|             "run_start", | ||||
|             "run_describe_start", | ||||
|             "test_start", | ||||
|             "hook_start", | ||||
|             "hook_success", | ||||
|             "test_fn_start", | ||||
|             "test_fn_success", | ||||
|             "test_done", | ||||
|             "run_describe_finish", | ||||
|             "run_finish", | ||||
|             "teardown", | ||||
|             "test_fn_failure", | ||||
|         ]; | ||||
|         if (!ignoredEvents.includes(event.name)) { | ||||
|             console.log( | ||||
|                 new Date().toString() + ` Unhandled event [${event.name}] ` + util.inspect(event) | ||||
|             ); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = DebugEnv; | ||||
| @@ -1,5 +0,0 @@ | ||||
| module.exports = { | ||||
|     "rootDir": "..", | ||||
|     "testRegex": "./test/frontend.spec.js", | ||||
| }; | ||||
|  | ||||
| @@ -1,20 +0,0 @@ | ||||
| module.exports = { | ||||
|     "launch": { | ||||
|         "dumpio": true, | ||||
|         "slowMo": 500, | ||||
|         "headless": process.env.HEADLESS_TEST || false, | ||||
|         "userDataDir": "./data/test-chrome-profile", | ||||
|         args: [ | ||||
|             "--disable-setuid-sandbox", | ||||
|             "--disable-gpu", | ||||
|             "--disable-dev-shm-usage", | ||||
|             "--no-default-browser-check", | ||||
|             "--no-experiments", | ||||
|             "--no-first-run", | ||||
|             "--no-pings", | ||||
|             "--no-sandbox", | ||||
|             "--no-zygote", | ||||
|             "--single-process", | ||||
|         ], | ||||
|     } | ||||
| }; | ||||
| @@ -1,12 +0,0 @@ | ||||
| module.exports = { | ||||
|     "verbose": true, | ||||
|     "preset": "jest-puppeteer", | ||||
|     "globals": { | ||||
|         "__DEV__": true | ||||
|     }, | ||||
|     "testRegex": "./test/e2e.spec.js", | ||||
|     "testEnvironment": "./config/jest-debug-env.js", | ||||
|     "rootDir": "..", | ||||
|     "testTimeout": 30000, | ||||
| }; | ||||
|  | ||||
| @@ -11,6 +11,9 @@ const viteCompressionFilter = /\.(js|mjs|json|css|html|svg)$/i; | ||||
|  | ||||
| // https://vitejs.dev/config/ | ||||
| export default defineConfig({ | ||||
|     server: { | ||||
|         port: 3000, | ||||
|     }, | ||||
|     define: { | ||||
|         "FRONTEND_VERSION": JSON.stringify(process.env.npm_package_version), | ||||
|     }, | ||||
|   | ||||
							
								
								
									
										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; | ||||
							
								
								
									
										5
									
								
								db/patch-add-gamedig-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								db/patch-add-gamedig-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
|  ALTER TABLE monitor | ||||
|      ADD game VARCHAR(255); | ||||
|  COMMIT | ||||
							
								
								
									
										4
									
								
								db/patch-add-google-analytics-status-page-tag.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								db/patch-add-google-analytics-status-page-tag.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
| ALTER TABLE status_page ADD google_analytics_tag_id VARCHAR; | ||||
| COMMIT; | ||||
							
								
								
									
										18
									
								
								db/patch-add-radius-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								db/patch-add-radius-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_username VARCHAR(255); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_password VARCHAR(255); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_calling_station_id VARCHAR(50); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_called_station_id VARCHAR(50); | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD radius_secret VARCHAR(255); | ||||
|  | ||||
| COMMIT | ||||
							
								
								
									
										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; | ||||
							
								
								
									
										25
									
								
								db/patch-grpc-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								db/patch-grpc-monitor.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_url VARCHAR(255) default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_protobuf TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_body TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_metadata TEXT default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_method VARCHAR(255) default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_service_name VARCHAR(255) default null; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD grpc_enable_tls BOOLEAN default 0 not null; | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										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; | ||||
							
								
								
									
										83
									
								
								db/patch-maintenance-table2.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								db/patch-maintenance-table2.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,83 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| -- Just for someone who tested maintenance before (patch-maintenance-table.sql) | ||||
| DROP TABLE IF EXISTS maintenance_status_page; | ||||
| DROP TABLE IF EXISTS monitor_maintenance; | ||||
| DROP TABLE IF EXISTS maintenance; | ||||
| DROP TABLE IF EXISTS maintenance_timeslot; | ||||
|  | ||||
| -- maintenance | ||||
| CREATE TABLE [maintenance] ( | ||||
|     [id] INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, | ||||
|     [title] VARCHAR(150) NOT NULL, | ||||
|     [description] TEXT NOT NULL, | ||||
|     [user_id] INTEGER REFERENCES [user]([id]) ON DELETE SET NULL ON UPDATE CASCADE, | ||||
|     [active] BOOLEAN NOT NULL DEFAULT 1, | ||||
|     [strategy] VARCHAR(50) NOT NULL DEFAULT 'single', | ||||
|     [start_date] DATETIME, | ||||
|     [end_date] DATETIME, | ||||
|     [start_time] TIME, | ||||
|     [end_time] TIME, | ||||
|     [weekdays] VARCHAR2(250) DEFAULT '[]', | ||||
|     [days_of_month] TEXT DEFAULT '[]', | ||||
|     [interval_day] INTEGER | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [manual_active] ON [maintenance] ( | ||||
|     [strategy], | ||||
|     [active] | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [active] ON [maintenance] ([active]); | ||||
|  | ||||
| CREATE INDEX [maintenance_user_id] ON [maintenance] ([user_id]); | ||||
|  | ||||
| -- maintenance_status_page | ||||
| CREATE TABLE maintenance_status_page ( | ||||
|     id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||
|     status_page_id INTEGER NOT NULL, | ||||
|     maintenance_id INTEGER NOT NULL, | ||||
|     CONSTRAINT FK_maintenance FOREIGN KEY (maintenance_id) REFERENCES maintenance (id) ON DELETE CASCADE ON UPDATE CASCADE, | ||||
|     CONSTRAINT FK_status_page FOREIGN KEY (status_page_id) REFERENCES status_page (id) ON DELETE CASCADE ON UPDATE CASCADE | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [status_page_id_index] | ||||
|     ON [maintenance_status_page]([status_page_id]); | ||||
|  | ||||
| CREATE INDEX [maintenance_id_index] | ||||
|     ON [maintenance_status_page]([maintenance_id]); | ||||
|  | ||||
| -- maintenance_timeslot | ||||
| CREATE TABLE [maintenance_timeslot] ( | ||||
|     [id] INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, | ||||
|     [maintenance_id] INTEGER NOT NULL CONSTRAINT [FK_maintenance] REFERENCES [maintenance]([id]) ON DELETE CASCADE ON UPDATE CASCADE, | ||||
|     [start_date] DATETIME NOT NULL, | ||||
|     [end_date] DATETIME, | ||||
|     [generated_next] BOOLEAN DEFAULT 0 | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [maintenance_id] ON [maintenance_timeslot] ([maintenance_id] DESC); | ||||
|  | ||||
| CREATE INDEX [active_timeslot_index] ON [maintenance_timeslot] ( | ||||
|     [maintenance_id] DESC, | ||||
|     [start_date] DESC, | ||||
|     [end_date] DESC | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [generated_next_index] ON [maintenance_timeslot] ([generated_next]); | ||||
|  | ||||
| -- monitor_maintenance | ||||
| CREATE TABLE monitor_maintenance ( | ||||
|     id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||
|     monitor_id INTEGER NOT NULL, | ||||
|     maintenance_id INTEGER NOT NULL, | ||||
|     CONSTRAINT FK_maintenance FOREIGN KEY (maintenance_id) REFERENCES maintenance (id) ON DELETE CASCADE ON UPDATE CASCADE, | ||||
|     CONSTRAINT FK_monitor FOREIGN KEY (monitor_id) REFERENCES monitor (id) ON DELETE CASCADE ON UPDATE CASCADE | ||||
| ); | ||||
|  | ||||
| CREATE INDEX [maintenance_id_index2] ON [monitor_maintenance]([maintenance_id]); | ||||
|  | ||||
| CREATE INDEX [monitor_id_index] ON [monitor_maintenance]([monitor_id]); | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										10
									
								
								db/patch-monitor-add-resend-interval.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								db/patch-monitor-add-resend-interval.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
|  | ||||
| ALTER TABLE monitor | ||||
|     ADD resend_interval INTEGER default 0 not null; | ||||
|  | ||||
| ALTER TABLE heartbeat | ||||
|     ADD down_count INTEGER default 0 not null; | ||||
|  | ||||
| COMMIT; | ||||
							
								
								
									
										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
									
								
								db/patch-ping-packet-size.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								db/patch-ping-packet-size.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| BEGIN TRANSACTION; | ||||
| ALTER TABLE monitor | ||||
|     ADD packet_size INTEGER DEFAULT 56 NOT NULL; | ||||
| COMMIT; | ||||
| @@ -3,6 +3,6 @@ FROM node:16-alpine3.12 | ||||
| WORKDIR /app | ||||
|  | ||||
| # Install apprise, iputils for non-root ping, setpriv | ||||
| RUN apk add --no-cache iputils setpriv dumb-init python3 py3-cryptography py3-pip py3-six py3-yaml py3-click py3-markdown py3-requests py3-requests-oauthlib && \ | ||||
|     pip3 --no-cache-dir install apprise==0.9.9 && \ | ||||
| RUN apk add --no-cache iputils setpriv dumb-init python3 py3-cryptography py3-pip py3-six py3-yaml py3-click py3-markdown py3-requests py3-requests-oauthlib git && \ | ||||
|     pip3 --no-cache-dir install apprise==1.3.0 && \ | ||||
|     rm -rf /root/.cache | ||||
|   | ||||
							
								
								
									
										16
									
								
								docker/builder-go.dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								docker/builder-go.dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| ############################################ | ||||
| # Build in Golang | ||||
| # Run npm run build-healthcheck-armv7 in the host first, another it will be super slow where it is building the armv7 healthcheck | ||||
| ############################################ | ||||
| FROM golang:1.19-buster | ||||
| WORKDIR /app | ||||
| ARG TARGETPLATFORM | ||||
| COPY ./extra/ ./extra/ | ||||
|  | ||||
| # Compile healthcheck.go | ||||
| RUN apt update && \ | ||||
|     apt --yes --no-install-recommends install curl && \ | ||||
|     curl -sL https://deb.nodesource.com/setup_18.x | bash && \ | ||||
|     apt --yes --no-install-recommends install nodejs && \ | ||||
|     node ./extra/build-healthcheck.js $TARGETPLATFORM && \ | ||||
|     apt --yes remove nodejs | ||||
| @@ -10,8 +10,8 @@ WORKDIR /app | ||||
| # 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 update && \ | ||||
|     apt --yes --no-install-recommends install python3 python3-pip python3-cryptography python3-six python3-yaml python3-click python3-markdown python3-requests python3-requests-oauthlib \ | ||||
|         sqlite3 iputils-ping util-linux dumb-init && \ | ||||
|     pip3 --no-cache-dir install apprise==0.9.9 && \ | ||||
|         sqlite3 iputils-ping util-linux dumb-init git && \ | ||||
|     pip3 --no-cache-dir install apprise==1.3.0 && \ | ||||
|     rm -rf /var/lib/apt/lists/* && \ | ||||
|     apt --yes autoremove | ||||
|  | ||||
|   | ||||
| @@ -1,31 +1,82 @@ | ||||
| ############################################ | ||||
| # Build in Golang | ||||
| # Run npm run build-healthcheck-armv7 in the host first, another it will be super slow where it is building the armv7 healthcheck | ||||
| # Check file: builder-go.dockerfile | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:builder-go AS build_healthcheck | ||||
|  | ||||
| ############################################ | ||||
| # Build in Node.js | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS build | ||||
| WORKDIR /app | ||||
|  | ||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | ||||
|  | ||||
| COPY .npmrc .npmrc | ||||
| COPY package.json package.json | ||||
| COPY package-lock.json package-lock.json | ||||
| RUN npm ci --omit=dev | ||||
| COPY . . | ||||
| RUN npm ci --production && \ | ||||
|     chmod +x /app/extra/entrypoint.sh | ||||
|  | ||||
| COPY --from=build_healthcheck /app/extra/healthcheck /app/extra/healthcheck | ||||
| RUN chmod +x /app/extra/entrypoint.sh | ||||
|  | ||||
| ############################################ | ||||
| # ⭐ Main Image | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS release | ||||
| WORKDIR /app | ||||
|  | ||||
| # Copy app files from build layer | ||||
| COPY --from=build /app /app | ||||
|  | ||||
|  | ||||
| EXPOSE 3001 | ||||
| VOLUME ["/app/data"] | ||||
| HEALTHCHECK --interval=60s --timeout=30s --start-period=180s --retries=5 CMD node extra/healthcheck.js | ||||
| HEALTHCHECK --interval=60s --timeout=30s --start-period=180s --retries=5 CMD extra/healthcheck | ||||
| ENTRYPOINT ["/usr/bin/dumb-init", "--", "extra/entrypoint.sh"] | ||||
| CMD ["node", "server/server.js"] | ||||
|  | ||||
|  | ||||
| ############################################ | ||||
| # Mark as Nightly | ||||
| ############################################ | ||||
| FROM release AS nightly | ||||
| RUN npm run mark-as-nightly | ||||
|  | ||||
| ############################################ | ||||
| # Build an image for testing pr | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS pr-test | ||||
|  | ||||
| WORKDIR /app | ||||
|  | ||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | ||||
|  | ||||
| ## Install Git | ||||
| RUN apt update \ | ||||
|     && apt --yes --no-install-recommends install curl \ | ||||
|     && curl -fsSL https://cli.github.com/packages/githubcli-archive-keyring.gpg | dd of=/usr/share/keyrings/githubcli-archive-keyring.gpg \ | ||||
|     && chmod go+r /usr/share/keyrings/githubcli-archive-keyring.gpg \ | ||||
|     && echo "deb [arch=$(dpkg --print-architecture) signed-by=/usr/share/keyrings/githubcli-archive-keyring.gpg] https://cli.github.com/packages stable main" | tee /etc/apt/sources.list.d/github-cli.list > /dev/null \ | ||||
|     && apt update \ | ||||
|     && apt --yes --no-install-recommends  install git | ||||
|  | ||||
| ## Empty the directory, because we have to clone the Git repo. | ||||
| RUN rm -rf ./* && chown node /app | ||||
|  | ||||
| USER node | ||||
| RUN git config --global user.email "no-reply@no-reply.com" | ||||
| RUN git config --global user.name "PR Tester" | ||||
| RUN git clone https://github.com/louislam/uptime-kuma.git . | ||||
| RUN npm ci | ||||
|  | ||||
| EXPOSE 3000 3001 | ||||
| VOLUME ["/app/data"] | ||||
| HEALTHCHECK --interval=60s --timeout=30s --start-period=180s --retries=5 CMD extra/healthcheck | ||||
| CMD ["npm", "run", "start-pr-test"] | ||||
|  | ||||
| ############################################ | ||||
| # Upload the artifact to Github | ||||
| ############################################ | ||||
| FROM louislam/uptime-kuma:base-debian AS upload-artifact | ||||
| WORKDIR / | ||||
| RUN apt update && \ | ||||
|   | ||||
| @@ -3,10 +3,12 @@ WORKDIR /app | ||||
|  | ||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | ||||
|  | ||||
| COPY .npmrc .npmrc | ||||
| COPY package.json package.json | ||||
| COPY package-lock.json package-lock.json | ||||
| RUN npm ci --omit=dev | ||||
| COPY . . | ||||
| RUN npm ci --production && \ | ||||
|     chmod +x /app/extra/entrypoint.sh | ||||
|  | ||||
| RUN chmod +x /app/extra/entrypoint.sh | ||||
|  | ||||
| FROM louislam/uptime-kuma:base-alpine AS release | ||||
| WORKDIR /app | ||||
|   | ||||
| @@ -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); | ||||
| @@ -32,6 +33,10 @@ if (! exists) { | ||||
|     process.exit(1); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Commit updated files | ||||
|  * @param {string} version Version to update to | ||||
|  */ | ||||
| function commit(version) { | ||||
|     let msg = "Update to " + version; | ||||
|  | ||||
| @@ -47,6 +52,10 @@ function commit(version) { | ||||
|     console.log(res.stdout.toString().trim()); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Create a tag with the specified version | ||||
|  * @param {string} version Tag to create | ||||
|  */ | ||||
| function tag(version) { | ||||
|     let res = childProcess.spawnSync("git", [ "tag", version ]); | ||||
|     console.log(res.stdout.toString().trim()); | ||||
| @@ -55,6 +64,11 @@ function tag(version) { | ||||
|     console.log(res.stdout.toString().trim()); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Check if a tag exists for the specified version | ||||
|  * @param {string} version Version to check | ||||
|  * @returns {boolean} Does the tag already exist | ||||
|  */ | ||||
| function tagExists(version) { | ||||
|     if (! version) { | ||||
|         throw new Error("invalid version"); | ||||
|   | ||||
							
								
								
									
										27
									
								
								extra/build-healthcheck.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								extra/build-healthcheck.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| const childProcess = require("child_process"); | ||||
| const fs = require("fs"); | ||||
| const platform = process.argv[2]; | ||||
|  | ||||
| if (!platform) { | ||||
|     console.error("No platform??"); | ||||
|     process.exit(1); | ||||
| } | ||||
|  | ||||
| if (platform === "linux/arm/v7") { | ||||
|     console.log("Arch: armv7"); | ||||
|     if (fs.existsSync("./extra/healthcheck-armv7")) { | ||||
|         fs.renameSync("./extra/healthcheck-armv7", "./extra/healthcheck"); | ||||
|         console.log("Already built in the host, skip."); | ||||
|         process.exit(0); | ||||
|     } else { | ||||
|         console.log("prebuilt not found, it will be slow! You should execute `npm run build-healthcheck-armv7` before build."); | ||||
|     } | ||||
| } else { | ||||
|     if (fs.existsSync("./extra/healthcheck-armv7")) { | ||||
|         fs.rmSync("./extra/healthcheck-armv7"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| const output = childProcess.execSync("go build -x -o ./extra/healthcheck ./extra/healthcheck.go").toString("utf8"); | ||||
| console.log(output); | ||||
|  | ||||
							
								
								
									
										33
									
								
								extra/checkout-pr.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								extra/checkout-pr.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| const childProcess = require("child_process"); | ||||
|  | ||||
| if (!process.env.UPTIME_KUMA_GH_REPO) { | ||||
|     console.error("Please set a repo to the environment variable 'UPTIME_KUMA_GH_REPO' (e.g. mhkarimi1383:goalert-notification)"); | ||||
|     process.exit(1); | ||||
| } | ||||
|  | ||||
| let inputArray = process.env.UPTIME_KUMA_GH_REPO.split(":"); | ||||
|  | ||||
| if (inputArray.length !== 2) { | ||||
|     console.error("Invalid format. Please set a repo to the environment variable 'UPTIME_KUMA_GH_REPO' (e.g. mhkarimi1383:goalert-notification)"); | ||||
| } | ||||
|  | ||||
| let name = inputArray[0]; | ||||
| let branch = inputArray[1]; | ||||
|  | ||||
| console.log("Checkout pr"); | ||||
|  | ||||
| // Checkout the pr | ||||
| let result = childProcess.spawnSync("git", [ "remote", "add", name, `https://github.com/${name}/uptime-kuma` ]); | ||||
|  | ||||
| console.log(result.stdout.toString()); | ||||
| console.error(result.stderr.toString()); | ||||
|  | ||||
| result = childProcess.spawnSync("git", [ "fetch", name, branch ]); | ||||
|  | ||||
| console.log(result.stdout.toString()); | ||||
| console.error(result.stderr.toString()); | ||||
|  | ||||
| result = childProcess.spawnSync("git", [ "checkout", `${name}/${branch}`, "--force" ]); | ||||
|  | ||||
| console.log(result.stdout.toString()); | ||||
| console.error(result.stderr.toString()); | ||||
							
								
								
									
										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)); | ||||
| @@ -25,6 +25,10 @@ if (platform === "linux/amd64") { | ||||
| 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) { | ||||
|   | ||||
| @@ -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 /> | ||||
| </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("")] | ||||
| [assembly: AssemblyConfiguration("")] | ||||
| [assembly: AssemblyCompany("")] | ||||
| [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("2DB53988-1D93-4AC0-90C4-96ADEAAC5C04")] | ||||
|  | ||||
| // 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.0.0")] | ||||
| [assembly: AssemblyFileVersion("1.0.0.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> | ||||
							
								
								
									
										90
									
								
								extra/healthcheck.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								extra/healthcheck.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | ||||
| /* | ||||
|  * If changed, have to run `npm run build-docker-builder-go`. | ||||
|  * This script should be run after a period of time (180s), because the server may need some time to prepare. | ||||
|  */ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"crypto/tls" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	isFreeBSD := runtime.GOOS == "freebsd" | ||||
|  | ||||
| 	// Is K8S + uptime-kuma as the container name | ||||
| 	// See #2083 | ||||
| 	isK8s := strings.HasPrefix(os.Getenv("UPTIME_KUMA_PORT"), "tcp://") | ||||
|  | ||||
| 	// process.env.NODE_TLS_REJECT_UNAUTHORIZED = "0"; | ||||
| 	http.DefaultTransport.(*http.Transport).TLSClientConfig = &tls.Config{ | ||||
| 		InsecureSkipVerify: true, | ||||
| 	} | ||||
|  | ||||
| 	client := http.Client{ | ||||
| 		Timeout: 28 * time.Second, | ||||
| 	} | ||||
|  | ||||
| 	sslKey := os.Getenv("UPTIME_KUMA_SSL_KEY") | ||||
| 	if len(sslKey) == 0 { | ||||
| 		sslKey = os.Getenv("SSL_KEY") | ||||
| 	} | ||||
|  | ||||
| 	sslCert := os.Getenv("UPTIME_KUMA_SSL_CERT") | ||||
| 	if len(sslCert) == 0 { | ||||
| 		sslCert = os.Getenv("SSL_CERT") | ||||
| 	} | ||||
|  | ||||
| 	hostname := os.Getenv("UPTIME_KUMA_HOST") | ||||
| 	if len(hostname) == 0 && !isFreeBSD { | ||||
| 		hostname = os.Getenv("HOST") | ||||
| 	} | ||||
| 	if len(hostname) == 0 { | ||||
| 		hostname = "127.0.0.1" | ||||
| 	} | ||||
|  | ||||
| 	port := "" | ||||
| 	// UPTIME_KUMA_PORT is override by K8S unexpectedly, | ||||
| 	if !isK8s { | ||||
| 		port = os.Getenv("UPTIME_KUMA_PORT") | ||||
| 	} | ||||
| 	if len(port) == 0 { | ||||
| 		port = os.Getenv("PORT") | ||||
| 	} | ||||
| 	if len(port) == 0 { | ||||
| 		port = "3001" | ||||
| 	} | ||||
|  | ||||
| 	protocol := "" | ||||
| 	if len(sslKey) != 0 && len(sslCert) != 0 { | ||||
| 		protocol = "https" | ||||
| 	} else { | ||||
| 		protocol = "http" | ||||
| 	} | ||||
|  | ||||
| 	url := protocol + "://" + hostname + ":" + port | ||||
|  | ||||
| 	log.Println("Checking " + url) | ||||
| 	resp, err := client.Get(url) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		log.Fatalln(err) | ||||
| 	} | ||||
|  | ||||
| 	defer resp.Body.Close() | ||||
|  | ||||
| 	_, err = ioutil.ReadAll(resp.Body) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		log.Fatalln(err) | ||||
| 	} | ||||
|  | ||||
| 	log.Printf("Health Check OK [Res Code: %d]\n", resp.StatusCode) | ||||
|  | ||||
| } | ||||
| @@ -1,4 +1,9 @@ | ||||
| /* | ||||
|  * ⚠️ ⚠️ ⚠️ ⚠️ 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. | ||||
|  */ | ||||
| const { FBSD } = require("../server/util-server"); | ||||
|   | ||||
| @@ -1,11 +1,12 @@ | ||||
| const pkg = require("../package.json"); | ||||
| const fs = require("fs"); | ||||
| const util = require("../src/util"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| util.polyfill(); | ||||
|  | ||||
| const oldVersion = pkg.version; | ||||
| const newVersion = oldVersion + "-nightly"; | ||||
| const newVersion = oldVersion + "-nightly-" + dayjs().format("YYYYMMDDHHmmss"); | ||||
|  | ||||
| console.log("Old Version: " + oldVersion); | ||||
| console.log("New Version: " + newVersion); | ||||
|   | ||||
| @@ -43,6 +43,11 @@ const main = async () => { | ||||
|     console.log("Finished."); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Ask question of user | ||||
|  * @param {string} question Question to ask | ||||
|  * @returns {Promise<string>} Users response | ||||
|  */ | ||||
| function question(question) { | ||||
|     return new Promise((resolve) => { | ||||
|         rl.question(question, (answer) => { | ||||
|   | ||||
| @@ -53,6 +53,11 @@ const main = async () => { | ||||
|     console.log("Finished."); | ||||
| }; | ||||
|  | ||||
| /** | ||||
|  * Ask question of user | ||||
|  * @param {string} question Question to ask | ||||
|  * @returns {Promise<string>} Users response | ||||
|  */ | ||||
| function question(question) { | ||||
|     return new Promise((resolve) => { | ||||
|         rl.question(question, (answer) => { | ||||
|   | ||||
| @@ -135,6 +135,11 @@ server.listen({ | ||||
|     udp: 5300 | ||||
| }); | ||||
|  | ||||
| /** | ||||
|  * Get human readable request type from request code | ||||
|  * @param {number} code Request code to translate | ||||
|  * @returns {string} Human readable request type | ||||
|  */ | ||||
| function type(code) { | ||||
|     for (let name in Packet.TYPE) { | ||||
|         if (Packet.TYPE[name] === code) { | ||||
|   | ||||
| @@ -11,6 +11,7 @@ class SimpleMqttServer { | ||||
|         this.port = port; | ||||
|     } | ||||
|  | ||||
|     /** Start the MQTT server */ | ||||
|     start() { | ||||
|         this.server.listen(this.port, () => { | ||||
|             console.log("server started and listening on port ", this.port); | ||||
|   | ||||
							
								
								
									
										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(" ")); | ||||
| @@ -1,51 +1,45 @@ | ||||
| // Need to use ES6 to read language files | ||||
|  | ||||
| import fs from "fs"; | ||||
| import path from "path"; | ||||
| import util from "util"; | ||||
| import rmSync from "../fs-rmSync.js"; | ||||
|  | ||||
| // https://stackoverflow.com/questions/13786160/copy-folder-recursively-in-node-js | ||||
| /** | ||||
|  * Look ma, it's cp -R. | ||||
|  * @param {string} src  The path to the thing to copy. | ||||
|  * @param {string} dest The path to the new copy. | ||||
|  * Copy across the required language files | ||||
|  * Creates a local directory (./languages) and copies the required files | ||||
|  * into it. | ||||
|  * @param {string} langCode Code of language to update. A file will be | ||||
|  * created with this code if one does not already exist | ||||
|  * @param {string} baseLang The second base language file to copy. This | ||||
|  * will be ignored if set to "en" as en.js is copied by default | ||||
|  */ | ||||
| const copyRecursiveSync = function (src, dest) { | ||||
|     let exists = fs.existsSync(src); | ||||
|     let stats = exists && fs.statSync(src); | ||||
|     let isDirectory = exists && stats.isDirectory(); | ||||
|  | ||||
|     if (isDirectory) { | ||||
|         fs.mkdirSync(dest); | ||||
|         fs.readdirSync(src).forEach(function (childItemName) { | ||||
|             copyRecursiveSync(path.join(src, childItemName), | ||||
|                 path.join(dest, childItemName)); | ||||
|         }); | ||||
|     } else { | ||||
|         fs.copyFileSync(src, dest); | ||||
|     } | ||||
| }; | ||||
|  | ||||
| console.log("Arguments:", process.argv); | ||||
| const baseLangCode = process.argv[2] || "en"; | ||||
| console.log("Base Lang: " + baseLangCode); | ||||
| function copyFiles(langCode, baseLang) { | ||||
|     if (fs.existsSync("./languages")) { | ||||
|         rmSync("./languages", { recursive: true }); | ||||
|     } | ||||
| copyRecursiveSync("../../src/languages", "./languages"); | ||||
|     fs.mkdirSync("./languages"); | ||||
|  | ||||
| const en = (await import("./languages/en.js")).default; | ||||
| const baseLang = (await import(`./languages/${baseLangCode}.js`)).default; | ||||
| const files = fs.readdirSync("./languages"); | ||||
| console.log("Files:", files); | ||||
|  | ||||
| for (const file of files) { | ||||
|     if (! file.endsWith(".js")) { | ||||
|         console.log("Skipping " + file); | ||||
|         continue; | ||||
|     if (!fs.existsSync(`../../src/languages/${langCode}.js`)) { | ||||
|         fs.closeSync(fs.openSync(`./languages/${langCode}.js`, "a")); | ||||
|     } else { | ||||
|         fs.copyFileSync(`../../src/languages/${langCode}.js`, `./languages/${langCode}.js`); | ||||
|     } | ||||
|     fs.copyFileSync("../../src/languages/en.js", "./languages/en.js"); | ||||
|     if (baseLang !== "en") { | ||||
|         fs.copyFileSync(`../../src/languages/${baseLang}.js`, `./languages/${baseLang}.js`); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Update the specified language file | ||||
|  * @param {string} langCode Language code to update | ||||
|  * @param {string} baseLang Second language to copy keys from | ||||
|  */ | ||||
| async function updateLanguage(langCode, baseLangCode) { | ||||
|     const en = (await import("./languages/en.js")).default; | ||||
|     const baseLang = (await import(`./languages/${baseLangCode}.js`)).default; | ||||
|  | ||||
|     let file = langCode + ".js"; | ||||
|     console.log("Processing " + file); | ||||
|     const lang = await import("./languages/" + file); | ||||
|  | ||||
| @@ -83,5 +77,20 @@ for (const file of files) { | ||||
|     fs.writeFileSync(`../../src/languages/${file}`, code); | ||||
| } | ||||
|  | ||||
| // Get command line arguments | ||||
| const baseLangCode = process.env.npm_config_baselang || "en"; | ||||
| const langCode = process.env.npm_config_language; | ||||
|  | ||||
| // We need the file to edit | ||||
| if (langCode == null) { | ||||
|     throw new Error("Argument --language=<code> must be provided"); | ||||
| } | ||||
|  | ||||
| console.log("Base Lang: " + baseLangCode); | ||||
| console.log("Updating: " + langCode); | ||||
|  | ||||
| copyFiles(langCode, baseLangCode); | ||||
| await updateLanguage(langCode, baseLangCode); | ||||
| rmSync("./languages", { recursive: true }); | ||||
|  | ||||
| console.log("Done. Fixing formatting by ESLint..."); | ||||
|   | ||||
| @@ -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); | ||||
| @@ -36,10 +37,8 @@ if (! exists) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Updates the version number in package.json and commits it to git. | ||||
|  * @param {string} version - The new version number | ||||
|  * | ||||
|  * Generated by Trelent | ||||
|  * Commit updated files | ||||
|  * @param {string} version Version to update to | ||||
|  */ | ||||
| function commit(version) { | ||||
|     let msg = "Update to " + version; | ||||
| @@ -53,16 +52,19 @@ function commit(version) { | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Create a tag with the specified version | ||||
|  * @param {string} version Tag to create | ||||
|  */ | ||||
| function tag(version) { | ||||
|     let res = childProcess.spawnSync("git", [ "tag", version ]); | ||||
|     console.log(res.stdout.toString().trim()); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Checks if a given version is already tagged in the git repository. | ||||
|  * @param {string} version - The version to check for. | ||||
|  * | ||||
|  * Generated by Trelent | ||||
|  * Check if a tag exists for the specified version | ||||
|  * @param {string} version Version to check | ||||
|  * @returns {boolean} Does the tag already exist | ||||
|  */ | ||||
| function tagExists(version) { | ||||
|     if (! version) { | ||||
|   | ||||
| @@ -10,6 +10,10 @@ if (!newVersion) { | ||||
|  | ||||
| updateWiki(newVersion); | ||||
|  | ||||
| /** | ||||
|  * Update the wiki with new version number | ||||
|  * @param {string} newVersion Version to update to | ||||
|  */ | ||||
| function updateWiki(newVersion) { | ||||
|     const wikiDir = "./tmp/wiki"; | ||||
|     const howToUpdateFilename = "./tmp/wiki/🆙-How-to-Update.md"; | ||||
| @@ -39,6 +43,10 @@ function updateWiki(newVersion) { | ||||
|     safeDelete(wikiDir); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Check if a directory exists and then delete it | ||||
|  * @param {string} dir Directory to delete | ||||
|  */ | ||||
| function safeDelete(dir) { | ||||
|     if (fs.existsSync(dir)) { | ||||
|         fs.rm(dir, { | ||||
|   | ||||
							
								
								
									
										14040
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										14040
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										116
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										116
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|     "name": "uptime-kuma", | ||||
|     "version": "1.17.1", | ||||
|     "version": "1.21.0", | ||||
|     "license": "MIT", | ||||
|     "repository": { | ||||
|         "type": "git", | ||||
| @@ -23,23 +23,23 @@ | ||||
|         "start-server": "node server/server.js", | ||||
|         "start-server-dev": "cross-env NODE_ENV=development node server/server.js", | ||||
|         "build": "vite build --config ./config/vite.config.js", | ||||
|         "test": "node test/prepare-test-server.js && node server/server.js --port=3002 --data-dir=./data/test/ --test", | ||||
|         "test": "node test/prepare-test-server.js && npm run jest-backend", | ||||
|         "test-with-build": "npm run build && npm test", | ||||
|         "jest": "node test/prepare-jest.js && npm run jest-frontend && npm run jest-backend", | ||||
|         "jest-frontend": "cross-env TEST_FRONTEND=1 jest --config=./config/jest-frontend.config.js", | ||||
|         "jest-backend": "cross-env TEST_BACKEND=1 jest --config=./config/jest-backend.config.js", | ||||
|         "jest-backend": "cross-env TEST_BACKEND=1 jest --runInBand --detectOpenHandles --forceExit --config=./config/jest-backend.config.js", | ||||
|         "tsc": "tsc", | ||||
|         "vite-preview-dist": "vite preview --host --config ./config/vite.config.js", | ||||
|         "build-docker": "npm run build && npm run build-docker-debian && npm run build-docker-alpine", | ||||
|         "build-docker-alpine-base": "docker buildx build -f docker/alpine-base.dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:base-alpine . --push", | ||||
|         "build-docker-debian-base": "docker buildx build -f docker/debian-base.dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:base-debian . --push", | ||||
|         "build-docker-builder-go": "docker buildx build -f docker/builder-go.dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:builder-go . --push", | ||||
|         "build-docker-alpine": "node ./extra/env2arg.js docker buildx build -f docker/dockerfile-alpine --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:alpine -t louislam/uptime-kuma:1-alpine -t louislam/uptime-kuma:$VERSION-alpine --target release . --push", | ||||
|         "build-docker-debian": "node ./extra/env2arg.js docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma -t louislam/uptime-kuma:1 -t louislam/uptime-kuma:$VERSION -t louislam/uptime-kuma:debian -t louislam/uptime-kuma:1-debian -t louislam/uptime-kuma:$VERSION-debian --target release . --push", | ||||
|         "build-docker-nightly": "npm run build && docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:nightly --target nightly . --push", | ||||
|         "build-docker-nightly-alpine": "docker buildx build -f docker/dockerfile-alpine --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:nightly-alpine --target nightly . --push", | ||||
|         "build-docker-nightly-amd64": "docker buildx build -f docker/dockerfile --platform linux/amd64 -t louislam/uptime-kuma:nightly-amd64 --target nightly . --push --progress plain", | ||||
|         "build-docker-pr-test": "docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64 -t louislam/uptime-kuma:pr-test --target pr-test . --push", | ||||
|         "upload-artifacts": "docker buildx build -f docker/dockerfile --platform linux/amd64 -t louislam/uptime-kuma:upload-artifact --build-arg VERSION --build-arg GITHUB_TOKEN --target upload-artifact . --progress plain", | ||||
|         "setup": "git checkout 1.17.1 && npm ci --production && npm run download-dist", | ||||
|         "setup": "git checkout 1.21.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", | ||||
| @@ -52,57 +52,76 @@ | ||||
|         "test-nodejs16": "docker build --progress plain -f test/ubuntu-nodejs16.dockerfile .", | ||||
|         "simple-dns-server": "node extra/simple-dns-server.js", | ||||
|         "simple-mqtt-server": "node extra/simple-mqtt-server.js", | ||||
|         "update-language-files-with-base-lang": "cd extra/update-language-files && node index.js %npm_config_base_lang% && eslint ../../src/languages/**.js --fix", | ||||
|         "update-language-files": "cd extra/update-language-files && node index.js && eslint ../../src/languages/**.js --fix", | ||||
|         "update-language-files": "cd extra/update-language-files && node index.js && cross-env-shell eslint ../../src/languages/$npm_config_language.js --fix", | ||||
|         "ncu-patch": "npm-check-updates -u -t patch", | ||||
|         "release-final": "node extra/update-version.js && npm run build-docker && node ./extra/press-any-key.js && npm run upload-artifacts && node ./extra/update-wiki-version.js", | ||||
|         "release-beta": "node extra/beta/update-version.js && npm run build && node ./extra/env2arg.js docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:$VERSION -t louislam/uptime-kuma:beta .  --target release --push && node ./extra/press-any-key.js && npm run upload-artifacts", | ||||
|         "git-remove-tag": "git tag -d", | ||||
|         "build-dist-and-restart": "npm run build && npm run start-server-dev" | ||||
|         "build-dist-and-restart": "npm run build && npm run start-server-dev", | ||||
|         "start-pr-test": "node extra/checkout-pr.js && npm install && npm run dev", | ||||
|         "cy:test": "node test/prepare-test-server.js && node server/server.js --port=3002 --data-dir=./data/test/ --e2e", | ||||
|         "cy:run": "npx cypress run --browser chrome --headless --config-file ./config/cypress.config.js", | ||||
|         "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", | ||||
|         "depoly-demo-server": "node extra/deploy-demo-server.js", | ||||
|         "sort-contributors": "node extra/sort-contributors.js" | ||||
|     }, | ||||
|     "dependencies": { | ||||
|         "@louislam/sqlite3": "~15.0.6", | ||||
|         "@grpc/grpc-js": "~1.7.3", | ||||
|         "@louislam/ping": "~0.4.4-mod.0", | ||||
|         "@louislam/sqlite3": "15.1.2", | ||||
|         "args-parser": "~1.3.0", | ||||
|         "axios": "~0.26.1", | ||||
|         "axios-ntlm": "^1.3.0", | ||||
|         "badge-maker": "^3.3.1", | ||||
|         "axios": "~0.27.0", | ||||
|         "axios-ntlm": "1.3.0", | ||||
|         "badge-maker": "~3.3.1", | ||||
|         "bcryptjs": "~2.4.3", | ||||
|         "bree": "~7.1.5", | ||||
|         "cacheable-lookup": "~6.0.4", | ||||
|         "chardet": "^1.3.0", | ||||
|         "chardet": "~1.4.0", | ||||
|         "check-password-strength": "^2.0.5", | ||||
|         "cheerio": "^1.0.0-rc.10", | ||||
|         "chroma-js": "^2.1.2", | ||||
|         "cheerio": "~1.0.0-rc.12", | ||||
|         "chroma-js": "~2.4.2", | ||||
|         "command-exists": "~1.2.9", | ||||
|         "compare-versions": "~3.6.0", | ||||
|         "compression": "^1.7.4", | ||||
|         "dayjs": "^1.11.0", | ||||
|         "compression": "~1.7.4", | ||||
|         "dayjs": "~1.11.5", | ||||
|         "dotenv": "~16.0.3", | ||||
|         "express": "~4.17.3", | ||||
|         "express-basic-auth": "~1.2.1", | ||||
|         "express-static-gzip": "^2.1.7", | ||||
|         "express-static-gzip": "~2.1.7", | ||||
|         "form-data": "~4.0.0", | ||||
|         "gamedig": "^4.0.5", | ||||
|         "http-graceful-shutdown": "~3.1.7", | ||||
|         "http-proxy-agent": "^5.0.0", | ||||
|         "https-proxy-agent": "^5.0.0", | ||||
|         "iconv-lite": "^0.6.3", | ||||
|         "jsonwebtoken": "~8.5.1", | ||||
|         "jwt-decode": "^3.1.2", | ||||
|         "limiter": "^2.1.0", | ||||
|         "mqtt": "^4.2.8", | ||||
|         "mssql": "^8.1.0", | ||||
|         "http-proxy-agent": "~5.0.0", | ||||
|         "https-proxy-agent": "~5.0.1", | ||||
|         "iconv-lite": "~0.6.3", | ||||
|         "jsesc": "~3.0.2", | ||||
|         "jsonwebtoken": "~9.0.0", | ||||
|         "jwt-decode": "~3.1.2", | ||||
|         "limiter": "~2.1.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", | ||||
|         "notp": "~2.0.3", | ||||
|         "password-hash": "~1.2.2", | ||||
|         "pg": "^8.7.3", | ||||
|         "pg-connection-string": "^2.5.0", | ||||
|         "pg": "~8.8.0", | ||||
|         "pg-connection-string": "~2.5.0", | ||||
|         "prom-client": "~13.2.0", | ||||
|         "prometheus-api-metrics": "~3.2.1", | ||||
|         "redbean-node": "0.1.4", | ||||
|         "socket.io": "~4.4.1", | ||||
|         "socket.io-client": "~4.4.1", | ||||
|         "protobufjs": "~7.1.1", | ||||
|         "qs": "~6.10.4", | ||||
|         "redbean-node": "~0.2.0", | ||||
|         "redis": "~4.5.1", | ||||
|         "socket.io": "~4.5.3", | ||||
|         "socket.io-client": "~4.5.3", | ||||
|         "socks-proxy-agent": "6.1.1", | ||||
|         "tar": "^6.1.11", | ||||
|         "tar": "~6.1.11", | ||||
|         "tcp-ping": "~0.1.1", | ||||
|         "thirty-two": "~1.0.2" | ||||
|     }, | ||||
| @@ -116,46 +135,51 @@ | ||||
|         "@fortawesome/vue-fontawesome": "~3.0.0-5", | ||||
|         "@popperjs/core": "~2.10.2", | ||||
|         "@types/bootstrap": "~5.1.9", | ||||
|         "@vitejs/plugin-legacy": "~1.8.2", | ||||
|         "@vitejs/plugin-vue": "~2.3.3", | ||||
|         "@vitejs/plugin-legacy": "~2.1.0", | ||||
|         "@vitejs/plugin-vue": "~3.1.0", | ||||
|         "@vue/compiler-sfc": "~3.2.36", | ||||
|         "@vuepic/vue-datepicker": "~3.4.8", | ||||
|         "aedes": "^0.46.3", | ||||
|         "babel-plugin-rewire": "~1.2.0", | ||||
|         "bootstrap": "5.1.3", | ||||
|         "chart.js": "~3.6.2", | ||||
|         "chartjs-adapter-dayjs": "~1.0.0", | ||||
|         "concurrently": "^7.1.0", | ||||
|         "core-js": "~3.18.3", | ||||
|         "core-js": "~3.26.1", | ||||
|         "cross-env": "~7.0.3", | ||||
|         "cypress": "^10.1.0", | ||||
|         "delay": "^5.0.0", | ||||
|         "dns2": "~2.0.1", | ||||
|         "dompurify": "~2.4.3", | ||||
|         "eslint": "~8.14.0", | ||||
|         "eslint-plugin-vue": "~8.7.1", | ||||
|         "favico.js": "^0.3.10", | ||||
|         "favico.js": "~0.3.10", | ||||
|         "jest": "~27.2.5", | ||||
|         "jest-puppeteer": "~6.0.3", | ||||
|         "postcss-html": "^1.3.1", | ||||
|         "postcss-rtlcss": "~3.4.1", | ||||
|         "postcss-scss": "~4.0.3", | ||||
|         "prismjs": "^1.27.0", | ||||
|         "puppeteer": "~13.1.3", | ||||
|         "marked": "~4.2.5", | ||||
|         "node-ssh": "~13.0.1", | ||||
|         "postcss-html": "~1.5.0", | ||||
|         "postcss-rtlcss": "~3.7.2", | ||||
|         "postcss-scss": "~4.0.4", | ||||
|         "prismjs": "~1.29.0", | ||||
|         "qrcode": "~1.5.0", | ||||
|         "rollup-plugin-visualizer": "^5.6.0", | ||||
|         "sass": "~1.42.1", | ||||
|         "stylelint": "~14.7.1", | ||||
|         "stylelint-config-standard": "~25.0.0", | ||||
|         "terser": "~5.15.0", | ||||
|         "timezones-list": "~3.0.1", | ||||
|         "typescript": "~4.4.4", | ||||
|         "v-pagination-3": "~0.1.7", | ||||
|         "vite": "~2.9.9", | ||||
|         "vite": "~3.1.0", | ||||
|         "vite-plugin-compression": "^0.5.1", | ||||
|         "vue": "next", | ||||
|         "vue-chart-3": "3.0.9", | ||||
|         "vue-confirm-dialog": "~1.0.2", | ||||
|         "vue-contenteditable": "~3.0.4", | ||||
|         "vue-i18n": "~9.1.9", | ||||
|         "vue-i18n": "~9.2.2", | ||||
|         "vue-image-crop-upload": "~3.0.3", | ||||
|         "vue-multiselect": "~3.0.0-alpha.2", | ||||
|         "vue-prism-editor": "^2.0.0-alpha.2", | ||||
|         "vue-prism-editor": "~2.0.0-alpha.2", | ||||
|         "vue-qrcode": "~1.0.0", | ||||
|         "vue-router": "~4.0.14", | ||||
|         "vue-toastification": "~2.0.0-rc.5", | ||||
|   | ||||
| @@ -2,7 +2,9 @@ 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 { loginRateLimiter, apiRateLimiter } = require("./rate-limiter"); | ||||
| const { Settings } = require("./settings"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| /** | ||||
|  * Login to web app | ||||
| @@ -34,8 +36,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 +74,31 @@ 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) => { | ||||
|                 callback(null, valid); | ||||
|                 // Only allow a set number of api requests per minute | ||||
|                 // (currently set to 60) | ||||
|                 apiRateLimiter.removeTokens(1); | ||||
|             }); | ||||
|         } else { | ||||
|             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) { | ||||
| @@ -63,9 +115,15 @@ function myAuthorizer(username, password, callback) { | ||||
|     }); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Use basic auth if auth is not disabled | ||||
|  * @param {express.Request} req Express request object | ||||
|  * @param {express.Response} res Express response object | ||||
|  * @param {express.NextFunction} next | ||||
|  */ | ||||
| exports.basicAuth = async function (req, res, next) { | ||||
|     const middleware = basicAuth({ | ||||
|         authorizer: myAuthorizer, | ||||
|         authorizer: userAuthorizer, | ||||
|         authorizeAsync: true, | ||||
|         challenge: true, | ||||
|     }); | ||||
| @@ -78,3 +136,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(); | ||||
|     } | ||||
| }; | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| const https = require("https"); | ||||
| const http = require("http"); | ||||
| const CacheableLookup = require("cacheable-lookup"); | ||||
| const { Settings } = require("./settings"); | ||||
| const { log } = require("../src/util"); | ||||
|  | ||||
| class CacheableDnsHttpAgent { | ||||
|  | ||||
| @@ -9,14 +11,36 @@ class CacheableDnsHttpAgent { | ||||
|     static httpAgentList = {}; | ||||
|     static httpsAgentList = {}; | ||||
|  | ||||
|     static enable = false; | ||||
|  | ||||
|     /** | ||||
|      * Register cacheable to global agents | ||||
|      * Register/Disable cacheable to global agents | ||||
|      */ | ||||
|     static registerGlobalAgent() { | ||||
|     static async update() { | ||||
|         log.debug("CacheableDnsHttpAgent", "update"); | ||||
|         let isEnable = await Settings.get("dnsCache"); | ||||
|  | ||||
|         if (isEnable !== this.enable) { | ||||
|             log.debug("CacheableDnsHttpAgent", "value changed"); | ||||
|  | ||||
|             if (isEnable) { | ||||
|                 log.debug("CacheableDnsHttpAgent", "enable"); | ||||
|                 this.cacheable.install(http.globalAgent); | ||||
|                 this.cacheable.install(https.globalAgent); | ||||
|             } else { | ||||
|                 log.debug("CacheableDnsHttpAgent", "disable"); | ||||
|                 this.cacheable.uninstall(http.globalAgent); | ||||
|                 this.cacheable.uninstall(https.globalAgent); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         this.enable = isEnable; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Attach cacheable to HTTP agent | ||||
|      * @param {http.Agent} agent Agent to install | ||||
|      */ | ||||
|     static install(agent) { | ||||
|         this.cacheable.install(agent); | ||||
|     } | ||||
| @@ -26,6 +50,10 @@ class CacheableDnsHttpAgent { | ||||
|      * @return {https.Agent} | ||||
|      */ | ||||
|     static getHttpsAgent(agentOptions) { | ||||
|         if (!this.enable) { | ||||
|             return new https.Agent(agentOptions); | ||||
|         } | ||||
|  | ||||
|         let key = JSON.stringify(agentOptions); | ||||
|         if (!(key in this.httpsAgentList)) { | ||||
|             this.httpsAgentList[key] = new https.Agent(agentOptions); | ||||
| @@ -39,6 +67,10 @@ class CacheableDnsHttpAgent { | ||||
|      * @return {https.Agents} | ||||
|      */ | ||||
|     static getHttpAgent(agentOptions) { | ||||
|         if (!this.enable) { | ||||
|             return new http.Agent(agentOptions); | ||||
|         } | ||||
|  | ||||
|         let key = JSON.stringify(agentOptions); | ||||
|         if (!(key in this.httpAgentList)) { | ||||
|             this.httpAgentList[key] = new http.Agent(agentOptions); | ||||
|   | ||||
| @@ -25,7 +25,7 @@ exports.startInterval = () => { | ||||
|             let checkBeta = await setting("checkBeta"); | ||||
|  | ||||
|             if (checkBeta && res.data.beta) { | ||||
|                 if (compareVersions.compare(res.data.beta, res.data.beta, ">")) { | ||||
|                 if (compareVersions.compare(res.data.beta, res.data.slow, ">")) { | ||||
|                     exports.latestVersion = res.data.beta; | ||||
|                     return; | ||||
|                 } | ||||
|   | ||||
| @@ -4,7 +4,8 @@ | ||||
| const { TimeLogger } = require("../src/util"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { UptimeKumaServer } = require("./uptime-kuma-server"); | ||||
| const io = UptimeKumaServer.getInstance().io; | ||||
| const server = UptimeKumaServer.getInstance(); | ||||
| const io = server.io; | ||||
| const { setting } = require("./util-server"); | ||||
| const checkVersion = require("./check-version"); | ||||
|  | ||||
| @@ -112,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 | ||||
| @@ -121,7 +147,9 @@ async function sendInfo(socket) { | ||||
|     socket.emit("info", { | ||||
|         version: checkVersion.version, | ||||
|         latestVersion: checkVersion.latestVersion, | ||||
|         primaryBaseURL: await setting("primaryBaseURL") | ||||
|         primaryBaseURL: await setting("primaryBaseURL"), | ||||
|         serverTimezone: await server.getTimezone(), | ||||
|         serverTimezoneOffset: server.getTimezoneOffset(), | ||||
|     }); | ||||
| } | ||||
|  | ||||
| @@ -154,6 +182,7 @@ module.exports = { | ||||
|     sendImportantHeartbeatList, | ||||
|     sendHeartbeatList, | ||||
|     sendProxyList, | ||||
|     sendAPIKeyList, | ||||
|     sendInfo, | ||||
|     sendDockerHostList | ||||
| }; | ||||
|   | ||||
| @@ -4,13 +4,21 @@ const demoMode = args["demo"] || false; | ||||
| const badgeConstants = { | ||||
|     naColor: "#999", | ||||
|     defaultUpColor: "#66c20a", | ||||
|     defaultWarnColor: "#eed202", | ||||
|     defaultDownColor: "#c2290a", | ||||
|     defaultPendingColor: "#f8a306", | ||||
|     defaultMaintenanceColor: "#1747f5", | ||||
|     defaultPingColor: "blue",  // as defined by badge-maker / shields.io | ||||
|     defaultStyle: "flat", | ||||
|     defaultPingValueSuffix: "ms", | ||||
|     defaultPingLabelSuffix: "h", | ||||
|     defaultUptimeValueSuffix: "%", | ||||
|     defaultUptimeLabelSuffix: "h", | ||||
|     defaultCertExpValueSuffix: " days", | ||||
|     defaultCertExpLabelSuffix: "h", | ||||
|     // Values Come From Default Notification Times | ||||
|     defaultCertExpireWarnDays: "14", | ||||
|     defaultCertExpireDownDays: "7" | ||||
| }; | ||||
|  | ||||
| module.exports = { | ||||
|   | ||||
| @@ -4,6 +4,7 @@ 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"); | ||||
|  | ||||
| /** | ||||
|  * Database & App Data Folder | ||||
| @@ -62,6 +63,17 @@ class Database { | ||||
|         "patch-add-clickable-status-page-link.sql": true, | ||||
|         "patch-add-sqlserver-monitor.sql": true, | ||||
|         "patch-add-other-auth.sql": { parents: [ "patch-monitor-basic-auth.sql" ] }, | ||||
|         "patch-grpc-monitor.sql": true, | ||||
|         "patch-add-radius-monitor.sql": true, | ||||
|         "patch-monitor-add-resend-interval.sql": true, | ||||
|         "patch-ping-packet-size.sql": true, | ||||
|         "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, | ||||
|     }; | ||||
|  | ||||
|     /** | ||||
| @@ -79,6 +91,13 @@ class Database { | ||||
|     static init(args) { | ||||
|         // Data Directory (must be end with "/") | ||||
|         Database.dataDir = process.env.DATA_DIR || args["data-dir"] || "./data/"; | ||||
|  | ||||
|         // Plugin feature is working only if the dataDir = "./data"; | ||||
|         if (Database.dataDir !== "./data/") { | ||||
|             log.warn("PLUGIN", "Warning: In order to enable plugin feature, you need to use the default data directory: ./data/"); | ||||
|             PluginsManager.disable = true; | ||||
|         } | ||||
|  | ||||
|         Database.path = Database.dataDir + "kuma.db"; | ||||
|         if (! fs.existsSync(Database.dataDir)) { | ||||
|             fs.mkdirSync(Database.dataDir, { recursive: true }); | ||||
| @@ -481,6 +500,16 @@ class Database { | ||||
|             const shmPath = Database.path + "-shm"; | ||||
|             const walPath = Database.path + "-wal"; | ||||
|  | ||||
|             // Make sure we have a backup to restore before deleting old db | ||||
|             if ( | ||||
|                 !fs.existsSync(this.backupPath) | ||||
|                 && !fs.existsSync(shmPath) | ||||
|                 && !fs.existsSync(walPath) | ||||
|             ) { | ||||
|                 log.error("db", "Backup file not found! Leaving database in failed state."); | ||||
|                 process.exit(1); | ||||
|             } | ||||
|  | ||||
|             // Delete patch failed db | ||||
|             try { | ||||
|                 if (fs.existsSync(Database.path)) { | ||||
|   | ||||
| @@ -75,7 +75,7 @@ class DockerHost { | ||||
|         if (dockerHost.dockerType === "socket") { | ||||
|             options.socketPath = dockerHost.dockerDaemon; | ||||
|         } else if (dockerHost.dockerType === "tcp") { | ||||
|             options.baseURL = dockerHost.dockerDaemon; | ||||
|             options.baseURL = DockerHost.patchDockerURL(dockerHost.dockerDaemon); | ||||
|         } | ||||
|  | ||||
|         let res = await axios.request(options); | ||||
| @@ -99,6 +99,18 @@ class DockerHost { | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Since axios 0.27.X, it does not accept `tcp://` protocol. | ||||
|      * Change it to `http://` on the fly in order to fix it. (https://github.com/louislam/uptime-kuma/issues/2165) | ||||
|      */ | ||||
|     static patchDockerURL(url) { | ||||
|         if (typeof url === "string") { | ||||
|             // Replace the first occurrence only with g | ||||
|             return url.replace(/tcp:\/\//g, "http://"); | ||||
|         } | ||||
|         return url; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|   | ||||
							
								
								
									
										24
									
								
								server/git.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								server/git.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| const childProcess = require("child_process"); | ||||
|  | ||||
| class Git { | ||||
|  | ||||
|     static clone(repoURL, cwd, targetDir = ".") { | ||||
|         let result = childProcess.spawnSync("git", [ | ||||
|             "clone", | ||||
|             repoURL, | ||||
|             targetDir, | ||||
|         ], { | ||||
|             cwd: cwd, | ||||
|         }); | ||||
|  | ||||
|         if (result.status !== 0) { | ||||
|             throw new Error(result.stderr.toString("utf-8")); | ||||
|         } else { | ||||
|             return result.stdout.toString("utf-8") + result.stderr.toString("utf-8"); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     Git, | ||||
| }; | ||||
							
								
								
									
										24
									
								
								server/google-analytics.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								server/google-analytics.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| const jsesc = require("jsesc"); | ||||
|  | ||||
| /** | ||||
|  * Returns a string that represents the javascript that is required to insert the Google Analytics scripts | ||||
|  * into a webpage. | ||||
|  * @param tagId Google UA/G/AW/DC Property ID to use with the Google Analytics script. | ||||
|  * @returns {string} | ||||
|  */ | ||||
| function getGoogleAnalyticsScript(tagId) { | ||||
|     let escapedTagId = jsesc(tagId, { isScriptContext: true }); | ||||
|  | ||||
|     if (escapedTagId) { | ||||
|         escapedTagId = escapedTagId.trim(); | ||||
|     } | ||||
|  | ||||
|     return ` | ||||
|         <script async src="https://www.googletagmanager.com/gtag/js?id=${escapedTagId}"></script> | ||||
|         <script>window.dataLayer = window.dataLayer || []; function gtag(){dataLayer.push(arguments);} gtag('js', new Date());gtag('config', '${escapedTagId}'); </script> | ||||
|     `; | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     getGoogleAnalyticsScript, | ||||
| }; | ||||
| @@ -32,6 +32,7 @@ const initBackgroundJobs = function (args) { | ||||
|     return bree; | ||||
| }; | ||||
|  | ||||
| /** Stop all background jobs if running */ | ||||
| const stopBackgroundJobs = function () { | ||||
|     if (bree) { | ||||
|         bree.stop(); | ||||
|   | ||||
| @@ -25,6 +25,10 @@ const DEFAULT_KEEP_PERIOD = 180; | ||||
|         parsedPeriod = DEFAULT_KEEP_PERIOD; | ||||
|     } | ||||
|  | ||||
|     if (parsedPeriod < 1) { | ||||
|         log(`Data deletion has been disabled as period is less than 1. Period is ${parsedPeriod} days.`); | ||||
|     } else { | ||||
|  | ||||
|         log(`Clearing Data older than ${parsedPeriod} days...`); | ||||
|  | ||||
|         try { | ||||
| @@ -35,6 +39,7 @@ const DEFAULT_KEEP_PERIOD = 180; | ||||
|         } catch (e) { | ||||
|             log(`Failed to clear old data: ${e.message}`); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     exit(); | ||||
| })(); | ||||
|   | ||||
							
								
								
									
										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,3 @@ | ||||
| const dayjs = require("dayjs"); | ||||
| const utc = require("dayjs/plugin/utc"); | ||||
| let timezone = require("dayjs/plugin/timezone"); | ||||
| dayjs.extend(utc); | ||||
| dayjs.extend(timezone); | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
|  | ||||
| /** | ||||
| @@ -10,6 +5,7 @@ const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
|  *      0 = DOWN | ||||
|  *      1 = UP | ||||
|  *      2 = PENDING | ||||
|  *      3 = MAINTENANCE | ||||
|  */ | ||||
| class Heartbeat extends BeanModel { | ||||
|  | ||||
|   | ||||
							
								
								
									
										240
									
								
								server/model/maintenance.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										240
									
								
								server/model/maintenance.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,240 @@ | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { parseTimeObject, parseTimeFromTimeObject, utcToLocal, localToUTC, log } = require("../../src/util"); | ||||
| const { timeObjectToUTC, timeObjectToLocal } = require("../util-server"); | ||||
| const { R } = require("redbean-node"); | ||||
| const dayjs = require("dayjs"); | ||||
|  | ||||
| class Maintenance extends BeanModel { | ||||
|  | ||||
|     /** | ||||
|      * Return an object that ready to parse to JSON for public | ||||
|      * Only show necessary data to public | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     async toPublicJSON() { | ||||
|  | ||||
|         let dateRange = []; | ||||
|         if (this.start_date) { | ||||
|             dateRange.push(utcToLocal(this.start_date)); | ||||
|             if (this.end_date) { | ||||
|                 dateRange.push(utcToLocal(this.end_date)); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let timeRange = []; | ||||
|         let startTime = timeObjectToLocal(parseTimeObject(this.start_time)); | ||||
|         timeRange.push(startTime); | ||||
|         let endTime = timeObjectToLocal(parseTimeObject(this.end_time)); | ||||
|         timeRange.push(endTime); | ||||
|  | ||||
|         let obj = { | ||||
|             id: this.id, | ||||
|             title: this.title, | ||||
|             description: this.description, | ||||
|             strategy: this.strategy, | ||||
|             intervalDay: this.interval_day, | ||||
|             active: !!this.active, | ||||
|             dateRange: dateRange, | ||||
|             timeRange: timeRange, | ||||
|             weekdays: (this.weekdays) ? JSON.parse(this.weekdays) : [], | ||||
|             daysOfMonth: (this.days_of_month) ? JSON.parse(this.days_of_month) : [], | ||||
|             timeslotList: [], | ||||
|         }; | ||||
|  | ||||
|         const timeslotList = await this.getTimeslotList(); | ||||
|  | ||||
|         for (let timeslot of timeslotList) { | ||||
|             obj.timeslotList.push(await timeslot.toPublicJSON()); | ||||
|         } | ||||
|  | ||||
|         if (!Array.isArray(obj.weekdays)) { | ||||
|             obj.weekdays = []; | ||||
|         } | ||||
|  | ||||
|         if (!Array.isArray(obj.daysOfMonth)) { | ||||
|             obj.daysOfMonth = []; | ||||
|         } | ||||
|  | ||||
|         // Maintenance Status | ||||
|         if (!obj.active) { | ||||
|             obj.status = "inactive"; | ||||
|         } else if (obj.strategy === "manual") { | ||||
|             obj.status = "under-maintenance"; | ||||
|         } else if (obj.timeslotList.length > 0) { | ||||
|             let currentTimestamp = dayjs().unix(); | ||||
|  | ||||
|             for (let timeslot of obj.timeslotList) { | ||||
|                 if (dayjs.utc(timeslot.startDate).unix() <= currentTimestamp && dayjs.utc(timeslot.endDate).unix() >= currentTimestamp) { | ||||
|                     log.debug("timeslot", "Timeslot ID: " + timeslot.id); | ||||
|                     log.debug("timeslot", "currentTimestamp:" + currentTimestamp); | ||||
|                     log.debug("timeslot", "timeslot.start_date:" + dayjs.utc(timeslot.startDate).unix()); | ||||
|                     log.debug("timeslot", "timeslot.end_date:" + dayjs.utc(timeslot.endDate).unix()); | ||||
|  | ||||
|                     obj.status = "under-maintenance"; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (!obj.status) { | ||||
|                 obj.status = "scheduled"; | ||||
|             } | ||||
|         } else if (obj.timeslotList.length === 0) { | ||||
|             obj.status = "ended"; | ||||
|         } else { | ||||
|             obj.status = "unknown"; | ||||
|         } | ||||
|  | ||||
|         return obj; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Only get future or current timeslots only | ||||
|      * @returns {Promise<[]>} | ||||
|      */ | ||||
|     async getTimeslotList() { | ||||
|         return R.convertToBeans("maintenance_timeslot", await R.getAll(` | ||||
|             SELECT maintenance_timeslot.* | ||||
|             FROM maintenance_timeslot, maintenance | ||||
|             WHERE maintenance_timeslot.maintenance_id = maintenance.id | ||||
|             AND maintenance.id = ? | ||||
|             AND ${Maintenance.getActiveAndFutureMaintenanceSQLCondition()} | ||||
|         `, [ | ||||
|             this.id | ||||
|         ])); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Return an object that ready to parse to JSON | ||||
|      * @param {string} timezone If not specified, the timeRange will be in UTC | ||||
|      * @returns {Object} | ||||
|      */ | ||||
|     async toJSON(timezone = null) { | ||||
|         return this.toPublicJSON(timezone); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a list of weekdays that the maintenance is active for | ||||
|      * Monday=1, Tuesday=2 etc. | ||||
|      * @returns {number[]} Array of active weekdays | ||||
|      */ | ||||
|     getDayOfWeekList() { | ||||
|         log.debug("timeslot", "List: " + this.weekdays); | ||||
|         return JSON.parse(this.weekdays).sort(function (a, b) { | ||||
|             return a - b; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get a list of days in month that maintenance is active for | ||||
|      * @returns {number[]} Array of active days in month | ||||
|      */ | ||||
|     getDayOfMonthList() { | ||||
|         return JSON.parse(this.days_of_month).sort(function (a, b) { | ||||
|             return a - b; | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 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 | ||||
|      * @returns {number} Duration of maintenance | ||||
|      */ | ||||
|     getDuration() { | ||||
|         let duration = dayjs.utc(this.end_time, "HH:mm").diff(dayjs.utc(this.start_time, "HH:mm"), "second"); | ||||
|         // Add 24hours if it is across day | ||||
|         if (duration < 0) { | ||||
|             duration += 24 * 3600; | ||||
|         } | ||||
|         return duration; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Convert data from socket to bean | ||||
|      * @param {Bean} bean Bean to fill in | ||||
|      * @param {Object} obj Data to fill bean with | ||||
|      * @returns {Bean} Filled bean | ||||
|      */ | ||||
|     static jsonToBean(bean, obj) { | ||||
|         if (obj.id) { | ||||
|             bean.id = obj.id; | ||||
|         } | ||||
|  | ||||
|         // Apply timezone offset to timeRange, as it cannot apply automatically. | ||||
|         if (obj.timeRange[0]) { | ||||
|             timeObjectToUTC(obj.timeRange[0]); | ||||
|             if (obj.timeRange[1]) { | ||||
|                 timeObjectToUTC(obj.timeRange[1]); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         bean.title = obj.title; | ||||
|         bean.description = obj.description; | ||||
|         bean.strategy = obj.strategy; | ||||
|         bean.interval_day = obj.intervalDay; | ||||
|         bean.active = obj.active; | ||||
|  | ||||
|         if (obj.dateRange[0]) { | ||||
|             bean.start_date = localToUTC(obj.dateRange[0]); | ||||
|  | ||||
|             if (obj.dateRange[1]) { | ||||
|                 bean.end_date = localToUTC(obj.dateRange[1]); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         bean.start_time = parseTimeFromTimeObject(obj.timeRange[0]); | ||||
|         bean.end_time = parseTimeFromTimeObject(obj.timeRange[1]); | ||||
|  | ||||
|         bean.weekdays = JSON.stringify(obj.weekdays); | ||||
|         bean.days_of_month = JSON.stringify(obj.daysOfMonth); | ||||
|  | ||||
|         return bean; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SQL conditions for active maintenance | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     static getActiveMaintenanceSQLCondition() { | ||||
|         return ` | ||||
|             ( | ||||
|                 (maintenance_timeslot.start_date <= DATETIME('now') | ||||
|                 AND maintenance_timeslot.end_date >= DATETIME('now') | ||||
|                 AND maintenance.active = 1) | ||||
|                 OR | ||||
|                 (maintenance.strategy = 'manual' AND active = 1) | ||||
|             ) | ||||
|         `; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SQL conditions for active and future maintenance | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     static getActiveAndFutureMaintenanceSQLCondition() { | ||||
|         return ` | ||||
|             ( | ||||
|                 ((maintenance_timeslot.end_date >= DATETIME('now') | ||||
|                 AND maintenance.active = 1) | ||||
|                 OR | ||||
|                 (maintenance.strategy = 'manual' AND active = 1)) | ||||
|             ) | ||||
|         `; | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Maintenance; | ||||
							
								
								
									
										223
									
								
								server/model/maintenance_timeslot.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										223
									
								
								server/model/maintenance_timeslot.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,223 @@ | ||||
| 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) { | ||||
|         log.info("maintenance", "Generate Timeslot for maintenance id: " + maintenance.id); | ||||
|  | ||||
|         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; | ||||
|  | ||||
|             if (!await this.isDuplicateTimeslot(bean)) { | ||||
|                 await R.store(bean); | ||||
|                 return bean; | ||||
|             } else { | ||||
|                 log.debug("maintenance", "Duplicate timeslot, skip"); | ||||
|                 return null; | ||||
|             } | ||||
|  | ||||
|         } 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"); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     static async isDuplicateTimeslot(timeslot) { | ||||
|         let bean = await R.findOne("maintenance_timeslot", "maintenance_id = ? AND start_date = ? AND end_date = ?", [ | ||||
|             timeslot.maintenance_id, | ||||
|             timeslot.start_date, | ||||
|             timeslot.end_date | ||||
|         ]); | ||||
|         return bean !== null; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * 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; | ||||
|  | ||||
|         if (!await this.isDuplicateTimeslot(bean)) { | ||||
|             await R.store(bean); | ||||
|             return bean; | ||||
|         } else { | ||||
|             log.debug("maintenance", "Duplicate timeslot, skip"); | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = MaintenanceTimeslot; | ||||
| @@ -1,13 +1,11 @@ | ||||
| const https = require("https"); | ||||
| const dayjs = require("dayjs"); | ||||
| const utc = require("dayjs/plugin/utc"); | ||||
| let timezone = require("dayjs/plugin/timezone"); | ||||
| dayjs.extend(utc); | ||||
| dayjs.extend(timezone); | ||||
| const axios = require("axios"); | ||||
| const { Prometheus } = require("../prometheus"); | ||||
| const { log, UP, DOWN, PENDING, flipStatus, TimeLogger } = require("../../src/util"); | ||||
| const { tcping, ping, dnsResolve, checkCertificate, checkStatusCode, getTotalClientInRoom, setting, mssqlQuery, postgresQuery, mqttAsync, setSetting, httpNtlm } = require("../util-server"); | ||||
| const { log, UP, DOWN, PENDING, MAINTENANCE, flipStatus, TimeLogger, MAX_INTERVAL_SECOND, MIN_INTERVAL_SECOND } = require("../../src/util"); | ||||
| const { tcping, ping, dnsResolve, checkCertificate, checkStatusCode, getTotalClientInRoom, setting, mssqlQuery, postgresQuery, mysqlQuery, mqttAsync, setSetting, httpNtlm, radius, grpcQuery, | ||||
|     redisPingAsync, mongodbPing, | ||||
| } = require("../util-server"); | ||||
| const { R } = require("redbean-node"); | ||||
| const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { Notification } = require("../notification"); | ||||
| @@ -17,12 +15,17 @@ const version = require("../../package.json").version; | ||||
| const apicache = require("../modules/apicache"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const { CacheableDnsHttpAgent } = require("../cacheable-dns-http-agent"); | ||||
| const { DockerHost } = require("../docker"); | ||||
| const Maintenance = require("./maintenance"); | ||||
| const { UptimeCacheList } = require("../uptime-cache-list"); | ||||
| const Gamedig = require("gamedig"); | ||||
|  | ||||
| /** | ||||
|  * status: | ||||
|  *      0 = DOWN | ||||
|  *      1 = UP | ||||
|  *      2 = PENDING | ||||
|  *      3 = MAINTENANCE | ||||
|  */ | ||||
| class Monitor extends BeanModel { | ||||
|  | ||||
| @@ -69,6 +72,7 @@ class Monitor extends BeanModel { | ||||
|         let data = { | ||||
|             id: this.id, | ||||
|             name: this.name, | ||||
|             description: this.description, | ||||
|             url: this.url, | ||||
|             method: this.method, | ||||
|             hostname: this.hostname, | ||||
| @@ -79,30 +83,36 @@ class Monitor extends BeanModel { | ||||
|             type: this.type, | ||||
|             interval: this.interval, | ||||
|             retryInterval: this.retryInterval, | ||||
|             resendInterval: this.resendInterval, | ||||
|             keyword: this.keyword, | ||||
|             expiryNotification: this.isEnabledExpiryNotification(), | ||||
|             ignoreTls: this.getIgnoreTls(), | ||||
|             upsideDown: this.isUpsideDown(), | ||||
|             packetSize: this.packetSize, | ||||
|             maxredirects: this.maxredirects, | ||||
|             accepted_statuscodes: this.getAcceptedStatuscodes(), | ||||
|             dns_resolve_type: this.dns_resolve_type, | ||||
|             dns_resolve_server: this.dns_resolve_server, | ||||
|             dns_last_result: this.dns_last_result, | ||||
|             pushToken: this.pushToken, | ||||
|             docker_container: this.docker_container, | ||||
|             docker_host: this.docker_host, | ||||
|             proxyId: this.proxy_id, | ||||
|             notificationIDList, | ||||
|             tags: tags, | ||||
|             mqttUsername: this.mqttUsername, | ||||
|             mqttPassword: this.mqttPassword, | ||||
|             maintenance: await Monitor.isUnderMaintenance(this.id), | ||||
|             mqttTopic: this.mqttTopic, | ||||
|             mqttSuccessMessage: this.mqttSuccessMessage, | ||||
|             databaseConnectionString: this.databaseConnectionString, | ||||
|             databaseQuery: this.databaseQuery, | ||||
|             authMethod: this.authMethod, | ||||
|             authWorkstation: this.authWorkstation, | ||||
|             authDomain: this.authDomain, | ||||
|             grpcUrl: this.grpcUrl, | ||||
|             grpcProtobuf: this.grpcProtobuf, | ||||
|             grpcMethod: this.grpcMethod, | ||||
|             grpcServiceName: this.grpcServiceName, | ||||
|             grpcEnableTls: this.getGrpcEnableTls(), | ||||
|             radiusCalledStationId: this.radiusCalledStationId, | ||||
|             radiusCallingStationId: this.radiusCallingStationId, | ||||
|             game: this.game, | ||||
|             httpBodyEncoding: this.httpBodyEncoding | ||||
|         }; | ||||
|  | ||||
|         if (includeSensitiveData) { | ||||
| @@ -110,12 +120,26 @@ class Monitor extends BeanModel { | ||||
|                 ...data, | ||||
|                 headers: this.headers, | ||||
|                 body: this.body, | ||||
|                 grpcBody: this.grpcBody, | ||||
|                 grpcMetadata: this.grpcMetadata, | ||||
|                 basic_auth_user: this.basic_auth_user, | ||||
|                 basic_auth_pass: this.basic_auth_pass, | ||||
|                 pushToken: this.pushToken, | ||||
|                 databaseConnectionString: this.databaseConnectionString, | ||||
|                 radiusUsername: this.radiusUsername, | ||||
|                 radiusPassword: this.radiusPassword, | ||||
|                 radiusSecret: this.radiusSecret, | ||||
|                 mqttUsername: this.mqttUsername, | ||||
|                 mqttPassword: this.mqttPassword, | ||||
|                 authWorkstation: this.authWorkstation, | ||||
|                 authDomain: this.authDomain, | ||||
|                 tlsCa: this.tlsCa, | ||||
|                 tlsCert: this.tlsCert, | ||||
|                 tlsKey: this.tlsKey, | ||||
|             }; | ||||
|         } | ||||
|  | ||||
|         data.includeSensitiveData = includeSensitiveData; | ||||
|         return data; | ||||
|     } | ||||
|  | ||||
| @@ -124,7 +148,7 @@ class Monitor extends BeanModel { | ||||
|      * @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 ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -160,6 +184,14 @@ class Monitor extends BeanModel { | ||||
|         return Boolean(this.upsideDown); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Parse to boolean | ||||
|      * @returns {boolean} | ||||
|      */ | ||||
|     getGrpcEnableTls() { | ||||
|         return Boolean(this.grpcEnableTls); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get accepted status codes | ||||
|      * @returns {Object} | ||||
| @@ -176,7 +208,7 @@ class Monitor extends BeanModel { | ||||
|         let previousBeat = null; | ||||
|         let retries = 0; | ||||
|  | ||||
|         let prometheus = new Prometheus(this); | ||||
|         this.prometheus = new Prometheus(this); | ||||
|  | ||||
|         const beat = async () => { | ||||
|  | ||||
| @@ -209,6 +241,7 @@ class Monitor extends BeanModel { | ||||
|             bean.monitor_id = this.id; | ||||
|             bean.time = R.isoDateTimeMillis(dayjs.utc()); | ||||
|             bean.status = DOWN; | ||||
|             bean.downCount = previousBeat?.downCount || 0; | ||||
|  | ||||
|             if (this.isUpsideDown()) { | ||||
|                 bean.status = flipStatus(bean.status); | ||||
| @@ -222,7 +255,10 @@ class Monitor extends BeanModel { | ||||
|             } | ||||
|  | ||||
|             try { | ||||
|                 if (this.type === "http" || this.type === "keyword") { | ||||
|                 if (await Monitor.isUnderMaintenance(this.id)) { | ||||
|                     bean.msg = "Monitor under maintenance"; | ||||
|                     bean.status = MAINTENANCE; | ||||
|                 } else if (this.type === "http" || this.type === "keyword") { | ||||
|                     // Do not do any queries/high loading things before the "bean.ping" | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
| @@ -241,16 +277,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) => { | ||||
| @@ -258,6 +312,10 @@ class Monitor extends BeanModel { | ||||
|                         }, | ||||
|                     }; | ||||
|  | ||||
|                     if (bodyValue) { | ||||
|                         options.data = bodyValue; | ||||
|                     } | ||||
|  | ||||
|                     if (this.proxy_id) { | ||||
|                         const proxy = await R.load("proxy", this.proxy_id); | ||||
|  | ||||
| @@ -276,23 +334,23 @@ 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`); | ||||
|  | ||||
|                     let res; | ||||
|                     if (this.auth_method === "ntlm") { | ||||
|                         options.httpsAgent.keepAlive = true; | ||||
|  | ||||
|                         res = await httpNtlm(options, { | ||||
|                             username: this.basic_auth_user, | ||||
|                             password: this.basic_auth_pass, | ||||
|                             domain: this.authDomain, | ||||
|                             workstation: this.authWorkstation ? this.authWorkstation : undefined | ||||
|                         }); | ||||
|  | ||||
|                     } else { | ||||
|                         res = await axios.request(options); | ||||
|                     } | ||||
|                     // Make Request | ||||
|                     let res = await this.makeAxiosRequest(options); | ||||
|  | ||||
|                     bean.msg = `${res.status} - ${res.statusText}`; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
| @@ -356,7 +414,7 @@ class Monitor extends BeanModel { | ||||
|                     bean.status = UP; | ||||
|  | ||||
|                 } else if (this.type === "ping") { | ||||
|                     bean.ping = await ping(this.hostname); | ||||
|                     bean.ping = await ping(this.hostname, this.packetSize); | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                 } else if (this.type === "dns") { | ||||
| @@ -466,39 +524,60 @@ class Monitor extends BeanModel { | ||||
|                         bean.msg = res.data.response.servers[0].name; | ||||
|  | ||||
|                         try { | ||||
|                             bean.ping = await ping(this.hostname); | ||||
|                             bean.ping = await ping(this.hostname, this.packetSize); | ||||
|                         } catch (_) { } | ||||
|                     } else { | ||||
|                         throw new Error("Server not found on Steam"); | ||||
|                     } | ||||
|                 } else if (this.type === "gamedig") { | ||||
|                     try { | ||||
|                         const state = await Gamedig.query({ | ||||
|                             type: this.game, | ||||
|                             host: this.hostname, | ||||
|                             port: this.port, | ||||
|                             givenPortOnly: true, | ||||
|                         }); | ||||
|  | ||||
|                         bean.msg = state.name; | ||||
|                         bean.status = UP; | ||||
|                         bean.ping = state.ping; | ||||
|                     } catch (e) { | ||||
|                         throw new Error(e.message); | ||||
|                     } | ||||
|                 } else if (this.type === "docker") { | ||||
|                     log.debug(`[${this.name}] Prepare Options for Axios`); | ||||
|                     log.debug("monitor", `[${this.name}] Prepare Options for Axios`); | ||||
|  | ||||
|                     const dockerHost = await R.load("docker_host", this.docker_host); | ||||
|  | ||||
|                     const options = { | ||||
|                         url: `/containers/${this.docker_container}/json`, | ||||
|                         timeout: this.interval * 1000 * 0.8, | ||||
|                         headers: { | ||||
|                             "Accept": "*/*", | ||||
|                             "User-Agent": "Uptime-Kuma/" + version, | ||||
|                         }, | ||||
|                         httpsAgent: new https.Agent({ | ||||
|                         httpsAgent: CacheableDnsHttpAgent.getHttpsAgent({ | ||||
|                             maxCachedSessions: 0,      // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940) | ||||
|                             rejectUnauthorized: !this.getIgnoreTls(), | ||||
|                         }), | ||||
|                         httpAgent: CacheableDnsHttpAgent.getHttpAgent({ | ||||
|                             maxCachedSessions: 0, | ||||
|                         }), | ||||
|                     }; | ||||
|  | ||||
|                     if (dockerHost._dockerType === "socket") { | ||||
|                         options.socketPath = dockerHost._dockerDaemon; | ||||
|                     } else if (dockerHost._dockerType === "tcp") { | ||||
|                         options.baseURL = dockerHost._dockerDaemon; | ||||
|                         options.baseURL = DockerHost.patchDockerURL(dockerHost._dockerDaemon); | ||||
|                     } | ||||
|  | ||||
|                     log.debug(`[${this.name}] Axios Request`); | ||||
|                     log.debug("monitor", `[${this.name}] Axios Request`); | ||||
|                     let res = await axios.request(options); | ||||
|                     if (res.data.State.Running) { | ||||
|                         bean.status = UP; | ||||
|                         bean.msg = ""; | ||||
|                         bean.msg = res.data.State.Status; | ||||
|                     } else { | ||||
|                         throw Error("Container State is " + res.data.State.Status); | ||||
|                     } | ||||
|                 } else if (this.type === "mqtt") { | ||||
|                     bean.msg = await mqttAsync(this.hostname, this.mqttTopic, this.mqttSuccessMessage, { | ||||
| @@ -516,6 +595,37 @@ class Monitor extends BeanModel { | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "grpc-keyword") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|                     const options = { | ||||
|                         grpcUrl: this.grpcUrl, | ||||
|                         grpcProtobufData: this.grpcProtobuf, | ||||
|                         grpcServiceName: this.grpcServiceName, | ||||
|                         grpcEnableTls: this.grpcEnableTls, | ||||
|                         grpcMethod: this.grpcMethod, | ||||
|                         grpcBody: this.grpcBody, | ||||
|                         keyword: this.keyword | ||||
|                     }; | ||||
|                     const response = await grpcQuery(options); | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                     log.debug("monitor:", `gRPC response: ${JSON.stringify(response)}`); | ||||
|                     let responseData = response.data; | ||||
|                     if (responseData.length > 50) { | ||||
|                         responseData = responseData.toString().substring(0, 47) + "..."; | ||||
|                     } | ||||
|                     if (response.code !== 1) { | ||||
|                         bean.status = DOWN; | ||||
|                         bean.msg = `Error in send gRPC ${response.code} ${response.errorMessage}`; | ||||
|                     } else { | ||||
|                         if (response.data.toString().includes(this.keyword)) { | ||||
|                             bean.status = UP; | ||||
|                             bean.msg = `${responseData}, keyword [${this.keyword}] is found`; | ||||
|                         } else { | ||||
|                             log.debug("monitor:", `GRPC response [${response.data}] + ", but keyword [${this.keyword}] is not in [" + ${response.data} + "]"`); | ||||
|                             bean.status = DOWN; | ||||
|                             bean.msg = `, but keyword [${this.keyword}] is not in [" + ${responseData} + "]`; | ||||
|                         } | ||||
|                     } | ||||
|                 } else if (this.type === "postgres") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
| @@ -524,9 +634,76 @@ class Monitor extends BeanModel { | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "mysql") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     await mysqlQuery(this.databaseConnectionString, this.databaseQuery); | ||||
|  | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "mongodb") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     await mongodbPing(this.databaseConnectionString); | ||||
|  | ||||
|                     bean.msg = ""; | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|  | ||||
|                 } else if (this.type === "radius") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     // Handle monitors that were created before the | ||||
|                     // update and as such don't have a value for | ||||
|                     // this.port. | ||||
|                     let port; | ||||
|                     if (this.port == null) { | ||||
|                         port = 1812; | ||||
|                     } else { | ||||
|                     bean.msg = "Unknown Monitor Type"; | ||||
|                     bean.status = PENDING; | ||||
|                         port = this.port; | ||||
|                     } | ||||
|  | ||||
|                     try { | ||||
|                         const resp = await radius( | ||||
|                             this.hostname, | ||||
|                             this.radiusUsername, | ||||
|                             this.radiusPassword, | ||||
|                             this.radiusCalledStationId, | ||||
|                             this.radiusCallingStationId, | ||||
|                             this.radiusSecret, | ||||
|                             port | ||||
|                         ); | ||||
|                         if (resp.code) { | ||||
|                             bean.msg = resp.code; | ||||
|                         } | ||||
|                         bean.status = UP; | ||||
|                     } catch (error) { | ||||
|                         bean.status = DOWN; | ||||
|                         if (error.response?.code) { | ||||
|                             bean.msg = error.response.code; | ||||
|                         } else { | ||||
|                             bean.msg = error.message; | ||||
|                         } | ||||
|                     } | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|                 } else if (this.type === "redis") { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|  | ||||
|                     bean.msg = await redisPingAsync(this.databaseConnectionString); | ||||
|                     bean.status = UP; | ||||
|                     bean.ping = dayjs().valueOf() - startTime; | ||||
|  | ||||
|                 } else if (this.type in UptimeKumaServer.monitorTypeList) { | ||||
|                     let startTime = dayjs().valueOf(); | ||||
|                     const monitorType = UptimeKumaServer.monitorTypeList[this.type]; | ||||
|                     await monitorType.check(this, bean); | ||||
|                     if (!bean.ping) { | ||||
|                         bean.ping = dayjs().valueOf() - startTime; | ||||
|                     } | ||||
|  | ||||
|                 } else { | ||||
|                     throw new Error("Unknown Monitor Type"); | ||||
|                 } | ||||
|  | ||||
|                 if (this.isUpsideDown()) { | ||||
| @@ -562,15 +739,36 @@ class Monitor extends BeanModel { | ||||
|             if (isImportant) { | ||||
|                 bean.important = true; | ||||
|  | ||||
|                 if (Monitor.isImportantForNotification(isFirstBeat, previousBeat?.status, bean.status)) { | ||||
|                     log.debug("monitor", `[${this.name}] sendNotification`); | ||||
|                     await Monitor.sendNotification(isFirstBeat, this, bean); | ||||
|                 } else { | ||||
|                     log.debug("monitor", `[${this.name}] will not sendNotification because it is (or was) under maintenance`); | ||||
|                 } | ||||
|  | ||||
|                 // Reset down count | ||||
|                 bean.downCount = 0; | ||||
|  | ||||
|                 // Clear Status Page Cache | ||||
|                 log.debug("monitor", `[${this.name}] apicache clear`); | ||||
|                 apicache.clear(); | ||||
|  | ||||
|                 UptimeKumaServer.getInstance().sendMaintenanceListByUserID(this.user_id); | ||||
|  | ||||
|             } else { | ||||
|                 bean.important = false; | ||||
|  | ||||
|                 if (bean.status === DOWN && this.resendInterval > 0) { | ||||
|                     ++bean.downCount; | ||||
|                     if (bean.downCount >= this.resendInterval) { | ||||
|                         // Send notification again, because we are still DOWN | ||||
|                         log.debug("monitor", `[${this.name}] sendNotification again: Down Count: ${bean.downCount} | Resend Interval: ${this.resendInterval}`); | ||||
|                         await Monitor.sendNotification(isFirstBeat, this, bean); | ||||
|  | ||||
|                         // Reset down count | ||||
|                         bean.downCount = 0; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             if (bean.status === UP) { | ||||
| @@ -580,11 +778,14 @@ class Monitor extends BeanModel { | ||||
|                     beatInterval = this.retryInterval; | ||||
|                 } | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Pending: ${bean.msg} | Max retries: ${this.maxretries} | Retry: ${retries} | Retry Interval: ${beatInterval} seconds | Type: ${this.type}`); | ||||
|             } else if (bean.status === MAINTENANCE) { | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Under Maintenance | Type: ${this.type}`); | ||||
|             } else { | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Failing: ${bean.msg} | Interval: ${beatInterval} seconds | Type: ${this.type}`); | ||||
|                 log.warn("monitor", `Monitor #${this.id} '${this.name}': Failing: ${bean.msg} | Interval: ${beatInterval} seconds | Type: ${this.type} | Down Count: ${bean.downCount} | Resend Interval: ${this.resendInterval}`); | ||||
|             } | ||||
|  | ||||
|             log.debug("monitor", `[${this.name}] Send to socket`); | ||||
|             UptimeCacheList.clearCache(this.id); | ||||
|             io.to(this.user_id).emit("heartbeat", bean.toJSON()); | ||||
|             Monitor.sendStats(io, this.id, this.user_id); | ||||
|  | ||||
| @@ -592,7 +793,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; | ||||
|  | ||||
| @@ -631,20 +832,60 @@ class Monitor extends BeanModel { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Make a request using axios | ||||
|      * @param {Object} options Options for Axios | ||||
|      * @param {boolean} finalCall Should this be the final call i.e | ||||
|      * don't retry on faliure | ||||
|      * @returns {Object} Axios response | ||||
|      */ | ||||
|     async makeAxiosRequest(options, finalCall = false) { | ||||
|         try { | ||||
|             let res; | ||||
|             if (this.auth_method === "ntlm") { | ||||
|                 options.httpsAgent.keepAlive = true; | ||||
|  | ||||
|                 res = await httpNtlm(options, { | ||||
|                     username: this.basic_auth_user, | ||||
|                     password: this.basic_auth_pass, | ||||
|                     domain: this.authDomain, | ||||
|                     workstation: this.authWorkstation ? this.authWorkstation : undefined | ||||
|                 }); | ||||
|             } else { | ||||
|                 res = await axios.request(options); | ||||
|             } | ||||
|  | ||||
|             return res; | ||||
|         } catch (e) { | ||||
|             // Fix #2253 | ||||
|             // Read more: https://stackoverflow.com/questions/1759956/curl-error-18-transfer-closed-with-outstanding-read-data-remaining | ||||
|             if (!finalCall && typeof e.message === "string" && e.message.includes("maxContentLength size of -1 exceeded")) { | ||||
|                 log.debug("monitor", "makeAxiosRequest with gzip"); | ||||
|                 options.headers["Accept-Encoding"] = "gzip, deflate"; | ||||
|                 return this.makeAxiosRequest(options, true); | ||||
|             } else { | ||||
|                 if (typeof e.message === "string" && e.message.includes("maxContentLength size of -1 exceeded")) { | ||||
|                     e.message = "response timeout: incomplete response within a interval"; | ||||
|                 } | ||||
|                 throw e; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** Stop monitor */ | ||||
|     stop() { | ||||
|         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; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -769,7 +1010,15 @@ class Monitor extends BeanModel { | ||||
|      * @param {number} duration Hours | ||||
|      * @param {number} monitorID ID of monitor to calculate | ||||
|      */ | ||||
|     static async calcUptime(duration, monitorID) { | ||||
|     static async calcUptime(duration, monitorID, forceNoCache = false) { | ||||
|  | ||||
|         if (!forceNoCache) { | ||||
|             let cachedUptime = UptimeCacheList.getUptime(monitorID, duration); | ||||
|             if (cachedUptime != null) { | ||||
|                 return cachedUptime; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         const timeLogger = new TimeLogger(); | ||||
|  | ||||
|         const startTime = R.isoDateTime(dayjs.utc().subtract(duration, "hour")); | ||||
| @@ -790,7 +1039,7 @@ class Monitor extends BeanModel { | ||||
|                -- SUM all uptime duration, also trim off the beat out of time window | ||||
|                 SUM( | ||||
|                     CASE | ||||
|                         WHEN (status = 1) | ||||
|                         WHEN (status = 1 OR status = 3) | ||||
|                         THEN | ||||
|                             CASE | ||||
|                                 WHEN (JULIANDAY(\`time\`) - JULIANDAY(?)) * 86400 < duration | ||||
| @@ -828,6 +1077,9 @@ class Monitor extends BeanModel { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Cache | ||||
|         UptimeCacheList.addUptime(monitorID, duration, uptime); | ||||
|  | ||||
|         return uptime; | ||||
|     } | ||||
|  | ||||
| @@ -861,11 +1113,49 @@ class Monitor extends BeanModel { | ||||
|         // DOWN -> PENDING = this case not exists | ||||
|         // DOWN -> DOWN = not important | ||||
|         // * DOWN -> UP = important | ||||
|         let isImportant = isFirstBeat || | ||||
|         // MAINTENANCE -> MAINTENANCE = not important | ||||
|         // * MAINTENANCE -> UP = important | ||||
|         // * MAINTENANCE -> DOWN = important | ||||
|         // * DOWN -> MAINTENANCE = important | ||||
|         // * UP -> MAINTENANCE = important | ||||
|         return isFirstBeat || | ||||
|             (previousBeatStatus === DOWN && currentBeatStatus === MAINTENANCE) || | ||||
|             (previousBeatStatus === UP && currentBeatStatus === MAINTENANCE) || | ||||
|             (previousBeatStatus === MAINTENANCE && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === MAINTENANCE && currentBeatStatus === UP) || | ||||
|             (previousBeatStatus === UP && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === DOWN && currentBeatStatus === UP) || | ||||
|             (previousBeatStatus === PENDING && currentBeatStatus === DOWN); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Is this beat important for notifications? | ||||
|      * @param {boolean} isFirstBeat Is this the first beat of this monitor? | ||||
|      * @param {const} previousBeatStatus Status of the previous beat | ||||
|      * @param {const} currentBeatStatus Status of the current beat | ||||
|      * @returns {boolean} True if is an important beat else false | ||||
|      */ | ||||
|     static isImportantForNotification(isFirstBeat, previousBeatStatus, currentBeatStatus) { | ||||
|         // * ? -> ANY STATUS = important [isFirstBeat] | ||||
|         // UP -> PENDING = not important | ||||
|         // * UP -> DOWN = important | ||||
|         // UP -> UP = not important | ||||
|         // PENDING -> PENDING = not important | ||||
|         // * PENDING -> DOWN = important | ||||
|         // PENDING -> UP = not important | ||||
|         // DOWN -> PENDING = this case not exists | ||||
|         // DOWN -> DOWN = not important | ||||
|         // * DOWN -> UP = important | ||||
|         // MAINTENANCE -> MAINTENANCE = not important | ||||
|         // MAINTENANCE -> UP = not important | ||||
|         // * MAINTENANCE -> DOWN = important | ||||
|         // DOWN -> MAINTENANCE = not important | ||||
|         // UP -> MAINTENANCE = not important | ||||
|         return isFirstBeat || | ||||
|             (previousBeatStatus === MAINTENANCE && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === UP && currentBeatStatus === DOWN) || | ||||
|             (previousBeatStatus === DOWN && currentBeatStatus === UP) || | ||||
|             (previousBeatStatus === PENDING && currentBeatStatus === DOWN); | ||||
|         return isImportant; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
| @@ -889,7 +1179,13 @@ class Monitor extends BeanModel { | ||||
|  | ||||
|             for (let notification of notificationList) { | ||||
|                 try { | ||||
|                     await Notification.send(JSON.parse(notification.config), msg, await monitor.toJSON(false), bean.toJSON()); | ||||
|                     // Prevent if the msg is undefined, notifications such as Discord cannot send out. | ||||
|                     const heartbeatJSON = bean.toJSON(); | ||||
|                     if (!heartbeatJSON["msg"]) { | ||||
|                         heartbeatJSON["msg"] = "N/A"; | ||||
|                     } | ||||
|  | ||||
|                     await Notification.send(JSON.parse(notification.config), msg, await monitor.toJSON(false), heartbeatJSON); | ||||
|                 } catch (e) { | ||||
|                     log.error("monitor", "Cannot send notification to " + notification.name); | ||||
|                     log.error("monitor", e); | ||||
| @@ -969,7 +1265,7 @@ class Monitor extends BeanModel { | ||||
|             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`); | ||||
|                     await Notification.send(JSON.parse(notification.config), `[${this.name}][${this.url}] Certificate will expire in ${daysRemaining} days`); | ||||
|                     sent = true; | ||||
|                 } catch (e) { | ||||
|                     log.error("monitor", "Cannot send cert notification to " + notification.name); | ||||
| @@ -996,12 +1292,42 @@ class Monitor extends BeanModel { | ||||
|      */ | ||||
|     static async getPreviousHeartbeat(monitorID) { | ||||
|         return await R.getRow(` | ||||
|             SELECT status, time FROM heartbeat | ||||
|             SELECT ping, status, time FROM heartbeat | ||||
|             WHERE id = (select MAX(id) from heartbeat where monitor_id = ?) | ||||
|         `, [ | ||||
|             monitorID | ||||
|         ]); | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Check if monitor is under maintenance | ||||
|      * @param {number} monitorID ID of monitor to check | ||||
|      * @returns {Promise<boolean>} | ||||
|      */ | ||||
|     static async isUnderMaintenance(monitorID) { | ||||
|         let activeCondition = Maintenance.getActiveMaintenanceSQLCondition(); | ||||
|         const maintenance = await R.getRow(` | ||||
|             SELECT COUNT(*) AS count | ||||
|             FROM monitor_maintenance mm | ||||
|             JOIN maintenance | ||||
|                 ON mm.maintenance_id = maintenance.id | ||||
|                 AND mm.monitor_id = ? | ||||
|             LEFT JOIN maintenance_timeslot | ||||
|                 ON maintenance_timeslot.maintenance_id = maintenance.id | ||||
|             WHERE ${activeCondition} | ||||
|             LIMIT 1`, [ monitorID ]); | ||||
|         return maintenance.count !== 0; | ||||
|     } | ||||
|  | ||||
|     /** Make sure monitor interval is between bounds */ | ||||
|     validate() { | ||||
|         if (this.interval > MAX_INTERVAL_SECOND) { | ||||
|             throw new Error(`Interval cannot be more than ${MAX_INTERVAL_SECOND} seconds`); | ||||
|         } | ||||
|         if (this.interval < MIN_INTERVAL_SECOND) { | ||||
|             throw new Error(`Interval cannot be less than ${MIN_INTERVAL_SECOND} seconds`); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Monitor; | ||||
|   | ||||
| @@ -2,6 +2,9 @@ const { BeanModel } = require("redbean-node/dist/bean-model"); | ||||
| const { R } = require("redbean-node"); | ||||
| const cheerio = require("cheerio"); | ||||
| const { UptimeKumaServer } = require("../uptime-kuma-server"); | ||||
| const jsesc = require("jsesc"); | ||||
| const Maintenance = require("./maintenance"); | ||||
| const googleAnalytics = require("../google-analytics"); | ||||
|  | ||||
| class StatusPage extends BeanModel { | ||||
|  | ||||
| @@ -36,7 +39,7 @@ class StatusPage extends BeanModel { | ||||
|      */ | ||||
|     static async renderHTML(indexHTML, statusPage) { | ||||
|         const $ = cheerio.load(indexHTML); | ||||
|         const description155 = statusPage.description?.substring(0, 155); | ||||
|         const description155 = statusPage.description?.substring(0, 155) ?? ""; | ||||
|  | ||||
|         $("title").text(statusPage.title); | ||||
|         $("meta[name=description]").attr("content", description155); | ||||
| @@ -51,18 +54,32 @@ class StatusPage extends BeanModel { | ||||
|  | ||||
|         const head = $("head"); | ||||
|  | ||||
|         if (statusPage.googleAnalyticsTagId) { | ||||
|             let escapedGoogleAnalyticsScript = googleAnalytics.getGoogleAnalyticsScript(statusPage.googleAnalyticsTagId); | ||||
|             head.append($(escapedGoogleAnalyticsScript)); | ||||
|         } | ||||
|  | ||||
|         // OG Meta Tags | ||||
|         head.append(`<meta property="og:title" content="${statusPage.title}" />`); | ||||
|         head.append(`<meta property="og:description" content="${description155}" />`); | ||||
|         let ogTitle = $("<meta property=\"og:title\" content=\"\" />").attr("content", statusPage.title); | ||||
|         head.append(ogTitle); | ||||
|  | ||||
|         let ogDescription = $("<meta property=\"og:description\" content=\"\" />").attr("content", description155); | ||||
|         head.append(ogDescription); | ||||
|  | ||||
|         // Preload data | ||||
|         const json = JSON.stringify(await StatusPage.getStatusPageData(statusPage)); | ||||
|         head.append(` | ||||
|             <script> | ||||
|                 window.preloadData = ${json} | ||||
|         // Add jsesc, fix https://github.com/louislam/uptime-kuma/issues/2186 | ||||
|         const escapedJSONObject = jsesc(await StatusPage.getStatusPageData(statusPage), { | ||||
|             "isScriptContext": true | ||||
|         }); | ||||
|  | ||||
|         const script = $(` | ||||
|             <script id="preload-data" data-json="{}"> | ||||
|                 window.preloadData = ${escapedJSONObject}; | ||||
|             </script> | ||||
|         `); | ||||
|  | ||||
|         head.append(script); | ||||
|  | ||||
|         // manifest.json | ||||
|         $("link[rel=manifest]").attr("href", `/api/status-page/${statusPage.slug}/manifest.json`); | ||||
|  | ||||
| @@ -83,6 +100,8 @@ class StatusPage extends BeanModel { | ||||
|             incident = incident.toPublicJSON(); | ||||
|         } | ||||
|  | ||||
|         let maintenanceList = await StatusPage.getMaintenanceList(statusPage.id); | ||||
|  | ||||
|         // Public Group List | ||||
|         const publicGroupList = []; | ||||
|         const showTags = !!statusPage.show_tags; | ||||
| @@ -100,7 +119,8 @@ class StatusPage extends BeanModel { | ||||
|         return { | ||||
|             config: await statusPage.toPublicJSON(), | ||||
|             incident, | ||||
|             publicGroupList | ||||
|             publicGroupList, | ||||
|             maintenanceList, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
| @@ -214,6 +234,7 @@ class StatusPage extends BeanModel { | ||||
|             customCSS: this.custom_css, | ||||
|             footerText: this.footer_text, | ||||
|             showPoweredBy: !!this.show_powered_by, | ||||
|             googleAnalyticsId: this.google_analytics_tag_id, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
| @@ -234,6 +255,7 @@ class StatusPage extends BeanModel { | ||||
|             customCSS: this.custom_css, | ||||
|             footerText: this.footer_text, | ||||
|             showPoweredBy: !!this.show_powered_by, | ||||
|             googleAnalyticsId: this.google_analytics_tag_id, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
| @@ -259,6 +281,38 @@ class StatusPage extends BeanModel { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get list of maintenances | ||||
|      * @param {number} statusPageId ID of status page to get maintenance for | ||||
|      * @returns {Object} Object representing maintenances sanitized for public | ||||
|      */ | ||||
|     static async getMaintenanceList(statusPageId) { | ||||
|         try { | ||||
|             const publicMaintenanceList = []; | ||||
|  | ||||
|             let activeCondition = Maintenance.getActiveMaintenanceSQLCondition(); | ||||
|             let maintenanceBeanList = R.convertToBeans("maintenance", await R.getAll(` | ||||
|                 SELECT 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 ])); | ||||
|  | ||||
|             for (const bean of maintenanceBeanList) { | ||||
|                 publicMaintenanceList.push(await bean.toPublicJSON()); | ||||
|             } | ||||
|  | ||||
|             return publicMaintenanceList; | ||||
|  | ||||
|         } catch (error) { | ||||
|             return []; | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = StatusPage; | ||||
|   | ||||
							
								
								
									
										20
									
								
								server/modules/dayjs/plugin/timezone.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								server/modules/dayjs/plugin/timezone.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| import { PluginFunc, ConfigType } from 'dayjs' | ||||
|  | ||||
| declare const plugin: PluginFunc | ||||
| export = plugin | ||||
|  | ||||
| declare module 'dayjs' { | ||||
|   interface Dayjs { | ||||
|     tz(timezone?: string, keepLocalTime?: boolean): Dayjs | ||||
|     offsetName(type?: 'short' | 'long'): string | undefined | ||||
|   } | ||||
|  | ||||
|   interface DayjsTimezone { | ||||
|     (date: ConfigType, timezone?: string): Dayjs | ||||
|     (date: ConfigType, format: string, timezone?: string): Dayjs | ||||
|     guess(): string | ||||
|     setDefault(timezone?: string): void | ||||
|   } | ||||
|  | ||||
|   const tz: DayjsTimezone | ||||
| } | ||||
							
								
								
									
										115
									
								
								server/modules/dayjs/plugin/timezone.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										115
									
								
								server/modules/dayjs/plugin/timezone.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,115 @@ | ||||
| /** | ||||
|  * Copy from node_modules/dayjs/plugin/timezone.js | ||||
|  * Try to fix https://github.com/louislam/uptime-kuma/issues/2318 | ||||
|  * Source: https://github.com/iamkun/dayjs/tree/dev/src/plugin/utc | ||||
|  * License: MIT | ||||
|  */ | ||||
| !function (t, e) { | ||||
|     // eslint-disable-next-line no-undef | ||||
|     typeof exports == "object" && typeof module != "undefined" ? module.exports = e() : typeof define == "function" && define.amd ? define(e) : (t = typeof globalThis != "undefined" ? globalThis : t || self).dayjs_plugin_timezone = e(); | ||||
| }(this, (function () { | ||||
|     "use strict"; | ||||
|     let t = { | ||||
|         year: 0, | ||||
|         month: 1, | ||||
|         day: 2, | ||||
|         hour: 3, | ||||
|         minute: 4, | ||||
|         second: 5 | ||||
|     }; | ||||
|     let e = {}; | ||||
|     return function (n, i, o) { | ||||
|         let r; | ||||
|         let a = function (t, n, i) { | ||||
|             void 0 === i && (i = {}); | ||||
|             let o = new Date(t); | ||||
|             let r = function (t, n) { | ||||
|                 void 0 === n && (n = {}); | ||||
|                 let i = n.timeZoneName || "short"; | ||||
|                 let o = t + "|" + i; | ||||
|                 let r = e[o]; | ||||
|                 return r || (r = new Intl.DateTimeFormat("en-US", { | ||||
|                     hour12: !1, | ||||
|                     timeZone: t, | ||||
|                     year: "numeric", | ||||
|                     month: "2-digit", | ||||
|                     day: "2-digit", | ||||
|                     hour: "2-digit", | ||||
|                     minute: "2-digit", | ||||
|                     second: "2-digit", | ||||
|                     timeZoneName: i | ||||
|                 }), e[o] = r), r; | ||||
|             }(n, i); | ||||
|             return r.formatToParts(o); | ||||
|         }; | ||||
|         let u = function (e, n) { | ||||
|             let i = a(e, n); | ||||
|             let r = []; | ||||
|             let u = 0; | ||||
|             for (; u < i.length; u += 1) { | ||||
|                 let f = i[u]; | ||||
|                 let s = f.type; | ||||
|                 let m = f.value; | ||||
|                 let c = t[s]; | ||||
|                 c >= 0 && (r[c] = parseInt(m, 10)); | ||||
|             } | ||||
|             let d = r[3]; | ||||
|             let l = d === 24 ? 0 : d; | ||||
|             let v = r[0] + "-" + r[1] + "-" + r[2] + " " + l + ":" + r[4] + ":" + r[5] + ":000"; | ||||
|             let h = +e; | ||||
|             return (o.utc(v).valueOf() - (h -= h % 1e3)) / 6e4; | ||||
|         }; | ||||
|         let f = i.prototype; | ||||
|         f.tz = function (t, e) { | ||||
|             void 0 === t && (t = r); | ||||
|             let n = this.utcOffset(); | ||||
|             let i = this.toDate(); | ||||
|             let a = i.toLocaleString("en-US", { timeZone: t }).replace("\u202f", " "); | ||||
|             let u = Math.round((i - new Date(a)) / 1e3 / 60); | ||||
|             let f = o(a).$set("millisecond", this.$ms).utcOffset(15 * -Math.round(i.getTimezoneOffset() / 15) - u, !0); | ||||
|             if (e) { | ||||
|                 let s = f.utcOffset(); | ||||
|                 f = f.add(n - s, "minute"); | ||||
|             } | ||||
|             return f.$x.$timezone = t, f; | ||||
|         }, f.offsetName = function (t) { | ||||
|             let e = this.$x.$timezone || o.tz.guess(); | ||||
|             let n = a(this.valueOf(), e, { timeZoneName: t }).find((function (t) { | ||||
|                 return t.type.toLowerCase() === "timezonename"; | ||||
|             })); | ||||
|             return n && n.value; | ||||
|         }; | ||||
|         let s = f.startOf; | ||||
|         f.startOf = function (t, e) { | ||||
|             if (!this.$x || !this.$x.$timezone) { | ||||
|                 return s.call(this, t, e); | ||||
|             } | ||||
|             let n = o(this.format("YYYY-MM-DD HH:mm:ss:SSS")); | ||||
|             return s.call(n, t, e).tz(this.$x.$timezone, !0); | ||||
|         }, o.tz = function (t, e, n) { | ||||
|             let i = n && e; | ||||
|             let a = n || e || r; | ||||
|             let f = u(+o(), a); | ||||
|             if (typeof t != "string") { | ||||
|                 return o(t).tz(a); | ||||
|             } | ||||
|             let s = function (t, e, n) { | ||||
|                 let i = t - 60 * e * 1e3; | ||||
|                 let o = u(i, n); | ||||
|                 if (e === o) { | ||||
|                     return [ i, e ]; | ||||
|                 } | ||||
|                 let r = u(i -= 60 * (o - e) * 1e3, n); | ||||
|                 return o === r ? [ i, o ] : [ t - 60 * Math.min(o, r) * 1e3, Math.max(o, r) ]; | ||||
|             }(o.utc(t, i).valueOf(), f, a); | ||||
|             let m = s[0]; | ||||
|             let c = s[1]; | ||||
|             let d = o(m).utcOffset(c); | ||||
|             return d.$x.$timezone = a, d; | ||||
|         }, o.tz.guess = function () { | ||||
|             return Intl.DateTimeFormat().resolvedOptions().timeZone; | ||||
|         }, o.tz.setDefault = function (t) { | ||||
|             r = t; | ||||
|         }; | ||||
|     }; | ||||
| })); | ||||
							
								
								
									
										19
									
								
								server/monitor-types/monitor-type.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								server/monitor-types/monitor-type.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| class MonitorType { | ||||
|  | ||||
|     name = undefined; | ||||
|  | ||||
|     /** | ||||
|      * | ||||
|      * @param {Monitor} monitor | ||||
|      * @param {Heartbeat} heartbeat | ||||
|      * @returns {Promise<void>} | ||||
|      */ | ||||
|     async check(monitor, heartbeat) { | ||||
|         throw new Error("You need to override check()"); | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| module.exports = { | ||||
|     MonitorType, | ||||
| }; | ||||
| @@ -12,9 +12,7 @@ const { default: axios } = require("axios"); | ||||
|  | ||||
| // bark is an APN bridge that sends notifications to Apple devices. | ||||
|  | ||||
| const barkNotificationGroup = "UptimeKuma"; | ||||
| const barkNotificationAvatar = "https://github.com/louislam/uptime-kuma/raw/master/public/icon.png"; | ||||
| const barkNotificationSound = "telegraph"; | ||||
| const successMessage = "Successes!"; | ||||
|  | ||||
| class Bark extends NotificationProvider { | ||||
| @@ -30,17 +28,17 @@ class Bark extends NotificationProvider { | ||||
|  | ||||
|         if (msg != null && heartbeatJSON != null && heartbeatJSON["status"] === UP) { | ||||
|             let title = "UptimeKuma Monitor Up"; | ||||
|             return await this.postNotification(title, msg, barkEndpoint); | ||||
|             return await this.postNotification(notification, title, msg, barkEndpoint); | ||||
|         } | ||||
|  | ||||
|         if (msg != null && heartbeatJSON != null && heartbeatJSON["status"] === DOWN) { | ||||
|             let title = "UptimeKuma Monitor Down"; | ||||
|             return await this.postNotification(title, msg, barkEndpoint); | ||||
|             return await this.postNotification(notification, title, msg, barkEndpoint); | ||||
|         } | ||||
|  | ||||
|         if (msg != null) { | ||||
|             let title = "UptimeKuma Message"; | ||||
|             return await this.postNotification(title, msg, barkEndpoint); | ||||
|             return await this.postNotification(notification, title, msg, barkEndpoint); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -50,13 +48,23 @@ class Bark extends NotificationProvider { | ||||
|      * @param {string} postUrl URL to append parameters to | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     appendAdditionalParameters(postUrl) { | ||||
|         // grouping all our notifications | ||||
|         postUrl += "?group=" + barkNotificationGroup; | ||||
|     appendAdditionalParameters(notification, postUrl) { | ||||
|         // set icon to uptime kuma icon, 11kb should be fine | ||||
|         postUrl += "&icon=" + barkNotificationAvatar; | ||||
|         postUrl += "?icon=" + barkNotificationAvatar; | ||||
|         // grouping all our notifications | ||||
|         if (notification.barkGroup != null) { | ||||
|             postUrl += "&group=" + notification.barkGroup; | ||||
|         } else { | ||||
|             // default name | ||||
|             postUrl += "&group=" + "UptimeKuma"; | ||||
|         } | ||||
|         // picked a sound, this should follow system's mute status when arrival | ||||
|         postUrl += "&sound=" + barkNotificationSound; | ||||
|         if (notification.barkSound != null) { | ||||
|             postUrl += "&sound=" + notification.barkSound; | ||||
|         } else { | ||||
|             // default sound | ||||
|             postUrl += "&sound=" + "telegraph"; | ||||
|         } | ||||
|         return postUrl; | ||||
|     } | ||||
|  | ||||
| @@ -81,12 +89,12 @@ class Bark extends NotificationProvider { | ||||
|      * @param {string} endpoint Endpoint to send request to | ||||
|      * @returns {string} | ||||
|      */ | ||||
|     async postNotification(title, subtitle, endpoint) { | ||||
|     async postNotification(notification, title, subtitle, endpoint) { | ||||
|         // url encode title and subtitle | ||||
|         title = encodeURIComponent(title); | ||||
|         subtitle = encodeURIComponent(subtitle); | ||||
|         let postUrl = endpoint + "/" + title + "/" + subtitle; | ||||
|         postUrl = this.appendAdditionalParameters(postUrl); | ||||
|         postUrl = this.appendAdditionalParameters(notification, postUrl); | ||||
|         let result = await axios.get(postUrl); | ||||
|         this.checkResult(result); | ||||
|         if (result.statusText != null) { | ||||
|   | ||||
| @@ -8,7 +8,6 @@ class ClickSendSMS extends NotificationProvider { | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             console.log({ notification }); | ||||
|             let config = { | ||||
|                 headers: { | ||||
|                     "Content-Type": "application/json", | ||||
|   | ||||
| @@ -64,7 +64,7 @@ class Discord extends NotificationProvider { | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: "Error", | ||||
|                                 value: heartbeatJSON["msg"], | ||||
|                                 value: heartbeatJSON["msg"] == null ? "N/A" : heartbeatJSON["msg"], | ||||
|                             }, | ||||
|                         ], | ||||
|                     }], | ||||
| @@ -91,7 +91,7 @@ class Discord extends NotificationProvider { | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: monitorJSON["type"] === "push" ? "Service Type" : "Service URL", | ||||
|                                 value: monitorJSON["type"] === "push" ? "Heartbeat" : address.startsWith("http") ? "[Visit Service](" + address + ")" : address, | ||||
|                                 value: monitorJSON["type"] === "push" ? "Heartbeat" : address, | ||||
|                             }, | ||||
|                             { | ||||
|                                 name: "Time (UTC)", | ||||
|   | ||||
							
								
								
									
										24
									
								
								server/notification-providers/freemobile.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								server/notification-providers/freemobile.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class FreeMobile extends NotificationProvider { | ||||
|  | ||||
|     name = "FreeMobile"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             await axios.post(`https://smsapi.free-mobile.fr/sendmsg?msg=${encodeURIComponent(msg.replace("🔴", "⛔️"))}`, { | ||||
|                 "user": notification.freemobileUser, | ||||
|                 "pass": notification.freemobilePass, | ||||
|             }); | ||||
|  | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = FreeMobile; | ||||
							
								
								
									
										35
									
								
								server/notification-providers/goalert.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								server/notification-providers/goalert.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
| const { UP } = require("../../src/util"); | ||||
|  | ||||
| class GoAlert extends NotificationProvider { | ||||
|  | ||||
|     name = "GoAlert"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         try { | ||||
|             let closeAction = "close"; | ||||
|             let data = { | ||||
|                 summary: msg, | ||||
|             }; | ||||
|             if (heartbeatJSON != null && heartbeatJSON["status"] === UP) { | ||||
|                 data["action"] = closeAction; | ||||
|             } | ||||
|             let headers = { | ||||
|                 "Content-Type": "multipart/form-data", | ||||
|             }; | ||||
|             let config = { | ||||
|                 headers: headers | ||||
|             }; | ||||
|             await axios.post(`${notification.goAlertBaseURL}/api/v2/generic/incoming?token=${notification.goAlertToken}`, data, config); | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             let msg = (error.response.data) ? error.response.data : "Error without response"; | ||||
|             throw new Error(msg); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = GoAlert; | ||||
							
								
								
									
										38
									
								
								server/notification-providers/home-assistant.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								server/notification-providers/home-assistant.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| const defaultNotificationService = "notify"; | ||||
|  | ||||
| class HomeAssistant extends NotificationProvider { | ||||
|     name = "HomeAssistant"; | ||||
|  | ||||
|     async send(notification, message, monitor = null, heartbeat = null) { | ||||
|         const notificationService = notification?.notificationService || defaultNotificationService; | ||||
|  | ||||
|         try { | ||||
|             await axios.post( | ||||
|                 `${notification.homeAssistantUrl}/api/services/notify/${notificationService}`, | ||||
|                 { | ||||
|                     title: "Uptime Kuma", | ||||
|                     message, | ||||
|                     ...(notificationService !== "persistent_notification" && { data: { | ||||
|                         name: monitor?.name, | ||||
|                         status: heartbeat?.status, | ||||
|                     } }), | ||||
|                 }, | ||||
|                 { | ||||
|                     headers: { | ||||
|                         Authorization: `Bearer ${notification.longLivedAccessToken}`, | ||||
|                         "Content-Type": "application/json", | ||||
|                     }, | ||||
|                 } | ||||
|             ); | ||||
|  | ||||
|             return "Sent Successfully."; | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = HomeAssistant; | ||||
							
								
								
									
										31
									
								
								server/notification-providers/kook.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								server/notification-providers/kook.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| const NotificationProvider = require("./notification-provider"); | ||||
| const axios = require("axios"); | ||||
|  | ||||
| class Kook extends NotificationProvider { | ||||
|  | ||||
|     name = "Kook"; | ||||
|  | ||||
|     async send(notification, msg, monitorJSON = null, heartbeatJSON = null) { | ||||
|         let okMsg = "Sent Successfully."; | ||||
|         let url = "https://www.kookapp.cn/api/v3/message/create"; | ||||
|         let data = { | ||||
|             target_id: notification.kookGuildID, | ||||
|             content: msg, | ||||
|         }; | ||||
|         let config = { | ||||
|             headers: { | ||||
|                 "Authorization": "Bot " + notification.kookBotToken, | ||||
|                 "Content-Type": "application/json", | ||||
|             }, | ||||
|         }; | ||||
|         try { | ||||
|             await axios.post(url, data, config); | ||||
|             return okMsg; | ||||
|  | ||||
|         } catch (error) { | ||||
|             this.throwGeneralAxiosError(error); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| module.exports = Kook; | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user