mirror of
git://git.code.sf.net/p/dhcpd-pools/code
synced 2025-12-16 15:57:00 +00:00
Compare commits
313 commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1722d54103 | ||
|
|
1c413b0653 | ||
|
|
d020e4d09f | ||
|
|
f917d5b223 | ||
|
|
5b7ab28314 | ||
|
|
dc649e27cd | ||
|
|
7fc13c60e0 | ||
|
|
5ed6e7688f | ||
|
|
9c6336e5b6 | ||
|
|
d94654e100 | ||
|
|
dc1f0b9b76 | ||
|
|
3d37ac0a2d | ||
|
|
102d017ed5 | ||
|
|
501bc15b43 | ||
|
|
0c5b8301dd | ||
|
|
1d20604519 | ||
|
|
5126e63665 | ||
|
|
c7e0058994 | ||
|
|
2ecfc86a06 | ||
|
|
0f19d44c1d | ||
|
|
b005ce9a25 | ||
|
|
45f7fc1514 | ||
|
|
ec613f762d | ||
|
|
e48768b041 | ||
|
|
544e7ec0a5 | ||
|
|
b96c6c14d2 | ||
|
|
21194d2e4a | ||
|
|
fb518b7a19 | ||
|
|
707eafa670 | ||
|
|
07b4eaa480 | ||
|
|
17d68b7e3e | ||
|
|
4f64902a9e | ||
|
|
008e9f17c1 | ||
|
|
71714a0993 | ||
|
|
6f6369f517 | ||
|
|
8731c6a11f | ||
|
|
d29b498cae | ||
|
|
48e46e8698 | ||
|
|
4befdeeb47 | ||
|
|
9727bb28b9 | ||
|
|
5027b50ade | ||
|
|
27c70a0efe | ||
|
|
7fe686b417 | ||
|
|
3f85360c64 | ||
|
|
e8e9d49ebb | ||
|
|
4e7ab66fd4 | ||
|
|
b568725e7d | ||
|
|
a192f51545 | ||
|
|
679d63dd9f | ||
|
|
0fe89808ee | ||
|
|
447241e6c3 | ||
|
|
2849dde21b | ||
|
|
813e320b68 | ||
|
|
e6e90b4d62 | ||
|
|
a1d2bd2cf7 | ||
|
|
46ec42182b | ||
|
|
444815f964 | ||
|
|
6d737a7607 | ||
|
|
ff3d9523e6 | ||
|
|
d24313d25a | ||
|
|
c687f38ed6 | ||
|
|
887845df2a | ||
|
|
74b697321f | ||
|
|
3369278fc0 | ||
|
|
e4f7259cf6 | ||
|
|
eabaa8adc2 | ||
|
|
8fba5c5e6b | ||
|
|
88a3f1eb53 | ||
|
|
8ae5fbf489 | ||
|
|
b2c764924b | ||
|
|
242c58f45e | ||
|
|
c029c7581a | ||
|
|
a64630aa49 | ||
|
|
15f08bbf02 | ||
|
|
ef5421ed05 | ||
|
|
a237c11d5a | ||
|
|
5a8c178924 | ||
|
|
d25e7afa1c | ||
|
|
7cd381ed83 | ||
|
|
1875a13733 | ||
|
|
adda925c1e | ||
|
|
3dda0a77a5 | ||
|
|
7575294c36 | ||
|
|
39b2811aa7 | ||
|
|
b96f8cd8ad | ||
|
|
54cedc1001 | ||
|
|
e4baff79bd | ||
|
|
4fc4bcd083 | ||
|
|
d5ae2a80c0 | ||
|
|
50fc4ebe9d | ||
|
|
66183bc7c7 | ||
|
|
2ec953858b | ||
|
|
39e6e65201 | ||
|
|
74fdf90980 | ||
|
|
7d9a5b5561 | ||
|
|
fe847bb9b1 | ||
|
|
1182ec4cc9 | ||
|
|
e9736f74e7 | ||
|
|
e5cd46e4e9 | ||
|
|
eb7547c742 | ||
|
|
8fe7dc2c03 | ||
|
|
c55c823753 | ||
|
|
e079cc16e0 | ||
|
|
344ed2900d | ||
|
|
48962004b8 | ||
|
|
05f8208518 | ||
|
|
abf5d04736 | ||
|
|
9deeae8c36 | ||
|
|
548ce63963 | ||
|
|
58a31513fe | ||
|
|
804aa3236d | ||
|
|
af1ae94112 | ||
|
|
fefd8d26cf | ||
|
|
1d47eb9a1c | ||
|
|
0c32a67ff1 | ||
|
|
bb0fa9adae | ||
|
|
48d0629881 | ||
|
|
782f63c3ad | ||
|
|
b9cff0d814 | ||
|
|
dff991666e | ||
|
|
242ef3109b | ||
|
|
1c8b799799 | ||
|
|
e93fc5dba4 | ||
|
|
c3c3fc6e40 | ||
|
|
e9223a852c | ||
|
|
9a52619385 | ||
|
|
2b75a0d78e | ||
|
|
344e01c1f6 | ||
|
|
c4a654a149 | ||
|
|
a905b50943 | ||
|
|
ae2edb0fbc | ||
|
|
5eae1b41a7 | ||
|
|
ea7fd91876 | ||
|
|
11c0b23d77 | ||
|
|
aaad35a8c9 | ||
|
|
a6b77ab785 | ||
|
|
7740927721 | ||
|
|
9bb30b29ca | ||
|
|
1b35a16d95 | ||
|
|
d26c858c13 | ||
|
|
840d2143e6 | ||
|
|
a3ef3d617f | ||
|
|
f6e256243d | ||
|
|
ccd5370d9c | ||
|
|
c4ebafb106 | ||
|
|
87c06a1b13 | ||
|
|
8a8c28a17e | ||
|
|
5a33b619d1 | ||
|
|
e5f9a77511 | ||
|
|
32e2d399a0 | ||
|
|
c305e2f82c | ||
|
|
10b06d88f0 | ||
|
|
3d0c510475 | ||
|
|
46641751b4 | ||
|
|
d717a043bf | ||
|
|
03b1edb188 | ||
|
|
1dcf762967 | ||
|
|
17f1fb7f5e | ||
|
|
4cd85b62ff | ||
|
|
ace697b472 | ||
|
|
1a7d982495 | ||
|
|
01aa13cf43 | ||
|
|
da4b7a783a | ||
|
|
15802d6648 | ||
|
|
d6486e7730 | ||
|
|
bd45266f74 | ||
|
|
8fca82f148 | ||
|
|
f5cd7383e4 | ||
|
|
49835cccb5 | ||
|
|
98bcdf9378 | ||
|
|
b524296016 | ||
|
|
5ed9958b69 | ||
|
|
0369340710 | ||
|
|
327691f34a | ||
|
|
4aff49ed80 | ||
|
|
f11ca0bec2 | ||
|
|
1299737d76 | ||
|
|
c7917152d3 | ||
|
|
5cede1ff31 | ||
|
|
7f3d553c7f | ||
|
|
99b6af70ef | ||
|
|
2528c4c9d6 | ||
|
|
c30c122027 | ||
|
|
535dfc4fc2 | ||
|
|
fae20302cf | ||
|
|
167c4b6989 | ||
|
|
d8aef85d01 | ||
|
|
cf26e17fe7 | ||
|
|
9d1241c006 | ||
|
|
c4e5ef6198 | ||
|
|
0d6c61d437 | ||
|
|
73b357484d | ||
|
|
735c5db817 | ||
|
|
34dd1dee6e | ||
|
|
c7379e3e25 | ||
|
|
cfbd69c20b | ||
|
|
33894fba74 | ||
|
|
8e076fcc4f | ||
|
|
0705b0c17f | ||
|
|
5519763ba9 | ||
|
|
190df198a4 | ||
|
|
e401c2c7e6 | ||
|
|
0d2b30b62d | ||
|
|
bd5877bf4f | ||
|
|
15502d3c97 | ||
|
|
74c6ef2566 | ||
|
|
e09f655a7b | ||
|
|
965875d20b | ||
|
|
4392a5b917 | ||
|
|
9090cfb1d7 | ||
|
|
6b267387cc | ||
|
|
6e680ee04f | ||
|
|
502daf8306 | ||
|
|
0f66becd57 | ||
|
|
eb55946595 | ||
|
|
5454ab7086 | ||
|
|
f3e282f122 | ||
|
|
ca0b1c3262 | ||
|
|
d089a19fb5 | ||
|
|
affb3d70a3 | ||
|
|
6b88e8d872 | ||
|
|
7fc354827a | ||
|
|
48caf5fb42 | ||
|
|
631bf0bf7c | ||
|
|
1eca347196 | ||
|
|
6379ace9c7 | ||
|
|
d2d8abd660 | ||
|
|
8da98bbc89 | ||
|
|
97c6f0292f | ||
|
|
5b8ad97611 | ||
|
|
5189333c95 | ||
|
|
20ec1c2b42 | ||
|
|
7c872ef8d7 | ||
|
|
e8f8c04979 | ||
|
|
47582ed8d8 | ||
|
|
06ffa5ed04 | ||
|
|
d70b08244f | ||
|
|
3944e289c8 | ||
|
|
57ddd4f785 | ||
|
|
28f1e8c54c | ||
|
|
4becf97fb3 | ||
|
|
978ddeddeb | ||
|
|
9f676f5276 | ||
|
|
c0e7b50bd3 | ||
|
|
08c9bcc502 | ||
|
|
7de719a80c | ||
|
|
3f3d447ad8 | ||
|
|
824a6aec73 | ||
|
|
8dba1c749d | ||
|
|
0002aa57e6 | ||
|
|
30238e9d1c | ||
|
|
0495ea4edf | ||
|
|
ab535d8ab1 | ||
|
|
ffffbc96cc | ||
|
|
b24fc42ec8 | ||
|
|
1a7649beb3 | ||
|
|
805d353584 | ||
|
|
5966360f12 | ||
|
|
90c1a4e4c5 | ||
|
|
3e677d8a24 | ||
|
|
7c34231004 | ||
|
|
d2b7d10dbc | ||
|
|
327b601ea4 | ||
|
|
5ee1aca2b9 | ||
|
|
b858a4778b | ||
|
|
cdf2c81f31 | ||
|
|
c3e492165c | ||
|
|
23aae87356 | ||
|
|
8e2da89c88 | ||
|
|
f3a476ba52 | ||
|
|
625c54d834 | ||
|
|
e50d0a7d38 | ||
|
|
34f73dbae1 | ||
|
|
632e97834b | ||
|
|
3be3f34bd4 | ||
|
|
478bbe1b44 | ||
|
|
ff7ca8c419 | ||
|
|
75d979bb00 | ||
|
|
9c4184bd9a | ||
|
|
afd9f77051 | ||
|
|
6820c9fc09 | ||
|
|
6642335c8a | ||
|
|
39e1fb9e5a | ||
|
|
137c1d37c2 | ||
|
|
4051a953bd | ||
|
|
70c65f4d59 | ||
|
|
51143b6b3c | ||
|
|
82df7c98d4 | ||
|
|
bd5ee58a3f | ||
|
|
4b014eba1c | ||
|
|
d3ee104a59 | ||
|
|
b59e980cab | ||
|
|
a9d2d9c82f | ||
|
|
2008e1a047 | ||
|
|
f1411858cb | ||
|
|
3471c91796 | ||
|
|
aa7928b3ab | ||
|
|
fdc560bab9 | ||
|
|
d8aae53968 | ||
|
|
a768049f3e | ||
|
|
443b197901 | ||
|
|
bc654bcc68 | ||
|
|
e9d196e816 | ||
|
|
48a23c94e5 | ||
|
|
d34259092d | ||
|
|
a302bc6829 | ||
|
|
179e2ac147 | ||
|
|
a2fa69a403 | ||
|
|
1424db9427 | ||
|
|
a57d399643 | ||
|
|
71bcee14e9 | ||
|
|
6684772550 | ||
|
|
761c9560d7 |
182 changed files with 13099 additions and 2534 deletions
41
.gitignore
vendored
41
.gitignore
vendored
|
|
@ -1,28 +1,33 @@
|
||||||
|
# Wildcard + in any subdir.
|
||||||
|
*.o
|
||||||
|
*.gcno
|
||||||
|
*.gcda
|
||||||
|
*.[ch].gcov
|
||||||
|
|
||||||
|
# Exact filename in any subdir.
|
||||||
|
.deps
|
||||||
|
.dirstamp
|
||||||
|
Makefile
|
||||||
|
Makefile.in
|
||||||
|
|
||||||
|
# Exact wildcard, e.g., not in subdirs.
|
||||||
|
/tests/*.log
|
||||||
|
/tests/*.trs
|
||||||
|
|
||||||
|
# Exact match, if a directory then everything in directory.
|
||||||
|
/.version
|
||||||
/aclocal.m4
|
/aclocal.m4
|
||||||
/autom4te.cache/
|
/autom4te.cache/
|
||||||
/autoscan.log
|
|
||||||
/config.guess
|
|
||||||
/config.h
|
/config.h
|
||||||
/config.h.in
|
/config.h.in
|
||||||
/config.h.in~
|
|
||||||
/config.log
|
/config.log
|
||||||
/config.status
|
/config.status
|
||||||
/config.sub
|
|
||||||
/configure
|
/configure
|
||||||
/configure.scan
|
/dhcpd-pools
|
||||||
/contrib/Makefile
|
|
||||||
/contrib/Makefile.in
|
|
||||||
/depcomp
|
|
||||||
/gnulib/
|
|
||||||
/INSTALL
|
/INSTALL
|
||||||
/install-sh
|
|
||||||
/libtool
|
/libtool
|
||||||
/ltmain.sh
|
/m4/
|
||||||
/m4
|
/man/dhcpd-pools.1
|
||||||
/Makefile
|
|
||||||
/Makefile.in
|
|
||||||
/man/Makefile
|
|
||||||
/man/Makefile.in
|
|
||||||
/missing
|
|
||||||
/stamp-h1
|
/stamp-h1
|
||||||
/tags
|
/test-suite.log
|
||||||
|
/tests/outputs/
|
||||||
|
|
|
||||||
2
.mailmap
Normal file
2
.mailmap
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
Sami Kerola <kerolasa@iki.fi>
|
||||||
|
Sami Kerola <kerolasa@iki.fi> <sami.kerola@tomtom.com>
|
||||||
54
Makefile.am
54
Makefile.am
|
|
@ -1,7 +1,55 @@
|
||||||
## Makefile.am -- Process this file with automake to produce Makefile.in
|
## Makefile.am -- Process this file with automake to produce Makefile.in
|
||||||
|
|
||||||
AUTOMAKE_OPTIONS = gnu
|
AUTOMAKE_OPTIONS = gnu
|
||||||
ACLOCAL_AMFLAGS = -I m4
|
|
||||||
|
|
||||||
EXTRA_DIST = m4/gnulib-cache.m4
|
EXTRA_DIST = \
|
||||||
SUBDIRS = lib src man contrib
|
.version \
|
||||||
|
build-aux/git-version-gen \
|
||||||
|
m4/gnulib-cache.m4
|
||||||
|
|
||||||
|
SUBDIRS = lib
|
||||||
|
|
||||||
|
BUILT_SOURCES = $(top_srcdir)/.version
|
||||||
|
$(top_srcdir)/.version:
|
||||||
|
echo $(VERSION) > $@-t && mv $@-t $@
|
||||||
|
dist-hook:
|
||||||
|
echo $(VERSION) > $(distdir)/.tarball-version
|
||||||
|
|
||||||
|
PATHFILES =
|
||||||
|
CLEANFILES = $(PATHFILES)
|
||||||
|
EXTRA_DIST += $(PATHFILES:=.in)
|
||||||
|
CLEAN_LOCALS =
|
||||||
|
|
||||||
|
edit_cmd = sed \
|
||||||
|
-e 's|@ALARM_CRIT[@]|$(ALARM_CRIT)|g' \
|
||||||
|
-e 's|@ALARM_WARN[@]|$(ALARM_WARN)|g' \
|
||||||
|
-e 's|@DHCPDCONF_FILE[@]|$(DHCPDCONF_FILE)|g' \
|
||||||
|
-e 's|@DHCPDLEASE_FILE[@]|$(DHCPDLEASE_FILE)|g' \
|
||||||
|
-e 's|@OUTPUT_FORMAT[@]|$(OUTPUT_FORMAT)|g' \
|
||||||
|
-e 's|@OUTPUT_LIMIT[@]|$(OUTPUT_LIMIT)|g' \
|
||||||
|
-e 's|@PACKAGE_BUGREPORT[@]|$(PACKAGE_BUGREPORT)|g' \
|
||||||
|
-e 's|@PACKAGE_MAINTAINER[@]|$(PACKAGE_MAINTAINER)|g' \
|
||||||
|
-e 's|@PACKAGE_NAME[@]|$(PACKAGE_NAME)|g' \
|
||||||
|
-e 's|@PACKAGE_URL[@]|$(PACKAGE_URL)|g' \
|
||||||
|
-e 's|@SHELL[@]|$(SHELL)|g' \
|
||||||
|
-e 's|@VERSION[@]|$(VERSION)|g' \
|
||||||
|
-e 's|@bindir[@]|$(bindir)|g' \
|
||||||
|
-e 's|@docdir[@]|$(docdir)|g' \
|
||||||
|
-e 's|@top_srcdir[@]|$(top_srcdir)|g'
|
||||||
|
|
||||||
|
$(PATHFILES): Makefile
|
||||||
|
@ rm -f $@ $@.tmp
|
||||||
|
$(AM_V_at) mkdir -p $$(dirname $@)
|
||||||
|
$(AM_V_GEN) srcdir=''; \
|
||||||
|
test -f ./$@.in || srcdir=$(srcdir)/; \
|
||||||
|
$(edit_cmd) $${srcdir}$@.in >$@.tmp
|
||||||
|
@ mv $@.tmp $@
|
||||||
|
|
||||||
|
include contrib/Makemodule.am
|
||||||
|
include doc/Makemodule.am
|
||||||
|
include man/Makemodule.am
|
||||||
|
include samples/Makemodule.am
|
||||||
|
include src/Makemodule.am
|
||||||
|
include tests/Makemodule.am
|
||||||
|
|
||||||
|
clean-local: $(CLEAN_LOCALS)
|
||||||
|
|
|
||||||
382
NEWS
382
NEWS
|
|
@ -5,6 +5,388 @@ See the end for copying conditions.
|
||||||
|
|
||||||
Please send dhcpd-pools bug reports to kerolasa@iki.fi.
|
Please send dhcpd-pools bug reports to kerolasa@iki.fi.
|
||||||
|
|
||||||
|
gpg: Signature is crated using RSA key ID 8ED396E37E38D471A00530D3A9553245FDE9B739.
|
||||||
|
|
||||||
|
|
||||||
|
Version 3.3
|
||||||
|
Belkacem Daheb (2):
|
||||||
|
add start, end and hostname printing support for xml and json
|
||||||
|
getdata output: add start, end and hostname printing support for xml and json
|
||||||
|
|
||||||
|
M. van Brummelen (1):
|
||||||
|
docs: fix manual page groff warning
|
||||||
|
|
||||||
|
Sami Kerola (9):
|
||||||
|
update project web page
|
||||||
|
website: use https
|
||||||
|
chore: update web links
|
||||||
|
getdata: only emit warning when config include file cannot be read
|
||||||
|
gnulib: update bootstrap and gitignore files
|
||||||
|
build-sys: quote subshell execution in the autotools file
|
||||||
|
build-sys: update bootstrap from gnulib
|
||||||
|
output: update html javascripts
|
||||||
|
release: 3.3
|
||||||
|
|
||||||
|
luisδμ (1):
|
||||||
|
fix: avoid generation of unvalid JSON in summary
|
||||||
|
|
||||||
|
Version 3.2
|
||||||
|
Jean Benoit (1):
|
||||||
|
contrib: snmptest.pl SNMPwalk can't access to all variables/wrong sort
|
||||||
|
|
||||||
|
Sami Kerola (5):
|
||||||
|
contrib: point out where one can find zabbix template
|
||||||
|
build-sys: update .gitignore files
|
||||||
|
config: remove unnecessary padding
|
||||||
|
build-sys: autotools and gnulib related updates
|
||||||
|
build-sys: routine update
|
||||||
|
release: 3.2
|
||||||
|
|
||||||
|
Version 3.1
|
||||||
|
Mark Sangster (1):
|
||||||
|
output: fix warn and crit counts on shared networks
|
||||||
|
|
||||||
|
Sami Kerola (31):
|
||||||
|
webpage: add instructions how to get output you need
|
||||||
|
build-sys: add coverage files to .gitignore
|
||||||
|
tests: improve coverage
|
||||||
|
docs: fix peoples name in THANKS file to have correct characters
|
||||||
|
add .mailmap
|
||||||
|
output: fix implicit conversion
|
||||||
|
main: simplify option parsing
|
||||||
|
fix typo
|
||||||
|
main: move print_mac_addreses to state structure
|
||||||
|
main: move output_format to state, and rename color_format
|
||||||
|
webpages: use html sample output
|
||||||
|
fix switch missing default case warnings
|
||||||
|
sort: ensure NaN will not trip over comp_double()
|
||||||
|
drop images, java scripts, and such from web sitemap file
|
||||||
|
getdata: remote dead code
|
||||||
|
other: use strftime() to generate date-time string
|
||||||
|
lib: update .gitignore
|
||||||
|
fix typos
|
||||||
|
various: fix few warnings
|
||||||
|
warnings: ensure optimal packing in structures
|
||||||
|
other: use IP string lengths from netinet/in.h
|
||||||
|
mustach: sync with most recent mustach upstream changes
|
||||||
|
various: tidy up variable scopes, and one name mismatch
|
||||||
|
build-sys: update .gitignore files
|
||||||
|
output: add warning and critical threshold counts to mustach
|
||||||
|
samples: make prometheus template less klunky
|
||||||
|
misc: fix spelling issues
|
||||||
|
output: add ethernet address priting support to --mustach
|
||||||
|
build-sys: update bootstrap from gnulib
|
||||||
|
output: update javascripts
|
||||||
|
release: 3.1
|
||||||
|
|
||||||
|
Version 3.0
|
||||||
|
Sami Kerola (56):
|
||||||
|
release: update web page meta data
|
||||||
|
update sitemap
|
||||||
|
output: make output_analysis() to be regular function
|
||||||
|
getdata: fix typo
|
||||||
|
output: add output helper functions
|
||||||
|
output: add color support to text output
|
||||||
|
output: include earlier missing data to json output
|
||||||
|
output: add --skip-ok option
|
||||||
|
other: add --skip-ok to usage() output
|
||||||
|
output: json nan values need quoting
|
||||||
|
other: do not use 'else' after 'return'
|
||||||
|
getdata: do not use 'else' after 'continue'
|
||||||
|
include: use project specific header guard
|
||||||
|
output: use range_output_helper() value in output_xml()
|
||||||
|
output: add mustach templating support
|
||||||
|
output: add separate first_ip and last_ip to json and mustach outputs
|
||||||
|
docs: add mustach sample files
|
||||||
|
tests: add mustach check
|
||||||
|
output: make mustach processing more robust
|
||||||
|
output: avoid mixing ntop_ipaddr() output buffers
|
||||||
|
output: do not skip over first range in mustach output
|
||||||
|
output: save and reuse output helper results
|
||||||
|
output: add must_put_err() utility function
|
||||||
|
output: include stdlib.h to avoid compilation error
|
||||||
|
gnulib: use nstrftime instead of strftime
|
||||||
|
build-sys: omit mustach compilation when it cannot work
|
||||||
|
usage: add error message informing mustach support is not available
|
||||||
|
output: deduplicate file closing code
|
||||||
|
clean up: remove unused variable
|
||||||
|
all files: replace global variables with runtime config state structure
|
||||||
|
all files: re-indent
|
||||||
|
analyze: bug fix shared networks counts
|
||||||
|
other: add Jose Bollo to version output credits
|
||||||
|
output: improve mustach template parsing error
|
||||||
|
analysis: shared networks to be linked list
|
||||||
|
output: make --skip-ok to effect --perfdata
|
||||||
|
output: move shared net andn range status check to output_helper
|
||||||
|
output: make warning and critical colors work in html output
|
||||||
|
output: display more entries in html table by default
|
||||||
|
output: shared net can be in suppressed state
|
||||||
|
output: improve html table
|
||||||
|
hash: include stdlib.h to avoid implicit declarations
|
||||||
|
clean up: fix couple compiler warnings
|
||||||
|
thanks: add Troy D. Hanson to credits about uthash
|
||||||
|
samples: add prometheus text file collector mustach template
|
||||||
|
output: make --skip to take arguments what will be skipped
|
||||||
|
misc: move command line option parsing to separate function
|
||||||
|
misc: move couple enums from global scope to file scope
|
||||||
|
docs: improve doxygen documentation
|
||||||
|
output: remove unused variable attribute
|
||||||
|
output: add more items to mustach tags
|
||||||
|
output: unify time stamp creations
|
||||||
|
orther: fix xstrstr_init() memcmp() return value usage
|
||||||
|
output: add some trivia data to json output
|
||||||
|
usage: --skip=ignored is actually 'suppressed'
|
||||||
|
release: 3.0
|
||||||
|
|
||||||
|
Version 2.29
|
||||||
|
Boris Lytochkin (1):
|
||||||
|
introduce -A arg: treat single subnets as shared-network with CIDR as their name
|
||||||
|
|
||||||
|
Manuel Hachtkemper (1):
|
||||||
|
alarming: add additional performance data
|
||||||
|
|
||||||
|
Sami Kerola (47):
|
||||||
|
tell in README when ./bootstrap is needed
|
||||||
|
add dhcpd-pools website content to a subdirectory
|
||||||
|
add sitemap url to robots.txt file
|
||||||
|
use long options in .indent.pro file
|
||||||
|
webpages: make index page mobile device friendly
|
||||||
|
webpages: compress sitemap.txt file
|
||||||
|
getdata: flip ranges if they are in greater smaller order
|
||||||
|
tests: add range definition flip test
|
||||||
|
contrib: add archlinux package build file
|
||||||
|
contrib: remove unnecessary cgi script
|
||||||
|
contrib: remove awk file duplicate
|
||||||
|
getdata: get rid of remaining stdbool usage
|
||||||
|
argument parsing: fix compiler warning
|
||||||
|
add dhcpd-pools Description Of A Project file
|
||||||
|
fix doap file git repository and license section
|
||||||
|
output: check alarming mode can output successfully
|
||||||
|
build-sys: update bootstrap from gnulib
|
||||||
|
build-sys: update gnulib .gitignore file
|
||||||
|
analyze: use while() when for() is less fit to purpose
|
||||||
|
build-sys: default to ./configure --enable-silent-rules
|
||||||
|
build-sys: always use restrict found by autoconf
|
||||||
|
docs: fix couple typos and improve a sentence in README
|
||||||
|
output: remove unnecessary increment
|
||||||
|
portability: add gnulib modules earlier missing
|
||||||
|
remove const and pure function attributes
|
||||||
|
man: improve synopsis and output limit
|
||||||
|
man: remove old html table only option argument from manual
|
||||||
|
getdata: report position in config file when parsing fails
|
||||||
|
output: fix timestamp localization on html page
|
||||||
|
output: add include avoid referringt to undefined definition
|
||||||
|
getdata: remove POSIX_FADV_NOREUSE
|
||||||
|
style: use same argument names in header and source file
|
||||||
|
contrib: Klaus Slott told about opensuse package
|
||||||
|
getdata: fpos_t is not easy to print correctly
|
||||||
|
lib: update .gitignore
|
||||||
|
fix typo
|
||||||
|
variable: add const to print_mac_addreses_tmp
|
||||||
|
lib: update .gitignore
|
||||||
|
add --ip-version option to force either IPv4 or IPv6 analysis
|
||||||
|
lib: update .gitignore
|
||||||
|
getdata: add cidr range support
|
||||||
|
build-sys: update bootstrap from gnulib
|
||||||
|
docs: update doxygen configuration file
|
||||||
|
docs: tell what needs to be done when releasing new version
|
||||||
|
docs: add build instruction link to the project web page
|
||||||
|
docs: update maintainer gpg key
|
||||||
|
release: 2.29
|
||||||
|
|
||||||
|
Version 2.28
|
||||||
|
Sami Kerola (26):
|
||||||
|
portability: Solaris 10 does not have err.h
|
||||||
|
fix couple compiler warnings
|
||||||
|
make binary exec path dynamic in scripts referring to it
|
||||||
|
update doxygen.conf file
|
||||||
|
add --snet-alarms option to suppress excess range alarms
|
||||||
|
add --warn-count and --crit-count options to suppress alarm noise
|
||||||
|
improve README file instructions
|
||||||
|
make html output to use Bootstrap and DataTables
|
||||||
|
fix typo in README
|
||||||
|
correct return value FIXME items
|
||||||
|
use more descriptive names in limit bits enum definition
|
||||||
|
remove upper limit of sort order definitions
|
||||||
|
fix protocol specifier in html output
|
||||||
|
use bitmap for booleans and other config that has known size
|
||||||
|
simplify output format selection, and passing
|
||||||
|
fix doxygen build
|
||||||
|
fix improve variable names
|
||||||
|
print include system error message when output fails
|
||||||
|
drop a core when bug condition happens
|
||||||
|
make usage() easier to read
|
||||||
|
unify quotation in error messages
|
||||||
|
add --warn-count and --crit-count test, and fix related bug
|
||||||
|
test all sorting options
|
||||||
|
change NAN markup to make tests work on mac
|
||||||
|
fix out of tree build tests
|
||||||
|
release: 2.28
|
||||||
|
|
||||||
|
Version 2.27
|
||||||
|
Sami Kerola (9):
|
||||||
|
docs: remove very basic git usage info
|
||||||
|
docs: remove todo items that will never happen
|
||||||
|
output: remove empty element from xml
|
||||||
|
add touched addresses counts to xml and json reports
|
||||||
|
add xml format check
|
||||||
|
improve html output
|
||||||
|
add hint about configure options to README
|
||||||
|
man: fix character class change
|
||||||
|
gitignore: update gnulib file list
|
||||||
|
release: 2.27
|
||||||
|
|
||||||
|
Version 2.26
|
||||||
|
$ git shortlog v2.25..v2.26
|
||||||
|
Sami Kerola (15):
|
||||||
|
news: add the note about .sig pgp key id
|
||||||
|
reindent all files
|
||||||
|
declare global variables only once
|
||||||
|
add appropriate sorting function for struct leases_t
|
||||||
|
update kernel.org url in README
|
||||||
|
getdata: fix buffer-overflows reported by address sanitizer
|
||||||
|
update bootstrap from gnulib
|
||||||
|
getdata: fix buffer overflow [AddressSanitizer]
|
||||||
|
other: reduce variable scope
|
||||||
|
output: prefer thread safe function localtime_r()
|
||||||
|
tests: add regression test to avoid shared-net off by one alarming
|
||||||
|
issue
|
||||||
|
output: avoid division by zero
|
||||||
|
other: disallow unsigned counter ever to have minus value
|
||||||
|
maint: remove unnecessary braces, spaces, update gnulib .gitignore
|
||||||
|
release: 2.26
|
||||||
|
|
||||||
|
Wolfgang Steudel (1):
|
||||||
|
alarming: include last shared network in alarming [off by one]
|
||||||
|
|
||||||
|
Version 2.25
|
||||||
|
$ git shortlog v2.24..v2.25
|
||||||
|
Sami Kerola (14):
|
||||||
|
build-sys: use more strict method to check __builtin_expect
|
||||||
|
getdata: add missing HAVE_POSIX_FADVISE protection
|
||||||
|
getdata: fix consecutive range definition regression
|
||||||
|
man: add tip analysis of include files can be useful
|
||||||
|
thanks: add Fredrik Lysén & Conor McCarthy
|
||||||
|
generic: use pure and const function attributes when possible
|
||||||
|
tests: fix testing error
|
||||||
|
output: make nagios output have performance data
|
||||||
|
build-sys: update gitinore file
|
||||||
|
output: use symbolic exit values for nagios commands
|
||||||
|
build-sys: update bootstrap script
|
||||||
|
build-sys: require automake 1.12 to get working test-driver
|
||||||
|
contrib: add release siging key to contrib/
|
||||||
|
release: 2.25
|
||||||
|
|
||||||
|
Version 2.24
|
||||||
|
$ git shortlog v2.23..v2.24
|
||||||
|
Sami Kerola (22):
|
||||||
|
build-sys: update configure.ac
|
||||||
|
output: make output limits to apply to alarm return values
|
||||||
|
tests: add alarming tests
|
||||||
|
output: allow alarming, in case of success, be silent
|
||||||
|
tests: fix tests to use the new OK printing limit
|
||||||
|
docs: inform how the limits affect to printing when alarming is in use
|
||||||
|
output: allow user to ignore small ranges and shared networks
|
||||||
|
tests: test small network alarm ignoring
|
||||||
|
docs: add an example how to use limits with alarming
|
||||||
|
thanks: Dan Pritts
|
||||||
|
xstrstr: speed up analysis by avoiding string comparisons
|
||||||
|
other: memcmp() is tiny bit quicker than strncmp()
|
||||||
|
build-sys: update gnulib modules
|
||||||
|
build-sys: AC_PROG_CC_C99 is obsolete; use AC_PROG_CC
|
||||||
|
getdata: use correct file caching advice
|
||||||
|
analyze: remove unnecessary function
|
||||||
|
getdata: memccpy() is better than own similar function
|
||||||
|
getdata: split busy function to segments used via function pointer
|
||||||
|
various: split functions to IPv4 and IPv6 versions
|
||||||
|
various: fix memory leaks
|
||||||
|
getdata: fix one ip ranges when they have trailing spaces in definition
|
||||||
|
release: 2.24
|
||||||
|
|
||||||
|
Version 2.23
|
||||||
|
$ git shortlog v2.22..v2.23
|
||||||
|
Sami Kerola (3):
|
||||||
|
gnulib: add gnulib modules
|
||||||
|
build-sys: update bootstrap from gnulib
|
||||||
|
release: 2.23
|
||||||
|
|
||||||
|
Version 2.22
|
||||||
|
$ git shortlog v2.21..v2.22
|
||||||
|
Cheer Xiao (1):
|
||||||
|
getdata: treat binding states abandoned, expired and release like free
|
||||||
|
|
||||||
|
Sami Kerola (19):
|
||||||
|
tests: add complete check including multiple ranges and shared networks
|
||||||
|
include: add missing declarations to dhcpd-pool.h
|
||||||
|
cppcheck: fix variable scoping
|
||||||
|
build-sys: update .gitignore
|
||||||
|
build-sys: add git-version-gen and test.sh to release package
|
||||||
|
docs: fix spelling mistakes in manual page
|
||||||
|
build-sys: ensure distribution has version file
|
||||||
|
build-sys: update git-version-gen from gnulib
|
||||||
|
tests: binding states commit needs to be reflected in expected results
|
||||||
|
build-sys: reorder gnulib macro in relation to C standard
|
||||||
|
getdata: posix_fadvise() error checking
|
||||||
|
getdata: certain input files caused SIGSEGV
|
||||||
|
tests: check how command works with invalid input
|
||||||
|
getdata: make a lease address detection more robust
|
||||||
|
getdata: make single address ranges work
|
||||||
|
tests: add a test for one ip range
|
||||||
|
build-sys: fix gnulib function order
|
||||||
|
build-sys: use non-recursive build
|
||||||
|
release: 2.22
|
||||||
|
|
||||||
|
Version 2.21
|
||||||
|
$ git shortlog v2.20..v2.21
|
||||||
|
Cheer Xiao (10):
|
||||||
|
IPv6: add DHCPv6 support
|
||||||
|
sort: Use a comparer instead of returner
|
||||||
|
sort: Get rid of global comparer
|
||||||
|
counting & output: Convert all counters to double
|
||||||
|
fix: Put prototype of field_selector back to dhcpd-pools.h
|
||||||
|
sort: let field_selector return NULL at the end
|
||||||
|
counting: Let get_range_size return double
|
||||||
|
sort: let comp_* functions compare double
|
||||||
|
tests: update expected/v6
|
||||||
|
output: fix doc of output_double
|
||||||
|
|
||||||
|
Gilles Bouthenot (5):
|
||||||
|
output: add json format support
|
||||||
|
json output: add newline at end
|
||||||
|
json output: remove useless, empty fields
|
||||||
|
json output: bugfix: correct range specify only first ip
|
||||||
|
contrib: add munin plugins
|
||||||
|
|
||||||
|
Sami Kerola (29):
|
||||||
|
build-sys: remove c++ compiler check
|
||||||
|
tools: add indent command configuration file
|
||||||
|
tests: add basic tests
|
||||||
|
build-sys: update .gitignore
|
||||||
|
tests: add v6 check
|
||||||
|
output: remove unnecessary new line printing
|
||||||
|
build-sys: more clean ups
|
||||||
|
json output: use defined output file for all printing
|
||||||
|
tests: add json tests, and make fail output loud
|
||||||
|
ouput format: fix active lease printing
|
||||||
|
output ethernet: do not crash if ethernet is missing
|
||||||
|
docs: add doxygen tags, configuration file, and autotooling
|
||||||
|
backup counting: fix output regression
|
||||||
|
clean up: remove unused functions
|
||||||
|
clean up: move code to more proper location
|
||||||
|
hash: make delete to use HASH_ITER if it is available
|
||||||
|
clean up: move global variables to config structure
|
||||||
|
clean up: use enum as symbolic bit masks
|
||||||
|
clean up: convert parsing magic numbers to symbolic values
|
||||||
|
clean up: rename get_order() to merge() and mark static
|
||||||
|
clean up: fix few indentation slips
|
||||||
|
getdata: be consistant with symbolic numbers
|
||||||
|
smatch scan: fix few warnings
|
||||||
|
build-sys: allow tests run parallel
|
||||||
|
build-sys: update .gitignore files
|
||||||
|
docs: thanks to Cheer Xiao this software has IPv6 support
|
||||||
|
tests: fix make distcheck
|
||||||
|
docs: fix doxygen make distcheck breakage
|
||||||
|
release: 2.21
|
||||||
|
|
||||||
Version 2.20
|
Version 2.20
|
||||||
$ git shortlog v2.19..v2.20
|
$ git shortlog v2.19..v2.20
|
||||||
|
|
|
||||||
220
README
220
README
|
|
@ -1,196 +1,62 @@
|
||||||
This is dhcpd-pools which is made for ISC dhcpd pool range analysis.
|
This is dhcpd-pools - ISC dhcpd lease status utility.
|
||||||
|
|
||||||
== Quick start
|
Quick start.
|
||||||
|
|
||||||
./bootstrap
|
Get the uthash, assuming you do not have it already.
|
||||||
./configure --prefix=/usr/local
|
|
||||||
|
cd /tmp
|
||||||
|
wget https://github.com/troydhanson/uthash/archive/master.zip
|
||||||
|
unzip master.zip
|
||||||
|
|
||||||
|
Build the dhcpd-pools project.
|
||||||
|
|
||||||
|
cd /tmp/dhcpd-pools
|
||||||
|
./bootstrap # only when building git clone
|
||||||
|
./configure --with-uthash=/tmp/uthash-master/include
|
||||||
make
|
make
|
||||||
|
make check
|
||||||
make install
|
make install
|
||||||
|
|
||||||
|
Notice that there are configuration options for default dhcpd.conf,
|
||||||
|
dhcpd.leases paths, among other things.
|
||||||
|
|
||||||
== General information
|
./configure --help
|
||||||
|
|
||||||
With this command you can check usage of pool addresses
|
Remember to read the friendly manual page.
|
||||||
when address space is great. Command is designed so that
|
|
||||||
it will not get slow even there is thousands of IPs in
|
|
||||||
lease file. This kind of huge dhcpd installation can be
|
|
||||||
near by DSL DSLAMs or some other public access connection
|
|
||||||
points.
|
|
||||||
|
|
||||||
This command will not print nice to know information like
|
man ./man/dhcpd-pools.1
|
||||||
DHCPStatus does. Output is limited only to list only
|
|
||||||
usage for ranges, shared network and total address space.
|
|
||||||
Limiting what is printed is the right thing to do when
|
|
||||||
there is thousands of addresses. If your address space is
|
|
||||||
small some other dhcp analyzer might be more suitable for
|
|
||||||
you.
|
|
||||||
|
|
||||||
== Dependencies to other projects
|
Dependencies to other projects.
|
||||||
|
|
||||||
http://www.gnu.org/software/gnulib/
|
https://www.gnu.org/software/gnulib/
|
||||||
|
|
||||||
If you have gnulib checked out somewhere at file
|
You can avoid repeated gnulib downloads by setting
|
||||||
system you can avoid download by setting
|
GNULIB_SRCDIR environment variable. For example:
|
||||||
GNULIB_SRCDIR environment variable.
|
|
||||||
|
|
||||||
http://uthash.sourceforge.net/
|
git clone git://git.savannah.gnu.org/gnulib.git ~/src/gnulib
|
||||||
|
export GNULIB_SRCDIR="$HOME/src/gnulib"
|
||||||
|
|
||||||
== Test data wanted
|
Assumign detached gnulib please remember to git pull the
|
||||||
|
latest updates before building dhcpd-pools.
|
||||||
|
|
||||||
|
https://troydhanson.github.io/uthash/
|
||||||
|
|
||||||
|
See quick start.
|
||||||
|
|
||||||
|
https://getbootstrap.com/
|
||||||
|
https://datatables.net/
|
||||||
|
|
||||||
|
Bootstrap and DataTables java scripts are used in html
|
||||||
|
output.
|
||||||
|
|
||||||
|
Test data wanted.
|
||||||
|
|
||||||
Maintainer is interested to get copy of your dhcpd.conf
|
Maintainer is interested to get copy of your dhcpd.conf
|
||||||
and dhcpd.leases file, with includes if you use them.
|
and dhcpd.leases files, and include files if you use them.
|
||||||
Intention is to collect large set of data to build
|
Intention is to collect large set of data to build
|
||||||
realistic regression test environment. By giving your
|
realistic regression test environment.
|
||||||
data to maintainer you can be sure that updates will work
|
|
||||||
for you.
|
|
||||||
|
|
||||||
If you are interested to help this way put all files into
|
If you are interested to help this way put all files into
|
||||||
tar.gz, and send them to kerolasa@iki.fi. It would be nice
|
tar.gz, and send them to:
|
||||||
that email subject line would have 'dhcpd-pools test data'.
|
|
||||||
|
|
||||||
|
Sami Kerola <kerolasa@iki.fi>
|
||||||
== Instructions for developers
|
|
||||||
|
|
||||||
AUTOTOOLS:
|
|
||||||
|
|
||||||
* "./bootstrap" generates all files needed to compile and
|
|
||||||
install the code (run it after checkout from git)
|
|
||||||
|
|
||||||
* "make distclean" removes all unnecessary files, but the
|
|
||||||
code can still be recompiled with "./configure; make"
|
|
||||||
|
|
||||||
PATCHES:
|
|
||||||
|
|
||||||
* First get familiar with git. In case you are completely
|
|
||||||
lost watch Greg Kroah-Hartman explaining the very
|
|
||||||
basics.
|
|
||||||
http://archive.fosdem.org/2010/schedule/events/linuxkernelpatch
|
|
||||||
|
|
||||||
* Get up to date version of the code base.
|
|
||||||
|
|
||||||
$ git clone git://dhcpd-pools.git.sourceforge.net/gitroot/dhcpd-pools/dhcpd-pools
|
|
||||||
|
|
||||||
* Don't include generated (autotools) stuff to your
|
|
||||||
patches (hint: use git-clean [-X])
|
|
||||||
|
|
||||||
* Add a Signed-off-by line, use "git commit -s"
|
|
||||||
|
|
||||||
* Patches are delivered via email only. The following
|
|
||||||
commands will do the correct thing.
|
|
||||||
|
|
||||||
$ git format-patch -C origin/master..yourbranch -o ~/patches
|
|
||||||
|
|
||||||
When you send only one patch use the following.
|
|
||||||
|
|
||||||
$ git send-email --to kerolasa@iki.fi 0001*
|
|
||||||
|
|
||||||
The command above expects you have configured email
|
|
||||||
sending properly. See git.wiki for help.
|
|
||||||
https://git.wiki.kernel.org/index.php/GitTips#Mail
|
|
||||||
|
|
||||||
* One patch per email, with the changelog in the body of
|
|
||||||
the email.
|
|
||||||
|
|
||||||
* When you send series of pathes include introductory
|
|
||||||
message.
|
|
||||||
|
|
||||||
$ git send-email --compose --to kerolasa@iki.fi ~/00*
|
|
||||||
|
|
||||||
Good introductory message will have at least
|
|
||||||
|
|
||||||
-- snip
|
|
||||||
Your Name (3):
|
|
||||||
firstfile.c: short description
|
|
||||||
secondfile.c: another description
|
|
||||||
|
|
||||||
firstfile.c | 2 +-
|
|
||||||
secondfile.c | 2 +-
|
|
||||||
secondfile.c | 2 +-
|
|
||||||
2 files changed, 3 insertions(+), 3 deletions(-)
|
|
||||||
-- snip
|
|
||||||
|
|
||||||
Above introductory can be generated with git.
|
|
||||||
|
|
||||||
$ git shortlog master..yourbranch
|
|
||||||
$ git diff --stat master..yourbranch
|
|
||||||
|
|
||||||
* Subject: [PATCH] subsystem: description. Following
|
|
||||||
~/.gitconfig will help you a little.
|
|
||||||
|
|
||||||
-- snip
|
|
||||||
[user]
|
|
||||||
name = Your Name
|
|
||||||
email = your.name@example.com
|
|
||||||
|
|
||||||
[format]
|
|
||||||
subjectprefix = PATCH
|
|
||||||
numbered = auto
|
|
||||||
signoff = yes
|
|
||||||
|
|
||||||
[sendemail]
|
|
||||||
chainreplyto = false
|
|
||||||
cc = your.name@example.com
|
|
||||||
-- snip
|
|
||||||
|
|
||||||
* If someone else wrote the patch, they should be
|
|
||||||
credited (and blamed) for it. To communicate this, add
|
|
||||||
a line:
|
|
||||||
|
|
||||||
From: John Doe <jdoe@wherever.com>
|
|
||||||
|
|
||||||
The sign-off is a simple line at the end of the
|
|
||||||
explanation for the patch, which certifies that you
|
|
||||||
wrote it or otherwise have the right to pass it on as a
|
|
||||||
open-source patch. The rules are pretty simple: if you
|
|
||||||
can certify the below:
|
|
||||||
|
|
||||||
By making a contribution to this project, I certify
|
|
||||||
that:
|
|
||||||
|
|
||||||
(a) The contribution was created in whole or in
|
|
||||||
part by me and I have the right to submit it
|
|
||||||
under the open source license indicated in the
|
|
||||||
file; or
|
|
||||||
|
|
||||||
(b) The contribution is based upon previous work
|
|
||||||
that, to the best of my knowledge, is covered
|
|
||||||
under an appropriate open source license and I
|
|
||||||
have the right under that license to submit
|
|
||||||
that work with modifications, whether created
|
|
||||||
in whole or in part by me, under the same open
|
|
||||||
source license (unless I am permitted to submit
|
|
||||||
under a different license), as indicated in the
|
|
||||||
file; or
|
|
||||||
|
|
||||||
(c) The contribution was provided directly to me by
|
|
||||||
some other person who certified (a), (b) or (c)
|
|
||||||
and I have not modified it.
|
|
||||||
|
|
||||||
(d) I understand and agree that this project and
|
|
||||||
the contribution are public and that a record
|
|
||||||
of the contribution (including all personal
|
|
||||||
information I submit with it, including my
|
|
||||||
sign-off) is maintained indefinitely and may be
|
|
||||||
redistributed consistent with this project or
|
|
||||||
the open source license(s) involved.
|
|
||||||
|
|
||||||
then you just add a line saying
|
|
||||||
|
|
||||||
Signed-off-by: Random J Developer <random@developer.example.org>
|
|
||||||
|
|
||||||
using your real name (sorry, no pseudonyms or anonymous
|
|
||||||
contributions.)
|
|
||||||
|
|
||||||
* If the the business with git is too difficult just send
|
|
||||||
source code files as is as email attachment to
|
|
||||||
maintainer.
|
|
||||||
|
|
||||||
CODING STYLE:
|
|
||||||
|
|
||||||
* The preferred coding style is based on the linux kernel
|
|
||||||
Documentation/CodingStyle. For more details see:
|
|
||||||
|
|
||||||
http://git.kernel.org/?p=linux/kernel/git/torvalds/linux-2.6.git;a=blob_plain;f=Documentation/CodingStyle
|
|
||||||
|
|
||||||
* Source code is pretty printed by using two, and only
|
|
||||||
the two, indent command switches -kr -i8
|
|
||||||
|
|
|
||||||
40
THANKS
40
THANKS
|
|
@ -5,22 +5,50 @@ Project is maintained by Sami Kerola <kerolasa@iki.fi>
|
||||||
People who reported problems, give improvement suggestions or even
|
People who reported problems, give improvement suggestions or even
|
||||||
contributed code.
|
contributed code.
|
||||||
|
|
||||||
Otto J. Mäkelä
|
Otto J. Mäkelä
|
||||||
Mika Paananen
|
Mika Paananen
|
||||||
Frank Bulk
|
Frank Bulk
|
||||||
Roar Pettersen
|
Roar Pettersen
|
||||||
Jeff Wieland
|
Jeff Wieland
|
||||||
Rusty
|
Rusty
|
||||||
Fredrik Vöcks
|
Fredrik Vöcks
|
||||||
Dan Thorson
|
Dan Thorson
|
||||||
Stian Øvrevåge
|
Stian Øvrevåge
|
||||||
Dominic Germain
|
Dominic Germain
|
||||||
Anders Låstad
|
Anders Låstad
|
||||||
Thor Eivind Brantzeg
|
Thor Eivind Brantzeg
|
||||||
Ahmed AL Dakhil
|
Ahmed AL Dakhil
|
||||||
Adam Ciarcinski
|
Adam Ciarcinski
|
||||||
Rezso Gajdóczy
|
Rezső Gajdóczy
|
||||||
Robert Viou
|
Robert Viou
|
||||||
Enno Gröper
|
Enno Gröper
|
||||||
|
Troy D. Hanson
|
||||||
Ryan Malek
|
Ryan Malek
|
||||||
Cheer Xiao
|
Cheer Xiao
|
||||||
|
Gilles Bouthenot
|
||||||
|
Helmut Grohne
|
||||||
|
Joey D.
|
||||||
|
Ryan Steinmetz
|
||||||
|
Dan Pritts
|
||||||
|
Fredrik Lysén
|
||||||
|
Conor McCarthy
|
||||||
|
Wolfgang Steudel
|
||||||
|
Aaron Paetznick
|
||||||
|
Tim Cantin
|
||||||
|
Martijn van Brummelen
|
||||||
|
Anton Tkachev
|
||||||
|
Derrick Lin
|
||||||
|
Ivanov Ivan
|
||||||
|
Manuel Hachtkemper
|
||||||
|
Klaus Slott
|
||||||
|
Boris Lytochkin
|
||||||
|
Jeff Bailey
|
||||||
|
José Bollo
|
||||||
|
Sebastián Cramatte
|
||||||
|
Mark Sangster
|
||||||
|
Brent Swingle
|
||||||
|
Mathieu Morier
|
||||||
|
Jean Benoit
|
||||||
|
Belkacem Daheb
|
||||||
|
Björn Lässig
|
||||||
|
Luisδμ
|
||||||
|
|
|
||||||
38
TODO
38
TODO
|
|
@ -8,7 +8,6 @@ Reported-by: Huangyu: I found dhcpd-pools didn't consider if the
|
||||||
|
|
||||||
### Feature requests
|
### Feature requests
|
||||||
|
|
||||||
o IPv6 support (Frank)
|
|
||||||
o Assigned IPs count (Nelson)
|
o Assigned IPs count (Nelson)
|
||||||
o Subnet counting class (Henryk)
|
o Subnet counting class (Henryk)
|
||||||
o When time stamps are part of lease situation evaluation (see
|
o When time stamps are part of lease situation evaluation (see
|
||||||
|
|
@ -16,26 +15,19 @@ o When time stamps are part of lease situation evaluation (see
|
||||||
expiry happens.
|
expiry happens.
|
||||||
o Add lease time histogram support.
|
o Add lease time histogram support.
|
||||||
|
|
||||||
### Next major version
|
### When releasing
|
||||||
|
|
||||||
1 Well structured code, with extendible sane architecture.
|
o Update gnulib
|
||||||
2 Server, which has analysis in cache for N seconds and will
|
o Update bootstrap
|
||||||
answer via socket.
|
o Update DataTables, and bootstrap versions; see https://datatables.net/download/index
|
||||||
3 Configuration file for server.
|
o Update NEWS, webpages/index.html dateModified & version
|
||||||
4 State file, where different ranges, shared networks etc have
|
o Make annotated git tag
|
||||||
unique identifiers (needed for graphs).
|
o ./configure --enable-doxygen && make distcheck
|
||||||
5 Support for snmp and munin protocols.
|
o BROWSER=cat man -H ./man/dhcpd-pools.1 | tee ./webpages/man.html
|
||||||
6 SNMP traps when limits get exceeded.
|
o gpg --armor --detach-sign dhcpd-pools*tar*
|
||||||
7 Some clever check_dhcpd_pools binary for Nagios & other
|
* Update sitemap
|
||||||
monitoring software.
|
o Upload files to sourceforge
|
||||||
|
o mv doc/html webpages/doxygen
|
||||||
8 Super server support, which will collect analysis from other
|
o sftp index.html man.html doxygen/* -> sourceforge web
|
||||||
dhcpd-pool servers.
|
remove old doxygen before uploading new
|
||||||
9 Nice http view interface on super server, much can be copied
|
o Send email to dhcpd-pools-announce@lists.sourceforge.net subject: Version 3.n is released
|
||||||
from http://piwik.org/
|
|
||||||
10 IPv6 support.
|
|
||||||
11 And some other things perhaps...
|
|
||||||
|
|
||||||
When items 1-7 are done v3 can be released. To support rewrite
|
|
||||||
there should be a design document with nice blue print, which is
|
|
||||||
a web page.
|
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,10 @@
|
||||||
# Bootstrap configuration.
|
# Bootstrap configuration. -*- sh -*-
|
||||||
|
|
||||||
# Copyright (C) 2006-2012 Free Software Foundation, Inc.
|
# Copyright (C) 2006-2025 Free Software Foundation, Inc.
|
||||||
|
|
||||||
# This program is free software: you can redistribute it and/or modify
|
# This program is free software: you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU General Public License as published by
|
# it under the terms of the GNU General Public License as published by
|
||||||
# the Free Software Foundation; either version 3 of the License, or
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
# (at your option) any later version.
|
# (at your option) any later version.
|
||||||
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
|
@ -13,30 +13,36 @@
|
||||||
# GNU General Public License for more details.
|
# GNU General Public License for more details.
|
||||||
|
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
|
||||||
# gnulib modules used by this package.
|
# gnulib modules used by this package.
|
||||||
gnulib_modules="
|
gnulib_modules="
|
||||||
close-stream
|
close-stream
|
||||||
closeout
|
closeout
|
||||||
|
error
|
||||||
fclose
|
fclose
|
||||||
fcntl-h
|
fcntl-h
|
||||||
fdopen
|
fdopen
|
||||||
fflush
|
fflush
|
||||||
fopen
|
fopen
|
||||||
free
|
|
||||||
getopt-gnu
|
getopt-gnu
|
||||||
langinfo
|
inet_pton
|
||||||
|
isnan
|
||||||
netinet_in
|
netinet_in
|
||||||
|
nstrftime
|
||||||
|
progname
|
||||||
|
quote
|
||||||
realloc-gnu
|
realloc-gnu
|
||||||
stat
|
stat
|
||||||
stddef
|
stddef
|
||||||
stdio
|
stdio
|
||||||
stdlib
|
stdlib
|
||||||
|
stpncpy
|
||||||
strdup-posix
|
strdup-posix
|
||||||
strftime
|
|
||||||
strstr
|
strstr
|
||||||
|
strtod
|
||||||
|
time_r
|
||||||
xalloc
|
xalloc
|
||||||
"
|
"
|
||||||
|
|
||||||
|
|
@ -58,7 +64,7 @@ XGETTEXT_OPTIONS=$XGETTEXT_OPTIONS'\\\
|
||||||
gettext_external=0
|
gettext_external=0
|
||||||
grep '^[ ]*AM_GNU_GETTEXT(external\>' configure.ac > /dev/null &&
|
grep '^[ ]*AM_GNU_GETTEXT(external\>' configure.ac > /dev/null &&
|
||||||
gettext_external=1
|
gettext_external=1
|
||||||
grep '^[ ]*AM_GNU_GETTEXT(\[external\]' configure.ac > /dev/null &&
|
grep '^[ ]*AM_GNU_GETTEXT(\[external]' configure.ac > /dev/null &&
|
||||||
gettext_external=1
|
gettext_external=1
|
||||||
|
|
||||||
if test $gettext_external = 1; then
|
if test $gettext_external = 1; then
|
||||||
|
|
|
||||||
4
build-aux/.gitignore
vendored
4
build-aux/.gitignore
vendored
|
|
@ -1,8 +1,12 @@
|
||||||
|
/ar-lib
|
||||||
|
/compile
|
||||||
/config.guess
|
/config.guess
|
||||||
|
/config.rpath
|
||||||
/config.sub
|
/config.sub
|
||||||
/depcomp
|
/depcomp
|
||||||
/install-sh
|
/install-sh
|
||||||
/ltmain.sh
|
/ltmain.sh
|
||||||
/mdate-sh
|
/mdate-sh
|
||||||
/missing
|
/missing
|
||||||
|
/test-driver
|
||||||
/texinfo.tex
|
/texinfo.tex
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,12 @@
|
||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
# Print a version string.
|
# Print a version string.
|
||||||
scriptversion=2011-02-19.19; # UTC
|
scriptversion=2022-01-27.18; # UTC
|
||||||
|
|
||||||
# Copyright (C) 2007-2011 Free Software Foundation, Inc.
|
# Copyright (C) 2007-2022 Free Software Foundation, Inc.
|
||||||
#
|
#
|
||||||
# This program is free software: you can redistribute it and/or modify
|
# This program is free software: you can redistribute it and/or modify
|
||||||
# it under the terms of the GNU General Public License as published by
|
# it under the terms of the GNU General Public License as published by
|
||||||
# the Free Software Foundation; either version 3 of the License, or
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
# (at your option) any later version.
|
# (at your option) any later version.
|
||||||
#
|
#
|
||||||
# This program is distributed in the hope that it will be useful,
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
|
@ -15,9 +15,9 @@ scriptversion=2011-02-19.19; # UTC
|
||||||
# GNU General Public License for more details.
|
# GNU General Public License for more details.
|
||||||
#
|
#
|
||||||
# You should have received a copy of the GNU General Public License
|
# You should have received a copy of the GNU General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
# This script is derived from GIT-VERSION-GEN from GIT: http://git.or.cz/.
|
# This script is derived from GIT-VERSION-GEN from GIT: https://git-scm.com/.
|
||||||
# It may be run two ways:
|
# It may be run two ways:
|
||||||
# - from a git repository in which the "git describe" command below
|
# - from a git repository in which the "git describe" command below
|
||||||
# produces useful output (thus requiring at least one signed tag)
|
# produces useful output (thus requiring at least one signed tag)
|
||||||
|
|
@ -44,8 +44,10 @@ scriptversion=2011-02-19.19; # UTC
|
||||||
# files to pick up a version string change; and leave it stale to
|
# files to pick up a version string change; and leave it stale to
|
||||||
# minimize rebuild time after unrelated changes to configure sources.
|
# minimize rebuild time after unrelated changes to configure sources.
|
||||||
#
|
#
|
||||||
# It is probably wise to add these two files to .gitignore, so that you
|
# As with any generated file in a VC'd directory, you should add
|
||||||
# don't accidentally commit either generated file.
|
# /.version to .gitignore, so that you don't accidentally commit it.
|
||||||
|
# .tarball-version is never generated in a VC'd directory, so needn't
|
||||||
|
# be listed there.
|
||||||
#
|
#
|
||||||
# Use the following line in your configure.ac, so that $(VERSION) will
|
# Use the following line in your configure.ac, so that $(VERSION) will
|
||||||
# automatically be up-to-date each time configure is run (and note that
|
# automatically be up-to-date each time configure is run (and note that
|
||||||
|
|
@ -57,24 +59,76 @@ scriptversion=2011-02-19.19; # UTC
|
||||||
# [bug-project@example])
|
# [bug-project@example])
|
||||||
#
|
#
|
||||||
# Then use the following lines in your Makefile.am, so that .version
|
# Then use the following lines in your Makefile.am, so that .version
|
||||||
# will be present for dependencies, and so that .tarball-version will
|
# will be present for dependencies, and so that .version and
|
||||||
# exist in distribution tarballs.
|
# .tarball-version will exist in distribution tarballs.
|
||||||
#
|
#
|
||||||
|
# EXTRA_DIST = $(top_srcdir)/.version
|
||||||
# BUILT_SOURCES = $(top_srcdir)/.version
|
# BUILT_SOURCES = $(top_srcdir)/.version
|
||||||
# $(top_srcdir)/.version:
|
# $(top_srcdir)/.version:
|
||||||
# echo $(VERSION) > $@-t && mv $@-t $@
|
# echo '$(VERSION)' > $@-t
|
||||||
|
# mv $@-t $@
|
||||||
# dist-hook:
|
# dist-hook:
|
||||||
# echo $(VERSION) > $(distdir)/.tarball-version
|
# echo '$(VERSION)' > $(distdir)/.tarball-version
|
||||||
|
|
||||||
case $# in
|
|
||||||
1|2) ;;
|
|
||||||
*) echo 1>&2 "Usage: $0 \$srcdir/.tarball-version" \
|
|
||||||
'[TAG-NORMALIZATION-SED-SCRIPT]'
|
|
||||||
exit 1;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
tarball_version_file=$1
|
me=$0
|
||||||
tag_sed_script="${2:-s/x/x/}"
|
|
||||||
|
year=`expr "$scriptversion" : '\([^-]*\)'`
|
||||||
|
version="git-version-gen $scriptversion
|
||||||
|
|
||||||
|
Copyright (C) ${year} Free Software Foundation, Inc.
|
||||||
|
License GPLv3+: GNU GPL version 3 or later <https://gnu.org/licenses/gpl.html>.
|
||||||
|
This is free software: you are free to change and redistribute it.
|
||||||
|
There is NO WARRANTY, to the extent permitted by law."
|
||||||
|
|
||||||
|
usage="\
|
||||||
|
Usage: $me [OPTION]... \$srcdir/.tarball-version [TAG-NORMALIZATION-SED-SCRIPT]
|
||||||
|
Print a version string.
|
||||||
|
|
||||||
|
Options:
|
||||||
|
|
||||||
|
--prefix PREFIX prefix of git tags (default 'v')
|
||||||
|
--fallback VERSION
|
||||||
|
fallback version to use if \"git --version\" fails
|
||||||
|
|
||||||
|
--help display this help and exit
|
||||||
|
--version output version information and exit
|
||||||
|
|
||||||
|
Running without arguments will suffice in most cases."
|
||||||
|
|
||||||
|
prefix=v
|
||||||
|
fallback=
|
||||||
|
|
||||||
|
while test $# -gt 0; do
|
||||||
|
case $1 in
|
||||||
|
--help) echo "$usage"; exit 0;;
|
||||||
|
--version) echo "$version"; exit 0;;
|
||||||
|
--prefix) shift; prefix=${1?};;
|
||||||
|
--fallback) shift; fallback=${1?};;
|
||||||
|
-*)
|
||||||
|
echo "$0: Unknown option '$1'." >&2
|
||||||
|
echo "$0: Try '--help' for more information." >&2
|
||||||
|
exit 1;;
|
||||||
|
*)
|
||||||
|
if test "x$tarball_version_file" = x; then
|
||||||
|
tarball_version_file="$1"
|
||||||
|
elif test "x$tag_sed_script" = x; then
|
||||||
|
tag_sed_script="$1"
|
||||||
|
else
|
||||||
|
echo "$0: extra non-option argument '$1'." >&2
|
||||||
|
exit 1
|
||||||
|
fi;;
|
||||||
|
esac
|
||||||
|
shift
|
||||||
|
done
|
||||||
|
|
||||||
|
if test "x$tarball_version_file" = x; then
|
||||||
|
echo "$usage"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
tag_sed_script="${tag_sed_script:-s/x/x/}"
|
||||||
|
|
||||||
nl='
|
nl='
|
||||||
'
|
'
|
||||||
|
|
||||||
|
|
@ -89,25 +143,23 @@ then
|
||||||
v=`cat $tarball_version_file` || v=
|
v=`cat $tarball_version_file` || v=
|
||||||
case $v in
|
case $v in
|
||||||
*$nl*) v= ;; # reject multi-line output
|
*$nl*) v= ;; # reject multi-line output
|
||||||
[0-9]*) ;;
|
|
||||||
*) v= ;;
|
|
||||||
esac
|
esac
|
||||||
test -z "$v" \
|
test "x$v" = x \
|
||||||
&& echo "$0: WARNING: $tarball_version_file is missing or damaged" 1>&2
|
&& echo "$0: WARNING: $tarball_version_file is damaged" 1>&2
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if test -n "$v"
|
if test "x$v" != x
|
||||||
then
|
then
|
||||||
: # use $v
|
: # use $v
|
||||||
# Otherwise, if there is at least one git commit involving the working
|
# Otherwise, if there is at least one git commit involving the working
|
||||||
# directory, and "git describe" output looks sensible, use that to
|
# directory, and "git describe" output looks sensible, use that to
|
||||||
# derive a version string.
|
# derive a version string.
|
||||||
elif test "`git log -1 --pretty=format:x . 2>&1`" = x \
|
elif test "`git log -1 --pretty=format:x . 2>&1`" = x \
|
||||||
&& v=`git describe --tags --abbrev=4 --match='v*' HEAD 2>/dev/null \
|
&& v=`git describe --abbrev=4 --match="$prefix*" HEAD 2>/dev/null \
|
||||||
|| git describe --tags --abbrev=4 HEAD 2>/dev/null` \
|
|| git describe --abbrev=4 HEAD 2>/dev/null` \
|
||||||
&& v=`printf '%s\n' "$v" | sed "$tag_sed_script"` \
|
&& v=`printf '%s\n' "$v" | sed "$tag_sed_script"` \
|
||||||
&& case $v in
|
&& case $v in
|
||||||
v[0-9]*) ;;
|
$prefix[0-9]*) ;;
|
||||||
*) (exit 1) ;;
|
*) (exit 1) ;;
|
||||||
esac
|
esac
|
||||||
then
|
then
|
||||||
|
|
@ -115,9 +167,10 @@ then
|
||||||
# tag or the previous older version that did not?
|
# tag or the previous older version that did not?
|
||||||
# Newer: v6.10-77-g0f8faeb
|
# Newer: v6.10-77-g0f8faeb
|
||||||
# Older: v6.10-g0f8faeb
|
# Older: v6.10-g0f8faeb
|
||||||
case $v in
|
vprefix=`expr "X$v" : 'X\(.*\)-g[^-]*$'` || vprefix=$v
|
||||||
*-*-*) : git describe is okay three part flavor ;;
|
case $vprefix in
|
||||||
*-*)
|
*-*) : git describe is probably okay three part flavor ;;
|
||||||
|
*)
|
||||||
: git describe is older two part flavor
|
: git describe is older two part flavor
|
||||||
# Recreate the number of commits and rewrite such that the
|
# Recreate the number of commits and rewrite such that the
|
||||||
# result is the same as if we were using the newer version
|
# result is the same as if we were using the newer version
|
||||||
|
|
@ -132,21 +185,23 @@ then
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
# Change the first '-' to a '.', so version-comparing tools work properly.
|
# Change the penultimate "-" to ".", for version-comparing tools.
|
||||||
# Remove the "g" in git describe's output string, to save a byte.
|
# Remove the "g" to save a byte.
|
||||||
v=`echo "$v" | sed 's/-/./;s/\(.*\)-g/\1-/'`;
|
v=`echo "$v" | sed 's/-\([^-]*\)-g\([^-]*\)$/.\1-\2/'`;
|
||||||
v_from_git=1
|
v_from_git=1
|
||||||
else
|
elif test "x$fallback" = x || git --version >/dev/null 2>&1; then
|
||||||
v=UNKNOWN
|
v=UNKNOWN
|
||||||
|
else
|
||||||
|
v=$fallback
|
||||||
fi
|
fi
|
||||||
|
|
||||||
v=`echo "$v" |sed 's/^v//'`
|
v=`echo "$v" |sed "s/^$prefix//"`
|
||||||
|
|
||||||
# Test whether to append the "-dirty" suffix only if the version
|
# Test whether to append the "-dirty" suffix only if the version
|
||||||
# string we're using came from git. I.e., skip the test if it's "UNKNOWN"
|
# string we're using came from git. I.e., skip the test if it's "UNKNOWN"
|
||||||
# or if it came from .tarball-version.
|
# or if it came from .tarball-version.
|
||||||
if test -n "$v_from_git"; then
|
if test "x$v_from_git" != x; then
|
||||||
# Don't declare a version "dirty" merely because a time stamp has changed.
|
# Don't declare a version "dirty" merely because a timestamp has changed.
|
||||||
git update-index --refresh > /dev/null 2>&1
|
git update-index --refresh > /dev/null 2>&1
|
||||||
|
|
||||||
dirty=`exec 2>/dev/null;git diff-index --name-only HEAD` || dirty=
|
dirty=`exec 2>/dev/null;git diff-index --name-only HEAD` || dirty=
|
||||||
|
|
@ -161,12 +216,12 @@ if test -n "$v_from_git"; then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Omit the trailing newline, so that m4_esyscmd can use the result directly.
|
# Omit the trailing newline, so that m4_esyscmd can use the result directly.
|
||||||
echo "$v" | tr -d "$nl"
|
printf %s "$v"
|
||||||
|
|
||||||
# Local variables:
|
# Local variables:
|
||||||
# eval: (add-hook 'write-file-hooks 'time-stamp)
|
# eval: (add-hook 'before-save-hook 'time-stamp)
|
||||||
# time-stamp-start: "scriptversion="
|
# time-stamp-start: "scriptversion="
|
||||||
# time-stamp-format: "%:y-%02m-%02d.%02H"
|
# time-stamp-format: "%:y-%02m-%02d.%02H"
|
||||||
# time-stamp-time-zone: "UTC"
|
# time-stamp-time-zone: "UTC0"
|
||||||
# time-stamp-end: "; # UTC"
|
# time-stamp-end: "; # UTC"
|
||||||
# End:
|
# End:
|
||||||
|
|
|
||||||
116
configure.ac
116
configure.ac
|
|
@ -1,29 +1,40 @@
|
||||||
# -*- Autoconf -*-
|
# -*- Autoconf -*-
|
||||||
# Process this file with autoconf to produce a configure script.
|
# Process this file with autoconf to produce a configure script.
|
||||||
|
|
||||||
AC_PREREQ(2.64)
|
AC_PREREQ([2.69])
|
||||||
AC_CONFIG_MACRO_DIR([m4])
|
AC_CONFIG_MACRO_DIRS([m4])
|
||||||
AC_INIT([dhcpd-pools],
|
AC_INIT([dhcpd-pools],
|
||||||
m4_esyscmd([build-aux/git-version-gen .tarball-version]),
|
[m4_esyscmd([build-aux/git-version-gen .tarball-version])],
|
||||||
[kerolasa@iki.fi],,[http://dhcpd-pools.sourceforge.net/])
|
[kerolasa@iki.fi],[],
|
||||||
|
[https://dhcpd-pools.sourceforge.net/])
|
||||||
PACKAGE_MAINTAINER="Sami Kerola"
|
PACKAGE_MAINTAINER="Sami Kerola"
|
||||||
AC_SUBST([PACKAGE_MAINTAINER])
|
AC_SUBST([PACKAGE_MAINTAINER])
|
||||||
AC_CONFIG_AUX_DIR([build-aux])
|
AC_CONFIG_AUX_DIR([build-aux])
|
||||||
AM_INIT_AUTOMAKE([-Wall foreign 1.10 tar-pax no-dist-gzip dist-xz -Wextra-portability])
|
AM_INIT_AUTOMAKE([
|
||||||
|
-Wall
|
||||||
|
-Wextra-portability
|
||||||
|
1.12
|
||||||
|
foreign
|
||||||
|
dist-xz
|
||||||
|
no-dist-gzip
|
||||||
|
parallel-tests
|
||||||
|
subdir-objects
|
||||||
|
tar-pax
|
||||||
|
])
|
||||||
AC_CONFIG_SRCDIR([src/dhcpd-pools.h])
|
AC_CONFIG_SRCDIR([src/dhcpd-pools.h])
|
||||||
AC_CONFIG_HEADERS([config.h])
|
AC_CONFIG_HEADERS([config.h])
|
||||||
|
|
||||||
AC_GNU_SOURCE
|
m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])],
|
||||||
|
[AC_SUBST([AM_DEFAULT_VERBOSITY], [1])])
|
||||||
|
|
||||||
# Checks for programs
|
# Checks for programs
|
||||||
AC_PROG_AWK
|
AC_USE_SYSTEM_EXTENSIONS
|
||||||
AC_PROG_CC_C99
|
AC_C_RESTRICT
|
||||||
|
AC_PROG_CC
|
||||||
gl_EARLY
|
gl_EARLY
|
||||||
gl_INIT
|
gl_INIT
|
||||||
AC_C_RESTRICT
|
|
||||||
AC_PROG_CXX
|
|
||||||
AC_PROG_MAKE_SET
|
AC_PROG_MAKE_SET
|
||||||
AC_PROG_LIBTOOL
|
LT_INIT
|
||||||
|
|
||||||
# Checks for libraries.
|
# Checks for libraries.
|
||||||
LT_INIT
|
LT_INIT
|
||||||
|
|
@ -37,20 +48,16 @@ AC_CHECK_HEADERS([\
|
||||||
sys/socket.h \
|
sys/socket.h \
|
||||||
])
|
])
|
||||||
|
|
||||||
AC_ARG_WITH(uthash,
|
AC_ARG_WITH(
|
||||||
AC_HELP_STRING([--with-uthash=DIR], [Use uthash from [DIR]/uthash.h)]), [
|
[uthash],
|
||||||
case "$with_uthash" in
|
[AS_HELP_STRING([--with-uthash=DIR], [Use uthash from [DIR]/uthash.h)])],
|
||||||
yes)
|
[AS_CASE([$with_uthash],
|
||||||
;;
|
[yes], [],
|
||||||
no)
|
[no], [AC_MSG_ERROR([the uthash is required])],
|
||||||
AC_MSG_ERROR([the uthash.h is required])
|
[CPPFLAGS="$CPPFLAGS -I$with_uthash"]
|
||||||
;;
|
)]
|
||||||
*)
|
)
|
||||||
CPPFLAGS="$CPPFLAGS -I$with_uthash"
|
AC_CHECK_HEADER([uthash.h], [], [AC_MSG_ERROR([Unable to find uthash.h])])
|
||||||
;;
|
|
||||||
esac
|
|
||||||
])
|
|
||||||
AC_CHECK_HEADER(uthash.h, [], [AC_MSG_ERROR([Unable to find uthash.h])])
|
|
||||||
|
|
||||||
# Checks for typedefs, structures, and compiler characteristics.
|
# Checks for typedefs, structures, and compiler characteristics.
|
||||||
AC_C_INLINE
|
AC_C_INLINE
|
||||||
|
|
@ -62,16 +69,32 @@ AC_TYPE_UINT32_T
|
||||||
AC_FUNC_ERROR_AT_LINE
|
AC_FUNC_ERROR_AT_LINE
|
||||||
AC_CHECK_FUNCS([\
|
AC_CHECK_FUNCS([\
|
||||||
__fpending \
|
__fpending \
|
||||||
|
open_memstream \
|
||||||
posix_fadvise \
|
posix_fadvise \
|
||||||
])
|
])
|
||||||
|
AC_CHECK_DECL([strndupa])
|
||||||
|
|
||||||
|
AC_CHECK_FUNCS([fpclassify], [],
|
||||||
|
[AC_CHECK_LIB([m], [fpclassify], [MATH_LIBS="-lm"])]
|
||||||
|
[AC_CHECK_LIB([m], [__fpclassify], [MATH_LIBS="-lm"])]
|
||||||
|
)
|
||||||
|
AC_SUBST([MATH_LIBS])
|
||||||
|
|
||||||
|
AS_IF([test "x$ac_cv_func_open_memstream" = "xyes" && test "x$ac_cv_have_decl_strndupa" == "xyes"], [
|
||||||
|
build_mustach=yes
|
||||||
|
AC_DEFINE([BUILD_MUSTACH], [1], [build mustach support])
|
||||||
|
], [
|
||||||
|
build_mustach=no
|
||||||
|
])
|
||||||
|
AM_CONDITIONAL([ENABLE_MUSTACH], [test "x$build_mustach" = xyes])
|
||||||
|
|
||||||
AC_MSG_CHECKING([if the compiler supports __builtin_expect])
|
AC_MSG_CHECKING([if the compiler supports __builtin_expect])
|
||||||
AC_TRY_COMPILE(, [
|
AC_LINK_IFELSE([AC_LANG_PROGRAM([[]], [[
|
||||||
return __builtin_expect(1, 1) ? 1 : 0
|
return __builtin_expect(1, 1) ? 1 : 0
|
||||||
], [
|
]])],[
|
||||||
have_builtin_expect=yes
|
have_builtin_expect=yes
|
||||||
AC_MSG_RESULT([yes])
|
AC_MSG_RESULT([yes])
|
||||||
], [
|
],[
|
||||||
have_builtin_expect=no
|
have_builtin_expect=no
|
||||||
AC_MSG_RESULT([no])
|
AC_MSG_RESULT([no])
|
||||||
])
|
])
|
||||||
|
|
@ -81,54 +104,69 @@ AS_IF([test x$have_builtin_expect = xyes], [
|
||||||
])
|
])
|
||||||
|
|
||||||
AC_ARG_WITH([dhcpd-conf],
|
AC_ARG_WITH([dhcpd-conf],
|
||||||
[AC_HELP_STRING([--with-dhcpd-conf=FILE], [default path of dhcpd.conf])],
|
[AS_HELP_STRING([--with-dhcpd-conf=FILE],[default path of dhcpd.conf])],
|
||||||
[dhcpd_conf_path="$withval"],
|
[dhcpd_conf_path="$withval"],
|
||||||
[dhcpd_conf_path="/etc/dhcpd.conf"])
|
[dhcpd_conf_path="/etc/dhcpd.conf"])
|
||||||
AC_DEFINE_UNQUOTED([DHCPDCONF_FILE], ["$dhcpd_conf_path"], [default path of dhcpd.conf])
|
AC_DEFINE_UNQUOTED([DHCPDCONF_FILE], ["$dhcpd_conf_path"], [default path of dhcpd.conf])
|
||||||
AC_SUBST([DHCPDCONF_FILE], [$dhcpd_conf_path])
|
AC_SUBST([DHCPDCONF_FILE], [$dhcpd_conf_path])
|
||||||
|
|
||||||
AC_ARG_WITH([dhcpd-leases],
|
AC_ARG_WITH([dhcpd-leases],
|
||||||
[AC_HELP_STRING([--with-dhcpd-leases=FILE], [default path of dhcpd.leases])],
|
[AS_HELP_STRING([--with-dhcpd-leases=FILE],[default path of dhcpd.leases])],
|
||||||
[dhcpd_leases_path="$withval"],
|
[dhcpd_leases_path="$withval"],
|
||||||
[dhcpd_leases_path="/var/lib/dhcp/dhcpd.leases"])
|
[dhcpd_leases_path="/var/lib/dhcp/dhcpd.leases"])
|
||||||
AC_DEFINE_UNQUOTED([DHCPDLEASE_FILE], ["$dhcpd_leases_path"], [default path of dhcpd.leases])
|
AC_DEFINE_UNQUOTED([DHCPDLEASE_FILE], ["$dhcpd_leases_path"], [default path of dhcpd.leases])
|
||||||
AC_SUBST([DHCPDLEASE_FILE], [$dhcpd_leases_path])
|
AC_SUBST([DHCPDLEASE_FILE], [$dhcpd_leases_path])
|
||||||
|
|
||||||
AC_ARG_WITH([output-format],
|
AC_ARG_WITH([output-format],
|
||||||
[AC_HELP_STRING([--with-output-format=FORMAT], [default output format])],
|
[AS_HELP_STRING([--with-output-format=FORMAT],[default output format])],
|
||||||
[output_format="$withval"],
|
[output_format="$withval"],
|
||||||
[output_format="text"])
|
[output_format="text"])
|
||||||
AC_DEFINE_UNQUOTED([OUTPUT_FORMAT], ["$output_format"], [default output format])
|
AC_DEFINE_UNQUOTED([OUTPUT_FORMAT], ["$output_format"], [default output format])
|
||||||
AC_SUBST([OUTPUT_FORMAT], [$output_format])
|
AC_SUBST([OUTPUT_FORMAT], [$output_format])
|
||||||
|
|
||||||
AC_ARG_WITH([output-limit],
|
AC_ARG_WITH([output-limit],
|
||||||
[AC_HELP_STRING([--with-output-limit=MASK], [default output limit])],
|
[AS_HELP_STRING([--with-output-limit=MASK],[default output limit])],
|
||||||
[output_limit="$withval"],
|
[output_limit="$withval"],
|
||||||
[output_limit="77"])
|
[output_limit="77"])
|
||||||
AC_DEFINE_UNQUOTED([OUTPUT_LIMIT], ["$output_limit"], [default output limit])
|
AC_DEFINE_UNQUOTED([OUTPUT_LIMIT], ["$output_limit"], [default output limit])
|
||||||
AC_SUBST([OUTPUT_LIMIT], [$output_limit])
|
AC_SUBST([OUTPUT_LIMIT], [$output_limit])
|
||||||
|
|
||||||
AC_ARG_WITH([warning],
|
AC_ARG_WITH([warning],
|
||||||
[AC_HELP_STRING([--with-warning=NUM], [monitoring warning default])],
|
[AS_HELP_STRING([--with-warning=NUM],[monitoring warning default])],
|
||||||
[warning_percent="$withval"],
|
[warning_percent="$withval"],
|
||||||
[warning_percent="80"])
|
[warning_percent="80"])
|
||||||
AC_DEFINE_UNQUOTED([ALARM_WARN], [$warning_percent], [monitoring warning default])
|
AC_DEFINE_UNQUOTED([ALARM_WARN], [$warning_percent], [monitoring warning default])
|
||||||
AC_SUBST([ALARM_WARN], [$warning_percent])
|
AC_SUBST([ALARM_WARN], [$warning_percent])
|
||||||
|
|
||||||
AC_ARG_WITH([critical],
|
AC_ARG_WITH([critical],
|
||||||
[AC_HELP_STRING([--with-critical=NUM], [monitoring critical default])],
|
[AS_HELP_STRING([--with-critical=NUM],[monitoring critical default])],
|
||||||
[critical_percent="$withval"],
|
[critical_percent="$withval"],
|
||||||
[critical_percent="90"])
|
[critical_percent="90"])
|
||||||
AC_DEFINE_UNQUOTED([ALARM_CRIT], [$critical_percent], [monitoring critical default])
|
AC_DEFINE_UNQUOTED([ALARM_CRIT], [$critical_percent], [monitoring critical default])
|
||||||
AC_SUBST([ALARM_CRIT], [$critical_percent])
|
AC_SUBST([ALARM_CRIT], [$critical_percent])
|
||||||
|
|
||||||
|
AC_ARG_ENABLE([doxygen],
|
||||||
|
AS_HELP_STRING([--enable-doxygen],[build internal api documentation @<:@default=disabled@:>@]),
|
||||||
|
[], [enable_doxygen=no])
|
||||||
|
AM_CONDITIONAL([ENABLE_DOXYGEN], [test "x$enable_doxygen" = "xyes"])
|
||||||
|
|
||||||
|
AS_IF([test x$enable_doxygen = xyes], [
|
||||||
|
AC_CHECK_PROGS([DOXYGEN], [doxygen])
|
||||||
|
AS_IF([test "x$DOXYGEN" = "x"],
|
||||||
|
AC_MSG_ERROR([doxygen not in path])
|
||||||
|
)
|
||||||
|
|
||||||
|
AC_CHECK_PROGS([DOXYGEN_DOT], [dot])
|
||||||
|
AS_IF([test "x$DOXYGEN" = "x"],
|
||||||
|
AC_MSG_WARN([dot not in path, internal documentation graphs are not generated])
|
||||||
|
)
|
||||||
|
])
|
||||||
|
AM_CONDITIONAL([HAVE_DOXYGEN], [test "x$DOXYGEN" != "x"])
|
||||||
|
AM_CONDITIONAL([HAVE_DOXYGEN_DOT], [test "x$DOXYGEN_DOT" != "x"])
|
||||||
|
|
||||||
AC_CONFIG_FILES([
|
AC_CONFIG_FILES([
|
||||||
Makefile
|
Makefile
|
||||||
lib/Makefile
|
lib/Makefile
|
||||||
src/Makefile
|
|
||||||
man/Makefile
|
|
||||||
man/dhcpd-pools.1
|
|
||||||
contrib/Makefile
|
|
||||||
])
|
])
|
||||||
|
|
||||||
AC_OUTPUT
|
AC_OUTPUT
|
||||||
|
|
|
||||||
1
contrib/.gitignore
vendored
Normal file
1
contrib/.gitignore
vendored
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
/nagios.conf
|
||||||
|
|
@ -1,3 +0,0 @@
|
||||||
contribdir = $(datadir)/dhcpd-pools/
|
|
||||||
dist_contrib_SCRIPTS = dhcpd-pools.cgi snmptest.pl
|
|
||||||
EXTRA_DIST = nagios.conf
|
|
||||||
5
contrib/Makemodule.am
Normal file
5
contrib/Makemodule.am
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
contribdir = $(datadir)/dhcpd-pools/
|
||||||
|
PATHFILES += contrib/nagios.conf
|
||||||
|
dist_contrib_SCRIPTS = contrib/snmptest.pl
|
||||||
|
dist_contrib_DATA = contrib/nagios.conf
|
||||||
|
EXTRA_DIST += contrib/munin_plugins
|
||||||
36
contrib/PKGBUILD
Normal file
36
contrib/PKGBUILD
Normal file
|
|
@ -0,0 +1,36 @@
|
||||||
|
# Archlinux package file. Just download this file, and
|
||||||
|
# makepkg PKGBUILD
|
||||||
|
# pacman -U ./dhcpd-pools*.pkg.tar.xz
|
||||||
|
|
||||||
|
pkgname=dhcpd-pools
|
||||||
|
pkgver=0
|
||||||
|
pkgrel=1
|
||||||
|
pkgdesc="ISC dhcpd lease status utility"
|
||||||
|
arch=('i686' 'x86_64')
|
||||||
|
url=https://dhcpd-pools.sourceforge.net/
|
||||||
|
license=('BSD')
|
||||||
|
depends=('pacman')
|
||||||
|
makedepends=('uthash' 'git')
|
||||||
|
source=("$pkgname"::'git://git.code.sf.net/p/dhcpd-pools/code')
|
||||||
|
md5sums=('SKIP')
|
||||||
|
|
||||||
|
pkgver() {
|
||||||
|
cd "$srcdir/$pkgname"
|
||||||
|
# Use the tag of the last commit
|
||||||
|
git describe --long | sed -E 's/([^-]*-g)/r\1/;s/-/./g'
|
||||||
|
}
|
||||||
|
|
||||||
|
build() {
|
||||||
|
cd "$srcdir/$pkgname"
|
||||||
|
./bootstrap
|
||||||
|
./configure \
|
||||||
|
--prefix=/usr \
|
||||||
|
--bindir=/usr/bin
|
||||||
|
make
|
||||||
|
}
|
||||||
|
|
||||||
|
package() {
|
||||||
|
cd "$srcdir/$pkgname"
|
||||||
|
make PREFIX=/ DESTDIR="$pkgdir" install
|
||||||
|
install -D -m644 COPYING "${pkgdir}/usr/share/licenses/${pkgname}/COPYING"
|
||||||
|
}
|
||||||
|
|
@ -1,34 +0,0 @@
|
||||||
#!/bin/sh
|
|
||||||
#
|
|
||||||
# Simple CGI for dhcpd-pools.
|
|
||||||
|
|
||||||
echo Content-type: text/html
|
|
||||||
echo
|
|
||||||
|
|
||||||
# To make lease table more fancy use CSS definition something
|
|
||||||
# like this in your style.css file.
|
|
||||||
#
|
|
||||||
# TABLE.dhcpd-pools {
|
|
||||||
# border-style : groove;
|
|
||||||
# margin-left : 2px;
|
|
||||||
# foo : bar;
|
|
||||||
# }
|
|
||||||
#
|
|
||||||
# http://www.w3.org/TR/REC-CSS2/tables.html
|
|
||||||
#
|
|
||||||
# And uncomment this line.
|
|
||||||
#
|
|
||||||
#echo <link type="text/css" rel="stylesheet" href="/style.css" />
|
|
||||||
|
|
||||||
echo "<html>"
|
|
||||||
echo "<body>"
|
|
||||||
echo "<p>This was situation at "
|
|
||||||
date
|
|
||||||
echo "</p>"
|
|
||||||
|
|
||||||
/usr/local/bin/dhcpd-pools --format html
|
|
||||||
|
|
||||||
echo "</body>"
|
|
||||||
echo "</html>"
|
|
||||||
|
|
||||||
# EOF
|
|
||||||
111
contrib/kerolasa.gpg
Normal file
111
contrib/kerolasa.gpg
Normal file
|
|
@ -0,0 +1,111 @@
|
||||||
|
-----BEGIN PGP PUBLIC KEY BLOCK-----
|
||||||
|
|
||||||
|
mQINBFLH/jcBEADNgkgZAr7b/qvHzNkme8eSplDA8UHhcCFQmBnQ8HYFEYBi1Y5a
|
||||||
|
GhICcuisRSuZIPLtqRF8ntoOXsOHOUxnn5hVR62HxMLgOC1PQMw8TzREobbkAxMp
|
||||||
|
Vg66c6JXD2+jEGuAbhRk1XyWJHaCN2ewT06ToqGi7jhlw6nz1AHyohhyxeRlyMff
|
||||||
|
a4daTIRREnh2qouImSOE0E8ETAb0eOnFOsweHhmbpC/RDzuY7Ns8YcNH3FbWT1Kl
|
||||||
|
4W9vOstB1JcgSz38/tnzoMG2Mf2jI8LgR+6xiHTb7i6bgyWiBOxcJWouTVam6SC0
|
||||||
|
Bc5cAe8EqHT6NMEjtNjDEzmb3ZFpFUnNKgKK5Wzghf17HAvYwpjYxNNoQl9jNIkf
|
||||||
|
XroquAyDDXW0IkNdGDARjJGW17r38oRH/R3q79DZNlqV/DPk4YLb0EAKzRAdBYGm
|
||||||
|
v8Bj+uLXc4JT9kTB5E8DT2ZR+61E4eT/zk4rpZ5j6tTktLNaSG58VYc+oQlPkPsB
|
||||||
|
ntraDsORpa5cK9uPUbHIHN1qd4gi35O4UDSqS023XnStQqsUD4vlzICedPDypn28
|
||||||
|
h9q2nlNQHSDjIRnZFD+z9IHfVoBNTfm5/UH7NKBAvSS6rP5zsgi2fittt5AFdbmQ
|
||||||
|
VBOtpYp5vbcdSt8gNdJRy8FRaiQ/2n74xF1Zso/PGni1xRmseS+qBGI95QARAQAB
|
||||||
|
tB1TYW1pIEtlcm9sYSA8a2Vyb2xhc2FAaWtpLmZpPokCcwQTAQgAXQIbAwULCQgH
|
||||||
|
AgYVCAkKCwIEFgIDAQIeAQIXgAIZARsYaHR0cDovL2h0dHAta2V5cy5nbnVwZy5u
|
||||||
|
ZXQWIQSO05bjfjjUcaAFMNOpVTJF/em3OQUCWb42mgUJGcI7YwAKCRCpVTJF/em3
|
||||||
|
OWNBD/4/N0OO6cA3bmEAum5qmHdcJVcUor9ClY3oY2QWdTyDCmSJLcTZUNsTrYYA
|
||||||
|
hY5vQsDxG6ZaKiITvhwp9y4PdJmnqqzRfnqnXusSVEsqG3vGHIVlpQj8Vgurr68d
|
||||||
|
1x2buHGRvRrjBMWR/Ro1TUeNOmM4HStcvjXRTWlHk9VXi9nk+uMQY9n1lXYNChda
|
||||||
|
0usj7q5QXXFXbOukKrvEZzt4I1tGbSEv7bBIghr5fMAJg8ZqAMDwfqsUKSM38HF4
|
||||||
|
leCUgMZS7XuXuEb35ImjR1VYYCb+L2Q/PFqDOEGQpUwyLOzXcU9TaIDjMhNQY19y
|
||||||
|
DTUjW1YdQtf+3XmQFa4X1Od0jP1PiVJhw1LNCQLOENs+SRqT9s0W8+nP3qjso/LR
|
||||||
|
T3vR3FcGSw6KrESMFvmwY7EEYGvOpEgaIv8sIJl7un4BoWHYq/2D6jDHK/YmbpgG
|
||||||
|
J2FIvZZ4vVYixqgrqCh54USpwTaKd4KQpdm4zZMJ2yIznR6+gXxTi2TyUwKFV67y
|
||||||
|
RY6o5n3jusWuRejB1BI2eXRbFSeZn6IxAG8QLlLHu1R/XBB1dYDkbtYZ5XuAJRTH
|
||||||
|
qjCL+s/4rjJIUj02dmmtiyObN3bs/48DTQsDqgCYbA2wmbBjT9lPmRWfncBSlqbx
|
||||||
|
XSRwyXnDql+UIi6wphBdESn5TNVJjWxYQ16ZRYoZlz4tV2bPl4kCHAQQAQoABgUC
|
||||||
|
WCOB1QAKCRCB/dUuZfDuelcKD/4iggO8sgYYmTf63y2LJhG7e2d3K/cbQ04Pn/iF
|
||||||
|
b6zBa5kCsLAWHJmWBPlPkQB3HjZ+j1OSs4sHvwuotuVde2cbtzEvuphQRamFSU83
|
||||||
|
lakXwXrk00IU7oVEz1Yo9c7IB6pOY0uDgQMIp/Vitm9pomZXAV3S1bYciYuvXdJC
|
||||||
|
6SaUbyrHzkVgxEV7TSK2XUgMFGnBCIUhqxNA+chkGg6KS715XH8MJyb1Izinsjux
|
||||||
|
Q07+tquoiYqRhSntTc+bJ83OpSB8bPlqP1FZdc+ypZjq8OEpFIZK8QtK+849vzYw
|
||||||
|
vORq19mwFAY9y6XZip7YZzSGzn7Q4v3XfVJlIwGrkHSnUkqOmQXGdoSHAiwX3WhU
|
||||||
|
nPtpYFZQoFFQjm0SsgdPWbqJmmLj2MjHUFbKzl9PVa0INnwp6RxfzNZjU6OxphoI
|
||||||
|
ocw+2hmXv/lHzb+cwFf8Nae0WOJkCoGwxk9mYA0omhuKJ/WWVvauu+uaswBkeX6O
|
||||||
|
1Yf1jSrWgK5u+jbDcmZj0kFL0zdB/pBxshngxCA3efPd1e2dJGcPlaoqkwZAtEmG
|
||||||
|
hwWDj9Gk5IhSX1/Wpv+ke2wprUPzwv7Z5loXroLub2R29FWyHzymwlnts5hSg7kR
|
||||||
|
1PzxD3iOHFgmnWcBCchf8UWmAjrutZQJZWLTDkzlC6Vj8rkJcgdb0v14zvYVJTWs
|
||||||
|
Q/hMDYkCHAQSAQoABgUCWCN/ZwAKCRCRAl9JYWN/KLmND/9k8ydscqaMW86NswOC
|
||||||
|
drJDXuOTDvJJlnLYhgO1+jKQizbHQqnszIVk7KvXY2/K++TdTNdG/+/LosvtIRKb
|
||||||
|
2DEvV2hPZhc3UxZ2Gnx9ocqjSCfmd9er9EY34WGLtr57eQnu93CJGe10F3qg4rr3
|
||||||
|
8zsgQcLXomvKnB2J5oRWBla01DXZhum1uIbehrqDmBUz9YxAIz9adY0AjDyLNvAB
|
||||||
|
IghkjobLHWhouQaTV1IT6Koixdrj6eKekjRojWVZ7Po5dcUe525NlA1yBJhI9VDF
|
||||||
|
TLy6DHEwmJJuAQqDpyRILzSDsyQ0B+wBgpGks5reCDZlwAjogLy0O6nV2qYKfVXK
|
||||||
|
ziHeXPvzLPEfSmkA0ZutLXnJBLNfod4zELLJrPJA27dDdQO4yImoAjjTklyBpHXy
|
||||||
|
Fp3r5RlDq5yA9VpNkC58KJQZoiplkUNBaQTjYgcYX4LDaU4bEq++CUUEnisIyqPO
|
||||||
|
KqJPJAvmlm1d2vOMxhzcjxt3xBAvDZ8ibRIXlV296Y56SEBe5UZhyUqpAQBa0zsp
|
||||||
|
oQm9MdkqoC+IDBTRVnbGSxsw0GiQRpAJyNOlkTzP6z6ySmRIQ1hH1TqXx7TChPqP
|
||||||
|
tcLHLOnRTY9/v5Kt8APa+qgn/0UVgGcetkarqXNdtq+zD1x9C0DFNURBl0uH/E1E
|
||||||
|
Dtyx4xI6xCY3jzv0Cb1y9Afrc7Q7U2FtaSBLZXJvbGEgKGh0dHA6Ly93d3cuaWtp
|
||||||
|
LmZpL2tlcm9sYXNhLykgPGtlcm9sYXNhQGlraS5maT6JAlYEEwECAEACGwMHCwkI
|
||||||
|
BwMCAQYVCAIJCgsEFgIDAQIeAQIXgBYhBI7TluN+ONRxoAUw06lVMkX96bc5BQJZ
|
||||||
|
vjaaBQkZwjtjAAoJEKlVMkX96bc58KYP/jvpHWmoKKFzqClLdDkN8tBo30GgHGoh
|
||||||
|
WKHbgW859mi/ZcYzTxlx6Dm88QwV5DRulgqAHvLTippncR3LTpL3Ys86ZDoIskF9
|
||||||
|
BnWOm/bC+Czl9mDL6BNfFEWWnfeGFsCaMq/Jx8nj8i2o0ezhbTjCeGeng4quWK1D
|
||||||
|
T6G8Kvnk7Dc+i0n20LUQf2WRCLrtJEl/PMElc+11X/jLm9ypjE/jr24xiNsZck45
|
||||||
|
4rdMsdkZqgbSHWNIHyxS66mWmD1vbHp3kq0ic09ISCx/Xc7OQpv7V0hN+lxE+AvA
|
||||||
|
3Mjp2i7A3ntwUbPwmZkdiWUrBqU4CYee0S4fbgroUjNQ/JhGZmCku5YbcyAg4KrP
|
||||||
|
quXEftY0HP3qQMMElcwna5ljDQdC9NGuSWkH0OKZsuXlgtIz583wjNTKupzii+jo
|
||||||
|
AJcNnbdRR4i/U7cI+/ySluTp+Az/odmTNzssGnlsXOSGYWk4o6ruKvU4HzUauMEO
|
||||||
|
XPrTvI7IlH/GIHvwrOcXAyAU90M0qppC4YjXK8oCawzTnbQVQsbedwoNaEPaD4Bz
|
||||||
|
N4ka9PtvgKnBzODvOeDsTz1O6vTomttW2C6nficc/PXQvxDkZ3NeJw3eGiiAQxeA
|
||||||
|
w2Ps3+oIasyQZZVLlvu8dfE2BsPArggPLNgd6bLk3yk/gQGR0D+/TyHuCgPM1nZt
|
||||||
|
mBG3utrXfMz3iQIcBBABCgAGBQJYI4HVAAoJEIH91S5l8O56YhYQAIfxcRP31w+6
|
||||||
|
htYZO5dGS1zF1CReY8VO/AyBa59YJHAlFSbOhqHQY1SAYD+H1IZhv/JCvQScZhyT
|
||||||
|
giI3KPP1RAPUa1xGZrf4hwl7Yu8ZpFn1Y0Gxd4bj2i9HWJcd1A+ci4kc+dU4rPEB
|
||||||
|
ac2U06vcIyuyCSjuK2RdYCyd7pQRkoUtZxWen6sxi1khsw3Qucfbr5QZyh/roA4i
|
||||||
|
hTb3VlFFIiCRcZOAqSYml9a+kbDxpgswbdCE3EMHVYBWtFwWImZOhnsun148H7EA
|
||||||
|
+b+hAcbkoNpXJOfC8q/EM0qm6C1GhryXe1voLxfeMdKKjXM78dc0z3L/Vz0E+stD
|
||||||
|
qrT7CG1jLSk/rIjSvRhbnEX/ZTKa+19ytht0B2zxkokjevvzw6cXSM5JjG8Q9k7R
|
||||||
|
PcPOTS0t7MSwRqCEFvhvswH3JYy0SH0U2nKbPmkY1Awh6BZNyGpf3dxHV648RVZv
|
||||||
|
lrXUbXv1azD8U+llaMkVx0PTbej0RcRHQM6B5zJ0qu1MC64V/v2uUK3MNl38JdFa
|
||||||
|
wkB494uf024FRIfJlQ8+WYrsGlHbUBZZsWzsyrPaeaqL0Xcy+hOc7wu7cZPxkqL8
|
||||||
|
71O3jbq2dFkBWA4ANsiRdJTh14qoen4YVdc9gjq6HsuFyu0gauDktLaLUHVCX27e
|
||||||
|
kWThHNxeNoGkcKKHoQdHrN8sII8C1q6fiQIcBBIBCgAGBQJYI39nAAoJEJECX0lh
|
||||||
|
Y38oIbEP/165gMnaD+g7cfGsHCHQk9/ZrE+4iXfZSwxlhaJ4kK88Ad/tPvoiOxmS
|
||||||
|
NaUIfH/koGBQNeE/OjjIkPiGlsldg9//4mGmSglM4jLeQBG0QuTLo30aNCl9xU6q
|
||||||
|
rv9PV/ROfjRUF5NxCAiDdA24uBZ8CftdK5sglD6fazeINQyNk5Wm8e7Jk9PRzIuK
|
||||||
|
jnZRyaGJl3egx5smWnCZZ8QfjsYXDzbp1uDavmF7Qczp+xbzBhdh6VdSZ0VwvnPr
|
||||||
|
kN3CzGiR3AuG982R41UGZQ3a+3d38QnmX8cdT1yCqJLrSh4+AJbuGrndOqxBhTN0
|
||||||
|
VQBIGHteZ6ezxDHs4kVBwfSc3XOzNimJKFVxWCJDXZQojKawu7PaDPYGxU9OWAjs
|
||||||
|
OWtJFTGqeEiBDQbnLwncT5l/I1O1zv9MIFhQ4MgJeRbvG/QjuLrP6RoIMGZLkX6K
|
||||||
|
o35dZ6gO/XAdhJmOhfon6Q5/RPdB11e/UkM58xZMbQLGJ3Gm9vjz039vwMabMjXf
|
||||||
|
28UE3xKzdzDg8Fb3ykNtIQSFNr/cf5YBZ7KMtfJ6ugExmJQQdWxZLQpHoNNCmknt
|
||||||
|
OHOajlB628N3J0PKlwPjjt5KveBO2ur/yAly6SSv2Oi/astGscgrJtIyPFKMmKar
|
||||||
|
gPzEnPv4DfLbHnskHkyzVBrQE16Pmcfx4fAe2LeDIO8KJMqSxKOouQINBFLH/jcB
|
||||||
|
EADGsDuJyhEGExcInkb4Zy8HlSNaT0nFxpQK+tI5RlqyC11TnxdUDCQwPPRh/hdN
|
||||||
|
s7RDQ6lHhmXydD04+Sjjn0nkQCxd8TlBTwOU6BV4vPF7BonskTMHb6mXxOo+6f7+
|
||||||
|
J5q4beQ7XzRSoFqLZ0kiump7B4m6L4WSqZmcfi8f6EOQZHK1HsTB2jE+cwaPm7vN
|
||||||
|
eNaKXYM6hHojZ7XnDavv4vMxfLRGGx5e6wNcZexD4t4pkMwmA+A576t5CnIm8JVB
|
||||||
|
2HLj2YZN6DfIfYQ9EcMYSYVLg/EuivPaMZIoIa9ki5TbeQDANluJqAG+i+CSoIfJ
|
||||||
|
X1V1p0WoeSEKJnGBhVFmajU+nek1lCqeM/VXQF4dWbFCQm8/eKl1M9PY77dwT90x
|
||||||
|
RZDeigM6o6CoB+6rfePAnwsmtA21LyQ9WMTCH9qii9SvlWkfCvLK3HnC/2d1ctws
|
||||||
|
F8yoc6KF+Qwj7RV23/iHYqjxB8OcOAyHuv+k0sD9o0MLlzAExOftLPkwyWS8ePs7
|
||||||
|
rJVLSA42N3diUYx5TO3NzmU8ifn4hGxbF9J/sIYQUfCxH9V+DBe2vhVSZ/bQKpVU
|
||||||
|
ZTE8KxINL32JvTR3HbEwHKIe0xIFnDOMGrWP7icPEsaSKj/0vEq/Is/R2nJWfD0V
|
||||||
|
zs7vil9zYU16vHQtZrv/5SzavzDRFGxLm7/jDfrzMuNRdQARAQABiQIlBBgBAgAP
|
||||||
|
BQJSx/43AhsMBQkJZgGAAAoJEKlVMkX96bc5cYEQAI8xVRdoza/MY9KFJCRiobii
|
||||||
|
4GjxAJqIXXvbY8mM7rvmiLnzfr5msLzSop7Epse6tkI8QvYXbY+EjXke27J8rM/z
|
||||||
|
qXyqdKJfcqKVzq6fIQyVVsz9o0hwvuSjkcSgCV2b8XHKbNygSnNh2VJ5YY/zL2x5
|
||||||
|
tsySa3tLukh46ydKWRQqOSMIxMT0TnCOiTMr5INwGnX68rB0kEuCSUdkzDuyulW2
|
||||||
|
T0oN7Yx4ASgrJixwSu/UNMO/fXvkG0NYIWeXQcnn0C8zQv/PAfzJ9O+24Md4m6tz
|
||||||
|
VuDpblSRB6E8xQhusysRc4BEHB/G3Ege3JR/tumT6solJD7pfFDYkC9rAKhTKK9h
|
||||||
|
1b2VpEGxN9o8qwwjtNljLakEYXuWNpCmZxKAR4Agg3UY8CGmgsTKyjsfB9iO2+LX
|
||||||
|
CS9uQMAtIHbnr++mPASBEKD6rP8QIyf12fbI9JRNGWcXQVZBGXWmu4UaKOrXBTVz
|
||||||
|
ooj2yG7kZmKO0FoIaCqqTUs2/KhnFj2EFvym2j1OMrYxOM/pT6w3cv3CwGb1ILOX
|
||||||
|
0qrTQiEpOExSMa21Mi4678jXt6F6NPWG/lwlgE1WM9i9jIcua+NU5ZJiqQ/rjuSR
|
||||||
|
MB/WFrQy66/jb9wWNp/G6DXrANgdgoC9Dq5oviB+dqSXlU5b0atG4ru506mXCHKm
|
||||||
|
OC+wu0oRlGQ/X6iuuDfS
|
||||||
|
=fVwo
|
||||||
|
-----END PGP PUBLIC KEY BLOCK-----
|
||||||
44
contrib/munin_plugins/dhcpstats_abs.js
Executable file
44
contrib/munin_plugins/dhcpstats_abs.js
Executable file
|
|
@ -0,0 +1,44 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
var command = "dhcpd-pools"
|
||||||
|
var timeout = 5000; // (ms) if the command take longer, then kill it
|
||||||
|
|
||||||
|
var execFile = require('child_process').execFile;
|
||||||
|
var echo = require('util').print;
|
||||||
|
|
||||||
|
var arg = process.argv[2];
|
||||||
|
|
||||||
|
if ("autoconf" === arg) {
|
||||||
|
echo("yes\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
execFile(command, [ "--format=j", "--sort=n" ], { timeout: timeout }, function(error, stdout, stderr) {
|
||||||
|
if (error !== null) {
|
||||||
|
console.log('exec error: ' + error);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
stdout = JSON.parse(stdout);
|
||||||
|
|
||||||
|
if ("config" === arg) {
|
||||||
|
echo("graph_title dhcp usage (number of machines)\n");
|
||||||
|
echo("graph_vlabel nb of machines\n");
|
||||||
|
echo("graph_scale no\n");
|
||||||
|
echo("graph_category network\n");
|
||||||
|
|
||||||
|
stdout["subnets"].forEach(function(subnet) {
|
||||||
|
var location = subnet["location"];
|
||||||
|
var range = subnet["range"];
|
||||||
|
echo(location + '.label ' + location + ' (' + range + ')\n');
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
stdout["shared-networks"].forEach(function(network) {
|
||||||
|
var location = network["location"];
|
||||||
|
var used = network["used"];
|
||||||
|
var defined = network["defined"];
|
||||||
|
echo(location + '.value ' + used + '\n');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
37
contrib/munin_plugins/dhcpstats_abs.php
Executable file
37
contrib/munin_plugins/dhcpstats_abs.php
Executable file
|
|
@ -0,0 +1,37 @@
|
||||||
|
#!/usr/bin/env php
|
||||||
|
<?php
|
||||||
|
error_reporting(E_ALL);
|
||||||
|
ini_set("display_errors", true);
|
||||||
|
|
||||||
|
$ret="";
|
||||||
|
$xml = exec("dhcpd-pools --format x --sort n", $ret);
|
||||||
|
$xml = join($ret);
|
||||||
|
|
||||||
|
$xml = new SimpleXMLElement($xml);
|
||||||
|
|
||||||
|
|
||||||
|
if ($argc>1 && $argv[1]=="autoconf") {
|
||||||
|
echo "yes\n";
|
||||||
|
return;
|
||||||
|
} elseif ($argc>1 && $argv[1]=="config") {
|
||||||
|
$xml2 = $xml->xpath("subnet");
|
||||||
|
echo "graph_title dhcp usage (number of machines)\n";
|
||||||
|
echo "graph_vlabel nb of machines\n";
|
||||||
|
echo "graph_scale no\n";
|
||||||
|
echo "graph_category network\n";
|
||||||
|
foreach ($xml2 as $xml3) {
|
||||||
|
$location = (string) $xml3->location;
|
||||||
|
$range = (string) $xml3->range;
|
||||||
|
echo "$location.label $location ($range)\n";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$xml2 = $xml->xpath("shared-network");
|
||||||
|
foreach ($xml2 as $xml3) {
|
||||||
|
$location = (string) $xml3->location;
|
||||||
|
$used = (int) $xml3->used;
|
||||||
|
$defined = (int) $xml3->defined;
|
||||||
|
$pourcent = ceil($used*10000/$defined)/100;
|
||||||
|
echo "$location.value $used\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
49
contrib/munin_plugins/dhcpstats_percent.js
Executable file
49
contrib/munin_plugins/dhcpstats_percent.js
Executable file
|
|
@ -0,0 +1,49 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
var command = "dhcpd-pools"
|
||||||
|
var timeout = 5000; // (ms) if the command take longer, then kill it
|
||||||
|
|
||||||
|
var execFile = require('child_process').execFile;
|
||||||
|
var echo = require('util').print;
|
||||||
|
|
||||||
|
var arg = process.argv[2];
|
||||||
|
|
||||||
|
if ("autoconf" === arg) {
|
||||||
|
echo("yes\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
execFile(command, [ "--format=j", "--sort=n" ], { timeout: timeout }, function(error, stdout, stderr) {
|
||||||
|
if (error !== null) {
|
||||||
|
console.log('exec error: ' + error);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
stdout = JSON.parse(stdout);
|
||||||
|
|
||||||
|
if ("config" === arg) {
|
||||||
|
echo("graph_title dhcp usage (in percent)\n");
|
||||||
|
echo("graph_args --upper-limit 100 -l 0 --rigid\n");
|
||||||
|
echo("graph_vlabel %\n");
|
||||||
|
echo("graph_scale no\n");
|
||||||
|
echo("graph_category network\n");
|
||||||
|
|
||||||
|
stdout["subnets"].forEach(function(subnet) {
|
||||||
|
var location = subnet["location"];
|
||||||
|
var range = subnet["range"];
|
||||||
|
echo(location + '.label ' + location + ' (' + range + ')\n');
|
||||||
|
echo(location + '.warning 75\n');
|
||||||
|
echo(location + '.critical 90\n');
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
stdout["shared-networks"].forEach(function(network) {
|
||||||
|
var location = network["location"];
|
||||||
|
var used = network["used"];
|
||||||
|
var defined = network["defined"];
|
||||||
|
// keep 1 digit after decimal point
|
||||||
|
var percent = Math.ceil(used * 100 / defined) / 10;
|
||||||
|
echo(location + '.value ' + percent + '\n');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
42
contrib/munin_plugins/dhcpstats_percent.php
Executable file
42
contrib/munin_plugins/dhcpstats_percent.php
Executable file
|
|
@ -0,0 +1,42 @@
|
||||||
|
#!/usr/bin/env php
|
||||||
|
<?php
|
||||||
|
error_reporting(E_ALL);
|
||||||
|
ini_set("display_errors", true);
|
||||||
|
|
||||||
|
//$xml = file_get_contents("test.xml");
|
||||||
|
$ret="";
|
||||||
|
$xml = exec("dhcpd-pools --format x --sort n", $ret);
|
||||||
|
$xml = join($ret);
|
||||||
|
|
||||||
|
$xml = new SimpleXMLElement($xml);
|
||||||
|
|
||||||
|
|
||||||
|
if ($argc>1 && $argv[1]=="autoconf") {
|
||||||
|
echo "yes\n";
|
||||||
|
return;
|
||||||
|
} elseif ($argc>1 && $argv[1]=="config") {
|
||||||
|
$xml2 = $xml->xpath("subnet");
|
||||||
|
echo "graph_title dhcp usage (in percent)\n";
|
||||||
|
echo "graph_args --upper-limit 100 -l 0 --rigid\n";
|
||||||
|
echo "graph_vlabel %\n";
|
||||||
|
echo "graph_scale no\n";
|
||||||
|
echo "graph_category network\n";
|
||||||
|
foreach ($xml2 as $xml3) {
|
||||||
|
$location = (string) $xml3->location;
|
||||||
|
$range = (string) $xml3->range;
|
||||||
|
echo "$location.label $location ($range)\n";
|
||||||
|
echo "$location.warning 75\n";
|
||||||
|
echo "$location.critical 90\n";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$xml2 = $xml->xpath("shared-network");
|
||||||
|
foreach ($xml2 as $xml3) {
|
||||||
|
$location = (string) $xml3->location;
|
||||||
|
$used = (int) $xml3->used;
|
||||||
|
$defined = (int) $xml3->defined;
|
||||||
|
// keep 1 digit after decimal point
|
||||||
|
$pourcent = ceil($used*1000/$defined)/10;
|
||||||
|
echo "$location.value $pourcent\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
BIN
contrib/munin_plugins/preview.png
Normal file
BIN
contrib/munin_plugins/preview.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 80 KiB |
|
|
@ -1 +0,0 @@
|
||||||
command[check_dhcpd_pools]=/usr/bin/dhcpd-pools --warning $ARG1$ --critical $ARG2$
|
|
||||||
1
contrib/nagios.conf.in
Normal file
1
contrib/nagios.conf.in
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
command[check_dhcpd_pools]=@bindir@/dhcpd-pools --warning $ARG1$ --critical $ARG2$
|
||||||
4
contrib/rpm.spec
Normal file
4
contrib/rpm.spec
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
Klaus Slott made a package that works for opensuse, and is
|
||||||
|
a good starting point for whom ever happens to need an rpm.
|
||||||
|
|
||||||
|
https://build.opensuse.org/package/show/home:Mr_Manor/dhcpd-pools
|
||||||
|
|
@ -23,9 +23,10 @@
|
||||||
# you set $dbg to 1 then output will be generated in /tmp.
|
# you set $dbg to 1 then output will be generated in /tmp.
|
||||||
|
|
||||||
use strict;
|
use strict;
|
||||||
|
use NetSNMP::OID;
|
||||||
|
|
||||||
# Version info:
|
# Version info:
|
||||||
my $SNMPver = "snmp1.0";
|
my $SNMPver = "snmp1.1";
|
||||||
my $DHCPver = "dhcp1.0";
|
my $DHCPver = "dhcp1.0";
|
||||||
my $VERSION = "$SNMPver/$DHCPver";
|
my $VERSION = "$SNMPver/$DHCPver";
|
||||||
#
|
#
|
||||||
|
|
@ -193,7 +194,7 @@ sub ParseDataFile () {
|
||||||
}
|
}
|
||||||
close IN;
|
close IN;
|
||||||
if ($dbg) {
|
if ($dbg) {
|
||||||
foreach (sort @validoidlist) { print DBG "ValidOID: $_\n"; }
|
foreach (sort Oidcmp @validoidlist) { print DBG "ValidOID: $_\n"; }
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($dbg) {
|
if ($dbg) {
|
||||||
|
|
@ -370,7 +371,7 @@ sub GetData ($) {
|
||||||
|
|
||||||
@userquery = split (/\./, $userqueryoid);
|
@userquery = split (/\./, $userqueryoid);
|
||||||
my $found = 0;
|
my $found = 0;
|
||||||
foreach (sort @validoidlist) {
|
foreach (sort Oidcmp @validoidlist) {
|
||||||
$next = $_;
|
$next = $_;
|
||||||
print DBG "Comparing $userqueryoid vs. $_\n" if $dbg;
|
print DBG "Comparing $userqueryoid vs. $_\n" if $dbg;
|
||||||
@validoid = split (/\./, $_);
|
@validoid = split (/\./, $_);
|
||||||
|
|
@ -432,6 +433,12 @@ sub Pong {
|
||||||
$line = 0;
|
$line = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub Oidcmp {
|
||||||
|
my $oida = new NetSNMP::OID($a);
|
||||||
|
my $oidb = new NetSNMP::OID($b);
|
||||||
|
snmp_oid_compare($oida, $oidb);
|
||||||
|
}
|
||||||
|
|
||||||
################################## START ##################################
|
################################## START ##################################
|
||||||
#
|
#
|
||||||
# Main
|
# Main
|
||||||
|
|
|
||||||
4
contrib/zabbix.txt
Normal file
4
contrib/zabbix.txt
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
Zabbix 5 template, and instructions how to use it, can be found from
|
||||||
|
Mathieu's repository.
|
||||||
|
|
||||||
|
https://github.com/mmorier86/dhcpd-pools-zabbix-template
|
||||||
3
doc/.gitignore
vendored
Normal file
3
doc/.gitignore
vendored
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
/doxy.conf
|
||||||
|
/doxyfile.stamp
|
||||||
|
/html
|
||||||
24
doc/Makemodule.am
Normal file
24
doc/Makemodule.am
Normal file
|
|
@ -0,0 +1,24 @@
|
||||||
|
PATHFILES += \
|
||||||
|
doc/doxy.conf
|
||||||
|
nodist_noinst_DATA = \
|
||||||
|
doc/doxy.conf
|
||||||
|
|
||||||
|
EXTRA_DIST += \
|
||||||
|
doc/introduction.dox
|
||||||
|
|
||||||
|
if ENABLE_DOXYGEN
|
||||||
|
|
||||||
|
doc/doxyfile.stamp:
|
||||||
|
$(DOXYGEN) doc/doxy.conf
|
||||||
|
date > doc/doxyfile.stamp
|
||||||
|
|
||||||
|
CLEANFILES += doc/doxyfile.stamp
|
||||||
|
|
||||||
|
all-local: doc/doxyfile.stamp
|
||||||
|
|
||||||
|
clean-local-doc:
|
||||||
|
-rm -rf doc/*html
|
||||||
|
|
||||||
|
CLEAN_LOCALS += clean-local-doc
|
||||||
|
|
||||||
|
endif
|
||||||
2424
doc/doxy.conf.in
Normal file
2424
doc/doxy.conf.in
Normal file
File diff suppressed because it is too large
Load diff
49
doc/introduction.dox
Normal file
49
doc/introduction.dox
Normal file
|
|
@ -0,0 +1,49 @@
|
||||||
|
/*!
|
||||||
|
@mainpage
|
||||||
|
@section introduction Introduction
|
||||||
|
|
||||||
|
This is dhcpd-pools ISC dhcp shared network and pool range usage
|
||||||
|
analysis. Purpose of command is to count usage ratio of each IP range
|
||||||
|
and shared network pool which ISC dhcpd is in control of. Users of the
|
||||||
|
command are most likely ISPs and other organizations that have large IP
|
||||||
|
space.
|
||||||
|
|
||||||
|
Program is written C. Design goal is to get analysis done quickly where
|
||||||
|
there is lots of data. On cheap laptop the speed of analysis is roughly
|
||||||
|
100k leases per second. Number of ranges, or shared networks, does not
|
||||||
|
make any significant difference in getting analysis done.
|
||||||
|
|
||||||
|
@section main-copyright Copyright Policy
|
||||||
|
|
||||||
|
The dhcpd-pools has BSD 2-clause license which also known as "Simplified
|
||||||
|
BSD License" or "FreeBSD License".
|
||||||
|
|
||||||
|
Copyright 2006- Sami Kerola. All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
modification, are permitted provided that the following conditions are
|
||||||
|
met:
|
||||||
|
|
||||||
|
1. Redistributions of source code must retain the above copyright
|
||||||
|
notice, this list of conditions and the following disclaimer.
|
||||||
|
2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
notice, this list of conditions and the following disclaimer in the
|
||||||
|
documentation and/or other materials provided with the distribution.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
||||||
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR AND CONTRIBUTORS OR
|
||||||
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||||||
|
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
||||||
|
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
|
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
|
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
The views and conclusions contained in the software and documentation are
|
||||||
|
those of the authors and should not be interpreted as representing
|
||||||
|
official policies, either expressed or implied, of Sami Kerola.
|
||||||
|
|
||||||
|
*/
|
||||||
268
lib/.gitignore
vendored
268
lib/.gitignore
vendored
|
|
@ -1,170 +1,262 @@
|
||||||
/.deps/
|
/alloca.h
|
||||||
/.gitignore~
|
/alloca.in.h
|
||||||
/.libs/
|
|
||||||
/Makefile
|
|
||||||
/Makefile.in
|
|
||||||
/Makefile.am
|
|
||||||
/_Exit.c
|
|
||||||
/arg-nonnull.h
|
/arg-nonnull.h
|
||||||
/atexit.c
|
/arpa/
|
||||||
/c++defs.h
|
/arpa_inet.c
|
||||||
|
/arpa_inet.in.h
|
||||||
|
/assert.in.h
|
||||||
|
/attribute.h
|
||||||
|
/basename-lgpl.c
|
||||||
|
/basename-lgpl.h
|
||||||
|
/c32is-impl.h
|
||||||
|
/c32isprint.c
|
||||||
|
/calloc.c
|
||||||
/c-ctype.c
|
/c-ctype.c
|
||||||
/c-ctype.h
|
/c-ctype.h
|
||||||
/c-ctype.lo
|
/c++defs.h
|
||||||
/c-ctype.o
|
/cdefs.h
|
||||||
/c-strcase.h
|
/cloexec.c
|
||||||
/c-strcasecmp.c
|
/cloexec.h
|
||||||
/c-strcasecmp.lo
|
|
||||||
/c-strcasecmp.o
|
|
||||||
/c-strcaseeq.h
|
|
||||||
/c-strncasecmp.c
|
|
||||||
/c-strncasecmp.lo
|
|
||||||
/c-strncasecmp.o
|
|
||||||
/calloc.c
|
|
||||||
/charset.alias
|
|
||||||
/close-stream.c
|
|
||||||
/close-stream.h
|
|
||||||
/close-stream.lo
|
|
||||||
/close-stream.o
|
|
||||||
/close.c
|
/close.c
|
||||||
/closeout.c
|
/closeout.c
|
||||||
/closeout.h
|
/closeout.h
|
||||||
/closeout.lo
|
/close-stream.c
|
||||||
/closeout.o
|
/close-stream.h
|
||||||
/config.charset
|
/c-strcasecmp.c
|
||||||
/configmake.h
|
/c-strcaseeq.h
|
||||||
/dosname.h
|
/c-strcase.h
|
||||||
|
/c-strncasecmp.c
|
||||||
|
/dup2.c
|
||||||
/errno.in.h
|
/errno.in.h
|
||||||
/error.c
|
/error.c
|
||||||
/error.h
|
/error.h
|
||||||
|
/error.in.h
|
||||||
/exitfail.c
|
/exitfail.c
|
||||||
/exitfail.h
|
/exitfail.h
|
||||||
/exitfail.lo
|
|
||||||
/exitfail.o
|
|
||||||
/fclose.c
|
/fclose.c
|
||||||
/fclose.lo
|
/fcntl.c
|
||||||
/fclose.o
|
/fcntl.h
|
||||||
|
/fcntl.in.h
|
||||||
/fd-hook.c
|
/fd-hook.c
|
||||||
/fd-hook.h
|
/fd-hook.h
|
||||||
/fd-hook.lo
|
|
||||||
/fd-hook.o
|
|
||||||
/fdopen.c
|
/fdopen.c
|
||||||
/fflush.c
|
/fflush.c
|
||||||
/fflush.lo
|
/filename.h
|
||||||
/fflush.o
|
/flexmember.h
|
||||||
|
/float.c
|
||||||
|
/float+.h
|
||||||
|
/float.h
|
||||||
|
/float.in.h
|
||||||
/fopen.c
|
/fopen.c
|
||||||
/fpending.c
|
/fpending.c
|
||||||
/fpending.h
|
/fpending.h
|
||||||
/fpurge.c
|
/fpurge.c
|
||||||
/fpurge.lo
|
|
||||||
/fpurge.o
|
|
||||||
/freading.c
|
/freading.c
|
||||||
/freading.h
|
/freading.h
|
||||||
/freading.lo
|
|
||||||
/freading.o
|
|
||||||
/free.c
|
/free.c
|
||||||
/fseek.c
|
/fseek.c
|
||||||
/fseek.lo
|
|
||||||
/fseek.o
|
|
||||||
/fseeko.c
|
/fseeko.c
|
||||||
/fseeko.lo
|
|
||||||
/fseeko.o
|
|
||||||
/fstat.c
|
/fstat.c
|
||||||
/ftell.c
|
/ftell.c
|
||||||
/ftello.c
|
/ftello.c
|
||||||
/getopt.c
|
/getdtablesize.c
|
||||||
/getopt.in.h
|
/getlocalename_l-unsafe.c
|
||||||
|
/getlocalename_l-unsafe.h
|
||||||
/getopt1.c
|
/getopt1.c
|
||||||
|
/getopt.c
|
||||||
|
/getopt-cdefs.in.h
|
||||||
|
/getopt-core.h
|
||||||
|
/getopt-ext.h
|
||||||
|
/getopt.in.h
|
||||||
/getopt_int.h
|
/getopt_int.h
|
||||||
|
/getopt-pfx-core.h
|
||||||
|
/getopt-pfx-ext.h
|
||||||
|
/getprogname.c
|
||||||
|
/getprogname.h
|
||||||
/gettext.h
|
/gettext.h
|
||||||
|
/hard-locale.c
|
||||||
|
/hard-locale.h
|
||||||
|
/ialloc.c
|
||||||
|
/ialloc.h
|
||||||
|
/idx.h
|
||||||
|
/inet_pton.c
|
||||||
/intprops.h
|
/intprops.h
|
||||||
|
/intprops-internal.h
|
||||||
|
/inttypes.h
|
||||||
|
/inttypes.in.h
|
||||||
|
/isnan.c
|
||||||
|
/isnand.c
|
||||||
|
/isnanf.c
|
||||||
|
/isnanl.c
|
||||||
|
/itold.c
|
||||||
|
/lc-charset-dispatch.c
|
||||||
|
/lc-charset-dispatch.h
|
||||||
|
/libc-config.h
|
||||||
/libdhcpd_pools.la
|
/libdhcpd_pools.la
|
||||||
|
/libdhcpd_pools_la-arpa_inet.lo
|
||||||
|
/libdhcpd_pools_la-basename-lgpl.lo
|
||||||
|
/libdhcpd_pools_la-c32isprint.lo
|
||||||
|
/libdhcpd_pools_la-c-ctype.lo
|
||||||
|
/libdhcpd_pools_la-cloexec.lo
|
||||||
|
/libdhcpd_pools_la-closeout.lo
|
||||||
|
/libdhcpd_pools_la-close-stream.lo
|
||||||
|
/libdhcpd_pools_la-c-strcasecmp.lo
|
||||||
|
/libdhcpd_pools_la-exitfail.lo
|
||||||
|
/libdhcpd_pools_la-fclose.lo
|
||||||
|
/libdhcpd_pools_la-fcntl.lo
|
||||||
|
/libdhcpd_pools_la-fd-hook.lo
|
||||||
|
/libdhcpd_pools_la-fflush.lo
|
||||||
|
/libdhcpd_pools_la-float.lo
|
||||||
|
/libdhcpd_pools_la-fopen.lo
|
||||||
|
/libdhcpd_pools_la-fpurge.lo
|
||||||
|
/libdhcpd_pools_la-freading.lo
|
||||||
|
/libdhcpd_pools_la-fseek.lo
|
||||||
|
/libdhcpd_pools_la-fseeko.lo
|
||||||
|
/libdhcpd_pools_la-getprogname.lo
|
||||||
|
/libdhcpd_pools_la-hard-locale.lo
|
||||||
|
/libdhcpd_pools_la-ialloc.lo
|
||||||
|
/libdhcpd_pools_la-localcharset.lo
|
||||||
|
/libdhcpd_pools_la-malloca.lo
|
||||||
|
/libdhcpd_pools_la-math.lo
|
||||||
|
/libdhcpd_pools_la-mbrtoc32.lo
|
||||||
|
/libdhcpd_pools_la-mbrtowc.lo
|
||||||
|
/libdhcpd_pools_la-mbszero.lo
|
||||||
|
/libdhcpd_pools_la-mktime.lo
|
||||||
|
/libdhcpd_pools_la-nstrftime.lo
|
||||||
|
/libdhcpd_pools_la-progname.lo
|
||||||
|
/libdhcpd_pools_la-quotearg.lo
|
||||||
|
/libdhcpd_pools_la-reallocarray.lo
|
||||||
|
/libdhcpd_pools_la-realloc.lo
|
||||||
|
/libdhcpd_pools_la-setlocale_null.lo
|
||||||
|
/libdhcpd_pools_la-setlocale_null-unlocked.lo
|
||||||
|
/libdhcpd_pools_la-stat-time.lo
|
||||||
|
/libdhcpd_pools_la-stdlib.lo
|
||||||
|
/libdhcpd_pools_la-sys_socket.lo
|
||||||
|
/libdhcpd_pools_la-timegm.lo
|
||||||
|
/libdhcpd_pools_la-time_rz.lo
|
||||||
|
/libdhcpd_pools_la-unistd.lo
|
||||||
|
/libdhcpd_pools_la-wctype-h.lo
|
||||||
|
/libdhcpd_pools_la-xalloc-die.lo
|
||||||
|
/libdhcpd_pools_la-xmalloc.lo
|
||||||
|
/.libs/
|
||||||
|
/limits.h
|
||||||
|
/limits.in.h
|
||||||
/localcharset.c
|
/localcharset.c
|
||||||
/localcharset.h
|
/localcharset.h
|
||||||
/localcharset.lo
|
/locale.h
|
||||||
/localcharset.o
|
/locale.in.h
|
||||||
|
/localename.h
|
||||||
|
/localename-unsafe.c
|
||||||
/lseek.c
|
/lseek.c
|
||||||
|
/Makefile.am
|
||||||
|
/malloca.c
|
||||||
|
/malloca.h
|
||||||
/malloc.c
|
/malloc.c
|
||||||
|
/math.c
|
||||||
|
/math.h
|
||||||
|
/math.in.h
|
||||||
|
/mbrtoc32.c
|
||||||
/mbrtowc.c
|
/mbrtowc.c
|
||||||
|
/mbrtowc-impl.h
|
||||||
|
/mbrtowc-impl-utf8.h
|
||||||
/mbsinit.c
|
/mbsinit.c
|
||||||
|
/mbszero.c
|
||||||
|
/mbtowc-lock.c
|
||||||
|
/mbtowc-lock.h
|
||||||
/memchr.c
|
/memchr.c
|
||||||
/memchr.valgrind
|
/memchr.valgrind
|
||||||
/memcpy.c
|
/minmax.h
|
||||||
|
/mktime.c
|
||||||
|
/mktime-internal.h
|
||||||
/msvc-inval.c
|
/msvc-inval.c
|
||||||
/msvc-inval.h
|
/msvc-inval.h
|
||||||
/msvc-nothrow.c
|
/msvc-nothrow.c
|
||||||
/msvc-nothrow.h
|
/msvc-nothrow.h
|
||||||
|
/netinet_in.in.h
|
||||||
|
/_Noreturn.h
|
||||||
|
/nstrftime.c
|
||||||
|
/open.c
|
||||||
/pathmax.h
|
/pathmax.h
|
||||||
/quote.h
|
/progname.c
|
||||||
|
/progname.h
|
||||||
/quotearg.c
|
/quotearg.c
|
||||||
/quotearg.h
|
/quotearg.h
|
||||||
/quotearg.lo
|
/quote.h
|
||||||
/quotearg.o
|
/reallocarray.c
|
||||||
/realloc.c
|
/realloc.c
|
||||||
/ref-add.sed
|
/setenv.c
|
||||||
/ref-add.sin
|
/setlocale-lock.c
|
||||||
/ref-del.sed
|
/setlocale_null.c
|
||||||
/ref-del.sin
|
/setlocale_null.h
|
||||||
|
/setlocale_null-unlocked.c
|
||||||
/stat.c
|
/stat.c
|
||||||
/stdarg.in.h
|
/stat-time.c
|
||||||
/stdbool.in.h
|
/stat-time.h
|
||||||
|
/stat-w32.c
|
||||||
|
/stat-w32.h
|
||||||
|
/stdckdint.h
|
||||||
|
/stdckdint.in.h
|
||||||
|
/stddef.h
|
||||||
/stddef.in.h
|
/stddef.in.h
|
||||||
/stdio-impl.h
|
/stdint.in.h
|
||||||
/stdio.h
|
/stdio.h
|
||||||
|
/stdio-impl.h
|
||||||
/stdio.in.h
|
/stdio.in.h
|
||||||
|
/stdio-read.c
|
||||||
|
/stdio-write.c
|
||||||
|
/stdlib.c
|
||||||
/stdlib.h
|
/stdlib.h
|
||||||
/stdlib.in.h
|
/stdlib.in.h
|
||||||
/str-two-way.h
|
/stpncpy.c
|
||||||
/strdup.c
|
/strdup.c
|
||||||
/streq.h
|
/streq.h
|
||||||
|
/strerror.c
|
||||||
/strerror-override.c
|
/strerror-override.c
|
||||||
/strerror-override.h
|
/strerror-override.h
|
||||||
/strerror.c
|
|
||||||
/strftime.c
|
/strftime.c
|
||||||
/strftime.h
|
/strftime.h
|
||||||
/strftime.lo
|
|
||||||
/strftime.o
|
|
||||||
/string.h
|
/string.h
|
||||||
/string.in.h
|
/string.in.h
|
||||||
/strstr.c
|
/strstr.c
|
||||||
/strtod.c
|
/strtod.c
|
||||||
|
/str-two-way.h
|
||||||
/sys/
|
/sys/
|
||||||
|
/sys_socket.c
|
||||||
|
/sys_socket.in.h
|
||||||
/sys_stat.in.h
|
/sys_stat.in.h
|
||||||
/sys_types.in.h
|
/sys_types.in.h
|
||||||
|
/sys_uio.in.h
|
||||||
|
/timegm.c
|
||||||
/time.h
|
/time.h
|
||||||
/time.in.h
|
/time.in.h
|
||||||
|
/time-internal.h
|
||||||
/time_r.c
|
/time_r.c
|
||||||
|
/time_rz.c
|
||||||
|
/tzset.c
|
||||||
|
/uchar.h
|
||||||
|
/uchar.in.h
|
||||||
|
/unictype/
|
||||||
|
/unictype/bitmap.h
|
||||||
|
/unictype/ctype_print.c
|
||||||
|
/unictype/ctype_print.h
|
||||||
|
/unictype/.gitignore
|
||||||
|
/unictype.h
|
||||||
|
/unictype.in.h
|
||||||
|
/unistd.c
|
||||||
/unistd.h
|
/unistd.h
|
||||||
/unistd.in.h
|
/unistd.in.h
|
||||||
/unused-parameter.h
|
/unitypes.h
|
||||||
|
/unitypes.in.h
|
||||||
|
/unsetenv.c
|
||||||
/verify.h
|
/verify.h
|
||||||
/warn-on-use.h
|
/warn-on-use.h
|
||||||
/wchar.h
|
/wchar.h
|
||||||
/wchar.in.h
|
/wchar.in.h
|
||||||
/wctype.h
|
/wctype.h
|
||||||
|
/wctype-h.c
|
||||||
/wctype.in.h
|
/wctype.in.h
|
||||||
|
/windows-initguard.h
|
||||||
/xalloc-die.c
|
/xalloc-die.c
|
||||||
/xalloc-die.lo
|
|
||||||
/xalloc-die.o
|
|
||||||
/xalloc-oversized.h
|
|
||||||
/xalloc.h
|
/xalloc.h
|
||||||
|
/xalloc-oversized.h
|
||||||
/xmalloc.c
|
/xmalloc.c
|
||||||
/xmalloc.lo
|
|
||||||
/xmalloc.o
|
|
||||||
/xprintf.c
|
|
||||||
/xprintf.h
|
|
||||||
/xprintf.lo
|
|
||||||
/xprintf.o
|
|
||||||
/xstrtod.c
|
|
||||||
/xstrtod.h
|
|
||||||
/xstrtod.lo
|
|
||||||
/xstrtod.o
|
|
||||||
/fcntl.in.h
|
|
||||||
/langinfo.in.h
|
|
||||||
/netinet_in.in.h
|
|
||||||
/stdalign.h
|
|
||||||
/stdalign.in.h
|
|
||||||
/sys_socket.in.h
|
|
||||||
/sys_uio.in.h
|
|
||||||
/fcntl.h
|
|
||||||
/langinfo.h
|
|
||||||
|
|
|
||||||
1
man/.gitignore
vendored
1
man/.gitignore
vendored
|
|
@ -1 +0,0 @@
|
||||||
/dhcpd-pools.1
|
|
||||||
|
|
@ -1,3 +0,0 @@
|
||||||
man_MANS = dhcpd-pools.1
|
|
||||||
EXTRA_DIST = dhcpd-pools.1.in
|
|
||||||
CLEANFILES = dhcpd-pools.1
|
|
||||||
2
man/Makemodule.am
Normal file
2
man/Makemodule.am
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
man_MANS = man/dhcpd-pools.1
|
||||||
|
PATHFILES += man/dhcpd-pools.1
|
||||||
|
|
@ -1,9 +1,28 @@
|
||||||
.TH DHCPD-POOLS "1" "2012-11-25" "@PACKAGE_STRING@" "User Commands"
|
'\" t
|
||||||
|
.TH DHCPD-POOLS "1" "2024-08-09" "@VERSION@" "User Commands"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
dhcpd-pools \- ISC dhcpd pools usage analysis
|
dhcpd-pools \- ISC dhcpd pools usage analysis
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B dhcpd-pools
|
.SY dhcpd-pools
|
||||||
[options]
|
.OP \-\-config file
|
||||||
|
.OP \-\-leases file
|
||||||
|
.OP \-\-sort nimcptTe
|
||||||
|
.OP \-\-reverse
|
||||||
|
.OP \-\-format tHcxXjJ
|
||||||
|
.OP \-\-mustach template
|
||||||
|
.OP \-\-output file
|
||||||
|
.OP \-\-limit nr
|
||||||
|
.OP \-\-color when
|
||||||
|
.OP \-\-warning percent
|
||||||
|
.OP \-\-critical percent
|
||||||
|
.OP \-\-warn\-count number
|
||||||
|
.OP \-\-crit\-count number
|
||||||
|
.OP \-\-snet\-alarms
|
||||||
|
.OP \-\-minsize size
|
||||||
|
.OP \-\-perfdata
|
||||||
|
.OP \-\-version
|
||||||
|
.OP \-\-help
|
||||||
|
.YS
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
The program analyses ISC dhcpd shared network and pool usage and outputs the
|
The program analyses ISC dhcpd shared network and pool usage and outputs the
|
||||||
results in a format selected by user.
|
results in a format selected by user.
|
||||||
|
|
@ -49,39 +68,55 @@ only if there is failover configuration.
|
||||||
.SH OPTIONS
|
.SH OPTIONS
|
||||||
.TP
|
.TP
|
||||||
\fB\-c\fR, \fB\-\-config\fR=\fIFILE\fR
|
\fB\-c\fR, \fB\-\-config\fR=\fIFILE\fR
|
||||||
Path to the dhcpd.conf file.
|
Path to the dhcpd.conf file. If the dhcpd.conf has include files they
|
||||||
|
can be analysed separately, that can be useful when trying to understand
|
||||||
|
or monitor subset of data.
|
||||||
.TP
|
.TP
|
||||||
\fB\-l\fR, \fB\-\-leases\fR=\fIFILE\fR
|
\fB\-l\fR, \fB\-\-leases\fR=\fIFILE\fR
|
||||||
Path to the dhcpd.leases file.
|
Path to the dhcpd.leases file.
|
||||||
.TP
|
.TP
|
||||||
\fB\-s\fR, \fB\-\-sort\fR=\fI[nimcptTe]\fR
|
\fB\-s\fR, \fB\-\-sort\fR=\fI[nimcptTe]\fR
|
||||||
Sort ranges by chosen fields as a sorting keys. Maximum of five sort keys
|
Sort ranges by chosen fields as a sorting keys. Keys weight from left to
|
||||||
can be defined. Keys weight from left to right, i.e., if more weighting keys
|
right, i.e., if more weighting keys are equal next one is used. The IP
|
||||||
are equal next one is used. The IP field is default sort key.
|
field is default sort key.
|
||||||
.TP
|
.TP
|
||||||
\fB\-r\fR, \fB\-\-reverse\fR
|
\fB\-r\fR, \fB\-\-reverse\fR
|
||||||
Sort results in reverse order.
|
Sort results in reverse order.
|
||||||
.TP
|
.TP
|
||||||
\fB\-f\fR, \fB\-\-format\fR=\fI[thHcxX]\fR
|
\fB\-f\fR, \fB\-\-format\fR=\fI[tHcxXjJ]\fR
|
||||||
Output format.
|
Output format.
|
||||||
Text
|
Text
|
||||||
.RI ( t ).
|
.RI ( t ).
|
||||||
Standard html
|
Full-html
|
||||||
.RI ( h )
|
|
||||||
outputs only the HTML tables, and is useful for embeding more complex web
|
|
||||||
pages. Full-html
|
|
||||||
.RI ( H )
|
.RI ( H )
|
||||||
provides complete HTML headers, etc., including in-line CSS. The
|
page output. In html page critical and warning thresholds can be visualized
|
||||||
|
with \-\-color=always option. The
|
||||||
.RI ( c )
|
.RI ( c )
|
||||||
stands for comma-separated values. Output format xml
|
stands for comma-separated values. Output format xml
|
||||||
.RI ( x )
|
.RI ( x )
|
||||||
is similar to the dhcpstatus Perl module output. The extended xml
|
is similar to the dhcpstatus Perl module output. The extended xml
|
||||||
.RI ( X )
|
.RI ( X )
|
||||||
format will print ethernet address details.
|
format will print ethernet address details. The
|
||||||
|
.RI ( j )
|
||||||
|
will output in json format, which can be extended with
|
||||||
|
.RI ( J )
|
||||||
|
to include ethernet address.
|
||||||
.IP
|
.IP
|
||||||
The default format is
|
The default format is
|
||||||
.IR @OUTPUT_FORMAT@ .
|
.IR @OUTPUT_FORMAT@ .
|
||||||
.TP
|
.TP
|
||||||
|
\fB\-\-mustach\fR=\fITEMPLATE\fR
|
||||||
|
Output using mustach
|
||||||
|
.I template
|
||||||
|
file. This is useful when the native output formats controlled with
|
||||||
|
.B \-\-format
|
||||||
|
option do not provide what you need. See below example mustach template
|
||||||
|
that is using all available {{tags}} to demonstrate what can be displayed
|
||||||
|
and how.
|
||||||
|
.IP
|
||||||
|
@bindir@/dhcpd-pools --config @docdir@/dhcpd.conf --leases
|
||||||
|
@docdir@/dhcpd.leases --mustach @docdir@/mustach.template
|
||||||
|
.TP
|
||||||
\fB\-o\fR, \fB\-\-output\fR=\fIFILE\fR
|
\fB\-o\fR, \fB\-\-output\fR=\fIFILE\fR
|
||||||
.I File
|
.I File
|
||||||
where output is written. Default is stdout.
|
where output is written. Default is stdout.
|
||||||
|
|
@ -100,29 +135,56 @@ determines which numeric analysis tables to include in the output. The
|
||||||
following values are "OR'd" together to create the desired output. The
|
following values are "OR'd" together to create the desired output. The
|
||||||
default is
|
default is
|
||||||
.IR @OUTPUT_LIMIT@ .
|
.IR @OUTPUT_LIMIT@ .
|
||||||
.PP
|
.IP
|
||||||
.RS
|
.TS
|
||||||
.PD 0
|
tab(:);
|
||||||
|
ll.
|
||||||
|
0\fI1\fR:Print ranges
|
||||||
|
0\fI2\fR:Print shared networks
|
||||||
|
0\fI4\fR:Print total summary
|
||||||
|
\fI1\fR0:Print range header
|
||||||
|
\fI2\fR0:Print shared network header
|
||||||
|
\fI4\fR0:Print total summary header
|
||||||
|
.TE
|
||||||
|
.IP
|
||||||
|
The output limit for total summary has special meaning in
|
||||||
|
.B \-\-warning
|
||||||
|
and
|
||||||
|
.B \-\-critical
|
||||||
|
alarming context. When the alarming is in use, and total is not wanted
|
||||||
|
to be seen then in the case of alarming returning success nothing is
|
||||||
|
printed.
|
||||||
.TP
|
.TP
|
||||||
.B 01
|
\fB\-\-color\fR=\fIwhen\fR
|
||||||
Print ranges
|
Use yellow for warning, red for critical, green for suppressed by \-\-minsize
|
||||||
|
and blue when \-\-snet\-alarms is the cause of suppression or shared network
|
||||||
|
does not have any ranges. The
|
||||||
|
.I when
|
||||||
|
string can be
|
||||||
|
.BR always ,
|
||||||
|
.BR never ,
|
||||||
|
or
|
||||||
|
.BR auto .
|
||||||
|
Default is auto, that uses colors when command is running in interactive
|
||||||
|
terminal. With use of
|
||||||
|
.B \-\-warning
|
||||||
|
or
|
||||||
|
.B \-\-critical
|
||||||
|
coloring thresholds can be changed, but one must also use
|
||||||
|
.B \-\-format=text
|
||||||
|
to avoid turning on alarting mode.
|
||||||
.TP
|
.TP
|
||||||
.B 02
|
\fB\-\-skip\fR=\fIwhen\fR
|
||||||
Print shared networks
|
The
|
||||||
.TP
|
.I when
|
||||||
.B 04
|
can be one of the following:
|
||||||
Print total summary
|
.IR ok ,
|
||||||
.TP
|
.IR warning ,
|
||||||
.B 10
|
.IR critical ,
|
||||||
Print range header
|
.IR minsize ,
|
||||||
.TP
|
or
|
||||||
.B 20
|
.IR suppressed .
|
||||||
Print shared network header
|
The skipping criteria is exact match with colors in \-\-color option.
|
||||||
.TP
|
|
||||||
.B 40
|
|
||||||
Print total summary header
|
|
||||||
.PD
|
|
||||||
.RE
|
|
||||||
.TP
|
.TP
|
||||||
\fB\-\-warning\fR=\fIpercent\fR
|
\fB\-\-warning\fR=\fIpercent\fR
|
||||||
Turn on alarm output format, and specify percentage number which will
|
Turn on alarm output format, and specify percentage number which will
|
||||||
|
|
@ -134,7 +196,7 @@ for example
|
||||||
.IR \-L10 .
|
.IR \-L10 .
|
||||||
To monitor shared network only the limit would be
|
To monitor shared network only the limit would be
|
||||||
.IR \-L20 .
|
.IR \-L20 .
|
||||||
If warning percentage is not speficied it defaults to
|
If warning percentage is not specified it defaults to
|
||||||
.BR @ALARM_WARN@ .
|
.BR @ALARM_WARN@ .
|
||||||
The
|
The
|
||||||
.I percent
|
.I percent
|
||||||
|
|
@ -147,6 +209,55 @@ is
|
||||||
If critical percentage is not specified it defaults to
|
If critical percentage is not specified it defaults to
|
||||||
.BR @ALARM_CRIT@ .
|
.BR @ALARM_CRIT@ .
|
||||||
.TP
|
.TP
|
||||||
|
\fB\-\-warn\-count\fR=\fInumber\fR
|
||||||
|
A
|
||||||
|
.I number
|
||||||
|
of free leases before alarm is raised. When specified both \-\-warning
|
||||||
|
.I percent
|
||||||
|
and count
|
||||||
|
.I number
|
||||||
|
are required to be exceeded in order to alarm criteria being fulfilled.
|
||||||
|
.IP
|
||||||
|
This option is intended to be used in setup where very large and small
|
||||||
|
shared-networks and ranges co-exists. In such environments percent based
|
||||||
|
alarming can lead to either flood of alarms about small ranges, or way too
|
||||||
|
great overhead of free addresses in large shared-networks. Suggested usage
|
||||||
|
is to set percentage to a level that makes small ranges to ring, and set the
|
||||||
|
count to match level when an enormous shared-network is too few free leases.
|
||||||
|
.IP
|
||||||
|
Defaults to 2^32, that is size of entire IPv4 address space.
|
||||||
|
.TP
|
||||||
|
\fB\-\-crit\-count\fR=\fInumber\fR
|
||||||
|
Same as \-\-warn\-count, but for critical alarms.
|
||||||
|
.TP
|
||||||
|
\fB\-\-snet\-alarms
|
||||||
|
Suppress range alarms that are part of shared networks. Use of this option
|
||||||
|
will keep alarm criteria applied to ranges that are not part of shared-net
|
||||||
|
along with shared-net alarms. This option may help reducing alarm noise for
|
||||||
|
configurations that has lots of small ranges in big shared-networks.
|
||||||
|
.TP
|
||||||
|
\fB\-\-minsize\fR=\fIsize\fR
|
||||||
|
Ignore ranges and shared networks that are smaller or equal to the
|
||||||
|
defined size. This option is meaningful only in context of alarming, and
|
||||||
|
will intended to suppress for example single host ranges. By default this
|
||||||
|
option is not in use.
|
||||||
|
.TP
|
||||||
|
\fB\-p\fR, \fB\-\-perfdata\fR
|
||||||
|
Print additional performance data, like lease count, touched leases and
|
||||||
|
backup leases. This option is meaningful only in context of alarming and
|
||||||
|
will print lots of data, if there are many networks. By default this option
|
||||||
|
is not in use.
|
||||||
|
.TP
|
||||||
|
\fB\-A\fR, \fB\-\-all\-as\-shared\fR
|
||||||
|
Treat all stand-alone subnets as shared-network with named formed from it's
|
||||||
|
CIDR. By default this option is not in use for backwards compatibility.
|
||||||
|
.TP
|
||||||
|
\fB\-\-ip\-version\fR=\fI4|6\fR
|
||||||
|
Force command to read configuration and leases files in IPv4 or IPv6 mode.
|
||||||
|
Notice that when inputs do not match with what is forced analysis output is
|
||||||
|
garbage. This option should not be necessary to use, and exists only to
|
||||||
|
allow debugging.
|
||||||
|
.TP
|
||||||
\fB\-v\fR, \fB\-\-version\fR
|
\fB\-v\fR, \fB\-\-version\fR
|
||||||
Print version information to standard output and exit successfully.
|
Print version information to standard output and exit successfully.
|
||||||
.TP
|
.TP
|
||||||
|
|
@ -155,21 +266,28 @@ Print help to standard output and exit successfully.
|
||||||
.SH EXAMPLES
|
.SH EXAMPLES
|
||||||
.TP
|
.TP
|
||||||
Print ranges header, and analysis.
|
Print ranges header, and analysis.
|
||||||
$ dhcpd-pools -L 11 -c dhcpd.conf -l dhcpd.leases
|
$ dhcpd-pools \-L 11 \-c dhcpd.conf \-l dhcpd.leases
|
||||||
.br
|
.br
|
||||||
Ranges:
|
Ranges:
|
||||||
.br
|
.br
|
||||||
shared net name [...]
|
shared net name [...]
|
||||||
.TP
|
.TP
|
||||||
Print shared networks and totals, both headers and results
|
Print shared networks and totals, both headers and results
|
||||||
$ dhcpd-pools -L 66 -c dhcpd.conf -l dhcpd.leases shared net name
|
$ dhcpd-pools \-L 66 \-c dhcpd.conf \-l dhcpd.leases shared net name
|
||||||
.br
|
.br
|
||||||
[...]
|
[...]
|
||||||
.TP
|
.TP
|
||||||
Alarming
|
Alarming
|
||||||
$ dhcpd-pools -c dhcpd.conf -l dhcpd.leases --critical 80.1 --warning 75
|
$ dhcpd-pools \-c dhcpd.conf \-l dhcpd.leases \-\-critical 80.1 \-\-warning 75
|
||||||
.br
|
.br
|
||||||
CRITICAL: dhcpd-pools: Ranges; crit: 14 warn: 22 ok: 220 Shared nets; crit: 1 warn: 0 ok: 4
|
CRITICAL: dhcpd-pools: Ranges; crit: 14 warn: 22 ok: 220 Shared nets; crit: 1 warn: 0 ok: 4
|
||||||
|
.IP
|
||||||
|
$ dhcpd-pools \-c dhcpd.conf \-l dhcpd.leases \-L 22 \-\-critical 70 \-\-warning 50
|
||||||
|
.br
|
||||||
|
[no-output]
|
||||||
|
.br
|
||||||
|
Suppress printing OK, and make alarm only to go off if shared networks
|
||||||
|
exceed critial or warning levels.
|
||||||
.SH FILES
|
.SH FILES
|
||||||
.TP
|
.TP
|
||||||
@DHCPDCONF_FILE@
|
@DHCPDCONF_FILE@
|
||||||
|
|
@ -177,12 +295,23 @@ ISC dhcpd configuration file.
|
||||||
.TP
|
.TP
|
||||||
@DHCPDLEASE_FILE@
|
@DHCPDLEASE_FILE@
|
||||||
ISC dhcpd lease file.
|
ISC dhcpd lease file.
|
||||||
.SH AUTHOR
|
.TP
|
||||||
Written by Sami Kerola.
|
@docdir@/prometheus.template
|
||||||
|
Prometheus text file collector mustach template.
|
||||||
|
.SH AUTHORS
|
||||||
|
Original design by Sami Kerola.
|
||||||
|
.br
|
||||||
|
uthash by Troy D. Hanson.
|
||||||
.br
|
.br
|
||||||
XML support by Dominic Germain, Sogetel inc.
|
XML support by Dominic Germain, Sogetel inc.
|
||||||
.PP
|
.br
|
||||||
The software has FreeBSD License.
|
IPv6 support by Cheer Xiao.
|
||||||
|
.br
|
||||||
|
Mustache templating support by José Bollo.
|
||||||
|
.SH LICENSE
|
||||||
|
The dhcpd-pools uses FreeBSD License, uthash uses BSD license, the mustache
|
||||||
|
uses Apache License, and the gnulib modules are mostly, but not entirely,
|
||||||
|
GPL.
|
||||||
.SH "REPORTING BUGS"
|
.SH "REPORTING BUGS"
|
||||||
Report bugs to
|
Report bugs to
|
||||||
.MT @PACKAGE_BUGREPORT@
|
.MT @PACKAGE_BUGREPORT@
|
||||||
|
|
@ -195,4 +324,6 @@ Home page
|
||||||
.SH "SEE ALSO"
|
.SH "SEE ALSO"
|
||||||
.BR dhcpd.leases (5),
|
.BR dhcpd.leases (5),
|
||||||
.BR dhcpd.conf (5),
|
.BR dhcpd.conf (5),
|
||||||
.BR chmod (1)
|
.BR chmod (1),
|
||||||
|
.UR https://mustache.github.io/
|
||||||
|
.UE
|
||||||
|
|
|
||||||
38
project.doap
Normal file
38
project.doap
Normal file
|
|
@ -0,0 +1,38 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<rdf:RDF xml:lang="en" xmlns="http://usefulinc.com/ns/doap#"
|
||||||
|
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
||||||
|
xmlns:rdfs="http://www.w3.org/2000/01/rdf-schema#"
|
||||||
|
xmlns:foaf="http://xmlns.com/foaf/0.1/">
|
||||||
|
<Project rdf:about="https://dhcpd-pools.sourceforge.net/">
|
||||||
|
<name>dhcpd-pools</name>
|
||||||
|
<homepage rdf:resource="https://dhcpd-pools.sourceforge.net/" />
|
||||||
|
<shortdesc>This is dhcpd-pools ISC dhcp shared network and pool
|
||||||
|
range usage analysis tool.</shortdesc>
|
||||||
|
<description>Purpose of command is to count usage ratio of each
|
||||||
|
IP range and shared network pool which ISC dhcpd is in control
|
||||||
|
of. Program is written C. Design goal is to get analysis done
|
||||||
|
quickly where there is lots of data. On cheap laptop the speed
|
||||||
|
of analysis is more than 100k leases per second. Number of
|
||||||
|
ranges, or shared networks, does not make any significant
|
||||||
|
difference in getting analysis done.</description>
|
||||||
|
<mailing-list rdf:resource="https://lists.sourceforge.net/lists/listinfo/dhcpd-pools-announce" />
|
||||||
|
<download-page rdf:resource="https://sourceforge.net/projects/dhcpd-pools/files/" />
|
||||||
|
<programming-language>C</programming-language>
|
||||||
|
<repository>
|
||||||
|
<GitRepository>
|
||||||
|
<location rdf:resource="git://git.code.sf.net/p/dhcpd-pools/code" />
|
||||||
|
<browse rdf:resource="https://sourceforge.net/p/dhcpd-pools/code/ci/master/tree/" />
|
||||||
|
</GitRepository>
|
||||||
|
</repository>
|
||||||
|
<License>
|
||||||
|
<rdfs:label>BSD 2-Clause license</rdfs:label>
|
||||||
|
<rdfs:seeAlso rdf:resource="https://sourceforge.net/p/dhcpd-pools/code/ci/master/tree/COPYING?format=raw" />
|
||||||
|
</License>
|
||||||
|
<maintainer>
|
||||||
|
<foaf:Person>
|
||||||
|
<foaf:name>Sami Kerola</foaf:name>
|
||||||
|
<foaf:mbox rdf:resource="mailto:kerolasa@iki.fi" />
|
||||||
|
</foaf:Person>
|
||||||
|
</maintainer>
|
||||||
|
</Project>
|
||||||
|
</rdf:RDF>
|
||||||
5
samples/Makemodule.am
Normal file
5
samples/Makemodule.am
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
dist_doc_DATA = \
|
||||||
|
samples/dhcpd.conf \
|
||||||
|
samples/dhcpd.leases \
|
||||||
|
samples/mustach.template \
|
||||||
|
samples/prometheus.template
|
||||||
31
samples/dhcpd.conf
Normal file
31
samples/dhcpd.conf
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
shared-network example1 {
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.0.0.1 10.0.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.1.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.1.0.1 10.1.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
shared-network example2 {
|
||||||
|
subnet 10.2.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.2.0.1 10.2.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.3.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.3.0.1 10.3.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
subnet 10.4.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.4.0.1 10.4.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
210
samples/dhcpd.leases
Normal file
210
samples/dhcpd.leases
Normal file
|
|
@ -0,0 +1,210 @@
|
||||||
|
lease 10.0.0.0 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:00;
|
||||||
|
}
|
||||||
|
lease 10.0.0.1 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:01;
|
||||||
|
}
|
||||||
|
lease 10.0.0.2 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:02;
|
||||||
|
}
|
||||||
|
lease 10.0.0.3 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:03;
|
||||||
|
}
|
||||||
|
lease 10.0.0.4 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:04;
|
||||||
|
}
|
||||||
|
lease 10.0.0.5 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:05;
|
||||||
|
}
|
||||||
|
lease 10.0.0.6 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:06;
|
||||||
|
}
|
||||||
|
lease 10.0.0.7 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:07;
|
||||||
|
}
|
||||||
|
lease 10.0.0.8 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:08;
|
||||||
|
}
|
||||||
|
lease 10.0.0.9 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:09;
|
||||||
|
}
|
||||||
|
lease 10.0.0.10 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:10;
|
||||||
|
}
|
||||||
|
lease 10.0.0.11 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:00:11;
|
||||||
|
}
|
||||||
|
lease 10.0.0.12 {
|
||||||
|
binding state backup;
|
||||||
|
hardware ethernet 00:00:00:00:00:12;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
lease 10.1.0.0 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:00;
|
||||||
|
}
|
||||||
|
lease 10.1.0.1 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:01;
|
||||||
|
}
|
||||||
|
lease 10.1.0.2 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:02;
|
||||||
|
}
|
||||||
|
lease 10.1.0.3 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:03;
|
||||||
|
}
|
||||||
|
lease 10.1.0.4 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:04;
|
||||||
|
}
|
||||||
|
lease 10.1.0.5 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:05;
|
||||||
|
}
|
||||||
|
lease 10.1.0.6 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:06;
|
||||||
|
}
|
||||||
|
lease 10.1.0.7 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:07;
|
||||||
|
}
|
||||||
|
lease 10.1.0.8 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:08;
|
||||||
|
}
|
||||||
|
lease 10.1.0.9 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:09;
|
||||||
|
}
|
||||||
|
lease 10.1.0.10 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:01:10;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
lease 10.2.0.0 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:00;
|
||||||
|
}
|
||||||
|
lease 10.2.0.1 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:01;
|
||||||
|
}
|
||||||
|
lease 10.2.0.2 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:02;
|
||||||
|
}
|
||||||
|
lease 10.2.0.3 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:03;
|
||||||
|
}
|
||||||
|
lease 10.2.0.4 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:04;
|
||||||
|
}
|
||||||
|
lease 10.2.0.5 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:05;
|
||||||
|
}
|
||||||
|
lease 10.2.0.6 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:06;
|
||||||
|
}
|
||||||
|
lease 10.2.0.7 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:07;
|
||||||
|
}
|
||||||
|
lease 10.2.0.8 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:02:08;
|
||||||
|
}
|
||||||
|
|
||||||
|
lease 10.3.0.0 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:00;
|
||||||
|
}
|
||||||
|
lease 10.3.0.1 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:01;
|
||||||
|
}
|
||||||
|
lease 10.3.0.2 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:02;
|
||||||
|
}
|
||||||
|
lease 10.3.0.3 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:03;
|
||||||
|
}
|
||||||
|
lease 10.3.0.4 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:04;
|
||||||
|
}
|
||||||
|
lease 10.3.0.5 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:05;
|
||||||
|
}
|
||||||
|
lease 10.3.0.6 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:06;
|
||||||
|
}
|
||||||
|
lease 10.3.0.7 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:07;
|
||||||
|
}
|
||||||
|
lease 10.3.0.8 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:08;
|
||||||
|
}
|
||||||
|
lease 10.3.0.9 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:09;
|
||||||
|
}
|
||||||
|
lease 10.3.0.9 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:03:09;
|
||||||
|
}
|
||||||
|
|
||||||
|
lease 10.4.0.0 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:04:00;
|
||||||
|
}
|
||||||
|
lease 10.4.0.1 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:04:01;
|
||||||
|
}
|
||||||
|
lease 10.4.0.2 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:04:02;
|
||||||
|
}
|
||||||
|
lease 10.4.0.3 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:04:03;
|
||||||
|
}
|
||||||
|
lease 10.4.0.4 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:04:04;
|
||||||
|
}
|
||||||
|
lease 10.4.0.5 {
|
||||||
|
binding state active;
|
||||||
|
hardware ethernet 00:00:00:00:04:05;
|
||||||
|
}
|
||||||
|
lease 10.4.0.6 {
|
||||||
|
binding state backup;
|
||||||
|
hardware ethernet 00:00:00:00:04:06;
|
||||||
|
}
|
||||||
76
samples/mustach.template
Normal file
76
samples/mustach.template
Normal file
|
|
@ -0,0 +1,76 @@
|
||||||
|
Ethernets:{{#active_lease}}
|
||||||
|
macaddress: {{macaddress}} ip: {{ip}}{{/active_lease}}
|
||||||
|
|
||||||
|
Subnets:{{#subnets}}
|
||||||
|
location: {{location}}
|
||||||
|
range: {{range}}
|
||||||
|
first_ip: {{first_ip}}
|
||||||
|
last_ip: {{last_ip}}
|
||||||
|
used: {{used}}
|
||||||
|
touched: {{touched}}
|
||||||
|
defined: {{defined}}
|
||||||
|
free: {{free}}
|
||||||
|
percent: {{percent}}
|
||||||
|
touch_count: {{touch_count}}
|
||||||
|
touch_percent: {{touch_percent}}
|
||||||
|
backup_count: {{backup_count}}
|
||||||
|
backup_percent: {{backup_percent}}
|
||||||
|
status: {{status}}
|
||||||
|
gettimeofday: {{gettimeofday}}
|
||||||
|
lease_file_epoch_mtime: {{lease_file_epoch_mtime}}
|
||||||
|
{{/subnets}}
|
||||||
|
|
||||||
|
Shared-networks:{{#shared-networks}}
|
||||||
|
location: {{location}}
|
||||||
|
defined: {{defined}}
|
||||||
|
used: {{used}}
|
||||||
|
touched: {{touched}}
|
||||||
|
free: {{free}}
|
||||||
|
percent: {{percent}}
|
||||||
|
touch_count: {{touch_count}}
|
||||||
|
touch_percent: {{touch_percent}}
|
||||||
|
backup_count: {{backup_count}}
|
||||||
|
backup_percent: {{backup_percent}}
|
||||||
|
status: {{status}}
|
||||||
|
gettimeofday: {{gettimeofday}}
|
||||||
|
lease_file_epoch_mtime: {{lease_file_epoch_mtime}}
|
||||||
|
{{/shared-networks}}
|
||||||
|
|
||||||
|
Summary:{{#summary}}
|
||||||
|
location: {{location}}
|
||||||
|
defined: {{defined}}
|
||||||
|
used: {{used}}
|
||||||
|
touched: {{touched}}
|
||||||
|
free: {{free}}
|
||||||
|
percent: {{percent}}
|
||||||
|
touch_count: {{touch_count}}
|
||||||
|
touch_percent: {{touch_percent}}
|
||||||
|
backup_count: {{backup_count}}
|
||||||
|
backup_percent: {{backup_percent}}
|
||||||
|
status: {{status}}
|
||||||
|
gettimeofday: {{gettimeofday}}
|
||||||
|
lease_file_epoch_mtime: {{lease_file_epoch_mtime}}
|
||||||
|
{{/summary}}
|
||||||
|
|
||||||
|
localtime: {{localtime}}
|
||||||
|
number_of_ranges: {{number_of_ranges}}
|
||||||
|
number_of_shared_networks: {{number_of_shared_networks}}
|
||||||
|
version: {{version}}
|
||||||
|
|
||||||
|
conf_file_path: {{conf_file_path}}
|
||||||
|
conf_file_local_mtime: {{conf_file_local_mtime}}
|
||||||
|
conf_file_epoch_mtime: {{conf_file_epoch_mtime}}
|
||||||
|
|
||||||
|
lease_file_path: {{lease_file_path}}
|
||||||
|
lease_file_local_mtime: {{lease_file_local_mtime}}
|
||||||
|
lease_file_epoch_mtime: {{lease_file_epoch_mtime}}
|
||||||
|
|
||||||
|
template_file_path: {{template_file_path}}
|
||||||
|
template_file_local_mtime: {{template_file_local_mtime}}
|
||||||
|
template_file_epoch_mtime: {{template_file_epoch_mtime}}
|
||||||
|
|
||||||
|
number_of_ranges_warning: {{number_of_ranges_warning}}
|
||||||
|
number_of_ranges_critical: {{number_of_ranges_critical}}
|
||||||
|
number_of_shared_networks: {{number_of_shared_networks}}
|
||||||
|
number_of_shared_networks_warning: {{number_of_shared_networks_warning}}
|
||||||
|
number_of_shared_networks_critical: {{number_of_shared_networks_critical}}
|
||||||
11
samples/prometheus.template
Normal file
11
samples/prometheus.template
Normal file
|
|
@ -0,0 +1,11 @@
|
||||||
|
# This mustach template can be used as Prometheus text file.
|
||||||
|
# https://prometheus.io/
|
||||||
|
# HELP dhcpd_pools ISC dhcpd statistics
|
||||||
|
# TYPE dhcpd_pools gauge
|
||||||
|
{{#subnets}}dhcpd_pools{location="{{location}}",range="{{first_ip}}",used="1"} {{used}} {{gettimeofday}}000
|
||||||
|
dhcpd_pools{location="{{location}}",range="{{first_ip}}",touched="1"} {{touched}} {{gettimeofday}}000
|
||||||
|
dhcpd_pools{location="{{location}}",range="{{first_ip}}",defined="1"} {{defined}} {{gettimeofday}}000
|
||||||
|
dhcpd_pools{location="{{location}}",range="{{first_ip}}",free="1"} {{free}} {{gettimeofday}}000
|
||||||
|
dhcpd_pools{location="{{location}}",range="{{first_ip}}",touch_count="1"} {{touch_count}} {{gettimeofday}}000
|
||||||
|
dhcpd_pools{location="{{location}}",range="{{first_ip}}",status="1"} {{status}} {{gettimeofday}}000
|
||||||
|
{{/subnets}}
|
||||||
12
src/.gitignore
vendored
12
src/.gitignore
vendored
|
|
@ -1,12 +0,0 @@
|
||||||
gmon.out
|
|
||||||
gmon.sum
|
|
||||||
*.c.gcov
|
|
||||||
*.gcda
|
|
||||||
*.gcno
|
|
||||||
*~
|
|
||||||
*.o
|
|
||||||
.deps/
|
|
||||||
dhcpd-pools
|
|
||||||
Makefile
|
|
||||||
Makefile.in
|
|
||||||
tags
|
|
||||||
16
src/.indent.pro
vendored
Normal file
16
src/.indent.pro
vendored
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
-linux
|
||||||
|
-TFILE
|
||||||
|
-Tconf_t
|
||||||
|
-Tipaddr_t
|
||||||
|
-Tleases_t
|
||||||
|
-Toff_t
|
||||||
|
-Trange_t
|
||||||
|
-Tsize_t
|
||||||
|
-Ttime_t
|
||||||
|
-Tuintmax_t
|
||||||
|
--blank-lines-after-declarations
|
||||||
|
--format-all-comments
|
||||||
|
--line-length100
|
||||||
|
--preprocessor-indentation1
|
||||||
|
--start-left-side-of-comments
|
||||||
|
--swallow-optional-blank-lines
|
||||||
|
|
@ -1,18 +0,0 @@
|
||||||
## Makefile.am -- Process this file with automake to produce Makefile.in
|
|
||||||
|
|
||||||
bin_PROGRAMS = dhcpd-pools
|
|
||||||
AC_PROG_RANLIB = resolv
|
|
||||||
AM_CPPFLAGS = -I$(top_srcdir)/src -I$(top_srcdir)/lib -I$(top_builddir)/lib
|
|
||||||
|
|
||||||
dhcpd_pools_LDADD = $(top_builddir)/lib/libdhcpd_pools.la
|
|
||||||
|
|
||||||
dhcpd_pools_SOURCES = \
|
|
||||||
analyze.c \
|
|
||||||
defaults.h \
|
|
||||||
dhcpd-pools.c \
|
|
||||||
dhcpd-pools.h \
|
|
||||||
getdata.c \
|
|
||||||
hash.c \
|
|
||||||
other.c \
|
|
||||||
output.c \
|
|
||||||
sort.c
|
|
||||||
24
src/Makemodule.am
Normal file
24
src/Makemodule.am
Normal file
|
|
@ -0,0 +1,24 @@
|
||||||
|
## Makefile.am -- Process this file with automake to produce Makefile.in
|
||||||
|
|
||||||
|
bin_PROGRAMS = dhcpd-pools
|
||||||
|
AC_PROG_RANLIB = resolv
|
||||||
|
AM_CPPFLAGS = -I$(top_srcdir)/src -I$(top_srcdir)/lib -I$(top_builddir)/lib
|
||||||
|
|
||||||
|
dhcpd_pools_LDADD = $(top_builddir)/lib/libdhcpd_pools.la $(MATH_LIBS)
|
||||||
|
|
||||||
|
dhcpd_pools_SOURCES = \
|
||||||
|
src/analyze.c \
|
||||||
|
src/dhcpd-pools.c \
|
||||||
|
src/dhcpd-pools.h \
|
||||||
|
src/getdata.c \
|
||||||
|
src/hash.c \
|
||||||
|
src/other.c \
|
||||||
|
src/output.c \
|
||||||
|
src/sort.c
|
||||||
|
|
||||||
|
if ENABLE_MUSTACH
|
||||||
|
dhcpd_pools_SOURCES += \
|
||||||
|
src/mustach-dhcpd-pools.c \
|
||||||
|
src/mustach.c \
|
||||||
|
src/mustach.h
|
||||||
|
endif
|
||||||
112
src/analyze.c
112
src/analyze.c
|
|
@ -33,6 +33,10 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*! \file analyze.c
|
||||||
|
* \brief Data analysis functions.
|
||||||
|
*/
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
|
|
@ -41,49 +45,34 @@
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
/* Clean up data */
|
/*! \brief Prepare data for analysis. The function will sort leases and
|
||||||
static int ip_sort(const struct leases_t *restrict a, const struct leases_t *restrict b)
|
* ranges. */
|
||||||
{
|
void prepare_data(struct conf_t *state)
|
||||||
if (a->ip < b->ip)
|
|
||||||
return -1;
|
|
||||||
if (a->ip > b->ip)
|
|
||||||
return 1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int prepare_data(void)
|
|
||||||
{
|
{
|
||||||
/* Sort leases */
|
/* Sort leases */
|
||||||
HASH_SORT(leases, ip_sort);
|
HASH_SORT(state->leases, leasecomp);
|
||||||
/* Sort ranges */
|
/* Sort ranges */
|
||||||
qsort(ranges, (size_t) num_ranges, sizeof(struct range_t), &rangecomp);
|
qsort(state->ranges, state->num_ranges, sizeof(struct range_t), &rangecomp);
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Join leases and ranges into counter structs */
|
/*!\brief Perform counting. Join leases with ranges, and update range and
|
||||||
int do_counting(void)
|
* shared network counters. */
|
||||||
|
void do_counting(struct conf_t *state)
|
||||||
{
|
{
|
||||||
struct range_t *restrict range_p;
|
struct range_t *restrict range_p = state->ranges;
|
||||||
const struct leases_t *restrict l = leases;
|
const struct leases_t *restrict l = state->leases;
|
||||||
unsigned long i, k, block_size;
|
unsigned long i;
|
||||||
|
double block_size;
|
||||||
unsigned long r_end;
|
|
||||||
range_p = ranges;
|
|
||||||
|
|
||||||
/* Walk through ranges */
|
/* Walk through ranges */
|
||||||
for (i = 0; i < num_ranges; i++) {
|
for (i = 0; i < state->num_ranges; i++, range_p++) {
|
||||||
for (; l != NULL && range_p->first_ip < l->ip; l = l->hh.prev)
|
while (l != NULL && ipcomp(&range_p->first_ip, &l->ip) < 0)
|
||||||
/* rewind */ ;
|
l = l->hh.prev; /* rewind */
|
||||||
if (l == NULL)
|
if (l == NULL)
|
||||||
l = leases;
|
l = state->leases;
|
||||||
/* last_ip + 1 make comparison to small bit quicker as it results to
|
for (; l != NULL && ipcomp(&l->ip, &range_p->last_ip) <= 0; l = l->hh.next) {
|
||||||
* be 'smaller than' not 'smaller or equal to' */
|
if (unlikely(ipcomp(&l->ip, &range_p->first_ip) < 0))
|
||||||
r_end = range_p->last_ip + 1;
|
continue; /* cannot happen? */
|
||||||
for (; l != NULL && l->ip < r_end; l = l->hh.next) {
|
|
||||||
if (l->ip < range_p->first_ip) {
|
|
||||||
/* should not be necessary */
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
/* IP in range */
|
/* IP in range */
|
||||||
switch (l->type) {
|
switch (l->type) {
|
||||||
case FREE:
|
case FREE:
|
||||||
|
|
@ -95,48 +84,23 @@ int do_counting(void)
|
||||||
case BACKUP:
|
case BACKUP:
|
||||||
range_p->backups++;
|
range_p->backups++;
|
||||||
break;
|
break;
|
||||||
}
|
default:
|
||||||
|
abort();
|
||||||
if (range_p->shared_net) {
|
|
||||||
switch (l->type) {
|
|
||||||
case FREE:
|
|
||||||
range_p->shared_net->touched++;
|
|
||||||
break;
|
|
||||||
case ACTIVE:
|
|
||||||
range_p->shared_net->used++;
|
|
||||||
break;
|
|
||||||
case BACKUP:
|
|
||||||
range_p->shared_net->backups++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
/* Size of range size. */
|
||||||
/* Size of range, shared net & all networks */
|
block_size = get_range_size(range_p);
|
||||||
block_size =
|
/* Count together ranges within shared network block. */
|
||||||
(unsigned int)(range_p->last_ip - range_p->first_ip + 1);
|
range_p->shared_net->available += block_size;
|
||||||
if (range_p->shared_net) {
|
range_p->shared_net->used += range_p->count;
|
||||||
range_p->shared_net->available += block_size;
|
range_p->shared_net->touched += range_p->touched;
|
||||||
|
range_p->shared_net->backups += range_p->backups;
|
||||||
|
/* When shared network is not 'all networks' add it as well. */
|
||||||
|
if (range_p->shared_net != state->shared_net_root) {
|
||||||
|
state->shared_net_root->available += block_size;
|
||||||
|
state->shared_net_root->used += range_p->count;
|
||||||
|
state->shared_net_root->touched += range_p->touched;
|
||||||
|
state->shared_net_root->backups += range_p->backups;
|
||||||
}
|
}
|
||||||
|
|
||||||
range_p++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* FIXME: During count of other shared networks default network and
|
|
||||||
* all networks got mixed together semantically. This fixes the
|
|
||||||
* problem, but is not elegant. */
|
|
||||||
shared_networks->available = 0;
|
|
||||||
shared_networks->used = 0;
|
|
||||||
shared_networks->touched = 0;
|
|
||||||
range_p = ranges;
|
|
||||||
for (k = 0; k < num_ranges; k++) {
|
|
||||||
shared_networks->available +=
|
|
||||||
range_p->last_ip - range_p->first_ip + 1;
|
|
||||||
shared_networks->used += range_p->count;
|
|
||||||
shared_networks->touched += range_p->touched;
|
|
||||||
shared_networks->backups += range_p->backups;
|
|
||||||
range_p++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,48 +0,0 @@
|
||||||
/*
|
|
||||||
* The dhcpd-pools has BSD 2-clause license which also known as "Simplified
|
|
||||||
* BSD License" or "FreeBSD License".
|
|
||||||
*
|
|
||||||
* Copyright 2006- Sami Kerola. All rights reserved.
|
|
||||||
*
|
|
||||||
* Redistribution and use in source and binary forms, with or without
|
|
||||||
* modification, are permitted provided that the following conditions are
|
|
||||||
* met:
|
|
||||||
*
|
|
||||||
* 1. Redistributions of source code must retain the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer.
|
|
||||||
*
|
|
||||||
* 2. Redistributions in binary form must reproduce the above copyright
|
|
||||||
* notice, this list of conditions and the following disclaimer in the
|
|
||||||
* documentation and/or other materials provided with the
|
|
||||||
* distribution.
|
|
||||||
*
|
|
||||||
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
||||||
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
||||||
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
||||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR AND CONTRIBUTORS OR
|
|
||||||
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
||||||
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
||||||
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
||||||
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
||||||
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
||||||
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
*
|
|
||||||
* The views and conclusions contained in the software and documentation are
|
|
||||||
* those of the authors and should not be interpreted as representing
|
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef DEFAULTS_H
|
|
||||||
# define DEFAULTS_H 1
|
|
||||||
|
|
||||||
/* Maximum line length in dhcpd.conf and dhcpd.leases */
|
|
||||||
static const size_t MAXLEN = 1024;
|
|
||||||
|
|
||||||
/* Maximum number of shared networks */
|
|
||||||
static const unsigned int SHARED_NETWORKS = 8192;
|
|
||||||
|
|
||||||
/* Maximum number of ranges */
|
|
||||||
unsigned int RANGES;
|
|
||||||
|
|
||||||
#endif /* DEFAULTS_H */
|
|
||||||
|
|
@ -33,143 +33,258 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*! \file dhcpd-pools.c
|
||||||
|
* \brief The main(), and core initialization.
|
||||||
|
*/
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <err.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <getopt.h>
|
#include <getopt.h>
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
#include "close-stream.h"
|
#include "close-stream.h"
|
||||||
#include "closeout.h"
|
#include "closeout.h"
|
||||||
#include "defaults.h"
|
#include "error.h"
|
||||||
#include "dhcpd-pools.h"
|
#include "progname.h"
|
||||||
|
#include "quote.h"
|
||||||
#include "xalloc.h"
|
#include "xalloc.h"
|
||||||
|
|
||||||
int main(int argc, char **argv)
|
#include "dhcpd-pools.h"
|
||||||
{
|
|
||||||
int i, c, sorts = 0;
|
/* Function pointers */
|
||||||
int option_index = 0;
|
int (*parse_ipaddr) (struct conf_t *state, const char *restrict src, union ipaddr_t *restrict dst);
|
||||||
char const *tmp;
|
void (*copy_ipaddr) (union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
||||||
struct range_t *tmp_ranges;
|
const char *(*ntop_ipaddr) (const union ipaddr_t *ip);
|
||||||
enum {
|
double (*get_range_size) (const struct range_t *r);
|
||||||
OPT_WARN = CHAR_MAX + 1,
|
int (*xstrstr) (struct conf_t *state, const char *restrict str);
|
||||||
OPT_CRIT
|
int (*ipcomp) (const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
||||||
};
|
int (*leasecomp) (const struct leases_t *restrict a, const struct leases_t *restrict b);
|
||||||
int ret_val;
|
void (*add_lease) (struct conf_t *state, union ipaddr_t *ip, enum ltype type);
|
||||||
|
struct leases_t *(*find_lease) (struct conf_t *state, union ipaddr_t *ip);
|
||||||
|
|
||||||
|
/*! \brief An option argument parser to populate state header_limit and
|
||||||
|
* number_limit values.
|
||||||
|
*/
|
||||||
|
static int return_limit(const char c)
|
||||||
|
{
|
||||||
|
if ('0' <= c && c < '8')
|
||||||
|
return c - '0';
|
||||||
|
error(EXIT_FAILURE, 0, "return_limit: output mask %s is illegal", quote(optarg));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Run time initialization. Global allocations, counter
|
||||||
|
* initializations, etc are here. */
|
||||||
|
static void prepare_memory(struct conf_t *state)
|
||||||
|
{
|
||||||
|
state->ranges = xmalloc(sizeof(struct range_t) * state->ranges_size);
|
||||||
|
/* First shared network entry is all networks */
|
||||||
|
state->shared_net_root = xcalloc(sizeof(struct shared_network_t), 1);
|
||||||
|
state->shared_net_root->name = xstrdup("All networks");
|
||||||
|
state->shared_net_head = state->shared_net_root;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief The --skip option argument parser. */
|
||||||
|
static void skip_arg_parse(struct conf_t *state, char *arg)
|
||||||
|
{
|
||||||
|
enum {
|
||||||
|
OPT_ARG_OK = 0,
|
||||||
|
OPT_ARG_WARNING,
|
||||||
|
OPT_ARG_CRITICAL,
|
||||||
|
OPT_ARG_MINSIZE,
|
||||||
|
OPT_ARG_SUPPRESSED
|
||||||
|
};
|
||||||
|
|
||||||
|
char *const tokens[] = {
|
||||||
|
[OPT_ARG_OK] = "ok",
|
||||||
|
[OPT_ARG_WARNING] = "warning",
|
||||||
|
[OPT_ARG_CRITICAL] = "critical",
|
||||||
|
[OPT_ARG_MINSIZE] = "minsize",
|
||||||
|
[OPT_ARG_SUPPRESSED] = "suppressed",
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
char *value;
|
||||||
|
|
||||||
|
while (*arg != '\0') {
|
||||||
|
switch (getsubopt(&arg, tokens, &value)) {
|
||||||
|
case OPT_ARG_OK:
|
||||||
|
state->skip_ok = 1;
|
||||||
|
break;
|
||||||
|
case OPT_ARG_WARNING:
|
||||||
|
state->skip_warning = 1;
|
||||||
|
break;
|
||||||
|
case OPT_ARG_CRITICAL:
|
||||||
|
state->skip_critical = 1;
|
||||||
|
break;
|
||||||
|
case OPT_ARG_MINSIZE:
|
||||||
|
state->skip_minsize = 1;
|
||||||
|
break;
|
||||||
|
case OPT_ARG_SUPPRESSED:
|
||||||
|
state->skip_suppressed = 1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
error(EXIT_FAILURE, 0, "unknown --skip specifier: %s", value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Command line options parser. */
|
||||||
|
static void parse_command_line_opts(struct conf_t *state, int argc, char **argv)
|
||||||
|
{
|
||||||
|
enum {
|
||||||
|
OPT_SNET_ALARMS = CHAR_MAX + 1,
|
||||||
|
OPT_WARN,
|
||||||
|
OPT_CRIT,
|
||||||
|
OPT_MINSIZE,
|
||||||
|
OPT_WARN_COUNT,
|
||||||
|
OPT_CRIT_COUNT,
|
||||||
|
OPT_COLOR,
|
||||||
|
OPT_SKIP,
|
||||||
|
OPT_SET_IPV,
|
||||||
|
OPT_MUSTACH
|
||||||
|
};
|
||||||
|
|
||||||
/* Options for getopt_long */
|
|
||||||
static struct option const long_options[] = {
|
static struct option const long_options[] = {
|
||||||
{"config", required_argument, NULL, 'c'},
|
{"config", required_argument, NULL, 'c'},
|
||||||
{"leases", required_argument, NULL, 'l'},
|
{"leases", required_argument, NULL, 'l'},
|
||||||
|
{"color", required_argument, NULL, OPT_COLOR},
|
||||||
|
{"skip", required_argument, NULL, OPT_SKIP},
|
||||||
{"format", required_argument, NULL, 'f'},
|
{"format", required_argument, NULL, 'f'},
|
||||||
{"sort", required_argument, NULL, 's'},
|
{"sort", required_argument, NULL, 's'},
|
||||||
{"reverse", no_argument, NULL, 'r'},
|
{"reverse", no_argument, NULL, 'r'},
|
||||||
{"output", required_argument, NULL, 'o'},
|
{"output", required_argument, NULL, 'o'},
|
||||||
{"limit", required_argument, NULL, 'L'},
|
{"limit", required_argument, NULL, 'L'},
|
||||||
|
{"mustach", required_argument, NULL, OPT_MUSTACH},
|
||||||
{"version", no_argument, NULL, 'v'},
|
{"version", no_argument, NULL, 'v'},
|
||||||
{"help", no_argument, NULL, 'h'},
|
{"help", no_argument, NULL, 'h'},
|
||||||
|
{"snet-alarms", no_argument, NULL, OPT_SNET_ALARMS},
|
||||||
{"warning", required_argument, NULL, OPT_WARN},
|
{"warning", required_argument, NULL, OPT_WARN},
|
||||||
{"critical", required_argument, NULL, OPT_CRIT},
|
{"critical", required_argument, NULL, OPT_CRIT},
|
||||||
|
{"warn-count", required_argument, NULL, OPT_WARN_COUNT},
|
||||||
|
{"crit-count", required_argument, NULL, OPT_CRIT_COUNT},
|
||||||
|
{"minsize", required_argument, NULL, OPT_MINSIZE},
|
||||||
|
{"perfdata", no_argument, NULL, 'p'},
|
||||||
|
{"all-as-shared", no_argument, NULL, 'A'},
|
||||||
|
{"ip-version", required_argument, NULL, OPT_SET_IPV},
|
||||||
{NULL, 0, NULL, 0}
|
{NULL, 0, NULL, 0}
|
||||||
};
|
};
|
||||||
|
int alarming = 0;
|
||||||
|
|
||||||
atexit(close_stdout);
|
|
||||||
|
|
||||||
/* FIXME: make these allocations dynamic up on need. */
|
|
||||||
config.dhcpdconf_file = xmalloc(sizeof(char) * MAXLEN);
|
|
||||||
config.dhcpdlease_file = xmalloc(sizeof(char) * MAXLEN);
|
|
||||||
config.output_file = xmalloc(sizeof(char) * MAXLEN);
|
|
||||||
|
|
||||||
/* Make sure string has zero length if there is no
|
|
||||||
* command line option */
|
|
||||||
config.output_file[0] = '\0';
|
|
||||||
/* Alarming defaults. */
|
|
||||||
config.warning = ALARM_WARN;
|
|
||||||
config.critical = ALARM_CRIT;
|
|
||||||
|
|
||||||
/* File location defaults */
|
|
||||||
strncpy(config.dhcpdconf_file, DHCPDCONF_FILE, MAXLEN - 1);
|
|
||||||
strncpy(config.dhcpdlease_file, DHCPDLEASE_FILE, MAXLEN - 1);
|
|
||||||
tmp = OUTPUT_LIMIT;
|
|
||||||
config.output_limit[0] = (*tmp - '0');
|
|
||||||
tmp++;
|
|
||||||
config.output_limit[1] = (*tmp - '0');
|
|
||||||
fullhtml = false;
|
|
||||||
|
|
||||||
/* Make sure some output format is selected by default */
|
|
||||||
strncpy(config.output_format, OUTPUT_FORMAT, (size_t) 1);
|
|
||||||
|
|
||||||
/* Default sort order is by IPs small to big */
|
|
||||||
config.reverse_order = false;
|
|
||||||
|
|
||||||
/* Parse command line options */
|
|
||||||
while (1) {
|
while (1) {
|
||||||
c = getopt_long(argc, argv, "c:l:f:o:s:rL:vh",
|
int c;
|
||||||
long_options, &option_index);
|
|
||||||
|
|
||||||
|
c = getopt_long(argc, argv, "c:l:f:o:s:rL:pAvh", long_options, NULL);
|
||||||
if (c == EOF)
|
if (c == EOF)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 'c':
|
case 'c':
|
||||||
/* config file */
|
/* config file */
|
||||||
strncpy(config.dhcpdconf_file, optarg, MAXLEN - 1);
|
state->dhcpdconf_file = optarg;
|
||||||
break;
|
break;
|
||||||
case 'l':
|
case 'l':
|
||||||
/* lease file */
|
/* lease file */
|
||||||
strncpy(config.dhcpdlease_file, optarg, MAXLEN - 1);
|
state->dhcpdlease_file = optarg;
|
||||||
break;
|
break;
|
||||||
case 'f':
|
case 'f':
|
||||||
/* Output format */
|
/* Output format */
|
||||||
strncpy(config.output_format, optarg, (size_t) 1);
|
state->output_format = optarg[0];
|
||||||
break;
|
break;
|
||||||
case 's':
|
case 's':
|
||||||
/* Output sorting option */
|
{
|
||||||
sorts = strlen(optarg);
|
/* Output sorting option */
|
||||||
if (5 < sorts) {
|
struct output_sort *p = state->sorts;
|
||||||
warnx
|
char *ptr = optarg;
|
||||||
("main: only first 5 sort orders will be used");
|
|
||||||
strncpy(config.sort, optarg, (size_t) 5);
|
while (p && p->next)
|
||||||
sorts = 5;
|
p = p->next;
|
||||||
} else {
|
while (*ptr) {
|
||||||
strncpy(config.sort, optarg, (size_t) sorts);
|
if (state->sorts == NULL) {
|
||||||
}
|
state->sorts =
|
||||||
for (i = 0; i < sorts; i++) {
|
xcalloc(1, sizeof(struct output_sort));
|
||||||
field_selector(config.sort[i]);
|
p = state->sorts;
|
||||||
|
} else {
|
||||||
|
p->next = xcalloc(1, sizeof(struct output_sort));
|
||||||
|
p = p->next;
|
||||||
|
}
|
||||||
|
p->func = field_selector(*ptr++);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 'r':
|
case 'r':
|
||||||
/* What ever sort in reverse order */
|
/* What ever sort in reverse order */
|
||||||
config.reverse_order = true;
|
state->reverse_order = 1;
|
||||||
break;
|
break;
|
||||||
case 'o':
|
case 'o':
|
||||||
/* Output file */
|
/* Output file */
|
||||||
strncpy(config.output_file, optarg, MAXLEN - 1);
|
state->output_file = optarg;
|
||||||
break;
|
break;
|
||||||
case 'L':
|
case 'L':
|
||||||
/* Specification what will be printed */
|
/* Specification what will be printed */
|
||||||
for (i = 0; i < 2; i++) {
|
state->header_limit = return_limit(optarg[0]);
|
||||||
if (optarg[i] >= '0' && optarg[i] < '8') {
|
state->number_limit = return_limit(optarg[1]);
|
||||||
config.output_limit[i] =
|
break;
|
||||||
optarg[i] - '0';
|
case OPT_MUSTACH:
|
||||||
} else {
|
#ifdef BUILD_MUSTACH
|
||||||
errx(EXIT_FAILURE,
|
state->mustach_template = optarg;
|
||||||
"main: output mask `%s' is illegal",
|
state->output_format = 'm';
|
||||||
optarg);
|
state->print_mac_addreses = 1;
|
||||||
}
|
#else
|
||||||
}
|
error(EXIT_FAILURE, 0, "compiled without mustach support");
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
|
case OPT_COLOR:
|
||||||
|
state->color_mode = parse_color_mode(optarg);
|
||||||
|
if (state->color_mode == color_unknown)
|
||||||
|
error(EXIT_FAILURE, errno, "unknown color mode: %s", quote(optarg));
|
||||||
|
break;
|
||||||
|
case OPT_SKIP:
|
||||||
|
skip_arg_parse(state, optarg);
|
||||||
|
break;
|
||||||
|
case OPT_SNET_ALARMS:
|
||||||
|
state->snet_alarms = 1;
|
||||||
break;
|
break;
|
||||||
case OPT_WARN:
|
case OPT_WARN:
|
||||||
strcpy(config.output_format, "a");
|
alarming = 1;
|
||||||
config.warning = strtod_or_err(optarg, "illegal argument");
|
state->warning = strtod_or_err(optarg, "illegal argument");
|
||||||
break;
|
break;
|
||||||
case OPT_CRIT:
|
case OPT_CRIT:
|
||||||
strcpy(config.output_format, "a");
|
alarming = 1;
|
||||||
config.critical = strtod_or_err(optarg, "illegal argument");
|
state->critical = strtod_or_err(optarg, "illegal argument");
|
||||||
|
break;
|
||||||
|
case OPT_WARN_COUNT:
|
||||||
|
alarming = 1;
|
||||||
|
state->warn_count = strtod_or_err(optarg, "illegal argument");
|
||||||
|
break;
|
||||||
|
case OPT_CRIT_COUNT:
|
||||||
|
alarming = 1;
|
||||||
|
state->crit_count = strtod_or_err(optarg, "illegal argument");
|
||||||
|
break;
|
||||||
|
case OPT_MINSIZE:
|
||||||
|
state->minsize = strtod_or_err(optarg, "illegal argument");
|
||||||
|
break;
|
||||||
|
case OPT_SET_IPV:
|
||||||
|
switch (optarg[0]) {
|
||||||
|
case '4':
|
||||||
|
set_ipv_functions(state, IPv4);
|
||||||
|
break;
|
||||||
|
case '6':
|
||||||
|
set_ipv_functions(state, IPv6);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
error(EXIT_FAILURE, 0, "unknown --ip-version argument: %s", optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'p':
|
||||||
|
/* Print additional performance data in alarming mode */
|
||||||
|
state->perfdata = 1;
|
||||||
|
break;
|
||||||
|
case 'A':
|
||||||
|
/* Treat single networks as shared with network CIDR as name */
|
||||||
|
state->all_as_shared = 1;
|
||||||
break;
|
break;
|
||||||
case 'v':
|
case 'v':
|
||||||
/* Print version */
|
/* Print version */
|
||||||
|
|
@ -178,78 +293,76 @@ int main(int argc, char **argv)
|
||||||
/* Print help */
|
/* Print help */
|
||||||
usage(EXIT_SUCCESS);
|
usage(EXIT_SUCCESS);
|
||||||
default:
|
default:
|
||||||
errx(EXIT_FAILURE,
|
error(EXIT_FAILURE, 0, "Try %s --help for more information.", program_name);
|
||||||
"Try `%s --help' for more information.",
|
|
||||||
program_invocation_short_name);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Output function selection */
|
/* Use default dhcpd.conf when user did not define anything. */
|
||||||
switch (config.output_format[0]) {
|
if (state->dhcpdconf_file == NULL)
|
||||||
case 't':
|
state->dhcpdconf_file = DHCPDCONF_FILE;
|
||||||
output_analysis = output_txt;
|
/* Use default dhcpd.leases when user did not define anything. */
|
||||||
break;
|
if (state->dhcpdlease_file == NULL)
|
||||||
case 'a':
|
state->dhcpdlease_file = DHCPDLEASE_FILE;
|
||||||
output_analysis = output_alarming;
|
/* Use default limits when user did not define anything. */
|
||||||
break;
|
if (state->header_limit == 8) {
|
||||||
case 'h':
|
char const *default_limit = OUTPUT_LIMIT;
|
||||||
output_analysis = output_html;
|
|
||||||
break;
|
state->header_limit = return_limit(default_limit[0]);
|
||||||
case 'H':
|
state->number_limit = return_limit(default_limit[1]);
|
||||||
output_analysis = output_html;
|
|
||||||
fullhtml = true;
|
|
||||||
break;
|
|
||||||
case 'x':
|
|
||||||
output_analysis = output_xml;
|
|
||||||
break;
|
|
||||||
case 'X':
|
|
||||||
output_analysis = output_xml;
|
|
||||||
break;
|
|
||||||
case 'c':
|
|
||||||
output_analysis = output_csv;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
errx(EXIT_FAILURE, "main: unknown output format `%c'",
|
|
||||||
config.output_format[0]);
|
|
||||||
}
|
}
|
||||||
|
/* Output format is not defined, if alarm thresholds are then it's alarming, else use the
|
||||||
|
* default. */
|
||||||
|
if (state->output_format == '\0') {
|
||||||
|
if (alarming == 1)
|
||||||
|
state->output_format = 'a';
|
||||||
|
else {
|
||||||
|
const char *const default_format = OUTPUT_FORMAT;
|
||||||
|
|
||||||
|
state->output_format = default_format[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (state->output_format == 'X' || state->output_format == 'J') {
|
||||||
|
state->print_mac_addreses = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*!\brief Start of execution. This will mostly call other functions one
|
||||||
|
* after another.
|
||||||
|
*
|
||||||
|
* \return Return value indicates success or fail or analysis, unless
|
||||||
|
* either --warning or --critical options are in use, which makes the
|
||||||
|
* return value in some cases to match with Nagios expectations about
|
||||||
|
* alarming. */
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
struct conf_t state = {
|
||||||
|
.warning = ALARM_WARN,
|
||||||
|
.critical = ALARM_CRIT,
|
||||||
|
.warn_count = 0x100000000, /* == 2^32 that is the entire IPv4 space */
|
||||||
|
.crit_count = 0x100000000, /* basically turns off the count criteria */
|
||||||
|
.header_limit = 8,
|
||||||
|
.ranges_size = 64,
|
||||||
|
.ip_version = IPvUNKNOWN,
|
||||||
|
.color_mode = color_auto
|
||||||
|
};
|
||||||
|
int ret_val;
|
||||||
|
|
||||||
|
atexit(close_stdout);
|
||||||
|
set_program_name(argv[0]);
|
||||||
|
prepare_memory(&state);
|
||||||
|
set_ipv_functions(&state, IPvUNKNOWN);
|
||||||
|
parse_command_line_opts(&state, argc, argv);
|
||||||
|
|
||||||
/* Do the job */
|
/* Do the job */
|
||||||
prepare_memory();
|
parse_config(&state, 1, state.dhcpdconf_file, state.shared_net_root);
|
||||||
parse_config(true, config.dhcpdconf_file, shared_networks);
|
parse_leases(&state);
|
||||||
|
prepare_data(&state);
|
||||||
parse_leases();
|
do_counting(&state);
|
||||||
prepare_data();
|
if (state.sorts != NULL)
|
||||||
do_counting();
|
mergesort_ranges(&state, state.ranges, state.num_ranges, NULL, 1);
|
||||||
tmp_ranges = xmalloc(sizeof(struct range_t) * num_ranges);
|
if (state.reverse_order == 1)
|
||||||
if (sorts != 0) {
|
flip_ranges(&state);
|
||||||
mergesort_ranges(ranges, num_ranges, tmp_ranges);
|
ret_val = output_analysis(&state);
|
||||||
}
|
clean_up(&state);
|
||||||
if (config.reverse_order == true) {
|
|
||||||
flip_ranges(ranges, tmp_ranges);
|
|
||||||
}
|
|
||||||
free(tmp_ranges);
|
|
||||||
ret_val = output_analysis();
|
|
||||||
|
|
||||||
clean_up();
|
|
||||||
return (ret_val);
|
return (ret_val);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Global allocations, counter resets etc */
|
|
||||||
int prepare_memory(void)
|
|
||||||
{
|
|
||||||
RANGES = 64;
|
|
||||||
num_ranges = num_shared_networks = 0;
|
|
||||||
shared_networks =
|
|
||||||
xmalloc(sizeof(struct shared_network_t) * SHARED_NETWORKS);
|
|
||||||
|
|
||||||
ranges = xmalloc(sizeof(struct range_t) * RANGES);
|
|
||||||
macaddr = NULL;
|
|
||||||
|
|
||||||
/* First shared network entry is all networks */
|
|
||||||
shared_networks->name = xstrdup("All networks");
|
|
||||||
shared_networks->used = 0;
|
|
||||||
shared_networks->touched = 0;
|
|
||||||
shared_networks->backups = 0;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -33,141 +33,340 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*! \file dhcpd-pools.h
|
||||||
|
* \brief Global definitions of structures, enums, and function prototypes.
|
||||||
|
*/
|
||||||
|
|
||||||
#ifndef DHCPD_POOLS_H
|
#ifndef DHCPD_POOLS_H
|
||||||
# define DHCPD_POOLS_H 1
|
# define DHCPD_POOLS_H 1
|
||||||
|
|
||||||
#include <config.h>
|
# include <config.h>
|
||||||
#include <arpa/inet.h>
|
# include <arpa/inet.h>
|
||||||
#include <stddef.h>
|
# include <stddef.h>
|
||||||
#include <stdio.h>
|
# include <stdio.h>
|
||||||
#include <string.h>
|
# include <string.h>
|
||||||
#include <uthash.h>
|
# include <uthash.h>
|
||||||
|
|
||||||
#ifdef HAVE_BUILTIN_EXPECT
|
/*! \def likely(x)
|
||||||
# define likely(x) __builtin_expect(!!(x), 1)
|
* \brief Symbolic call to __builtin_expect'ed branch.
|
||||||
# define unlikely(x) __builtin_expect(!!(x), 0)
|
*/
|
||||||
#else
|
/*! \def unlikely(x)
|
||||||
# define likely(x) (x)
|
* \brief Symbolic call to not-__builtin_expect'ed branch.
|
||||||
# define unlikely(x) (x)
|
*/
|
||||||
#endif
|
# ifdef HAVE_BUILTIN_EXPECT
|
||||||
|
# define likely(x) __builtin_expect(!!(x), 1)
|
||||||
|
# define unlikely(x) __builtin_expect(!!(x), 0)
|
||||||
|
# else
|
||||||
|
# define likely(x) (x)
|
||||||
|
# define unlikely(x) (x)
|
||||||
|
# endif
|
||||||
|
|
||||||
/* Structures and unions */
|
/*! \def _DP_ATTRIBUTE_HOT
|
||||||
struct configuration_t {
|
* \brief The function attribute __hot__ was added in gcc 4.3. See gnu
|
||||||
char *dhcpdconf_file;
|
* documentation for further information.
|
||||||
char *dhcpdlease_file;
|
* https://gcc.gnu.org/onlinedocs/gcc/Common-Function-Attributes.html#index-hot-function-attribute
|
||||||
char output_format[2];
|
*/
|
||||||
char sort[6];
|
# if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
||||||
int reverse_order;
|
# define _DP_ATTRIBUTE_HOT __attribute__ ((__hot__))
|
||||||
char *output_file;
|
# else
|
||||||
int output_limit[2];
|
# define _DP_ATTRIBUTE_HOT /* empty */
|
||||||
double warning;
|
# endif
|
||||||
double critical;
|
|
||||||
|
/*! \union ipaddr_t
|
||||||
|
* \brief Memory space for a binary IP address saving. */
|
||||||
|
union ipaddr_t {
|
||||||
|
uint32_t v4;
|
||||||
|
unsigned char v6[16];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*! \enum dhcp_version
|
||||||
|
* \brief The IP version, IPv4 or IPv6, served by the dhcpd.
|
||||||
|
*/
|
||||||
|
enum dhcp_version {
|
||||||
|
IPvUNKNOWN,
|
||||||
|
IPv4,
|
||||||
|
IPv6
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \enum prefix_t
|
||||||
|
* \brief Enumeration of interesting data in dhcpd.leases file, that has to
|
||||||
|
* be further examined, and saved. Functions xstrstr_v4() and xstrstr_v6()
|
||||||
|
* return one of these values to parse_leases().
|
||||||
|
*/
|
||||||
|
enum prefix_t {
|
||||||
|
PREFIX_LEASE,
|
||||||
|
PREFIX_BINDING_STATE_FREE,
|
||||||
|
PREFIX_BINDING_STATE_ABANDONED,
|
||||||
|
PREFIX_BINDING_STATE_EXPIRED,
|
||||||
|
PREFIX_BINDING_STATE_RELEASED,
|
||||||
|
PREFIX_BINDING_STATE_ACTIVE,
|
||||||
|
PREFIX_BINDING_STATE_BACKUP,
|
||||||
|
PREFIX_HARDWARE_ETHERNET,
|
||||||
|
PREFIX_STARTS,
|
||||||
|
PREFIX_ENDS,
|
||||||
|
PREFIX_HOSTNAME,
|
||||||
|
NUM_OF_PREFIX
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \enum color_mode
|
||||||
|
* \brief Enumeration whether to use or not color output.
|
||||||
|
*/
|
||||||
|
enum color_mode {
|
||||||
|
color_unknown,
|
||||||
|
color_off,
|
||||||
|
color_on,
|
||||||
|
color_auto /*!< Default, use colors when output terminal is interactive. */
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \struct shared_network_t
|
||||||
|
* \brief Counters for an individual shared network. This data entry is
|
||||||
|
* also used for 'all networks' counting.
|
||||||
|
*/
|
||||||
struct shared_network_t {
|
struct shared_network_t {
|
||||||
char *name;
|
char *name;
|
||||||
unsigned long int available;
|
double available;
|
||||||
unsigned long int used;
|
double used;
|
||||||
unsigned long int touched;
|
double touched;
|
||||||
unsigned long int backups;
|
double backups;
|
||||||
|
struct shared_network_t *next;
|
||||||
|
int netmask;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*! \struct range_t
|
||||||
|
* \brief Counters for an individual range.
|
||||||
|
*/
|
||||||
struct range_t {
|
struct range_t {
|
||||||
struct shared_network_t *shared_net;
|
struct shared_network_t *shared_net;
|
||||||
uint32_t first_ip;
|
union ipaddr_t first_ip;
|
||||||
uint32_t last_ip;
|
union ipaddr_t last_ip;
|
||||||
unsigned long int count;
|
double count;
|
||||||
unsigned long int touched;
|
double touched;
|
||||||
unsigned long int backups;
|
double backups;
|
||||||
};
|
};
|
||||||
struct macaddr_t {
|
|
||||||
char *ethernet;
|
/*! \struct output_helper_t
|
||||||
char *ip;
|
* \brief Various per range and shared net temporary calculation results.
|
||||||
struct macaddr_t *next;
|
*/
|
||||||
|
struct output_helper_t {
|
||||||
|
double range_size;
|
||||||
|
double percent;
|
||||||
|
double tc;
|
||||||
|
double tcp;
|
||||||
|
double bup;
|
||||||
|
int status;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*! \struct status_counts_t
|
||||||
|
* \brief Range and shared network alarming status counts.
|
||||||
|
*/
|
||||||
|
struct status_counts_t {
|
||||||
|
unsigned int warning;
|
||||||
|
unsigned int critical;
|
||||||
|
unsigned int ok;
|
||||||
|
unsigned int ignored;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \enum ltype
|
||||||
|
* \brief Lease state types. These are the possible values in struct leases_t.
|
||||||
|
*/
|
||||||
enum ltype {
|
enum ltype {
|
||||||
ACTIVE,
|
ACTIVE,
|
||||||
FREE,
|
FREE,
|
||||||
BACKUP
|
BACKUP
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*! \struct leases_t
|
||||||
|
* \brief An individual lease. These leaases are hashed.
|
||||||
|
*/
|
||||||
struct leases_t {
|
struct leases_t {
|
||||||
uint32_t ip; /* ip as key */
|
union ipaddr_t ip; /* ip as key */
|
||||||
enum ltype type;
|
char *ethernet;
|
||||||
UT_hash_handle hh;
|
UT_hash_handle hh;
|
||||||
|
enum ltype type;
|
||||||
|
char *ends;
|
||||||
|
char *starts;
|
||||||
|
char *hostname;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Global variables */
|
/*! \enum limbits
|
||||||
struct configuration_t config;
|
* \brief Output limit bits.
|
||||||
static int const output_limit_bit_1 = 1;
|
*/
|
||||||
static int const output_limit_bit_2 = 2;
|
enum limbits {
|
||||||
static int const output_limit_bit_3 = 4;
|
R_BIT = (1 << 0), /*!< Range limit. */
|
||||||
unsigned int fullhtml;
|
S_BIT = (1 << 1), /*!< Shared networks limit. */
|
||||||
struct shared_network_t *shared_networks;
|
A_BIT = (1 << 2) /*!< All networks summary limit. */
|
||||||
unsigned int num_shared_networks;
|
};
|
||||||
struct range_t *ranges;
|
|
||||||
unsigned int num_ranges;
|
/*! \def STATE_OK
|
||||||
struct leases_t *leases;
|
* \brief Nagios alarm exit value.
|
||||||
unsigned long int num_leases;
|
*/
|
||||||
unsigned long int num_touches;
|
# define STATE_OK 0
|
||||||
unsigned long int num_backups;
|
# define STATE_WARNING 1
|
||||||
struct macaddr_t *macaddr;
|
# define STATE_CRITICAL 2
|
||||||
|
|
||||||
|
/*! \var comparer_t
|
||||||
|
* \brief Function pointer holding sort algorithm.
|
||||||
|
*/
|
||||||
|
typedef int (*comparer_t) (struct range_t *r1, struct range_t *r2);
|
||||||
|
|
||||||
|
/*! \struct output_sort
|
||||||
|
* \brief Linked list of sort functions.
|
||||||
|
*/
|
||||||
|
struct output_sort {
|
||||||
|
comparer_t func;
|
||||||
|
struct output_sort *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \struct conf_t
|
||||||
|
* \brief Runtime configuration state.
|
||||||
|
*/
|
||||||
|
struct conf_t {
|
||||||
|
struct shared_network_t *shared_net_root; /*!< First entry in shared network linked list, that is the 'all networks', */
|
||||||
|
struct shared_network_t *shared_net_head; /*!< Last entry in shared network linked list. */
|
||||||
|
struct range_t *ranges; /*!< Array of ranges. */
|
||||||
|
unsigned int num_ranges; /*!< Number of ranges in the ranges array. */
|
||||||
|
enum dhcp_version ip_version; /*!< Designator if the dhcpd is running in IPv4 or IPv6 mode. */
|
||||||
|
size_t ranges_size; /*!< Size of the ranges array. */
|
||||||
|
struct leases_t *leases; /*!< An array of individual leases from dhcpd.leases file. */
|
||||||
|
const char *dhcpdconf_file; /*!< Path to dhcpd.conf file. */
|
||||||
|
const char *dhcpdlease_file; /*!< Path to dhcpd.leases file. */
|
||||||
|
struct output_sort *sorts; /*!< Linked list how to sort ranges. */
|
||||||
|
const char *output_file; /*!< Output file path. */
|
||||||
|
const char *mustach_template; /*!< Mustach template file path. */
|
||||||
|
double warning; /*!< Warning percent threshold. */
|
||||||
|
double critical; /*!< Critical percent threshold. */
|
||||||
|
double warn_count; /*!< Maximum number of free IP's before warning. */
|
||||||
|
double crit_count; /*!< Maximum number of free IP's before critical. */
|
||||||
|
double minsize; /*!< Minimum size of range or shared network to be considered exceeding threshold. */
|
||||||
|
int color_format; /*!< Column to use in color_tags array. */
|
||||||
|
char output_format; /*!< Output format, such as text, json, xml, .... */
|
||||||
|
uint32_t
|
||||||
|
print_mac_addreses:1, /*!< Print mac address in xml or json. */
|
||||||
|
reverse_order:1, /*!< Reverse sort order. */
|
||||||
|
backups_found:1, /*!< Indicator if dhcpd.leases file has leases in backup state. */
|
||||||
|
snet_alarms:1, /*!< Suppress alarming thresholds for ranges that are part of a shared network. */
|
||||||
|
perfdata:1, /*!< Include performance statistics when using Nagios alarm output format. */
|
||||||
|
all_as_shared:1, /*!< Treat stand-alone subnets as a shared network. */
|
||||||
|
header_limit:4, /*!< Bits to suppress header output. */
|
||||||
|
number_limit:3, /*!< Bits to suppress value output. */
|
||||||
|
skip_ok:1, /*!< Skip none-alarming values from output. */
|
||||||
|
skip_warning:1, /*!< Skip warning values from output. */
|
||||||
|
skip_critical:1, /*!< Skip critical values from output. */
|
||||||
|
skip_minsize:1, /*!< Skip alarming values that are below minsize from output. */
|
||||||
|
skip_suppressed:1, /*!< Skip alarming values that are suppressed with --snet-alarms option, or they are shared networks without IP availability. */
|
||||||
|
color_mode:2; /*!< Indicator if colors should be used in output. */
|
||||||
|
};
|
||||||
|
|
||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
int prepare_memory(void);
|
|
||||||
int parse_leases(void);
|
|
||||||
void parse_config(int, const char *__restrict, struct shared_network_t *__restrict)
|
|
||||||
__attribute__ ((nonnull(2, 3)));
|
|
||||||
void nth_field(char *__restrict dest, const char *__restrict src)
|
|
||||||
__attribute__ ((nonnull(1, 2)))
|
|
||||||
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
|
||||||
__attribute__ ((__hot__))
|
|
||||||
#endif
|
|
||||||
;
|
|
||||||
int prepare_data(void);
|
|
||||||
int do_counting(void);
|
|
||||||
void flip_ranges(struct range_t *__restrict ranges, struct range_t *__restrict tmp_ranges)
|
|
||||||
__attribute__ ((nonnull(1, 2)));
|
|
||||||
/* support functions */
|
|
||||||
int xstrstr(const char *__restrict a, const char *__restrict b, int len)
|
|
||||||
__attribute__ ((nonnull(1, 2)))
|
|
||||||
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
|
||||||
__attribute__ ((__hot__))
|
|
||||||
#endif
|
|
||||||
;
|
|
||||||
double strtod_or_err(const char *__restrict str, const char *__restrict errmesg);
|
|
||||||
void print_version(void) __attribute__ ((noreturn));
|
|
||||||
void usage(int status) __attribute__ ((noreturn));
|
|
||||||
/* qsort required functions... */
|
|
||||||
/* ...for ranges and... */
|
|
||||||
int intcomp(const void *__restrict x, const void *__restrict y) __attribute__ ((nonnull(1, 2)));
|
|
||||||
int rangecomp(const void *__restrict r1, const void *__restrict r2)
|
|
||||||
__attribute__ ((nonnull(1, 2)));
|
|
||||||
/* sort function pointer and functions */
|
|
||||||
int sort_name(void);
|
|
||||||
unsigned long int (*returner) (struct range_t r);
|
|
||||||
unsigned long int ret_ip(struct range_t r);
|
|
||||||
unsigned long int ret_cur(struct range_t r);
|
|
||||||
unsigned long int ret_max(struct range_t r);
|
|
||||||
unsigned long int ret_percent(struct range_t r);
|
|
||||||
unsigned long int ret_touched(struct range_t r);
|
|
||||||
unsigned long int ret_tc(struct range_t r);
|
|
||||||
unsigned long int ret_tcperc(struct range_t r);
|
|
||||||
void field_selector(char c);
|
|
||||||
int get_order(struct range_t *__restrict left, struct range_t *__restrict right)
|
|
||||||
__attribute__ ((nonnull(1, 2)));
|
|
||||||
void mergesort_ranges(struct range_t *__restrict orig, int size, struct range_t *__restrict temp)
|
|
||||||
__attribute__ ((nonnull(1, 3)));
|
|
||||||
/* output function pointer and functions */
|
|
||||||
int (*output_analysis) (void);
|
|
||||||
int output_txt(void);
|
|
||||||
int output_html(void);
|
|
||||||
int output_xml(void);
|
|
||||||
int output_csv(void);
|
|
||||||
int output_alarming(void);
|
|
||||||
/* Memory release, file closing etc */
|
|
||||||
void clean_up(void);
|
|
||||||
/* Hash functions */
|
|
||||||
void add_lease(int ip, enum ltype type);
|
|
||||||
struct leases_t * find_lease(int ip);
|
|
||||||
void delete_lease(struct leases_t * lease);
|
|
||||||
void delete_all_leases();
|
|
||||||
|
|
||||||
#endif /* DHCPD_POOLS_H */
|
/* analyze.c */
|
||||||
|
extern void prepare_data(struct conf_t *state);
|
||||||
|
extern void do_counting(struct conf_t *state);
|
||||||
|
|
||||||
|
/* getdata.c */
|
||||||
|
extern int parse_leases(struct conf_t *state);
|
||||||
|
extern void parse_config(struct conf_t *state, const int is_include,
|
||||||
|
const char *restrict config_file,
|
||||||
|
struct shared_network_t *restrict shared_p);
|
||||||
|
|
||||||
|
/* hash.c */
|
||||||
|
extern void (*add_lease) (struct conf_t *state, union ipaddr_t *addr, enum ltype type);
|
||||||
|
extern void add_lease_init(struct conf_t *state, union ipaddr_t *addr, enum ltype type);
|
||||||
|
extern void add_lease_v4(struct conf_t *state, union ipaddr_t *addr, enum ltype type);
|
||||||
|
extern void add_lease_v6(struct conf_t *state, union ipaddr_t *addr, enum ltype type);
|
||||||
|
|
||||||
|
extern struct leases_t *(*find_lease) (struct conf_t *state, union ipaddr_t *addr);
|
||||||
|
extern struct leases_t *find_lease_init(struct conf_t *state, union ipaddr_t *addr);
|
||||||
|
extern struct leases_t *find_lease_v4(struct conf_t *state, union ipaddr_t *addr);
|
||||||
|
extern struct leases_t *find_lease_v6(struct conf_t *state, union ipaddr_t *addr);
|
||||||
|
|
||||||
|
extern void delete_lease(struct conf_t *state, struct leases_t *lease);
|
||||||
|
extern void delete_all_leases(struct conf_t *state);
|
||||||
|
|
||||||
|
/* mustach-dhcpd-pools.c */
|
||||||
|
extern int mustach_dhcpd_pools(struct conf_t *state);
|
||||||
|
|
||||||
|
/* other.c */
|
||||||
|
extern void set_ipv_functions(struct conf_t *state, int version);
|
||||||
|
extern void flip_ranges(struct conf_t *state);
|
||||||
|
extern void clean_up(struct conf_t *state);
|
||||||
|
extern void parse_cidr(struct conf_t *state, struct range_t *range_p, const char *word);
|
||||||
|
extern int parse_color_mode(const char *restrict arg);
|
||||||
|
extern double strtod_or_err(const char *restrict str, const char *restrict errmesg);
|
||||||
|
extern void __attribute__ ((noreturn)) print_version(void);
|
||||||
|
extern void __attribute__ ((noreturn)) usage(int status);
|
||||||
|
extern void dp_time_tool(FILE *file, const char *path, int epoch);
|
||||||
|
|
||||||
|
extern int (*parse_ipaddr) (struct conf_t *state, const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst);
|
||||||
|
extern int parse_ipaddr_init(struct conf_t *state, const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst);
|
||||||
|
extern int parse_ipaddr_v4(struct conf_t *state, const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst);
|
||||||
|
extern int parse_ipaddr_v6(struct conf_t *state, const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst);
|
||||||
|
|
||||||
|
extern int (*xstrstr) (struct conf_t *state, const char *restrict str);
|
||||||
|
extern int xstrstr_init(struct conf_t *state, const char *restrict str);
|
||||||
|
extern int xstrstr_v4(struct conf_t *state, const char *restrict str);
|
||||||
|
extern int xstrstr_v6(struct conf_t *state, const char *restrict str);
|
||||||
|
|
||||||
|
extern void (*copy_ipaddr) (union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
||||||
|
extern void copy_ipaddr_init(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
||||||
|
extern void copy_ipaddr_v4(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
||||||
|
extern void copy_ipaddr_v6(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
||||||
|
|
||||||
|
extern const char *(*ntop_ipaddr) (const union ipaddr_t *ip);
|
||||||
|
extern const char *ntop_ipaddr_init(const union ipaddr_t *ip);
|
||||||
|
extern const char *ntop_ipaddr_v4(const union ipaddr_t *ip);
|
||||||
|
extern const char *ntop_ipaddr_v6(const union ipaddr_t *ip);
|
||||||
|
|
||||||
|
extern double (*get_range_size) (const struct range_t *r);
|
||||||
|
extern double get_range_size_init(const struct range_t *r);
|
||||||
|
extern double get_range_size_v4(const struct range_t *r);
|
||||||
|
extern double get_range_size_v6(const struct range_t *r);
|
||||||
|
|
||||||
|
/* output.c */
|
||||||
|
extern int range_output_helper(struct conf_t *state, struct output_helper_t *oh,
|
||||||
|
struct range_t *range_p);
|
||||||
|
extern int shnet_output_helper(struct conf_t *state, struct output_helper_t *oh,
|
||||||
|
struct shared_network_t *shared_p);
|
||||||
|
extern int output_analysis(struct conf_t *state);
|
||||||
|
extern void range_alarms(struct conf_t *state, struct status_counts_t *rangstat);
|
||||||
|
extern void shared_net_alarms(struct conf_t *state, struct status_counts_t *sharstat);
|
||||||
|
|
||||||
|
/* sort.c */
|
||||||
|
extern void mergesort_ranges(struct conf_t *state,
|
||||||
|
struct range_t *restrict orig, unsigned int size,
|
||||||
|
struct range_t *restrict temp, const int root_call);
|
||||||
|
|
||||||
|
extern int (*leasecomp) (const struct leases_t *restrict a, const struct leases_t *restrict b);
|
||||||
|
extern int leasecomp_init(const struct leases_t *restrict a
|
||||||
|
__attribute__ ((unused)),
|
||||||
|
const struct leases_t *restrict b __attribute__ ((unused)));
|
||||||
|
extern int leasecomp_v4(const struct leases_t *restrict a, const struct leases_t *restrict b);
|
||||||
|
extern int leasecomp_v6(const struct leases_t *restrict a, const struct leases_t *restrict b);
|
||||||
|
|
||||||
|
extern int (*ipcomp) (const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
||||||
|
extern int ipcomp_init(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
||||||
|
extern int ipcomp_v4(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
||||||
|
extern int ipcomp_v6(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
||||||
|
|
||||||
|
extern int rangecomp(const void *restrict r1, const void *restrict r2)
|
||||||
|
__attribute__ ((nonnull(1, 2)));
|
||||||
|
|
||||||
|
extern int comp_cur(struct range_t *r1, struct range_t *r2);
|
||||||
|
extern int comp_double(double f1, double f2);
|
||||||
|
extern int comp_ip(struct range_t *r1, struct range_t *r2);
|
||||||
|
extern int comp_max(struct range_t *r1, struct range_t *r2);
|
||||||
|
extern int comp_percent(struct range_t *r1, struct range_t *r2);
|
||||||
|
extern int comp_tc(struct range_t *r1, struct range_t *r2);
|
||||||
|
extern int comp_tcperc(struct range_t *r1, struct range_t *r2);
|
||||||
|
extern int comp_touched(struct range_t *r1, struct range_t *r2);
|
||||||
|
|
||||||
|
extern comparer_t field_selector(char c);
|
||||||
|
extern double ret_percent(struct range_t r);
|
||||||
|
extern double ret_tc(struct range_t r);
|
||||||
|
extern double ret_tcperc(struct range_t r);
|
||||||
|
|
||||||
|
#endif /* DHCPD_POOLS_H */
|
||||||
|
|
|
||||||
529
src/getdata.c
529
src/getdata.c
|
|
@ -33,218 +33,244 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*! \file getdata.c
|
||||||
|
* \brief Functions to read data from dhcpd.conf and dhcdp.leases files.
|
||||||
|
*/
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include <assert.h>
|
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
#include <err.h>
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
|
#include <limits.h>
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
|
||||||
#include "defaults.h"
|
#include "error.h"
|
||||||
#include "dhcpd-pools.h"
|
|
||||||
#include "xalloc.h"
|
#include "xalloc.h"
|
||||||
|
|
||||||
/* Parse dhcpd.leases file. All performance boosts for this function are
|
#include "dhcpd-pools.h"
|
||||||
* welcome */
|
|
||||||
int parse_leases(void)
|
/*! \enum dhcpd_magic_numbers
|
||||||
|
* \brief MAXLEN is maximum expected line length in dhcpd.conf and
|
||||||
|
* dhcpd.leases.
|
||||||
|
*/
|
||||||
|
enum dhcpd_magic_numbers {
|
||||||
|
MAXLEN = 1024
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \enum isc_conf_parser
|
||||||
|
* \brief Configuration file parsing state flags. The
|
||||||
|
* is_interesting_config_clause() will return one of these to parse_config().
|
||||||
|
*/
|
||||||
|
enum isc_conf_parser {
|
||||||
|
ITS_NOTHING_INTERESTING,
|
||||||
|
ITS_A_RANGE_FIRST_IP,
|
||||||
|
ITS_A_RANGE_SECOND_IP,
|
||||||
|
ITS_A_SHAREDNET,
|
||||||
|
ITS_AN_INCLUDE,
|
||||||
|
ITS_A_SUBNET,
|
||||||
|
ITS_A_NETMASK
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \brief Lease file parser. The parser can only read ISC DHCPD
|
||||||
|
* dhcpd.leases file format. */
|
||||||
|
int parse_leases(struct conf_t *state)
|
||||||
{
|
{
|
||||||
FILE *dhcpd_leases;
|
FILE *dhcpd_leases;
|
||||||
char *line, *ipstring, *macstring = NULL;
|
char *line, *ipstring, macstring[20], *stop, endsstr[30], startsstr[30], hostnamestr[MAXLEN];
|
||||||
struct in_addr inp;
|
union ipaddr_t addr;
|
||||||
struct stat lease_file_stats;
|
|
||||||
struct macaddr_t *macaddr_p = NULL;
|
|
||||||
int sw_active_lease = 0;
|
|
||||||
struct leases_t *lease;
|
struct leases_t *lease;
|
||||||
|
|
||||||
num_touches = num_leases = num_backups = 0;
|
dhcpd_leases = fopen(state->dhcpdlease_file, "r");
|
||||||
|
if (dhcpd_leases == NULL)
|
||||||
dhcpd_leases = fopen(config.dhcpdlease_file, "r");
|
error(EXIT_FAILURE, errno, "parse_leases: %s", state->dhcpdlease_file);
|
||||||
if (dhcpd_leases == NULL) {
|
|
||||||
err(EXIT_FAILURE, "parse_leases: %s", config.dhcpdlease_file);
|
|
||||||
}
|
|
||||||
#ifdef HAVE_POSIX_FADVISE
|
#ifdef HAVE_POSIX_FADVISE
|
||||||
# ifdef POSIX_FADV_WILLNEED
|
|
||||||
posix_fadvise(fileno(dhcpd_leases), 0, 0, POSIX_FADV_WILLNEED);
|
|
||||||
if (errno) {
|
|
||||||
err(EXIT_FAILURE, "parse_leases: fadvise %s",
|
|
||||||
config.dhcpdlease_file);
|
|
||||||
}
|
|
||||||
# endif /* POSIX_FADV_WILLNEED */
|
|
||||||
# ifdef POSIX_FADV_SEQUENTIAL
|
# ifdef POSIX_FADV_SEQUENTIAL
|
||||||
posix_fadvise(fileno(dhcpd_leases), 0, 0, POSIX_FADV_SEQUENTIAL);
|
if (posix_fadvise(fileno(dhcpd_leases), 0, 0, POSIX_FADV_SEQUENTIAL) != 0)
|
||||||
if (errno) {
|
error(EXIT_FAILURE, errno, "parse_leases: fadvise %s", state->dhcpdlease_file);
|
||||||
err(EXIT_FAILURE, "parse_leases: fadvise %s",
|
|
||||||
config.dhcpdlease_file);
|
|
||||||
}
|
|
||||||
# endif /* POSIX_FADV_SEQUENTIAL */
|
# endif /* POSIX_FADV_SEQUENTIAL */
|
||||||
#endif /* HAVE_POSIX_FADVISE */
|
#endif /* HAVE_POSIX_FADVISE */
|
||||||
|
|
||||||
/* I found out that there's one lease address per 300 bytes in
|
|
||||||
* dhcpd.leases file. Malloc is little bit pessimistic and uses 250.
|
|
||||||
* If someone has higher density in lease file I'm interested to
|
|
||||||
* hear about that. */
|
|
||||||
if (stat(config.dhcpdlease_file, &lease_file_stats)) {
|
|
||||||
err(EXIT_FAILURE, "parse_leases: %s", config.dhcpdlease_file);
|
|
||||||
}
|
|
||||||
|
|
||||||
line = xmalloc(sizeof(char) * MAXLEN);
|
line = xmalloc(sizeof(char) * MAXLEN);
|
||||||
|
line[0] = '\0';
|
||||||
ipstring = xmalloc(sizeof(char) * MAXLEN);
|
ipstring = xmalloc(sizeof(char) * MAXLEN);
|
||||||
if (config.output_format[0] == 'X') {
|
ipstring[0] = '\0';
|
||||||
macstring = xmalloc(sizeof(char) * 18);
|
endsstr[0] = '\0';
|
||||||
macaddr = xmalloc(sizeof(struct macaddr_t));
|
startsstr[0] = '\0';
|
||||||
macaddr_p = macaddr;
|
hostnamestr[0] = '\0';
|
||||||
macaddr_p->next = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (!feof(dhcpd_leases)) {
|
while (!feof(dhcpd_leases)) {
|
||||||
if (!fgets(line, MAXLEN, dhcpd_leases) && ferror(dhcpd_leases)) {
|
if (!fgets(line, MAXLEN, dhcpd_leases) && ferror(dhcpd_leases))
|
||||||
err(EXIT_FAILURE, "parse_leases: %s",
|
error(EXIT_FAILURE, errno, "parse_leases: %s", state->dhcpdlease_file);
|
||||||
config.dhcpdlease_file);
|
switch (xstrstr(state, line)) {
|
||||||
}
|
/* It's a lease, save IP */
|
||||||
/* It's a lease, save IP */
|
case PREFIX_LEASE:
|
||||||
if (xstrstr(line, "lease", 5)) {
|
stop =
|
||||||
memcpy(ipstring, line + 6, 16);
|
memccpy(ipstring,
|
||||||
nth_field(ipstring, ipstring);
|
line + (state->ip_version ==
|
||||||
inet_aton(ipstring, &inp);
|
IPv4 ? 6 : 9), ' ', strlen(line));
|
||||||
sw_active_lease = 0;
|
if (stop != NULL) {
|
||||||
continue;
|
--stop;
|
||||||
}
|
*stop = '\0';
|
||||||
if (xstrstr(line, " binding state free", 20)) {
|
}
|
||||||
|
parse_ipaddr(state, ipstring, &addr);
|
||||||
|
break;
|
||||||
|
case PREFIX_BINDING_STATE_FREE:
|
||||||
|
case PREFIX_BINDING_STATE_ABANDONED:
|
||||||
|
case PREFIX_BINDING_STATE_EXPIRED:
|
||||||
|
case PREFIX_BINDING_STATE_RELEASED:
|
||||||
|
if ((lease = find_lease(state, &addr)) != NULL)
|
||||||
|
delete_lease(state, lease);
|
||||||
|
add_lease(state, &addr, FREE);
|
||||||
|
break;
|
||||||
|
case PREFIX_BINDING_STATE_ACTIVE:
|
||||||
/* remove old entry, if exists */
|
/* remove old entry, if exists */
|
||||||
if ((lease = find_lease(ntohl(inp.s_addr))) != NULL) {
|
if ((lease = find_lease(state, &addr)) != NULL)
|
||||||
delete_lease(lease);
|
delete_lease(state, lease);
|
||||||
}
|
add_lease(state, &addr, ACTIVE);
|
||||||
add_lease(ntohl(inp.s_addr), FREE);
|
break;
|
||||||
continue;
|
case PREFIX_BINDING_STATE_BACKUP:
|
||||||
}
|
|
||||||
/* Copy IP to correct array */
|
|
||||||
if (xstrstr(line, " binding state active", 22)) {
|
|
||||||
/* remove old entry, if exists */
|
/* remove old entry, if exists */
|
||||||
if ((lease = find_lease(ntohl(inp.s_addr))) != NULL) {
|
if ((lease = find_lease(state, &addr)) != NULL)
|
||||||
delete_lease(lease);
|
delete_lease(state, lease);
|
||||||
|
add_lease(state, &addr, BACKUP);
|
||||||
|
state->backups_found = 1;
|
||||||
|
break;
|
||||||
|
case PREFIX_HARDWARE_ETHERNET:
|
||||||
|
if (state->print_mac_addreses == 0)
|
||||||
|
break;
|
||||||
|
memcpy(macstring, line + 20, 17);
|
||||||
|
macstring[17] = '\0';
|
||||||
|
if ((lease = find_lease(state, &addr)) != NULL) {
|
||||||
|
lease->ethernet = xstrdup(macstring);
|
||||||
|
lease->starts = xstrdup(startsstr);
|
||||||
|
startsstr[0] = '\0';
|
||||||
|
lease->ends = xstrdup(endsstr);
|
||||||
|
endsstr[0] = '\0';
|
||||||
}
|
}
|
||||||
add_lease(ntohl(inp.s_addr), ACTIVE);
|
break;
|
||||||
sw_active_lease = 1;
|
case PREFIX_ENDS:
|
||||||
continue;
|
if (state->print_mac_addreses == 0)
|
||||||
}
|
break;
|
||||||
if (xstrstr(line, " binding state backup", 22)) {
|
strncpy(endsstr, line + 7, sizeof(endsstr)-1);
|
||||||
/* remove old entry, if exists */
|
endsstr[strlen(endsstr)-2] = '\0';
|
||||||
if ((lease = find_lease(ntohl(inp.s_addr))) != NULL) {
|
break;
|
||||||
delete_lease(lease);
|
case PREFIX_STARTS:
|
||||||
}
|
if (state->print_mac_addreses == 0)
|
||||||
add_lease(ntohl(inp.s_addr), BACKUP);
|
break;
|
||||||
continue;
|
strncpy(startsstr, line + 9, sizeof(startsstr)-1);
|
||||||
}
|
startsstr[strlen(startsstr)-2] = '\0';
|
||||||
if ((macaddr != NULL)
|
break;
|
||||||
&& (sw_active_lease == 1)
|
case PREFIX_HOSTNAME:
|
||||||
&& (xstrstr(line, " hardware ethernet", 19))) {
|
if (state->print_mac_addreses == 0)
|
||||||
nth_field(macstring, line + 20);
|
break;
|
||||||
if (macstring) {
|
strncpy(hostnamestr, line + 19, sizeof(hostnamestr)-1);
|
||||||
macstring[17] = '\0';
|
hostnamestr[strlen(hostnamestr)-3] = '\0';
|
||||||
macaddr_p->ethernet = xstrdup(macstring);
|
if ((lease = find_lease(state, &addr)) != NULL) {
|
||||||
macaddr_p->ip = xstrdup(ipstring);
|
lease->hostname = xstrdup(hostnamestr);
|
||||||
macaddr_p->next =
|
|
||||||
xmalloc(sizeof(struct macaddr_t));
|
|
||||||
macaddr_p = macaddr_p->next;
|
|
||||||
macaddr_p->next = NULL;
|
|
||||||
}
|
}
|
||||||
|
hostnamestr[0] = '\0';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* do nothing */ ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#undef HAS_PREFIX
|
||||||
free(line);
|
free(line);
|
||||||
free(ipstring);
|
free(ipstring);
|
||||||
if (macaddr != NULL) {
|
|
||||||
free(macstring);
|
|
||||||
}
|
|
||||||
fclose(dhcpd_leases);
|
fclose(dhcpd_leases);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Like strcpy but for field which is separated by white spaces. */
|
/*! \brief Keyword search in dhcpd.conf file.
|
||||||
void nth_field(char *restrict dest, const char *restrict src)
|
* \param s A line from the dhcpd.conf file.
|
||||||
|
* \return Indicator what configuration was found. */
|
||||||
|
static int is_interesting_config_clause(struct conf_t *state, char const *restrict s)
|
||||||
{
|
{
|
||||||
size_t i, len;
|
if (strstr(s, "range"))
|
||||||
len = strlen(src);
|
return ITS_A_RANGE_FIRST_IP;
|
||||||
for (i = 0; i < len; i++) {
|
if (strstr(s, "shared-network"))
|
||||||
dest[i] = src[i];
|
return ITS_A_SHAREDNET;
|
||||||
if (unlikely(src[i] == ' ')) {
|
if (state->all_as_shared) {
|
||||||
dest[i] = '\0';
|
if (strstr(s, "subnet"))
|
||||||
break;
|
return ITS_A_SUBNET;
|
||||||
}
|
if (strstr(s, "netmask"))
|
||||||
|
return ITS_A_NETMASK;
|
||||||
|
}
|
||||||
|
if (strstr(s, "include"))
|
||||||
|
return ITS_AN_INCLUDE;
|
||||||
|
return ITS_NOTHING_INTERESTING;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Flip first and last IP in range if they are in unusual order.
|
||||||
|
*/
|
||||||
|
static void reorder_last_first(struct range_t *range_p)
|
||||||
|
{
|
||||||
|
if (ipcomp(&range_p->first_ip, &range_p->last_ip) > 0) {
|
||||||
|
union ipaddr_t tmp;
|
||||||
|
|
||||||
|
tmp = range_p->first_ip;
|
||||||
|
range_p->first_ip = range_p->last_ip;
|
||||||
|
range_p->last_ip = tmp;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* dhcpd.conf interesting words */
|
/*! \brief The dhcpd.conf file parser.
|
||||||
static int is_interesting_config_clause(char const *restrict s)
|
* FIXME: This spaghetti monster function needs to be rewrote at least
|
||||||
{
|
* ones more.
|
||||||
if (strstr(s, "range"))
|
*/
|
||||||
return 3;
|
void parse_config(struct conf_t *state, const int is_include, const char *restrict config_file,
|
||||||
if (strstr(s, "shared-network"))
|
|
||||||
return 1;
|
|
||||||
if (strstr(s, "include"))
|
|
||||||
return 4;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* FIXME: This spaghetti monster function need to be rewrote at least ones. */
|
|
||||||
void parse_config(int is_include, const char *restrict config_file,
|
|
||||||
struct shared_network_t *restrict shared_p)
|
struct shared_network_t *restrict shared_p)
|
||||||
{
|
{
|
||||||
FILE *dhcpd_config;
|
FILE *dhcpd_config;
|
||||||
bool newclause = true, comment = false;
|
int newclause = 1, comment = 0, one_ip_range = 0; /* booleans */
|
||||||
int quote = 0, braces = 0, argument = 0;
|
int quote = 0, braces = 0, argument = ITS_NOTHING_INTERESTING;
|
||||||
size_t i = 0;
|
size_t i = 0;
|
||||||
char *word, c;
|
char *word;
|
||||||
int braces_shared = 1000;
|
int braces_shared = 1000;
|
||||||
struct in_addr inp;
|
union ipaddr_t addr;
|
||||||
struct range_t *range_p;
|
struct range_t *range_p = NULL;
|
||||||
|
|
||||||
word = xmalloc(sizeof(char) * MAXLEN);
|
word = xmalloc(sizeof(char) * MAXLEN);
|
||||||
|
if (is_include)
|
||||||
if (is_include) {
|
|
||||||
/* Default place holder for ranges "All networks". */
|
/* Default place holder for ranges "All networks". */
|
||||||
shared_p->name = shared_networks->name;
|
shared_p->name = state->shared_net_root->name;
|
||||||
}
|
|
||||||
|
|
||||||
/* Open configuration file */
|
/* Open configuration file */
|
||||||
dhcpd_config = fopen(config_file, "r");
|
dhcpd_config = fopen(config_file, "r");
|
||||||
if (dhcpd_config == NULL) {
|
if (dhcpd_config == NULL) {
|
||||||
err(EXIT_FAILURE, "parse_config: %s", config_file);
|
if (is_include) {
|
||||||
|
error(0, errno, "cannot open inlude: %s", config_file);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
/* config if from command line, just exit with error */
|
||||||
|
error(EXIT_FAILURE, errno, "parse_config: %s", config_file);
|
||||||
}
|
}
|
||||||
#ifdef POSIX_FADV_WILLNEED
|
#ifdef HAVE_POSIX_FADVISE
|
||||||
posix_fadvise(fileno(dhcpd_config), 0, 0, POSIX_FADV_WILLNEED);
|
# ifdef POSIX_FADV_SEQUENTIAL
|
||||||
if (errno) {
|
if (posix_fadvise(fileno(dhcpd_config), 0, 0, POSIX_FADV_SEQUENTIAL) != 0)
|
||||||
err(EXIT_FAILURE, "parse_config: fadvise %s", config_file);
|
error(EXIT_FAILURE, errno, "parse_config: fadvise %s", config_file);
|
||||||
}
|
# endif /* POSIX_FADV_SEQUENTIAL */
|
||||||
#endif /* POSIX_FADV_WILLNEED */
|
#endif /* HAVE_POSIX_FADVISE */
|
||||||
#ifdef POSIX_FADV_SEQUENTIAL
|
|
||||||
posix_fadvise(fileno(dhcpd_config), 0, 0, POSIX_FADV_SEQUENTIAL);
|
|
||||||
if (errno) {
|
|
||||||
err(EXIT_FAILURE, "parse_config: fadvise %s", config_file);
|
|
||||||
}
|
|
||||||
#endif /* POSIX_FADV_SEQUENTIAL */
|
|
||||||
|
|
||||||
/* Very hairy stuff begins. */
|
/* Very hairy stuff begins. */
|
||||||
while (unlikely(!feof(dhcpd_config))) {
|
while (unlikely(!feof(dhcpd_config))) {
|
||||||
|
int c;
|
||||||
|
|
||||||
c = fgetc(dhcpd_config);
|
c = fgetc(dhcpd_config);
|
||||||
|
if (CHAR_MAX < c)
|
||||||
|
continue;
|
||||||
/* Certain characters are magical */
|
/* Certain characters are magical */
|
||||||
switch (c) {
|
switch (c) {
|
||||||
/* Handle comments if they are not quoted */
|
/* Handle comments if they are not quoted */
|
||||||
case '#':
|
case '#':
|
||||||
if (quote == false) {
|
if (quote == 0)
|
||||||
comment = true;
|
comment = 1;
|
||||||
}
|
|
||||||
continue;
|
continue;
|
||||||
case '"':
|
case '"':
|
||||||
if (comment == false) {
|
if (comment == 0) {
|
||||||
quote++;
|
quote++;
|
||||||
/* Either one or zero */
|
/* Either one or zero */
|
||||||
quote = quote % 2;
|
quote = quote % 2;
|
||||||
|
|
@ -253,19 +279,22 @@ void parse_config(int is_include, const char *restrict config_file,
|
||||||
case '\n':
|
case '\n':
|
||||||
/* New line resets comment section, but
|
/* New line resets comment section, but
|
||||||
* not if quoted */
|
* not if quoted */
|
||||||
if (quote == false) {
|
if (quote == 0)
|
||||||
comment = false;
|
comment = 0;
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case ';':
|
case ';':
|
||||||
/* Quoted colon does not mean new clause */
|
/* Quoted colon does not mean new clause */
|
||||||
if (0 < quote) {
|
if (0 < quote)
|
||||||
break;
|
break;
|
||||||
}
|
if (comment == 0
|
||||||
if (comment == false && argument != 2 && argument != 4) {
|
&& argument != ITS_A_RANGE_FIRST_IP
|
||||||
newclause = true;
|
&& argument != ITS_A_RANGE_SECOND_IP && argument != ITS_AN_INCLUDE) {
|
||||||
|
newclause = 1;
|
||||||
i = 0;
|
i = 0;
|
||||||
} else if (argument == 2) {
|
} else if (argument == ITS_A_RANGE_FIRST_IP && one_ip_range == 1) {
|
||||||
|
argument = ITS_A_RANGE_SECOND_IP;
|
||||||
|
c = ' ';
|
||||||
|
} else if (argument == ITS_A_RANGE_SECOND_IP && 0 < i) {
|
||||||
/* Range ends to ; and this hair in code
|
/* Range ends to ; and this hair in code
|
||||||
* make two ranges wrote together like...
|
* make two ranges wrote together like...
|
||||||
*
|
*
|
||||||
|
|
@ -273,152 +302,178 @@ void parse_config(int is_include, const char *restrict config_file,
|
||||||
*
|
*
|
||||||
* ...to be interpreted correctly. */
|
* ...to be interpreted correctly. */
|
||||||
c = ' ';
|
c = ' ';
|
||||||
|
break;
|
||||||
|
} else if (argument == ITS_A_RANGE_SECOND_IP && i == 0) {
|
||||||
|
if (!range_p) {
|
||||||
|
long int pos;
|
||||||
|
pos = ftell(dhcpd_config);
|
||||||
|
error(EXIT_FAILURE, 0, "parse_config: parsing failed at position: %ld", pos);
|
||||||
|
}
|
||||||
|
range_p->last_ip = range_p->first_ip;
|
||||||
|
goto newrange;
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
case '{':
|
case '{':
|
||||||
if (0 < quote) {
|
if (0 < quote)
|
||||||
break;
|
break;
|
||||||
}
|
if (comment == 0)
|
||||||
if (comment == 0) {
|
|
||||||
braces++;
|
braces++;
|
||||||
}
|
|
||||||
/* i == 0 detects word that ends to brace like:
|
/* i == 0 detects word that ends to brace like:
|
||||||
*
|
*
|
||||||
* shared-network DSL{ ... */
|
* shared-network DSL{ ... */
|
||||||
if (i == 0) {
|
if (i == 0) {
|
||||||
newclause = true;
|
newclause = 1;
|
||||||
continue;
|
continue;
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
case '}':
|
case '}':
|
||||||
if (0 < quote) {
|
if (0 < quote)
|
||||||
break;
|
break;
|
||||||
}
|
if (comment == 0) {
|
||||||
if (comment == false) {
|
|
||||||
braces--;
|
braces--;
|
||||||
/* End of shared-network */
|
/* End of shared-network */
|
||||||
if (braces_shared == braces) {
|
if (braces_shared == braces) {
|
||||||
/* FIXME: Using 1000 is lame, but
|
/* FIXME: Using 1000 is lame, but
|
||||||
* works. */
|
* works. */
|
||||||
braces_shared = 1000;
|
braces_shared = 1000;
|
||||||
shared_p = shared_networks;
|
shared_p = state->shared_net_root;
|
||||||
}
|
}
|
||||||
/* Not literally true, but works for this
|
/* Not literally 1, but works for this
|
||||||
* program */
|
* program */
|
||||||
newclause = true;
|
newclause = 1;
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Either inside comment or Nth word of clause. */
|
/* Either inside comment or Nth word of clause. */
|
||||||
if (comment == true || (newclause == false && argument == 0)) {
|
if (comment == 1 || (newclause == 0 && argument == ITS_NOTHING_INTERESTING))
|
||||||
continue;
|
continue;
|
||||||
}
|
|
||||||
/* Strip white spaces before new clause word. */
|
/* Strip white spaces before new clause word. */
|
||||||
if ((newclause == true || argument != 0) && isspace(c)
|
if ((newclause == 1 || argument != ITS_NOTHING_INTERESTING)
|
||||||
&& i == 0) {
|
&& isspace(c) && i == 0 && one_ip_range == 0)
|
||||||
continue;
|
continue;
|
||||||
}
|
|
||||||
/* Save to word which clause this is. */
|
/* Save to word which clause this is. */
|
||||||
if ((newclause == true || argument != 0)
|
if ((newclause == 1 || argument != ITS_NOTHING_INTERESTING)
|
||||||
&& (!isspace(c) || 0 < quote)) {
|
&& (!isspace(c) || 0 < quote)) {
|
||||||
word[i] = c;
|
word[i] = (char) c;
|
||||||
i++;
|
i++;
|
||||||
/* Long word which is almost causing overflow. None
|
/* Long word which is almost causing overflow. None
|
||||||
* of words are this long which the program is
|
* of words are this long which the program is
|
||||||
* searching. */
|
* searching. */
|
||||||
if (MAXLEN < i) {
|
if (MAXLEN == i) {
|
||||||
newclause = false;
|
newclause = 0;
|
||||||
i = 0;
|
i = 0;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* See if clause is something that parser is looking for. */
|
/* See if clause is something that parser is looking for. */
|
||||||
else if (newclause == true) {
|
else if (newclause == 1) {
|
||||||
/* Insert string end & set state */
|
/* Insert string end & set state */
|
||||||
word[i] = '\0';
|
word[i] = '\0';
|
||||||
if (word[i - 1] != '{') {
|
if (word[i - 1] != '{')
|
||||||
newclause = false;
|
newclause = 0;
|
||||||
}
|
|
||||||
i = 0;
|
i = 0;
|
||||||
argument = is_interesting_config_clause(word);
|
argument = is_interesting_config_clause(state, word);
|
||||||
|
if (argument == ITS_A_RANGE_FIRST_IP)
|
||||||
|
one_ip_range = 1;
|
||||||
}
|
}
|
||||||
/* words after range, shared-network or include */
|
/* words after range, shared-network or include */
|
||||||
else if (argument != 0) {
|
else if (argument != ITS_NOTHING_INTERESTING) {
|
||||||
word[i] = '\0';
|
word[i] = '\0';
|
||||||
newclause = false;
|
newclause = 0;
|
||||||
i = 0;
|
i = 0;
|
||||||
|
|
||||||
switch (argument) {
|
switch (argument) {
|
||||||
case 2:
|
case ITS_A_RANGE_SECOND_IP:
|
||||||
/* printf ("range 2nd ip: %s\n", word); */
|
/* printf ("range 2nd ip: %s\n", word); */
|
||||||
range_p = ranges + num_ranges;
|
range_p = state->ranges + state->num_ranges;
|
||||||
inet_aton(word, &inp);
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
argument = 0;
|
if (strchr(word, '/')) {
|
||||||
range_p->last_ip = ntohl(inp.s_addr);
|
parse_cidr(state, range_p, word);
|
||||||
|
one_ip_range = 0;
|
||||||
|
} else {
|
||||||
|
/* not cidr */
|
||||||
|
parse_ipaddr(state, word, &addr);
|
||||||
|
if (one_ip_range == 1) {
|
||||||
|
one_ip_range = 0;
|
||||||
|
copy_ipaddr(&range_p->first_ip, &addr);
|
||||||
|
}
|
||||||
|
copy_ipaddr(&range_p->last_ip, &addr);
|
||||||
|
reorder_last_first(range_p);
|
||||||
|
}
|
||||||
|
newrange:
|
||||||
range_p->count = 0;
|
range_p->count = 0;
|
||||||
range_p->touched = 0;
|
range_p->touched = 0;
|
||||||
range_p->backups = 0;
|
range_p->backups = 0;
|
||||||
range_p->shared_net = shared_p;
|
range_p->shared_net = shared_p;
|
||||||
num_ranges++;
|
state->num_ranges++;
|
||||||
if (RANGES < num_ranges + 1) {
|
if (state->ranges_size <= state->num_ranges) {
|
||||||
RANGES *= 2;
|
state->ranges_size *= 2;
|
||||||
ranges =
|
state->ranges = xrealloc(state->ranges, sizeof(struct range_t) * state->ranges_size);
|
||||||
xrealloc(ranges,
|
range_p = state->ranges + state->num_ranges;
|
||||||
sizeof(struct
|
|
||||||
range_t) *
|
|
||||||
RANGES);
|
|
||||||
range_p = ranges + num_ranges;
|
|
||||||
}
|
}
|
||||||
newclause = true;
|
newclause = 1;
|
||||||
break;
|
break;
|
||||||
case 3:
|
case ITS_A_RANGE_FIRST_IP:
|
||||||
/* printf ("range 1nd ip: %s\n", word); */
|
/* printf ("range 1st ip: %s\n", word); */
|
||||||
range_p = ranges + num_ranges;
|
range_p = state->ranges + state->num_ranges;
|
||||||
if (!(inet_aton(word, &inp))) {
|
if (!(parse_ipaddr(state, word, &addr)))
|
||||||
/* word was not ip, try
|
/* word was not ip, try again */
|
||||||
* again */
|
break;
|
||||||
|
copy_ipaddr(&range_p->first_ip, &addr);
|
||||||
|
one_ip_range = 0;
|
||||||
|
argument = ITS_A_RANGE_SECOND_IP;
|
||||||
|
break;
|
||||||
|
case ITS_A_SHAREDNET:
|
||||||
|
case ITS_A_SUBNET:
|
||||||
|
/* ignore subnets inside a shared-network */
|
||||||
|
if (argument == ITS_A_SUBNET && shared_p != state->shared_net_root) {
|
||||||
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
range_p->first_ip = ntohl(inp.s_addr);
|
state->shared_net_head->next = xcalloc(sizeof(struct shared_network_t), 1);
|
||||||
argument = 2;
|
state->shared_net_head = state->shared_net_head->next;
|
||||||
break;
|
shared_p = state->shared_net_head;
|
||||||
case 1:
|
|
||||||
/* printf ("shared-network named: %s\n", word); */
|
|
||||||
num_shared_networks++;
|
|
||||||
shared_p =
|
|
||||||
shared_networks + num_shared_networks;
|
|
||||||
shared_p->name = xstrdup(word);
|
shared_p->name = xstrdup(word);
|
||||||
shared_p->available = 0;
|
shared_p->netmask = (argument == ITS_A_SUBNET ? -1 : 0); /* do not fill in netmask */
|
||||||
shared_p->used = 0;
|
/* record network's mask too */
|
||||||
shared_p->touched = 0;
|
if (argument == ITS_A_SUBNET)
|
||||||
shared_p->backups = 0;
|
newclause = 1;
|
||||||
if (SHARED_NETWORKS < num_shared_networks + 2) {
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
/* FIXME: make this
|
|
||||||
* away by reallocating
|
|
||||||
* more space. */
|
|
||||||
errx(EXIT_FAILURE,
|
|
||||||
"parse_config: increase default.h SHARED_NETWORKS and recompile");
|
|
||||||
}
|
|
||||||
argument = 0;
|
|
||||||
braces_shared = braces;
|
braces_shared = braces;
|
||||||
break;
|
break;
|
||||||
case 4:
|
case ITS_A_NETMASK:
|
||||||
/* printf ("include file: %s\n", word); */
|
/* fill in only when requested to do so */
|
||||||
argument = 0;
|
if (shared_p->netmask) {
|
||||||
parse_config(false, word, shared_p);
|
if (!(parse_ipaddr(state, word, &addr)))
|
||||||
newclause = true;
|
break;
|
||||||
|
shared_p->netmask = 32;
|
||||||
|
while ((addr.v4 & 0x01) == 0) {
|
||||||
|
addr.v4 >>= 1;
|
||||||
|
shared_p->netmask--;
|
||||||
|
}
|
||||||
|
snprintf(word, MAXLEN-1, "%s/%d", shared_p->name, shared_p->netmask);
|
||||||
|
if (shared_p->name)
|
||||||
|
free(shared_p->name);
|
||||||
|
shared_p->name = xstrdup(word);
|
||||||
|
}
|
||||||
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
|
braces_shared = braces;
|
||||||
break;
|
break;
|
||||||
case 0:
|
case ITS_AN_INCLUDE:
|
||||||
|
/* printf ("include file: %s\n", word); */
|
||||||
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
|
parse_config(state, 0, word, shared_p);
|
||||||
|
newclause = 1;
|
||||||
|
break;
|
||||||
|
case ITS_NOTHING_INTERESTING:
|
||||||
/* printf ("nothing interesting: %s\n", word); */
|
/* printf ("nothing interesting: %s\n", word); */
|
||||||
argument = 0;
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
warnx("impossible occurred, report a bug");
|
puts("impossible occurred, report a bug");
|
||||||
assert(0);
|
abort();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
125
src/hash.c
125
src/hash.c
|
|
@ -33,49 +33,124 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
/*! \file hash.c
|
||||||
|
* \brief The leases hash functions. The hash sorting is key to make
|
||||||
|
* analysis happen as quick as possible..
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "xalloc.h"
|
#include "xalloc.h"
|
||||||
|
|
||||||
void add_lease(int ip, enum ltype type)
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
|
#define HASH_FIND_V6(head, findv6, out) HASH_FIND(hh, head, findv6, 16, out)
|
||||||
|
#define HASH_ADD_V6(head, v6field, add) HASH_ADD(hh, head, v6field, 16, add)
|
||||||
|
|
||||||
|
/*! \brief Add a lease to hash array.
|
||||||
|
* \param addr Binary IP to be added in leases hash.
|
||||||
|
* \param type Lease state of the IP. */
|
||||||
|
void add_lease_init(struct conf_t *state __attribute__ ((unused)), union ipaddr_t *addr
|
||||||
|
__attribute__ ((unused)), enum ltype type __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
struct leases_t *l;
|
|
||||||
l = xmalloc(sizeof(struct leases_t));
|
|
||||||
l->ip = ip;
|
|
||||||
l->type = type;
|
|
||||||
HASH_ADD_INT(leases, ip, l);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct leases_t *find_lease(int ip)
|
void add_lease_v4(struct conf_t *state, union ipaddr_t *addr, enum ltype type)
|
||||||
{
|
{
|
||||||
struct leases_t *l;
|
struct leases_t *l;
|
||||||
|
|
||||||
HASH_FIND_INT(leases, &ip, l);
|
l = xmalloc(sizeof(struct leases_t));
|
||||||
|
copy_ipaddr(&l->ip, addr);
|
||||||
|
l->type = type;
|
||||||
|
HASH_ADD_INT(state->leases, ip.v4, l);
|
||||||
|
l->ethernet = NULL;
|
||||||
|
l->ends = NULL;
|
||||||
|
l->starts = NULL;
|
||||||
|
l->hostname = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_lease_v6(struct conf_t *state, union ipaddr_t *addr, enum ltype type)
|
||||||
|
{
|
||||||
|
struct leases_t *l;
|
||||||
|
|
||||||
|
l = xmalloc(sizeof(struct leases_t));
|
||||||
|
copy_ipaddr(&l->ip, addr);
|
||||||
|
l->type = type;
|
||||||
|
HASH_ADD_V6(state->leases, ip.v6, l);
|
||||||
|
l->ethernet = NULL;
|
||||||
|
l->ends = NULL;
|
||||||
|
l->starts = NULL;
|
||||||
|
l->hostname = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Find pointer to lease from hash array.
|
||||||
|
* \param addr Binary IP searched from leases hash.
|
||||||
|
* \return A lease structure about requested IP, or NULL.
|
||||||
|
*/
|
||||||
|
struct leases_t *find_lease_init(struct conf_t *state __attribute__ ((unused)), union ipaddr_t *addr
|
||||||
|
__attribute__ ((unused)))
|
||||||
|
{
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct leases_t *find_lease_v4(struct conf_t *state, union ipaddr_t *addr)
|
||||||
|
{
|
||||||
|
struct leases_t *l;
|
||||||
|
|
||||||
|
HASH_FIND_INT(state->leases, &addr->v4, l);
|
||||||
return l;
|
return l;
|
||||||
}
|
}
|
||||||
|
|
||||||
void delete_lease(struct leases_t *lease)
|
struct leases_t *find_lease_v6(struct conf_t *state, union ipaddr_t *addr)
|
||||||
{
|
{
|
||||||
HASH_DEL(leases, lease);
|
struct leases_t *l;
|
||||||
|
|
||||||
|
HASH_FIND_V6(state->leases, &addr->v4, l);
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Delete a lease from hash array.
|
||||||
|
* \param lease Pointer to lease hash. */
|
||||||
|
void delete_lease(struct conf_t *state, struct leases_t *lease)
|
||||||
|
{
|
||||||
|
free(lease->ethernet);
|
||||||
|
free(lease->ends);
|
||||||
|
free(lease->starts);
|
||||||
|
free(lease->hostname);
|
||||||
|
HASH_DEL(state->leases, lease);
|
||||||
free(lease);
|
free(lease);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* uthash >= 1.9.2
|
/*! \brief Delete all leases from hash array. */
|
||||||
void delete_all_leases()
|
#ifdef HASH_ITER
|
||||||
|
void delete_all_leases(struct conf_t *state)
|
||||||
{
|
{
|
||||||
struct leases_t *l, *tmp;
|
struct leases_t *l, *tmp;
|
||||||
HASH_ITER(hh, leases, l, tmp) {
|
|
||||||
HASH_DEL(leases, l);
|
|
||||||
free(l);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
void delete_all_leases()
|
HASH_ITER(hh, state->leases, l, tmp) {
|
||||||
{
|
free(l->ethernet);
|
||||||
struct leases_t *l;
|
free(l->ends);
|
||||||
while (leases) {
|
free(l->starts);
|
||||||
l = leases;
|
free(l->hostname);
|
||||||
HASH_DEL(leases, l); /* leases advances to next on delete */
|
HASH_DEL(state->leases, l);
|
||||||
free(l);
|
free(l);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
void delete_all_leases(struct conf_t *state)
|
||||||
|
{
|
||||||
|
while (state->leases) {
|
||||||
|
struct leases_t *l;
|
||||||
|
|
||||||
|
l = state->leases;
|
||||||
|
free(l->ethernet);
|
||||||
|
free(l->ends);
|
||||||
|
free(l->starts);
|
||||||
|
free(l->hostname);
|
||||||
|
HASH_DEL(state->leases, l); /* leases advances to next on delete */
|
||||||
|
free(l);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
|
||||||
519
src/mustach-dhcpd-pools.c
Normal file
519
src/mustach-dhcpd-pools.c
Normal file
|
|
@ -0,0 +1,519 @@
|
||||||
|
/*
|
||||||
|
* The dhcpd-pools has BSD 2-clause license which also known as "Simplified
|
||||||
|
* BSD License" or "FreeBSD License".
|
||||||
|
*
|
||||||
|
* Copyright 2006- Sami Kerola. All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are
|
||||||
|
* met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the
|
||||||
|
* distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
||||||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR AND CONTRIBUTORS OR
|
||||||
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||||||
|
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
||||||
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
||||||
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
||||||
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*
|
||||||
|
* The views and conclusions contained in the software and documentation are
|
||||||
|
* those of the authors and should not be interpreted as representing
|
||||||
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*! \file mustach-dhcpd-pools.c
|
||||||
|
* \brief Mustache templating specific functions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <error.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "close-stream.h"
|
||||||
|
#include "dhcpd-pools.h"
|
||||||
|
#include "error.h"
|
||||||
|
#include "mustach.h"
|
||||||
|
#include "xalloc.h"
|
||||||
|
|
||||||
|
/*! \struct expl
|
||||||
|
* \brief A structure that travels through mustach via closure void pointer.
|
||||||
|
*/
|
||||||
|
struct expl {
|
||||||
|
struct conf_t *state;
|
||||||
|
struct leases_t *lease_p;
|
||||||
|
struct range_t *range_p;
|
||||||
|
struct shared_network_t *shnet_p;
|
||||||
|
struct output_helper_t oh;
|
||||||
|
int current;
|
||||||
|
uint32_t pad;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int must_enter(void *closure, const char *name);
|
||||||
|
static int must_leave(void *closure);
|
||||||
|
|
||||||
|
/*! \brief Template base level tag parser and printer. */
|
||||||
|
static int must_put_base(void *closure, const char *name, int escape
|
||||||
|
__attribute__ ((unused)), FILE *file)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
if (!strcmp(name, "localtime")) {
|
||||||
|
dp_time_tool(file, NULL, 0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "number_of_ranges")) {
|
||||||
|
fprintf(file, "%u", e->state->num_ranges);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "number_of_ranges_warning")) {
|
||||||
|
struct status_counts_t stat = { 0 };
|
||||||
|
|
||||||
|
range_alarms(e->state, &stat);
|
||||||
|
fprintf(file, "%u", stat.warning);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "number_of_ranges_critical")) {
|
||||||
|
struct status_counts_t stat = { 0 };
|
||||||
|
|
||||||
|
range_alarms(e->state, &stat);
|
||||||
|
fprintf(file, "%u", stat.critical);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "number_of_shared_networks")) {
|
||||||
|
uint32_t num = 0;
|
||||||
|
struct shared_network_t *shared_p;
|
||||||
|
|
||||||
|
for (shared_p = e->state->shared_net_root->next; shared_p;
|
||||||
|
shared_p = shared_p->next) {
|
||||||
|
num++;
|
||||||
|
}
|
||||||
|
fprintf(file, "%u", num);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "number_of_shared_networks_warning")) {
|
||||||
|
struct status_counts_t stat = { 0 };
|
||||||
|
|
||||||
|
shared_net_alarms(e->state, &stat);
|
||||||
|
fprintf(file, "%u", stat.warning);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "number_of_shared_networks_critical")) {
|
||||||
|
struct status_counts_t stat = { 0 };
|
||||||
|
|
||||||
|
shared_net_alarms(e->state, &stat);
|
||||||
|
fprintf(file, "%u", stat.critical);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "version")) {
|
||||||
|
fprintf(file, "%s", PACKAGE_VERSION);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
/* lease file */
|
||||||
|
if (!strcmp(name, "lease_file_path")) {
|
||||||
|
fprintf(file, "%s", e->state->dhcpdlease_file);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "lease_file_local_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->dhcpdlease_file, 0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "lease_file_epoch_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->dhcpdlease_file, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
/* conf file */
|
||||||
|
if (!strcmp(name, "conf_file_path")) {
|
||||||
|
fprintf(file, "%s", e->state->dhcpdconf_file);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "conf_file_local_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->dhcpdconf_file, 0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "conf_file_epoch_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->dhcpdconf_file, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
/* template file */
|
||||||
|
if (!strcmp(name, "template_file_path")) {
|
||||||
|
fprintf(file, "%s", e->state->mustach_template);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "template_file_local_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->mustach_template, 0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "template_file_epoch_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->mustach_template, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unexpected tag: %s", name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \struct mustach_itf
|
||||||
|
* \brief Mustach function pointers. */
|
||||||
|
static struct mustach_itf itf = {
|
||||||
|
.start = NULL,
|
||||||
|
.enter = must_enter,
|
||||||
|
.put = must_put_base,
|
||||||
|
.next = NULL,
|
||||||
|
.leave = must_leave
|
||||||
|
};
|
||||||
|
|
||||||
|
/*! \brief Mustach active lease aka {{#active_lease}} tag parser and printer. */
|
||||||
|
static int must_put_active_lease(void *closure, const char *name, int escape
|
||||||
|
__attribute__((unused)), FILE *file)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
if (!strcmp(name, "ip")) {
|
||||||
|
fprintf(file, "%s", ntop_ipaddr(&e->lease_p->ip));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "macaddress")) {
|
||||||
|
fprintf(file, "%s", e->lease_p->ethernet);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unexpected tag: %s", name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief A function to move to next lease when {{/active_lease}} is encountered. */
|
||||||
|
static int must_next_active_lease(void *closure)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
e->lease_p = e->lease_p->hh.next;
|
||||||
|
if (e->lease_p == NULL)
|
||||||
|
return 0;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*! \brief Mustach range aka {{#subnets}} tag parser and printer. */
|
||||||
|
static int must_put_range(void *closure, const char *name, int escape
|
||||||
|
__attribute__ ((unused)), FILE *file)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
if (!strcmp(name, "location")) {
|
||||||
|
fprintf(file, "%s", e->range_p->shared_net->name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "range")) {
|
||||||
|
fprintf(file, "%s - ", ntop_ipaddr(&e->range_p->first_ip));
|
||||||
|
fprintf(file, "%s", ntop_ipaddr(&e->range_p->last_ip));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "first_ip")) {
|
||||||
|
fprintf(file, "%s", ntop_ipaddr(&e->range_p->first_ip));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "last_ip")) {
|
||||||
|
fprintf(file, "%s", ntop_ipaddr(&e->range_p->last_ip));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "used")) {
|
||||||
|
fprintf(file, "%g", e->range_p->count);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "touched")) {
|
||||||
|
fprintf(file, "%g", e->range_p->touched);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "defined")) {
|
||||||
|
fprintf(file, "%g", e->oh.range_size);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "free")) {
|
||||||
|
fprintf(file, "%g", e->oh.range_size - e->range_p->count);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "percent")) {
|
||||||
|
fprintf(file, "%g", e->oh.percent);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "touch_count")) {
|
||||||
|
fprintf(file, "%g", e->oh.tc);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "touch_percent")) {
|
||||||
|
fprintf(file, "%g", e->oh.tcp);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (e->state->backups_found == 1) {
|
||||||
|
if (!strcmp(name, "backup_count")) {
|
||||||
|
fprintf(file, "%g", e->range_p->backups);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "backup_percent")) {
|
||||||
|
fprintf(file, "%g", e->oh.bup);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "status")) {
|
||||||
|
fprintf(file, "%d", e->oh.status);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "gettimeofday")) {
|
||||||
|
dp_time_tool(file, NULL, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "lease_file_epoch_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->dhcpdlease_file, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unexpected tag: %s", name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Mustach shared networks aka {{#shared-networks}} tag parser and printer. */
|
||||||
|
static int must_put_shnet(void *closure, const char *name, int escape
|
||||||
|
__attribute__ ((unused)), FILE *file)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
if (!strcmp(name, "location")) {
|
||||||
|
fprintf(file, "%s", e->shnet_p->name);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "defined")) {
|
||||||
|
fprintf(file, "%g", e->shnet_p->available);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "used")) {
|
||||||
|
fprintf(file, "%g", e->shnet_p->used);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "touched")) {
|
||||||
|
fprintf(file, "%g", e->shnet_p->touched);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "free")) {
|
||||||
|
fprintf(file, "%g", e->shnet_p->available - e->shnet_p->used);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "percent")) {
|
||||||
|
fprintf(file, "%g", e->oh.percent);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "touch_count")) {
|
||||||
|
fprintf(file, "%g", e->oh.tc);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "touch_percent")) {
|
||||||
|
fprintf(file, "%g", e->oh.tcp);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (e->state->backups_found == 1) {
|
||||||
|
if (!strcmp(name, "backup_count")) {
|
||||||
|
fprintf(file, "%g", e->shnet_p->backups);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "backup_percent")) {
|
||||||
|
fprintf(file, "%g", e->oh.bup);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "status")) {
|
||||||
|
fprintf(file, "%d", e->oh.status);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "gettimeofday")) {
|
||||||
|
dp_time_tool(file, NULL, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "lease_file_epoch_mtime")) {
|
||||||
|
dp_time_tool(file, e->state->dhcpdlease_file, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unexpected tag: %s", name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief A function to move to next range when {{/subnets}} is encountered. */
|
||||||
|
static int must_next_range(void *closure)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
do {
|
||||||
|
e->range_p++;
|
||||||
|
e->current--;
|
||||||
|
if (e->current <= 0)
|
||||||
|
return 0;
|
||||||
|
} while (range_output_helper(e->state, &e->oh, e->range_p));
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief A function to move to next shared network when {{/shared-networks}}
|
||||||
|
* is encountered. */
|
||||||
|
static int must_next_shnet(void *closure)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
if (e->current == 1 || e->shnet_p == NULL)
|
||||||
|
return 0;
|
||||||
|
while (1) {
|
||||||
|
e->shnet_p = e->shnet_p->next;
|
||||||
|
if (e->shnet_p == NULL)
|
||||||
|
break;
|
||||||
|
if (shnet_output_helper(e->state, &e->oh, e->shnet_p))
|
||||||
|
continue;
|
||||||
|
else
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Function that is called when mustach is searching output loops from
|
||||||
|
* template file. */
|
||||||
|
static int must_enter(void *closure, const char *name)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
if (!strcmp(name, "active_lease")) {
|
||||||
|
itf.put = must_put_active_lease;
|
||||||
|
itf.next = must_next_active_lease;
|
||||||
|
e->current = 0;
|
||||||
|
e->lease_p = e->state->leases;
|
||||||
|
return must_next_active_lease(closure);
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "subnets")) {
|
||||||
|
itf.put = must_put_range;
|
||||||
|
itf.next = must_next_range;
|
||||||
|
e->current = e->state->num_ranges + 1;
|
||||||
|
e->range_p = e->state->ranges;
|
||||||
|
/* must_next_range() will skip_ok when needed */
|
||||||
|
e->range_p--;
|
||||||
|
return must_next_range(closure);
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "shared-networks")) {
|
||||||
|
itf.put = must_put_shnet;
|
||||||
|
itf.next = must_next_shnet;
|
||||||
|
e->shnet_p = e->state->shared_net_root;
|
||||||
|
e->current = 0;
|
||||||
|
return must_next_shnet(closure);
|
||||||
|
}
|
||||||
|
if (!strcmp(name, "summary")) {
|
||||||
|
itf.put = must_put_shnet;
|
||||||
|
itf.next = must_next_shnet;
|
||||||
|
e->shnet_p = e->state->shared_net_root;
|
||||||
|
e->current = 1;
|
||||||
|
shnet_output_helper(e->state, &e->oh, e->shnet_p);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unexpected tag: %s", name);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Function that is called when all elements within a print loop are outputted. */
|
||||||
|
static int must_leave(void *closure)
|
||||||
|
{
|
||||||
|
struct expl *e = closure;
|
||||||
|
|
||||||
|
e->shnet_p = e->state->shared_net_root;
|
||||||
|
e->range_p = e->state->ranges;
|
||||||
|
itf.put = must_put_base;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Read mustach template to memory. */
|
||||||
|
static char *must_read_template(const char *filename)
|
||||||
|
{
|
||||||
|
int f;
|
||||||
|
struct stat s;
|
||||||
|
char *result;
|
||||||
|
|
||||||
|
if (filename == NULL)
|
||||||
|
error(EXIT_FAILURE, 0, "must_read_template: --mustach argument missing");
|
||||||
|
if ((f = open(filename, O_RDONLY)) < 0) {
|
||||||
|
error(EXIT_FAILURE, errno, "must_read_template: open: %s", filename);
|
||||||
|
}
|
||||||
|
fstat(f, &s);
|
||||||
|
result = xmalloc(s.st_size + 1);
|
||||||
|
if (read(f, result, s.st_size) != s.st_size) {
|
||||||
|
error(EXIT_FAILURE, errno, "must_read_template: read: %s", filename);
|
||||||
|
}
|
||||||
|
close(f);
|
||||||
|
result[s.st_size] = 0;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Start mustach processing. */
|
||||||
|
int mustach_dhcpd_pools(struct conf_t *state)
|
||||||
|
{
|
||||||
|
struct expl e = { .state = state };
|
||||||
|
char *template;
|
||||||
|
FILE *outfile;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
template = must_read_template(state->mustach_template);
|
||||||
|
if (state->output_file) {
|
||||||
|
outfile = fopen(state->output_file, "w+");
|
||||||
|
if (outfile == NULL) {
|
||||||
|
error(EXIT_FAILURE, errno, "mustach_dhcpd_pools: fopen: %s",
|
||||||
|
state->output_file);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
outfile = stdout;
|
||||||
|
}
|
||||||
|
ret = fmustach(template, &itf, &e, outfile);
|
||||||
|
free(template);
|
||||||
|
if (outfile == stdout) {
|
||||||
|
if (fflush(stdout))
|
||||||
|
error(EXIT_FAILURE, errno, "mustach_dhcpd_pools: fflush");
|
||||||
|
} else {
|
||||||
|
if (close_stream(outfile))
|
||||||
|
error(EXIT_FAILURE, errno, "mustach_dhcpd_pools: fclose");
|
||||||
|
}
|
||||||
|
switch (ret) {
|
||||||
|
case MUSTACH_OK:
|
||||||
|
return 0;
|
||||||
|
case MUSTACH_ERROR_SYSTEM:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: system error");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_UNEXPECTED_END:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unexpected end");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_EMPTY_TAG:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: empty tag");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_TAG_TOO_LONG:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: too long tag");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_BAD_SEPARATORS:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: bad separator");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_TOO_DEEP:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: too deep");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_CLOSING:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: closing");
|
||||||
|
break;
|
||||||
|
case MUSTACH_ERROR_BAD_UNESCAPE_TAG:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: bad escape tag");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
error(EXIT_FAILURE, 0, "mustach_dhcpd_pools: fmustach: unknown error");
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
325
src/mustach.c
Normal file
325
src/mustach.c
Normal file
|
|
@ -0,0 +1,325 @@
|
||||||
|
/*
|
||||||
|
Author: José Bollo <jobol@nonadev.net>
|
||||||
|
Author: José Bollo <jose.bollo@iot.bzh>
|
||||||
|
|
||||||
|
https://gitlab.com/jobol/mustach
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define _GNU_SOURCE
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
|
#include "mustach.h"
|
||||||
|
|
||||||
|
#define NAME_LENGTH_MAX 1024
|
||||||
|
#define DEPTH_MAX 256
|
||||||
|
|
||||||
|
#if !defined(NO_OPEN_MEMSTREAM)
|
||||||
|
static FILE *memfile_open(char **buffer, size_t *size)
|
||||||
|
{
|
||||||
|
return open_memstream(buffer, size);
|
||||||
|
}
|
||||||
|
static void memfile_abort(FILE *file, char **buffer, size_t *size)
|
||||||
|
{
|
||||||
|
fclose(file);
|
||||||
|
free(*buffer);
|
||||||
|
*buffer = NULL;
|
||||||
|
*size = 0;
|
||||||
|
}
|
||||||
|
static int memfile_close(FILE *file, char **buffer, size_t *size)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
/* adds terminating null */
|
||||||
|
rc = fputc(0, file) ? MUSTACH_ERROR_SYSTEM : 0;
|
||||||
|
fclose(file);
|
||||||
|
if (rc == 0)
|
||||||
|
/* removes terminating null of the length */
|
||||||
|
(*size)--;
|
||||||
|
else {
|
||||||
|
free(*buffer);
|
||||||
|
*buffer = NULL;
|
||||||
|
*size = 0;
|
||||||
|
}
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
static FILE *memfile_open(char **buffer, size_t *size)
|
||||||
|
{
|
||||||
|
return tmpfile();
|
||||||
|
}
|
||||||
|
static void memfile_abort(FILE *file, char **buffer, size_t *size)
|
||||||
|
{
|
||||||
|
fclose(file);
|
||||||
|
*buffer = NULL;
|
||||||
|
*size = 0;
|
||||||
|
}
|
||||||
|
static int memfile_close(FILE *file, char **buffer, size_t *size)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
size_t s;
|
||||||
|
char *b;
|
||||||
|
|
||||||
|
s = (size_t)ftell(file);
|
||||||
|
b = malloc(s + 1);
|
||||||
|
if (b == NULL) {
|
||||||
|
rc = MUSTACH_ERROR_SYSTEM;
|
||||||
|
errno = ENOMEM;
|
||||||
|
s = 0;
|
||||||
|
} else {
|
||||||
|
rewind(file);
|
||||||
|
if (1 == fread(b, s, 1, file)) {
|
||||||
|
rc = 0;
|
||||||
|
b[s] = 0;
|
||||||
|
} else {
|
||||||
|
rc = MUSTACH_ERROR_SYSTEM;
|
||||||
|
free(b);
|
||||||
|
b = NULL;
|
||||||
|
s = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*buffer = b;
|
||||||
|
*size = s;
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static int getpartial(struct mustach_itf *itf, void *closure, const char *name, char **result)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
FILE *file;
|
||||||
|
size_t size;
|
||||||
|
|
||||||
|
*result = NULL;
|
||||||
|
file = memfile_open(result, &size);
|
||||||
|
if (file == NULL)
|
||||||
|
rc = MUSTACH_ERROR_SYSTEM;
|
||||||
|
else {
|
||||||
|
rc = itf->put(closure, name, 0, file);
|
||||||
|
if (rc < 0)
|
||||||
|
memfile_abort(file, result, &size);
|
||||||
|
else
|
||||||
|
rc = memfile_close(file, result, &size);
|
||||||
|
}
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int process(const char *template, struct mustach_itf *itf, void *closure, FILE *file, const char *opstr, const char *clstr)
|
||||||
|
{
|
||||||
|
char name[NAME_LENGTH_MAX + 1], *partial, c, *tmp;
|
||||||
|
const char *beg, *term;
|
||||||
|
struct { const char *name, *again; size_t length; int emit, entered; } stack[DEPTH_MAX];
|
||||||
|
size_t oplen, cllen, len, l;
|
||||||
|
int depth, rc, emit;
|
||||||
|
|
||||||
|
emit = 1;
|
||||||
|
oplen = strlen(opstr);
|
||||||
|
cllen = strlen(clstr);
|
||||||
|
depth = 0;
|
||||||
|
for(;;) {
|
||||||
|
beg = strstr(template, opstr);
|
||||||
|
if (beg == NULL) {
|
||||||
|
/* no more mustach */
|
||||||
|
if (emit)
|
||||||
|
fwrite(template, strlen(template), 1, file);
|
||||||
|
return depth ? MUSTACH_ERROR_UNEXPECTED_END : 0;
|
||||||
|
}
|
||||||
|
if (emit)
|
||||||
|
fwrite(template, (size_t)(beg - template), 1, file);
|
||||||
|
beg += oplen;
|
||||||
|
term = strstr(beg, clstr);
|
||||||
|
if (term == NULL)
|
||||||
|
return MUSTACH_ERROR_UNEXPECTED_END;
|
||||||
|
template = term + cllen;
|
||||||
|
len = (size_t)(term - beg);
|
||||||
|
c = *beg;
|
||||||
|
switch(c) {
|
||||||
|
case '!':
|
||||||
|
case '=':
|
||||||
|
break;
|
||||||
|
case '{':
|
||||||
|
for (l = 0 ; clstr[l] == '}' ; l++);
|
||||||
|
if (clstr[l]) {
|
||||||
|
if (!len || beg[len-1] != '}')
|
||||||
|
return MUSTACH_ERROR_BAD_UNESCAPE_TAG;
|
||||||
|
len--;
|
||||||
|
} else {
|
||||||
|
if (term[l] != '}')
|
||||||
|
return MUSTACH_ERROR_BAD_UNESCAPE_TAG;
|
||||||
|
template++;
|
||||||
|
}
|
||||||
|
c = '&';
|
||||||
|
/*@fallthrough@*/
|
||||||
|
case '^':
|
||||||
|
case '#':
|
||||||
|
case '/':
|
||||||
|
case '&':
|
||||||
|
case '>':
|
||||||
|
#if !defined(NO_EXTENSION_FOR_MUSTACH) && !defined(NO_COLON_EXTENSION_FOR_MUSTACH)
|
||||||
|
case ':':
|
||||||
|
#endif
|
||||||
|
beg++; len--;
|
||||||
|
default:
|
||||||
|
while (len && isspace(beg[0])) { beg++; len--; }
|
||||||
|
while (len && isspace(beg[len-1])) len--;
|
||||||
|
#if defined(NO_EXTENSION_FOR_MUSTACH) || defined(NO_ALLOW_EMPTY_TAG)
|
||||||
|
if (len == 0)
|
||||||
|
return MUSTACH_ERROR_EMPTY_TAG;
|
||||||
|
#endif
|
||||||
|
if (len > NAME_LENGTH_MAX)
|
||||||
|
return MUSTACH_ERROR_TAG_TOO_LONG;
|
||||||
|
memcpy(name, beg, len);
|
||||||
|
name[len] = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
switch(c) {
|
||||||
|
case '!':
|
||||||
|
/* comment */
|
||||||
|
/* nothing to do */
|
||||||
|
break;
|
||||||
|
case '=':
|
||||||
|
/* defines separators */
|
||||||
|
if (len < 5 || beg[len - 1] != '=')
|
||||||
|
return MUSTACH_ERROR_BAD_SEPARATORS;
|
||||||
|
beg++;
|
||||||
|
len -= 2;
|
||||||
|
for (l = 0; l < len && !isspace(beg[l]) ; l++);
|
||||||
|
if (l == len)
|
||||||
|
return MUSTACH_ERROR_BAD_SEPARATORS;
|
||||||
|
oplen = l;
|
||||||
|
tmp = alloca(oplen + 1);
|
||||||
|
memcpy(tmp, beg, oplen);
|
||||||
|
tmp[oplen] = 0;
|
||||||
|
opstr = tmp;
|
||||||
|
while (l < len && isspace(beg[l])) l++;
|
||||||
|
if (l == len)
|
||||||
|
return MUSTACH_ERROR_BAD_SEPARATORS;
|
||||||
|
cllen = len - l;
|
||||||
|
tmp = alloca(cllen + 1);
|
||||||
|
memcpy(tmp, beg + l, cllen);
|
||||||
|
tmp[cllen] = 0;
|
||||||
|
clstr = tmp;
|
||||||
|
break;
|
||||||
|
case '^':
|
||||||
|
case '#':
|
||||||
|
/* begin section */
|
||||||
|
if (depth == DEPTH_MAX)
|
||||||
|
return MUSTACH_ERROR_TOO_DEEP;
|
||||||
|
rc = emit;
|
||||||
|
if (rc) {
|
||||||
|
rc = itf->enter(closure, name);
|
||||||
|
if (rc < 0)
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
stack[depth].name = beg;
|
||||||
|
stack[depth].again = template;
|
||||||
|
stack[depth].length = len;
|
||||||
|
stack[depth].emit = emit;
|
||||||
|
stack[depth].entered = rc;
|
||||||
|
if ((c == '#') == (rc == 0))
|
||||||
|
emit = 0;
|
||||||
|
depth++;
|
||||||
|
break;
|
||||||
|
case '/':
|
||||||
|
/* end section */
|
||||||
|
if (depth-- == 0 || len != stack[depth].length || memcmp(stack[depth].name, name, len))
|
||||||
|
return MUSTACH_ERROR_CLOSING;
|
||||||
|
rc = emit && stack[depth].entered ? itf->next(closure) : 0;
|
||||||
|
if (rc < 0)
|
||||||
|
return rc;
|
||||||
|
if (rc) {
|
||||||
|
template = stack[depth++].again;
|
||||||
|
} else {
|
||||||
|
emit = stack[depth].emit;
|
||||||
|
if (emit && stack[depth].entered)
|
||||||
|
itf->leave(closure);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case '>':
|
||||||
|
/* partials */
|
||||||
|
if (emit) {
|
||||||
|
rc = getpartial(itf, closure, name, &partial);
|
||||||
|
if (rc == 0) {
|
||||||
|
rc = process(partial, itf, closure, file, opstr, clstr);
|
||||||
|
free(partial);
|
||||||
|
}
|
||||||
|
if (rc < 0)
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* replacement */
|
||||||
|
if (emit) {
|
||||||
|
rc = itf->put(closure, name, c != '&', file);
|
||||||
|
if (rc < 0)
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int fmustach(const char *template, struct mustach_itf *itf, void *closure, FILE *file)
|
||||||
|
{
|
||||||
|
int rc = itf->start ? itf->start(closure) : 0;
|
||||||
|
if (rc == 0)
|
||||||
|
rc = process(template, itf, closure, file, "{{", "}}");
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fdmustach(const char *template, struct mustach_itf *itf, void *closure, int fd)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
FILE *file;
|
||||||
|
|
||||||
|
file = fdopen(fd, "w");
|
||||||
|
if (file == NULL) {
|
||||||
|
rc = MUSTACH_ERROR_SYSTEM;
|
||||||
|
errno = ENOMEM;
|
||||||
|
} else {
|
||||||
|
rc = fmustach(template, itf, closure, file);
|
||||||
|
fclose(file);
|
||||||
|
}
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
int mustach(const char *template, struct mustach_itf *itf, void *closure, char **result, size_t *size)
|
||||||
|
{
|
||||||
|
int rc;
|
||||||
|
FILE *file;
|
||||||
|
size_t s;
|
||||||
|
|
||||||
|
*result = NULL;
|
||||||
|
if (size == NULL)
|
||||||
|
size = &s;
|
||||||
|
file = memfile_open(result, size);
|
||||||
|
if (file == NULL)
|
||||||
|
rc = MUSTACH_ERROR_SYSTEM;
|
||||||
|
else {
|
||||||
|
rc = fmustach(template, itf, closure, file);
|
||||||
|
if (rc < 0)
|
||||||
|
memfile_abort(file, result, size);
|
||||||
|
else
|
||||||
|
rc = memfile_close(file, result, size);
|
||||||
|
}
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
120
src/mustach.h
Normal file
120
src/mustach.h
Normal file
|
|
@ -0,0 +1,120 @@
|
||||||
|
/*
|
||||||
|
Author: José Bollo <jobol@nonadev.net>
|
||||||
|
Author: José Bollo <jose.bollo@iot.bzh>
|
||||||
|
|
||||||
|
https://gitlab.com/jobol/mustach
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _mustach_h_included_
|
||||||
|
#define _mustach_h_included_
|
||||||
|
|
||||||
|
/**
|
||||||
|
* mustach_itf - interface for callbacks
|
||||||
|
*
|
||||||
|
* All of this function should return a negative value to stop
|
||||||
|
* the mustache processing. The returned negative value will be
|
||||||
|
* then returned to the caller of mustach as is.
|
||||||
|
*
|
||||||
|
* The functions enter and next should return 0 or 1.
|
||||||
|
*
|
||||||
|
* All other functions should normally return 0. If it returns
|
||||||
|
* a negative value, it means an error that stop the process
|
||||||
|
* and that is reported to the caller.
|
||||||
|
*
|
||||||
|
* @start: Starts the mustach processing of the closure
|
||||||
|
* 'start' is optional (can be NULL)
|
||||||
|
*
|
||||||
|
* @put: Writes the value of 'name' to 'file' with 'escape' or not
|
||||||
|
* As an extension (see NO_ALLOW_EMPTY_TAG), the 'name' can be
|
||||||
|
* the empty string. In that later case an implemntation can
|
||||||
|
* return MUSTACH_ERROR_EMPTY_TAG to refuse empty names.
|
||||||
|
*
|
||||||
|
* @enter: Enters the section of 'name' if possible.
|
||||||
|
* Musts return 1 if entered or 0 if not entered.
|
||||||
|
* When 1 is returned, the function 'leave' will always be called.
|
||||||
|
* Conversely 'leave' is never called when enter returns 0 or
|
||||||
|
* a negative value.
|
||||||
|
* When 1 is returned, the function must activate the first
|
||||||
|
* item of the section.
|
||||||
|
*
|
||||||
|
* @next: Activates the next item of the section if it exists.
|
||||||
|
* Musts return 1 when the next item is activated.
|
||||||
|
* Musts return 0 when there is no item to activate.
|
||||||
|
*
|
||||||
|
* @leave: Leaves the last entered section
|
||||||
|
*/
|
||||||
|
struct mustach_itf {
|
||||||
|
int (*start)(void *closure);
|
||||||
|
int (*put)(void *closure, const char *name, int escape, FILE *file);
|
||||||
|
int (*enter)(void *closure, const char *name);
|
||||||
|
int (*next)(void *closure);
|
||||||
|
int (*leave)(void *closure);
|
||||||
|
};
|
||||||
|
|
||||||
|
#define MUSTACH_OK 0
|
||||||
|
#define MUSTACH_ERROR_SYSTEM -1
|
||||||
|
#define MUSTACH_ERROR_UNEXPECTED_END -2
|
||||||
|
#define MUSTACH_ERROR_EMPTY_TAG -3
|
||||||
|
#define MUSTACH_ERROR_TAG_TOO_LONG -4
|
||||||
|
#define MUSTACH_ERROR_BAD_SEPARATORS -5
|
||||||
|
#define MUSTACH_ERROR_TOO_DEEP -6
|
||||||
|
#define MUSTACH_ERROR_CLOSING -7
|
||||||
|
#define MUSTACH_ERROR_BAD_UNESCAPE_TAG -8
|
||||||
|
|
||||||
|
/* compatibility with older bad name */
|
||||||
|
#define MUSTACH_ERROR_TOO_DEPTH MUSTACH_ERROR_TOO_DEEP
|
||||||
|
|
||||||
|
/**
|
||||||
|
* fmustach - Renders the mustache 'template' in 'file' for 'itf' and 'closure'.
|
||||||
|
*
|
||||||
|
* @template: the template string to instanciate
|
||||||
|
* @itf: the interface to the functions that mustach calls
|
||||||
|
* @closure: the closure to pass to functions called
|
||||||
|
* @file: the file where to write the result
|
||||||
|
*
|
||||||
|
* Returns 0 in case of success, -1 with errno set in case of system error
|
||||||
|
* a other negative value in case of error.
|
||||||
|
*/
|
||||||
|
extern int fmustach(const char *template, struct mustach_itf *itf, void *closure, FILE *file);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* fmustach - Renders the mustache 'template' in 'fd' for 'itf' and 'closure'.
|
||||||
|
*
|
||||||
|
* @template: the template string to instanciate
|
||||||
|
* @itf: the interface to the functions that mustach calls
|
||||||
|
* @closure: the closure to pass to functions called
|
||||||
|
* @fd: the file descriptor number where to write the result
|
||||||
|
*
|
||||||
|
* Returns 0 in case of success, -1 with errno set in case of system error
|
||||||
|
* a other negative value in case of error.
|
||||||
|
*/
|
||||||
|
extern int fdmustach(const char *template, struct mustach_itf *itf, void *closure, int fd);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* fmustach - Renders the mustache 'template' in 'result' for 'itf' and 'closure'.
|
||||||
|
*
|
||||||
|
* @template: the template string to instanciate
|
||||||
|
* @itf: the interface to the functions that mustach calls
|
||||||
|
* @closure: the closure to pass to functions called
|
||||||
|
* @result: the pointer receiving the result when 0 is returned
|
||||||
|
* @size: the size of the returned result
|
||||||
|
*
|
||||||
|
* Returns 0 in case of success, -1 with errno set in case of system error
|
||||||
|
* a other negative value in case of error.
|
||||||
|
*/
|
||||||
|
extern int mustach(const char *template, struct mustach_itf *itf, void *closure, char **result, size_t *size);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
662
src/other.c
662
src/other.c
|
|
@ -33,46 +33,482 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*! \file other.c
|
||||||
|
* \brief Collection of various functions.
|
||||||
|
*/
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
#include <arpa/inet.h>
|
||||||
|
|
||||||
#include <err.h>
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <stdbool.h>
|
#include <limits.h>
|
||||||
|
#include <netinet/in.h>
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <time.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "error.h"
|
||||||
|
#include "progname.h"
|
||||||
|
#include "quote.h"
|
||||||
|
#include "xalloc.h"
|
||||||
|
|
||||||
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
|
static char *(*cidr_last) (union ipaddr_t *restrict addr, const int mask);
|
||||||
|
static char *cidr_last_v4(union ipaddr_t *restrict addr, const int mask);
|
||||||
|
static char *cidr_last_v6(union ipaddr_t *restrict addr, const int mask);
|
||||||
|
|
||||||
|
/*! \brief Set function pointers depending on IP version.
|
||||||
|
* \param ip IP version.
|
||||||
|
*/
|
||||||
|
void set_ipv_functions(struct conf_t *state, int version)
|
||||||
|
{
|
||||||
|
switch (version) {
|
||||||
|
|
||||||
|
case IPv4:
|
||||||
|
state->ip_version = version;
|
||||||
|
add_lease = add_lease_v4;
|
||||||
|
copy_ipaddr = copy_ipaddr_v4;
|
||||||
|
find_lease = find_lease_v4;
|
||||||
|
get_range_size = get_range_size_v4;
|
||||||
|
ipcomp = ipcomp_v4;
|
||||||
|
leasecomp = leasecomp_v4;
|
||||||
|
ntop_ipaddr = ntop_ipaddr_v4;
|
||||||
|
parse_ipaddr = parse_ipaddr_v4;
|
||||||
|
cidr_last = cidr_last_v4;
|
||||||
|
xstrstr = xstrstr_v4;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case IPv6:
|
||||||
|
state->ip_version = version;
|
||||||
|
add_lease = add_lease_v6;
|
||||||
|
copy_ipaddr = copy_ipaddr_v6;
|
||||||
|
find_lease = find_lease_v6;
|
||||||
|
get_range_size = get_range_size_v6;
|
||||||
|
ipcomp = ipcomp_v6;
|
||||||
|
leasecomp = leasecomp_v6;
|
||||||
|
ntop_ipaddr = ntop_ipaddr_v6;
|
||||||
|
parse_ipaddr = parse_ipaddr_v6;
|
||||||
|
cidr_last = cidr_last_v6;
|
||||||
|
xstrstr = xstrstr_v6;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case IPvUNKNOWN:
|
||||||
|
state->ip_version = version;
|
||||||
|
add_lease = add_lease_init;
|
||||||
|
copy_ipaddr = copy_ipaddr_init;
|
||||||
|
find_lease = find_lease_init;
|
||||||
|
get_range_size = get_range_size_init;
|
||||||
|
ipcomp = ipcomp_init;
|
||||||
|
leasecomp = leasecomp_init;
|
||||||
|
ntop_ipaddr = ntop_ipaddr_init;
|
||||||
|
parse_ipaddr = parse_ipaddr_init;
|
||||||
|
cidr_last = NULL;
|
||||||
|
xstrstr = xstrstr_init;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
abort();
|
||||||
|
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Convert text string IP address from either IPv4 or IPv6 to an integer.
|
||||||
|
* \param src An IP string in either format.
|
||||||
|
* \param dst An union which will hold conversion result.
|
||||||
|
* \return Was parsing successful.
|
||||||
|
*/
|
||||||
|
int parse_ipaddr_init(struct conf_t *state, const char *restrict src, union ipaddr_t *restrict dst)
|
||||||
|
{
|
||||||
|
struct in_addr addr;
|
||||||
|
struct in6_addr addr6;
|
||||||
|
|
||||||
|
if (inet_aton(src, &addr) == 1)
|
||||||
|
set_ipv_functions(state, IPv4);
|
||||||
|
else if (inet_pton(AF_INET6, src, &addr6) == 1)
|
||||||
|
set_ipv_functions(state, IPv6);
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
return parse_ipaddr(state, src, dst);
|
||||||
|
}
|
||||||
|
|
||||||
|
int parse_ipaddr_v4(struct conf_t *state
|
||||||
|
__attribute__ ((unused)), const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst)
|
||||||
|
{
|
||||||
|
int rv;
|
||||||
|
struct in_addr addr;
|
||||||
|
|
||||||
|
rv = inet_aton(src, &addr);
|
||||||
|
dst->v4 = ntohl(addr.s_addr);
|
||||||
|
return rv == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int parse_ipaddr_v6(struct conf_t *state
|
||||||
|
__attribute__ ((unused)), const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst)
|
||||||
|
{
|
||||||
|
int rv;
|
||||||
|
struct in6_addr addr;
|
||||||
|
|
||||||
|
rv = inet_pton(AF_INET6, src, &addr);
|
||||||
|
memcpy(&dst->v6, addr.s6_addr, sizeof(addr.s6_addr));
|
||||||
|
return rv == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Convert string to a desimal format network marks.
|
||||||
|
* \param src Digit that should be a network mask.
|
||||||
|
* \return Network mask, or -1 when failing.
|
||||||
|
*/
|
||||||
|
static int strtol_mask(const char *str)
|
||||||
|
{
|
||||||
|
long num;
|
||||||
|
char *end = NULL;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
if (str == NULL || *str == '\0')
|
||||||
|
goto err;
|
||||||
|
num = strtol(str, &end, 10);
|
||||||
|
|
||||||
|
if (errno || str == end || (end && *end))
|
||||||
|
goto err;
|
||||||
|
if (num < 0 || 128 < num)
|
||||||
|
goto err;
|
||||||
|
return (int)num;
|
||||||
|
err:
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Find last address in IPv4 range by using cidr format.
|
||||||
|
* \param addr Pointer to memory where address needs to be stored.
|
||||||
|
* \return Allocated string format of the address.
|
||||||
|
*/
|
||||||
|
static char *cidr_last_v4(union ipaddr_t *restrict addr, const int mask)
|
||||||
|
{
|
||||||
|
union ipaddr_t last_ip;
|
||||||
|
uint32_t netmask;
|
||||||
|
const char *ip;
|
||||||
|
|
||||||
|
if (mask)
|
||||||
|
netmask = (1U << (32 - mask)) - 1;
|
||||||
|
else
|
||||||
|
netmask = 0;
|
||||||
|
last_ip.v4 = addr->v4 | netmask;
|
||||||
|
|
||||||
|
ip = ntop_ipaddr(&last_ip);
|
||||||
|
return xstrdup(ip);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Find last address in IPv6 range by using cidr format.
|
||||||
|
* \param addr Pointer to memory where address needs to be stored.
|
||||||
|
* \return Allocated string format of the address.
|
||||||
|
*/
|
||||||
|
static char *cidr_last_v6(union ipaddr_t *restrict addr, const int mask)
|
||||||
|
{
|
||||||
|
union ipaddr_t bitmask;
|
||||||
|
int i, j;
|
||||||
|
char ip[128];
|
||||||
|
|
||||||
|
memset(&bitmask, 0x0, sizeof(bitmask));
|
||||||
|
for (i = mask, j = 0; i > 0; i -= 8, j++) {
|
||||||
|
if (i >= 8)
|
||||||
|
bitmask.v6[j] = 0xff;
|
||||||
|
else
|
||||||
|
bitmask.v6[j] = (unsigned char)(0xffU << (8 - i));
|
||||||
|
}
|
||||||
|
for (i = 0; i < (int)sizeof(bitmask); i++)
|
||||||
|
addr->v6[i] |= ~bitmask.v6[i];
|
||||||
|
inet_ntop(AF_INET6, addr, ip, sizeof(ip));
|
||||||
|
return xstrdup(ip);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Convert a cidr notated address to a range.
|
||||||
|
* \param range_p Pointer to memory where addresses need to be stored.
|
||||||
|
* \param word A range as a cidr string.
|
||||||
|
*/
|
||||||
|
void parse_cidr(struct conf_t *state, struct range_t *range_p, const char *word)
|
||||||
|
{
|
||||||
|
char *divider;
|
||||||
|
int mask;
|
||||||
|
union ipaddr_t addr;
|
||||||
|
char *last;
|
||||||
|
|
||||||
|
/* determine cidr */
|
||||||
|
divider = strchr(word, '/');
|
||||||
|
*divider++ = '\0';
|
||||||
|
mask = strtol_mask(divider);
|
||||||
|
if (mask < 0)
|
||||||
|
error(EXIT_FAILURE, 0, "cidr %s invalid mask %s", word, divider);
|
||||||
|
if (state->ip_version == IPvUNKNOWN) {
|
||||||
|
if (!strchr(word, ':'))
|
||||||
|
set_ipv_functions(state, IPv4);
|
||||||
|
else
|
||||||
|
set_ipv_functions(state, IPv6);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* start of the range is easy */
|
||||||
|
parse_ipaddr(state, word, &addr);
|
||||||
|
copy_ipaddr(&range_p->first_ip, &addr);
|
||||||
|
|
||||||
|
/* end of the range depends cidr size */
|
||||||
|
last = cidr_last(&addr, mask);
|
||||||
|
parse_ipaddr(state, last, &addr);
|
||||||
|
copy_ipaddr(&range_p->last_ip, &addr);
|
||||||
|
free(last);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Copy IP address to union.
|
||||||
|
*
|
||||||
|
* \param dst Destination for a binary IP address.
|
||||||
|
* \param src Sourse of an IP address. */
|
||||||
|
void copy_ipaddr_init(union ipaddr_t *restrict dst __attribute__ ((unused)),
|
||||||
|
const union ipaddr_t *restrict src __attribute__ ((unused)))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void copy_ipaddr_v4(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src)
|
||||||
|
{
|
||||||
|
dst->v4 = src->v4;
|
||||||
|
}
|
||||||
|
|
||||||
|
void copy_ipaddr_v6(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src)
|
||||||
|
{
|
||||||
|
memcpy(&dst->v6, &src->v6, sizeof(src->v6));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Convert an address to string. This function will convert the
|
||||||
|
* IPv4 addresses to 123.45.65.78 format, and the IPv6 addresses to it's
|
||||||
|
* native format depending on which version of the addressing is found to
|
||||||
|
* be in use.
|
||||||
|
*
|
||||||
|
* \param ip Binary IP address.
|
||||||
|
* \return Printable address.
|
||||||
|
*/
|
||||||
|
const char *ntop_ipaddr_init(const union ipaddr_t *ip __attribute__ ((unused)))
|
||||||
|
{
|
||||||
|
static char buffer = '\0';
|
||||||
|
|
||||||
|
return &buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *ntop_ipaddr_v4(const union ipaddr_t *ip)
|
||||||
|
{
|
||||||
|
static char buffer[INET_ADDRSTRLEN];
|
||||||
|
struct in_addr addr;
|
||||||
|
|
||||||
|
addr.s_addr = htonl(ip->v4);
|
||||||
|
return inet_ntop(AF_INET, &addr, buffer, sizeof(buffer));
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *ntop_ipaddr_v6(const union ipaddr_t *ip)
|
||||||
|
{
|
||||||
|
static char buffer[INET6_ADDRSTRLEN];
|
||||||
|
struct in6_addr addr;
|
||||||
|
|
||||||
|
memcpy(addr.s6_addr, ip->v6, sizeof(addr.s6_addr));
|
||||||
|
return inet_ntop(AF_INET6, &addr, buffer, sizeof(buffer));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Calculate how many addresses there are in a range.
|
||||||
|
*
|
||||||
|
* \param r Pointer to range structure, which has information about first
|
||||||
|
* and last IP in the range.
|
||||||
|
* \return Size of a range.
|
||||||
|
*/
|
||||||
|
double get_range_size_init(const struct range_t *r __attribute__ ((unused)))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
double get_range_size_v4(const struct range_t *r)
|
||||||
|
{
|
||||||
|
return r->last_ip.v4 - r->first_ip.v4 + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
double get_range_size_v6(const struct range_t *r)
|
||||||
|
{
|
||||||
|
double size = 0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* When calculating the size of an IPv6 range overflow may occur.
|
||||||
|
* In that case only the last LONG_BIT bits are preserved, thus
|
||||||
|
* we just skip the first (16 - LONG_BIT) bits... */
|
||||||
|
for (i = 0; i < 16; i++) {
|
||||||
|
size *= 256;
|
||||||
|
size += (int)r->last_ip.v6[i] - (int)r->first_ip.v6[i];
|
||||||
|
}
|
||||||
|
return size + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \fn xstrstr_init(const char *restrict str)
|
||||||
|
* \brief Determine if the dhcpd is in IPv4 or IPv6 mode. This function
|
||||||
|
* may be needed when dhcpd.conf file has zero IP version hints.
|
||||||
|
*
|
||||||
|
* \param str A line from dhcpd.conf
|
||||||
|
* \return prefix_t enum value
|
||||||
|
*/
|
||||||
|
int xstrstr_init(struct conf_t *state, const char *restrict str)
|
||||||
|
{
|
||||||
|
if (!memcmp("lease ", str, 6)) {
|
||||||
|
set_ipv_functions(state, IPv4);
|
||||||
|
return PREFIX_LEASE;
|
||||||
|
}
|
||||||
|
if (!memcmp(" iaaddr ", str, 9)) {
|
||||||
|
set_ipv_functions(state, IPv6);
|
||||||
|
return PREFIX_LEASE;
|
||||||
|
}
|
||||||
|
return NUM_OF_PREFIX;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \fn xstrstr_v4(const char *restrict str)
|
||||||
|
* \brief parse lease file in IPv4 mode
|
||||||
|
*
|
||||||
|
* \param str A line from dhcpd.conf
|
||||||
|
* \return prefix_t enum value
|
||||||
|
*/
|
||||||
int
|
int
|
||||||
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
||||||
__attribute__ ((hot))
|
__attribute__ ((hot))
|
||||||
#endif
|
#endif
|
||||||
xstrstr(const char *restrict a, const char *restrict b, const int len)
|
xstrstr_v4(struct conf_t *state __attribute__ ((unused)), const char *restrict str)
|
||||||
{
|
{
|
||||||
int i;
|
size_t len;
|
||||||
/* two spaces are very common in lease file, after them
|
|
||||||
* nearly everything differs */
|
if (str[2] == 'b' || str[2] == 'h')
|
||||||
if (likely(a[2] != b[2])) {
|
len = strlen(str);
|
||||||
return false;
|
else
|
||||||
}
|
len = 0;
|
||||||
/* " binding state " == 16 chars, this will skip right
|
if (15 < len) {
|
||||||
* to first differing line. */
|
switch (str[16]) {
|
||||||
if (17 < len && a[17] != b[17]) {
|
case 'f':
|
||||||
return false;
|
if (!memcmp(" binding state free;", str, 21))
|
||||||
}
|
return PREFIX_BINDING_STATE_FREE;
|
||||||
/* looking good, double check the whole thing... */
|
break;
|
||||||
for (i = 0; a[i] != '\0' && b[i] != '\0'; i++) {
|
case 'a':
|
||||||
if (a[i] != b[i]) {
|
if (!memcmp(" binding state active;", str, 23))
|
||||||
return false;
|
return PREFIX_BINDING_STATE_ACTIVE;
|
||||||
|
if (!memcmp(" binding state abandoned;", str, 25))
|
||||||
|
return PREFIX_BINDING_STATE_ABANDONED;
|
||||||
|
break;
|
||||||
|
case 'e':
|
||||||
|
if (!memcmp(" binding state expired;", str, 24))
|
||||||
|
return PREFIX_BINDING_STATE_EXPIRED;
|
||||||
|
break;
|
||||||
|
case 'r':
|
||||||
|
if (!memcmp(" binding state released;", str, 25))
|
||||||
|
return PREFIX_BINDING_STATE_RELEASED;
|
||||||
|
break;
|
||||||
|
case 'b':
|
||||||
|
if (!memcmp(" binding state backup;", str, 23))
|
||||||
|
return PREFIX_BINDING_STATE_BACKUP;
|
||||||
|
break;
|
||||||
|
case 'n':
|
||||||
|
if (!memcmp(" hardware ethernet", str, 19))
|
||||||
|
return PREFIX_HARDWARE_ETHERNET;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true;
|
if (!memcmp("lease ", str, 6))
|
||||||
|
return PREFIX_LEASE;
|
||||||
|
else if (!memcmp(" starts ", str, 9))
|
||||||
|
return PREFIX_STARTS;
|
||||||
|
else if (!memcmp(" ends ", str, 7))
|
||||||
|
return PREFIX_ENDS;
|
||||||
|
else if (!memcmp(" client-hostname ", str, 18))
|
||||||
|
return PREFIX_HOSTNAME;
|
||||||
|
return NUM_OF_PREFIX;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return percentage value */
|
/*! \fn xstrstr_v4(const char *restrict str)
|
||||||
|
* \brief parse lease file in IPv6 mode
|
||||||
|
*
|
||||||
|
* \param str A line from dhcpd.conf
|
||||||
|
* \return prefix_t enum value
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
||||||
|
__attribute__ ((hot))
|
||||||
|
#endif
|
||||||
|
xstrstr_v6(struct conf_t *state __attribute__ ((unused)), const char *restrict str)
|
||||||
|
{
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
if (str[4] == 'b' || str[2] == 'h')
|
||||||
|
len = strlen(str);
|
||||||
|
else
|
||||||
|
len = 0;
|
||||||
|
if (17 < len) {
|
||||||
|
switch (str[18]) {
|
||||||
|
case 'f':
|
||||||
|
if (!memcmp(" binding state free;", str, 23))
|
||||||
|
return PREFIX_BINDING_STATE_FREE;
|
||||||
|
break;
|
||||||
|
case 'a':
|
||||||
|
if (!memcmp(" binding state active;", str, 25))
|
||||||
|
return PREFIX_BINDING_STATE_ACTIVE;
|
||||||
|
if (!memcmp(" binding state abandoned;", str, 27))
|
||||||
|
return PREFIX_BINDING_STATE_ABANDONED;
|
||||||
|
break;
|
||||||
|
case 'e':
|
||||||
|
if (!memcmp(" binding state expired;", str, 26))
|
||||||
|
return PREFIX_BINDING_STATE_EXPIRED;
|
||||||
|
break;
|
||||||
|
case 'r':
|
||||||
|
if (!memcmp(" binding state released;", str, 27))
|
||||||
|
return PREFIX_BINDING_STATE_RELEASED;
|
||||||
|
break;
|
||||||
|
case 'b':
|
||||||
|
if (!memcmp(" binding state backup;", str, 25))
|
||||||
|
return PREFIX_BINDING_STATE_BACKUP;
|
||||||
|
break;
|
||||||
|
case 'n':
|
||||||
|
if (!memcmp(" hardware ethernet", str, 19))
|
||||||
|
return PREFIX_HARDWARE_ETHERNET;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!memcmp(" iaaddr ", str, 9))
|
||||||
|
return PREFIX_LEASE;
|
||||||
|
else if (!memcmp(" starts ", str, 9))
|
||||||
|
return PREFIX_STARTS;
|
||||||
|
else if (!memcmp(" ends ", str, 7))
|
||||||
|
return PREFIX_ENDS;
|
||||||
|
else if (!memcmp(" client-hostname ", str, 18))
|
||||||
|
return PREFIX_HOSTNAME;
|
||||||
|
return NUM_OF_PREFIX;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Parse option argument color mode.
|
||||||
|
*
|
||||||
|
* \param Color mode string.
|
||||||
|
* \return color mode enum.
|
||||||
|
*/
|
||||||
|
int parse_color_mode(const char *restrict arg)
|
||||||
|
{
|
||||||
|
if (!strcmp(arg, "always"))
|
||||||
|
return color_on;
|
||||||
|
if (!strcmp(arg, "auto"))
|
||||||
|
return color_auto;
|
||||||
|
if (!strcmp(arg, "never"))
|
||||||
|
return color_off;
|
||||||
|
return color_unknown;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Return a double floating point value.
|
||||||
|
*
|
||||||
|
* \param str String to be converted to a double.
|
||||||
|
* \param errmesg Exit error message if conversion fails.
|
||||||
|
* \return Binary result of string to double conversion.
|
||||||
|
*/
|
||||||
double strtod_or_err(const char *restrict str, const char *restrict errmesg)
|
double strtod_or_err(const char *restrict str, const char *restrict errmesg)
|
||||||
{
|
{
|
||||||
double num;
|
double num;
|
||||||
|
|
@ -82,102 +518,142 @@ double strtod_or_err(const char *restrict str, const char *restrict errmesg)
|
||||||
goto err;
|
goto err;
|
||||||
errno = 0;
|
errno = 0;
|
||||||
num = strtod(str, &end);
|
num = strtod(str, &end);
|
||||||
|
|
||||||
if (errno || str == end || (end && *end))
|
if (errno || str == end || (end && *end))
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
return num;
|
return num;
|
||||||
err:
|
err:
|
||||||
if (errno)
|
error(EXIT_FAILURE, errno, "%s: %s", errmesg, quote(str));
|
||||||
err(EXIT_FAILURE, "%s: '%s'", errmesg, str);
|
return 0;
|
||||||
|
|
||||||
errx(EXIT_FAILURE, "%s: '%s'", errmesg, str);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void flip_ranges(struct range_t *restrict flip_me,
|
/*! \brief Reverse range.
|
||||||
struct range_t *restrict tmp_ranges)
|
* Used before output, if a caller has requested reverse sorting. */
|
||||||
|
void flip_ranges(struct conf_t *state)
|
||||||
{
|
{
|
||||||
unsigned int i = num_ranges - 1, j;
|
unsigned int i = state->num_ranges - 1, j;
|
||||||
|
struct range_t *tmp_ranges;
|
||||||
|
|
||||||
for (j = 0; j < num_ranges; j++) {
|
tmp_ranges = xmalloc(sizeof(struct range_t) * state->num_ranges);
|
||||||
*(tmp_ranges + j) = *(flip_me + i);
|
for (j = 0; j < state->num_ranges; j++, i--)
|
||||||
i--;
|
*(tmp_ranges + j) = *(state->ranges + i);
|
||||||
}
|
memcpy(state->ranges, tmp_ranges, state->num_ranges * sizeof(struct range_t));
|
||||||
|
free(tmp_ranges);
|
||||||
memcpy(flip_me, tmp_ranges, num_ranges * sizeof(struct range_t));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free memory, flush buffers etc */
|
/*! \brief Free memory, flush buffers etc. */
|
||||||
void clean_up(void)
|
void clean_up(struct conf_t *state)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
struct output_sort *cur, *next;
|
||||||
|
struct shared_network_t *c, *n;
|
||||||
|
|
||||||
/* Just in case there something in buffers */
|
/* Just in case there something in buffers */
|
||||||
if (fflush(NULL)) {
|
if (fflush(NULL))
|
||||||
warn("clean_up: fflush");
|
error(EXIT_FAILURE, errno, "clean_up: fflush");
|
||||||
|
free(state->ranges);
|
||||||
|
delete_all_leases(state);
|
||||||
|
for (cur = state->sorts; cur; cur = next) {
|
||||||
|
next = cur->next;
|
||||||
|
free(cur);
|
||||||
}
|
}
|
||||||
num_shared_networks++;
|
for (c = state->shared_net_root; c; c = n) {
|
||||||
for (i = 0; i < num_shared_networks; i++) {
|
n = c->next;
|
||||||
free((shared_networks + i)->name);
|
free(c->name);
|
||||||
|
free(c);
|
||||||
}
|
}
|
||||||
free(config.dhcpdconf_file);
|
|
||||||
free(config.dhcpdlease_file);
|
|
||||||
free(config.output_file);
|
|
||||||
free(ranges);
|
|
||||||
delete_all_leases();
|
|
||||||
free(shared_networks);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*! \brief Print a time stamp of a path or now to output file. */
|
||||||
|
void dp_time_tool(FILE *file, const char *path, const int epoch)
|
||||||
|
{
|
||||||
|
struct stat st;
|
||||||
|
|
||||||
|
if (path)
|
||||||
|
stat(path, &st);
|
||||||
|
else
|
||||||
|
clock_gettime(CLOCK_REALTIME, &st.st_mtim);
|
||||||
|
if (epoch)
|
||||||
|
fprintf(file, "%ld", st.st_mtim.tv_sec);
|
||||||
|
else {
|
||||||
|
char time_stamp[64];
|
||||||
|
const time_t mtime = st.st_mtim.tv_sec;
|
||||||
|
struct tm tm;
|
||||||
|
|
||||||
|
localtime_r(&mtime, &tm);
|
||||||
|
strftime(time_stamp, sizeof(time_stamp), "%FT%T%z", &tm);
|
||||||
|
fprintf(file, "%s", time_stamp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief A version printing. */
|
||||||
void __attribute__ ((__noreturn__)) print_version(void)
|
void __attribute__ ((__noreturn__)) print_version(void)
|
||||||
{
|
{
|
||||||
|
#define stringify(s) #s
|
||||||
|
#define stringify_value(s) stringify(s)
|
||||||
fprintf(stdout, "%s\n"
|
fprintf(stdout, "%s\n"
|
||||||
"Written by Sami Kerola.\n"
|
"Original design and maintainer Sami Kerola.\n"
|
||||||
"XML support by Dominic Germain, Sogetel inc.\n\n"
|
"uthash %s by Troy D. Hanson.\n"
|
||||||
"The software has FreeBSD License.\n", PACKAGE_STRING);
|
"XML support by Dominic Germain, Sogetel inc.\n"
|
||||||
|
"IPv6 support by Cheer Xiao.\n"
|
||||||
|
"Mustach templating support by Jose Bollo.\n"
|
||||||
|
" The dhcpd-pools is FreeBSD Licensed,\n"
|
||||||
|
" uthash uses BSD license,\n"
|
||||||
|
" gnulib parts are mostly GPL,\n"
|
||||||
|
" and mustache uses Apache License.\n", PACKAGE_STRING,
|
||||||
|
stringify_value(UTHASH_VERSION));
|
||||||
exit(EXIT_SUCCESS);
|
exit(EXIT_SUCCESS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*! \brief Command line help screen. */
|
||||||
void __attribute__ ((__noreturn__)) usage(int status)
|
void __attribute__ ((__noreturn__)) usage(int status)
|
||||||
{
|
{
|
||||||
FILE *out;
|
FILE *out = status == EXIT_SUCCESS ? stdout : stderr;
|
||||||
out = status != 0 ? stderr : stdout;
|
|
||||||
|
|
||||||
fprintf(out, "\
|
fprintf(out, "Usage: %s [OPTIONS]\n", program_name);
|
||||||
Usage: %s [OPTIONS]\n\n\
|
fputs( "\n", out);
|
||||||
This is ISC dhcpd pools usage analyzer.\n\
|
fputs( "This is ISC dhcpd pools usage analyzer.\n", out);
|
||||||
\n", program_invocation_short_name);
|
fputs( "\n", out);
|
||||||
fprintf(out, "\
|
fputs( " -c, --config=FILE path to the dhcpd.conf file\n", out);
|
||||||
-c, --config=FILE path to the dhcpd.conf file\n\
|
fputs( " -l, --leases=FILE path to the dhcpd.leases file\n", out);
|
||||||
-l, --leases=FILE path to the dhcpd.leases file\n\
|
fputs( " -f, --format=[thHcxXjJ] output format\n", out);
|
||||||
-f, --format=[thHcxX] output format\n\
|
fputs( " t for text\n", out);
|
||||||
t for text\n\
|
fputs( " H for full html page\n", out);
|
||||||
h for html table\n\
|
fputs( " x for xml\n", out);
|
||||||
H for full html page\n\
|
fputs( " X for xml with active lease details\n", out);
|
||||||
x for xml\n\
|
fputs( " j for json\n", out);
|
||||||
X for xml with active lease details\n\
|
fputs( " J for json with active lease details\n", out);
|
||||||
c for comma separated values\n");
|
fputs( " c for comma separated values\n", out);
|
||||||
fprintf(out, "\
|
#ifdef BUILD_MUSTACH
|
||||||
-s, --sort=[nimcptTe] sort ranges by\n\
|
fputs( " --mustach=FILE output using mustach template file\n", out);
|
||||||
n name\n\
|
#endif
|
||||||
i IP\n\
|
fputs( " -s, --sort=[nimcptTe] sort ranges by\n", out);
|
||||||
m maximum\n\
|
fputs( " n name\n", out);
|
||||||
c current\n\
|
fputs( " i IP\n", out);
|
||||||
p percent\n\
|
fputs( " m maximum\n", out);
|
||||||
t touched\n\
|
fputs( " c current\n", out);
|
||||||
T t+c\n\
|
fputs( " p percent\n", out);
|
||||||
e t+c perc\n\
|
fputs( " t touched\n", out);
|
||||||
-r, --reverse reverse order sort\n\
|
fputs( " T t+c\n", out);
|
||||||
-o, --output=FILE output into a file\n\
|
fputs( " e t+c perc\n", out);
|
||||||
-L, --limit=NR output limit mask 77 - 00\n");
|
fputs( " -r, --reverse reverse order sort\n", out);
|
||||||
fprintf(out, "\
|
fputs( " -o, --output=FILE output into a file\n", out);
|
||||||
--warning=PERC set warning alarming limit\n\
|
fputs( " -L, --limit=NR output limit mask 77 - 00\n", out);
|
||||||
--critical=PERC set critical alarming limit\n");
|
fputs( " --color=WHEN use colors 'always', 'never', or 'auto'\n", out);
|
||||||
fprintf(out, "\
|
fputs( " --warning=PERC set warning alarming threshold\n", out);
|
||||||
-v, --version version information\n\
|
fputs( " --critical=PERC set critical alarming threshold\n", out);
|
||||||
-h, --help this screen\n\
|
fputs( " --skip=WHAT do not print threshold 'ok', 'warning', 'critical',\n", out);
|
||||||
\n\
|
fputs( " 'minsize', or 'suppressed'\n", out);
|
||||||
Report bugs to <%s>\n\
|
fputs( " --warn-count=NR a number of free leases before warning raised\n", out);
|
||||||
Homepage: %s\n", PACKAGE_BUGREPORT, PACKAGE_URL);
|
fputs( " --crit-count=NR a number of free leases before critical raised\n", out);
|
||||||
|
fputs( " --minsize=size disable alarms for small ranges and shared-nets\n", out);
|
||||||
|
fputs( " --snet-alarms suppress range alarms that are part of a shared-net\n", out);
|
||||||
|
fputs( " -p, --perfdata print additional perfdata in alarming mode\n", out);
|
||||||
|
fputs( " -A, --all-as-shared treat single subnets as shared-network with CIDR as their name\n", out);
|
||||||
|
fputs( " --ip-version=4|6 force analysis to use either IPv4 or IPv6 functions\n", out);
|
||||||
|
fputs( " -v, --version output version information and exit\n", out);
|
||||||
|
fputs( " -h, --help display this help and exit\n", out);
|
||||||
|
fputs( "\n", out);
|
||||||
|
fprintf(out, "Report bugs to <%s>\n", PACKAGE_BUGREPORT);
|
||||||
|
fprintf(out, "Homepage: %s\n", PACKAGE_URL);
|
||||||
|
|
||||||
exit(out == stderr ? EXIT_FAILURE : EXIT_SUCCESS);
|
exit(status);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
1599
src/output.c
1599
src/output.c
File diff suppressed because it is too large
Load diff
316
src/sort.c
316
src/sort.c
|
|
@ -3,6 +3,7 @@
|
||||||
* BSD License" or "FreeBSD License".
|
* BSD License" or "FreeBSD License".
|
||||||
*
|
*
|
||||||
* Copyright 2006- Sami Kerola. All rights reserved.
|
* Copyright 2006- Sami Kerola. All rights reserved.
|
||||||
|
* Copyright 2012 Cheer Xiao.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
* modification, are permitted provided that the following conditions are
|
* modification, are permitted provided that the following conditions are
|
||||||
|
|
@ -33,182 +34,293 @@
|
||||||
* official policies, either expressed or implied, of Sami Kerola.
|
* official policies, either expressed or implied, of Sami Kerola.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*! \file sort.c
|
||||||
|
* \brief Functions to sort output.
|
||||||
|
*/
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include <err.h>
|
#include <math.h>
|
||||||
#include <errno.h>
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "error.h"
|
||||||
|
#include "progname.h"
|
||||||
|
#include "quote.h"
|
||||||
|
#include "xalloc.h"
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
/* Sort functions for range sorting */
|
/*! \brief Compare IP address, with IPv4/v6 determination.
|
||||||
int intcomp(const void *restrict x, const void *restrict y)
|
* \param a Binary IP address.
|
||||||
|
* \param b Binary IP address.
|
||||||
|
* \return If a < b return -1, if a < b return 1, when they are equal return 0.
|
||||||
|
*/
|
||||||
|
int ipcomp_init(const union ipaddr_t *restrict a __attribute__ ((unused)),
|
||||||
|
const union ipaddr_t *restrict b __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
if (*(uint32_t *) x < *(uint32_t *) y)
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ipcomp_v4(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b)
|
||||||
|
{
|
||||||
|
if (a->v4 < b->v4)
|
||||||
return -1;
|
return -1;
|
||||||
if (*(uint32_t *) y < *(uint32_t *) x)
|
if (a->v4 > b->v4)
|
||||||
return 1;
|
return 1;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int ipcomp_v6(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b)
|
||||||
|
{
|
||||||
|
return memcmp(&a->v6, &b->v6, sizeof(a->v6));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Compare IP address in leases_t structure, with IPv4/v6 determination.
|
||||||
|
* \param a Binary IP address.
|
||||||
|
* \param b Binary IP address.
|
||||||
|
* \return If a < b return -1, if a < b return 1, when they are equal return 0.
|
||||||
|
*/
|
||||||
|
int leasecomp_init(const struct leases_t *restrict a __attribute__ ((unused)),
|
||||||
|
const struct leases_t *restrict b __attribute__ ((unused)))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int leasecomp_v4(const struct leases_t *restrict a, const struct leases_t *restrict b)
|
||||||
|
{
|
||||||
|
if (a->ip.v4 < b->ip.v4)
|
||||||
|
return -1;
|
||||||
|
if (a->ip.v4 > b->ip.v4)
|
||||||
|
return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int leasecomp_v6(const struct leases_t *restrict a, const struct leases_t *restrict b)
|
||||||
|
{
|
||||||
|
return memcmp(&a->ip.v6, &b->ip.v6, sizeof(a->ip.v6));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Compare IP address in leases. Suitable for sorting range table.
|
||||||
|
* \param r1 A range structure.
|
||||||
|
* \param r2 A range structure.
|
||||||
|
* \return Return pas through from ipcomp.
|
||||||
|
*/
|
||||||
int rangecomp(const void *restrict r1, const void *restrict r2)
|
int rangecomp(const void *restrict r1, const void *restrict r2)
|
||||||
{
|
{
|
||||||
if ((((struct range_t *)r1)->first_ip) <
|
return ipcomp(&((const struct range_t *)r1)->first_ip,
|
||||||
(((struct range_t *)r2)->first_ip))
|
&((const struct range_t *)r2)->first_ip);
|
||||||
return -1;
|
|
||||||
if ((((struct range_t *)r2)->first_ip) <
|
|
||||||
(((struct range_t *)r1)->first_ip))
|
|
||||||
return 1;
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_ip(struct range_t r)
|
/*! \brief Compare two doubles.
|
||||||
|
* \param f1,f2 Data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_double(double f1, double f2)
|
||||||
{
|
{
|
||||||
return (r.first_ip);
|
if (isless(f1, f2))
|
||||||
|
return -1;
|
||||||
|
else if (isless(f2, f1))
|
||||||
|
return 1;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_cur(struct range_t r)
|
|
||||||
|
/*! \brief Compare two range_t by their first_ip.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_ip(struct range_t *r1, struct range_t *r2)
|
||||||
{
|
{
|
||||||
return (r.count);
|
return ipcomp(&r1->first_ip, &r2->first_ip);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_max(struct range_t r)
|
/*! \brief Compare two range_t by their capacity.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_max(struct range_t *r1, struct range_t *r2)
|
||||||
{
|
{
|
||||||
return (r.last_ip - r.first_ip);
|
return comp_double(get_range_size(r1), get_range_size(r2));
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_percent(struct range_t r)
|
/*! \brief Compare two range_t by their current usage.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_cur(struct range_t *r1, struct range_t *r2)
|
||||||
{
|
{
|
||||||
float f;
|
return comp_double(r1->count, r2->count);
|
||||||
f = (float)r.count / (r.last_ip - r.first_ip - 1);
|
|
||||||
return ((unsigned long int)(f * 100000));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_touched(struct range_t r)
|
/*! \brief Compare two range_t by their current usage percentage.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_percent(struct range_t *r1, struct range_t *r2)
|
||||||
{
|
{
|
||||||
return (r.touched);
|
return comp_double(ret_percent(*r1), ret_percent(*r2));
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_tc(struct range_t r)
|
/*! \brief Compare two range_t by their touched addresses.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_touched(struct range_t *r1, struct range_t *r2)
|
||||||
|
{
|
||||||
|
return comp_double(r1->touched, r2->touched);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Compare two range_t by their touched and in use addresses.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_tc(struct range_t *r1, struct range_t *r2)
|
||||||
|
{
|
||||||
|
return comp_double(ret_tc(*r1), ret_tc(*r2));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Compare two range_t by their touched and in use percentage.
|
||||||
|
* \param r1,r2 Pointers to data to compare.
|
||||||
|
* \return Like strcmp.
|
||||||
|
*/
|
||||||
|
int comp_tcperc(struct range_t *r1, struct range_t *r2)
|
||||||
|
{
|
||||||
|
return comp_double(ret_tcperc(*r1), ret_tcperc(*r2));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Percentage in use in range.
|
||||||
|
* \param r A range structure.
|
||||||
|
* \return Usage percentage of the given range.
|
||||||
|
*/
|
||||||
|
double ret_percent(struct range_t r)
|
||||||
|
{
|
||||||
|
return r.count / get_range_size(&r);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief Touched and in use in range
|
||||||
|
* \param r A range structure.
|
||||||
|
* \return Number of touched or in use addresses in the given range.
|
||||||
|
*/
|
||||||
|
double ret_tc(struct range_t r)
|
||||||
{
|
{
|
||||||
return (r.count + r.touched);
|
return (r.count + r.touched);
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long int ret_tcperc(struct range_t r)
|
/*! \brief Return percentage of addresses touched and in use in range.
|
||||||
|
* \param r A range structure.
|
||||||
|
* \return Percentage of touched or in use addresses in the given range.
|
||||||
|
*/
|
||||||
|
double ret_tcperc(struct range_t r)
|
||||||
{
|
{
|
||||||
float f;
|
return ret_tc(r) / get_range_size(&r);
|
||||||
f = (float)(r.count + r.touched) / (r.last_ip - r.first_ip - 1);
|
|
||||||
return ((unsigned long int)(f * 10000));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void field_selector(char c)
|
/*! \brief Sort field selector.
|
||||||
|
* \param c Symbolic name of a sort by character.
|
||||||
|
* The sort algorithms are stabile, which means multiple sorts can be
|
||||||
|
* specified and they do not mess the result of previous sort. The sort
|
||||||
|
* algorithms are used via function pointer, that gets to be reassigned.
|
||||||
|
* \return Return the selected compare function.
|
||||||
|
*/
|
||||||
|
comparer_t field_selector(char c)
|
||||||
{
|
{
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 'n':
|
case 'n':
|
||||||
break;
|
return NULL;
|
||||||
case 'i':
|
case 'i':
|
||||||
returner = ret_ip;
|
return comp_ip;
|
||||||
break;
|
|
||||||
case 'm':
|
case 'm':
|
||||||
returner = ret_max;
|
return comp_max;
|
||||||
break;
|
|
||||||
case 'c':
|
case 'c':
|
||||||
returner = ret_cur;
|
return comp_cur;
|
||||||
break;
|
|
||||||
case 'p':
|
case 'p':
|
||||||
returner = ret_percent;
|
return comp_percent;
|
||||||
break;
|
|
||||||
case 't':
|
case 't':
|
||||||
returner = ret_touched;
|
return comp_touched;
|
||||||
break;
|
|
||||||
case 'T':
|
case 'T':
|
||||||
returner = ret_tc;
|
return comp_tc;
|
||||||
break;
|
|
||||||
case 'e':
|
case 'e':
|
||||||
returner = ret_tcperc;
|
return comp_tcperc;
|
||||||
break;
|
|
||||||
default:
|
default:
|
||||||
warnx("field_selector: unknown sort order `%c'", c);
|
{
|
||||||
errx(EXIT_FAILURE, "Try `%s --help' for more information.",
|
char str[2] = { c, '\0' };
|
||||||
program_invocation_short_name);
|
error(EXIT_FAILURE, 0, "field_selector: unknown sort order: %s",
|
||||||
|
quote(str));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Needed to support multiple key sorting. */
|
/*! \brief Perform requested sorting.
|
||||||
int get_order(struct range_t *restrict left, struct range_t *restrict right)
|
* \param left The left side of the merge sort.
|
||||||
|
* \param right The right side of the merge sort.
|
||||||
|
* \return Relevant for merge sort decision.
|
||||||
|
*/
|
||||||
|
static int merge(struct conf_t *state, struct range_t *restrict left, struct range_t *restrict right)
|
||||||
{
|
{
|
||||||
int i, len, ret;
|
struct output_sort *p;
|
||||||
unsigned long int lint, rint;
|
int ret;
|
||||||
|
|
||||||
len = strlen(config.sort);
|
for (p = state->sorts; p; p = p->next) {
|
||||||
for (i = 0; i < len; i++) {
|
if (p->func == NULL) {
|
||||||
/* Handling strings is case of it's own */
|
/* String sorting is special. */
|
||||||
if (config.sort[i] == 'n') {
|
ret = strcmp(left->shared_net->name, right->shared_net->name);
|
||||||
ret =
|
} else {
|
||||||
strcmp(left->shared_net->name,
|
/* Range sorts are common. */
|
||||||
right->shared_net->name);
|
ret = p->func(left, right);
|
||||||
if (0 < ret)
|
|
||||||
return (0);
|
|
||||||
if (ret < 0)
|
|
||||||
return (1);
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
if (0 < ret)
|
||||||
/* Select which function is pointed by returner */
|
return (0);
|
||||||
field_selector(config.sort[i]);
|
if (ret < 0)
|
||||||
lint = returner(*left);
|
|
||||||
rint = returner(*right);
|
|
||||||
/* If fields are equal use next sort method */
|
|
||||||
if (lint == rint) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (lint < rint) {
|
|
||||||
return (1);
|
return (1);
|
||||||
}
|
|
||||||
return (0);
|
|
||||||
}
|
}
|
||||||
|
/* this is reached if nothing was sorted */
|
||||||
/* If all returners where equal */
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void mergesort_ranges(struct range_t *restrict orig, int size, struct range_t *restrict temp)
|
/*! \brief Mergesort for range table.
|
||||||
|
*/
|
||||||
|
void mergesort_ranges(struct conf_t *state, struct range_t *restrict orig, unsigned int size,
|
||||||
|
struct range_t *restrict temp, const int root_call)
|
||||||
{
|
{
|
||||||
int left, right, i;
|
unsigned int left, i, u_right;
|
||||||
struct range_t hold;
|
struct range_t hold;
|
||||||
/* Merge sort split size */
|
/* Merge sort split size */
|
||||||
static const int MIN_MERGE_SIZE = 8;
|
static const unsigned int MIN_MERGE_SIZE = 8;
|
||||||
|
|
||||||
|
if (temp == NULL)
|
||||||
|
temp = xmalloc(sizeof(struct range_t) * size);
|
||||||
|
|
||||||
if (size < MIN_MERGE_SIZE) {
|
if (size < MIN_MERGE_SIZE) {
|
||||||
for (left = 0; left < size; left++) {
|
for (left = 0; left < size; left++) {
|
||||||
|
int s_right;
|
||||||
|
|
||||||
hold = *(orig + left);
|
hold = *(orig + left);
|
||||||
for (right = left - 1; 0 <= right; right--) {
|
for (s_right = left - 1; 0 <= s_right; s_right--) {
|
||||||
if (get_order((orig + right), &hold)) {
|
if (merge(state, (orig + s_right), &hold))
|
||||||
break;
|
break;
|
||||||
}
|
*(orig + s_right + 1) = *(orig + s_right);
|
||||||
*(orig + right + 1) = *(orig + right);
|
|
||||||
}
|
}
|
||||||
*(orig + right + 1) = hold;
|
*(orig + s_right + 1) = hold;
|
||||||
}
|
}
|
||||||
|
if (root_call)
|
||||||
|
free(temp);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
mergesort_ranges(state, orig, size / 2, temp, 0);
|
||||||
mergesort_ranges(orig, size / 2, temp);
|
mergesort_ranges(state, orig + size / 2, size - size / 2, temp, 0);
|
||||||
mergesort_ranges(orig + size / 2, size - size / 2, temp);
|
|
||||||
|
|
||||||
left = 0;
|
left = 0;
|
||||||
right = size / 2;
|
u_right = size / 2;
|
||||||
i = 0;
|
i = 0;
|
||||||
|
while (left < size / 2 && u_right < size) {
|
||||||
while (left < size / 2 && right < size) {
|
if (merge(state, (orig + left), (orig + u_right))) {
|
||||||
if (get_order((orig + left), (orig + right))) {
|
|
||||||
*(temp + i) = *(orig + left);
|
*(temp + i) = *(orig + left);
|
||||||
left++;
|
left++;
|
||||||
} else {
|
} else {
|
||||||
*(temp + i) = *(orig + right);
|
*(temp + i) = *(orig + u_right);
|
||||||
right++;
|
u_right++;
|
||||||
}
|
}
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
@ -217,11 +329,13 @@ void mergesort_ranges(struct range_t *restrict orig, int size, struct range_t *r
|
||||||
left++;
|
left++;
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
while (right < size) {
|
while (u_right < size) {
|
||||||
*(temp + i) = *(orig + right);
|
*(temp + i) = *(orig + u_right);
|
||||||
right++;
|
u_right++;
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(orig, temp, size * sizeof(struct range_t));
|
memcpy(orig, temp, size * sizeof(struct range_t));
|
||||||
|
|
||||||
|
if (root_call)
|
||||||
|
free(temp);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
59
tests/Makemodule.am
Normal file
59
tests/Makemodule.am
Normal file
|
|
@ -0,0 +1,59 @@
|
||||||
|
TESTS = \
|
||||||
|
tests/alarm-count-option \
|
||||||
|
tests/alarm-critical \
|
||||||
|
tests/alarm-critical-ranges \
|
||||||
|
tests/alarm-critical-snets \
|
||||||
|
tests/alarm-ignore \
|
||||||
|
tests/alarm-ok \
|
||||||
|
tests/alarm-shared-ok \
|
||||||
|
tests/alarm-warning \
|
||||||
|
tests/alarm-warning-ranges \
|
||||||
|
tests/alarm-warning-snets \
|
||||||
|
tests/shnet-alarm \
|
||||||
|
tests/big-small \
|
||||||
|
tests/binding-states \
|
||||||
|
tests/bootp \
|
||||||
|
tests/cidr-v4 \
|
||||||
|
tests/cidr-v6 \
|
||||||
|
tests/complete \
|
||||||
|
tests/complete-perfdata \
|
||||||
|
tests/empty \
|
||||||
|
tests/errors \
|
||||||
|
tests/full-json \
|
||||||
|
tests/full-xml \
|
||||||
|
tests/formats \
|
||||||
|
tests/leading0 \
|
||||||
|
tests/one-ip \
|
||||||
|
tests/one-line \
|
||||||
|
tests/parser \
|
||||||
|
tests/range4 \
|
||||||
|
tests/range6 \
|
||||||
|
tests/same-twice \
|
||||||
|
tests/shufled \
|
||||||
|
tests/simple \
|
||||||
|
tests/skip \
|
||||||
|
tests/sorts \
|
||||||
|
tests/statuses \
|
||||||
|
tests/v6 \
|
||||||
|
tests/v6-perfdata
|
||||||
|
|
||||||
|
if ENABLE_MUSTACH
|
||||||
|
TESTS += \
|
||||||
|
tests/mustach
|
||||||
|
endif
|
||||||
|
|
||||||
|
EXTRA_DIST += \
|
||||||
|
tests/confs \
|
||||||
|
tests/expected \
|
||||||
|
tests/leases \
|
||||||
|
tests/test.sh \
|
||||||
|
$(TESTS)
|
||||||
|
|
||||||
|
TESTS_ENVIRONMENT = top_srcdir=$(top_srcdir) PATH=$(top_builddir)$(PATH_SEPARATOR)$$PATH
|
||||||
|
|
||||||
|
check-local: $(SHELL) $(top_srcdir)/tests/test.sh
|
||||||
|
|
||||||
|
clean-local-tests:
|
||||||
|
rm -rf $(top_builddir)/tests/outputs
|
||||||
|
|
||||||
|
CLEAN_LOCALS += clean-local-tests
|
||||||
44
tests/alarm-count-option
Executable file
44
tests/alarm-count-option
Executable file
|
|
@ -0,0 +1,44 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm warning both ranges and shared networks.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo '== warn count ==' > tests/outputs/$IAM
|
||||||
|
dhcpd-pools --config $top_srcdir/tests/confs/complete --leases $top_srcdir/tests/leases/complete \
|
||||||
|
--color=never --warning=40 --warn-count=20 --output=tests/outputs/$IAM-too
|
||||||
|
echo $? >> tests/outputs/$IAM-too
|
||||||
|
cat tests/outputs/$IAM-too >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
echo '== crit count ==' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools --config $top_srcdir/tests/confs/complete --leases $top_srcdir/tests/leases/complete \
|
||||||
|
--color=never --critical=40 --crit-count=20 --output=tests/outputs/$IAM-too
|
||||||
|
echo $? >> tests/outputs/$IAM-too
|
||||||
|
cat tests/outputs/$IAM-too >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
echo '== minsize ==' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--color=never --warning=40 --warn-count=20 --minsize=40 -o tests/outputs/$IAM-too
|
||||||
|
echo $? >> tests/outputs/$IAM-too
|
||||||
|
cat tests/outputs/$IAM-too >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
echo '== snet alarms ==' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--color=never --warning=40 --snet-alarms -o tests/outputs/$IAM-too
|
||||||
|
echo $? >> tests/outputs/$IAM-too
|
||||||
|
cat tests/outputs/$IAM-too >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
echo '== shared net count alarms ==' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--color=always --format t --crit-count=20 --critical=0 --warning=0 --warn-count=24 \
|
||||||
|
--output=tests/outputs/$IAM-too
|
||||||
|
echo $? >> tests/outputs/$IAM-too
|
||||||
|
cat tests/outputs/$IAM-too >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
rm -f tests/outputs/$IAM-too
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-critical
Executable file
16
tests/alarm-critical
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm critical both ranges & shared nets.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools --critical=50 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-critical-ranges
Executable file
16
tests/alarm-critical-ranges
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm critical ranges only.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -L 11 --critical=50 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-critical-snets
Executable file
16
tests/alarm-critical-snets
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm critical shared networks only.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -L 22 --critical=50 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
15
tests/alarm-ignore
Executable file
15
tests/alarm-ignore
Executable file
|
|
@ -0,0 +1,15 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Minimal regression test suite.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools --warning 1 --critical 101 --min 2 \
|
||||||
|
-c $top_srcdir/tests/confs/one-ip \
|
||||||
|
-l $top_srcdir/tests/leases/one-ip -o tests/outputs/$IAM
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-ok
Executable file
16
tests/alarm-ok
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm warning both ranges and shared networks.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools --critical=100 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-shared-ok
Executable file
16
tests/alarm-shared-ok
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Shared network should be OK, and critical range ignored.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -L 66 --critical=54 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-warning
Executable file
16
tests/alarm-warning
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm warning both ranges and shared networks.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools --warning=50 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-warning-ranges
Executable file
16
tests/alarm-warning-ranges
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm warning ranges only.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -L 11 --warning=50 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
16
tests/alarm-warning-snets
Executable file
16
tests/alarm-warning-snets
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Alarm warning shared networks only.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -L 22 --warning=50 -c $top_srcdir/tests/confs/complete \
|
||||||
|
-l $top_srcdir/tests/leases/complete -o tests/outputs/$IAM
|
||||||
|
echo $? >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
1
tests/big-small
Symbolic link
1
tests/big-small
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
test.sh
|
||||||
14
tests/binding-states
Executable file
14
tests/binding-states
Executable file
|
|
@ -0,0 +1,14 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Minimal regression test suite.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/$IAM --color=never \
|
||||||
|
-l $top_srcdir/tests/leases/$IAM -o tests/outputs/$IAM
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
1
tests/bootp
Symbolic link
1
tests/bootp
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
test.sh
|
||||||
1
tests/cidr-v4
Symbolic link
1
tests/cidr-v4
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
test.sh
|
||||||
1
tests/cidr-v6
Symbolic link
1
tests/cidr-v6
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
test.sh
|
||||||
1
tests/complete
Symbolic link
1
tests/complete
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
test.sh
|
||||||
16
tests/complete-perfdata
Executable file
16
tests/complete-perfdata
Executable file
|
|
@ -0,0 +1,16 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Minimal regression test suite.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
TESTDATA=${IAM%-*}
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/$TESTDATA \
|
||||||
|
-l $top_srcdir/tests/leases/$TESTDATA -o tests/outputs/$IAM \
|
||||||
|
--warning 80 --perfdata
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
5
tests/confs/big-small
Normal file
5
tests/confs/big-small
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.0.0.10 10.0.0.1;
|
||||||
|
}
|
||||||
|
}
|
||||||
1
tests/confs/binding-states
Symbolic link
1
tests/confs/binding-states
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
5
tests/confs/bootp
Normal file
5
tests/confs/bootp
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range dynamic-bootp 10.0.0.1 10.0.0.10;
|
||||||
|
}
|
||||||
|
}
|
||||||
31
tests/confs/cidr-v4
Normal file
31
tests/confs/cidr-v4
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
shared-network example1 {
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.0.0.1/27;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.1.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.1.0.1/27;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
shared-network example2 {
|
||||||
|
subnet 10.2.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.2.0.1/27;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.3.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.3.0.1/27;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
subnet 10.4.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.4.0.1/28;
|
||||||
|
}
|
||||||
|
}
|
||||||
10
tests/confs/cidr-v6
Normal file
10
tests/confs/cidr-v6
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
subnet6 dead:abba:1000::/56 {
|
||||||
|
range6 dead:abba:1000::2/120;
|
||||||
|
prefix6 dead:abba:1000:0100:: dead:abba:1000:ff00::/56;
|
||||||
|
}
|
||||||
|
|
||||||
|
subnet6 dead:abba:40ff::/56 {
|
||||||
|
range6 dead:abba:4000::2/120;
|
||||||
|
prefix6 dead:abba:4000:0100:: dead:abba:4000:ff00::/56;
|
||||||
|
}
|
||||||
|
|
||||||
31
tests/confs/complete
Normal file
31
tests/confs/complete
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
shared-network example1 {
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.0.0.1 10.0.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.1.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.1.0.1 10.1.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
shared-network example2 {
|
||||||
|
subnet 10.2.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.2.0.1 10.2.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.3.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.3.0.1 10.3.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
subnet 10.4.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.4.0.1 10.4.0.20;
|
||||||
|
}
|
||||||
|
}
|
||||||
0
tests/confs/empty
Normal file
0
tests/confs/empty
Normal file
1
tests/confs/formats
Symbolic link
1
tests/confs/formats
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
5
tests/confs/leading0
Normal file
5
tests/confs/leading0
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range dynamic-bootp 10.0.0.001 10.0.0.010;
|
||||||
|
}
|
||||||
|
}
|
||||||
1
tests/confs/one-ip
Normal file
1
tests/confs/one-ip
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
range 10.20.30.40;
|
||||||
1
tests/confs/one-line
Normal file
1
tests/confs/one-line
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
pool {range 10.0.0.1 10.0.0.5;range 10.0.0.6 10.0.0.10;}
|
||||||
31
tests/confs/range4
Normal file
31
tests/confs/range4
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
shared-network example1 {
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.0.0.0/27;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.1.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.1.0.1/27;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
shared-network example2 {
|
||||||
|
subnet 10.2.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.2.0.1/28;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
subnet 10.3.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.3.0.1/29;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
subnet 10.4.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.4.0.1/32;
|
||||||
|
}
|
||||||
|
}
|
||||||
4
tests/confs/range6
Normal file
4
tests/confs/range6
Normal file
|
|
@ -0,0 +1,4 @@
|
||||||
|
subnet6 dead:abba:1000::/56 {
|
||||||
|
range6 dead:abba:1000::/56;
|
||||||
|
prefix6 dead:abba:1000:0100:: dead:abba:1000:ff00::/56;
|
||||||
|
}
|
||||||
1
tests/confs/same-twice
Symbolic link
1
tests/confs/same-twice
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
simple
|
||||||
1
tests/confs/shnet-alarm
Symbolic link
1
tests/confs/shnet-alarm
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
1
tests/confs/shufled
Symbolic link
1
tests/confs/shufled
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
5
tests/confs/simple
Normal file
5
tests/confs/simple
Normal file
|
|
@ -0,0 +1,5 @@
|
||||||
|
subnet 10.0.0.0 netmask 255.255.255.0 {
|
||||||
|
pool {
|
||||||
|
range 10.0.0.1 10.0.0.10;
|
||||||
|
}
|
||||||
|
}
|
||||||
1
tests/confs/statuses
Symbolic link
1
tests/confs/statuses
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
10
tests/confs/v6
Normal file
10
tests/confs/v6
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
subnet6 dead:abba:1000::/56 {
|
||||||
|
range6 dead:abba:1000::2 dead:abba:1000:00ff:ffff:ffff:ffff:ffff;
|
||||||
|
prefix6 dead:abba:1000:0100:: dead:abba:1000:ff00::/56;
|
||||||
|
}
|
||||||
|
|
||||||
|
subnet6 dead:abba:40ff::/56 {
|
||||||
|
range6 dead:abba:4000::2 dead:abba:4000::00ff;
|
||||||
|
prefix6 dead:abba:4000:0100:: dead:abba:4000:ff00::/56;
|
||||||
|
}
|
||||||
|
|
||||||
1
tests/empty
Symbolic link
1
tests/empty
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
test.sh
|
||||||
48
tests/errors
Executable file
48
tests/errors
Executable file
|
|
@ -0,0 +1,48 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# Test error inputs.
|
||||||
|
|
||||||
|
IAM=$(basename $0)
|
||||||
|
|
||||||
|
if [ ! -d tests/outputs ]; then
|
||||||
|
mkdir tests/outputs
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo '=== output mask' >| tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
-L 88 >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
echo '=== unknown specifier' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--skip=okish >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
echo '=== color mode' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--color=sometimes >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
echo '=== IPv5' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--ip-version=5 >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
echo '=== missing conf' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete_NXFILE -l $top_srcdir/tests/leases/complete 2>&1 | \
|
||||||
|
sed 's/: ..\/..\/tests/: .\/tests/'>> tests/outputs/$IAM
|
||||||
|
|
||||||
|
echo '=== missing leases' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete_NXFILE 2>&1 | \
|
||||||
|
sed 's/: ..\/..\/tests/: .\/tests/' >> tests/outputs/$IAM
|
||||||
|
|
||||||
|
echo '=== html table' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--format=html >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
echo '=== none existing format' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
-fz >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
echo '=== broken percent input' >> tests/outputs/$IAM
|
||||||
|
dhcpd-pools -c $top_srcdir/tests/confs/complete -l $top_srcdir/tests/leases/complete \
|
||||||
|
--warning=eighty >> tests/outputs/$IAM 2>&1
|
||||||
|
|
||||||
|
diff -u $top_srcdir/tests/expected/$IAM tests/outputs/$IAM
|
||||||
|
exit $?
|
||||||
34
tests/expected/alarm-count-option
Normal file
34
tests/expected/alarm-count-option
Normal file
|
|
@ -0,0 +1,34 @@
|
||||||
|
== warn count ==
|
||||||
|
WARNING: dhcpd-pools: Ranges - crit: 0 warn: 3 ok: 2; | range_crit=0 range_warn=3 range_ok=2
|
||||||
|
Shared nets - crit: 0 warn: 1 ok: 1; | snet_crit=0 snet_warn=1 snet_ok=1
|
||||||
|
1
|
||||||
|
== crit count ==
|
||||||
|
CRITICAL: dhcpd-pools: Ranges - crit: 3 warn: 0 ok: 2; | range_crit=3 range_warn=0 range_ok=2
|
||||||
|
Shared nets - crit: 1 warn: 0 ok: 1; | snet_crit=1 snet_warn=0 snet_ok=1
|
||||||
|
2
|
||||||
|
== minsize ==
|
||||||
|
OK: Ranges - crit: 0 warn: 0 ok: 2 ignored: 3; | range_crit=0 range_warn=0 range_ok=2 range_ignored=3
|
||||||
|
Shared nets - crit: 0 warn: 0 ok: 0 ignored: 2; | snet_crit=0 snet_warn=0 snet_ok=0 snet_ignored=2
|
||||||
|
0
|
||||||
|
== snet alarms ==
|
||||||
|
WARNING: dhcpd-pools: Ranges - crit: 0 warn: 0 ok: 2; | range_crit=0 range_warn=0 range_ok=2
|
||||||
|
Shared nets - crit: 0 warn: 2 ok: 0; | snet_crit=0 snet_warn=2 snet_ok=0
|
||||||
|
1
|
||||||
|
== shared net count alarms ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
[1;31mexample1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000[0m
|
||||||
|
[1;31mexample1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000[0m
|
||||||
|
[1;31mexample2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000[0m
|
||||||
|
[1;31mexample2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000[0m
|
||||||
|
[1;31mAll networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000[0m
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
[1;31mexample1 40 21 52.500 0 21 52.500[0m
|
||||||
|
[1;33mexample2 40 17 42.500 0 17 42.500[0m
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 100 43 43.000 0 43 43.000
|
||||||
|
0
|
||||||
3
tests/expected/alarm-critical
Normal file
3
tests/expected/alarm-critical
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
CRITICAL: dhcpd-pools: Ranges - crit: 1 warn: 0 ok: 4; | range_crit=1 range_warn=0 range_ok=4
|
||||||
|
Shared nets - crit: 1 warn: 0 ok: 1; | snet_crit=1 snet_warn=0 snet_ok=1
|
||||||
|
2
|
||||||
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue