| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
 | "use strict";
/* Basic editor for switches, and possibly networks and whatever.
 * This is the first real use of both the nmsBox and nmsType, so
 * expect changes as the need(s) arise.
 * 
 * The general idea is simple, though: Editing and adding is to be treated
 * as similar as possible, and there should be no hard-coding anywhere. If
 * we need a "one-off" for whatever, we should find a genric way of solving
 * it to avoid complicating things. 
 * 
 */
class nmsNewSwitch extends nmsPanel {
	constructor() {
		super("Add new switch")
		this.add(new nmsModSwitch(undefined))
	}
}
class nmsNewNet extends nmsPanel {
	constructor() {
		super("Add new network")
		this.add(new nmsModNet(undefined))
	}
}
class nmsModThing extends nmsBox {
	constructor(data) {
		super("div",{html:{className: "panel-body"}});
		this.identifier = data.identifier;
		this.invalidate = data.invalidate;
		this.api = data.api;
		this.item = data.item;
		this.generateBaseTemplate()
		this.populate()
		var save = new nmsButton("Save","btn-primary");
		save.panel = this;
		save.html.onclick = this.save;
		this.add(save)
		var del = new nmsButton("Delete","btn btn-danger");
		del.panel = this
		del.html.onclick = this.del;
		this.add(del)
	}
	commit(data) {
		$.ajax({
			type: "POST", 
			url: this.api,
			dataType: "text",
			nmsBox:this,
			data:JSON.stringify(data),
			success: function (data, textStatus, jqXHR) {
				var msg = new nmsString("Changed...");
				msg.attach(this.nmsBox._root);
				msg.show()
				this.nmsBox.destroy()
				//nmsInfoBox.hide();
				for (var x of this.nmsBox.invalidate) {
					nmsData.invalidate(x);
				}
			}
		});
	}
	del(e) {
		if(confirm("This will delete the " + this.typeName + ":" + this.nmsBox.panel.item)) {
			this.nmsBox.panel.commit([{'sysname': this.nmsBox.panel.item, 'deleted': true}]); 
		};
	}
	save(e) { 
		var diff = this.nmsBox.panel.diff()
		if (diff != undefined) {
			this.nmsBox.panel.commit([diff])
		}
	}
	/* Pretty sure that the type-thing is OK, but what I want is to
	 * generate a nmsTemplate or something that can be used to get/set
	 * variables generically, and replaces nmsEditRow. Since right now,
	 * both the template and the row is fiddling with values, luckily
	 * all through the same actual object, but still....
	 * This is because I wrote nmsEditRow before I added a type-system.
	 *
	 * The fundamental problem is that the row-rendering is obviously
	 * affected by the type, and the overall "template"/parent
	 * (nmsModSwitch right now) is also obviously affected by changes to
	 * the individual rows.
	 *
	 * Right now a change in a value means nmsEditRow will get the
	 * event, it will use the nmsType to validate, and ultimately set,
	 * but it ALSO has to store a text-representation of the value if it
	 * changes from other sources (e.g.: auto-complete), and we need to
	 * alert nmsModSwitch that a change has occurred so it can act
	 * approrpiately (e.g.: Enabling/disabling a save button).
	 * 
	 * This means that nmsType instances, nmsEditRow instances and
	 * nmsModSwitch instance is tightly coupled in non-obvious ways.
	 * 
	 * Which means bugs.
	 */
	populate() {
		if (this.item != undefined) {
			this._populateTemplate(this.item);
		}
		this.table = new nmsTable();
		this.rows = {}
		for (var v in this._template) {
			this.rows[v] = new nmsEditRow(v, this._template[v]);
			this.rows[v].parent = this;
			this.table.add(this.rows[v]);
		}
		this.add(this.table);
	}
	changed(row) {
		this.title = "saw row change on " + row.name + " to " + row.value;
	}
	get value() {
		var ret = {};
		for (var idx in this.rows) {
			ret[idx] = this.rows[idx].value;
		}
		return ret;
	}
	diff() {
		var ret = {};
		var changed = 0;
		for (var idx in this.rows) {
			if (this.rows[idx].value != this.rows[idx].original) {
				ret[idx] = this.rows[idx].value;
				changed++;
			}
		}
		if (!changed) {
			return undefined;
		}
		ret[this.identifier] = this.rows[this.identifier].value;
		return ret;
	}
}
class nmsEditRow extends nmsBox {
	constructor(text,value) {
		super("tr")
		// This should/could be smarter in the future.
		console.assert(value instanceof nmsType)
		this.name = text;
		this._value = value;
		this.original = value.value;
		var td1 = new nmsBox("td")
		var name = new nmsString(text+" ");
		name.html.title = value.description;
		td1.add(name)
		this.add(td1);
		td1.html.width="50%"
		this._state = new nmsBox("span",{html:{className:"label label-default",textContent:"Original"}})
		this._valid = new nmsBox("span",{html:{className:"label label-default",textContent:"Not verified"}})
		name.add(this._state)
		name.add(this._valid)
		this._valid.hide()
		this.changed(false)
		var content = new nmsBox("td")
		var input = new nmsBox("input")
		input.html.value = value.value;
		input.html.className = "form-control";
		input.html.type = "text";
		input.row = this;
		if (value.ro) {
			input.html.disabled = true;
			input.html.title = "Read/only attribute";
		}
		if (value instanceof nmsTypeSecret) {
			input.html.type = "password"
			input.html.autocomplete = "off"
			input.html.onfocus = function f() { this.type = "text" }
			input.html.oninput = function f() { this.type = "text" }
			input.html.onblur = function f() { this.type = "password" }
		}
		input.html.onchange = function() {
			this.nmsBox.row.value = this.value
		}
		input.html.oninput = function() {
			this.nmsBox.row.value = this.value
		}
		this._input = input;
		this._content = content;
		content.add(input)
		this.add(content)
	}
	get value() {
		return this._value.value;
	}
	changed(val) {
		if (val) {
			this._state.show()
			this._state.html.textContent = "Changed"
			this._state.html.classList.remove("label-default")
			this._state.html.classList.add("label-warning")
		} else {
			this._state.hide()
		}
	}
	valid(val) {
		this._valid.html.classList.remove("label-default")
		this._valid.show()
		if (val) {
			this._valid.html.textContent = "Valid"
			this._valid.html.classList.remove("label-danger")
			this._valid.html.classList.add("label-success")
		} else {
			this._valid.html.textContent = "Invalid"
			this._valid.html.classList.add("label-danger")
			this._valid.html.classList.remove("label-success")
		}
	}
	/* THIS IS A MESS */
	set value(value) {
		if (!this._value.validate(value)) {
			this.valid(false)
			this._content.html.classList.add("has-error");
			return;
		} else {
			this.valid(true)
			this._content.html.classList.remove("has-error");
			this._value.value = value;
		}
		if (this._input.html.value != this._value.value) {
			this._input.html.value = this._value.value
		}
		if (this._value.value != this.original) {
			this.changed(true)
			this._content.html.classList.add("has-success");
		} else {
			this.changed(false)
			this._content.html.classList.remove("has-success");
		}
		this.parent.changed(this) 
	}
}
class nmsModSwitch extends nmsModThing {
	constructor(sw) {
		super({item: sw, identifier: "sysname", invalidate: ["switches","smanagement"], api: "/api/write/switches"})	
	}
	generateBaseTemplate() {
		this._template = {
			sysname: new nmsTypeSysname("Unique systemname/switch name. Only required field.  Read/only on existing equipment." ),
			mgmt_v4_addr: new nmsTypeIP("Management address IPv4"),
			mgmt_v6_addr: new nmsTypeIP("Management address IPv6"),
			mgmt_vlan: new nmsTypeNetwork("Management VLAN"),
			traffic_vlan: new nmsTypeNetwork("Traffic VLAN"),
			distro_name: new nmsTypeSysnameReference("Distro switch upstream of this system. Required for provisioning."),
			distro_phy_port: new nmsTypePort("Name of port we connect to at the distro switch. Used for provisioning, among other things."),
			poll_frequency: new nmsTypeInterval("Poll frequency for SNMP (will use default from backend)"),
			community: new nmsTypeSecret("SNMP community (will use default from backend)"),
			placement: new nmsTypePlace("Map placement (If following a regular naming scheme, the backend will place it poperly, otherwise a random place will be chose)"),
			tags: new nmsTypeTags("Additional tags in JSON text array format. Can be anything. Used to provide a simple escape hatch mechanism to tag systems.")
	      }
	}
	_populateTemplate(sw) {
		var swi = [];
		var swm = [];
		try {
			swi = nmsData.switches["switches"][sw];
			swm = nmsData.smanagement.switches[sw];
		} catch(e) {}
		var template = {}
		for (var v in swi) {
			console.assert(this._template[v] instanceof nmsType)
			if (swi[v] != null) {
				this._template[v].initial(swi[v]);
			}
		}
		for (var v in swm) {
			if (v == "last_updated") {
				continue;
			}
			console.assert(this._template[v] instanceof nmsType)
			if (swm[v] != null) {
				this._template[v].initial(swm[v]);
			}
		}
	}
}
class nmsModNet extends nmsModThing {
	constructor(net) {
		super({item: net, identifier: "name", invalidate: ["networks","smanagement"], api: "/api/write/networks"})	
	}
	generateBaseTemplate() {
		this._template = {
			name: new nmsTypeNetwork("Unique networkname. Only required field. Read/only on existing nets."),
			gw4: new nmsTypeIP("Gateway address, IPv4"),
			gw6: new nmsTypeIP("Gateway address, IPv6"),
			subnet4: new nmsTypeIP("Subnet, IPv4"),
			subnet6: new nmsTypeIP("Subnet, IPv6"),
			router: new nmsTypeSysnameReference("Router where net is terminated. E.g.: r1.noc for floor traffic nets"),
			tags: new nmsTypeTags("Additional tags in JSON text array format. Can be anything. Used to provide a simple escape hatch mechanism to tag systems.")
	      }
	}
	_populateTemplate(net) {
		var nets = [];
		try {
			nets = nmsData.networks["networks"][net];
		} catch(e) {}
		var template = {}
		for (var v in nets) {
			console.assert(this._template[v] instanceof nmsType)
			if (nets[v] != null) {
				this._template[v].initial(nets[v]);
			}
		}
	}
}
 |