Merge remote-tracking branch 'origin/master' into feature/482-add-description-to-monitor
# Conflicts: # server/database.js # server/model/monitor.js # src/icon.js # src/languages/en.js # src/languages/es-ES.js
This commit is contained in:
		
						commit
						7f9332c753
					
				
					 393 changed files with 56675 additions and 19206 deletions
				
			
		|  | @ -1,6 +1,7 @@ | ||||||
| /.idea | /.idea | ||||||
| /node_modules | /node_modules | ||||||
| /data | /data | ||||||
|  | /cypress | ||||||
| /out | /out | ||||||
| /test | /test | ||||||
| /kubernetes | /kubernetes | ||||||
|  | @ -28,6 +29,11 @@ SECURITY.md | ||||||
| tsconfig.json | tsconfig.json | ||||||
| .env | .env | ||||||
| /tmp | /tmp | ||||||
|  | /babel.config.js | ||||||
|  | /ecosystem.config.js | ||||||
|  | /extra/healthcheck.exe | ||||||
|  | /extra/healthcheck | ||||||
|  | extra/exe-builder | ||||||
| 
 | 
 | ||||||
| ### .gitignore content (commented rules are duplicated) | ### .gitignore content (commented rules are duplicated) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -19,3 +19,6 @@ indent_size = 2 | ||||||
| 
 | 
 | ||||||
| [*.vue] | [*.vue] | ||||||
| trim_trailing_whitespace = false | trim_trailing_whitespace = false | ||||||
|  | 
 | ||||||
|  | [*.go] | ||||||
|  | indent_style = tab | ||||||
|  |  | ||||||
							
								
								
									
										28
									
								
								.eslintrc.js
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								.eslintrc.js
									
									
									
									
									
								
							|  | @ -1,4 +1,9 @@ | ||||||
| module.exports = { | module.exports = { | ||||||
|  |     ignorePatterns: [ | ||||||
|  |         "test/*", | ||||||
|  |         "server/modules/apicache/*", | ||||||
|  |         "src/util.js" | ||||||
|  |     ], | ||||||
|     root: true, |     root: true, | ||||||
|     env: { |     env: { | ||||||
|         browser: true, |         browser: true, | ||||||
|  | @ -17,14 +22,16 @@ module.exports = { | ||||||
|         requireConfigFile: false, |         requireConfigFile: false, | ||||||
|     }, |     }, | ||||||
|     rules: { |     rules: { | ||||||
|  |         "yoda": "error", | ||||||
|  |         eqeqeq: [ "warn", "smart" ], | ||||||
|         "linebreak-style": [ "error", "unix" ], |         "linebreak-style": [ "error", "unix" ], | ||||||
|         "camelcase": [ "warn", { |         "camelcase": [ "warn", { | ||||||
|             "properties": "never", |             "properties": "never", | ||||||
|             "ignoreImports": true |             "ignoreImports": true | ||||||
|         }], |         }], | ||||||
|         // override/add rules settings here, such as:
 |         "no-unused-vars": [ "warn", { | ||||||
|         // 'vue/no-unused-vars': 'error'
 |             "args": "none" | ||||||
|         "no-unused-vars": "warn", |         }], | ||||||
|         indent: [ |         indent: [ | ||||||
|             "error", |             "error", | ||||||
|             4, |             4, | ||||||
|  | @ -33,16 +40,23 @@ module.exports = { | ||||||
|                 SwitchCase: 1, |                 SwitchCase: 1, | ||||||
|             }, |             }, | ||||||
|         ], |         ], | ||||||
|         quotes: ["warn", "double"], |         quotes: [ "error", "double" ], | ||||||
|         semi: "warn", |         semi: "error", | ||||||
|         "vue/html-indent": ["warn", 4], // default: 2
 |         "vue/html-indent": [ "error", 4 ], // default: 2
 | ||||||
|         "vue/max-attributes-per-line": "off", |         "vue/max-attributes-per-line": "off", | ||||||
|         "vue/singleline-html-element-content-newline": "off", |         "vue/singleline-html-element-content-newline": "off", | ||||||
|         "vue/html-self-closing": "off", |         "vue/html-self-closing": "off", | ||||||
|  |         "vue/require-component-is": "off",      // not allow is="style" https://github.com/vuejs/eslint-plugin-vue/issues/462#issuecomment-430234675
 | ||||||
|         "vue/attribute-hyphenation": "off",     // This change noNL to "no-n-l" unexpectedly
 |         "vue/attribute-hyphenation": "off",     // This change noNL to "no-n-l" unexpectedly
 | ||||||
|  |         "vue/multi-word-component-names": "off", | ||||||
|         "no-multi-spaces": [ "error", { |         "no-multi-spaces": [ "error", { | ||||||
|             ignoreEOLComments: true, |             ignoreEOLComments: true, | ||||||
|         }], |         }], | ||||||
|  |         "array-bracket-spacing": [ "warn", "always", { | ||||||
|  |             "singleValue": true, | ||||||
|  |             "objectsInArrays": false, | ||||||
|  |             "arraysInArrays": false | ||||||
|  |         }], | ||||||
|         "space-before-function-paren": [ "error", { |         "space-before-function-paren": [ "error", { | ||||||
|             "anonymous": "always", |             "anonymous": "always", | ||||||
|             "named": "never", |             "named": "never", | ||||||
|  | @ -59,7 +73,7 @@ module.exports = { | ||||||
|         "keyword-spacing": "warn", |         "keyword-spacing": "warn", | ||||||
|         "space-infix-ops": "warn", |         "space-infix-ops": "warn", | ||||||
|         "arrow-spacing": "warn", |         "arrow-spacing": "warn", | ||||||
|         "no-trailing-spaces": "warn", |         "no-trailing-spaces": "error", | ||||||
|         "no-constant-condition": [ "error", { |         "no-constant-condition": [ "error", { | ||||||
|             "checkLoops": false, |             "checkLoops": false, | ||||||
|         }], |         }], | ||||||
|  |  | ||||||
							
								
								
									
										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: | ||||||
|  | 
 | ||||||
							
								
								
									
										14
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							|  | @ -1,16 +1,21 @@ | ||||||
|  | ⚠️⚠️⚠️ 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 | # Description | ||||||
| 
 | 
 | ||||||
| Fixes #(issue) | Fixes #(issue) | ||||||
| 
 | 
 | ||||||
| ## Type of change | ## Type of change | ||||||
| 
 | 
 | ||||||
| Please delete options that are not relevant. | Please delete any options that are not relevant. | ||||||
| 
 | 
 | ||||||
| - Bug fix (non-breaking change which fixes an issue) | - Bug fix (non-breaking change which fixes an issue) | ||||||
| - User Interface | - User interface (UI) | ||||||
| - New feature (non-breaking change which adds functionality) | - New feature (non-breaking change which adds functionality) | ||||||
| - Breaking change (fix or feature that would cause existing functionality to not work as expected) | - Breaking change (fix or feature that would cause existing functionality to not work as expected) | ||||||
| - Translation update |  | ||||||
| - Other | - Other | ||||||
| - This change requires a documentation update | - This change requires a documentation update | ||||||
| 
 | 
 | ||||||
|  | @ -18,8 +23,9 @@ Please delete options that are not relevant. | ||||||
| 
 | 
 | ||||||
| - [ ] My code follows the style guidelines of this project | - [ ] My code follows the style guidelines of this project | ||||||
| - [ ] I ran ESLint and other linters for modified files | - [ ] I ran ESLint and other linters for modified files | ||||||
| - [ ] I have performed a self-review of my own code and test it | - [ ] I have performed a self-review of my own code and tested it | ||||||
| - [ ] I have commented my code, particularly in hard-to-understand areas | - [ ] I have commented my code, particularly in hard-to-understand areas | ||||||
|  |   (including JSDoc for methods) | ||||||
| - [ ] My changes generate no new warnings | - [ ] My changes generate no new warnings | ||||||
| - [ ] My code needed automated testing. I have added them (this is optional task) | - [ ] My code needed automated testing. I have added them (this is optional task) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										66
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										66
									
								
								.github/workflows/auto-test.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -6,30 +6,84 @@ name: Auto Test | ||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: [ master ] |     branches: [ master ] | ||||||
|  |     paths-ignore: | ||||||
|  |       - '*.md' | ||||||
|   pull_request: |   pull_request: | ||||||
|     branches: [ master ] |     branches: [ master ] | ||||||
|  |     paths-ignore: | ||||||
|  |       - '*.md' | ||||||
| 
 | 
 | ||||||
| jobs: | jobs: | ||||||
|   auto-test: |   auto-test: | ||||||
|  |     needs: [ check-linters ] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|  |     timeout-minutes: 15 | ||||||
| 
 | 
 | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [macos-latest, ubuntu-latest, windows-latest] |         os: [macos-latest, ubuntu-latest, windows-latest] | ||||||
|         node-version: [14.x, 16.x, 17.x] |         node: [ 14, 16, 18, 19 ] | ||||||
|         # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ |         # See supported Node.js release schedule at https://nodejs.org/en/about/releases/ | ||||||
| 
 | 
 | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v2 |     - run: git config --global core.autocrlf false  # Mainly for Windows | ||||||
|  |     - uses: actions/checkout@v3 | ||||||
| 
 | 
 | ||||||
|     - name: Use Node.js ${{ matrix.node-version }} |     - name: Use Node.js ${{ matrix.node }} | ||||||
|       uses: actions/setup-node@v2 |       uses: actions/setup-node@v3 | ||||||
|       with: |       with: | ||||||
|         node-version: ${{ matrix.node-version }} |         node-version: ${{ matrix.node }} | ||||||
|         cache: 'npm' |         cache: 'npm' | ||||||
|     - run: npm run install-legacy |     - run: npm install | ||||||
|     - run: npm run build |     - run: npm run build | ||||||
|     - run: npm test |     - run: npm test | ||||||
|       env: |       env: | ||||||
|         HEADLESS_TEST: 1 |         HEADLESS_TEST: 1 | ||||||
|         JUST_FOR_TEST: ${{ secrets.JUST_FOR_TEST }} |         JUST_FOR_TEST: ${{ secrets.JUST_FOR_TEST }} | ||||||
|  | 
 | ||||||
|  |   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 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 | name: Close Incorrect Issue | ||||||
| 
 | 
 | ||||||
| on: | on: | ||||||
|  | @ -12,13 +11,13 @@ jobs: | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest] |         os: [ubuntu-latest] | ||||||
|         node-version: [16.x] |         node-version: [16] | ||||||
| 
 | 
 | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v2 |     - uses: actions/checkout@v3 | ||||||
| 
 | 
 | ||||||
|     - name: Use Node.js ${{ matrix.node-version }} |     - name: Use Node.js ${{ matrix.node-version }} | ||||||
|       uses: actions/setup-node@v2 |       uses: actions/setup-node@v3 | ||||||
|       with: |       with: | ||||||
|         node-version: ${{ matrix.node-version }} |         node-version: ${{ matrix.node-version }} | ||||||
|         cache: 'npm' |         cache: 'npm' | ||||||
|  |  | ||||||
							
								
								
									
										24
									
								
								.github/workflows/stale-bot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								.github/workflows/stale-bot.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -1,22 +1,22 @@ | ||||||
| name: 'Automatically close stale issues and PRs' | name: 'Automatically close stale issues and PRs' | ||||||
| on: | on: | ||||||
|  |   workflow_dispatch: | ||||||
|   schedule: |   schedule: | ||||||
|     - cron: '0 0 * * *' |     - cron: '0 */6 * * *' | ||||||
| #Run once a day at midnight  | #Run every 6 hours | ||||||
| 
 | 
 | ||||||
| jobs: | jobs: | ||||||
|   stale: |   stale: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/stale@v4 |       - uses: actions/stale@v7 | ||||||
|         with: |         with: | ||||||
|           stale-issue-message: 'We are clearing up our old issues and your ticket has been open for 6 months with no activity. Remove stale label or comment or this will be closed in 7 days.' |           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.' | ||||||
|           stale-pr-message: 'We are clearing up our old Pull Requests and yours has been open for 6 months with no activity. Remove stale label or comment or this will be closed in 7 days.' |           close-issue-message: 'This issue was closed because it has been stalled for 2 days with no activity.' | ||||||
|           close-issue-message: 'This issue was closed because it has been stalled for 7 days with no activity.' |           days-before-stale: 90 | ||||||
|           close-pr-message: 'This PR was closed because it has been stalled for 7 days with no activity.' |           days-before-close: 2 | ||||||
|           days-before-stale: 180 |           days-before-pr-stale: 999999999 | ||||||
|           days-before-close: 7 |           days-before-pr-close: 1 | ||||||
|           exempt-issue-labels: 'News,Medium,High,discussion,bug,doc,' |           exempt-issue-labels: 'News,Medium,High,discussion,bug,doc,feature-request' | ||||||
|           exempt-pr-labels: 'awaiting-approval,work-in-progress,enhancement,' |  | ||||||
|           exempt-issue-assignees: 'louislam' |           exempt-issue-assignees: 'louislam' | ||||||
|           exempt-pr-assignees: 'louislam' |           operations-per-run: 200 | ||||||
|  |  | ||||||
							
								
								
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							|  | @ -13,3 +13,13 @@ dist-ssr | ||||||
| /out | /out | ||||||
| /tmp | /tmp | ||||||
| .env | .env | ||||||
|  | 
 | ||||||
|  | cypress/videos | ||||||
|  | cypress/screenshots | ||||||
|  | 
 | ||||||
|  | /extra/healthcheck.exe | ||||||
|  | /extra/healthcheck | ||||||
|  | /extra/healthcheck-armv7 | ||||||
|  | 
 | ||||||
|  | extra/exe-builder/bin | ||||||
|  | extra/exe-builder/obj | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								.npmrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.npmrc
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | legacy-peer-deps=true | ||||||
|  | @ -1,9 +1,15 @@ | ||||||
| { | { | ||||||
|     "extends": "stylelint-config-standard", |     "extends": "stylelint-config-standard", | ||||||
|  |     "customSyntax": "postcss-html", | ||||||
|     "rules": { |     "rules": { | ||||||
|         "indentation": 4, |         "indentation": 4, | ||||||
|         "no-descending-specificity": null, |         "no-descending-specificity": null, | ||||||
|         "selector-list-comma-newline-after": null, |         "selector-list-comma-newline-after": null, | ||||||
|         "declaration-empty-line-before": null |         "declaration-empty-line-before": null, | ||||||
|  |         "alpha-value-notation": "number", | ||||||
|  |         "color-function-notation": "legacy", | ||||||
|  |         "shorthand-property-no-redundant-values": null, | ||||||
|  |         "color-hex-length": null, | ||||||
|  |         "declaration-block-no-redundant-longhand-properties": null | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										218
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							
							
						
						
									
										218
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							|  | @ -1,6 +1,6 @@ | ||||||
| # Project Info | # 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. | 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 | ## Directories | ||||||
| 
 | 
 | ||||||
|  | - config (dev config files) | ||||||
| - data (App data) | - data (App data) | ||||||
|  | - db (Base database and migration scripts) | ||||||
| - dist (Frontend build) | - dist (Frontend build) | ||||||
|  | - docker (Dockerfiles) | ||||||
| - extra (Extra useful scripts) | - extra (Extra useful scripts) | ||||||
| - public (Frontend resources for dev only) | - public (Frontend resources for dev only) | ||||||
| - server (Server source code) | - server (Server source code) | ||||||
|  | @ -27,12 +30,46 @@ The frontend code build into "dist" directory. The server (express.js) exposes t | ||||||
| 
 | 
 | ||||||
| ## Can I create a pull request for Uptime Kuma? | ## Can I create a pull request for Uptime Kuma? | ||||||
| 
 | 
 | ||||||
| Generally, if the pull request is working fine, and it does not affect any existing logic, workflow and performance, I will merge into the master branch once it is tested. | 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. | ||||||
|  | 
 | ||||||
|  | Here are some references: | ||||||
|  | 
 | ||||||
|  | ✅ 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. | ||||||
| 
 | 
 | ||||||
| If you are not sure whether I will accept your pull request, feel free to create an empty pull request draft first. |  | ||||||
| 
 | 
 | ||||||
| ### Recommended Pull Request Guideline | ### Recommended Pull Request Guideline | ||||||
| 
 | 
 | ||||||
|  | Before deep into coding, discussion first is preferred. Creating an empty pull request for discussion would be recommended. | ||||||
|  | 
 | ||||||
| 1. Fork the project | 1. Fork the project | ||||||
| 1. Clone your fork repo to local | 1. Clone your fork repo to local | ||||||
| 1. Create a new branch | 1. Create a new branch | ||||||
|  | @ -42,118 +79,93 @@ If you are not sure whether I will accept your pull request, feel free to create | ||||||
| 1. Create a pull request: https://github.com/louislam/uptime-kuma/compare | 1. Create a pull request: https://github.com/louislam/uptime-kuma/compare | ||||||
| 1. Write a proper description | 1. Write a proper description | ||||||
| 1. Click "Change to draft" | 1. Click "Change to draft" | ||||||
| 
 | 1. Discussion | ||||||
| ### Pull Request Examples |  | ||||||
| 
 |  | ||||||
| Here are some example situations in the past. |  | ||||||
| 
 |  | ||||||
| #### ✅ High - Medium Priority |  | ||||||
| 
 |  | ||||||
| Easy to review, no breaking change and not touching the existing code |  | ||||||
| 
 |  | ||||||
| - Add a new notification |  | ||||||
| - Add a chart |  | ||||||
| - Fix a bug |  | ||||||
| - Translations |  | ||||||
| - Add a independent new feature |  | ||||||
| 
 |  | ||||||
| #### *️⃣ Requires one more reviewer |  | ||||||
| 
 |  | ||||||
| I do not have such knowledge to test it. |  | ||||||
| 
 |  | ||||||
| - Add k8s supports |  | ||||||
| 
 |  | ||||||
| #### ⚠ Low Priority - Harsh Mode |  | ||||||
| 
 |  | ||||||
| Some pull requests are required to modify the core. To be honest, I do not want anyone to try to do that, because it would spend a lot of your time. I will review your pull request harshly. Also, you may need to write a lot of unit tests to ensure that there is no breaking change. |  | ||||||
| 
 |  | ||||||
| - Touch large parts of code of any very important features |  | ||||||
| - Touch monitoring logic |  | ||||||
| - Drop a table or drop a column for any reason |  | ||||||
| - Touch the entry point of Docker or Node.js |  | ||||||
| - Modify auth |  | ||||||
| 
 |  | ||||||
| #### *️⃣ Low Priority |  | ||||||
| 
 |  | ||||||
| It changed my current workflow and require further studies. |  | ||||||
| 
 |  | ||||||
| - Change my release approach |  | ||||||
| 
 |  | ||||||
| #### ❌ Won't Merge |  | ||||||
| 
 |  | ||||||
| - Any breaking changes |  | ||||||
| - Duplicated pull request |  | ||||||
| - Buggy |  | ||||||
| - Existing logic is completely modified or deleted |  | ||||||
| - A function that is completely out of scope |  | ||||||
| 
 | 
 | ||||||
| ## Project Styles | ## 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 | - 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. Env var is not encouraged. | - 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 | - Easy to use | ||||||
|  | - The web UI styling should be consistent and nice | ||||||
| 
 | 
 | ||||||
| ## Coding Styles | ## Coding Styles | ||||||
| 
 | 
 | ||||||
| - 4 spaces indentation | - 4 spaces indentation | ||||||
| - Follow `.editorconfig` | - Follow `.editorconfig` | ||||||
| - Follow ESLint | - Follow ESLint | ||||||
|  | - Methods and functions should be documented with JSDoc | ||||||
| 
 | 
 | ||||||
| ## Name convention | ## Name Conventions | ||||||
| 
 | 
 | ||||||
| - Javascript/Typescript: camelCaseType | - Javascript/Typescript: camelCaseType | ||||||
| - SQLite: underscore_type | - SQLite: snake_case (Underscore) | ||||||
| - CSS/SCSS: dash-type | - CSS/SCSS: kebab-case (Dash) | ||||||
| 
 | 
 | ||||||
| ## Tools | ## Tools | ||||||
| 
 | 
 | ||||||
| - Node.js >= 14 | - Node.js >= 14 | ||||||
|  | - NPM >= 8.5 | ||||||
| - Git | - Git | ||||||
| - IDE that supports ESLint and EditorConfig (I am using IntelliJ IDEA) | - IDE that supports ESLint and EditorConfig (I am using IntelliJ IDEA) | ||||||
| - A SQLite tool (SQLite Expert Personal is suggested) | - A SQLite GUI tool (SQLite Expert Personal is suggested) | ||||||
| 
 | 
 | ||||||
| ## Install dependencies | ## Install Dependencies for Development | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| npm ci | npm ci | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ## How to start the Backend Dev Server | ## Dev Server | ||||||
| 
 | 
 | ||||||
| (2021-09-23 Update) | (2022-04-26 Update) | ||||||
| 
 | 
 | ||||||
| ```bash | We can start the frontend dev server and the backend dev server in one command. | ||||||
| npm run start-server-dev |  | ||||||
| ``` |  | ||||||
| 
 | 
 | ||||||
| It binds to `0.0.0.0:3001` by default. | Port `3000` and port `3001` will be used. | ||||||
| 
 |  | ||||||
| ### Backend Details |  | ||||||
| 
 |  | ||||||
| It is mainly a socket.io app + express.js. |  | ||||||
| 
 |  | ||||||
| express.js is just used for serving the frontend built files (index.html, .js and .css etc.) |  | ||||||
| 
 |  | ||||||
| - model/ (Object model, auto mapping to the database table name) |  | ||||||
| - modules/ (Modified 3rd-party modules) |  | ||||||
| - notification-providers/ (individual notification logic) |  | ||||||
| - routers/ (Express Routers) |  | ||||||
| - socket-handler (Socket.io Handlers) |  | ||||||
| - server.js (Server main logic) |  | ||||||
| 
 |  | ||||||
| ## How to start the Frontend Dev Server |  | ||||||
| 
 |  | ||||||
| 1. Set the env var `NODE_ENV` to "development". |  | ||||||
| 2. Start the frontend dev server by the following command. |  | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| npm run dev | npm run dev | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
|    It binds to `0.0.0.0:3000` by default. | 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. | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | It is mainly a socket.io app + express.js. | ||||||
|  | 
 | ||||||
|  | express.js is used for:  | ||||||
|  | - entry point such as redirecting to a status page or the dashboard | ||||||
|  | - serving the frontend built files (index.html, .js and .css etc.) | ||||||
|  | - serving internal APIs of status page | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ### 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) | ||||||
|  | - uptime-kuma-server.js (UptimeKumaServer class, main logic should be here, but some still in `server.js`) | ||||||
|  | 
 | ||||||
|  | ## Frontend Dev Server | ||||||
|  | 
 | ||||||
|  | It binds to `0.0.0.0:3000` by default. Frontend dev server is used for development only.  | ||||||
|  | 
 | ||||||
|  | For production, it is not used. It will be compiled to `dist` directory instead.  | ||||||
| 
 | 
 | ||||||
| You can use Vue.js devtools Chrome extension for debugging. | You can use Vue.js devtools Chrome extension for debugging. | ||||||
| 
 | 
 | ||||||
|  | @ -180,29 +192,30 @@ The data and socket logic are in `src/mixins/socket.js`. | ||||||
| 
 | 
 | ||||||
| ## Unit Test | ## Unit Test | ||||||
| 
 | 
 | ||||||
| It is an end-to-end testing. It is using Jest and Puppeteer. |  | ||||||
| 
 |  | ||||||
| ```bash | ```bash | ||||||
| npm run build | npm run build | ||||||
| npm test | 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` | - Frontend dependencies = "devDependencies" | ||||||
| https://github.com/raineorshine/npm-check-updates |   - Examples: vue, chart.js | ||||||
|  | - Backend dependencies = "dependencies" | ||||||
|  |   - Examples: socket.io, sqlite3 | ||||||
|  | - Development dependencies = "devDependencies" | ||||||
|  |   - Examples: eslint, sass | ||||||
| 
 | 
 | ||||||
| ```bash | ### Update Dependencies | ||||||
| ncu -u -t patch |  | ||||||
| npm install |  | ||||||
| ``` |  | ||||||
| 
 | 
 | ||||||
| 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. | 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/)) | 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 | ## Translations | ||||||
| 
 | 
 | ||||||
| Please read: https://github.com/louislam/uptime-kuma/tree/master/src/languages | Please read: https://github.com/louislam/uptime-kuma/tree/master/src/languages | ||||||
|  | @ -221,14 +234,14 @@ https://github.com/louislam/uptime-kuma/issues?q=sort%3Aupdated-desc | ||||||
| ### Release Procedures | ### Release Procedures | ||||||
| 
 | 
 | ||||||
| 1. Draft a release note | 1. Draft a release note | ||||||
| 1. Make sure the repo is cleared | 2. Make sure the repo is cleared | ||||||
| 1. `npm run update-version 1.X.X` | 3. If the healthcheck is updated, remember to re-compile it: `npm run build-docker-builder-go` | ||||||
| 1. `npm run build` | 3. `npm run release-final with env vars: `VERSION` and `GITHUB_TOKEN` | ||||||
| 1. `npm run build-docker` | 4. Wait until the `Press any key to continue` | ||||||
| 1. `git push` | 5. `git push` | ||||||
| 1. Publish the release note as 1.X.X  | 6. Publish the release note as 1.X.X  | ||||||
| 1. `npm run upload-artifacts` | 7. Press any key to continue | ||||||
| 1. SSH to demo site server and update to 1.X.X | 8. Deploy to the demo server: `npm run deploy-demo-server` | ||||||
| 
 | 
 | ||||||
| Checking: | Checking: | ||||||
| 
 | 
 | ||||||
|  | @ -236,6 +249,15 @@ Checking: | ||||||
| - Try the Docker image with tag 1.X.X (Clean install / amd64 / arm64 / armv7) | - Try the Docker image with tag 1.X.X (Clean install / amd64 / arm64 / armv7) | ||||||
| - Try clean installation with Node.js | - Try clean installation with Node.js | ||||||
| 
 | 
 | ||||||
|  | ### Release Beta Procedures | ||||||
|  | 
 | ||||||
|  | 1. Draft a release note, check "This is a pre-release" | ||||||
|  | 2. Make sure the repo is cleared | ||||||
|  | 3. `npm run release-beta` with env vars: `VERSION` and `GITHUB_TOKEN` | ||||||
|  | 4. Wait until the `Press any key to continue` | ||||||
|  | 5. Publish the release note as 1.X.X-beta.X | ||||||
|  | 6. Press any key to continue | ||||||
|  | 
 | ||||||
| ### Release Wiki | ### Release Wiki | ||||||
| 
 | 
 | ||||||
| #### Setup Repo | #### Setup Repo | ||||||
|  |  | ||||||
							
								
								
									
										96
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										96
									
								
								README.md
									
									
									
									
									
								
							|  | @ -1,53 +1,59 @@ | ||||||
| # Uptime Kuma | # 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> | <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%"> | <div align="center" width="100%"> | ||||||
|     <img src="./public/icon.svg" width="128" alt="" /> |     <img src="./public/icon.svg" width="128" alt="" /> | ||||||
| </div> | </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 | ## 🥔 Live Demo | ||||||
| 
 | 
 | ||||||
| Try it! | 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. | 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. | ||||||
| 
 |  | ||||||
| VPS is sponsored by Uptime Kuma sponsors on [Open Collective](https://opencollective.com/uptime-kuma)! Thank you so much! |  | ||||||
| 
 | 
 | ||||||
| ## ⭐ Features | ## ⭐ Features | ||||||
| 
 | 
 | ||||||
| * Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / Ping / DNS Record / Push / Steam Game Server. | * Monitoring uptime for HTTP(s) / TCP / HTTP(s) Keyword / Ping / DNS Record / Push / Steam Game Server / Docker Containers | ||||||
| * Fancy, Reactive, Fast UI/UX. | * Fancy, Reactive, Fast UI/UX | ||||||
| * Notifications via Telegram, Discord, Gotify, Slack, Pushover, Email (SMTP), and [70+ notification services, click here for the full list](https://github.com/louislam/uptime-kuma/tree/master/src/components/notifications). | * 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. | * 20 second intervals | ||||||
| * [Multi Languages](https://github.com/louislam/uptime-kuma/tree/master/src/languages) | * [Multi Languages](https://github.com/louislam/uptime-kuma/tree/master/src/lang) | ||||||
| * Simple Status Page | * Multiple status pages | ||||||
| * Ping Chart | * Map status pages to specific domains | ||||||
| * Certificate Info | * Ping chart | ||||||
|  | * Certificate info | ||||||
|  | * Proxy support | ||||||
|  | * 2FA support | ||||||
| 
 | 
 | ||||||
| ## 🔧 How to Install | ## 🔧 How to Install | ||||||
| 
 | 
 | ||||||
| ### 🐳 Docker | ### 🐳 Docker | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| docker volume create uptime-kuma |  | ||||||
| docker run -d --restart=always -p 3001:3001 -v uptime-kuma:/app/data --name uptime-kuma louislam/uptime-kuma:1 | docker run -d --restart=always -p 3001:3001 -v uptime-kuma:/app/data --name uptime-kuma louislam/uptime-kuma:1 | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| ⚠️ Please use a **local volume** only. Other types such as NFS are not supported. | ⚠️ 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 | ### 💪🏻 Non-Docker | ||||||
| 
 | 
 | ||||||
| Required Tools: Node.js >= 14, git and pm2. | 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 running Uptime Kuma in the background | ||||||
| 
 | 
 | ||||||
| ```bash | ```bash | ||||||
| # Update your npm to the latest version | # Update your npm to the latest version | ||||||
|  | @ -61,11 +67,25 @@ npm run setup | ||||||
| node server/server.js | node server/server.js | ||||||
| 
 | 
 | ||||||
| # (Recommended) Option 2. Run in background using PM2 | # (Recommended) Option 2. Run in background using PM2 | ||||||
| # Install PM2 if you don't have it: npm install pm2 -g | # Install PM2 if you don't have it:  | ||||||
| pm2 start server/server.js --name uptime-kuma | npm install pm2 -g && pm2 install pm2-logrotate | ||||||
| ``` |  | ||||||
| 
 | 
 | ||||||
| Browse to http://localhost:3001 after starting. | # Start Server | ||||||
|  | pm2 start server/server.js --name uptime-kuma | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | Uptime Kuma is now running on http://localhost:3001 | ||||||
|  | 
 | ||||||
|  | More useful PM2 Commands | ||||||
|  | 
 | ||||||
|  | ```bash | ||||||
|  | # If you want to see the current console output | ||||||
|  | pm2 monit | ||||||
|  | 
 | ||||||
|  | # If you want to add it to startup | ||||||
|  | pm2 save && pm2 startup | ||||||
|  | ``` | ||||||
| 
 | 
 | ||||||
| ### Advanced Installation | ### Advanced Installation | ||||||
| 
 | 
 | ||||||
|  | @ -87,13 +107,13 @@ https://github.com/louislam/uptime-kuma/milestones | ||||||
| 
 | 
 | ||||||
| Project Plan: | Project Plan: | ||||||
| 
 | 
 | ||||||
| https://github.com/louislam/uptime-kuma/projects/1 | https://github.com/users/louislam/projects/4/views/1 | ||||||
| 
 | 
 | ||||||
| ## ❤️ Sponsors | ## ❤️ Sponsors | ||||||
| 
 | 
 | ||||||
| Thank you so much! (GitHub Sponsors will be updated manually. OpenCollective sponsors will be updated automatically, the list will be cached by GitHub though. It may need some time to be updated) | Thank you so much! (GitHub Sponsors will be updated manually. OpenCollective sponsors will be updated automatically, the list will be cached by GitHub though. It may need some time to be updated) | ||||||
| 
 | 
 | ||||||
| <img src="https://uptime.kuma.pet/sponsors?v=3" alt /> | <img src="https://uptime.kuma.pet/sponsors?v=6" alt /> | ||||||
| 
 | 
 | ||||||
| ## 🖼 More Screenshots | ## 🖼 More Screenshots | ||||||
| 
 | 
 | ||||||
|  | @ -115,7 +135,7 @@ Telegram Notification Sample: | ||||||
| 
 | 
 | ||||||
| ## Motivation | ## Motivation | ||||||
| 
 | 
 | ||||||
| * I was looking for a self-hosted monitoring tool like "Uptime Robot", but it is hard to find a suitable one. One of the close ones is statping. Unfortunately, it is not stable and unmaintained. | * I was looking for a self-hosted monitoring tool like "Uptime Robot", but it is hard to find a suitable one. One of the close ones is statping. Unfortunately, it is not stable and no longer maintained. | ||||||
| * Want to build a fancy UI. | * Want to build a fancy UI. | ||||||
| * Learn Vue 3 and vite.js. | * Learn Vue 3 and vite.js. | ||||||
| * Show the power of Bootstrap 5. | * Show the power of Bootstrap 5. | ||||||
|  | @ -132,16 +152,30 @@ You can discuss or ask for help in [issues](https://github.com/louislam/uptime-k | ||||||
| 
 | 
 | ||||||
| ### Subreddit | ### Subreddit | ||||||
| 
 | 
 | ||||||
| My Reddit account: louislamlam | My Reddit account: [u/louislamlam](https://reddit.com/u/louislamlam).   | ||||||
| You can mention me if you ask a question on Reddit. | You can mention me if you ask a question on Reddit. | ||||||
| https://www.reddit.com/r/UptimeKuma/ | [r/Uptime kuma](https://www.reddit.com/r/UptimeKuma/) | ||||||
| 
 | 
 | ||||||
| ## Contribute | ## Contribute | ||||||
| 
 | 
 | ||||||
| If you want to report a bug or request a new feature. Free feel to open a [new issue](https://github.com/louislam/uptime-kuma/issues). | ### Test Pull Requests | ||||||
| 
 | 
 | ||||||
| If you want to translate Uptime Kuma into your language, please read: https://github.com/louislam/uptime-kuma/tree/master/src/languages | There are a lot of pull requests right now, but I don't have time to test them all. | ||||||
| 
 | 
 | ||||||
| If you want to modify Uptime Kuma, this guideline may be useful for you: https://github.com/louislam/uptime-kuma/blob/master/CONTRIBUTING.md | If you want to help, you can check this: | ||||||
|  | https://github.com/louislam/uptime-kuma/wiki/Test-Pull-Requests | ||||||
| 
 | 
 | ||||||
| English proofreading is needed too because my grammar is not that great, sadly. Feel free to correct my grammar in this README, source code, or wiki. | ### Test Beta Version | ||||||
|  | 
 | ||||||
|  | Check out the latest beta release here: https://github.com/louislam/uptime-kuma/releases | ||||||
|  | 
 | ||||||
|  | ### Bug Reports / Feature Requests | ||||||
|  | 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 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. | ||||||
|  | 
 | ||||||
|  | ### 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 | ||||||
|  |  | ||||||
							
								
								
									
										21
									
								
								SECURITY.md
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								SECURITY.md
									
									
									
									
									
								
							|  | @ -2,21 +2,20 @@ | ||||||
| 
 | 
 | ||||||
| ## Reporting a Vulnerability | ## 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 | ## Supported Versions | ||||||
| 
 | 
 | ||||||
| Use this section to tell people about which versions of your project are |  | ||||||
| currently being supported with security updates. |  | ||||||
| 
 |  | ||||||
| ### Uptime Kuma Versions | ### Uptime Kuma Versions | ||||||
| 
 | 
 | ||||||
| | Version | Supported          | | You should use or upgrade to the latest version of Uptime Kuma. All `1.X.X` versions are upgradable to the lastest version. | ||||||
| | ------- | ------------------ | |  | ||||||
| | 1.9.X  | :white_check_mark: | |  | ||||||
| | <= 1.8.X  | ❌ | |  | ||||||
| 
 | 
 | ||||||
| ### Upgradable Docker Tags | ### Upgradable Docker Tags | ||||||
| 
 | 
 | ||||||
|  | @ -24,8 +23,8 @@ currently being supported with security updates. | ||||||
| | ------- | ------------------ | | | ------- | ------------------ | | ||||||
| | 1 | :white_check_mark: | | | 1 | :white_check_mark: | | ||||||
| | 1-debian | :white_check_mark: | | | 1-debian | :white_check_mark: | | ||||||
| | 1-alpine | :white_check_mark: | |  | ||||||
| | latest | :white_check_mark: | | | latest | :white_check_mark: | | ||||||
| | debian | :white_check_mark: | | | debian | :white_check_mark: | | ||||||
| | alpine | :white_check_mark: | | | 1-alpine | ⚠️ Deprecated | | ||||||
|  | | alpine | ⚠️ Deprecated | | ||||||
| | All other tags  | ❌ | | | All other tags  | ❌ | | ||||||
|  |  | ||||||
							
								
								
									
										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, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
|  | @ -1,18 +1,38 @@ | ||||||
| import legacy from "@vitejs/plugin-legacy"; | import legacy from "@vitejs/plugin-legacy"; | ||||||
| import vue from "@vitejs/plugin-vue"; | import vue from "@vitejs/plugin-vue"; | ||||||
| import { defineConfig } from "vite"; | import { defineConfig } from "vite"; | ||||||
|  | import visualizer from "rollup-plugin-visualizer"; | ||||||
|  | import viteCompression from "vite-plugin-compression"; | ||||||
| 
 | 
 | ||||||
| const postCssScss = require("postcss-scss"); | const postCssScss = require("postcss-scss"); | ||||||
| const postcssRTLCSS = require("postcss-rtlcss"); | const postcssRTLCSS = require("postcss-rtlcss"); | ||||||
| 
 | 
 | ||||||
|  | const viteCompressionFilter = /\.(js|mjs|json|css|html|svg)$/i; | ||||||
|  | 
 | ||||||
| // https://vitejs.dev/config/
 | // https://vitejs.dev/config/
 | ||||||
| export default defineConfig({ | export default defineConfig({ | ||||||
|  |     server: { | ||||||
|  |         port: 3000, | ||||||
|  |     }, | ||||||
|  |     define: { | ||||||
|  |         "FRONTEND_VERSION": JSON.stringify(process.env.npm_package_version), | ||||||
|  |     }, | ||||||
|     plugins: [ |     plugins: [ | ||||||
|         vue(), |         vue(), | ||||||
|         legacy({ |         legacy({ | ||||||
|             targets: ["ie > 11"], |             targets: [ "since 2015" ], | ||||||
|             additionalLegacyPolyfills: ["regenerator-runtime/runtime"] |         }), | ||||||
|         }) |         visualizer({ | ||||||
|  |             filename: "tmp/dist-stats.html" | ||||||
|  |         }), | ||||||
|  |         viteCompression({ | ||||||
|  |             algorithm: "gzip", | ||||||
|  |             filter: viteCompressionFilter, | ||||||
|  |         }), | ||||||
|  |         viteCompression({ | ||||||
|  |             algorithm: "brotliCompress", | ||||||
|  |             filter: viteCompressionFilter, | ||||||
|  |         }), | ||||||
|     ], |     ], | ||||||
|     css: { |     css: { | ||||||
|         postcss: { |         postcss: { | ||||||
|  | @ -21,4 +41,13 @@ export default defineConfig({ | ||||||
|             "plugins": [ postcssRTLCSS ] |             "plugins": [ postcssRTLCSS ] | ||||||
|         } |         } | ||||||
|     }, |     }, | ||||||
|  |     build: { | ||||||
|  |         rollupOptions: { | ||||||
|  |             output: { | ||||||
|  |                 manualChunks(id, { getModuleInfo, getModuleIds }) { | ||||||
|  | 
 | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |     } | ||||||
| }); | }); | ||||||
|  |  | ||||||
							
								
								
									
										5
									
								
								db/patch-add-clickable-status-page-link.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								db/patch-add-clickable-status-page-link.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_group | ||||||
|  |     ADD send_url BOOLEAN DEFAULT 0 NOT NULL; | ||||||
|  | COMMIT; | ||||||
							
								
								
									
										18
									
								
								db/patch-add-docker-columns.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								db/patch-add-docker-columns.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | ||||||
|  | -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  | CREATE TABLE docker_host ( | ||||||
|  | 	id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||||
|  | 	user_id INT NOT NULL, | ||||||
|  | 	docker_daemon VARCHAR(255), | ||||||
|  | 	docker_type VARCHAR(255), | ||||||
|  | 	name VARCHAR(255) | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  | 	ADD docker_host INTEGER REFERENCES docker_host(id); | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  | 	ADD docker_container VARCHAR(255); | ||||||
|  | 
 | ||||||
|  | COMMIT; | ||||||
							
								
								
									
										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-other-auth.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								db/patch-add-other-auth.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  |   ALTER TABLE monitor | ||||||
|  |       ADD auth_method VARCHAR(250); | ||||||
|  | 
 | ||||||
|  |   ALTER TABLE monitor | ||||||
|  |       ADD auth_domain TEXT; | ||||||
|  |   ALTER TABLE monitor | ||||||
|  | 
 | ||||||
|  |       ADD auth_workstation TEXT; | ||||||
|  | 
 | ||||||
|  | COMMIT; | ||||||
|  | 
 | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  |   UPDATE monitor | ||||||
|  |         SET auth_method = 'basic' | ||||||
|  |         WHERE basic_auth_user is not null; | ||||||
|  | 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 | ||||||
							
								
								
									
										10
									
								
								db/patch-add-sqlserver-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								db/patch-add-sqlserver-monitor.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  |  ALTER TABLE monitor | ||||||
|  |      ADD database_connection_string VARCHAR(2000); | ||||||
|  | 
 | ||||||
|  |  ALTER TABLE monitor | ||||||
|  |      ADD database_query TEXT; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |  COMMIT | ||||||
							
								
								
									
										16
									
								
								db/patch-added-mqtt-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								db/patch-added-mqtt-monitor.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,16 @@ | ||||||
|  | -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  | 	ADD mqtt_topic TEXT; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  | 	ADD mqtt_success_message VARCHAR(255); | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  | 	ADD mqtt_username VARCHAR(255); | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  | 	ADD mqtt_password VARCHAR(255); | ||||||
|  | 
 | ||||||
|  | COMMIT; | ||||||
							
								
								
									
										25
									
								
								db/patch-grpc-monitor.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								db/patch-grpc-monitor.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | ||||||
|  | -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_url VARCHAR(255) default null; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_protobuf TEXT default null; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_body TEXT default null; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_metadata TEXT default null; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_method VARCHAR(255) default null; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_service_name VARCHAR(255) default null; | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor | ||||||
|  |     ADD grpc_enable_tls BOOLEAN default 0 not null; | ||||||
|  | 
 | ||||||
|  | COMMIT; | ||||||
							
								
								
									
										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; | ||||||
							
								
								
									
										7
									
								
								db/patch-monitor-expiry-notification.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								db/patch-monitor-expiry-notification.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 expiry_notification BOOLEAN default 1; | ||||||
|  | 
 | ||||||
|  | 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; | ||||||
							
								
								
									
										23
									
								
								db/patch-proxy.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								db/patch-proxy.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,23 @@ | ||||||
|  | -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  | CREATE TABLE proxy ( | ||||||
|  |     id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, | ||||||
|  |     user_id INT NOT NULL, | ||||||
|  |     protocol VARCHAR(10) NOT NULL, | ||||||
|  |     host VARCHAR(255) NOT NULL, | ||||||
|  |     port SMALLINT NOT NULL, | ||||||
|  |     auth BOOLEAN NOT NULL, | ||||||
|  |     username VARCHAR(255) NULL, | ||||||
|  |     password VARCHAR(255) NULL, | ||||||
|  |     active BOOLEAN NOT NULL DEFAULT 1, | ||||||
|  |     'default' BOOLEAN NOT NULL DEFAULT 0, | ||||||
|  |     created_date DATETIME DEFAULT (DATETIME('now')) NOT NULL | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | ALTER TABLE monitor ADD COLUMN proxy_id INTEGER REFERENCES proxy(id); | ||||||
|  | 
 | ||||||
|  | CREATE INDEX proxy_id ON monitor (proxy_id); | ||||||
|  | CREATE INDEX proxy_user_id ON proxy (user_id); | ||||||
|  | 
 | ||||||
|  | COMMIT; | ||||||
							
								
								
									
										6
									
								
								db/patch-status-page-footer-css.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								db/patch-status-page-footer-css.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | ||||||
|  | -- 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 footer_text TEXT; | ||||||
|  | ALTER TABLE status_page ADD custom_css TEXT; | ||||||
|  | ALTER TABLE status_page ADD show_powered_by BOOLEAN NOT NULL DEFAULT 1; | ||||||
|  | COMMIT; | ||||||
							
								
								
									
										31
									
								
								db/patch-status-page.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								db/patch-status-page.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,31 @@ | ||||||
|  | -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||||
|  | BEGIN TRANSACTION; | ||||||
|  | 
 | ||||||
|  | CREATE TABLE [status_page]( | ||||||
|  |     [id] INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, | ||||||
|  |     [slug] VARCHAR(255) NOT NULL UNIQUE, | ||||||
|  |     [title] VARCHAR(255) NOT NULL, | ||||||
|  |     [description] TEXT, | ||||||
|  |     [icon] VARCHAR(255) NOT NULL, | ||||||
|  |     [theme] VARCHAR(30) NOT NULL, | ||||||
|  |     [published] BOOLEAN NOT NULL DEFAULT 1, | ||||||
|  |     [search_engine_index] BOOLEAN NOT NULL DEFAULT 1, | ||||||
|  |     [show_tags] BOOLEAN NOT NULL DEFAULT 0, | ||||||
|  |     [password] VARCHAR, | ||||||
|  |     [created_date] DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP, | ||||||
|  |     [modified_date] DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | CREATE UNIQUE INDEX [slug] ON [status_page]([slug]); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | CREATE TABLE [status_page_cname]( | ||||||
|  |     [id] INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, | ||||||
|  |     [status_page_id] INTEGER NOT NULL REFERENCES [status_page]([id]) ON DELETE CASCADE ON UPDATE CASCADE, | ||||||
|  |     [domain] VARCHAR NOT NULL UNIQUE | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | ALTER TABLE incident ADD status_page_id INTEGER; | ||||||
|  | ALTER TABLE [group] ADD status_page_id INTEGER; | ||||||
|  | 
 | ||||||
|  | COMMIT; | ||||||
|  | @ -1,8 +1,8 @@ | ||||||
| # DON'T UPDATE TO alpine3.13, 1.14, see #41. | # DON'T UPDATE TO alpine3.13, 1.14, see #41. | ||||||
| FROM node:14-alpine3.12 | FROM node:16-alpine3.12 | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| 
 | 
 | ||||||
| # Install apprise, iputils for non-root ping, setpriv | # 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 && \ | 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==0.9.6 && \ |     pip3 --no-cache-dir install apprise==1.2.1 && \ | ||||||
|     rm -rf /root/.cache |     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 | ||||||
|  | @ -1,12 +1,28 @@ | ||||||
| # DON'T UPDATE TO node:14-bullseye-slim, see #372. | # DON'T UPDATE TO node:14-bullseye-slim, see #372. | ||||||
| # If the image changed, the second stage image should be changed too | # If the image changed, the second stage image should be changed too | ||||||
| FROM node:14-buster-slim | FROM node:16-buster-slim | ||||||
|  | ARG TARGETPLATFORM | ||||||
|  | 
 | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| 
 | 
 | ||||||
|  | # Install Curl | ||||||
| # Install Apprise, add sqlite3 cli for debugging in the future, iputils-ping for ping, util-linux for setpriv | # Install Apprise, add sqlite3 cli for debugging in the future, iputils-ping for ping, util-linux for setpriv | ||||||
| # Stupid python3 and python3-pip actually install a lot of useless things into Debian, specify --no-install-recommends to skip them, make the base even smaller than alpine! | # 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 && \ | 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 \ |     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 && \ |         sqlite3 iputils-ping util-linux dumb-init git && \ | ||||||
|     pip3 --no-cache-dir install apprise==0.9.6 && \ |     pip3 --no-cache-dir install apprise==1.2.1 && \ | ||||||
|     rm -rf /var/lib/apt/lists/* |     rm -rf /var/lib/apt/lists/* && \ | ||||||
|  |     apt --yes autoremove | ||||||
|  | 
 | ||||||
|  | # Install cloudflared | ||||||
|  | # dpkg --add-architecture arm: cloudflared do not provide armhf, this is workaround. Read more: https://github.com/cloudflare/cloudflared/issues/583 | ||||||
|  | COPY extra/download-cloudflared.js ./extra/download-cloudflared.js | ||||||
|  | RUN node ./extra/download-cloudflared.js $TARGETPLATFORM && \ | ||||||
|  |     dpkg --add-architecture arm && \ | ||||||
|  |     apt update && \ | ||||||
|  |     apt --yes --no-install-recommends install ./cloudflared.deb && \ | ||||||
|  |     rm -rf /var/lib/apt/lists/* && \ | ||||||
|  |     rm -f cloudflared.deb && \ | ||||||
|  |     apt --yes autoremove | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -1,13 +1,14 @@ | ||||||
| # Simple docker-composer.yml | # Simple docker-compose.yml | ||||||
| # You can change your port or volume location | # You can change your port or volume location | ||||||
| 
 | 
 | ||||||
| version: '3.3' | version: '3.3' | ||||||
| 
 | 
 | ||||||
| services: | services: | ||||||
|   uptime-kuma: |   uptime-kuma: | ||||||
|     image: louislam/uptime-kuma |     image: louislam/uptime-kuma:1 | ||||||
|     container_name: uptime-kuma |     container_name: uptime-kuma | ||||||
|     volumes: |     volumes: | ||||||
|       - ./uptime-kuma:/app/data |       - ./uptime-kuma-data:/app/data | ||||||
|     ports: |     ports: | ||||||
|       - 3001:3001 |       - 3001:3001  # <Host Port>:<Container Port> | ||||||
|  |     restart: always | ||||||
|  |  | ||||||
|  | @ -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 | FROM louislam/uptime-kuma:base-debian AS build | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| 
 | 
 | ||||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | 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 . . | COPY . . | ||||||
| RUN npm ci --production && \ | COPY --from=build_healthcheck /app/extra/healthcheck /app/extra/healthcheck | ||||||
|     chmod +x /app/extra/entrypoint.sh | RUN chmod +x /app/extra/entrypoint.sh | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
|  | ############################################ | ||||||
|  | # ⭐ Main Image | ||||||
|  | ############################################ | ||||||
| FROM louislam/uptime-kuma:base-debian AS release | FROM louislam/uptime-kuma:base-debian AS release | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
| 
 | 
 | ||||||
| # Copy app files from build layer | # Copy app files from build layer | ||||||
| COPY --from=build /app /app | COPY --from=build /app /app | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| EXPOSE 3001 | EXPOSE 3001 | ||||||
| VOLUME ["/app/data"] | 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"] | ENTRYPOINT ["/usr/bin/dumb-init", "--", "extra/entrypoint.sh"] | ||||||
| CMD ["node", "server/server.js"] | CMD ["node", "server/server.js"] | ||||||
| 
 | 
 | ||||||
| 
 | ############################################ | ||||||
|  | # Mark as Nightly | ||||||
|  | ############################################ | ||||||
| FROM release AS nightly | FROM release AS nightly | ||||||
| RUN npm run mark-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 | # Upload the artifact to Github | ||||||
|  | ############################################ | ||||||
| FROM louislam/uptime-kuma:base-debian AS upload-artifact | FROM louislam/uptime-kuma:base-debian AS upload-artifact | ||||||
| WORKDIR / | WORKDIR / | ||||||
| RUN apt update && \ | RUN apt update && \ | ||||||
|  |  | ||||||
|  | @ -3,10 +3,12 @@ WORKDIR /app | ||||||
| 
 | 
 | ||||||
| ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 | 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 . . | COPY . . | ||||||
| RUN npm ci --production && \ | RUN chmod +x /app/extra/entrypoint.sh | ||||||
|     chmod +x /app/extra/entrypoint.sh |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| FROM louislam/uptime-kuma:base-alpine AS release | FROM louislam/uptime-kuma:base-alpine AS release | ||||||
| WORKDIR /app | WORKDIR /app | ||||||
|  |  | ||||||
|  | @ -3,4 +3,4 @@ module.exports = { | ||||||
|         name: "uptime-kuma", |         name: "uptime-kuma", | ||||||
|         script: "./server/server.js", |         script: "./server/server.js", | ||||||
|     }] |     }] | ||||||
| } | }; | ||||||
|  |  | ||||||
							
								
								
									
										80
									
								
								extra/beta/update-version.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								extra/beta/update-version.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,80 @@ | ||||||
|  | const pkg = require("../../package.json"); | ||||||
|  | const fs = require("fs"); | ||||||
|  | const childProcess = require("child_process"); | ||||||
|  | const util = require("../../src/util"); | ||||||
|  | 
 | ||||||
|  | util.polyfill(); | ||||||
|  | 
 | ||||||
|  | const version = process.env.VERSION; | ||||||
|  | 
 | ||||||
|  | console.log("Beta Version: " + version); | ||||||
|  | 
 | ||||||
|  | if (!version || !version.includes("-beta.")) { | ||||||
|  |     console.error("invalid version, beta version only"); | ||||||
|  |     process.exit(1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const exists = tagExists(version); | ||||||
|  | 
 | ||||||
|  | if (! exists) { | ||||||
|  |     // Process package.json
 | ||||||
|  |     pkg.version = version; | ||||||
|  |     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n"); | ||||||
|  | 
 | ||||||
|  |     // Also update package-lock.json
 | ||||||
|  |     const npm = /^win/.test(process.platform) ? "npm.cmd" : "npm"; | ||||||
|  |     childProcess.spawnSync(npm, [ "install" ]); | ||||||
|  | 
 | ||||||
|  |     commit(version); | ||||||
|  |     tag(version); | ||||||
|  | 
 | ||||||
|  | } else { | ||||||
|  |     console.log("version tag exists, please delete the tag or use another tag"); | ||||||
|  |     process.exit(1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Commit updated files | ||||||
|  |  * @param {string} version Version to update to | ||||||
|  |  */ | ||||||
|  | function commit(version) { | ||||||
|  |     let msg = "Update to " + version; | ||||||
|  | 
 | ||||||
|  |     let res = childProcess.spawnSync("git", [ "commit", "-m", msg, "-a" ]); | ||||||
|  |     let stdout = res.stdout.toString().trim(); | ||||||
|  |     console.log(stdout); | ||||||
|  | 
 | ||||||
|  |     if (stdout.includes("no changes added to commit")) { | ||||||
|  |         throw new Error("commit error"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     res = childProcess.spawnSync("git", [ "push", "origin", "master" ]); | ||||||
|  |     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()); | ||||||
|  | 
 | ||||||
|  |     res = childProcess.spawnSync("git", [ "push", "origin", 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"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     let res = childProcess.spawnSync("git", [ "tag", "-l", version ]); | ||||||
|  | 
 | ||||||
|  |     return res.stdout.toString().trim() === 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()); | ||||||
|  | @ -37,7 +37,7 @@ const github = require("@actions/github"); | ||||||
|                 owner: issue.owner, |                 owner: issue.owner, | ||||||
|                 repo: issue.repo, |                 repo: issue.repo, | ||||||
|                 issue_number: issue.number, |                 issue_number: issue.number, | ||||||
|                 body: `@${username}: Hello! :wave:\n\nThis issue is being automatically closed because it does not follow the issue template. Please DO NOT open a blank issue` |                 body: `@${username}: Hello! :wave:\n\nThis issue is being automatically closed because it does not follow the issue template. Please DO NOT open a blank issue.` | ||||||
|             }); |             }); | ||||||
| 
 | 
 | ||||||
|             // Close the issue
 |             // Close the issue
 | ||||||
|  |  | ||||||
							
								
								
									
										59
									
								
								extra/deploy-demo-server.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								extra/deploy-demo-server.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | ||||||
|  | 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)); | ||||||
							
								
								
									
										48
									
								
								extra/download-cloudflared.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								extra/download-cloudflared.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,48 @@ | ||||||
|  | //
 | ||||||
|  | 
 | ||||||
|  | const http = require("https"); // or 'https' for https:// URLs
 | ||||||
|  | const fs = require("fs"); | ||||||
|  | 
 | ||||||
|  | const platform = process.argv[2]; | ||||||
|  | 
 | ||||||
|  | if (!platform) { | ||||||
|  |     console.error("No platform??"); | ||||||
|  |     process.exit(1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | let arch = null; | ||||||
|  | 
 | ||||||
|  | if (platform === "linux/amd64") { | ||||||
|  |     arch = "amd64"; | ||||||
|  | } else if (platform === "linux/arm64") { | ||||||
|  |     arch = "arm64"; | ||||||
|  | } else if (platform === "linux/arm/v7") { | ||||||
|  |     arch = "arm"; | ||||||
|  | } else { | ||||||
|  |     console.error("Invalid platform?? " + platform); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const file = fs.createWriteStream("cloudflared.deb"); | ||||||
|  | get("https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-linux-" + arch + ".deb"); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Download specified file | ||||||
|  |  * @param {string} url URL to request | ||||||
|  |  */ | ||||||
|  | function get(url) { | ||||||
|  |     http.get(url, function (res) { | ||||||
|  |         if (res.statusCode >= 300 && res.statusCode < 400 && res.headers.location) { | ||||||
|  |             console.log("Redirect to " + res.headers.location); | ||||||
|  |             get(res.headers.location); | ||||||
|  |         } else if (res.statusCode >= 200 && res.statusCode < 300) { | ||||||
|  |             res.pipe(file); | ||||||
|  | 
 | ||||||
|  |             res.on("end", function () { | ||||||
|  |                 console.log("Downloaded"); | ||||||
|  |             }); | ||||||
|  |         } else { | ||||||
|  |             console.error(res.statusCode); | ||||||
|  |             process.exit(1); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | @ -4,6 +4,7 @@ const tar = require("tar"); | ||||||
| 
 | 
 | ||||||
| const packageJSON = require("../package.json"); | const packageJSON = require("../package.json"); | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
|  | const rmSync = require("./fs-rmSync.js"); | ||||||
| const version = packageJSON.version; | const version = packageJSON.version; | ||||||
| 
 | 
 | ||||||
| const filename = "dist.tar.gz"; | const filename = "dist.tar.gz"; | ||||||
|  | @ -11,6 +12,12 @@ const filename = "dist.tar.gz"; | ||||||
| const url = `https://github.com/louislam/uptime-kuma/releases/download/${version}/${filename}`; | const url = `https://github.com/louislam/uptime-kuma/releases/download/${version}/${filename}`; | ||||||
| download(url); | download(url); | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Downloads the latest version of the dist from a GitHub release. | ||||||
|  |  * @param {string} url The URL to download from. | ||||||
|  |  * | ||||||
|  |  * Generated by Trelent | ||||||
|  |  */ | ||||||
| function download(url) { | function download(url) { | ||||||
|     console.log(url); |     console.log(url); | ||||||
| 
 | 
 | ||||||
|  | @ -21,7 +28,7 @@ function download(url) { | ||||||
|             if (fs.existsSync("./dist")) { |             if (fs.existsSync("./dist")) { | ||||||
| 
 | 
 | ||||||
|                 if (fs.existsSync("./dist-backup")) { |                 if (fs.existsSync("./dist-backup")) { | ||||||
|                     fs.rmdirSync("./dist-backup", { |                     rmSync("./dist-backup", { | ||||||
|                         recursive: true |                         recursive: true | ||||||
|                     }); |                     }); | ||||||
|                 } |                 } | ||||||
|  | @ -35,11 +42,12 @@ function download(url) { | ||||||
| 
 | 
 | ||||||
|             tarStream.on("close", () => { |             tarStream.on("close", () => { | ||||||
|                 if (fs.existsSync("./dist-backup")) { |                 if (fs.existsSync("./dist-backup")) { | ||||||
|                     fs.rmdirSync("./dist-backup", { |                     rmSync("./dist-backup", { | ||||||
|                         recursive: true |                         recursive: true | ||||||
|                     }); |                     }); | ||||||
|                 } |                 } | ||||||
|                 console.log("Done"); |                 console.log("Done"); | ||||||
|  |                 process.exit(0); | ||||||
|             }); |             }); | ||||||
| 
 | 
 | ||||||
|             tarStream.on("error", () => { |             tarStream.on("error", () => { | ||||||
|  |  | ||||||
							
								
								
									
										19
									
								
								extra/env2arg.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								extra/env2arg.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,19 @@ | ||||||
|  | #!/usr/bin/env node
 | ||||||
|  | 
 | ||||||
|  | const childProcess = require("child_process"); | ||||||
|  | let env = process.env; | ||||||
|  | 
 | ||||||
|  | let cmd = process.argv[2]; | ||||||
|  | let args = process.argv.slice(3); | ||||||
|  | let replacedArgs = []; | ||||||
|  | 
 | ||||||
|  | for (let arg of args) { | ||||||
|  |     for (let key in env) { | ||||||
|  |         arg = arg.replaceAll(`$${key}`, env[key]); | ||||||
|  |     } | ||||||
|  |     replacedArgs.push(arg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | let child = childProcess.spawn(cmd, replacedArgs); | ||||||
|  | child.stdout.pipe(process.stdout); | ||||||
|  | child.stderr.pipe(process.stderr); | ||||||
							
								
								
									
										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> | ||||||
							
								
								
									
										65
									
								
								extra/exe-builder/FS.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								extra/exe-builder/FS.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,65 @@ | ||||||
|  | using System.IO; | ||||||
|  | using System.Reflection; | ||||||
|  | 
 | ||||||
|  | namespace UptimeKuma { | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Current Directory using App location | ||||||
|  |      */ | ||||||
|  |     public class Directory { | ||||||
|  |         private static string baseDir; | ||||||
|  | 
 | ||||||
|  |         public static string FullPath(string path) { | ||||||
|  |             return Path.Combine(GetBaseDir(), path); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static string GetBaseDir() { | ||||||
|  |             if (baseDir == null) { | ||||||
|  |                 baseDir = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location); | ||||||
|  |             } | ||||||
|  |             return baseDir; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static bool Exists(string path) { | ||||||
|  |             return System.IO.Directory.Exists(FullPath(path)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void Delete(string path, bool recursive) { | ||||||
|  |             System.IO.Directory.Delete(FullPath(path), recursive); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void Move(string src, string dest) { | ||||||
|  |             System.IO.Directory.Move(FullPath(src), FullPath(dest)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static string[] GetDirectories(string path) { | ||||||
|  |             return System.IO.Directory.GetDirectories(FullPath(path)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public class File { | ||||||
|  | 
 | ||||||
|  |         private static string FullPath(string path) { | ||||||
|  |             return Directory.FullPath(path); | ||||||
|  |         } | ||||||
|  |         public static bool Exists(string path) { | ||||||
|  |             return System.IO.File.Exists(FullPath(path)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static FileStream Create(string path) { | ||||||
|  |             return System.IO.File.Create(FullPath(path)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static string ReadAllText(string path) { | ||||||
|  |             return System.IO.File.ReadAllText(FullPath(path)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void Delete(string path) { | ||||||
|  |             System.IO.File.Delete(FullPath(path)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void WriteAllText(string path, string content) { | ||||||
|  |             System.IO.File.WriteAllText(FullPath(path), content); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										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> | ||||||
							
								
								
									
										197
									
								
								extra/exe-builder/Program.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										197
									
								
								extra/exe-builder/Program.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,197 @@ | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Diagnostics; | ||||||
|  | using System.Drawing; | ||||||
|  | using System.IO; | ||||||
|  | using System.Linq; | ||||||
|  | using System.Net; | ||||||
|  | using System.Reflection; | ||||||
|  | using System.Runtime.InteropServices; | ||||||
|  | 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) { | ||||||
|  |             Application.EnableVisualStyles(); | ||||||
|  |             Application.SetCompatibleTextRenderingDefault(false); | ||||||
|  |             Application.Run(new UptimeKumaApplicationContext()); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public class UptimeKumaApplicationContext : ApplicationContext | ||||||
|  |     { | ||||||
|  |         const string appName = "Uptime Kuma"; | ||||||
|  | 
 | ||||||
|  |         private NotifyIcon trayIcon; | ||||||
|  |         private Process process; | ||||||
|  | 
 | ||||||
|  |         private MenuItem runWhenStarts; | ||||||
|  | 
 | ||||||
|  |         private RegistryKey registryKey = Registry.CurrentUser.OpenSubKey("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run", true); | ||||||
|  | 
 | ||||||
|  |         public UptimeKumaApplicationContext() | ||||||
|  |         { | ||||||
|  |             trayIcon = new NotifyIcon(); | ||||||
|  | 
 | ||||||
|  |             runWhenStarts = new MenuItem("Run when system starts", RunWhenStarts); | ||||||
|  |             runWhenStarts.Checked = registryKey.GetValue(appName) != null; | ||||||
|  | 
 | ||||||
|  |             trayIcon.Icon = Icon.ExtractAssociatedIcon(Assembly.GetExecutingAssembly().Location); | ||||||
|  |             trayIcon.ContextMenu = new ContextMenu(new MenuItem[] { | ||||||
|  |                 new("Open", Open), | ||||||
|  |                 //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); | ||||||
|  | 
 | ||||||
|  |             } 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 © 2022 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> | ||||||
							
								
								
									
										213
									
								
								extra/exe-builder/UptimeKuma.csproj
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								extra/exe-builder/UptimeKuma.csproj
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,213 @@ | ||||||
|  | <?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="FS.cs" /> | ||||||
|  |         <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> | ||||||
							
								
								
									
										23
									
								
								extra/fs-rmSync.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								extra/fs-rmSync.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,23 @@ | ||||||
|  | const fs = require("fs"); | ||||||
|  | /** | ||||||
|  |  * Detect if `fs.rmSync` is available | ||||||
|  |  * to avoid the runtime deprecation warning triggered for using `fs.rmdirSync` with `{ recursive: true }` in Node.js v16, | ||||||
|  |  * or the `recursive` property removing completely in the future Node.js version. | ||||||
|  |  * See the link below. | ||||||
|  |  * | ||||||
|  |  * @todo Once we drop the support for Node.js v14 (or at least versions before v14.14.0), we can safely replace this function with `fs.rmSync`, since `fs.rmSync` was add in Node.js v14.14.0 and currently we supports all the Node.js v14 versions that include the versions before the v14.14.0, and this function have almost the same signature with `fs.rmSync`. | ||||||
|  |  * @link https://nodejs.org/docs/latest-v16.x/api/deprecations.html#dep0147-fsrmdirpath--recursive-true- the deprecation infomation of `fs.rmdirSync`
 | ||||||
|  |  * @link https://nodejs.org/docs/latest-v16.x/api/fs.html#fsrmsyncpath-options the document of `fs.rmSync`
 | ||||||
|  |  * @param {fs.PathLike} path Valid types for path values in "fs". | ||||||
|  |  * @param {fs.RmDirOptions} [options] options for `fs.rmdirSync`, if `fs.rmSync` is available and property `recursive` is true, it will automatically have property `force` with value `true`. | ||||||
|  |  */ | ||||||
|  | const rmSync = (path, options) => { | ||||||
|  |     if (typeof fs.rmSync === "function") { | ||||||
|  |         if (options.recursive) { | ||||||
|  |             options.force = true; | ||||||
|  |         } | ||||||
|  |         return fs.rmSync(path, options); | ||||||
|  |     } | ||||||
|  |     return fs.rmdirSync(path, options); | ||||||
|  | }; | ||||||
|  | module.exports = rmSync; | ||||||
							
								
								
									
										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. |  * 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"); | const { FBSD } = require("../server/util-server"); | ||||||
|  | @ -20,7 +25,7 @@ if (sslKey && sslCert) { | ||||||
| // Dual-stack support for (::)
 | // Dual-stack support for (::)
 | ||||||
| let hostname = process.env.UPTIME_KUMA_HOST; | let hostname = process.env.UPTIME_KUMA_HOST; | ||||||
| 
 | 
 | ||||||
| // Also read HOST if not FreeBSD, as HOST is a system environment variable in FreeBSD
 | // Also read HOST if not *BSD, as HOST is a system environment variable in FreeBSD
 | ||||||
| if (!hostname && !FBSD) { | if (!hostname && !FBSD) { | ||||||
|     hostname = process.env.HOST; |     hostname = process.env.HOST; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -189,7 +189,7 @@ if (type == "local") { | ||||||
|    bash("check=$(pm2 --version)"); |    bash("check=$(pm2 --version)"); | ||||||
|    if (check == "") { |    if (check == "") { | ||||||
|        println("Installing PM2"); |        println("Installing PM2"); | ||||||
|        bash("npm install pm2 -g"); |        bash("npm install pm2 -g && pm2 install pm2-logrotate"); | ||||||
|        bash("pm2 startup"); |        bash("pm2 startup"); | ||||||
|    } |    } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,24 +1,25 @@ | ||||||
| const pkg = require("../package.json"); | const pkg = require("../package.json"); | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const util = require("../src/util"); | const util = require("../src/util"); | ||||||
|  | const dayjs = require("dayjs"); | ||||||
| 
 | 
 | ||||||
| util.polyfill(); | util.polyfill(); | ||||||
| 
 | 
 | ||||||
| const oldVersion = pkg.version | const oldVersion = pkg.version; | ||||||
| const newVersion = oldVersion + "-nightly" | const newVersion = oldVersion + "-nightly-" + dayjs().format("YYYYMMDDHHmmss"); | ||||||
| 
 | 
 | ||||||
| console.log("Old Version: " + oldVersion) | console.log("Old Version: " + oldVersion); | ||||||
| console.log("New Version: " + newVersion) | console.log("New Version: " + newVersion); | ||||||
| 
 | 
 | ||||||
| if (newVersion) { | if (newVersion) { | ||||||
|     // Process package.json
 |     // Process package.json
 | ||||||
|     pkg.version = newVersion |     pkg.version = newVersion; | ||||||
|     pkg.scripts.setup = pkg.scripts.setup.replaceAll(oldVersion, newVersion) |     pkg.scripts.setup = pkg.scripts.setup.replaceAll(oldVersion, newVersion); | ||||||
|     pkg.scripts["build-docker"] = pkg.scripts["build-docker"].replaceAll(oldVersion, newVersion) |     pkg.scripts["build-docker"] = pkg.scripts["build-docker"].replaceAll(oldVersion, newVersion); | ||||||
|     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n") |     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n"); | ||||||
| 
 | 
 | ||||||
|     // Process README.md
 |     // Process README.md
 | ||||||
|     if (fs.existsSync("README.md")) { |     if (fs.existsSync("README.md")) { | ||||||
|         fs.writeFileSync("README.md", fs.readFileSync("README.md", "utf8").replaceAll(oldVersion, newVersion)) |         fs.writeFileSync("README.md", fs.readFileSync("README.md", "utf8").replaceAll(oldVersion, newVersion)); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										6
									
								
								extra/press-any-key.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								extra/press-any-key.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,6 @@ | ||||||
|  | console.log("Git Push and Publish the release note on github, then press any key to continue"); | ||||||
|  | 
 | ||||||
|  | process.stdin.setRawMode(true); | ||||||
|  | process.stdin.resume(); | ||||||
|  | process.stdin.on("data", process.exit.bind(process, 0)); | ||||||
|  | 
 | ||||||
|  | @ -43,6 +43,11 @@ const main = async () => { | ||||||
|     console.log("Finished."); |     console.log("Finished."); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Ask question of user | ||||||
|  |  * @param {string} question Question to ask | ||||||
|  |  * @returns {Promise<string>} Users response | ||||||
|  |  */ | ||||||
| function question(question) { | function question(question) { | ||||||
|     return new Promise((resolve) => { |     return new Promise((resolve) => { | ||||||
|         rl.question(question, (answer) => { |         rl.question(question, (answer) => { | ||||||
|  |  | ||||||
|  | @ -1,11 +1,10 @@ | ||||||
| console.log("== Uptime Kuma Reset Password Tool =="); | console.log("== Uptime Kuma Reset Password Tool =="); | ||||||
| 
 | 
 | ||||||
| console.log("Loading the database"); |  | ||||||
| 
 |  | ||||||
| const Database = require("../server/database"); | const Database = require("../server/database"); | ||||||
| const { R } = require("redbean-node"); | const { R } = require("redbean-node"); | ||||||
| const readline = require("readline"); | const readline = require("readline"); | ||||||
| const { initJWTSecret } = require("../server/util-server"); | const { initJWTSecret } = require("../server/util-server"); | ||||||
|  | const User = require("../server/model/user"); | ||||||
| const args = require("args-parser")(process.argv); | const args = require("args-parser")(process.argv); | ||||||
| const rl = readline.createInterface({ | const rl = readline.createInterface({ | ||||||
|     input: process.stdin, |     input: process.stdin, | ||||||
|  | @ -13,8 +12,9 @@ const rl = readline.createInterface({ | ||||||
| }); | }); | ||||||
| 
 | 
 | ||||||
| const main = async () => { | const main = async () => { | ||||||
|  |     console.log("Connecting the database"); | ||||||
|     Database.init(args); |     Database.init(args); | ||||||
|     await Database.connect(); |     await Database.connect(false, false, true); | ||||||
| 
 | 
 | ||||||
|     try { |     try { | ||||||
|         // No need to actually reset the password for testing, just make sure no connection problem. It is ok for now.
 |         // No need to actually reset the password for testing, just make sure no connection problem. It is ok for now.
 | ||||||
|  | @ -31,7 +31,7 @@ const main = async () => { | ||||||
|                 let confirmPassword = await question("Confirm New Password: "); |                 let confirmPassword = await question("Confirm New Password: "); | ||||||
| 
 | 
 | ||||||
|                 if (password === confirmPassword) { |                 if (password === confirmPassword) { | ||||||
|                     await user.resetPassword(password); |                     await User.resetPassword(user.id, password); | ||||||
| 
 | 
 | ||||||
|                     // Reset all sessions by reset jwt secret
 |                     // Reset all sessions by reset jwt secret
 | ||||||
|                     await initJWTSecret(); |                     await initJWTSecret(); | ||||||
|  | @ -53,6 +53,11 @@ const main = async () => { | ||||||
|     console.log("Finished."); |     console.log("Finished."); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Ask question of user | ||||||
|  |  * @param {string} question Question to ask | ||||||
|  |  * @returns {Promise<string>} Users response | ||||||
|  |  */ | ||||||
| function question(question) { | function question(question) { | ||||||
|     return new Promise((resolve) => { |     return new Promise((resolve) => { | ||||||
|         rl.question(question, (answer) => { |         rl.question(question, (answer) => { | ||||||
|  |  | ||||||
|  | @ -26,7 +26,7 @@ server.on("request", (request, send, rinfo) => { | ||||||
|                     ttl: 300, |                     ttl: 300, | ||||||
|                     address: "1.2.3.4" |                     address: "1.2.3.4" | ||||||
|                 }); |                 }); | ||||||
|             } if (question.type === Packet.TYPE.AAAA) { |             } else if (question.type === Packet.TYPE.AAAA) { | ||||||
|                 response.answers.push({ |                 response.answers.push({ | ||||||
|                     name: question.name, |                     name: question.name, | ||||||
|                     type: question.type, |                     type: question.type, | ||||||
|  | @ -135,6 +135,11 @@ server.listen({ | ||||||
|     udp: 5300 |     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) { | function type(code) { | ||||||
|     for (let name in Packet.TYPE) { |     for (let name in Packet.TYPE) { | ||||||
|         if (Packet.TYPE[name] === code) { |         if (Packet.TYPE[name] === code) { | ||||||
|  |  | ||||||
							
								
								
									
										51
									
								
								extra/simple-mqtt-server.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								extra/simple-mqtt-server.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,51 @@ | ||||||
|  | const { log } = require("../src/util"); | ||||||
|  | 
 | ||||||
|  | const mqttUsername = "louis1"; | ||||||
|  | const mqttPassword = "!@#$LLam"; | ||||||
|  | 
 | ||||||
|  | class SimpleMqttServer { | ||||||
|  |     aedes = require("aedes")(); | ||||||
|  |     server = require("net").createServer(this.aedes.handle); | ||||||
|  | 
 | ||||||
|  |     constructor(port) { | ||||||
|  |         this.port = port; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Start the MQTT server */ | ||||||
|  |     start() { | ||||||
|  |         this.server.listen(this.port, () => { | ||||||
|  |             console.log("server started and listening on port ", this.port); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | let server1 = new SimpleMqttServer(10000); | ||||||
|  | 
 | ||||||
|  | server1.aedes.authenticate = function (client, username, password, callback) { | ||||||
|  |     if (username && password) { | ||||||
|  |         console.log(password.toString("utf-8")); | ||||||
|  |         callback(null, username === mqttUsername && password.toString("utf-8") === mqttPassword); | ||||||
|  |     } else { | ||||||
|  |         callback(null, false); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | server1.aedes.on("subscribe", (subscriptions, client) => { | ||||||
|  |     console.log(subscriptions); | ||||||
|  | 
 | ||||||
|  |     for (let s of subscriptions) { | ||||||
|  |         if (s.topic === "test") { | ||||||
|  |             server1.aedes.publish({ | ||||||
|  |                 topic: "test", | ||||||
|  |                 payload: Buffer.from("ok"), | ||||||
|  |             }, (error) => { | ||||||
|  |                 if (error) { | ||||||
|  |                     log.error("mqtt_server", error); | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | server1.start(); | ||||||
|  | @ -1,50 +1,45 @@ | ||||||
| // Need to use ES6 to read language files
 | // Need to use ES6 to read language files
 | ||||||
| 
 | 
 | ||||||
| import fs from "fs"; | import fs from "fs"; | ||||||
| import path from "path"; |  | ||||||
| import util from "util"; | 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. |  * Copy across the required language files | ||||||
|  * @param {string} src  The path to the thing to copy. |  * Creates a local directory (./languages) and copies the required files | ||||||
|  * @param {string} dest The path to the new copy. |  * 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) { | function copyFiles(langCode, baseLang) { | ||||||
|     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); |  | ||||||
|     if (fs.existsSync("./languages")) { |     if (fs.existsSync("./languages")) { | ||||||
|     fs.rmdirSync("./languages", { recursive: true }); |         rmSync("./languages", { recursive: true }); | ||||||
|     } |     } | ||||||
| copyRecursiveSync("../../src/languages", "./languages"); |     fs.mkdirSync("./languages"); | ||||||
| 
 | 
 | ||||||
|  |     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 en = (await import("./languages/en.js")).default; | ||||||
|     const baseLang = (await import(`./languages/${baseLangCode}.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; |  | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|  |     let file = langCode + ".js"; | ||||||
|     console.log("Processing " + file); |     console.log("Processing " + file); | ||||||
|     const lang = await import("./languages/" + file); |     const lang = await import("./languages/" + file); | ||||||
| 
 | 
 | ||||||
|  | @ -82,5 +77,20 @@ for (const file of files) { | ||||||
|     fs.writeFileSync(`../../src/languages/${file}`, code); |     fs.writeFileSync(`../../src/languages/${file}`, code); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| fs.rmdirSync("./languages", { recursive: true }); | // 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..."); | console.log("Done. Fixing formatting by ESLint..."); | ||||||
|  |  | ||||||
|  | @ -1,14 +1,12 @@ | ||||||
| const pkg = require("../package.json"); | const pkg = require("../package.json"); | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const child_process = require("child_process"); | const childProcess = require("child_process"); | ||||||
| const util = require("../src/util"); | const util = require("../src/util"); | ||||||
| 
 | 
 | ||||||
| util.polyfill(); | util.polyfill(); | ||||||
| 
 | 
 | ||||||
| const oldVersion = pkg.version; | const newVersion = process.env.VERSION; | ||||||
| const newVersion = process.argv[2]; |  | ||||||
| 
 | 
 | ||||||
| console.log("Old Version: " + oldVersion); |  | ||||||
| console.log("New Version: " + newVersion); | console.log("New Version: " + newVersion); | ||||||
| 
 | 
 | ||||||
| if (! newVersion) { | if (! newVersion) { | ||||||
|  | @ -22,25 +20,30 @@ if (! exists) { | ||||||
| 
 | 
 | ||||||
|     // Process package.json
 |     // Process package.json
 | ||||||
|     pkg.version = newVersion; |     pkg.version = newVersion; | ||||||
|     pkg.scripts.setup = pkg.scripts.setup.replaceAll(oldVersion, newVersion); | 
 | ||||||
|     pkg.scripts["build-docker"] = pkg.scripts["build-docker"].replaceAll(oldVersion, newVersion); |     // Replace the version: https://regex101.com/r/hmj2Bc/1
 | ||||||
|     pkg.scripts["build-docker-alpine"] = pkg.scripts["build-docker-alpine"].replaceAll(oldVersion, newVersion); |     pkg.scripts.setup = pkg.scripts.setup.replace(/(git checkout )([^\s]+)/, `$1${newVersion}`); | ||||||
|     pkg.scripts["build-docker-debian"] = pkg.scripts["build-docker-debian"].replaceAll(oldVersion, newVersion); |  | ||||||
|     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n"); |     fs.writeFileSync("package.json", JSON.stringify(pkg, null, 4) + "\n"); | ||||||
| 
 | 
 | ||||||
|  |     // Also update package-lock.json
 | ||||||
|  |     const npm = /^win/.test(process.platform) ? "npm.cmd" : "npm"; | ||||||
|  |     childProcess.spawnSync(npm, [ "install" ]); | ||||||
|  | 
 | ||||||
|     commit(newVersion); |     commit(newVersion); | ||||||
|     tag(newVersion); |     tag(newVersion); | ||||||
| 
 | 
 | ||||||
|     updateWiki(oldVersion, newVersion); |  | ||||||
| 
 |  | ||||||
| } else { | } else { | ||||||
|     console.log("version exists"); |     console.log("version exists"); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Commit updated files | ||||||
|  |  * @param {string} version Version to update to | ||||||
|  |  */ | ||||||
| function commit(version) { | function commit(version) { | ||||||
|     let msg = "update to " + version; |     let msg = "Update to " + version; | ||||||
| 
 | 
 | ||||||
|     let res = child_process.spawnSync("git", ["commit", "-m", msg, "-a"]); |     let res = childProcess.spawnSync("git", [ "commit", "-m", msg, "-a" ]); | ||||||
|     let stdout = res.stdout.toString().trim(); |     let stdout = res.stdout.toString().trim(); | ||||||
|     console.log(stdout); |     console.log(stdout); | ||||||
| 
 | 
 | ||||||
|  | @ -49,52 +52,26 @@ function commit(version) { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Create a tag with the specified version | ||||||
|  |  * @param {string} version Tag to create | ||||||
|  |  */ | ||||||
| function tag(version) { | function tag(version) { | ||||||
|     let res = child_process.spawnSync("git", ["tag", version]); |     let res = childProcess.spawnSync("git", [ "tag", version ]); | ||||||
|     console.log(res.stdout.toString().trim()); |     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) { | function tagExists(version) { | ||||||
|     if (! version) { |     if (! version) { | ||||||
|         throw new Error("invalid version"); |         throw new Error("invalid version"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let res = child_process.spawnSync("git", ["tag", "-l", version]); |     let res = childProcess.spawnSync("git", [ "tag", "-l", version ]); | ||||||
| 
 | 
 | ||||||
|     return res.stdout.toString().trim() === version; |     return res.stdout.toString().trim() === version; | ||||||
| } | } | ||||||
| 
 |  | ||||||
| function updateWiki(oldVersion, newVersion) { |  | ||||||
|     const wikiDir = "./tmp/wiki"; |  | ||||||
|     const howToUpdateFilename = "./tmp/wiki/🆙-How-to-Update.md"; |  | ||||||
| 
 |  | ||||||
|     safeDelete(wikiDir); |  | ||||||
| 
 |  | ||||||
|     child_process.spawnSync("git", ["clone", "https://github.com/louislam/uptime-kuma.wiki.git", wikiDir]); |  | ||||||
|     let content = fs.readFileSync(howToUpdateFilename).toString(); |  | ||||||
|     content = content.replaceAll(`git checkout ${oldVersion}`, `git checkout ${newVersion}`); |  | ||||||
|     fs.writeFileSync(howToUpdateFilename, content); |  | ||||||
| 
 |  | ||||||
|     child_process.spawnSync("git", ["add", "-A"], { |  | ||||||
|         cwd: wikiDir, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     child_process.spawnSync("git", ["commit", "-m", `Update to ${newVersion} from ${oldVersion}`], { |  | ||||||
|         cwd: wikiDir, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     console.log("Pushing to Github"); |  | ||||||
|     child_process.spawnSync("git", ["push"], { |  | ||||||
|         cwd: wikiDir, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     safeDelete(wikiDir); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function safeDelete(dir) { |  | ||||||
|     if (fs.existsSync(dir)) { |  | ||||||
|         fs.rmdirSync(dir, { |  | ||||||
|             recursive: true, |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
							
								
								
									
										56
									
								
								extra/update-wiki-version.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								extra/update-wiki-version.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,56 @@ | ||||||
|  | const childProcess = require("child_process"); | ||||||
|  | const fs = require("fs"); | ||||||
|  | 
 | ||||||
|  | const newVersion = process.env.VERSION; | ||||||
|  | 
 | ||||||
|  | if (!newVersion) { | ||||||
|  |     console.log("Missing version"); | ||||||
|  |     process.exit(1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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"; | ||||||
|  | 
 | ||||||
|  |     safeDelete(wikiDir); | ||||||
|  | 
 | ||||||
|  |     childProcess.spawnSync("git", [ "clone", "https://github.com/louislam/uptime-kuma.wiki.git", wikiDir ]); | ||||||
|  |     let content = fs.readFileSync(howToUpdateFilename).toString(); | ||||||
|  | 
 | ||||||
|  |     // Replace the version: https://regex101.com/r/hmj2Bc/1
 | ||||||
|  |     content = content.replace(/(git checkout )([^\s]+)/, `$1${newVersion}`); | ||||||
|  |     fs.writeFileSync(howToUpdateFilename, content); | ||||||
|  | 
 | ||||||
|  |     childProcess.spawnSync("git", [ "add", "-A" ], { | ||||||
|  |         cwd: wikiDir, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     childProcess.spawnSync("git", [ "commit", "-m", `Update to ${newVersion}` ], { | ||||||
|  |         cwd: wikiDir, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     console.log("Pushing to Github"); | ||||||
|  |     childProcess.spawnSync("git", [ "push" ], { | ||||||
|  |         cwd: wikiDir, | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     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, { | ||||||
|  |             recursive: true, | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -159,7 +159,7 @@ fi | ||||||
|   check=$(pm2 --version) |   check=$(pm2 --version) | ||||||
|   if [ "$check" == "" ]; then |   if [ "$check" == "" ]; then | ||||||
|     "echo" "-e" "Installing PM2" |     "echo" "-e" "Installing PM2" | ||||||
|     npm install pm2 -g |     npm install pm2 -g && pm2 install pm2-logrotate | ||||||
|     pm2 startup   |     pm2 startup   | ||||||
| fi | fi | ||||||
|   mkdir -p $installPath |   mkdir -p $installPath | ||||||
|  |  | ||||||
							
								
								
									
										24971
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										24971
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										205
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										205
									
								
								package.json
									
									
									
									
									
								
							|  | @ -1,6 +1,6 @@ | ||||||
| { | { | ||||||
|     "name": "uptime-kuma", |     "name": "uptime-kuma", | ||||||
|     "version": "1.11.1", |     "version": "1.20.1", | ||||||
|     "license": "MIT", |     "license": "MIT", | ||||||
|     "repository": { |     "repository": { | ||||||
|         "type": "git", |         "type": "git", | ||||||
|  | @ -10,35 +10,37 @@ | ||||||
|         "node": "14.* || >=16.*" |         "node": "14.* || >=16.*" | ||||||
|     }, |     }, | ||||||
|     "scripts": { |     "scripts": { | ||||||
|         "install-legacy": "npm install --legacy-peer-deps", |         "install-legacy": "npm install", | ||||||
|         "update-legacy": "npm update --legacy-peer-deps", |         "update-legacy": "npm update", | ||||||
|         "lint:js": "eslint --ext \".js,.vue\" --ignore-path .gitignore .", |         "lint:js": "eslint --ext \".js,.vue\" --ignore-path .gitignore .", | ||||||
|  |         "lint-fix:js": "eslint --ext \".js,.vue\" --fix --ignore-path .gitignore .", | ||||||
|         "lint:style": "stylelint \"**/*.{vue,css,scss}\" --ignore-path .gitignore", |         "lint:style": "stylelint \"**/*.{vue,css,scss}\" --ignore-path .gitignore", | ||||||
|  |         "lint-fix:style": "stylelint \"**/*.{vue,css,scss}\" --fix --ignore-path .gitignore", | ||||||
|         "lint": "npm run lint:js && npm run lint:style", |         "lint": "npm run lint:js && npm run lint:style", | ||||||
|         "dev": "vite --host --config ./config/vite.config.js", |         "dev": "concurrently -k -r \"wait-on tcp:3000 && npm run start-server-dev \" \"npm run start-frontend-dev\"", | ||||||
|  |         "start-frontend-dev": "cross-env NODE_ENV=development vite --host --config ./config/vite.config.js", | ||||||
|         "start": "npm run start-server", |         "start": "npm run start-server", | ||||||
|         "start-server": "node server/server.js", |         "start-server": "node server/server.js", | ||||||
|         "start-server-dev": "cross-env NODE_ENV=development node server/server.js", |         "start-server-dev": "cross-env NODE_ENV=development node server/server.js", | ||||||
|         "build": "vite build --config ./config/vite.config.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", |         "test-with-build": "npm run build && npm test", | ||||||
|         "jest": "node test/prepare-jest.js && npm run jest-frontend && npm run jest-backend", |         "jest-backend": "cross-env TEST_BACKEND=1 jest --runInBand --detectOpenHandles --forceExit --config=./config/jest-backend.config.js", | ||||||
|         "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", |  | ||||||
|         "tsc": "tsc", |         "tsc": "tsc", | ||||||
|         "vite-preview-dist": "vite preview --host --config ./config/vite.config.js", |         "vite-preview-dist": "vite preview --host --config ./config/vite.config.js", | ||||||
|         "build-docker": "npm run build-docker-debian && npm run build-docker-alpine", |         "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-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-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-alpine": "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:1.11.1-alpine --target release . --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-debian": "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:1.11.1 -t louislam/uptime-kuma:debian -t louislam/uptime-kuma:1-debian -t louislam/uptime-kuma:1.11.1-debian --target release . --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-nightly": "docker buildx build -f docker/dockerfile --platform linux/amd64,linux/arm64,linux/arm/v7 -t louislam/uptime-kuma:nightly --target nightly . --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-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-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", |         "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.11.1 && npm ci --production && npm run download-dist", |         "setup": "git checkout 1.20.1 && npm ci --production && npm run download-dist", | ||||||
|         "download-dist": "node extra/download-dist.js", |         "download-dist": "node extra/download-dist.js", | ||||||
|         "update-version": "node extra/update-version.js", |  | ||||||
|         "mark-as-nightly": "node extra/mark-as-nightly.js", |         "mark-as-nightly": "node extra/mark-as-nightly.js", | ||||||
|         "reset-password": "node extra/reset-password.js", |         "reset-password": "node extra/reset-password.js", | ||||||
|         "remove-2fa": "node extra/remove-2fa.js", |         "remove-2fa": "node extra/remove-2fa.js", | ||||||
|  | @ -49,86 +51,137 @@ | ||||||
|         "test-install-script-ubuntu1604": "npm run compile-install-script && docker build --progress plain -f test/test_install_script/ubuntu1604.dockerfile .", |         "test-install-script-ubuntu1604": "npm run compile-install-script && docker build --progress plain -f test/test_install_script/ubuntu1604.dockerfile .", | ||||||
|         "test-nodejs16": "docker build --progress plain -f test/ubuntu-nodejs16.dockerfile .", |         "test-nodejs16": "docker build --progress plain -f test/ubuntu-nodejs16.dockerfile .", | ||||||
|         "simple-dns-server": "node extra/simple-dns-server.js", |         "simple-dns-server": "node extra/simple-dns-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", |         "simple-mqtt-server": "node extra/simple-mqtt-server.js", | ||||||
|         "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": "ncu -u -t patch" |         "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", | ||||||
|  |         "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" | ||||||
|     }, |     }, | ||||||
|     "dependencies": { |     "dependencies": { | ||||||
|  |         "@grpc/grpc-js": "~1.7.3", | ||||||
|  |         "@louislam/ping": "~0.4.2-mod.1", | ||||||
|  |         "@louislam/sqlite3": "15.1.2", | ||||||
|  |         "args-parser": "~1.3.0", | ||||||
|  |         "axios": "~0.27.0", | ||||||
|  |         "axios-ntlm": "1.3.0", | ||||||
|  |         "badge-maker": "~3.3.1", | ||||||
|  |         "bcryptjs": "~2.4.3", | ||||||
|  |         "bree": "~7.1.5", | ||||||
|  |         "cacheable-lookup": "~6.0.4", | ||||||
|  |         "chardet": "~1.4.0", | ||||||
|  |         "check-password-strength": "^2.0.5", | ||||||
|  |         "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.5", | ||||||
|  |         "dotenv": "~16.0.3", | ||||||
|  |         "express": "~4.17.3", | ||||||
|  |         "express-basic-auth": "~1.2.1", | ||||||
|  |         "express-static-gzip": "~2.1.7", | ||||||
|  |         "form-data": "~4.0.0", | ||||||
|  |         "gamedig": "^4.0.5", | ||||||
|  |         "http-graceful-shutdown": "~3.1.7", | ||||||
|  |         "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.13.0", | ||||||
|  |         "mqtt": "~4.3.7", | ||||||
|  |         "mssql": "~8.1.4", | ||||||
|  |         "mysql2": "~2.3.3", | ||||||
|  |         "node-cloudflared-tunnel": "~1.0.9", | ||||||
|  |         "node-radius-client": "~1.0.0", | ||||||
|  |         "nodemailer": "~6.6.5", | ||||||
|  |         "notp": "~2.0.3", | ||||||
|  |         "password-hash": "~1.2.2", | ||||||
|  |         "pg": "~8.8.0", | ||||||
|  |         "pg-connection-string": "~2.5.0", | ||||||
|  |         "prom-client": "~13.2.0", | ||||||
|  |         "prometheus-api-metrics": "~3.2.1", | ||||||
|  |         "protobufjs": "~7.1.1", | ||||||
|  |         "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", | ||||||
|  |         "tcp-ping": "~0.1.1", | ||||||
|  |         "thirty-two": "~1.0.2" | ||||||
|  |     }, | ||||||
|  |     "devDependencies": { | ||||||
|  |         "@actions/github": "~5.0.1", | ||||||
|  |         "@babel/eslint-parser": "~7.17.0", | ||||||
|  |         "@babel/preset-env": "^7.15.8", | ||||||
|         "@fortawesome/fontawesome-svg-core": "~1.2.36", |         "@fortawesome/fontawesome-svg-core": "~1.2.36", | ||||||
|         "@fortawesome/free-regular-svg-icons": "~5.15.4", |         "@fortawesome/free-regular-svg-icons": "~5.15.4", | ||||||
|         "@fortawesome/free-solid-svg-icons": "~5.15.4", |         "@fortawesome/free-solid-svg-icons": "~5.15.4", | ||||||
|         "@fortawesome/vue-fontawesome": "~3.0.0-5", |         "@fortawesome/vue-fontawesome": "~3.0.0-5", | ||||||
|         "@louislam/sqlite3": "~6.0.1", |  | ||||||
|         "@popperjs/core": "~2.10.2", |         "@popperjs/core": "~2.10.2", | ||||||
|         "args-parser": "~1.3.0", |         "@types/bootstrap": "~5.1.9", | ||||||
|         "axios": "~0.21.4", |         "@vitejs/plugin-legacy": "~2.1.0", | ||||||
|         "bcryptjs": "~2.4.3", |         "@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", |         "bootstrap": "5.1.3", | ||||||
|         "bree": "~7.1.0", |         "chart.js": "~3.6.2", | ||||||
|         "chardet": "^1.3.0", |  | ||||||
|         "chart.js": "~3.6.0", |  | ||||||
|         "chartjs-adapter-dayjs": "~1.0.0", |         "chartjs-adapter-dayjs": "~1.0.0", | ||||||
|         "check-password-strength": "^2.0.3", |         "concurrently": "^7.1.0", | ||||||
|         "command-exists": "~1.2.9", |         "core-js": "~3.26.1", | ||||||
|         "compare-versions": "~3.6.0", |         "cross-env": "~7.0.3", | ||||||
|         "dayjs": "~1.10.7", |         "cypress": "^10.1.0", | ||||||
|         "express": "~4.17.1", |         "delay": "^5.0.0", | ||||||
|         "express-basic-auth": "~1.2.0", |         "dns2": "~2.0.1", | ||||||
|         "form-data": "~4.0.0", |         "dompurify": "~2.4.3", | ||||||
|         "http-graceful-shutdown": "~3.1.5", |         "eslint": "~8.14.0", | ||||||
|         "iconv-lite": "^0.6.3", |         "eslint-plugin-vue": "~8.7.1", | ||||||
|         "jsonwebtoken": "~8.5.1", |         "favico.js": "~0.3.10", | ||||||
|         "jwt-decode": "^3.1.2", |         "jest": "~27.2.5", | ||||||
|         "limiter": "^2.1.0", |         "marked": "~4.2.5", | ||||||
|         "nodemailer": "~6.6.5", |         "node-ssh": "~13.0.1", | ||||||
|         "notp": "~2.0.3", |         "postcss-html": "~1.5.0", | ||||||
|         "password-hash": "~1.2.2", |         "postcss-rtlcss": "~3.7.2", | ||||||
|         "postcss-rtlcss": "~3.4.1", |         "postcss-scss": "~4.0.4", | ||||||
|         "postcss-scss": "~4.0.2", |         "prismjs": "~1.29.0", | ||||||
|         "prom-client": "~13.2.0", |  | ||||||
|         "prometheus-api-metrics": "~3.2.0", |  | ||||||
|         "qrcode": "~1.5.0", |         "qrcode": "~1.5.0", | ||||||
|         "redbean-node": "0.1.3", |         "rollup-plugin-visualizer": "^5.6.0", | ||||||
|         "socket.io": "~4.2.0", |         "sass": "~1.42.1", | ||||||
|         "socket.io-client": "~4.2.0", |         "stylelint": "~14.7.1", | ||||||
|         "tar": "^6.1.11", |         "stylelint-config-standard": "~25.0.0", | ||||||
|         "tcp-ping": "~0.1.1", |         "terser": "~5.15.0", | ||||||
|         "thirty-two": "~1.0.2", |  | ||||||
|         "timezones-list": "~3.0.1", |         "timezones-list": "~3.0.1", | ||||||
|  |         "typescript": "~4.4.4", | ||||||
|         "v-pagination-3": "~0.1.7", |         "v-pagination-3": "~0.1.7", | ||||||
|  |         "vite": "~3.1.0", | ||||||
|  |         "vite-plugin-compression": "^0.5.1", | ||||||
|         "vue": "next", |         "vue": "next", | ||||||
|         "vue-chart-3": "~0.5.11", |         "vue-chart-3": "3.0.9", | ||||||
|         "vue-confirm-dialog": "~1.0.2", |         "vue-confirm-dialog": "~1.0.2", | ||||||
|         "vue-contenteditable": "~3.0.4", |         "vue-contenteditable": "~3.0.4", | ||||||
|         "vue-i18n": "~9.1.9", |         "vue-i18n": "~9.2.2", | ||||||
|         "vue-image-crop-upload": "~3.0.3", |         "vue-image-crop-upload": "~3.0.3", | ||||||
|         "vue-multiselect": "~3.0.0-alpha.2", |         "vue-multiselect": "~3.0.0-alpha.2", | ||||||
|  |         "vue-prism-editor": "~2.0.0-alpha.2", | ||||||
|         "vue-qrcode": "~1.0.0", |         "vue-qrcode": "~1.0.0", | ||||||
|         "vue-router": "~4.0.12", |         "vue-router": "~4.0.14", | ||||||
|         "vue-toastification": "~2.0.0-rc.5", |         "vue-toastification": "~2.0.0-rc.5", | ||||||
|         "vuedraggable": "~4.1.0" |         "vuedraggable": "~4.1.0", | ||||||
|     }, |         "wait-on": "^6.0.1" | ||||||
|     "devDependencies": { |  | ||||||
|         "@actions/github": "~5.0.0", |  | ||||||
|         "@babel/eslint-parser": "~7.15.8", |  | ||||||
|         "@babel/preset-env": "^7.15.8", |  | ||||||
|         "@types/bootstrap": "~5.1.6", |  | ||||||
|         "@vitejs/plugin-legacy": "~1.6.3", |  | ||||||
|         "@vitejs/plugin-vue": "~1.9.4", |  | ||||||
|         "@vue/compiler-sfc": "~3.2.22", |  | ||||||
|         "babel-plugin-rewire": "~1.2.0", |  | ||||||
|         "core-js": "~3.18.3", |  | ||||||
|         "cross-env": "~7.0.3", |  | ||||||
|         "dns2": "~2.0.1", |  | ||||||
|         "eslint": "~7.32.0", |  | ||||||
|         "eslint-plugin-vue": "~7.18.0", |  | ||||||
|         "jest": "~27.2.5", |  | ||||||
|         "jest-puppeteer": "~6.0.0", |  | ||||||
|         "puppeteer": "~10.4.0", |  | ||||||
|         "sass": "~1.42.1", |  | ||||||
|         "stylelint": "~13.13.1", |  | ||||||
|         "stylelint-config-standard": "~22.0.0", |  | ||||||
|         "typescript": "~4.4.4", |  | ||||||
|         "vite": "~2.6.14" |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| Before Width: | Height: | Size: 6.4 KiB After Width: | Height: | Size: 893 B | 
|  | @ -1,8 +1,12 @@ | ||||||
| const { checkLogin } = require("./util-server"); |  | ||||||
| const { R } = require("redbean-node"); | const { R } = require("redbean-node"); | ||||||
| 
 | 
 | ||||||
| class TwoFA { | class TwoFA { | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Disable 2FA for specified user | ||||||
|  |      * @param {number} userID ID of user to disable | ||||||
|  |      * @returns {Promise<void>} | ||||||
|  |      */ | ||||||
|     static async disable2FA(userID) { |     static async disable2FA(userID) { | ||||||
|         return await R.exec("UPDATE `user` SET twofa_status = 0 WHERE id = ? ", [ |         return await R.exec("UPDATE `user` SET twofa_status = 0 WHERE id = ? ", [ | ||||||
|             userID, |             userID, | ||||||
|  |  | ||||||
|  | @ -2,16 +2,19 @@ const basicAuth = require("express-basic-auth"); | ||||||
| const passwordHash = require("./password-hash"); | const passwordHash = require("./password-hash"); | ||||||
| const { R } = require("redbean-node"); | const { R } = require("redbean-node"); | ||||||
| const { setting } = require("./util-server"); | const { setting } = require("./util-server"); | ||||||
| const { debug } = require("../src/util"); |  | ||||||
| const { loginRateLimiter } = require("./rate-limiter"); | const { loginRateLimiter } = require("./rate-limiter"); | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * |  * Login to web app | ||||||
|  * @param username : string |  * @param {string} username | ||||||
|  * @param password : string |  * @param {string} password | ||||||
|  * @returns {Promise<Bean|null>} |  * @returns {Promise<(Bean|null)>} | ||||||
|  */ |  */ | ||||||
| exports.login = async function (username, password) { | exports.login = async function (username, password) { | ||||||
|  |     if (typeof username !== "string" || typeof password !== "string") { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     let user = await R.findOne("user", " username = ? AND active = 1 ", [ |     let user = await R.findOne("user", " username = ? AND active = 1 ", [ | ||||||
|         username, |         username, | ||||||
|     ]); |     ]); | ||||||
|  | @ -30,11 +33,20 @@ exports.login = async function (username, password) { | ||||||
|     return null; |     return null; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Callback for myAuthorizer | ||||||
|  |  * @callback myAuthorizerCB | ||||||
|  |  * @param {any} err Any error encountered | ||||||
|  |  * @param {boolean} authorized Is the client authorized? | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Custom authorizer for express-basic-auth | ||||||
|  |  * @param {string} username | ||||||
|  |  * @param {string} password | ||||||
|  |  * @param {myAuthorizerCB} callback | ||||||
|  |  */ | ||||||
| function myAuthorizer(username, password, callback) { | function myAuthorizer(username, password, callback) { | ||||||
|     setting("disableAuth").then((result) => { |  | ||||||
|         if (result) { |  | ||||||
|             callback(null, true); |  | ||||||
|         } else { |  | ||||||
|     // Login Rate Limit
 |     // Login Rate Limit
 | ||||||
|     loginRateLimiter.pass(null, 0).then((pass) => { |     loginRateLimiter.pass(null, 0).then((pass) => { | ||||||
|         if (pass) { |         if (pass) { | ||||||
|  | @ -49,13 +61,26 @@ function myAuthorizer(username, password, callback) { | ||||||
|             callback(null, false); |             callback(null, false); | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
| 
 |  | ||||||
|         } |  | ||||||
|     }); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| exports.basicAuth = basicAuth({ | /** | ||||||
|  |  * 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: myAuthorizer, | ||||||
|         authorizeAsync: true, |         authorizeAsync: true, | ||||||
|         challenge: true, |         challenge: true, | ||||||
|     }); |     }); | ||||||
|  | 
 | ||||||
|  |     const disabledAuth = await setting("disableAuth"); | ||||||
|  | 
 | ||||||
|  |     if (!disabledAuth) { | ||||||
|  |         middleware(req, res, next); | ||||||
|  |     } else { | ||||||
|  |         next(); | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  |  | ||||||
							
								
								
									
										86
									
								
								server/cacheable-dns-http-agent.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								server/cacheable-dns-http-agent.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,86 @@ | ||||||
|  | const https = require("https"); | ||||||
|  | const http = require("http"); | ||||||
|  | const CacheableLookup = require("cacheable-lookup"); | ||||||
|  | const { Settings } = require("./settings"); | ||||||
|  | const { log } = require("../src/util"); | ||||||
|  | 
 | ||||||
|  | class CacheableDnsHttpAgent { | ||||||
|  | 
 | ||||||
|  |     static cacheable = new CacheableLookup(); | ||||||
|  | 
 | ||||||
|  |     static httpAgentList = {}; | ||||||
|  |     static httpsAgentList = {}; | ||||||
|  | 
 | ||||||
|  |     static enable = false; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Register/Disable cacheable to global agents | ||||||
|  |      */ | ||||||
|  |     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); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * @var {https.AgentOptions} agentOptions | ||||||
|  |      * @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); | ||||||
|  |             this.cacheable.install(this.httpsAgentList[key]); | ||||||
|  |         } | ||||||
|  |         return this.httpsAgentList[key]; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * @var {http.AgentOptions} agentOptions | ||||||
|  |      * @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); | ||||||
|  |             this.cacheable.install(this.httpAgentList[key]); | ||||||
|  |         } | ||||||
|  |         return this.httpAgentList[key]; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |     CacheableDnsHttpAgent, | ||||||
|  | }; | ||||||
|  | @ -1,11 +1,13 @@ | ||||||
| const { setSetting } = require("./util-server"); | const { setSetting, setting } = require("./util-server"); | ||||||
| const axios = require("axios"); | const axios = require("axios"); | ||||||
|  | const compareVersions = require("compare-versions"); | ||||||
| 
 | 
 | ||||||
| exports.version = require("../package.json").version; | exports.version = require("../package.json").version; | ||||||
| exports.latestVersion = null; | exports.latestVersion = null; | ||||||
| 
 | 
 | ||||||
| let interval; | let interval; | ||||||
| 
 | 
 | ||||||
|  | /** Start 48 hour check interval */ | ||||||
| exports.startInterval = () => { | exports.startInterval = () => { | ||||||
|     let check = async () => { |     let check = async () => { | ||||||
|         try { |         try { | ||||||
|  | @ -16,6 +18,19 @@ exports.startInterval = () => { | ||||||
|                 res.data.slow = "1000.0.0"; |                 res.data.slow = "1000.0.0"; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |             if (await setting("checkUpdate") === false) { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             let checkBeta = await setting("checkBeta"); | ||||||
|  | 
 | ||||||
|  |             if (checkBeta && res.data.beta) { | ||||||
|  |                 if (compareVersions.compare(res.data.beta, res.data.slow, ">")) { | ||||||
|  |                     exports.latestVersion = res.data.beta; | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             if (res.data.slow) { |             if (res.data.slow) { | ||||||
|                 exports.latestVersion = res.data.slow; |                 exports.latestVersion = res.data.slow; | ||||||
|             } |             } | ||||||
|  | @ -28,6 +43,11 @@ exports.startInterval = () => { | ||||||
|     interval = setInterval(check, 3600 * 1000 * 48); |     interval = setInterval(check, 3600 * 1000 * 48); | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Enable the check update feature | ||||||
|  |  * @param {boolean} value Should the check update feature be enabled? | ||||||
|  |  * @returns {Promise<void>} | ||||||
|  |  */ | ||||||
| exports.enableCheckUpdate = async (value) => { | exports.enableCheckUpdate = async (value) => { | ||||||
|     await setSetting("checkUpdate", value); |     await setSetting("checkUpdate", value); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -3,10 +3,17 @@ | ||||||
|  */ |  */ | ||||||
| const { TimeLogger } = require("../src/util"); | const { TimeLogger } = require("../src/util"); | ||||||
| const { R } = require("redbean-node"); | const { R } = require("redbean-node"); | ||||||
| const { io } = require("./server"); | const { UptimeKumaServer } = require("./uptime-kuma-server"); | ||||||
|  | const server = UptimeKumaServer.getInstance(); | ||||||
|  | const io = server.io; | ||||||
| const { setting } = require("./util-server"); | const { setting } = require("./util-server"); | ||||||
| const checkVersion = require("./check-version"); | const checkVersion = require("./check-version"); | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Send list of notification providers to client | ||||||
|  |  * @param {Socket} socket Socket.io socket instance | ||||||
|  |  * @returns {Promise<Bean[]>} | ||||||
|  |  */ | ||||||
| async function sendNotificationList(socket) { | async function sendNotificationList(socket) { | ||||||
|     const timeLogger = new TimeLogger(); |     const timeLogger = new TimeLogger(); | ||||||
| 
 | 
 | ||||||
|  | @ -16,7 +23,10 @@ async function sendNotificationList(socket) { | ||||||
|     ]); |     ]); | ||||||
| 
 | 
 | ||||||
|     for (let bean of list) { |     for (let bean of list) { | ||||||
|         result.push(bean.export()); |         let notificationObject = bean.export(); | ||||||
|  |         notificationObject.isDefault = (notificationObject.isDefault === 1); | ||||||
|  |         notificationObject.active = (notificationObject.active === 1); | ||||||
|  |         result.push(notificationObject); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     io.to(socket.userID).emit("notificationList", result); |     io.to(socket.userID).emit("notificationList", result); | ||||||
|  | @ -28,8 +38,11 @@ async function sendNotificationList(socket) { | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Send Heartbeat History list to socket |  * Send Heartbeat History list to socket | ||||||
|  * @param toUser  True = send to all browsers with the same user id, False = send to the current browser only |  * @param {Socket} socket Socket.io instance | ||||||
|  * @param overwrite Overwrite client-side's heartbeat list |  * @param {number} monitorID ID of monitor to send heartbeat history | ||||||
|  |  * @param {boolean} [toUser=false]  True = send to all browsers with the same user id, False = send to the current browser only | ||||||
|  |  * @param {boolean} [overwrite=false] Overwrite client-side's heartbeat list | ||||||
|  |  * @returns {Promise<void>} | ||||||
|  */ |  */ | ||||||
| async function sendHeartbeatList(socket, monitorID, toUser = false, overwrite = false) { | async function sendHeartbeatList(socket, monitorID, toUser = false, overwrite = false) { | ||||||
|     const timeLogger = new TimeLogger(); |     const timeLogger = new TimeLogger(); | ||||||
|  | @ -56,10 +69,11 @@ async function sendHeartbeatList(socket, monitorID, toUser = false, overwrite = | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Important Heart beat list (aka event list) |  * Important Heart beat list (aka event list) | ||||||
|  * @param socket |  * @param {Socket} socket Socket.io instance | ||||||
|  * @param monitorID |  * @param {number} monitorID ID of monitor to send heartbeat history | ||||||
|  * @param toUser  True = send to all browsers with the same user id, False = send to the current browser only |  * @param {boolean} [toUser=false]  True = send to all browsers with the same user id, False = send to the current browser only | ||||||
|  * @param overwrite Overwrite client-side's heartbeat list |  * @param {boolean} [overwrite=false] Overwrite client-side's heartbeat list | ||||||
|  |  * @returns {Promise<void>} | ||||||
|  */ |  */ | ||||||
| async function sendImportantHeartbeatList(socket, monitorID, toUser = false, overwrite = false) { | async function sendImportantHeartbeatList(socket, monitorID, toUser = false, overwrite = false) { | ||||||
|     const timeLogger = new TimeLogger(); |     const timeLogger = new TimeLogger(); | ||||||
|  | @ -83,18 +97,66 @@ async function sendImportantHeartbeatList(socket, monitorID, toUser = false, ove | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Emit proxy list to client | ||||||
|  |  * @param {Socket} socket Socket.io socket instance | ||||||
|  |  * @return {Promise<Bean[]>} | ||||||
|  |  */ | ||||||
|  | async function sendProxyList(socket) { | ||||||
|  |     const timeLogger = new TimeLogger(); | ||||||
|  | 
 | ||||||
|  |     const list = await R.find("proxy", " user_id = ? ", [ socket.userID ]); | ||||||
|  |     io.to(socket.userID).emit("proxyList", list.map(bean => bean.export())); | ||||||
|  | 
 | ||||||
|  |     timeLogger.print("Send Proxy List"); | ||||||
|  | 
 | ||||||
|  |     return list; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Emits the version information to the client. | ||||||
|  |  * @param {Socket} socket Socket.io socket instance | ||||||
|  |  * @returns {Promise<void>} | ||||||
|  |  */ | ||||||
| async function sendInfo(socket) { | async function sendInfo(socket) { | ||||||
|     socket.emit("info", { |     socket.emit("info", { | ||||||
|         version: checkVersion.version, |         version: checkVersion.version, | ||||||
|         latestVersion: checkVersion.latestVersion, |         latestVersion: checkVersion.latestVersion, | ||||||
|         primaryBaseURL: await setting("primaryBaseURL") |         primaryBaseURL: await setting("primaryBaseURL"), | ||||||
|  |         serverTimezone: await server.getTimezone(), | ||||||
|  |         serverTimezoneOffset: server.getTimezoneOffset(), | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /** | ||||||
|  |  * Send list of docker hosts to client | ||||||
|  |  * @param {Socket} socket Socket.io socket instance | ||||||
|  |  * @returns {Promise<Bean[]>} | ||||||
|  |  */ | ||||||
|  | async function sendDockerHostList(socket) { | ||||||
|  |     const timeLogger = new TimeLogger(); | ||||||
|  | 
 | ||||||
|  |     let result = []; | ||||||
|  |     let list = await R.find("docker_host", " user_id = ? ", [ | ||||||
|  |         socket.userID, | ||||||
|  |     ]); | ||||||
|  | 
 | ||||||
|  |     for (let bean of list) { | ||||||
|  |         result.push(bean.toJSON()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     io.to(socket.userID).emit("dockerHostList", result); | ||||||
|  | 
 | ||||||
|  |     timeLogger.print("Send Docker Host List"); | ||||||
|  | 
 | ||||||
|  |     return list; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| module.exports = { | module.exports = { | ||||||
|     sendNotificationList, |     sendNotificationList, | ||||||
|     sendImportantHeartbeatList, |     sendImportantHeartbeatList, | ||||||
|     sendHeartbeatList, |     sendHeartbeatList, | ||||||
|     sendInfo |     sendProxyList, | ||||||
|  |     sendInfo, | ||||||
|  |     sendDockerHostList | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
|  |  | ||||||
|  | @ -1,7 +1,28 @@ | ||||||
| const args = require("args-parser")(process.argv); | const args = require("args-parser")(process.argv); | ||||||
| const demoMode = args["demo"] || false; | 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 = { | module.exports = { | ||||||
|     args, |     args, | ||||||
|     demoMode |     demoMode, | ||||||
|  |     badgeConstants, | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -1,9 +1,10 @@ | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const { R } = require("redbean-node"); | const { R } = require("redbean-node"); | ||||||
| const { setSetting, setting } = require("./util-server"); | const { setSetting, setting } = require("./util-server"); | ||||||
| const { debug, sleep } = require("../src/util"); | const { log, sleep } = require("../src/util"); | ||||||
| const dayjs = require("dayjs"); | const dayjs = require("dayjs"); | ||||||
| const knex = require("knex"); | const knex = require("knex"); | ||||||
|  | const { PluginsManager } = require("./plugins-manager"); | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Database & App Data Folder |  * Database & App Data Folder | ||||||
|  | @ -53,8 +54,25 @@ class Database { | ||||||
|         "patch-2fa-invalidate-used-token.sql": true, |         "patch-2fa-invalidate-used-token.sql": true, | ||||||
|         "patch-notification_sent_history.sql": true, |         "patch-notification_sent_history.sql": true, | ||||||
|         "patch-monitor-basic-auth.sql": true, |         "patch-monitor-basic-auth.sql": true, | ||||||
|  |         "patch-add-docker-columns.sql": true, | ||||||
|  |         "patch-status-page.sql": true, | ||||||
|  |         "patch-proxy.sql": true, | ||||||
|  |         "patch-monitor-expiry-notification.sql": true, | ||||||
|  |         "patch-status-page-footer-css.sql": true, | ||||||
|  |         "patch-added-mqtt-monitor.sql": true, | ||||||
|  |         "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-add-description-monitor.sql": true, | ||||||
|     } |     }; | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * The final version should be 10 after merged tag feature |      * The final version should be 10 after merged tag feature | ||||||
|  | @ -64,9 +82,20 @@ class Database { | ||||||
| 
 | 
 | ||||||
|     static noReject = true; |     static noReject = true; | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Initialize the database | ||||||
|  |      * @param {Object} args Arguments to initialize DB with | ||||||
|  |      */ | ||||||
|     static init(args) { |     static init(args) { | ||||||
|         // Data Directory (must be end with "/")
 |         // Data Directory (must be end with "/")
 | ||||||
|         Database.dataDir = process.env.DATA_DIR || args["data-dir"] || "./data/"; |         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"; |         Database.path = Database.dataDir + "kuma.db"; | ||||||
|         if (! fs.existsSync(Database.dataDir)) { |         if (! fs.existsSync(Database.dataDir)) { | ||||||
|             fs.mkdirSync(Database.dataDir, { recursive: true }); |             fs.mkdirSync(Database.dataDir, { recursive: true }); | ||||||
|  | @ -78,10 +107,19 @@ class Database { | ||||||
|             fs.mkdirSync(Database.uploadDir, { recursive: true }); |             fs.mkdirSync(Database.uploadDir, { recursive: true }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         console.log(`Data Dir: ${Database.dataDir}`); |         log.info("db", `Data Dir: ${Database.dataDir}`); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     static async connect(testMode = false) { |     /** | ||||||
|  |      * Connect to the database | ||||||
|  |      * @param {boolean} [testMode=false] Should the connection be | ||||||
|  |      * started in test mode? | ||||||
|  |      * @param {boolean} [autoloadModels=true] Should models be | ||||||
|  |      * automatically loaded? | ||||||
|  |      * @param {boolean} [noLog=false] Should logs not be output? | ||||||
|  |      * @returns {Promise<void>} | ||||||
|  |      */ | ||||||
|  |     static async connect(testMode = false, autoloadModels = true, noLog = false) { | ||||||
|         const acquireConnectionTimeout = 120 * 1000; |         const acquireConnectionTimeout = 120 * 1000; | ||||||
| 
 | 
 | ||||||
|         const Dialect = require("knex/lib/dialects/sqlite3/index.js"); |         const Dialect = require("knex/lib/dialects/sqlite3/index.js"); | ||||||
|  | @ -111,7 +149,10 @@ class Database { | ||||||
| 
 | 
 | ||||||
|         // Auto map the model to a bean object
 |         // Auto map the model to a bean object
 | ||||||
|         R.freeze(true); |         R.freeze(true); | ||||||
|  | 
 | ||||||
|  |         if (autoloadModels) { | ||||||
|             await R.autoloadModels("./server/model"); |             await R.autoloadModels("./server/model"); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         await R.exec("PRAGMA foreign_keys = ON"); |         await R.exec("PRAGMA foreign_keys = ON"); | ||||||
|         if (testMode) { |         if (testMode) { | ||||||
|  | @ -124,12 +165,20 @@ class Database { | ||||||
|         await R.exec("PRAGMA cache_size = -12000"); |         await R.exec("PRAGMA cache_size = -12000"); | ||||||
|         await R.exec("PRAGMA auto_vacuum = FULL"); |         await R.exec("PRAGMA auto_vacuum = FULL"); | ||||||
| 
 | 
 | ||||||
|         console.log("SQLite config:"); |         // This ensures that an operating system crash or power failure will not corrupt the database.
 | ||||||
|         console.log(await R.getAll("PRAGMA journal_mode")); |         // FULL synchronous is very safe, but it is also slower.
 | ||||||
|         console.log(await R.getAll("PRAGMA cache_size")); |         // Read more: https://sqlite.org/pragma.html#pragma_synchronous
 | ||||||
|         console.log("SQLite Version: " + await R.getCell("SELECT sqlite_version()")); |         await R.exec("PRAGMA synchronous = FULL"); | ||||||
|  | 
 | ||||||
|  |         if (!noLog) { | ||||||
|  |             log.info("db", "SQLite config:"); | ||||||
|  |             log.info("db", await R.getAll("PRAGMA journal_mode")); | ||||||
|  |             log.info("db", await R.getAll("PRAGMA cache_size")); | ||||||
|  |             log.info("db", "SQLite Version: " + await R.getCell("SELECT sqlite_version()")); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** Patch the database */ | ||||||
|     static async patch() { |     static async patch() { | ||||||
|         let version = parseInt(await setting("database_version")); |         let version = parseInt(await setting("database_version")); | ||||||
| 
 | 
 | ||||||
|  | @ -137,33 +186,39 @@ class Database { | ||||||
|             version = 0; |             version = 0; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         console.info("Your database version: " + version); |         log.info("db", "Your database version: " + version); | ||||||
|         console.info("Latest database version: " + this.latestVersion); |         log.info("db", "Latest database version: " + this.latestVersion); | ||||||
| 
 | 
 | ||||||
|         if (version === this.latestVersion) { |         if (version === this.latestVersion) { | ||||||
|             console.info("Database patch not needed"); |             log.info("db", "Database patch not needed"); | ||||||
|         } else if (version > this.latestVersion) { |         } else if (version > this.latestVersion) { | ||||||
|             console.info("Warning: Database version is newer than expected"); |             log.info("db", "Warning: Database version is newer than expected"); | ||||||
|         } else { |         } else { | ||||||
|             console.info("Database patch is needed"); |             log.info("db", "Database patch is needed"); | ||||||
| 
 | 
 | ||||||
|  |             try { | ||||||
|                 this.backup(version); |                 this.backup(version); | ||||||
|  |             } catch (e) { | ||||||
|  |                 log.error("db", e); | ||||||
|  |                 log.error("db", "Unable to create a backup before patching the database. Please make sure you have enough space and permission."); | ||||||
|  |                 process.exit(1); | ||||||
|  |             } | ||||||
| 
 | 
 | ||||||
|             // Try catch anything here, if gone wrong, restore the backup
 |             // Try catch anything here, if gone wrong, restore the backup
 | ||||||
|             try { |             try { | ||||||
|                 for (let i = version + 1; i <= this.latestVersion; i++) { |                 for (let i = version + 1; i <= this.latestVersion; i++) { | ||||||
|                     const sqlFile = `./db/patch${i}.sql`; |                     const sqlFile = `./db/patch${i}.sql`; | ||||||
|                     console.info(`Patching ${sqlFile}`); |                     log.info("db", `Patching ${sqlFile}`); | ||||||
|                     await Database.importSQLFile(sqlFile); |                     await Database.importSQLFile(sqlFile); | ||||||
|                     console.info(`Patched ${sqlFile}`); |                     log.info("db", `Patched ${sqlFile}`); | ||||||
|                     await setSetting("database_version", i); |                     await setSetting("database_version", i); | ||||||
|                 } |                 } | ||||||
|             } catch (ex) { |             } catch (ex) { | ||||||
|                 await Database.close(); |                 await Database.close(); | ||||||
| 
 | 
 | ||||||
|                 console.error(ex); |                 log.error("db", ex); | ||||||
|                 console.error("Start Uptime-Kuma failed due to issue patching the database"); |                 log.error("db", "Start Uptime-Kuma failed due to issue patching the database"); | ||||||
|                 console.error("Please submit a bug report if you still encounter the problem after restart: https://github.com/louislam/uptime-kuma/issues"); |                 log.error("db", "Please submit a bug report if you still encounter the problem after restart: https://github.com/louislam/uptime-kuma/issues"); | ||||||
| 
 | 
 | ||||||
|                 this.restore(); |                 this.restore(); | ||||||
|                 process.exit(1); |                 process.exit(1); | ||||||
|  | @ -171,22 +226,25 @@ class Database { | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         await this.patch2(); |         await this.patch2(); | ||||||
|  |         await this.migrateNewStatusPage(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  |      * Patch DB using new process | ||||||
|      * Call it from patch() only |      * Call it from patch() only | ||||||
|  |      * @private | ||||||
|      * @returns {Promise<void>} |      * @returns {Promise<void>} | ||||||
|      */ |      */ | ||||||
|     static async patch2() { |     static async patch2() { | ||||||
|         console.log("Database Patch 2.0 Process"); |         log.info("db", "Database Patch 2.0 Process"); | ||||||
|         let databasePatchedFiles = await setting("databasePatchedFiles"); |         let databasePatchedFiles = await setting("databasePatchedFiles"); | ||||||
| 
 | 
 | ||||||
|         if (! databasePatchedFiles) { |         if (! databasePatchedFiles) { | ||||||
|             databasePatchedFiles = {}; |             databasePatchedFiles = {}; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         debug("Patched files:"); |         log.debug("db", "Patched files:"); | ||||||
|         debug(databasePatchedFiles); |         log.debug("db", databasePatchedFiles); | ||||||
| 
 | 
 | ||||||
|         try { |         try { | ||||||
|             for (let sqlFilename in this.patchList) { |             for (let sqlFilename in this.patchList) { | ||||||
|  | @ -194,15 +252,15 @@ class Database { | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             if (this.patched) { |             if (this.patched) { | ||||||
|                 console.log("Database Patched Successfully"); |                 log.info("db", "Database Patched Successfully"); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|         } catch (ex) { |         } catch (ex) { | ||||||
|             await Database.close(); |             await Database.close(); | ||||||
| 
 | 
 | ||||||
|             console.error(ex); |             log.error("db", ex); | ||||||
|             console.error("Start Uptime-Kuma failed due to issue patching the database"); |             log.error("db", "Start Uptime-Kuma failed due to issue patching the database"); | ||||||
|             console.error("Please submit the bug report if you still encounter the problem after restart: https://github.com/louislam/uptime-kuma/issues"); |             log.error("db", "Please submit the bug report if you still encounter the problem after restart: https://github.com/louislam/uptime-kuma/issues"); | ||||||
| 
 | 
 | ||||||
|             this.restore(); |             this.restore(); | ||||||
| 
 | 
 | ||||||
|  | @ -213,24 +271,95 @@ class Database { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|  |      * Migrate status page value in setting to "status_page" table | ||||||
|  |      * @returns {Promise<void>} | ||||||
|  |      */ | ||||||
|  |     static async migrateNewStatusPage() { | ||||||
|  | 
 | ||||||
|  |         // Fix 1.13.0 empty slug bug
 | ||||||
|  |         await R.exec("UPDATE status_page SET slug = 'empty-slug-recover' WHERE TRIM(slug) = ''"); | ||||||
|  | 
 | ||||||
|  |         let title = await setting("title"); | ||||||
|  | 
 | ||||||
|  |         if (title) { | ||||||
|  |             console.log("Migrating Status Page"); | ||||||
|  | 
 | ||||||
|  |             let statusPageCheck = await R.findOne("status_page", " slug = 'default' "); | ||||||
|  | 
 | ||||||
|  |             if (statusPageCheck !== null) { | ||||||
|  |                 console.log("Migrating Status Page - Skip, default slug record is already existing"); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             let statusPage = R.dispense("status_page"); | ||||||
|  |             statusPage.slug = "default"; | ||||||
|  |             statusPage.title = title; | ||||||
|  |             statusPage.description = await setting("description"); | ||||||
|  |             statusPage.icon = await setting("icon"); | ||||||
|  |             statusPage.theme = await setting("statusPageTheme"); | ||||||
|  |             statusPage.published = !!await setting("statusPagePublished"); | ||||||
|  |             statusPage.search_engine_index = !!await setting("searchEngineIndex"); | ||||||
|  |             statusPage.show_tags = !!await setting("statusPageTags"); | ||||||
|  |             statusPage.password = null; | ||||||
|  | 
 | ||||||
|  |             if (!statusPage.title) { | ||||||
|  |                 statusPage.title = "My Status Page"; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (!statusPage.icon) { | ||||||
|  |                 statusPage.icon = ""; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (!statusPage.theme) { | ||||||
|  |                 statusPage.theme = "light"; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             let id = await R.store(statusPage); | ||||||
|  | 
 | ||||||
|  |             await R.exec("UPDATE incident SET status_page_id = ? WHERE status_page_id IS NULL", [ | ||||||
|  |                 id | ||||||
|  |             ]); | ||||||
|  | 
 | ||||||
|  |             await R.exec("UPDATE [group] SET status_page_id = ? WHERE status_page_id IS NULL", [ | ||||||
|  |                 id | ||||||
|  |             ]); | ||||||
|  | 
 | ||||||
|  |             await R.exec("DELETE FROM setting WHERE type = 'statusPage'"); | ||||||
|  | 
 | ||||||
|  |             // Migrate Entry Page if it is status page
 | ||||||
|  |             let entryPage = await setting("entryPage"); | ||||||
|  | 
 | ||||||
|  |             if (entryPage === "statusPage") { | ||||||
|  |                 await setSetting("entryPage", "statusPage-default", "general"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             console.log("Migrating Status Page - Done"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Patch database using new patching process | ||||||
|      * Used it patch2() only |      * Used it patch2() only | ||||||
|  |      * @private | ||||||
|      * @param sqlFilename |      * @param sqlFilename | ||||||
|      * @param databasePatchedFiles |      * @param databasePatchedFiles | ||||||
|  |      * @returns {Promise<void>} | ||||||
|      */ |      */ | ||||||
|     static async patch2Recursion(sqlFilename, databasePatchedFiles) { |     static async patch2Recursion(sqlFilename, databasePatchedFiles) { | ||||||
|         let value = this.patchList[sqlFilename]; |         let value = this.patchList[sqlFilename]; | ||||||
| 
 | 
 | ||||||
|         if (! value) { |         if (! value) { | ||||||
|             console.log(sqlFilename + " skip"); |             log.info("db", sqlFilename + " skip"); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // Check if patched
 |         // Check if patched
 | ||||||
|         if (! databasePatchedFiles[sqlFilename]) { |         if (! databasePatchedFiles[sqlFilename]) { | ||||||
|             console.log(sqlFilename + " is not patched"); |             log.info("db", sqlFilename + " is not patched"); | ||||||
| 
 | 
 | ||||||
|             if (value.parents) { |             if (value.parents) { | ||||||
|                 console.log(sqlFilename + " need parents"); |                 log.info("db", sqlFilename + " need parents"); | ||||||
|                 for (let parentSQLFilename of value.parents) { |                 for (let parentSQLFilename of value.parents) { | ||||||
|                     await this.patch2Recursion(parentSQLFilename, databasePatchedFiles); |                     await this.patch2Recursion(parentSQLFilename, databasePatchedFiles); | ||||||
|                 } |                 } | ||||||
|  | @ -238,24 +367,24 @@ class Database { | ||||||
| 
 | 
 | ||||||
|             this.backup(dayjs().format("YYYYMMDDHHmmss")); |             this.backup(dayjs().format("YYYYMMDDHHmmss")); | ||||||
| 
 | 
 | ||||||
|             console.log(sqlFilename + " is patching"); |             log.info("db", sqlFilename + " is patching"); | ||||||
|             this.patched = true; |             this.patched = true; | ||||||
|             await this.importSQLFile("./db/" + sqlFilename); |             await this.importSQLFile("./db/" + sqlFilename); | ||||||
|             databasePatchedFiles[sqlFilename] = true; |             databasePatchedFiles[sqlFilename] = true; | ||||||
|             console.log(sqlFilename + " was patched successfully"); |             log.info("db", sqlFilename + " was patched successfully"); | ||||||
| 
 | 
 | ||||||
|         } else { |         } else { | ||||||
|             debug(sqlFilename + " is already patched, skip"); |             log.debug("db", sqlFilename + " is already patched, skip"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Sadly, multi sql statements is not supported by many sqlite libraries, I have to implement it myself |      * Load an SQL file and execute it | ||||||
|      * @param filename |      * @param filename Filename of SQL file to import | ||||||
|      * @returns {Promise<void>} |      * @returns {Promise<void>} | ||||||
|      */ |      */ | ||||||
|     static async importSQLFile(filename) { |     static async importSQLFile(filename) { | ||||||
| 
 |         // Sadly, multi sql statements is not supported by many sqlite libraries, I have to implement it myself
 | ||||||
|         await R.getCell("SELECT 1"); |         await R.getCell("SELECT 1"); | ||||||
| 
 | 
 | ||||||
|         let text = fs.readFileSync(filename).toString(); |         let text = fs.readFileSync(filename).toString(); | ||||||
|  | @ -283,6 +412,10 @@ class Database { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Aquire a direct connection to database | ||||||
|  |      * @returns {any} | ||||||
|  |      */ | ||||||
|     static getBetterSQLite3Database() { |     static getBetterSQLite3Database() { | ||||||
|         return R.knex.client.acquireConnection(); |         return R.knex.client.acquireConnection(); | ||||||
|     } |     } | ||||||
|  | @ -297,7 +430,7 @@ class Database { | ||||||
|         }; |         }; | ||||||
|         process.addListener("unhandledRejection", listener); |         process.addListener("unhandledRejection", listener); | ||||||
| 
 | 
 | ||||||
|         console.log("Closing the database"); |         log.info("db", "Closing the database"); | ||||||
| 
 | 
 | ||||||
|         while (true) { |         while (true) { | ||||||
|             Database.noReject = true; |             Database.noReject = true; | ||||||
|  | @ -307,10 +440,10 @@ class Database { | ||||||
|             if (Database.noReject) { |             if (Database.noReject) { | ||||||
|                 break; |                 break; | ||||||
|             } else { |             } else { | ||||||
|                 console.log("Waiting to close the database"); |                 log.info("db", "Waiting to close the database"); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         console.log("SQLite closed"); |         log.info("db", "SQLite closed"); | ||||||
| 
 | 
 | ||||||
|         process.removeListener("unhandledRejection", listener); |         process.removeListener("unhandledRejection", listener); | ||||||
|     } |     } | ||||||
|  | @ -318,11 +451,11 @@ class Database { | ||||||
|     /** |     /** | ||||||
|      * One backup one time in this process. |      * One backup one time in this process. | ||||||
|      * Reset this.backupPath if you want to backup again |      * Reset this.backupPath if you want to backup again | ||||||
|      * @param version |      * @param {string} version Version code of backup | ||||||
|      */ |      */ | ||||||
|     static backup(version) { |     static backup(version) { | ||||||
|         if (! this.backupPath) { |         if (! this.backupPath) { | ||||||
|             console.info("Backing up the database"); |             log.info("db", "Backing up the database"); | ||||||
|             this.backupPath = this.dataDir + "kuma.db.bak" + version; |             this.backupPath = this.dataDir + "kuma.db.bak" + version; | ||||||
|             fs.copyFileSync(Database.path, this.backupPath); |             fs.copyFileSync(Database.path, this.backupPath); | ||||||
| 
 | 
 | ||||||
|  | @ -337,19 +470,44 @@ class Database { | ||||||
|                 this.backupWalPath = walPath + ".bak" + version; |                 this.backupWalPath = walPath + ".bak" + version; | ||||||
|                 fs.copyFileSync(walPath, this.backupWalPath); |                 fs.copyFileSync(walPath, this.backupWalPath); | ||||||
|             } |             } | ||||||
|  | 
 | ||||||
|  |             // Double confirm if all files actually backup
 | ||||||
|  |             if (!fs.existsSync(this.backupPath)) { | ||||||
|  |                 throw new Error("Backup failed! " + this.backupPath); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (fs.existsSync(shmPath)) { | ||||||
|  |                 if (!fs.existsSync(this.backupShmPath)) { | ||||||
|  |                     throw new Error("Backup failed! " + this.backupShmPath); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|     /** |             if (fs.existsSync(walPath)) { | ||||||
|      * |                 if (!fs.existsSync(this.backupWalPath)) { | ||||||
|      */ |                     throw new Error("Backup failed! " + this.backupWalPath); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** Restore from most recent backup */ | ||||||
|     static restore() { |     static restore() { | ||||||
|         if (this.backupPath) { |         if (this.backupPath) { | ||||||
|             console.error("Patching the database failed!!! Restoring the backup"); |             log.error("db", "Patching the database failed!!! Restoring the backup"); | ||||||
| 
 | 
 | ||||||
|             const shmPath = Database.path + "-shm"; |             const shmPath = Database.path + "-shm"; | ||||||
|             const walPath = Database.path + "-wal"; |             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
 |             // Delete patch failed db
 | ||||||
|             try { |             try { | ||||||
|                 if (fs.existsSync(Database.path)) { |                 if (fs.existsSync(Database.path)) { | ||||||
|  | @ -364,7 +522,7 @@ class Database { | ||||||
|                     fs.unlinkSync(walPath); |                     fs.unlinkSync(walPath); | ||||||
|                 } |                 } | ||||||
|             } catch (e) { |             } catch (e) { | ||||||
|                 console.log("Restore failed; you may need to restore the backup manually"); |                 log.error("db", "Restore failed; you may need to restore the backup manually"); | ||||||
|                 process.exit(1); |                 process.exit(1); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | @ -380,17 +538,22 @@ class Database { | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|         } else { |         } else { | ||||||
|             console.log("Nothing to restore"); |             log.info("db", "Nothing to restore"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** Get the size of the database */ | ||||||
|     static getSize() { |     static getSize() { | ||||||
|         debug("Database.getSize()"); |         log.debug("db", "Database.getSize()"); | ||||||
|         let stats = fs.statSync(Database.path); |         let stats = fs.statSync(Database.path); | ||||||
|         debug(stats); |         log.debug("db", stats); | ||||||
|         return stats.size; |         return stats.size; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Shrink the database | ||||||
|  |      * @returns {Promise<void>} | ||||||
|  |      */ | ||||||
|     static async shrink() { |     static async shrink() { | ||||||
|         await R.exec("VACUUM"); |         await R.exec("VACUUM"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
							
								
								
									
										118
									
								
								server/docker.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								server/docker.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,118 @@ | ||||||
|  | const axios = require("axios"); | ||||||
|  | const { R } = require("redbean-node"); | ||||||
|  | const version = require("../package.json").version; | ||||||
|  | const https = require("https"); | ||||||
|  | 
 | ||||||
|  | class DockerHost { | ||||||
|  |     /** | ||||||
|  |      * Save a docker host | ||||||
|  |      * @param {Object} dockerHost Docker host to save | ||||||
|  |      * @param {?number} dockerHostID ID of the docker host to update | ||||||
|  |      * @param {number} userID ID of the user who adds the docker host | ||||||
|  |      * @returns {Promise<Bean>} | ||||||
|  |      */ | ||||||
|  |     static async save(dockerHost, dockerHostID, userID) { | ||||||
|  |         let bean; | ||||||
|  | 
 | ||||||
|  |         if (dockerHostID) { | ||||||
|  |             bean = await R.findOne("docker_host", " id = ? AND user_id = ? ", [ dockerHostID, userID ]); | ||||||
|  | 
 | ||||||
|  |             if (!bean) { | ||||||
|  |                 throw new Error("docker host not found"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  |             bean = R.dispense("docker_host"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         bean.user_id = userID; | ||||||
|  |         bean.docker_daemon = dockerHost.dockerDaemon; | ||||||
|  |         bean.docker_type = dockerHost.dockerType; | ||||||
|  |         bean.name = dockerHost.name; | ||||||
|  | 
 | ||||||
|  |         await R.store(bean); | ||||||
|  | 
 | ||||||
|  |         return bean; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Delete a Docker host | ||||||
|  |      * @param {number} dockerHostID ID of the Docker host to delete | ||||||
|  |      * @param {number} userID ID of the user who created the Docker host | ||||||
|  |      * @returns {Promise<void>} | ||||||
|  |      */ | ||||||
|  |     static async delete(dockerHostID, userID) { | ||||||
|  |         let bean = await R.findOne("docker_host", " id = ? AND user_id = ? ", [ dockerHostID, userID ]); | ||||||
|  | 
 | ||||||
|  |         if (!bean) { | ||||||
|  |             throw new Error("docker host not found"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Delete removed proxy from monitors if exists
 | ||||||
|  |         await R.exec("UPDATE monitor SET docker_host = null WHERE docker_host = ?", [ dockerHostID ]); | ||||||
|  | 
 | ||||||
|  |         await R.trash(bean); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Fetches the amount of containers on the Docker host | ||||||
|  |      * @param {Object} dockerHost Docker host to check for | ||||||
|  |      * @returns {number} Total amount of containers on the host | ||||||
|  |      */ | ||||||
|  |     static async testDockerHost(dockerHost) { | ||||||
|  |         const options = { | ||||||
|  |             url: "/containers/json?all=true", | ||||||
|  |             headers: { | ||||||
|  |                 "Accept": "*/*", | ||||||
|  |                 "User-Agent": "Uptime-Kuma/" + version | ||||||
|  |             }, | ||||||
|  |             httpsAgent: new https.Agent({ | ||||||
|  |                 maxCachedSessions: 0,      // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940)
 | ||||||
|  |                 rejectUnauthorized: false, | ||||||
|  |             }), | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         if (dockerHost.dockerType === "socket") { | ||||||
|  |             options.socketPath = dockerHost.dockerDaemon; | ||||||
|  |         } else if (dockerHost.dockerType === "tcp") { | ||||||
|  |             options.baseURL = DockerHost.patchDockerURL(dockerHost.dockerDaemon); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let res = await axios.request(options); | ||||||
|  | 
 | ||||||
|  |         if (Array.isArray(res.data)) { | ||||||
|  | 
 | ||||||
|  |             if (res.data.length > 1) { | ||||||
|  | 
 | ||||||
|  |                 if ("ImageID" in res.data[0]) { | ||||||
|  |                     return res.data.length; | ||||||
|  |                 } else { | ||||||
|  |                     throw new Error("Invalid Docker response, is it Docker really a daemon?"); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |             } else { | ||||||
|  |                 return res.data.length; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  |             throw new Error("Invalid Docker response, is it Docker really a daemon?"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Since axios 0.27.X, it does not accept `tcp://` protocol. | ||||||
|  |      * Change it to `http://` on the fly in order to fix it. (https://github.com/louislam/uptime-kuma/issues/2165)
 | ||||||
|  |      */ | ||||||
|  |     static patchDockerURL(url) { | ||||||
|  |         if (typeof url === "string") { | ||||||
|  |             // Replace the first occurrence only with g
 | ||||||
|  |             return url.replace(/tcp:\/\//g, "http://"); | ||||||
|  |         } | ||||||
|  |         return url; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | module.exports = { | ||||||
|  |     DockerHost, | ||||||
|  | }; | ||||||
							
								
								
									
										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, | ||||||
|  | }; | ||||||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
		Reference in a new issue