Merge pull request #86 from Spiritreader/master
Implement retries (#56)
This commit is contained in:
		
						commit
						248b5292dc
					
				
					 13 changed files with 81 additions and 8 deletions
				
			
		|  | @ -11,3 +11,4 @@ | |||
| LICENSE | ||||
| README.md | ||||
| .editorconfig | ||||
| .vscode | ||||
|  |  | |||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							|  | @ -7,3 +7,4 @@ dist-ssr | |||
| 
 | ||||
| /data | ||||
| !/data/.gitkeep | ||||
| .vscode | ||||
							
								
								
									
										
											BIN
										
									
								
								db/kuma.db
									
									
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								db/kuma.db
									
									
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										37
									
								
								db/patch3.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								db/patch3.sql
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,37 @@ | |||
| -- You should not modify if this have pushed to Github, unless it does serious wrong with the db. | ||||
| -- Add maxretries column to monitor | ||||
| PRAGMA foreign_keys=off; | ||||
| 
 | ||||
| BEGIN TRANSACTION; | ||||
| 
 | ||||
| create table monitor_dg_tmp | ||||
| ( | ||||
|     id INTEGER not null | ||||
|         primary key autoincrement, | ||||
|     name VARCHAR(150), | ||||
|     active BOOLEAN default 1 not null, | ||||
|     user_id INTEGER | ||||
|         references user | ||||
|                    on update cascade on delete set null, | ||||
|     interval INTEGER default 20 not null, | ||||
|     url TEXT, | ||||
|     type VARCHAR(20), | ||||
|     weight INTEGER default 2000, | ||||
|     hostname VARCHAR(255), | ||||
|     port INTEGER, | ||||
|     created_date DATETIME, | ||||
|     keyword VARCHAR(255), | ||||
|     maxretries INTEGER NOT NULL DEFAULT 0 | ||||
| ); | ||||
| 
 | ||||
| insert into monitor_dg_tmp(id, name, active, user_id, interval, url, type, weight, hostname, port, created_date, keyword) select id, name, active, user_id, interval, url, type, weight, hostname, port, created_date, keyword from monitor; | ||||
| 
 | ||||
| drop table monitor; | ||||
| 
 | ||||
| alter table monitor_dg_tmp rename to monitor; | ||||
| 
 | ||||
| create index user_id on monitor (user_id); | ||||
| 
 | ||||
| COMMIT; | ||||
| 
 | ||||
| PRAGMA foreign_keys=on; | ||||
|  | @ -16,7 +16,6 @@ const {Notification} = require("../notification") | |||
|  *      1 = UP | ||||
|  */ | ||||
| class Monitor extends BeanModel { | ||||
| 
 | ||||
|     async toJSON() { | ||||
| 
 | ||||
|         let notificationIDList = {}; | ||||
|  | @ -35,6 +34,7 @@ class Monitor extends BeanModel { | |||
|             url: this.url, | ||||
|             hostname: this.hostname, | ||||
|             port: this.port, | ||||
|             maxretries: this.maxretries, | ||||
|             weight: this.weight, | ||||
|             active: this.active, | ||||
|             type: this.type, | ||||
|  | @ -46,6 +46,7 @@ class Monitor extends BeanModel { | |||
| 
 | ||||
|     start(io) { | ||||
|         let previousBeat = null; | ||||
|         let retries = 0; | ||||
| 
 | ||||
|         const beat = async () => { | ||||
|             if (! previousBeat) { | ||||
|  | @ -107,12 +108,19 @@ class Monitor extends BeanModel { | |||
|                     bean.status = 1; | ||||
|                 } | ||||
| 
 | ||||
|                 retries = 0; | ||||
| 
 | ||||
|             } catch (error) { | ||||
|                 if ((this.maxretries > 0) && (retries < this.maxretries)) { | ||||
|                     retries++; | ||||
|                     bean.status = 2; | ||||
|                 } | ||||
|                 bean.msg = error.message; | ||||
|             } | ||||
| 
 | ||||
|             // Mark as important if status changed
 | ||||
|             if (! previousBeat || previousBeat.status !== bean.status) { | ||||
|             // Mark as important if status changed, ignore pending pings,
 | ||||
|             // Don't notify if disrupted changes to up
 | ||||
|             if ((! previousBeat || previousBeat.status !== bean.status) && bean.status !== 2 && !(previousBeat.status === 2 && bean.status !== 0)) { | ||||
|                 bean.important = true; | ||||
| 
 | ||||
|                 // Do not send if first beat is UP
 | ||||
|  | @ -237,7 +245,7 @@ class Monitor extends BeanModel { | |||
|                 } | ||||
| 
 | ||||
|                 total += value; | ||||
|                 if (row.status === 0) { | ||||
|                 if (row.status === 0 || row.status === 2) { | ||||
|                     downtime += value; | ||||
|                 } | ||||
|             } | ||||
|  |  | |||
|  | @ -233,6 +233,7 @@ let needSetup = false; | |||
|                 bean.url = monitor.url | ||||
|                 bean.interval = monitor.interval | ||||
|                 bean.hostname = monitor.hostname; | ||||
|                 bean.maxretries = monitor.maxretries; | ||||
|                 bean.port = monitor.port; | ||||
|                 bean.keyword = monitor.keyword; | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,7 +1,8 @@ | |||
| $primary: #5CDD8B; | ||||
| $danger: #DC3545; | ||||
| $warning: #f8a306; | ||||
| $link-color: #111; | ||||
| $border-radius: 50rem; | ||||
| 
 | ||||
| $highlight: #7ce8a4; | ||||
| $highlight-white: #e7faec; | ||||
| $highlight-white: #e7faec; | ||||
|  | @ -3,7 +3,7 @@ | |||
|         <div class="hp-bar-big" :style="barStyle"> | ||||
|             <div | ||||
|                 class="beat" | ||||
|                 :class="{ 'empty' : (beat === 0), 'down' : (beat.status === 0) }" | ||||
|                 :class="{ 'empty' : (beat === 0), 'down' : (beat.status === 0), 'pending' : (beat.status === 2) }" | ||||
|                 :style="beatStyle" | ||||
|                 v-for="(beat, index) in shortBeatList" | ||||
|                 :key="index" | ||||
|  | @ -166,6 +166,10 @@ export default { | |||
|             background-color: $danger; | ||||
|         } | ||||
| 
 | ||||
|         &.pending { | ||||
|             background-color: $warning; | ||||
|         } | ||||
| 
 | ||||
|         &:not(.empty):hover { | ||||
|             transition: all ease-in-out 0.15s; | ||||
|             opacity: 0.8; | ||||
|  |  | |||
|  | @ -14,6 +14,8 @@ export default { | |||
|                 return "danger" | ||||
|             } else if (this.status === 1) { | ||||
|                 return "primary" | ||||
|             } else if (this.status === 2) { | ||||
|                 return "warning" | ||||
|             } else { | ||||
|                 return "secondary" | ||||
|             } | ||||
|  | @ -24,6 +26,8 @@ export default { | |||
|                 return "Down" | ||||
|             } else if (this.status === 1) { | ||||
|                 return "Up" | ||||
|             } else if (this.status === 2) { | ||||
|                 return "Pending" | ||||
|             } else { | ||||
|                 return "Unknown" | ||||
|             } | ||||
|  | @ -34,6 +38,6 @@ export default { | |||
| 
 | ||||
| <style scoped> | ||||
|     span { | ||||
|         width: 45px; | ||||
|         width: 54px; | ||||
|     } | ||||
| </style> | ||||
|  |  | |||
|  | @ -30,6 +30,8 @@ export default { | |||
|                 return "danger" | ||||
|             } else if (this.lastHeartBeat.status === 1) { | ||||
|                 return "primary" | ||||
|             } else if (this.lastHeartBeat.status === 2) { | ||||
|                 return "warning" | ||||
|             } else { | ||||
|                 return "secondary" | ||||
|             } | ||||
|  |  | |||
|  | @ -279,6 +279,11 @@ export default { | |||
|                         text: "Down", | ||||
|                         color: "danger" | ||||
|                     }; | ||||
|                 } else if (lastHeartBeat.status === 2) { | ||||
|                     result[monitorID] = { | ||||
|                         text: "Pending", | ||||
|                         color: "warning" | ||||
|                     }; | ||||
|                 } else { | ||||
|                     result[monitorID] = unknown; | ||||
|                 } | ||||
|  |  | |||
|  | @ -110,6 +110,8 @@ export default { | |||
|                         result.up++; | ||||
|                     } else if (beat.status === 0) { | ||||
|                         result.down++; | ||||
|                     } else if (beat.status === 2) { | ||||
|                         result.up++;                 | ||||
|                     } else { | ||||
|                         result.unknown++; | ||||
|                     } | ||||
|  |  | |||
|  | @ -48,6 +48,12 @@ | |||
|                         <input type="number" class="form-control" id="interval" v-model="monitor.interval" required min="20" step="1"> | ||||
|                     </div> | ||||
| 
 | ||||
|                     <div class="mb-3"> | ||||
|                         <label for="maxRetries" class="form-label">Retries</label> | ||||
|                         <input type="number" class="form-control" id="maxRetries" v-model="monitor.maxretries" required min="0" step="1"> | ||||
|                         <div class="form-text">Maximum retries before the service is marked as down and a notification is sent</div> | ||||
|                     </div> | ||||
| 
 | ||||
|                     <div> | ||||
|                         <button class="btn btn-primary" type="submit" :disabled="processing">Save</button> | ||||
|                     </div> | ||||
|  | @ -61,7 +67,7 @@ | |||
|                 <h2>Notifications</h2> | ||||
|                 <p v-if="$root.notificationList.length === 0">Not available, please setup.</p> | ||||
| 
 | ||||
|                 <div class="form-check form-switch mb-3" v-for="(notification, index) in $root.notificationList" :key="index"> | ||||
|                 <div class="form-check form-switch mb-3" :key="notification.id" v-for="notification in $root.notificationList"> | ||||
|                     <input class="form-check-input" type="checkbox" :id=" 'notification' + notification.id" v-model="monitor.notificationIDList[notification.id]"> | ||||
| 
 | ||||
|                     <label class="form-check-label" :for=" 'notification' + notification.id"> | ||||
|  | @ -119,6 +125,7 @@ export default { | |||
|                     name: "", | ||||
|                     url: "https://", | ||||
|                     interval: 60, | ||||
|                     maxretries: 0, | ||||
|                     notificationIDList: {}, | ||||
|                 } | ||||
|             } else if (this.isEdit) { | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue