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
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
|
/*
* Copyright (c) 2020 Tano Systems LLC. All Rights Reserved.
* Author: Anton Kikin <a.kikin@tano-systems.com>
* Copyright (c) 2023-2024. All Rights Reserved.
* Paul Donald <newtwen+github@gmail.com>
*/
'use strict';
'require rpc';
'require form';
'require lldpd';
'require network';
'require uci';
'require tools.widgets as widgets';
var callInitList = rpc.declare({
object: 'luci',
method: 'getInitList',
params: [ 'name' ],
expect: { '': {} },
filter: function(res) {
for (var k in res)
return +res[k].enabled;
return null;
}
});
var callInitAction = rpc.declare({
object: 'luci',
method: 'setInitAction',
params: [ 'name', 'action' ],
expect: { result: false }
});
var usage = _('See syntax <a %s>here</a>.').format('href=https://lldpd.github.io/usage.html target="_blank"');
const validateioentries = function(section_id, value) {
if (value) {
const emsg = _('Cannot have both interface %s and its exclusion %s');
const a = value.split(' ');
const noex = a.filter(el=> !el.startsWith('!'));
const ex = a.filter(el=> el.startsWith('!') && !el.startsWith('!!'));
for (var i of noex) {
for (var e of ex) {
if ('!'+i == e){
return emsg.format(i, e);
}
}
}
}
return true;
};
return L.view.extend({
__init__: function() {
this.super('__init__', arguments);
// Inject CSS
var head = document.getElementsByTagName('head')[0];
var css = E('link', { 'href':
L.resource('lldpd/lldpd.css')
+ '?v=#PKG_VERSION', 'rel': 'stylesheet' });
head.appendChild(css);
},
load: function() {
return Promise.all([
callInitList('lldpd'),
lldpd.init(),
uci.load('lldpd'),
network.getDevices()
]);
},
// -----------------------------------------------------------------------------------------
//
// Basic Options
//
// -----------------------------------------------------------------------------------------
/** @private */
populateBasicOptions: function(s, tab, data) {
var o;
var serviceEnabled = data[0];
var net_devices = data[3];
// Service enable/disable
o = s.taboption(tab, form.Flag, 'enabled', _('Enable service'));
o.optional = false;
o.rmempty = false;
o.cfgvalue = function() {
return serviceEnabled ? this.enabled : this.disabled;
};
o.write = function(section_id, value) {
uci.set('lldpd', section_id, 'enabled', value);
if (value == '1') {
// Enable and start
callInitAction('lldpd', 'enable').then(function() {
return callInitAction('lldpd', 'start');
});
}
else {
// Stop and disable
callInitAction('lldpd', 'stop').then(function() {
return callInitAction('lldpd', 'disable');
});
}
};
// System description
o = s.taboption(tab, form.Value, 'lldp_description',
_('System description'),
_('Override %s.').format('<code>system description</code>'));
o.placeholder = 'System description';
// System hostname
o = s.taboption(tab, form.Value, 'lldp_hostname',
_('System hostname'),
_('Override %s.').format('<code>system hostname</code>'));
o.placeholder = 'System hostname';
// Host location
o = s.taboption(tab, form.Value, 'lldp_location',
_('Host location'),
_('Override the announced location of the host.') + '<br />' +
usage);
// multiple syntaxes alert for location parameter
o.placeholder = 'address country EU';
o.rmempty = true;
o.validate = function(section_id, value) {
if (value) {
if (!value.match(/^coordinate |^address |^elin /))
return _("Must start: 'coordinate ...', 'address ...' or 'elin ...'");
}
return true;
};
// Platform
o = s.taboption(tab, form.Value, 'lldp_platform',
_('System platform description'),
_('Override %s.').format('<code>system platform</code>') + '<br />' +
_('The default description is the kernel name (Linux).'));
o.placeholder = 'System platform description';
o = s.taboption(tab, form.Flag, 'lldp_capability_advertisements', _('System capability advertisements'));
o.default = '1'; //lldpd internal default
// Capabilities override
o = s.taboption(tab, form.MultiValue, 'lldp_syscapabilities',
_('System capabilities'),
_('Override %s.').format('<code>system capabilities</code>') + '<br />' +
_('The default is derived from kernel information.'));
o.depends({lldp_capability_advertisements: '1'});
o.value('bridge');
o.value('docsis');
o.value('other');
o.value('repeater');
o.value('router');
o.value('station');
o.value('telephone');
o.value('wlan');
o.cfgvalue = function(section_id) {
return String(this.super('load', [section_id]) || this.default).split(',');
};
o.write = function(section_id, value) {
return this.super('write', [ section_id, L.toArray(value).join(',') ]);
};
o = s.taboption(tab, form.Flag, 'lldp_mgmt_addr_advertisements', _('System management IO advertisements'));
o.default = '1'; //lldpd internal default
// Management addresses of this system
// This value: lldpd.init handles as a single value, and needs a CSV for lldpd.conf: 'configure system ip management pattern'
o = s.taboption(tab, lldpd.CBIMultiIOSelect, 'lldp_mgmt_ip',
_('System management IO'),
_('Defaults to the first IPv4 and IPv6. ' +
'If an exact IP address is provided, it is used ' +
'as a management address without any check. To ' +
'blacklist IPv6 addresses, use <code>!*:*</code>.') + '<br />' +
usage);
o.placeholder = 'Addresses and interfaces';
o.depends({lldp_mgmt_addr_advertisements: '1'});
o.cfgvalue = function(section_id) {
const opt = uci.get(this.config, section_id, this.option);
return opt ? opt.split(',') : '';
};
net_devices.forEach(nd => {
o.value(nd.getName());
o.value('!'+nd.getName());
nd.getIPAddrs().forEach(addr => o.value(addr.split('/')[0], E([], [addr.split('/')[0], ' (', E('strong', {}, nd.getName()), ')'])));
nd.getIP6Addrs().forEach(addr => o.value(addr.split('/')[0], E([], [addr.split('/')[0], ' (', E('strong', {}, nd.getName()), ')'])));
});
o.value('!*:*');
o.validate = validateioentries;
o.write = function(section_id, value, sep) {
return this.super('write', [ section_id, value.join(',') ]);
}
// LLDP tx interval
o = s.taboption(tab, form.Value, 'lldp_tx_interval',
_('Transmit delay'),
_('The delay between ' +
'transmissions of LLDP PDU. The default value ' +
'is 30 seconds.') + '<br />' +
_('Suffix %s for millisecond values.').format('<code>ms</code>'));
o.default = 30;
o.placeholder = 30;
o.rmempty = false;
o.validate = function(section_id, value) {
const pattern = /^(\d+)(?:ms)?$/;
if (!value.match(pattern) || parseInt(value) <= 0)
return _('Must be a greater than zero number optionally suffixed with "ms"');
return true;
};
// LLDP tx hold
o = s.taboption(tab, form.Value, 'lldp_tx_hold',
_('Transmit hold value'),
_('Determines the transmitted ' +
'packet TTL (== this value * transmit delay). ' +
'The default value is 4 ∴ ' +
'the default TTL is 120 seconds.'));
o.datatype = 'uinteger';
o.default = 4;
o.placeholder = 4;
o.rmempty = false;
o.validate = function(section_id, value) {
if (value != parseInt(value))
return _('Must be a number');
else if (value <= 0)
return _('Transmit hold value must be greater than 0');
return true;
};
// Received-only mode (-r)
o = s.taboption(tab, form.Flag, 'readonly_mode',
_('Receive-only mode'),
_("LLDPd won't send any frames; " +
'only listen to neighbors.'));
o.rmempty = false;
o.optional = false;
o.default = '0';
},
// -----------------------------------------------------------------------------------------
//
// Network Interfaces
//
// -----------------------------------------------------------------------------------------
/** @private */
populateIfacesOptions: function(s, tab, data) {
var o;
var net_devices = data[3];
// Interfaces to listen on
// This value: lldpd.init handles as a list value, and produces a CSV for lldpd.conf: 'configure system interface pattern'
o = s.taboption(tab, lldpd.CBIMultiIOSelect, 'interface',
_('Network IO'),
_('Specify which interface (not) to listen upon and send LLDPDU from. ' +
'Absent any value, LLDPd uses all available physical interfaces.'));
o.value('*');
net_devices.forEach(nd => {
o.value(nd.getName());
o.value('!'+nd.getName());
o.value('!!'+nd.getName());
});
o.value('!*:*');
o.validate = validateioentries;
// ChassisID interfaces
// This value: lldpd.init handles as a list value, and produces a CSV for the -C param
o = s.taboption(tab, lldpd.CBIMultiIOSelect, 'cid_interface',
_('Network IO for chassis ID'),
_('Specify which interfaces (not) to use for computing chassis ID. ' +
'Absent any value, all interfaces are considered. ' +
'LLDPd takes the first MAC address from all the considered ' +
'interfaces to compute the chassis ID.'));
o.value('*');
o.value('!*');
net_devices.forEach(nd => {
o.value(nd.getName());
o.value('!'+nd.getName());
o.value('!!'+nd.getName());
});
o.value('!*:*');
o.validate = validateioentries;
},
// -----------------------------------------------------------------------------------------
//
// Advanced Options
//
// -----------------------------------------------------------------------------------------
/** @private */
populateAdvancedOptions: function(s, tab, data) {
var o;
// SNMP agentX socket
// **Note**: lldpd is compiled in OpenWrt without SNMP support by default. Setting this action will then cause the lldpd daemon to stop starting and thus lldpd will stop working. To fix this, the value must then be deleted and lldpd restarted.
// o = s.taboption(tab, form.Value, 'agentxsocket',
// _('SNMP agentX socket path'),
// _('If the path to the socket is set, then LLDPd will enable an ' +
// 'SNMP subagent using AgentX protocol. This allows you to get ' +
// 'information about local system and remote systems through SNMP.'));
// o.rmempty = true;
// o.placeholder = '/var/run/agentx.sock';
// o.default = '';
// LLDP-MED class
o = s.taboption(tab, form.ListValue, 'lldp_class',
_('LLDP-MED device class'));
o.value('1', _('Generic Endpoint (Class I)'));
o.value('2', _('Media Endpoint (Class II)'));
o.value('3', _('Communication Device Endpoints (Class III)'));
o.value('4', _('Network Connectivity Device (Class IV)'));
o.default = '4';
// LLDP-MED policy
o = s.taboption(tab, form.Value, 'lldp_policy',
_('LLDP-MED policy'));
o.depends({lldp_class: '2'});
o.depends({lldp_class: '3'});
o.rmempty = true;
o.placeholder = 'application streaming-video';
o.value('application voice');
o.value('application voice unknown');
o.value('application voice-signaling');
o.value('application voice-signaling unknown');
o.value('application guest-voice');
o.value('application guest-voice unknown');
o.value('application guest-voice-signaling');
o.value('application guest-voice-signaling unknown');
o.value('application softphone-voice');
o.value('application softphone-voice unknown');
o.value('application video-conferencing');
o.value('application video-conferencing unknown');
o.value('application streaming-video');
o.value('application streaming-video unknown');
o.value('application video-signaling');
o.value('application video-signaling unknown');
o.validate = function(section_id, value) {
if (value && !value.startsWith('application '))
return _('Must start: application ...');
return true;
};
// LLDP-MED fast-start
o = s.taboption(tab, form.Flag, 'lldpmed_fast_start',
_('LLDP-MED fast-start'));
// LLDP-MED fast-start
o = s.taboption(tab, form.Value, 'lldpmed_fast_start_tx_interval',
_('LLDP-MED fast-start tx-interval'));
o.depends({lldpmed_fast_start: '1'});
o.datatype = 'uinteger';
o.placeholder = '10';
o.rmempty = true;
// LLDP-MED inventory TLV transmission (-i)
o = s.taboption(tab, form.Flag, 'lldpmed_no_inventory',
_('Disable LLDP-MED inventory TLV transmission'),
_('LLDPd will still receive (and publish using SNMP if enabled) ' +
'those LLDP-MED TLV but will not send them. Use this option ' +
'if you do not want to transmit sensitive information like serial numbers.'));
o.default = '0';
// Disable advertising of kernel release, version and machine. (-k)
o = s.taboption(tab, form.Flag, 'lldp_no_version',
_('Disable advertising of kernel release, version and machine'),
_('Kernel name (ie: Linux) will still be shared, and Inventory ' +
'software version will be set to %s.').format('<code>Unknown</code>'));
o.default = '0';
// Filter neighbors (-H)
o = s.taboption(tab, lldpd.cbiFilterSelect, 'filter',
_('Specify the behaviour when detecting multiple neighbors'),
_('The default filter is 15. Refer to "FILTERING NEIGHBORS".') + '<br />' +
usage);
o.default = 15;
// Force port ID subtype
o = s.taboption(tab, form.ListValue, 'lldp_portidsubtype',
_('Force port ID subtype'),
_('With this option, you can force the port identifier ' +
'to be the interface name or the MAC address.'));
o.value('macaddress', _('Interface MAC address'));
o.value('ifname', _('Interface name'));
o.default = 'macaddress';
// The destination MAC address used to send LLDPDU
o = s.taboption(tab, form.ListValue, 'lldp_agenttype',
_('LLDPDU destination MAC'),
_('Allows an agent ' +
'to control the propagation of LLDPDUs. By default, the ' +
'MAC address %s is used and limits the propagation ' +
'of the LLDPDU to the nearest bridge.').format('<code>01:80:c2:00:00:0e</code>'));
o.value('nearest-bridge', '01:80:c2:00:00:0e (nearest-bridge)');
o.value('nearest-nontpmr-bridge', '01:80:c2:00:00:03 (nearest-nontpmr-bridge)');
o.value('nearest-customer-bridge', '01:80:c2:00:00:00 (nearest-customer-bridge)');
o.default = 'nearest-bridge';
},
// -----------------------------------------------------------------------------------------
//
// Protocols Support
//
// -----------------------------------------------------------------------------------------
/** @private */
populateProtocolsOptions: function(s, tab, data) {
var o;
o = s.taboption(tab, form.SectionValue, '_protocols', form.TypedSection, 'lldpd');
var ss = o.subsection;
ss.anonymous = true;
ss.addremove = false;
//
// LLDPD
// Link Layer Discovery Protocol
//
ss.tab('lldp', _('LLDP'));
o = ss.taboption('lldp', form.Flag, 'enable_lldp',
_('Enable LLDP'));
o.default = '1';
o.rmempty = true;
o = ss.taboption('lldp', form.Flag, 'force_lldp',
_('Force sending LLDP packets'),
_('Even when there is no LLDP peer ' +
'detected but there is a peer speaking another protocol detected.') + '<br />' +
_('By default, LLDP packets are sent when there is a peer speaking ' +
'LLDP detected or when there is no peer at all.'));
o.default = '0';
o.rmempty = true;
o.depends('enable_lldp', '1');
//
// CDP
// Cisco Discovery Protocol
//
ss.tab('cdp', _('CDP'));
o = ss.taboption('cdp', form.Flag, 'enable_cdp',
_('Enable CDP'),
_('Enable the support of CDP protocol to deal with Cisco routers ' +
'that do not speak LLDP'));
o.default = '1';
o.rmempty = false;
o = ss.taboption('cdp', form.ListValue, 'cdp_version',
_('CDP version'));
o.value('cdpv1v2', _('CDPv1 and CDPv2'));
o.value('cdpv2', _('Only CDPv2'));
o.depends('enable_cdp', '1');
o.default = 'cdpv1v2';
o = ss.taboption('cdp', form.Flag, 'force_cdp',
_('Send CDP packets even if no CDP peer detected'));
o.default = '0';
o.rmempty = true;
o.depends('enable_cdp', '1');
o = ss.taboption('cdp', form.Flag, 'force_cdpv2',
_('Force sending CDPv2 packets'));
o.default = '0';
o.rmempty = true;
o.depends({
force_cdp: '1',
enable_cdp: '1',
cdp_version: 'cdpv1v2'
});
//
// FDP
// Foundry Discovery Protocol
//
ss.tab('fdp', _('FDP'));
o = ss.taboption('fdp', form.Flag, 'enable_fdp',
_('Enable FDP'),
_('Enable the support of FDP protocol to deal with Foundry routers ' +
'that do not speak LLDP'));
o.default = '1';
o.rmempty = false;
o = ss.taboption('fdp', form.Flag, 'force_fdp',
_('Send FDP packets even if no FDP peer detected'));
o.default = '0';
o.rmempty = true;
o.depends('enable_fdp', '1');
//
// EDP
// Extreme Discovery Protocol
//
ss.tab('edp', _('EDP'));
o = ss.taboption('edp', form.Flag, 'enable_edp',
_('Enable EDP'),
_('Enable the support of EDP protocol to deal with Extreme routers ' +
'and switches that do not speak LLDP.'));
o.default = '1';
o.rmempty = false;
o = ss.taboption('edp', form.Flag, 'force_edp',
_('Send EDP packets even if no EDP peer detected'));
o.default = '0';
o.rmempty = true;
o.depends('enable_edp', '1');
//
// SONMP
// SynOptics Network Management Protocol
//
// a.k.a.
// Nortel Topology Discovery Protocol (NTDP)
// Nortel Discovery Protocol (NDP)
// Bay Network Management Protocol (BNMP)
// Bay Discovery Protocol (BDP)
//
ss.tab('sonmp', _('SONMP (NTDP, NDP, BNMP, BDP)'));
o = ss.taboption('sonmp', form.Flag, 'enable_sonmp',
_('Enable SONMP'),
_('Enable the support of SONMP protocol to deal with Nortel ' +
'routers and switches that do not speak LLDP.'));
o.default = '1';
o.rmempty = false;
o = ss.taboption('sonmp', form.Flag, 'force_sonmp',
_('Send SONMP packets even if no SONMP peer detected'));
o.default = '0';
o.rmempty = true;
o.depends('enable_sonmp', '1');
},
/** @private */
populateOptions: function(s, data) {
var o;
s.tab('basic', _('Basic Settings'));
this.populateBasicOptions(s, 'basic', data);
s.tab('ifaces', _('Network Interfaces'));
this.populateIfacesOptions(s, 'ifaces', data);
s.tab('advanced', _('Advanced Settings'));
this.populateAdvancedOptions(s, 'advanced', data);
s.tab('protocols', _('Protocols Support'));
this.populateProtocolsOptions(s, 'protocols', data);
},
render: function(data) {
var m, s;
m = new form.Map('lldpd', _('LLDPd Settings'),
_('LLDPd is an implementation of IEEE 802.1ab') + ' ' +
'(<abbr title="Link Layer Discovery Protocol">LLDP</abbr>).' + ' ' +
_('On this page you may configure LLDPd parameters.'));
s = m.section(form.TypedSection, 'lldpd');
s.addremove = false;
s.anonymous = true;
this.populateOptions(s, data);
return m.render();
},
});
|