mirror of
git://git.code.sf.net/p/dhcpd-pools/code
synced 2025-12-16 15:57:00 +00:00
Compare commits
224 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 |
132 changed files with 10239 additions and 3509 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
|
@ -1,5 +1,8 @@
|
||||||
# Wildcard + in any subdir.
|
# Wildcard + in any subdir.
|
||||||
*.o
|
*.o
|
||||||
|
*.gcno
|
||||||
|
*.gcda
|
||||||
|
*.[ch].gcov
|
||||||
|
|
||||||
# Exact filename in any subdir.
|
# Exact filename in any subdir.
|
||||||
.deps
|
.deps
|
||||||
|
|
|
||||||
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>
|
||||||
31
Makefile.am
31
Makefile.am
|
|
@ -1,7 +1,6 @@
|
||||||
## 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 = \
|
EXTRA_DIST = \
|
||||||
.version \
|
.version \
|
||||||
|
|
@ -16,12 +15,40 @@ $(top_srcdir)/.version:
|
||||||
dist-hook:
|
dist-hook:
|
||||||
echo $(VERSION) > $(distdir)/.tarball-version
|
echo $(VERSION) > $(distdir)/.tarball-version
|
||||||
|
|
||||||
CLEANFILES =
|
PATHFILES =
|
||||||
|
CLEANFILES = $(PATHFILES)
|
||||||
|
EXTRA_DIST += $(PATHFILES:=.in)
|
||||||
CLEAN_LOCALS =
|
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 contrib/Makemodule.am
|
||||||
include doc/Makemodule.am
|
include doc/Makemodule.am
|
||||||
include man/Makemodule.am
|
include man/Makemodule.am
|
||||||
|
include samples/Makemodule.am
|
||||||
include src/Makemodule.am
|
include src/Makemodule.am
|
||||||
include tests/Makemodule.am
|
include tests/Makemodule.am
|
||||||
|
|
||||||
|
|
|
||||||
271
NEWS
271
NEWS
|
|
@ -5,6 +5,277 @@ 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
|
Version 2.24
|
||||||
$ git shortlog v2.23..v2.24
|
$ git shortlog v2.23..v2.24
|
||||||
|
|
|
||||||
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
|
|
||||||
|
|
|
||||||
35
THANKS
35
THANKS
|
|
@ -5,23 +5,24 @@ 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
|
Gilles Bouthenot
|
||||||
|
|
@ -29,3 +30,25 @@ Helmut Grohne
|
||||||
Joey D.
|
Joey D.
|
||||||
Ryan Steinmetz
|
Ryan Steinmetz
|
||||||
Dan Pritts
|
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δμ
|
||||||
|
|
|
||||||
36
TODO
36
TODO
|
|
@ -15,25 +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 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,13 +13,14 @@
|
||||||
# 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
|
||||||
|
|
@ -27,9 +28,11 @@ gnulib_modules="
|
||||||
fopen
|
fopen
|
||||||
getopt-gnu
|
getopt-gnu
|
||||||
inet_pton
|
inet_pton
|
||||||
langinfo
|
isnan
|
||||||
netinet_in
|
netinet_in
|
||||||
|
nstrftime
|
||||||
progname
|
progname
|
||||||
|
quote
|
||||||
realloc-gnu
|
realloc-gnu
|
||||||
stat
|
stat
|
||||||
stddef
|
stddef
|
||||||
|
|
@ -37,8 +40,9 @@ gnulib_modules="
|
||||||
stdlib
|
stdlib
|
||||||
stpncpy
|
stpncpy
|
||||||
strdup-posix
|
strdup-posix
|
||||||
strftime
|
|
||||||
strstr
|
strstr
|
||||||
|
strtod
|
||||||
|
time_r
|
||||||
xalloc
|
xalloc
|
||||||
"
|
"
|
||||||
|
|
||||||
|
|
@ -60,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
|
||||||
|
|
|
||||||
2
build-aux/.gitignore
vendored
2
build-aux/.gitignore
vendored
|
|
@ -1,5 +1,7 @@
|
||||||
|
/ar-lib
|
||||||
/compile
|
/compile
|
||||||
/config.guess
|
/config.guess
|
||||||
|
/config.rpath
|
||||||
/config.sub
|
/config.sub
|
||||||
/depcomp
|
/depcomp
|
||||||
/install-sh
|
/install-sh
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,12 @@
|
||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
# Print a version string.
|
# Print a version string.
|
||||||
scriptversion=2012-12-31.23; # UTC
|
scriptversion=2022-01-27.18; # UTC
|
||||||
|
|
||||||
# Copyright (C) 2007-2013 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=2012-12-31.23; # 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)
|
||||||
|
|
@ -65,19 +65,21 @@ scriptversion=2012-12-31.23; # UTC
|
||||||
# EXTRA_DIST = $(top_srcdir)/.version
|
# 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
|
||||||
|
|
||||||
|
|
||||||
me=$0
|
me=$0
|
||||||
|
|
||||||
|
year=`expr "$scriptversion" : '\([^-]*\)'`
|
||||||
version="git-version-gen $scriptversion
|
version="git-version-gen $scriptversion
|
||||||
|
|
||||||
Copyright 2011 Free Software Foundation, Inc.
|
Copyright (C) ${year} Free Software Foundation, Inc.
|
||||||
There is NO warranty. You may redistribute this software
|
License GPLv3+: GNU GPL version 3 or later <https://gnu.org/licenses/gpl.html>.
|
||||||
under the terms of the GNU General Public License.
|
This is free software: you are free to change and redistribute it.
|
||||||
For more information about these matters, see the files named COPYING."
|
There is NO WARRANTY, to the extent permitted by law."
|
||||||
|
|
||||||
usage="\
|
usage="\
|
||||||
Usage: $me [OPTION]... \$srcdir/.tarball-version [TAG-NORMALIZATION-SED-SCRIPT]
|
Usage: $me [OPTION]... \$srcdir/.tarball-version [TAG-NORMALIZATION-SED-SCRIPT]
|
||||||
|
|
@ -85,8 +87,9 @@ Print a version string.
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
|
|
||||||
--prefix prefix of git tags (default 'v')
|
--prefix PREFIX prefix of git tags (default 'v')
|
||||||
--fallback fallback version to use if \"git --version\" fails
|
--fallback VERSION
|
||||||
|
fallback version to use if \"git --version\" fails
|
||||||
|
|
||||||
--help display this help and exit
|
--help display this help and exit
|
||||||
--version output version information and exit
|
--version output version information and exit
|
||||||
|
|
@ -100,8 +103,8 @@ while test $# -gt 0; do
|
||||||
case $1 in
|
case $1 in
|
||||||
--help) echo "$usage"; exit 0;;
|
--help) echo "$usage"; exit 0;;
|
||||||
--version) echo "$version"; exit 0;;
|
--version) echo "$version"; exit 0;;
|
||||||
--prefix) shift; prefix="$1";;
|
--prefix) shift; prefix=${1?};;
|
||||||
--fallback) shift; fallback="$1";;
|
--fallback) shift; fallback=${1?};;
|
||||||
-*)
|
-*)
|
||||||
echo "$0: Unknown option '$1'." >&2
|
echo "$0: Unknown option '$1'." >&2
|
||||||
echo "$0: Try '--help' for more information." >&2
|
echo "$0: Try '--help' for more information." >&2
|
||||||
|
|
@ -140,11 +143,9 @@ 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 "x$v" = x \
|
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 "x$v" != x
|
if test "x$v" != x
|
||||||
|
|
@ -166,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
|
||||||
|
|
@ -183,9 +185,9 @@ 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
|
||||||
elif test "x$fallback" = x || git --version >/dev/null 2>&1; then
|
elif test "x$fallback" = x || git --version >/dev/null 2>&1; then
|
||||||
v=UNKNOWN
|
v=UNKNOWN
|
||||||
|
|
@ -199,7 +201,7 @@ v=`echo "$v" |sed "s/^$prefix//"`
|
||||||
# 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 "x$v_from_git" != x; 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=
|
||||||
|
|
@ -214,12 +216,12 @@ if test "x$v_from_git" != x; 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:
|
||||||
|
|
|
||||||
31
configure.ac
31
configure.ac
|
|
@ -2,18 +2,18 @@
|
||||||
# Process this file with autoconf to produce a configure script.
|
# Process this file with autoconf to produce a configure script.
|
||||||
|
|
||||||
AC_PREREQ([2.69])
|
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],[],
|
[kerolasa@iki.fi],[],
|
||||||
[http://dhcpd-pools.sourceforge.net/])
|
[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([
|
AM_INIT_AUTOMAKE([
|
||||||
-Wall
|
-Wall
|
||||||
-Wextra-portability
|
-Wextra-portability
|
||||||
1.10
|
1.12
|
||||||
foreign
|
foreign
|
||||||
dist-xz
|
dist-xz
|
||||||
no-dist-gzip
|
no-dist-gzip
|
||||||
|
|
@ -24,6 +24,9 @@ AM_INIT_AUTOMAKE([
|
||||||
AC_CONFIG_SRCDIR([src/dhcpd-pools.h])
|
AC_CONFIG_SRCDIR([src/dhcpd-pools.h])
|
||||||
AC_CONFIG_HEADERS([config.h])
|
AC_CONFIG_HEADERS([config.h])
|
||||||
|
|
||||||
|
m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])],
|
||||||
|
[AC_SUBST([AM_DEFAULT_VERBOSITY], [1])])
|
||||||
|
|
||||||
# Checks for programs
|
# Checks for programs
|
||||||
AC_USE_SYSTEM_EXTENSIONS
|
AC_USE_SYSTEM_EXTENSIONS
|
||||||
AC_C_RESTRICT
|
AC_C_RESTRICT
|
||||||
|
|
@ -66,11 +69,27 @@ 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_COMPILE_IFELSE([AC_LANG_PROGRAM([[]], [[
|
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
|
||||||
|
|
@ -134,8 +153,7 @@ AM_CONDITIONAL([ENABLE_DOXYGEN], [test "x$enable_doxygen" = "xyes"])
|
||||||
AS_IF([test x$enable_doxygen = xyes], [
|
AS_IF([test x$enable_doxygen = xyes], [
|
||||||
AC_CHECK_PROGS([DOXYGEN], [doxygen])
|
AC_CHECK_PROGS([DOXYGEN], [doxygen])
|
||||||
AS_IF([test "x$DOXYGEN" = "x"],
|
AS_IF([test "x$DOXYGEN" = "x"],
|
||||||
AC_MSG_ERROR([doxygen not in path]),
|
AC_MSG_ERROR([doxygen not in path])
|
||||||
AC_CONFIG_FILES([doc/doxy.conf])
|
|
||||||
)
|
)
|
||||||
|
|
||||||
AC_CHECK_PROGS([DOXYGEN_DOT], [dot])
|
AC_CHECK_PROGS([DOXYGEN_DOT], [dot])
|
||||||
|
|
@ -149,7 +167,6 @@ AM_CONDITIONAL([HAVE_DOXYGEN_DOT], [test "x$DOXYGEN_DOT" != "x"])
|
||||||
AC_CONFIG_FILES([
|
AC_CONFIG_FILES([
|
||||||
Makefile
|
Makefile
|
||||||
lib/Makefile
|
lib/Makefile
|
||||||
man/dhcpd-pools.1
|
|
||||||
])
|
])
|
||||||
|
|
||||||
AC_OUTPUT
|
AC_OUTPUT
|
||||||
|
|
|
||||||
1
contrib/.gitignore
vendored
Normal file
1
contrib/.gitignore
vendored
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
/nagios.conf
|
||||||
|
|
@ -1,3 +1,5 @@
|
||||||
contribdir = $(datadir)/dhcpd-pools/
|
contribdir = $(datadir)/dhcpd-pools/
|
||||||
dist_contrib_SCRIPTS = contrib/dhcpd-pools.cgi contrib/snmptest.pl
|
PATHFILES += contrib/nagios.conf
|
||||||
EXTRA_DIST += contrib/nagios.conf contrib/munin_plugins
|
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-----
|
||||||
|
|
@ -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
|
||||||
|
|
@ -1,10 +1,14 @@
|
||||||
|
PATHFILES += \
|
||||||
|
doc/doxy.conf
|
||||||
|
nodist_noinst_DATA = \
|
||||||
|
doc/doxy.conf
|
||||||
|
|
||||||
EXTRA_DIST += \
|
EXTRA_DIST += \
|
||||||
doc/doxy.conf.in \
|
|
||||||
doc/introduction.dox
|
doc/introduction.dox
|
||||||
|
|
||||||
if ENABLE_DOXYGEN
|
if ENABLE_DOXYGEN
|
||||||
|
|
||||||
doxyfile.stamp:
|
doc/doxyfile.stamp:
|
||||||
$(DOXYGEN) doc/doxy.conf
|
$(DOXYGEN) doc/doxy.conf
|
||||||
date > doc/doxyfile.stamp
|
date > doc/doxyfile.stamp
|
||||||
|
|
||||||
|
|
|
||||||
2492
doc/doxy.conf.in
2492
doc/doxy.conf.in
File diff suppressed because it is too large
Load diff
218
lib/.gitignore
vendored
218
lib/.gitignore
vendored
|
|
@ -1,45 +1,52 @@
|
||||||
/*.o
|
/alloca.h
|
||||||
/*.lo
|
/alloca.in.h
|
||||||
/.deps/
|
|
||||||
/.gitignore~
|
|
||||||
/.libs/
|
|
||||||
/arpa/
|
|
||||||
/Makefile
|
|
||||||
/Makefile.am
|
|
||||||
/Makefile.in
|
|
||||||
/_Exit.c
|
|
||||||
/arg-nonnull.h
|
/arg-nonnull.h
|
||||||
|
/arpa/
|
||||||
|
/arpa_inet.c
|
||||||
/arpa_inet.in.h
|
/arpa_inet.in.h
|
||||||
/atexit.c
|
/assert.in.h
|
||||||
/c++defs.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-strcase.h
|
/c++defs.h
|
||||||
/c-strcasecmp.c
|
/cdefs.h
|
||||||
/c-strcaseeq.h
|
/cloexec.c
|
||||||
/c-strncasecmp.c
|
/cloexec.h
|
||||||
/calloc.c
|
|
||||||
/charset.alias
|
|
||||||
/close-stream.c
|
|
||||||
/close-stream.h
|
|
||||||
/close.c
|
/close.c
|
||||||
/closeout.c
|
/closeout.c
|
||||||
/closeout.h
|
/closeout.h
|
||||||
/config.charset
|
/close-stream.c
|
||||||
/configmake.h
|
/close-stream.h
|
||||||
/dosname.h
|
/c-strcasecmp.c
|
||||||
|
/c-strcaseeq.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
|
||||||
/fclose.c
|
/fclose.c
|
||||||
|
/fcntl.c
|
||||||
/fcntl.h
|
/fcntl.h
|
||||||
/fcntl.in.h
|
/fcntl.in.h
|
||||||
/fd-hook.c
|
/fd-hook.c
|
||||||
/fd-hook.h
|
/fd-hook.h
|
||||||
/fdopen.c
|
/fdopen.c
|
||||||
/fflush.c
|
/fflush.c
|
||||||
|
/filename.h
|
||||||
|
/flexmember.h
|
||||||
|
/float.c
|
||||||
|
/float+.h
|
||||||
|
/float.h
|
||||||
|
/float.in.h
|
||||||
/fopen.c
|
/fopen.c
|
||||||
/fpending.c
|
/fpending.c
|
||||||
/fpending.h
|
/fpending.h
|
||||||
|
|
@ -52,91 +59,204 @@
|
||||||
/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
|
/inet_pton.c
|
||||||
/intprops.h
|
/intprops.h
|
||||||
/langinfo.h
|
/intprops-internal.h
|
||||||
/langinfo.in.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
|
||||||
|
/locale.h
|
||||||
|
/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
|
/netinet_in.in.h
|
||||||
|
/_Noreturn.h
|
||||||
|
/nstrftime.c
|
||||||
|
/open.c
|
||||||
/pathmax.h
|
/pathmax.h
|
||||||
/progname.c
|
/progname.c
|
||||||
/progname.h
|
/progname.h
|
||||||
/quote.h
|
|
||||||
/quotearg.c
|
/quotearg.c
|
||||||
/quotearg.h
|
/quotearg.h
|
||||||
|
/quote.h
|
||||||
|
/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
|
||||||
/stdalign.h
|
/stat-time.c
|
||||||
/stdalign.in.h
|
/stat-time.h
|
||||||
/stdarg.in.h
|
/stat-w32.c
|
||||||
/stdbool.in.h
|
/stat-w32.h
|
||||||
|
/stdckdint.h
|
||||||
|
/stdckdint.in.h
|
||||||
|
/stddef.h
|
||||||
/stddef.in.h
|
/stddef.in.h
|
||||||
/stdio-impl.h
|
/stdint.in.h
|
||||||
/stdio.c
|
|
||||||
/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
|
||||||
/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.c
|
||||||
/sys_socket.in.h
|
/sys_socket.in.h
|
||||||
/sys_stat.in.h
|
/sys_stat.in.h
|
||||||
/sys_types.in.h
|
/sys_types.in.h
|
||||||
/sys_uio.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.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.c
|
|
||||||
/wctype.h
|
/wctype.h
|
||||||
|
/wctype-h.c
|
||||||
/wctype.in.h
|
/wctype.in.h
|
||||||
|
/windows-initguard.h
|
||||||
/xalloc-die.c
|
/xalloc-die.c
|
||||||
/xalloc-oversized.h
|
|
||||||
/xalloc.h
|
/xalloc.h
|
||||||
|
/xalloc-oversized.h
|
||||||
/xmalloc.c
|
/xmalloc.c
|
||||||
/xprintf.c
|
|
||||||
/xprintf.h
|
|
||||||
/xstrtod.c
|
|
||||||
/xstrtod.h
|
|
||||||
/stpncpy.c
|
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,2 @@
|
||||||
man_MANS = man/dhcpd-pools.1
|
man_MANS = man/dhcpd-pools.1
|
||||||
EXTRA_DIST += man/dhcpd-pools.1.in
|
PATHFILES += man/dhcpd-pools.1
|
||||||
CLEANFILES += 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,29 +68,29 @@ 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[thHcxXjJ]\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 embedding 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 )
|
||||||
|
|
@ -86,6 +105,18 @@ to include ethernet address.
|
||||||
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.
|
||||||
|
|
@ -104,29 +135,17 @@ 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(:);
|
||||||
.TP
|
ll.
|
||||||
.B 01
|
0\fI1\fR:Print ranges
|
||||||
Print ranges
|
0\fI2\fR:Print shared networks
|
||||||
.TP
|
0\fI4\fR:Print total summary
|
||||||
.B 02
|
\fI1\fR0:Print range header
|
||||||
Print shared networks
|
\fI2\fR0:Print shared network header
|
||||||
.TP
|
\fI4\fR0:Print total summary header
|
||||||
.B 04
|
.TE
|
||||||
Print total summary
|
|
||||||
.TP
|
|
||||||
.B 10
|
|
||||||
Print range header
|
|
||||||
.TP
|
|
||||||
.B 20
|
|
||||||
Print shared network header
|
|
||||||
.TP
|
|
||||||
.B 40
|
|
||||||
Print total summary header
|
|
||||||
.PD
|
|
||||||
.RE
|
|
||||||
.IP
|
.IP
|
||||||
The output limit for total summary has special meaning in
|
The output limit for total summary has special meaning in
|
||||||
.B \-\-warning
|
.B \-\-warning
|
||||||
|
|
@ -136,6 +155,37 @@ 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
|
to be seen then in the case of alarming returning success nothing is
|
||||||
printed.
|
printed.
|
||||||
.TP
|
.TP
|
||||||
|
\fB\-\-color\fR=\fIwhen\fR
|
||||||
|
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
|
||||||
|
\fB\-\-skip\fR=\fIwhen\fR
|
||||||
|
The
|
||||||
|
.I when
|
||||||
|
can be one of the following:
|
||||||
|
.IR ok ,
|
||||||
|
.IR warning ,
|
||||||
|
.IR critical ,
|
||||||
|
.IR minsize ,
|
||||||
|
or
|
||||||
|
.IR suppressed .
|
||||||
|
The skipping criteria is exact match with colors in \-\-color option.
|
||||||
|
.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
|
||||||
cause an alarm. If either a range or shared network will exceed
|
cause an alarm. If either a range or shared network will exceed
|
||||||
|
|
@ -159,12 +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\-\-minsize\fR=\fIsize\f
|
\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
|
Ignore ranges and shared networks that are smaller or equal to the
|
||||||
defined size. This option is meaningful only in context of alarming, and
|
defined size. This option is meaningful only in context of alarming, and
|
||||||
will intented to supress for example single host ranges. By default this
|
will intended to suppress for example single host ranges. By default this
|
||||||
option is not in use.
|
option is not in use.
|
||||||
.TP
|
.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
|
||||||
|
|
@ -193,7 +286,7 @@ $ dhcpd-pools \-c dhcpd.conf \-l dhcpd.leases \-L 22 \-\-critical 70 \-\-warning
|
||||||
.br
|
.br
|
||||||
[no-output]
|
[no-output]
|
||||||
.br
|
.br
|
||||||
Supress printing OK, and make alarm only to go off if shared networks
|
Suppress printing OK, and make alarm only to go off if shared networks
|
||||||
exceed critial or warning levels.
|
exceed critial or warning levels.
|
||||||
.SH FILES
|
.SH FILES
|
||||||
.TP
|
.TP
|
||||||
|
|
@ -202,14 +295,23 @@ ISC dhcpd configuration file.
|
||||||
.TP
|
.TP
|
||||||
@DHCPDLEASE_FILE@
|
@DHCPDLEASE_FILE@
|
||||||
ISC dhcpd lease file.
|
ISC dhcpd lease file.
|
||||||
|
.TP
|
||||||
|
@docdir@/prometheus.template
|
||||||
|
Prometheus text file collector mustach template.
|
||||||
.SH AUTHORS
|
.SH AUTHORS
|
||||||
Original design by Sami Kerola.
|
Original design by Sami Kerola.
|
||||||
.br
|
.br
|
||||||
|
uthash by Troy D. Hanson.
|
||||||
|
.br
|
||||||
XML support by Dominic Germain, Sogetel inc.
|
XML support by Dominic Germain, Sogetel inc.
|
||||||
.br
|
.br
|
||||||
IPv6 support by Cheer Xiao.
|
IPv6 support by Cheer Xiao.
|
||||||
.PP
|
.br
|
||||||
The software has FreeBSD License.
|
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@
|
||||||
|
|
@ -222,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}}
|
||||||
18
src/.indent.pro
vendored
18
src/.indent.pro
vendored
|
|
@ -1,2 +1,16 @@
|
||||||
-linux -Tuintmax_t -TFILE -Tsize_t -Toff_t -Ttime_t -ppi1
|
-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
|
||||||
|
|
|
||||||
|
|
@ -4,11 +4,10 @@ bin_PROGRAMS = dhcpd-pools
|
||||||
AC_PROG_RANLIB = resolv
|
AC_PROG_RANLIB = resolv
|
||||||
AM_CPPFLAGS = -I$(top_srcdir)/src -I$(top_srcdir)/lib -I$(top_builddir)/lib
|
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_LDADD = $(top_builddir)/lib/libdhcpd_pools.la $(MATH_LIBS)
|
||||||
|
|
||||||
dhcpd_pools_SOURCES = \
|
dhcpd_pools_SOURCES = \
|
||||||
src/analyze.c \
|
src/analyze.c \
|
||||||
src/defaults.h \
|
|
||||||
src/dhcpd-pools.c \
|
src/dhcpd-pools.c \
|
||||||
src/dhcpd-pools.h \
|
src/dhcpd-pools.h \
|
||||||
src/getdata.c \
|
src/getdata.c \
|
||||||
|
|
@ -16,3 +15,10 @@ dhcpd_pools_SOURCES = \
|
||||||
src/other.c \
|
src/other.c \
|
||||||
src/output.c \
|
src/output.c \
|
||||||
src/sort.c
|
src/sort.c
|
||||||
|
|
||||||
|
if ENABLE_MUSTACH
|
||||||
|
dhcpd_pools_SOURCES += \
|
||||||
|
src/mustach-dhcpd-pools.c \
|
||||||
|
src/mustach.c \
|
||||||
|
src/mustach.h
|
||||||
|
endif
|
||||||
|
|
|
||||||
|
|
@ -46,40 +46,33 @@
|
||||||
#include "dhcpd-pools.h"
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
/*! \brief Prepare data for analysis. The function will sort leases and
|
/*! \brief Prepare data for analysis. The function will sort leases and
|
||||||
* ranges.
|
* ranges. */
|
||||||
* FIXME: This function should return void. */
|
void prepare_data(struct conf_t *state)
|
||||||
int prepare_data(void)
|
|
||||||
{
|
{
|
||||||
/* Sort leases */
|
/* Sort leases */
|
||||||
HASH_SORT(leases, ipcomp);
|
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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Perform counting. Join leases with ranges, and update counters.
|
/*!\brief Perform counting. Join leases with ranges, and update range and
|
||||||
* FIXME: This function should return void. */
|
* shared network counters. */
|
||||||
int do_counting(void)
|
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;
|
||||||
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 && ipcomp(&range_p->first_ip, &l->ip) < 0;
|
while (l != NULL && ipcomp(&range_p->first_ip, &l->ip) < 0)
|
||||||
l = l->hh.prev)
|
l = l->hh.prev; /* rewind */
|
||||||
/* rewind */ ;
|
|
||||||
if (l == NULL)
|
if (l == NULL)
|
||||||
l = leases;
|
l = state->leases;
|
||||||
for (; l != NULL && ipcomp(&l->ip, &range_p->last_ip) <= 0;
|
for (; l != NULL && ipcomp(&l->ip, &range_p->last_ip) <= 0; l = l->hh.next) {
|
||||||
l = l->hh.next) {
|
if (unlikely(ipcomp(&l->ip, &range_p->first_ip) < 0))
|
||||||
if (ipcomp(&l->ip, &range_p->first_ip) < 0) {
|
continue; /* cannot happen? */
|
||||||
/* should not be necessary */
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
/* IP in range */
|
/* IP in range */
|
||||||
switch (l->type) {
|
switch (l->type) {
|
||||||
case FREE:
|
case FREE:
|
||||||
|
|
@ -91,46 +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 = get_range_size(range_p);
|
||||||
if (range_p->shared_net) {
|
/* Count together ranges within shared network block. */
|
||||||
range_p->shared_net->available += block_size;
|
range_p->shared_net->available += block_size;
|
||||||
|
range_p->shared_net->used += range_p->count;
|
||||||
|
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. The below
|
|
||||||
* 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 += get_range_size(range_p);
|
|
||||||
shared_networks->used += range_p->count;
|
|
||||||
shared_networks->touched += range_p->touched;
|
|
||||||
shared_networks->backups += range_p->backups;
|
|
||||||
range_p++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,55 +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.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*! \file defaults.h
|
|
||||||
* \brief Default settings which cannot be changed without recompiling
|
|
||||||
* the software.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef DEFAULTS_H
|
|
||||||
# define DEFAULTS_H 1
|
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
|
||||||
|
|
||||||
/*! \var MAXLEN
|
|
||||||
* \brief Maximum expected line length in dhcpd.conf and dhcpd.leases
|
|
||||||
* files. */
|
|
||||||
static const size_t MAXLEN = 1024;
|
|
||||||
|
|
||||||
/*! \var SHARED_NETWORKS
|
|
||||||
* \brief Maximum number of different shared networks in dhcpd.conf file. */
|
|
||||||
static const unsigned int SHARED_NETWORKS = 8192;
|
|
||||||
|
|
||||||
#endif /* DEFAULTS_H */
|
|
||||||
|
|
@ -38,166 +38,253 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#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 "progname.h"
|
||||||
|
#include "quote.h"
|
||||||
#include "xalloc.h"
|
#include "xalloc.h"
|
||||||
|
|
||||||
/*! \brief Start of execution. Parse options, and call other other
|
#include "dhcpd-pools.h"
|
||||||
* functions one after another. At the moment adding threading support
|
|
||||||
* would be difficult, but there does not seem to be valid reason to
|
/* Function pointers */
|
||||||
* consider that. Overall the analysis already quick enough even without
|
int (*parse_ipaddr) (struct conf_t *state, const char *restrict src, union ipaddr_t *restrict dst);
|
||||||
* making it parallel.
|
void (*copy_ipaddr) (union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
||||||
*
|
const char *(*ntop_ipaddr) (const union ipaddr_t *ip);
|
||||||
* \return Return value indicates success or fail or analysis, unless
|
double (*get_range_size) (const struct range_t *r);
|
||||||
* either --warning or --critical options are in use, which makes the
|
int (*xstrstr) (struct conf_t *state, const char *restrict str);
|
||||||
* return value in some cases to match with Nagios expectations about
|
int (*ipcomp) (const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
||||||
* alarming. */
|
int (*leasecomp) (const struct leases_t *restrict a, const struct leases_t *restrict b);
|
||||||
int main(int argc, char **argv)
|
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);
|
||||||
int i, sorts = 0;
|
|
||||||
int option_index = 0;
|
/*! \brief An option argument parser to populate state header_limit and
|
||||||
char const *tmp;
|
* number_limit values.
|
||||||
struct range_t *tmp_ranges;
|
*/
|
||||||
enum {
|
static int return_limit(const char c)
|
||||||
OPT_WARN = CHAR_MAX + 1,
|
{
|
||||||
OPT_CRIT,
|
if ('0' <= c && c < '8')
|
||||||
OPT_MINSIZE
|
return c - '0';
|
||||||
};
|
error(EXIT_FAILURE, 0, "return_limit: output mask %s is illegal", quote(optarg));
|
||||||
int ret_val;
|
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},
|
{"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);
|
|
||||||
set_program_name(argv[0]);
|
|
||||||
|
|
||||||
/* FIXME: These allocations should be fully dynamic, e.g., grow
|
|
||||||
* if needed. */
|
|
||||||
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');
|
|
||||||
config.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;
|
|
||||||
config.backups_found = false;
|
|
||||||
|
|
||||||
/* Parse command line options */
|
|
||||||
while (1) {
|
while (1) {
|
||||||
int c;
|
int c;
|
||||||
c = getopt_long(argc, argv, "c:l:f:o:s:rL:vh",
|
|
||||||
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
|
||||||
clean_up();
|
state->mustach_template = optarg;
|
||||||
errx(EXIT_FAILURE,
|
state->output_format = 'm';
|
||||||
"main: output mask `%s' is illegal",
|
state->print_mac_addreses = 1;
|
||||||
optarg);
|
#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 =
|
state->warning = strtod_or_err(optarg, "illegal argument");
|
||||||
strtod_or_err(optarg, "illegal argument");
|
|
||||||
break;
|
break;
|
||||||
case OPT_CRIT:
|
case OPT_CRIT:
|
||||||
strcpy(config.output_format, "a");
|
alarming = 1;
|
||||||
config.critical =
|
state->critical = strtod_or_err(optarg, "illegal argument");
|
||||||
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;
|
break;
|
||||||
case OPT_MINSIZE:
|
case OPT_MINSIZE:
|
||||||
config.minsize =
|
state->minsize = strtod_or_err(optarg, "illegal argument");
|
||||||
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 */
|
||||||
|
|
@ -206,88 +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_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;
|
|
||||||
config.fullhtml = true;
|
|
||||||
break;
|
|
||||||
case 'x':
|
|
||||||
output_analysis = output_xml;
|
|
||||||
break;
|
|
||||||
case 'X':
|
|
||||||
output_analysis = output_xml;
|
|
||||||
break;
|
|
||||||
case 'j':
|
|
||||||
output_analysis = output_json;
|
|
||||||
break;
|
|
||||||
case 'J':
|
|
||||||
output_analysis = output_json;
|
|
||||||
break;
|
|
||||||
case 'c':
|
|
||||||
output_analysis = output_csv;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
clean_up();
|
|
||||||
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);
|
||||||
set_ipv_functions(VERSION_UNKNOWN);
|
parse_leases(&state);
|
||||||
parse_config(true, config.dhcpdconf_file, shared_networks);
|
prepare_data(&state);
|
||||||
|
do_counting(&state);
|
||||||
parse_leases();
|
if (state.sorts != NULL)
|
||||||
prepare_data();
|
mergesort_ranges(&state, state.ranges, state.num_ranges, NULL, 1);
|
||||||
do_counting();
|
if (state.reverse_order == 1)
|
||||||
tmp_ranges = xmalloc(sizeof(struct range_t) * num_ranges);
|
flip_ranges(&state);
|
||||||
if (sorts != 0) {
|
ret_val = output_analysis(&state);
|
||||||
mergesort_ranges(ranges, num_ranges, tmp_ranges);
|
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);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Run time initialization. Global allocations, counter
|
|
||||||
* initializations, etc are here.
|
|
||||||
* FIXME: This function should return void. */
|
|
||||||
int prepare_memory(void)
|
|
||||||
{
|
|
||||||
config.dhcp_version = VERSION_UNKNOWN;
|
|
||||||
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);
|
|
||||||
|
|
||||||
/* 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;
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -35,8 +35,6 @@
|
||||||
|
|
||||||
/*! \file dhcpd-pools.h
|
/*! \file dhcpd-pools.h
|
||||||
* \brief Global definitions of structures, enums, and function prototypes.
|
* \brief Global definitions of structures, enums, and function prototypes.
|
||||||
* FIXME: The file has too many global variables. Most of them should be
|
|
||||||
* removed, if not all.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef DHCPD_POOLS_H
|
#ifndef DHCPD_POOLS_H
|
||||||
|
|
@ -44,7 +42,6 @@
|
||||||
|
|
||||||
# include <config.h>
|
# include <config.h>
|
||||||
# include <arpa/inet.h>
|
# include <arpa/inet.h>
|
||||||
# include <stdbool.h>
|
|
||||||
# include <stddef.h>
|
# include <stddef.h>
|
||||||
# include <stdio.h>
|
# include <stdio.h>
|
||||||
# include <string.h>
|
# include <string.h>
|
||||||
|
|
@ -64,23 +61,37 @@
|
||||||
# define unlikely(x) (x)
|
# define unlikely(x) (x)
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
|
/*! \def _DP_ATTRIBUTE_HOT
|
||||||
|
* \brief The function attribute __hot__ was added in gcc 4.3. See gnu
|
||||||
|
* documentation for further information.
|
||||||
|
* https://gcc.gnu.org/onlinedocs/gcc/Common-Function-Attributes.html#index-hot-function-attribute
|
||||||
|
*/
|
||||||
|
# if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
||||||
|
# define _DP_ATTRIBUTE_HOT __attribute__ ((__hot__))
|
||||||
|
# else
|
||||||
|
# define _DP_ATTRIBUTE_HOT /* empty */
|
||||||
|
# endif
|
||||||
|
|
||||||
/*! \union ipaddr_t
|
/*! \union ipaddr_t
|
||||||
* \brief Memory space for a binary IP address saving. */
|
* \brief Memory space for a binary IP address saving. */
|
||||||
union ipaddr_t {
|
union ipaddr_t {
|
||||||
uint32_t v4;
|
uint32_t v4;
|
||||||
unsigned char v6[16];
|
unsigned char v6[16];
|
||||||
};
|
};
|
||||||
|
|
||||||
/*! \enum dhcp_version
|
/*! \enum dhcp_version
|
||||||
* \brief Indicator which IP version is in use.
|
* \brief The IP version, IPv4 or IPv6, served by the dhcpd.
|
||||||
*/
|
*/
|
||||||
enum dhcp_version {
|
enum dhcp_version {
|
||||||
VERSION_4,
|
IPvUNKNOWN,
|
||||||
VERSION_6,
|
IPv4,
|
||||||
VERSION_UNKNOWN
|
IPv6
|
||||||
};
|
};
|
||||||
|
|
||||||
/*! \enum prefix_t
|
/*! \enum prefix_t
|
||||||
* \brief Enumeration of interesting data in dhcpd.leases file, that has
|
* \brief Enumeration of interesting data in dhcpd.leases file, that has to
|
||||||
* to be further examined, and saved.
|
* be further examined, and saved. Functions xstrstr_v4() and xstrstr_v6()
|
||||||
|
* return one of these values to parse_leases().
|
||||||
*/
|
*/
|
||||||
enum prefix_t {
|
enum prefix_t {
|
||||||
PREFIX_LEASE,
|
PREFIX_LEASE,
|
||||||
|
|
@ -91,29 +102,25 @@ enum prefix_t {
|
||||||
PREFIX_BINDING_STATE_ACTIVE,
|
PREFIX_BINDING_STATE_ACTIVE,
|
||||||
PREFIX_BINDING_STATE_BACKUP,
|
PREFIX_BINDING_STATE_BACKUP,
|
||||||
PREFIX_HARDWARE_ETHERNET,
|
PREFIX_HARDWARE_ETHERNET,
|
||||||
|
PREFIX_STARTS,
|
||||||
|
PREFIX_ENDS,
|
||||||
|
PREFIX_HOSTNAME,
|
||||||
NUM_OF_PREFIX
|
NUM_OF_PREFIX
|
||||||
};
|
};
|
||||||
/*! \struct configuration_t
|
|
||||||
* \brief Runtime configuration.
|
/*! \enum color_mode
|
||||||
|
* \brief Enumeration whether to use or not color output.
|
||||||
*/
|
*/
|
||||||
struct configuration_t {
|
enum color_mode {
|
||||||
char dhcpv6;
|
color_unknown,
|
||||||
enum dhcp_version dhcp_version;
|
color_off,
|
||||||
char *dhcpdconf_file;
|
color_on,
|
||||||
char *dhcpdlease_file;
|
color_auto /*!< Default, use colors when output terminal is interactive. */
|
||||||
char output_format[2];
|
|
||||||
bool fullhtml;
|
|
||||||
char sort[6];
|
|
||||||
bool reverse_order;
|
|
||||||
char *output_file;
|
|
||||||
int output_limit[2];
|
|
||||||
bool backups_found;
|
|
||||||
double warning;
|
|
||||||
double critical;
|
|
||||||
double minsize;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*! \struct shared_network_t
|
/*! \struct shared_network_t
|
||||||
* \brief Counters for an individual shared network.
|
* \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;
|
||||||
|
|
@ -121,7 +128,10 @@ struct shared_network_t {
|
||||||
double used;
|
double used;
|
||||||
double touched;
|
double touched;
|
||||||
double backups;
|
double backups;
|
||||||
|
struct shared_network_t *next;
|
||||||
|
int netmask;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*! \struct range_t
|
/*! \struct range_t
|
||||||
* \brief Counters for an individual range.
|
* \brief Counters for an individual range.
|
||||||
*/
|
*/
|
||||||
|
|
@ -133,162 +143,230 @@ struct range_t {
|
||||||
double touched;
|
double touched;
|
||||||
double backups;
|
double backups;
|
||||||
};
|
};
|
||||||
/*! \enum isc_conf_parser
|
|
||||||
* \brief Configuration file parsing state flags.
|
/*! \struct output_helper_t
|
||||||
|
* \brief Various per range and shared net temporary calculation results.
|
||||||
*/
|
*/
|
||||||
enum isc_conf_parser {
|
struct output_helper_t {
|
||||||
ITS_NOTHING_INTERESTING,
|
double range_size;
|
||||||
ITS_A_RANGE_FIRST_IP,
|
double percent;
|
||||||
ITS_A_RANGE_SECOND_IP,
|
double tc;
|
||||||
ITS_A_SHAREDNET,
|
double tcp;
|
||||||
ITS_AN_INCLUCE
|
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
|
/*! \enum ltype
|
||||||
* \brief Lease state types.
|
* \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
|
/*! \struct leases_t
|
||||||
* \brief An individual lease. The leaases are hashed.
|
* \brief An individual lease. These leaases are hashed.
|
||||||
*/
|
*/
|
||||||
struct leases_t {
|
struct leases_t {
|
||||||
union ipaddr_t ip; /* ip as key */
|
union ipaddr_t ip; /* ip as key */
|
||||||
enum ltype type;
|
|
||||||
char *ethernet;
|
char *ethernet;
|
||||||
UT_hash_handle hh;
|
UT_hash_handle hh;
|
||||||
|
enum ltype type;
|
||||||
|
char *ends;
|
||||||
|
char *starts;
|
||||||
|
char *hostname;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*! \enum limbits
|
/*! \enum limbits
|
||||||
* \brief Output limit bits
|
* \brief Output limit bits.
|
||||||
*/
|
*/
|
||||||
enum limbits {
|
enum limbits {
|
||||||
BIT1 = 0x1,
|
R_BIT = (1 << 0), /*!< Range limit. */
|
||||||
BIT2 = 0x2,
|
S_BIT = (1 << 1), /*!< Shared networks limit. */
|
||||||
BIT3 = 0x4
|
A_BIT = (1 << 2) /*!< All networks summary limit. */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Global variables */
|
/*! \def STATE_OK
|
||||||
/* \var prefix_length Length of each prefix. */
|
* \brief Nagios alarm exit value.
|
||||||
int prefix_length[2][NUM_OF_PREFIX];
|
*/
|
||||||
/* \var config Runtime configuration. */
|
# define STATE_OK 0
|
||||||
struct configuration_t config;
|
# define STATE_WARNING 1
|
||||||
/* \var shared_networks Pointer holding shared network count results. */
|
# define STATE_CRITICAL 2
|
||||||
struct shared_network_t *shared_networks;
|
|
||||||
/* \var num_shared_networks Number of shared networks found. */
|
/*! \var comparer_t
|
||||||
unsigned int num_shared_networks;
|
* \brief Function pointer holding sort algorithm.
|
||||||
/* \var ranges Pointer holding range count results. */
|
*/
|
||||||
struct range_t *ranges;
|
typedef int (*comparer_t) (struct range_t *r1, struct range_t *r2);
|
||||||
/* \var num_ranges Number of ranges found. */
|
|
||||||
unsigned int num_ranges;
|
/*! \struct output_sort
|
||||||
/* \var leases Pointer holding all leases. */
|
* \brief Linked list of sort functions.
|
||||||
struct leases_t *leases;
|
*/
|
||||||
/*! \var RANGES Maximum number of ranges. */
|
struct output_sort {
|
||||||
unsigned int RANGES;
|
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);
|
|
||||||
void set_ipv_functions(int version);
|
/* analyze.c */
|
||||||
int parse_leases(void);
|
extern void prepare_data(struct conf_t *state);
|
||||||
void parse_config(int, const char *__restrict,
|
extern void do_counting(struct conf_t *state);
|
||||||
struct shared_network_t *__restrict)
|
|
||||||
__attribute__ ((nonnull(2, 3)));
|
/* getdata.c */
|
||||||
int prepare_data(void);
|
extern int parse_leases(struct conf_t *state);
|
||||||
int do_counting(void);
|
extern void parse_config(struct conf_t *state, const int is_include,
|
||||||
void flip_ranges(struct range_t *__restrict ranges,
|
const char *restrict config_file,
|
||||||
struct range_t *__restrict tmp_ranges)
|
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)));
|
__attribute__ ((nonnull(1, 2)));
|
||||||
/* support functions */
|
|
||||||
int (*parse_ipaddr)(const char *restrict src, union ipaddr_t *restrict dst);
|
|
||||||
int parse_ipaddr_init(const char *restrict src, union ipaddr_t *restrict dst);
|
|
||||||
int parse_ipaddr_v4(const char *restrict src, union ipaddr_t *restrict dst);
|
|
||||||
int parse_ipaddr_v6(const char *restrict src, union ipaddr_t *restrict dst);
|
|
||||||
|
|
||||||
void (*copy_ipaddr)(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
extern int comp_cur(struct range_t *r1, struct range_t *r2);
|
||||||
void copy_ipaddr_init(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
extern int comp_double(double f1, double f2);
|
||||||
void copy_ipaddr_v4(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
extern int comp_ip(struct range_t *r1, struct range_t *r2);
|
||||||
void copy_ipaddr_v6(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src);
|
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);
|
||||||
|
|
||||||
const char *(*ntop_ipaddr)(const union ipaddr_t *ip);
|
extern comparer_t field_selector(char c);
|
||||||
const char *ntop_ipaddr_init(const union ipaddr_t *ip);
|
extern double ret_percent(struct range_t r);
|
||||||
const char *ntop_ipaddr_v4(const union ipaddr_t *ip);
|
extern double ret_tc(struct range_t r);
|
||||||
const char *ntop_ipaddr_v6(const union ipaddr_t *ip);
|
extern double ret_tcperc(struct range_t r);
|
||||||
|
|
||||||
double (*get_range_size)(const struct range_t *r);
|
#endif /* DHCPD_POOLS_H */
|
||||||
double get_range_size_init(const struct range_t *r);
|
|
||||||
double get_range_size_v4(const struct range_t *r);
|
|
||||||
double get_range_size_v6(const struct range_t *r);
|
|
||||||
|
|
||||||
int (*xstrstr)(const char *__restrict str);
|
|
||||||
int xstrstr_init(const char *__restrict str)
|
|
||||||
# if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
|
||||||
__attribute__ ((__hot__))
|
|
||||||
# endif
|
|
||||||
;
|
|
||||||
int xstrstr_v4(const char *__restrict str)
|
|
||||||
# if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
|
||||||
__attribute__ ((__hot__))
|
|
||||||
# endif
|
|
||||||
;
|
|
||||||
int xstrstr_v6(const char *__restrict str)
|
|
||||||
# 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 (*ipcomp)(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
|
||||||
int ipcomp_init(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
|
||||||
int ipcomp_v4(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
|
||||||
int ipcomp_v6(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b);
|
|
||||||
|
|
||||||
int comp_cur(struct range_t *r1, struct range_t *r2);
|
|
||||||
int comp_double(double f1, double f2);
|
|
||||||
int comp_ip(struct range_t *r1, struct range_t *r2);
|
|
||||||
int comp_max(struct range_t *r1, struct range_t *r2);
|
|
||||||
int comp_percent(struct range_t *r1, struct range_t *r2);
|
|
||||||
int comp_tc(struct range_t *r1, struct range_t *r2);
|
|
||||||
int comp_tcperc(struct range_t *r1, struct range_t *r2);
|
|
||||||
int comp_touched(struct range_t *r1, struct range_t *r2);
|
|
||||||
int rangecomp(const void *__restrict r1, const void *__restrict r2)
|
|
||||||
__attribute__ ((nonnull(1, 2)));
|
|
||||||
/* sort function pointer and functions */
|
|
||||||
typedef int (*comparer_t) (struct range_t *r1, struct range_t *r2);
|
|
||||||
comparer_t field_selector(char c);
|
|
||||||
double ret_percent(struct range_t r);
|
|
||||||
double ret_tc(struct range_t r);
|
|
||||||
double ret_tcperc(struct range_t r);
|
|
||||||
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_json(void);
|
|
||||||
int output_csv(void);
|
|
||||||
int output_alarming(void);
|
|
||||||
/* Memory release, file closing etc */
|
|
||||||
void clean_up(void);
|
|
||||||
/* Hash functions */
|
|
||||||
void (*add_lease)(union ipaddr_t *ip, enum ltype type);
|
|
||||||
void add_lease_init(union ipaddr_t *ip, enum ltype type);
|
|
||||||
void add_lease_v4(union ipaddr_t *ip, enum ltype type);
|
|
||||||
void add_lease_v6(union ipaddr_t *ip, enum ltype type);
|
|
||||||
|
|
||||||
struct leases_t *(*find_lease)(union ipaddr_t *ip);
|
|
||||||
struct leases_t *find_lease_init(union ipaddr_t *ip);
|
|
||||||
struct leases_t *find_lease_v4(union ipaddr_t *ip);
|
|
||||||
struct leases_t *find_lease_v6(union ipaddr_t *ip);
|
|
||||||
|
|
||||||
void delete_lease(struct leases_t *lease);
|
|
||||||
void delete_all_leases(void);
|
|
||||||
|
|
||||||
#endif /* DHCPD_POOLS_H */
|
|
||||||
|
|
|
||||||
404
src/getdata.c
404
src/getdata.c
|
|
@ -40,117 +40,143 @@
|
||||||
#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"
|
||||||
|
|
||||||
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
|
/*! \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
|
/*! \brief Lease file parser. The parser can only read ISC DHCPD
|
||||||
* dhcpd.leases file format. */
|
* dhcpd.leases file format. */
|
||||||
int parse_leases(void)
|
int parse_leases(struct conf_t *state)
|
||||||
{
|
{
|
||||||
FILE *dhcpd_leases;
|
FILE *dhcpd_leases;
|
||||||
char *line, *ipstring, macstring[20], *stop;
|
char *line, *ipstring, macstring[20], *stop, endsstr[30], startsstr[30], hostnamestr[MAXLEN];
|
||||||
union ipaddr_t addr;
|
union ipaddr_t addr;
|
||||||
struct stat lease_file_stats;
|
|
||||||
bool ethernets = false;
|
|
||||||
struct leases_t *lease;
|
struct leases_t *lease;
|
||||||
|
|
||||||
dhcpd_leases = fopen(config.dhcpdlease_file, "r");
|
dhcpd_leases = fopen(state->dhcpdlease_file, "r");
|
||||||
if (dhcpd_leases == NULL) {
|
if (dhcpd_leases == NULL)
|
||||||
err(EXIT_FAILURE, "parse_leases: %s", config.dhcpdlease_file);
|
error(EXIT_FAILURE, errno, "parse_leases: %s", state->dhcpdlease_file);
|
||||||
}
|
|
||||||
#ifdef HAVE_POSIX_FADVISE
|
#ifdef HAVE_POSIX_FADVISE
|
||||||
# ifdef POSIX_FADV_NOREUSE
|
|
||||||
if (posix_fadvise(fileno(dhcpd_leases), 0, 0, POSIX_FADV_NOREUSE) != 0) {
|
|
||||||
err(EXIT_FAILURE, "parse_leases: fadvise %s",
|
|
||||||
config.dhcpdlease_file);
|
|
||||||
}
|
|
||||||
# endif /* POSIX_FADV_NOREUSE */
|
|
||||||
# ifdef POSIX_FADV_SEQUENTIAL
|
# ifdef POSIX_FADV_SEQUENTIAL
|
||||||
if (posix_fadvise(fileno(dhcpd_leases), 0, 0, POSIX_FADV_SEQUENTIAL) != 0) {
|
if (posix_fadvise(fileno(dhcpd_leases), 0, 0, POSIX_FADV_SEQUENTIAL) != 0)
|
||||||
err(EXIT_FAILURE, "parse_leases: fadvise %s",
|
error(EXIT_FAILURE, errno, "parse_leases: fadvise %s", state->dhcpdlease_file);
|
||||||
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' || config.output_format[0] == 'J') {
|
ipstring[0] = '\0';
|
||||||
ethernets = true;
|
endsstr[0] = '\0';
|
||||||
}
|
startsstr[0] = '\0';
|
||||||
|
hostnamestr[0] = '\0';
|
||||||
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 */
|
||||||
switch(xstrstr(line)) {
|
|
||||||
/* It's a lease, save IP */
|
|
||||||
case PREFIX_LEASE:
|
case PREFIX_LEASE:
|
||||||
stop = memccpy(ipstring, line + (config.dhcp_version == VERSION_4 ? 6 : 9), ' ', strlen(line));
|
stop =
|
||||||
|
memccpy(ipstring,
|
||||||
|
line + (state->ip_version ==
|
||||||
|
IPv4 ? 6 : 9), ' ', strlen(line));
|
||||||
if (stop != NULL) {
|
if (stop != NULL) {
|
||||||
--stop;
|
--stop;
|
||||||
*stop = '\0';
|
*stop = '\0';
|
||||||
}
|
}
|
||||||
parse_ipaddr(ipstring, &addr);
|
parse_ipaddr(state, ipstring, &addr);
|
||||||
break;
|
break;
|
||||||
case PREFIX_BINDING_STATE_FREE:
|
case PREFIX_BINDING_STATE_FREE:
|
||||||
case PREFIX_BINDING_STATE_ABANDONED:
|
case PREFIX_BINDING_STATE_ABANDONED:
|
||||||
case PREFIX_BINDING_STATE_EXPIRED:
|
case PREFIX_BINDING_STATE_EXPIRED:
|
||||||
case PREFIX_BINDING_STATE_RELEASED:
|
case PREFIX_BINDING_STATE_RELEASED:
|
||||||
if ((lease = find_lease(&addr)) != NULL) {
|
if ((lease = find_lease(state, &addr)) != NULL)
|
||||||
delete_lease(lease);
|
delete_lease(state, lease);
|
||||||
}
|
add_lease(state, &addr, FREE);
|
||||||
add_lease(&addr, FREE);
|
|
||||||
break;
|
break;
|
||||||
case PREFIX_BINDING_STATE_ACTIVE:
|
case PREFIX_BINDING_STATE_ACTIVE:
|
||||||
/* remove old entry, if exists */
|
/* remove old entry, if exists */
|
||||||
if ((lease = find_lease(&addr)) != NULL) {
|
if ((lease = find_lease(state, &addr)) != NULL)
|
||||||
delete_lease(lease);
|
delete_lease(state, lease);
|
||||||
}
|
add_lease(state, &addr, ACTIVE);
|
||||||
add_lease(&addr, ACTIVE);
|
|
||||||
break;
|
break;
|
||||||
case PREFIX_BINDING_STATE_BACKUP:
|
case PREFIX_BINDING_STATE_BACKUP:
|
||||||
/* remove old entry, if exists */
|
/* remove old entry, if exists */
|
||||||
if ((lease = find_lease(&addr)) != NULL) {
|
if ((lease = find_lease(state, &addr)) != NULL)
|
||||||
delete_lease(lease);
|
delete_lease(state, lease);
|
||||||
}
|
add_lease(state, &addr, BACKUP);
|
||||||
add_lease(&addr, BACKUP);
|
state->backups_found = 1;
|
||||||
config.backups_found = true;
|
|
||||||
break;
|
break;
|
||||||
case PREFIX_HARDWARE_ETHERNET:
|
case PREFIX_HARDWARE_ETHERNET:
|
||||||
if (ethernets == false)
|
if (state->print_mac_addreses == 0)
|
||||||
break;
|
break;
|
||||||
memcpy(macstring, line + 20, 17);
|
memcpy(macstring, line + 20, 17);
|
||||||
macstring[17] = '\0';
|
macstring[17] = '\0';
|
||||||
if ((lease = find_lease(&addr)) != NULL) {
|
if ((lease = find_lease(state, &addr)) != NULL) {
|
||||||
lease->ethernet = xstrdup(macstring);
|
lease->ethernet = xstrdup(macstring);
|
||||||
|
lease->starts = xstrdup(startsstr);
|
||||||
|
startsstr[0] = '\0';
|
||||||
|
lease->ends = xstrdup(endsstr);
|
||||||
|
endsstr[0] = '\0';
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case PREFIX_ENDS:
|
||||||
|
if (state->print_mac_addreses == 0)
|
||||||
|
break;
|
||||||
|
strncpy(endsstr, line + 7, sizeof(endsstr)-1);
|
||||||
|
endsstr[strlen(endsstr)-2] = '\0';
|
||||||
|
break;
|
||||||
|
case PREFIX_STARTS:
|
||||||
|
if (state->print_mac_addreses == 0)
|
||||||
|
break;
|
||||||
|
strncpy(startsstr, line + 9, sizeof(startsstr)-1);
|
||||||
|
startsstr[strlen(startsstr)-2] = '\0';
|
||||||
|
break;
|
||||||
|
case PREFIX_HOSTNAME:
|
||||||
|
if (state->print_mac_addreses == 0)
|
||||||
|
break;
|
||||||
|
strncpy(hostnamestr, line + 19, sizeof(hostnamestr)-1);
|
||||||
|
hostnamestr[strlen(hostnamestr)-3] = '\0';
|
||||||
|
if ((lease = find_lease(state, &addr)) != NULL) {
|
||||||
|
lease->hostname = xstrdup(hostnamestr);
|
||||||
|
}
|
||||||
|
hostnamestr[0] = '\0';
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
/* do nothing */;
|
/* do nothing */ ;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#undef HAS_PREFIX
|
#undef HAS_PREFIX
|
||||||
|
|
@ -163,70 +189,88 @@ int parse_leases(void)
|
||||||
/*! \brief Keyword search in dhcpd.conf file.
|
/*! \brief Keyword search in dhcpd.conf file.
|
||||||
* \param s A line from the dhcpd.conf file.
|
* \param s A line from the dhcpd.conf file.
|
||||||
* \return Indicator what configuration was found. */
|
* \return Indicator what configuration was found. */
|
||||||
static int is_interesting_config_clause(char const *restrict s)
|
static int is_interesting_config_clause(struct conf_t *state, char const *restrict s)
|
||||||
{
|
{
|
||||||
if (strstr(s, "range"))
|
if (strstr(s, "range"))
|
||||||
return ITS_A_RANGE_FIRST_IP;
|
return ITS_A_RANGE_FIRST_IP;
|
||||||
if (strstr(s, "shared-network"))
|
if (strstr(s, "shared-network"))
|
||||||
return ITS_A_SHAREDNET;
|
return ITS_A_SHAREDNET;
|
||||||
|
if (state->all_as_shared) {
|
||||||
|
if (strstr(s, "subnet"))
|
||||||
|
return ITS_A_SUBNET;
|
||||||
|
if (strstr(s, "netmask"))
|
||||||
|
return ITS_A_NETMASK;
|
||||||
|
}
|
||||||
if (strstr(s, "include"))
|
if (strstr(s, "include"))
|
||||||
return ITS_AN_INCLUCE;
|
return ITS_AN_INCLUDE;
|
||||||
return ITS_NOTHING_INTERESTING;
|
return ITS_NOTHING_INTERESTING;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief The dhcpd.conf file parser.
|
/*! \brief Flip first and last IP in range if they are in unusual order.
|
||||||
* FIXME: This spaghetti monster function need to be rewrote at least
|
|
||||||
* ones.
|
|
||||||
*/
|
*/
|
||||||
void parse_config(int is_include, const char *restrict config_file,
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*! \brief The dhcpd.conf file parser.
|
||||||
|
* FIXME: This spaghetti monster function needs to be rewrote at least
|
||||||
|
* ones more.
|
||||||
|
*/
|
||||||
|
void parse_config(struct conf_t *state, const 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, one_ip_range = false;
|
int newclause = 1, comment = 0, one_ip_range = 0; /* booleans */
|
||||||
int quote = 0, braces = 0, argument = ITS_NOTHING_INTERESTING;
|
int quote = 0, braces = 0, argument = ITS_NOTHING_INTERESTING;
|
||||||
size_t i = 0;
|
size_t i = 0;
|
||||||
char *word;
|
char *word;
|
||||||
int braces_shared = 1000;
|
int braces_shared = 1000;
|
||||||
union ipaddr_t addr;
|
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_NOREUSE
|
#ifdef HAVE_POSIX_FADVISE
|
||||||
if (posix_fadvise(fileno(dhcpd_config), 0, 0, POSIX_FADV_NOREUSE) != 0) {
|
# ifdef POSIX_FADV_SEQUENTIAL
|
||||||
err(EXIT_FAILURE, "parse_config: fadvise %s", config_file);
|
if (posix_fadvise(fileno(dhcpd_config), 0, 0, POSIX_FADV_SEQUENTIAL) != 0)
|
||||||
}
|
error(EXIT_FAILURE, errno, "parse_config: fadvise %s", config_file);
|
||||||
#endif /* POSIX_FADV_NOREUSE */
|
# endif /* POSIX_FADV_SEQUENTIAL */
|
||||||
#ifdef POSIX_FADV_SEQUENTIAL
|
#endif /* HAVE_POSIX_FADVISE */
|
||||||
if (posix_fadvise(fileno(dhcpd_config), 0, 0, POSIX_FADV_SEQUENTIAL) != 0) {
|
|
||||||
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))) {
|
||||||
char c;
|
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 == 0) {
|
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;
|
||||||
|
|
@ -235,22 +279,19 @@ 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 == 0) {
|
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 != ITS_A_RANGE_FIRST_IP
|
&& argument != ITS_A_RANGE_FIRST_IP
|
||||||
&& argument != ITS_A_RANGE_SECOND_IP
|
&& argument != ITS_A_RANGE_SECOND_IP && argument != ITS_AN_INCLUDE) {
|
||||||
&& argument != ITS_AN_INCLUCE) {
|
newclause = 1;
|
||||||
newclause = true;
|
|
||||||
i = 0;
|
i = 0;
|
||||||
} else if (argument == ITS_A_RANGE_FIRST_IP && one_ip_range == true) {
|
} else if (argument == ITS_A_RANGE_FIRST_IP && one_ip_range == 1) {
|
||||||
argument = ITS_A_RANGE_SECOND_IP;
|
argument = ITS_A_RANGE_SECOND_IP;
|
||||||
c = ' ';
|
c = ' ';
|
||||||
} else if (argument == ITS_A_RANGE_SECOND_IP && 0 < i) {
|
} else if (argument == ITS_A_RANGE_SECOND_IP && 0 < i) {
|
||||||
|
|
@ -261,165 +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) {
|
} 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;
|
range_p->last_ip = range_p->first_ip;
|
||||||
goto newrange;
|
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
|
if (comment == 1 || (newclause == 0 && argument == ITS_NOTHING_INTERESTING))
|
||||||
|| (newclause == false
|
|
||||||
&& argument == ITS_NOTHING_INTERESTING)) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
|
||||||
/* Strip white spaces before new clause word. */
|
/* Strip white spaces before new clause word. */
|
||||||
if ((newclause == true || argument != ITS_NOTHING_INTERESTING)
|
if ((newclause == 1 || argument != ITS_NOTHING_INTERESTING)
|
||||||
&& isspace(c) && i == 0 && one_ip_range == false) {
|
&& 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 != ITS_NOTHING_INTERESTING)
|
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) {
|
if (argument == ITS_A_RANGE_FIRST_IP)
|
||||||
one_ip_range = true;
|
one_ip_range = 1;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
/* words after range, shared-network or include */
|
/* words after range, shared-network or include */
|
||||||
else if (argument != ITS_NOTHING_INTERESTING) {
|
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 ITS_A_RANGE_SECOND_IP:
|
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;
|
||||||
argument = ITS_NOTHING_INTERESTING;
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
parse_ipaddr(word, &addr);
|
if (strchr(word, '/')) {
|
||||||
if (one_ip_range == true) {
|
parse_cidr(state, range_p, word);
|
||||||
one_ip_range = false;
|
one_ip_range = 0;
|
||||||
copy_ipaddr(&range_p->first_ip, &addr);
|
} 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);
|
||||||
}
|
}
|
||||||
copy_ipaddr(&range_p->last_ip, &addr);
|
|
||||||
newrange:
|
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 ITS_A_RANGE_FIRST_IP:
|
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 (!(parse_ipaddr(word, &addr))) {
|
if (!(parse_ipaddr(state, word, &addr)))
|
||||||
/* word was not ip, try
|
/* word was not ip, try again */
|
||||||
* again */
|
|
||||||
break;
|
break;
|
||||||
}
|
|
||||||
copy_ipaddr(&range_p->first_ip, &addr);
|
copy_ipaddr(&range_p->first_ip, &addr);
|
||||||
one_ip_range = false;
|
one_ip_range = 0;
|
||||||
argument = ITS_A_RANGE_SECOND_IP;
|
argument = ITS_A_RANGE_SECOND_IP;
|
||||||
break;
|
break;
|
||||||
case ITS_A_SHAREDNET:
|
case ITS_A_SHAREDNET:
|
||||||
/* printf ("shared-network named: %s\n", word); */
|
case ITS_A_SUBNET:
|
||||||
num_shared_networks++;
|
/* ignore subnets inside a shared-network */
|
||||||
shared_p =
|
if (argument == ITS_A_SUBNET && shared_p != state->shared_net_root) {
|
||||||
shared_networks + num_shared_networks;
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
state->shared_net_head->next = xcalloc(sizeof(struct shared_network_t), 1);
|
||||||
|
state->shared_net_head = state->shared_net_head->next;
|
||||||
|
shared_p = state->shared_net_head;
|
||||||
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
|
braces_shared = braces;
|
||||||
* away by reallocating
|
break;
|
||||||
* more space. */
|
case ITS_A_NETMASK:
|
||||||
errx(EXIT_FAILURE,
|
/* fill in only when requested to do so */
|
||||||
"parse_config: increase default.h SHARED_NETWORKS and recompile");
|
if (shared_p->netmask) {
|
||||||
|
if (!(parse_ipaddr(state, word, &addr)))
|
||||||
|
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;
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
braces_shared = braces;
|
braces_shared = braces;
|
||||||
break;
|
break;
|
||||||
case ITS_AN_INCLUCE:
|
case ITS_AN_INCLUDE:
|
||||||
/* printf ("include file: %s\n", word); */
|
/* printf ("include file: %s\n", word); */
|
||||||
argument = ITS_NOTHING_INTERESTING;
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
parse_config(false, word, shared_p);
|
parse_config(state, 0, word, shared_p);
|
||||||
newclause = true;
|
newclause = 1;
|
||||||
break;
|
break;
|
||||||
case ITS_NOTHING_INTERESTING:
|
case ITS_NOTHING_INTERESTING:
|
||||||
/* printf ("nothing interesting: %s\n", word); */
|
/* printf ("nothing interesting: %s\n", word); */
|
||||||
argument = ITS_NOTHING_INTERESTING;
|
argument = ITS_NOTHING_INTERESTING;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
warnx("impossible occurred, report a bug");
|
puts("impossible occurred, report a bug");
|
||||||
assert(0);
|
abort();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
68
src/hash.c
68
src/hash.c
|
|
@ -38,90 +38,118 @@
|
||||||
* analysis happen as quick as possible..
|
* analysis happen as quick as possible..
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
#include "xalloc.h"
|
#include "xalloc.h"
|
||||||
|
|
||||||
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
#define HASH_FIND_V6(head, findv6, out) HASH_FIND(hh, head, findv6, 16, out)
|
#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)
|
#define HASH_ADD_V6(head, v6field, add) HASH_ADD(hh, head, v6field, 16, add)
|
||||||
|
|
||||||
/*! \brief Add a lease to hash array.
|
/*! \brief Add a lease to hash array.
|
||||||
* \param addr Binary IP to be added in leases hash.
|
* \param addr Binary IP to be added in leases hash.
|
||||||
* \param type Lease state of the IP. */
|
* \param type Lease state of the IP. */
|
||||||
void add_lease_init(union ipaddr_t *addr __attribute__((unused)), enum ltype type __attribute__((unused)))
|
void add_lease_init(struct conf_t *state __attribute__ ((unused)), union ipaddr_t *addr
|
||||||
|
__attribute__ ((unused)), enum ltype type __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
void add_lease_v4(union ipaddr_t *addr, enum ltype type)
|
void add_lease_v4(struct conf_t *state, union ipaddr_t *addr, enum ltype type)
|
||||||
{
|
{
|
||||||
struct leases_t *l;
|
struct leases_t *l;
|
||||||
|
|
||||||
l = xmalloc(sizeof(struct leases_t));
|
l = xmalloc(sizeof(struct leases_t));
|
||||||
copy_ipaddr(&l->ip, addr);
|
copy_ipaddr(&l->ip, addr);
|
||||||
l->type = type;
|
l->type = type;
|
||||||
HASH_ADD_INT(leases, ip.v4, l);
|
HASH_ADD_INT(state->leases, ip.v4, l);
|
||||||
l->ethernet = NULL;
|
l->ethernet = NULL;
|
||||||
|
l->ends = NULL;
|
||||||
|
l->starts = NULL;
|
||||||
|
l->hostname = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void add_lease_v6(union ipaddr_t *addr, enum ltype type)
|
void add_lease_v6(struct conf_t *state, union ipaddr_t *addr, enum ltype type)
|
||||||
{
|
{
|
||||||
struct leases_t *l;
|
struct leases_t *l;
|
||||||
|
|
||||||
l = xmalloc(sizeof(struct leases_t));
|
l = xmalloc(sizeof(struct leases_t));
|
||||||
copy_ipaddr(&l->ip, addr);
|
copy_ipaddr(&l->ip, addr);
|
||||||
l->type = type;
|
l->type = type;
|
||||||
HASH_ADD_V6(leases, ip.v6, l);
|
HASH_ADD_V6(state->leases, ip.v6, l);
|
||||||
l->ethernet = NULL;
|
l->ethernet = NULL;
|
||||||
|
l->ends = NULL;
|
||||||
|
l->starts = NULL;
|
||||||
|
l->hostname = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Find pointer to lease from hash array.
|
/*! \brief Find pointer to lease from hash array.
|
||||||
* \param addr Binary IP searched from leases hash.
|
* \param addr Binary IP searched from leases hash.
|
||||||
* \return A lease structure about requested IP, or NULL.
|
* \return A lease structure about requested IP, or NULL.
|
||||||
*/
|
*/
|
||||||
struct leases_t *find_lease_init(union ipaddr_t *addr __attribute__((unused)))
|
struct leases_t *find_lease_init(struct conf_t *state __attribute__ ((unused)), union ipaddr_t *addr
|
||||||
|
__attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct leases_t *find_lease_v4(union ipaddr_t *addr)
|
struct leases_t *find_lease_v4(struct conf_t *state, union ipaddr_t *addr)
|
||||||
{
|
{
|
||||||
struct leases_t *l;
|
struct leases_t *l;
|
||||||
HASH_FIND_INT(leases, &addr->v4, l);
|
|
||||||
|
HASH_FIND_INT(state->leases, &addr->v4, l);
|
||||||
return l;
|
return l;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct leases_t *find_lease_v6(union ipaddr_t *addr)
|
struct leases_t *find_lease_v6(struct conf_t *state, union ipaddr_t *addr)
|
||||||
{
|
{
|
||||||
struct leases_t *l;
|
struct leases_t *l;
|
||||||
HASH_FIND_V6(leases, &addr->v4, l);
|
|
||||||
|
HASH_FIND_V6(state->leases, &addr->v4, l);
|
||||||
return l;
|
return l;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Delete a lease from hash array.
|
/*! \brief Delete a lease from hash array.
|
||||||
* \param lease Pointer to lease hash. */
|
* \param lease Pointer to lease hash. */
|
||||||
void delete_lease(struct leases_t *lease)
|
void delete_lease(struct conf_t *state, struct leases_t *lease)
|
||||||
{
|
{
|
||||||
free(lease->ethernet);
|
free(lease->ethernet);
|
||||||
HASH_DEL(leases, lease);
|
free(lease->ends);
|
||||||
|
free(lease->starts);
|
||||||
|
free(lease->hostname);
|
||||||
|
HASH_DEL(state->leases, lease);
|
||||||
free(lease);
|
free(lease);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Delete all leases from hash array. */
|
/*! \brief Delete all leases from hash array. */
|
||||||
#ifdef HASH_ITER
|
#ifdef HASH_ITER
|
||||||
void delete_all_leases(void)
|
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_ITER(hh, state->leases, l, tmp) {
|
||||||
free(l->ethernet);
|
free(l->ethernet);
|
||||||
HASH_DEL(leases, l);
|
free(l->ends);
|
||||||
|
free(l->starts);
|
||||||
|
free(l->hostname);
|
||||||
|
HASH_DEL(state->leases, l);
|
||||||
free(l);
|
free(l);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
void delete_all_leases(void)
|
void delete_all_leases(struct conf_t *state)
|
||||||
{
|
{
|
||||||
while (leases) {
|
while (state->leases) {
|
||||||
struct leases_t *l;
|
struct leases_t *l;
|
||||||
l = leases;
|
|
||||||
|
l = state->leases;
|
||||||
free(l->ethernet);
|
free(l->ethernet);
|
||||||
HASH_DEL(leases, l); /* leases advances to next on delete */
|
free(l->ends);
|
||||||
|
free(l->starts);
|
||||||
|
free(l->hostname);
|
||||||
|
HASH_DEL(state->leases, l); /* leases advances to next on delete */
|
||||||
free(l);
|
free(l);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
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
|
||||||
|
|
||||||
448
src/other.c
448
src/other.c
|
|
@ -39,62 +39,75 @@
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include "dhcpd-pools.h"
|
|
||||||
#include "defaults.h"
|
|
||||||
#include "progname.h"
|
|
||||||
|
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include <err.h>
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
|
#include <limits.h>
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#include <stdbool.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 <limits.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.
|
/*! \brief Set function pointers depending on IP version.
|
||||||
* \param ip IP version.
|
* \param ip IP version.
|
||||||
*/
|
*/
|
||||||
void set_ipv_functions(int version)
|
void set_ipv_functions(struct conf_t *state, int version)
|
||||||
{
|
{
|
||||||
switch (version) {
|
switch (version) {
|
||||||
|
|
||||||
case VERSION_4:
|
case IPv4:
|
||||||
config.dhcp_version = version;
|
state->ip_version = version;
|
||||||
add_lease = add_lease_v4;
|
add_lease = add_lease_v4;
|
||||||
copy_ipaddr = copy_ipaddr_v4;
|
copy_ipaddr = copy_ipaddr_v4;
|
||||||
find_lease = find_lease_v4;
|
find_lease = find_lease_v4;
|
||||||
get_range_size = get_range_size_v4;
|
get_range_size = get_range_size_v4;
|
||||||
ipcomp = ipcomp_v4;
|
ipcomp = ipcomp_v4;
|
||||||
|
leasecomp = leasecomp_v4;
|
||||||
ntop_ipaddr = ntop_ipaddr_v4;
|
ntop_ipaddr = ntop_ipaddr_v4;
|
||||||
parse_ipaddr = parse_ipaddr_v4;
|
parse_ipaddr = parse_ipaddr_v4;
|
||||||
|
cidr_last = cidr_last_v4;
|
||||||
xstrstr = xstrstr_v4;
|
xstrstr = xstrstr_v4;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case VERSION_6:
|
case IPv6:
|
||||||
config.dhcp_version = version;
|
state->ip_version = version;
|
||||||
add_lease = add_lease_v6;
|
add_lease = add_lease_v6;
|
||||||
copy_ipaddr = copy_ipaddr_v6;
|
copy_ipaddr = copy_ipaddr_v6;
|
||||||
find_lease = find_lease_v6;
|
find_lease = find_lease_v6;
|
||||||
get_range_size = get_range_size_v6;
|
get_range_size = get_range_size_v6;
|
||||||
ipcomp = ipcomp_v6;
|
ipcomp = ipcomp_v6;
|
||||||
|
leasecomp = leasecomp_v6;
|
||||||
ntop_ipaddr = ntop_ipaddr_v6;
|
ntop_ipaddr = ntop_ipaddr_v6;
|
||||||
parse_ipaddr = parse_ipaddr_v6;
|
parse_ipaddr = parse_ipaddr_v6;
|
||||||
|
cidr_last = cidr_last_v6;
|
||||||
xstrstr = xstrstr_v6;
|
xstrstr = xstrstr_v6;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case VERSION_UNKNOWN:
|
case IPvUNKNOWN:
|
||||||
config.dhcp_version = version;
|
state->ip_version = version;
|
||||||
add_lease = add_lease_init;
|
add_lease = add_lease_init;
|
||||||
copy_ipaddr = copy_ipaddr_init;
|
copy_ipaddr = copy_ipaddr_init;
|
||||||
find_lease = find_lease_init;
|
find_lease = find_lease_init;
|
||||||
get_range_size = get_range_size_init;
|
get_range_size = get_range_size_init;
|
||||||
ipcomp = ipcomp_init;
|
ipcomp = ipcomp_init;
|
||||||
|
leasecomp = leasecomp_init;
|
||||||
ntop_ipaddr = ntop_ipaddr_init;
|
ntop_ipaddr = ntop_ipaddr_init;
|
||||||
parse_ipaddr = parse_ipaddr_init;
|
parse_ipaddr = parse_ipaddr_init;
|
||||||
|
cidr_last = NULL;
|
||||||
xstrstr = xstrstr_init;
|
xstrstr = xstrstr_init;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
@ -110,55 +123,160 @@ void set_ipv_functions(int version)
|
||||||
* \param dst An union which will hold conversion result.
|
* \param dst An union which will hold conversion result.
|
||||||
* \return Was parsing successful.
|
* \return Was parsing successful.
|
||||||
*/
|
*/
|
||||||
int parse_ipaddr_init(const char *restrict src, union ipaddr_t *restrict dst)
|
int parse_ipaddr_init(struct conf_t *state, const char *restrict src, union ipaddr_t *restrict dst)
|
||||||
{
|
{
|
||||||
struct in_addr addr;
|
struct in_addr addr;
|
||||||
struct in6_addr addr6;
|
struct in6_addr addr6;
|
||||||
if (inet_aton(src, &addr) == 1) {
|
|
||||||
set_ipv_functions(VERSION_4);
|
if (inet_aton(src, &addr) == 1)
|
||||||
} else if (inet_pton(AF_INET6, src, &addr6) == 1) {
|
set_ipv_functions(state, IPv4);
|
||||||
set_ipv_functions(VERSION_6);
|
else if (inet_pton(AF_INET6, src, &addr6) == 1)
|
||||||
} else {
|
set_ipv_functions(state, IPv6);
|
||||||
|
else
|
||||||
return 0;
|
return 0;
|
||||||
}
|
return parse_ipaddr(state, src, dst);
|
||||||
return parse_ipaddr(src, dst);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int parse_ipaddr_v4(const char *restrict src, union ipaddr_t *restrict dst)
|
int parse_ipaddr_v4(struct conf_t *state
|
||||||
|
__attribute__ ((unused)), const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst)
|
||||||
{
|
{
|
||||||
int rv;
|
int rv;
|
||||||
struct in_addr addr;
|
struct in_addr addr;
|
||||||
|
|
||||||
rv = inet_aton(src, &addr);
|
rv = inet_aton(src, &addr);
|
||||||
dst->v4 = ntohl(addr.s_addr);
|
dst->v4 = ntohl(addr.s_addr);
|
||||||
return rv == 1;
|
return rv == 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int parse_ipaddr_v6(const char *restrict src, union ipaddr_t *restrict dst)
|
int parse_ipaddr_v6(struct conf_t *state
|
||||||
|
__attribute__ ((unused)), const char *restrict src,
|
||||||
|
union ipaddr_t *restrict dst)
|
||||||
{
|
{
|
||||||
int rv;
|
int rv;
|
||||||
struct in6_addr addr;
|
struct in6_addr addr;
|
||||||
|
|
||||||
rv = inet_pton(AF_INET6, src, &addr);
|
rv = inet_pton(AF_INET6, src, &addr);
|
||||||
memcpy(&dst->v6, addr.s6_addr, sizeof(addr.s6_addr));
|
memcpy(&dst->v6, addr.s6_addr, sizeof(addr.s6_addr));
|
||||||
return rv == 1;
|
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.
|
/*! \brief Copy IP address to union.
|
||||||
*
|
*
|
||||||
* \param dst Destination for a binary IP address.
|
* \param dst Destination for a binary IP address.
|
||||||
* \param src Sourse of an IP address. */
|
* \param src Sourse of an IP address. */
|
||||||
void copy_ipaddr_init(union ipaddr_t *restrict dst __attribute__((unused)),
|
void copy_ipaddr_init(union ipaddr_t *restrict dst __attribute__ ((unused)),
|
||||||
const union ipaddr_t *restrict src __attribute__((unused)))
|
const union ipaddr_t *restrict src __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
void copy_ipaddr_v4(union ipaddr_t *restrict dst,
|
void copy_ipaddr_v4(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src)
|
||||||
const union ipaddr_t *restrict src)
|
|
||||||
{
|
{
|
||||||
dst->v4 = src->v4;
|
dst->v4 = src->v4;
|
||||||
}
|
}
|
||||||
|
|
||||||
void copy_ipaddr_v6(union ipaddr_t *restrict dst,
|
void copy_ipaddr_v6(union ipaddr_t *restrict dst, const union ipaddr_t *restrict src)
|
||||||
const union ipaddr_t *restrict src)
|
|
||||||
{
|
{
|
||||||
memcpy(&dst->v6, &src->v6, sizeof(src->v6));
|
memcpy(&dst->v6, &src->v6, sizeof(src->v6));
|
||||||
}
|
}
|
||||||
|
|
@ -171,25 +289,27 @@ void copy_ipaddr_v6(union ipaddr_t *restrict dst,
|
||||||
* \param ip Binary IP address.
|
* \param ip Binary IP address.
|
||||||
* \return Printable address.
|
* \return Printable address.
|
||||||
*/
|
*/
|
||||||
const char *ntop_ipaddr_init(const union ipaddr_t *ip __attribute__((unused)))
|
const char *ntop_ipaddr_init(const union ipaddr_t *ip __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
static char buffer = '\0';
|
static char buffer = '\0';
|
||||||
|
|
||||||
return &buffer;
|
return &buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
const char *ntop_ipaddr_v4(const union ipaddr_t *ip)
|
const char *ntop_ipaddr_v4(const union ipaddr_t *ip)
|
||||||
{
|
{
|
||||||
static char buffer[sizeof("255.255.255.255")];
|
static char buffer[INET_ADDRSTRLEN];
|
||||||
struct in_addr addr;
|
struct in_addr addr;
|
||||||
|
|
||||||
addr.s_addr = htonl(ip->v4);
|
addr.s_addr = htonl(ip->v4);
|
||||||
return inet_ntop(AF_INET, &addr, buffer, sizeof(buffer));
|
return inet_ntop(AF_INET, &addr, buffer, sizeof(buffer));
|
||||||
}
|
}
|
||||||
|
|
||||||
const char *ntop_ipaddr_v6(const union ipaddr_t *ip)
|
const char *ntop_ipaddr_v6(const union ipaddr_t *ip)
|
||||||
{
|
{
|
||||||
static char
|
static char buffer[INET6_ADDRSTRLEN];
|
||||||
buffer[sizeof("ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255")];
|
|
||||||
struct in6_addr addr;
|
struct in6_addr addr;
|
||||||
|
|
||||||
memcpy(addr.s6_addr, ip->v6, sizeof(addr.s6_addr));
|
memcpy(addr.s6_addr, ip->v6, sizeof(addr.s6_addr));
|
||||||
return inet_ntop(AF_INET6, &addr, buffer, sizeof(buffer));
|
return inet_ntop(AF_INET6, &addr, buffer, sizeof(buffer));
|
||||||
}
|
}
|
||||||
|
|
@ -200,7 +320,7 @@ const char *ntop_ipaddr_v6(const union ipaddr_t *ip)
|
||||||
* and last IP in the range.
|
* and last IP in the range.
|
||||||
* \return Size of a range.
|
* \return Size of a range.
|
||||||
*/
|
*/
|
||||||
double get_range_size_init(const struct range_t *r __attribute__((unused)))
|
double get_range_size_init(const struct range_t *r __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
@ -214,6 +334,7 @@ double get_range_size_v6(const struct range_t *r)
|
||||||
{
|
{
|
||||||
double size = 0;
|
double size = 0;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* When calculating the size of an IPv6 range overflow may occur.
|
/* When calculating the size of an IPv6 range overflow may occur.
|
||||||
* In that case only the last LONG_BIT bits are preserved, thus
|
* In that case only the last LONG_BIT bits are preserved, thus
|
||||||
* we just skip the first (16 - LONG_BIT) bits... */
|
* we just skip the first (16 - LONG_BIT) bits... */
|
||||||
|
|
@ -231,17 +352,14 @@ double get_range_size_v6(const struct range_t *r)
|
||||||
* \param str A line from dhcpd.conf
|
* \param str A line from dhcpd.conf
|
||||||
* \return prefix_t enum value
|
* \return prefix_t enum value
|
||||||
*/
|
*/
|
||||||
int
|
int xstrstr_init(struct conf_t *state, const char *restrict str)
|
||||||
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
|
||||||
__attribute__ ((hot))
|
|
||||||
#endif
|
|
||||||
xstrstr_init(const char *restrict str)
|
|
||||||
{
|
{
|
||||||
if (memcmp("lease ", str, 6)) {
|
if (!memcmp("lease ", str, 6)) {
|
||||||
set_ipv_functions(VERSION_4);
|
set_ipv_functions(state, IPv4);
|
||||||
return PREFIX_LEASE;
|
return PREFIX_LEASE;
|
||||||
} else if (memcmp(" iaaddr ", str, 9)) {
|
}
|
||||||
set_ipv_functions(VERSION_6);
|
if (!memcmp(" iaaddr ", str, 9)) {
|
||||||
|
set_ipv_functions(state, IPv6);
|
||||||
return PREFIX_LEASE;
|
return PREFIX_LEASE;
|
||||||
}
|
}
|
||||||
return NUM_OF_PREFIX;
|
return NUM_OF_PREFIX;
|
||||||
|
|
@ -257,14 +375,14 @@ int
|
||||||
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
||||||
__attribute__ ((hot))
|
__attribute__ ((hot))
|
||||||
#endif
|
#endif
|
||||||
xstrstr_v4(const char *restrict str)
|
xstrstr_v4(struct conf_t *state __attribute__ ((unused)), const char *restrict str)
|
||||||
{
|
{
|
||||||
size_t len;
|
size_t len;
|
||||||
if (str[2] == 'b' || str[2] == 'h') {
|
|
||||||
|
if (str[2] == 'b' || str[2] == 'h')
|
||||||
len = strlen(str);
|
len = strlen(str);
|
||||||
} else {
|
else
|
||||||
len = 0;
|
len = 0;
|
||||||
}
|
|
||||||
if (15 < len) {
|
if (15 < len) {
|
||||||
switch (str[16]) {
|
switch (str[16]) {
|
||||||
case 'f':
|
case 'f':
|
||||||
|
|
@ -293,11 +411,18 @@ int
|
||||||
if (!memcmp(" hardware ethernet", str, 19))
|
if (!memcmp(" hardware ethernet", str, 19))
|
||||||
return PREFIX_HARDWARE_ETHERNET;
|
return PREFIX_HARDWARE_ETHERNET;
|
||||||
break;
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!memcmp("lease ", str, 6)) {
|
if (!memcmp("lease ", str, 6))
|
||||||
return PREFIX_LEASE;
|
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 NUM_OF_PREFIX;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -311,14 +436,14 @@ int
|
||||||
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
#if __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3)
|
||||||
__attribute__ ((hot))
|
__attribute__ ((hot))
|
||||||
#endif
|
#endif
|
||||||
xstrstr_v6(const char *restrict str)
|
xstrstr_v6(struct conf_t *state __attribute__ ((unused)), const char *restrict str)
|
||||||
{
|
{
|
||||||
size_t len;
|
size_t len;
|
||||||
if (str[4] == 'b' || str[2] == 'h') {
|
|
||||||
|
if (str[4] == 'b' || str[2] == 'h')
|
||||||
len = strlen(str);
|
len = strlen(str);
|
||||||
} else {
|
else
|
||||||
len = 0;
|
len = 0;
|
||||||
}
|
|
||||||
if (17 < len) {
|
if (17 < len) {
|
||||||
switch (str[18]) {
|
switch (str[18]) {
|
||||||
case 'f':
|
case 'f':
|
||||||
|
|
@ -347,14 +472,37 @@ int
|
||||||
if (!memcmp(" hardware ethernet", str, 19))
|
if (!memcmp(" hardware ethernet", str, 19))
|
||||||
return PREFIX_HARDWARE_ETHERNET;
|
return PREFIX_HARDWARE_ETHERNET;
|
||||||
break;
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!memcmp(" iaaddr ", str, 9)) {
|
if (!memcmp(" iaaddr ", str, 9))
|
||||||
return PREFIX_LEASE;
|
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 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.
|
/*! \brief Return a double floating point value.
|
||||||
*
|
*
|
||||||
* \param str String to be converted to a double.
|
* \param str String to be converted to a double.
|
||||||
|
|
@ -370,118 +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);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Reverse range.
|
/*! \brief Reverse range.
|
||||||
* Used before output, if a caller has requested reverse sorting.
|
* Used before output, if a caller has requested reverse sorting. */
|
||||||
* FIXME: The temporary memory area handling should be internal to this
|
void flip_ranges(struct conf_t *state)
|
||||||
* function, not a parameter.
|
|
||||||
*
|
|
||||||
* \param flip_me The range that needs to be inverted.
|
|
||||||
* \param tmp_ranges Temporary memory area for the flip. */
|
|
||||||
void flip_ranges(struct range_t *restrict flip_me,
|
|
||||||
struct range_t *restrict tmp_ranges)
|
|
||||||
{
|
{
|
||||||
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));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief 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);
|
||||||
}
|
}
|
||||||
free(config.dhcpdconf_file);
|
for (c = state->shared_net_root; c; c = n) {
|
||||||
free(config.dhcpdlease_file);
|
n = c->next;
|
||||||
free(config.output_file);
|
free(c->name);
|
||||||
free(ranges);
|
free(c);
|
||||||
delete_all_leases();
|
}
|
||||||
if (shared_networks) {
|
}
|
||||||
num_shared_networks++;
|
|
||||||
for (i = 0; i < num_shared_networks; i++) {
|
/*! \brief Print a time stamp of a path or now to output file. */
|
||||||
free((shared_networks + i)->name);
|
void dp_time_tool(FILE *file, const char *path, const int epoch)
|
||||||
}
|
{
|
||||||
free(shared_networks);
|
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. */
|
/*! \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"
|
||||||
"Original design by Sami Kerola.\n"
|
"Original design and maintainer Sami Kerola.\n"
|
||||||
|
"uthash %s by Troy D. Hanson.\n"
|
||||||
"XML support by Dominic Germain, Sogetel inc.\n"
|
"XML support by Dominic Germain, Sogetel inc.\n"
|
||||||
"IPv6 support by Cheer Xiao.\n\n"
|
"IPv6 support by Cheer Xiao.\n"
|
||||||
"The software has FreeBSD License.\n", PACKAGE_STRING);
|
"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. */
|
/*! \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_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);
|
||||||
fprintf(out, "\
|
fputs( " t for text\n", out);
|
||||||
-f, --format=[thHcxXjJ] output format\n\
|
fputs( " H for full html page\n", out);
|
||||||
t for text\n\
|
fputs( " x for xml\n", out);
|
||||||
h for html table\n\
|
fputs( " X for xml with active lease details\n", out);
|
||||||
H for full html page\n\
|
fputs( " j for json\n", out);
|
||||||
x for xml\n\
|
fputs( " J for json with active lease details\n", out);
|
||||||
X for xml with active lease details\n\
|
fputs( " c for comma separated values\n", out);
|
||||||
j for json\n\
|
#ifdef BUILD_MUSTACH
|
||||||
J for json with active lease details\n\
|
fputs( " --mustach=FILE output using mustach template file\n", out);
|
||||||
c for comma separated values\n");
|
#endif
|
||||||
fprintf(out, "\
|
fputs( " -s, --sort=[nimcptTe] sort ranges by\n", out);
|
||||||
-s, --sort=[nimcptTe] sort ranges by\n\
|
fputs( " n name\n", out);
|
||||||
n name\n\
|
fputs( " i IP\n", out);
|
||||||
i IP\n\
|
fputs( " m maximum\n", out);
|
||||||
m maximum\n\
|
fputs( " c current\n", out);
|
||||||
c current\n\
|
fputs( " p percent\n", out);
|
||||||
p percent\n\
|
fputs( " t touched\n", out);
|
||||||
t touched\n\
|
fputs( " T t+c\n", out);
|
||||||
T t+c\n\
|
fputs( " e t+c perc\n", out);
|
||||||
e t+c perc\n\
|
fputs( " -r, --reverse reverse order sort\n", out);
|
||||||
-r, --reverse reverse order sort\n\
|
fputs( " -o, --output=FILE output into a file\n", out);
|
||||||
-o, --output=FILE output into a file\n\
|
fputs( " -L, --limit=NR output limit mask 77 - 00\n", out);
|
||||||
-L, --limit=NR output limit mask 77 - 00\n");
|
fputs( " --color=WHEN use colors 'always', 'never', or 'auto'\n", out);
|
||||||
fprintf(out, "\
|
fputs( " --warning=PERC set warning alarming threshold\n", out);
|
||||||
--warning=PERC set warning alarming limit\n\
|
fputs( " --critical=PERC set critical alarming threshold\n", out);
|
||||||
--critical=PERC set critical alarming limit\n\
|
fputs( " --skip=WHAT do not print threshold 'ok', 'warning', 'critical',\n", out);
|
||||||
--minsize=size disable alarms for small ranges and shared-nets\n");
|
fputs( " 'minsize', or 'suppressed'\n", out);
|
||||||
fprintf(out, "\
|
fputs( " --warn-count=NR a number of free leases before warning raised\n", out);
|
||||||
-v, --version version information\n\
|
fputs( " --crit-count=NR a number of free leases before critical raised\n", out);
|
||||||
-h, --help this screen\n\
|
fputs( " --minsize=size disable alarms for small ranges and shared-nets\n", out);
|
||||||
\n\
|
fputs( " --snet-alarms suppress range alarms that are part of a shared-net\n", out);
|
||||||
Report bugs to <%s>\n\
|
fputs( " -p, --perfdata print additional perfdata in alarming mode\n", out);
|
||||||
Homepage: %s\n", PACKAGE_BUGREPORT, PACKAGE_URL);
|
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);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
1524
src/output.c
1524
src/output.c
File diff suppressed because it is too large
Load diff
160
src/sort.c
160
src/sort.c
|
|
@ -40,28 +40,30 @@
|
||||||
|
|
||||||
#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 "dhcpd-pools.h"
|
#include "error.h"
|
||||||
#include "progname.h"
|
#include "progname.h"
|
||||||
|
#include "quote.h"
|
||||||
|
#include "xalloc.h"
|
||||||
|
|
||||||
|
#include "dhcpd-pools.h"
|
||||||
|
|
||||||
/*! \brief Compare IP address, with IPv4/v6 determination.
|
/*! \brief Compare IP address, with IPv4/v6 determination.
|
||||||
* \param a Binary IP address.
|
* \param a Binary IP address.
|
||||||
* \param b 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.
|
* \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)),
|
int ipcomp_init(const union ipaddr_t *restrict a __attribute__ ((unused)),
|
||||||
const union ipaddr_t *restrict b __attribute__((unused)))
|
const union ipaddr_t *restrict b __attribute__ ((unused)))
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ipcomp_v4(const union ipaddr_t *restrict a,
|
int ipcomp_v4(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b)
|
||||||
const union ipaddr_t *restrict b)
|
|
||||||
{
|
{
|
||||||
if (a->v4 < b->v4)
|
if (a->v4 < b->v4)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
@ -70,12 +72,36 @@ int ipcomp_v4(const union ipaddr_t *restrict a,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ipcomp_v6(const union ipaddr_t *restrict a,
|
int ipcomp_v6(const union ipaddr_t *restrict a, const union ipaddr_t *restrict b)
|
||||||
const union ipaddr_t *restrict b)
|
|
||||||
{
|
{
|
||||||
return memcmp(&a->v6, &b->v6, sizeof(a->v6));
|
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.
|
/*! \brief Compare IP address in leases. Suitable for sorting range table.
|
||||||
* \param r1 A range structure.
|
* \param r1 A range structure.
|
||||||
* \param r2 A range structure.
|
* \param r2 A range structure.
|
||||||
|
|
@ -93,9 +119,14 @@ int rangecomp(const void *restrict r1, const void *restrict r2)
|
||||||
*/
|
*/
|
||||||
int comp_double(double f1, double f2)
|
int comp_double(double f1, double f2)
|
||||||
{
|
{
|
||||||
return f1 < f2 ? -1 : f1 > f2 ? 1 : 0;
|
if (isless(f1, f2))
|
||||||
|
return -1;
|
||||||
|
else if (isless(f2, f1))
|
||||||
|
return 1;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*! \brief Compare two range_t by their first_ip.
|
/*! \brief Compare two range_t by their first_ip.
|
||||||
* \param r1,r2 Pointers to data to compare.
|
* \param r1,r2 Pointers to data to compare.
|
||||||
* \return Like strcmp.
|
* \return Like strcmp.
|
||||||
|
|
@ -197,7 +228,7 @@ comparer_t field_selector(char c)
|
||||||
{
|
{
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 'n':
|
case 'n':
|
||||||
break;
|
return NULL;
|
||||||
case 'i':
|
case 'i':
|
||||||
return comp_ip;
|
return comp_ip;
|
||||||
case 'm':
|
case 'm':
|
||||||
|
|
@ -213,10 +244,11 @@ comparer_t field_selector(char c)
|
||||||
case 'e':
|
case 'e':
|
||||||
return comp_tcperc;
|
return comp_tcperc;
|
||||||
default:
|
default:
|
||||||
clean_up();
|
{
|
||||||
warnx("field_selector: unknown sort order `%c'", c);
|
char str[2] = { c, '\0' };
|
||||||
errx(EXIT_FAILURE, "Try `%s --help' for more information.",
|
error(EXIT_FAILURE, 0, "field_selector: unknown sort order: %s",
|
||||||
program_name);
|
quote(str));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
@ -226,85 +258,69 @@ comparer_t field_selector(char c)
|
||||||
* \param right The right side of the merge sort.
|
* \param right The right side of the merge sort.
|
||||||
* \return Relevant for merge sort decision.
|
* \return Relevant for merge sort decision.
|
||||||
*/
|
*/
|
||||||
static int merge(struct range_t *restrict left, struct range_t *restrict right)
|
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;
|
||||||
comparer_t comparer;
|
int ret;
|
||||||
int cmp;
|
|
||||||
|
|
||||||
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 comparer */
|
return (0);
|
||||||
comparer = field_selector(config.sort[i]);
|
if (ret < 0)
|
||||||
cmp = comparer(left, right);
|
|
||||||
/* If fields are equal use next sort method */
|
|
||||||
if (cmp == 0) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (cmp < 0) {
|
|
||||||
return (1);
|
return (1);
|
||||||
}
|
|
||||||
return (0);
|
|
||||||
}
|
}
|
||||||
|
/* this is reached if nothing was sorted */
|
||||||
/* If all comparers where equal */
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*! \brief Mergesort for range table.
|
/*! \brief Mergesort for range table.
|
||||||
* \param orig Pointer to range that is requested to be sorted.
|
|
||||||
* \param size Number of ranges to be sorted.
|
|
||||||
* \param temp Temporary memory space, needed when a values has to be
|
|
||||||
* flipped.
|
|
||||||
*/
|
*/
|
||||||
void mergesort_ranges(struct range_t *restrict orig, int size,
|
void mergesort_ranges(struct conf_t *state, struct range_t *restrict orig, unsigned int size,
|
||||||
struct range_t *restrict temp)
|
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 (merge((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 (merge((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++;
|
||||||
}
|
}
|
||||||
|
|
@ -313,11 +329,13 @@ void mergesort_ranges(struct range_t *restrict orig, int size,
|
||||||
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);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,5 @@
|
||||||
TESTS = \
|
TESTS = \
|
||||||
|
tests/alarm-count-option \
|
||||||
tests/alarm-critical \
|
tests/alarm-critical \
|
||||||
tests/alarm-critical-ranges \
|
tests/alarm-critical-ranges \
|
||||||
tests/alarm-critical-snets \
|
tests/alarm-critical-snets \
|
||||||
|
|
@ -8,15 +9,38 @@ TESTS = \
|
||||||
tests/alarm-warning \
|
tests/alarm-warning \
|
||||||
tests/alarm-warning-ranges \
|
tests/alarm-warning-ranges \
|
||||||
tests/alarm-warning-snets \
|
tests/alarm-warning-snets \
|
||||||
|
tests/shnet-alarm \
|
||||||
|
tests/big-small \
|
||||||
|
tests/binding-states \
|
||||||
tests/bootp \
|
tests/bootp \
|
||||||
|
tests/cidr-v4 \
|
||||||
|
tests/cidr-v6 \
|
||||||
tests/complete \
|
tests/complete \
|
||||||
|
tests/complete-perfdata \
|
||||||
tests/empty \
|
tests/empty \
|
||||||
|
tests/errors \
|
||||||
tests/full-json \
|
tests/full-json \
|
||||||
|
tests/full-xml \
|
||||||
|
tests/formats \
|
||||||
tests/leading0 \
|
tests/leading0 \
|
||||||
tests/one-ip \
|
tests/one-ip \
|
||||||
|
tests/one-line \
|
||||||
|
tests/parser \
|
||||||
|
tests/range4 \
|
||||||
|
tests/range6 \
|
||||||
tests/same-twice \
|
tests/same-twice \
|
||||||
|
tests/shufled \
|
||||||
tests/simple \
|
tests/simple \
|
||||||
tests/v6
|
tests/skip \
|
||||||
|
tests/sorts \
|
||||||
|
tests/statuses \
|
||||||
|
tests/v6 \
|
||||||
|
tests/v6-perfdata
|
||||||
|
|
||||||
|
if ENABLE_MUSTACH
|
||||||
|
TESTS += \
|
||||||
|
tests/mustach
|
||||||
|
endif
|
||||||
|
|
||||||
EXTRA_DIST += \
|
EXTRA_DIST += \
|
||||||
tests/confs \
|
tests/confs \
|
||||||
|
|
|
||||||
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 $?
|
||||||
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/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
|
||||||
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
|
||||||
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;
|
||||||
|
}
|
||||||
|
|
||||||
1
tests/confs/formats
Symbolic link
1
tests/confs/formats
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
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/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
|
||||||
1
tests/confs/statuses
Symbolic link
1
tests/confs/statuses
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
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
|
||||||
|
|
@ -1,2 +1,3 @@
|
||||||
CRITICAL: dhcpd-pools: Ranges; crit: 1 warn: 0 ok: 4 Shared nets; crit: 1 warn: 0 ok: 1
|
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
|
2
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,3 @@
|
||||||
CRITICAL: dhcpd-pools: Ranges; crit: 1 warn: 0 ok: 4
|
CRITICAL: dhcpd-pools: Ranges - crit: 1 warn: 0 ok: 4; | range_crit=1 range_warn=0 range_ok=4
|
||||||
|
|
||||||
2
|
2
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,2 @@
|
||||||
CRITICAL: dhcpd-pools: Shared nets; crit: 1 warn: 0 ok: 1
|
CRITICAL: dhcpd-pools: Shared nets - crit: 1 warn: 0 ok: 1; | snet_crit=1 snet_warn=0 snet_ok=1
|
||||||
2
|
2
|
||||||
|
|
|
||||||
|
|
@ -1 +1,2 @@
|
||||||
OK: Ranges; crit: 0 warn: 0 ok: 0 ignored: 1 Shared nets; crit: 0 warn: 0 ok: 0
|
OK: Ranges - crit: 0 warn: 0 ok: 0 ignored: 1; | range_crit=0 range_warn=0 range_ok=0 range_ignored=1
|
||||||
|
Shared nets - crit: 0 warn: 0 ok: 0; | snet_crit=0 snet_warn=0 snet_ok=0
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,3 @@
|
||||||
OK: Ranges; crit: 0 warn: 0 ok: 5 Shared nets; crit: 0 warn: 0 ok: 2
|
OK: Ranges - crit: 0 warn: 0 ok: 5; | range_crit=0 range_warn=0 range_ok=5
|
||||||
|
Shared nets - crit: 0 warn: 0 ok: 2; | snet_crit=0 snet_warn=0 snet_ok=2
|
||||||
0
|
0
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,2 @@
|
||||||
OK: Shared nets; crit: 0 warn: 0 ok: 2
|
OK: Shared nets - crit: 0 warn: 0 ok: 2; | snet_crit=0 snet_warn=0 snet_ok=2
|
||||||
0
|
0
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,3 @@
|
||||||
WARNING: dhcpd-pools: Ranges; crit: 0 warn: 1 ok: 4 Shared nets; crit: 0 warn: 1 ok: 1
|
WARNING: dhcpd-pools: Ranges - crit: 0 warn: 1 ok: 4; | range_crit=0 range_warn=1 range_ok=4
|
||||||
|
Shared nets - crit: 0 warn: 1 ok: 1; | snet_crit=0 snet_warn=1 snet_ok=1
|
||||||
1
|
1
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,3 @@
|
||||||
WARNING: dhcpd-pools: Ranges; crit: 0 warn: 1 ok: 4
|
WARNING: dhcpd-pools: Ranges - crit: 0 warn: 1 ok: 4; | range_crit=0 range_warn=1 range_ok=4
|
||||||
|
|
||||||
1
|
1
|
||||||
|
|
|
||||||
|
|
@ -1,2 +1,2 @@
|
||||||
WARNING: dhcpd-pools: Shared nets; crit: 0 warn: 1 ok: 1
|
WARNING: dhcpd-pools: Shared nets - crit: 0 warn: 1 ok: 1; | snet_crit=0 snet_warn=1 snet_ok=1
|
||||||
1
|
1
|
||||||
|
|
|
||||||
1
tests/expected/big-small
Symbolic link
1
tests/expected/big-small
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
simple
|
||||||
16
tests/expected/binding-states
Normal file
16
tests/expected/binding-states
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 4 20.000 3 7 35.000 2 10.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000 0 0.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000 0 0.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000 0 0.000
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000 0 0.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
example1 40 14 35.000 3 17 42.500 2 5.000
|
||||||
|
example2 40 17 42.500 0 17 42.500 0 0.000
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
All networks 100 36 36.000 3 39 39.000 2 2.000
|
||||||
|
|
@ -1,10 +0,0 @@
|
||||||
Ranges:
|
|
||||||
shared net name first ip last ip max cur percent touch t+c t+c perc
|
|
||||||
All networks 10.0.0.1 - 10.0.0.10 10 10 100.000 0 10 100.000
|
|
||||||
|
|
||||||
Shared networks:
|
|
||||||
name max cur percent touch t+c t+c perc
|
|
||||||
|
|
||||||
Sum of all ranges:
|
|
||||||
name max cur percent touch t+c t+c perc
|
|
||||||
All networks 10 10 100.000 0 10 100.000
|
|
||||||
1
tests/expected/bootp
Symbolic link
1
tests/expected/bootp
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
simple
|
||||||
16
tests/expected/cidr-v4
Normal file
16
tests/expected/cidr-v4
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
example1 10.0.0.1 - 10.0.0.31 31 11 35.484 0 11 35.484
|
||||||
|
example1 10.1.0.1 - 10.1.0.31 31 10 32.258 0 10 32.258
|
||||||
|
example2 10.2.0.1 - 10.2.0.31 31 8 25.806 0 8 25.806
|
||||||
|
example2 10.3.0.1 - 10.3.0.31 31 9 29.032 0 9 29.032
|
||||||
|
All networks 10.4.0.1 - 10.4.0.15 15 5 33.333 0 5 33.333
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 62 21 33.871 0 21 33.871
|
||||||
|
example2 62 17 27.419 0 17 27.419
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 139 43 30.935 0 43 30.935
|
||||||
11
tests/expected/cidr-v6
Normal file
11
tests/expected/cidr-v6
Normal file
|
|
@ -0,0 +1,11 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks dead:abba:1000::2 - dead:abba:1000::ff 254 0 0.000 0 0 0.000
|
||||||
|
All networks dead:abba:4000::2 - dead:abba:4000::ff 254 1 0.394 0 1 0.394
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 508 1 0.197 0 1 0.197
|
||||||
3
tests/expected/complete-perfdata
Normal file
3
tests/expected/complete-perfdata
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
OK: Ranges - crit: 0 warn: 0 ok: 5; | range_crit=0 range_warn=0 range_ok=5 10.4.0.1_r=5;16;18;0;20 10.4.0.1_rt=0 10.3.0.1_r=9;16;18;0;20 10.3.0.1_rt=0 10.2.0.1_r=8;16;18;0;20 10.2.0.1_rt=0 10.1.0.1_r=10;16;18;0;20 10.1.0.1_rt=0 10.0.0.1_r=11;16;18;0;20 10.0.0.1_rt=0
|
||||||
|
Shared nets - crit: 0 warn: 0 ok: 2; | snet_crit=0 snet_warn=0 snet_ok=2 'example1_s'=21;32;36;0;40 'example1_st'=0 'example2_s'=17;32;36;0;40 'example2_st'=0
|
||||||
|
|
||||||
|
|
@ -6,4 +6,4 @@ name max cur percent touch t+c t+c perc
|
||||||
|
|
||||||
Sum of all ranges:
|
Sum of all ranges:
|
||||||
name max cur percent touch t+c t+c perc
|
name max cur percent touch t+c t+c perc
|
||||||
All networks 0 0 -nan 0 0 -nan
|
All networks 0 0 nan 0 0 nan
|
||||||
|
|
|
||||||
27
tests/expected/errors
Normal file
27
tests/expected/errors
Normal file
|
|
@ -0,0 +1,27 @@
|
||||||
|
=== output mask
|
||||||
|
dhcpd-pools: return_limit: output mask '88' is illegal
|
||||||
|
=== unknown specifier
|
||||||
|
dhcpd-pools: unknown --skip specifier: okish
|
||||||
|
=== color mode
|
||||||
|
dhcpd-pools: unknown color mode: 'sometimes'
|
||||||
|
=== IPv5
|
||||||
|
dhcpd-pools: unknown --ip-version argument: 5
|
||||||
|
=== missing conf
|
||||||
|
dhcpd-pools: cannot open inlude: ./tests/confs/complete_NXFILE: No such file or directory
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
[1;34mAll networks 0 0 nan 0 0 nan[0m
|
||||||
|
=== missing leases
|
||||||
|
dhcpd-pools: parse_leases: ./tests/leases/complete_NXFILE: No such file or directory
|
||||||
|
=== html table
|
||||||
|
dhcpd-pools: html table only output format is deprecated
|
||||||
|
=== none existing format
|
||||||
|
dhcpd-pools: unknown output format: 'z'
|
||||||
|
=== broken percent input
|
||||||
|
dhcpd-pools: illegal argument: 'eighty'
|
||||||
284
tests/expected/formats
Normal file
284
tests/expected/formats
Normal file
|
|
@ -0,0 +1,284 @@
|
||||||
|
=== color text
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 100 43 43.000 0 43 43.000
|
||||||
|
=== html
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>ISC dhcpd dhcpd-pools output</title>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" type="text/css">
|
||||||
|
<link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/v/dt/dt-2.2.2/datatables.min.css">
|
||||||
|
<style type="text/css">
|
||||||
|
table.dhcpd-pools th { text-transform: capitalize }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="container">
|
||||||
|
<h2>ISC DHCPD status</h2>
|
||||||
|
<h3>Sum of all</h3>
|
||||||
|
<table id="a" class="dhcpd-pools order-column table table-hover" summary="all">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>name</th>
|
||||||
|
<th>max</th>
|
||||||
|
<th>cur</th>
|
||||||
|
<th>free</th>
|
||||||
|
<th>percent</th>
|
||||||
|
<th>touch</th>
|
||||||
|
<th>t+c</th>
|
||||||
|
<th>t+c perc</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>All networks</td>
|
||||||
|
<td>100</td>
|
||||||
|
<td>43</td>
|
||||||
|
<td>57</td>
|
||||||
|
<td>43.000</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>43</td>
|
||||||
|
<td>43.000</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<h3>Shared networks</h3>
|
||||||
|
<table id="s" class="dhcpd-pools order-column table table-hover" summary="snet">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>name</th>
|
||||||
|
<th>max</th>
|
||||||
|
<th>cur</th>
|
||||||
|
<th>free</th>
|
||||||
|
<th>percent</th>
|
||||||
|
<th>touch</th>
|
||||||
|
<th>t+c</th>
|
||||||
|
<th>t+c perc</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>example1</td>
|
||||||
|
<td>40</td>
|
||||||
|
<td>21</td>
|
||||||
|
<td>19</td>
|
||||||
|
<td>52.5</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>21</td>
|
||||||
|
<td>52.500</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>example2</td>
|
||||||
|
<td>40</td>
|
||||||
|
<td>17</td>
|
||||||
|
<td>23</td>
|
||||||
|
<td>42.5</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>17</td>
|
||||||
|
<td>42.500</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<h3>Ranges</h3>
|
||||||
|
<table id="r" class="dhcpd-pools order-column table table-hover" summary="ranges">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>shared net name</th>
|
||||||
|
<th>first ip</th>
|
||||||
|
<th>last ip</th>
|
||||||
|
<th>max</th>
|
||||||
|
<th>cur</th>
|
||||||
|
<th>free</th>
|
||||||
|
<th>percent</th>
|
||||||
|
<th>touch</th>
|
||||||
|
<th>t+c</th>
|
||||||
|
<th>t+c perc</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>example1</td>
|
||||||
|
<td>10.0.0.1</td>
|
||||||
|
<td>10.0.0.20</td>
|
||||||
|
<td>20</td>
|
||||||
|
<td>11</td>
|
||||||
|
<td>9</td>
|
||||||
|
<td>55</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>11</td>
|
||||||
|
<td>55.000</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>example1</td>
|
||||||
|
<td>10.1.0.1</td>
|
||||||
|
<td>10.1.0.20</td>
|
||||||
|
<td>20</td>
|
||||||
|
<td>10</td>
|
||||||
|
<td>10</td>
|
||||||
|
<td>50</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>10</td>
|
||||||
|
<td>50.000</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>example2</td>
|
||||||
|
<td>10.2.0.1</td>
|
||||||
|
<td>10.2.0.20</td>
|
||||||
|
<td>20</td>
|
||||||
|
<td>8</td>
|
||||||
|
<td>12</td>
|
||||||
|
<td>40</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>8</td>
|
||||||
|
<td>40.000</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>example2</td>
|
||||||
|
<td>10.3.0.1</td>
|
||||||
|
<td>10.3.0.20</td>
|
||||||
|
<td>20</td>
|
||||||
|
<td>9</td>
|
||||||
|
<td>11</td>
|
||||||
|
<td>45</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>9</td>
|
||||||
|
<td>45.000</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>All networks</td>
|
||||||
|
<td>10.4.0.1</td>
|
||||||
|
<td>10.4.0.20</td>
|
||||||
|
<td>20</td>
|
||||||
|
<td>5</td>
|
||||||
|
<td>15</td>
|
||||||
|
<td>25</td>
|
||||||
|
<td>0</td>
|
||||||
|
<td>5</td>
|
||||||
|
<td>25.000</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<br /><div class="well well-lg">
|
||||||
|
More info at <a href="https://dhcpd-pools.sourceforge.net/">https://dhcpd-pools.sourceforge.net/</a>
|
||||||
|
</small></div></div>
|
||||||
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/js/bootstrap.min.js" type="text/javascript"></script>
|
||||||
|
<script type="text/javascript" src="https://cdn.datatables.net/v/dt/dt-2.2.2/datatables.min.js"></script>
|
||||||
|
<script type="text/javascript" class="init">$(document).ready(function() { $('#s').DataTable({ "iDisplayLength": 50, "lengthMenu": [ [25, 50, 100, -1], [25, 50, 100, "All"] ], "order": [[ 4, "desc" ]] } ); } );</script>
|
||||||
|
<script type="text/javascript" class="init">$(document).ready(function() { $('#r').DataTable({ "iDisplayLength": 100, "lengthMenu": [ [25, 50, 100, -1], [25, 50, 100, "All"] ], "order": [[ 6, "desc" ]] } ); } );</script>
|
||||||
|
</body></html>
|
||||||
|
=== xml
|
||||||
|
<dhcpstatus>
|
||||||
|
<subnet>
|
||||||
|
<location>example1</location>
|
||||||
|
<range>10.0.0.1 - 10.0.0.20</range>
|
||||||
|
<defined>20</defined>
|
||||||
|
<used>11</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>9</free>
|
||||||
|
</subnet>
|
||||||
|
<subnet>
|
||||||
|
<location>example1</location>
|
||||||
|
<range>10.1.0.1 - 10.1.0.20</range>
|
||||||
|
<defined>20</defined>
|
||||||
|
<used>10</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>10</free>
|
||||||
|
</subnet>
|
||||||
|
<subnet>
|
||||||
|
<location>example2</location>
|
||||||
|
<range>10.2.0.1 - 10.2.0.20</range>
|
||||||
|
<defined>20</defined>
|
||||||
|
<used>8</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>12</free>
|
||||||
|
</subnet>
|
||||||
|
<subnet>
|
||||||
|
<location>example2</location>
|
||||||
|
<range>10.3.0.1 - 10.3.0.20</range>
|
||||||
|
<defined>20</defined>
|
||||||
|
<used>9</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>11</free>
|
||||||
|
</subnet>
|
||||||
|
<subnet>
|
||||||
|
<location>All networks</location>
|
||||||
|
<range>10.4.0.1 - 10.4.0.20</range>
|
||||||
|
<defined>20</defined>
|
||||||
|
<used>5</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>15</free>
|
||||||
|
</subnet>
|
||||||
|
<shared-network>
|
||||||
|
<location>example1</location>
|
||||||
|
<defined>40</defined>
|
||||||
|
<used>21</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>19</free>
|
||||||
|
</shared-network>
|
||||||
|
<shared-network>
|
||||||
|
<location>example2</location>
|
||||||
|
<defined>40</defined>
|
||||||
|
<used>17</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>23</free>
|
||||||
|
</shared-network>
|
||||||
|
<summary>
|
||||||
|
<location>All networks</location>
|
||||||
|
<defined>100</defined>
|
||||||
|
<used>43</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>57</free>
|
||||||
|
</summary>
|
||||||
|
</dhcpstatus>
|
||||||
|
=== csv
|
||||||
|
"Ranges:"
|
||||||
|
"shared net name","first ip","last ip","max","cur","percent","touch","t+c","t+c perc"
|
||||||
|
"example1","10.0.0.1","10.0.0.20","20","11","55.000","0","11","55.000"
|
||||||
|
"example1","10.1.0.1","10.1.0.20","20","10","50.000","0","10","50.000"
|
||||||
|
"example2","10.2.0.1","10.2.0.20","20","8","40.000","0","8","40.000"
|
||||||
|
"example2","10.3.0.1","10.3.0.20","20","9","45.000","0","9","45.000"
|
||||||
|
"All networks","10.4.0.1","10.4.0.20","20","5","25.000","0","5","25.000"
|
||||||
|
|
||||||
|
"Shared networks:"
|
||||||
|
"name","max","cur","percent","touch","t+c","t+c perc"
|
||||||
|
"example1","40","21","52.500","0","21","52.500"
|
||||||
|
"example2","40","17","42.500","0","17","42.500"
|
||||||
|
|
||||||
|
"Sum of all ranges:"
|
||||||
|
"name","max","cur","percent","touch","t+c","t+c perc"
|
||||||
|
"All networks","100","43","43.000","0","43","43.000"
|
||||||
|
=== json
|
||||||
|
{
|
||||||
|
"subnets": [
|
||||||
|
{ "location":"example1", "range":"10.0.0.1 - 10.0.0.20", "first_ip":"10.0.0.1", "last_ip":"10.0.0.20", "defined":20, "used":11, "touched":0, "free":9, "percent":55, "touch_count":11, "touch_percent":55, "status":0 },
|
||||||
|
{ "location":"example1", "range":"10.1.0.1 - 10.1.0.20", "first_ip":"10.1.0.1", "last_ip":"10.1.0.20", "defined":20, "used":10, "touched":0, "free":10, "percent":50, "touch_count":10, "touch_percent":50, "status":0 },
|
||||||
|
{ "location":"example2", "range":"10.2.0.1 - 10.2.0.20", "first_ip":"10.2.0.1", "last_ip":"10.2.0.20", "defined":20, "used":8, "touched":0, "free":12, "percent":40, "touch_count":8, "touch_percent":40, "status":0 },
|
||||||
|
{ "location":"example2", "range":"10.3.0.1 - 10.3.0.20", "first_ip":"10.3.0.1", "last_ip":"10.3.0.20", "defined":20, "used":9, "touched":0, "free":11, "percent":45, "touch_count":9, "touch_percent":45, "status":0 },
|
||||||
|
{ "location":"All networks", "range":"10.4.0.1 - 10.4.0.20", "first_ip":"10.4.0.1", "last_ip":"10.4.0.20", "defined":20, "used":5, "touched":0, "free":15, "percent":25, "touch_count":5, "touch_percent":25, "status":0 }
|
||||||
|
],
|
||||||
|
"shared-networks": [
|
||||||
|
{ "location":"example1", "defined":40, "used":21, "touched":0, "free":19, "percent":52.5, "touch_count":21, "touch_percent":52.5, "status":0 },
|
||||||
|
{ "location":"example2", "defined":40, "used":17, "touched":0, "free":23, "percent":42.5, "touch_count":17, "touch_percent":42.5, "status":0 }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
=== perfdata
|
||||||
|
CRITICAL: dhcpd-pools: Ranges - crit: 3 warn: 2 ok: 0; | range_crit=3 range_warn=2 range_ok=0 10.4.0.1_r=5;4;8;0;20 10.4.0.1_rt=0 10.3.0.1_r=9;4;8;0;20 10.3.0.1_rt=0 10.2.0.1_r=8;4;8;0;20 10.2.0.1_rt=0 10.1.0.1_r=10;4;8;0;20 10.1.0.1_rt=0 10.0.0.1_r=11;4;8;0;20 10.0.0.1_rt=0
|
||||||
|
Shared nets - crit: 2 warn: 0 ok: 0; | snet_crit=2 snet_warn=0 snet_ok=0 'example1_s'=21;8;16;0;40 'example1_st'=0 'example2_s'=17;8;16;0;40 'example2_st'=0
|
||||||
|
|
||||||
317
tests/expected/mustach
Normal file
317
tests/expected/mustach
Normal file
|
|
@ -0,0 +1,317 @@
|
||||||
|
Ethernets:
|
||||||
|
macaddress: 00:00:00:00:00:01 ip: 10.0.0.1
|
||||||
|
macaddress: 00:00:00:00:00:02 ip: 10.0.0.2
|
||||||
|
macaddress: 00:00:00:00:00:03 ip: 10.0.0.3
|
||||||
|
macaddress: 00:00:00:00:00:04 ip: 10.0.0.4
|
||||||
|
macaddress: 00:00:00:00:00:05 ip: 10.0.0.5
|
||||||
|
macaddress: 00:00:00:00:00:06 ip: 10.0.0.6
|
||||||
|
macaddress: 00:00:00:00:00:07 ip: 10.0.0.7
|
||||||
|
macaddress: 00:00:00:00:00:08 ip: 10.0.0.8
|
||||||
|
macaddress: 00:00:00:00:00:09 ip: 10.0.0.9
|
||||||
|
macaddress: 00:00:00:00:00:10 ip: 10.0.0.10
|
||||||
|
macaddress: 00:00:00:00:00:11 ip: 10.0.0.11
|
||||||
|
macaddress: 00:00:00:00:00:12 ip: 10.0.0.12
|
||||||
|
macaddress: 00:00:00:00:01:00 ip: 10.1.0.0
|
||||||
|
macaddress: 00:00:00:00:01:01 ip: 10.1.0.1
|
||||||
|
macaddress: 00:00:00:00:01:02 ip: 10.1.0.2
|
||||||
|
macaddress: 00:00:00:00:01:03 ip: 10.1.0.3
|
||||||
|
macaddress: 00:00:00:00:01:04 ip: 10.1.0.4
|
||||||
|
macaddress: 00:00:00:00:01:05 ip: 10.1.0.5
|
||||||
|
macaddress: 00:00:00:00:01:06 ip: 10.1.0.6
|
||||||
|
macaddress: 00:00:00:00:01:07 ip: 10.1.0.7
|
||||||
|
macaddress: 00:00:00:00:01:08 ip: 10.1.0.8
|
||||||
|
macaddress: 00:00:00:00:01:09 ip: 10.1.0.9
|
||||||
|
macaddress: 00:00:00:00:01:10 ip: 10.1.0.10
|
||||||
|
macaddress: 00:00:00:00:02:00 ip: 10.2.0.0
|
||||||
|
macaddress: 00:00:00:00:02:01 ip: 10.2.0.1
|
||||||
|
macaddress: 00:00:00:00:02:02 ip: 10.2.0.2
|
||||||
|
macaddress: 00:00:00:00:02:03 ip: 10.2.0.3
|
||||||
|
macaddress: 00:00:00:00:02:04 ip: 10.2.0.4
|
||||||
|
macaddress: 00:00:00:00:02:05 ip: 10.2.0.5
|
||||||
|
macaddress: 00:00:00:00:02:06 ip: 10.2.0.6
|
||||||
|
macaddress: 00:00:00:00:02:07 ip: 10.2.0.7
|
||||||
|
macaddress: 00:00:00:00:02:08 ip: 10.2.0.8
|
||||||
|
macaddress: 00:00:00:00:03:00 ip: 10.3.0.0
|
||||||
|
macaddress: 00:00:00:00:03:01 ip: 10.3.0.1
|
||||||
|
macaddress: 00:00:00:00:03:02 ip: 10.3.0.2
|
||||||
|
macaddress: 00:00:00:00:03:03 ip: 10.3.0.3
|
||||||
|
macaddress: 00:00:00:00:03:04 ip: 10.3.0.4
|
||||||
|
macaddress: 00:00:00:00:03:05 ip: 10.3.0.5
|
||||||
|
macaddress: 00:00:00:00:03:06 ip: 10.3.0.6
|
||||||
|
macaddress: 00:00:00:00:03:07 ip: 10.3.0.7
|
||||||
|
macaddress: 00:00:00:00:03:08 ip: 10.3.0.8
|
||||||
|
macaddress: 00:00:00:00:03:09 ip: 10.3.0.9
|
||||||
|
macaddress: 00:00:00:00:04:00 ip: 10.4.0.0
|
||||||
|
macaddress: 00:00:00:00:04:01 ip: 10.4.0.1
|
||||||
|
macaddress: 00:00:00:00:04:02 ip: 10.4.0.2
|
||||||
|
macaddress: 00:00:00:00:04:03 ip: 10.4.0.3
|
||||||
|
macaddress: 00:00:00:00:04:04 ip: 10.4.0.4
|
||||||
|
macaddress: 00:00:00:00:04:05 ip: 10.4.0.5
|
||||||
|
macaddress: 00:00:00:00:04:06 ip: 10.4.0.6
|
||||||
|
|
||||||
|
Subnets:
|
||||||
|
location: example1
|
||||||
|
range: 10.0.0.1 - 10.0.0.20
|
||||||
|
first_ip: 10.0.0.1
|
||||||
|
last_ip: 10.0.0.20
|
||||||
|
used: 11
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 9
|
||||||
|
percent: 55
|
||||||
|
touch_count: 11
|
||||||
|
touch_percent: 55
|
||||||
|
backup_count: 1
|
||||||
|
backup_percent: 5
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
location: example1
|
||||||
|
range: 10.1.0.1 - 10.1.0.20
|
||||||
|
first_ip: 10.1.0.1
|
||||||
|
last_ip: 10.1.0.20
|
||||||
|
used: 10
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 10
|
||||||
|
percent: 50
|
||||||
|
touch_count: 10
|
||||||
|
touch_percent: 50
|
||||||
|
backup_count: 0
|
||||||
|
backup_percent: 0
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
location: example2
|
||||||
|
range: 10.2.0.1 - 10.2.0.20
|
||||||
|
first_ip: 10.2.0.1
|
||||||
|
last_ip: 10.2.0.20
|
||||||
|
used: 8
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 12
|
||||||
|
percent: 40
|
||||||
|
touch_count: 8
|
||||||
|
touch_percent: 40
|
||||||
|
backup_count: 0
|
||||||
|
backup_percent: 0
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
location: example2
|
||||||
|
range: 10.3.0.1 - 10.3.0.20
|
||||||
|
first_ip: 10.3.0.1
|
||||||
|
last_ip: 10.3.0.20
|
||||||
|
used: 9
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 11
|
||||||
|
percent: 45
|
||||||
|
touch_count: 9
|
||||||
|
touch_percent: 45
|
||||||
|
backup_count: 0
|
||||||
|
backup_percent: 0
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
location: All networks
|
||||||
|
range: 10.4.0.1 - 10.4.0.20
|
||||||
|
first_ip: 10.4.0.1
|
||||||
|
last_ip: 10.4.0.20
|
||||||
|
used: 5
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 15
|
||||||
|
percent: 25
|
||||||
|
touch_count: 5
|
||||||
|
touch_percent: 25
|
||||||
|
backup_count: 1
|
||||||
|
backup_percent: 5
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
|
||||||
|
Shared-networks:
|
||||||
|
location: example1
|
||||||
|
defined: 40
|
||||||
|
used: 21
|
||||||
|
touched: 0
|
||||||
|
free: 19
|
||||||
|
percent: 52.5
|
||||||
|
touch_count: 21
|
||||||
|
touch_percent: 52.5
|
||||||
|
backup_count: 1
|
||||||
|
backup_percent: 2.5
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
location: example2
|
||||||
|
defined: 40
|
||||||
|
used: 17
|
||||||
|
touched: 0
|
||||||
|
free: 23
|
||||||
|
percent: 42.5
|
||||||
|
touch_count: 17
|
||||||
|
touch_percent: 42.5
|
||||||
|
backup_count: 0
|
||||||
|
backup_percent: 0
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
|
||||||
|
Summary:
|
||||||
|
location: All networks
|
||||||
|
defined: 100
|
||||||
|
used: 43
|
||||||
|
touched: 0
|
||||||
|
free: 57
|
||||||
|
percent: 43
|
||||||
|
touch_count: 43
|
||||||
|
touch_percent: 43
|
||||||
|
backup_count: 2
|
||||||
|
backup_percent: 2
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
|
||||||
|
number_of_ranges: 5
|
||||||
|
number_of_shared_networks: 2
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
number_of_ranges_warning: 0
|
||||||
|
number_of_ranges_critical: 0
|
||||||
|
number_of_shared_networks: 2
|
||||||
|
number_of_shared_networks_warning: 0
|
||||||
|
number_of_shared_networks_critical: 0
|
||||||
|
--- skip ok ---
|
||||||
|
Ethernets:
|
||||||
|
macaddress: 00:00:00:00:00:01 ip: 10.0.0.1
|
||||||
|
macaddress: 00:00:00:00:00:02 ip: 10.0.0.2
|
||||||
|
macaddress: 00:00:00:00:00:03 ip: 10.0.0.3
|
||||||
|
macaddress: 00:00:00:00:00:04 ip: 10.0.0.4
|
||||||
|
macaddress: 00:00:00:00:00:05 ip: 10.0.0.5
|
||||||
|
macaddress: 00:00:00:00:00:06 ip: 10.0.0.6
|
||||||
|
macaddress: 00:00:00:00:00:07 ip: 10.0.0.7
|
||||||
|
macaddress: 00:00:00:00:00:08 ip: 10.0.0.8
|
||||||
|
macaddress: 00:00:00:00:00:09 ip: 10.0.0.9
|
||||||
|
macaddress: 00:00:00:00:00:10 ip: 10.0.0.10
|
||||||
|
macaddress: 00:00:00:00:00:11 ip: 10.0.0.11
|
||||||
|
macaddress: 00:00:00:00:00:12 ip: 10.0.0.12
|
||||||
|
macaddress: 00:00:00:00:01:00 ip: 10.1.0.0
|
||||||
|
macaddress: 00:00:00:00:01:01 ip: 10.1.0.1
|
||||||
|
macaddress: 00:00:00:00:01:02 ip: 10.1.0.2
|
||||||
|
macaddress: 00:00:00:00:01:03 ip: 10.1.0.3
|
||||||
|
macaddress: 00:00:00:00:01:04 ip: 10.1.0.4
|
||||||
|
macaddress: 00:00:00:00:01:05 ip: 10.1.0.5
|
||||||
|
macaddress: 00:00:00:00:01:06 ip: 10.1.0.6
|
||||||
|
macaddress: 00:00:00:00:01:07 ip: 10.1.0.7
|
||||||
|
macaddress: 00:00:00:00:01:08 ip: 10.1.0.8
|
||||||
|
macaddress: 00:00:00:00:01:09 ip: 10.1.0.9
|
||||||
|
macaddress: 00:00:00:00:01:10 ip: 10.1.0.10
|
||||||
|
macaddress: 00:00:00:00:02:00 ip: 10.2.0.0
|
||||||
|
macaddress: 00:00:00:00:02:01 ip: 10.2.0.1
|
||||||
|
macaddress: 00:00:00:00:02:02 ip: 10.2.0.2
|
||||||
|
macaddress: 00:00:00:00:02:03 ip: 10.2.0.3
|
||||||
|
macaddress: 00:00:00:00:02:04 ip: 10.2.0.4
|
||||||
|
macaddress: 00:00:00:00:02:05 ip: 10.2.0.5
|
||||||
|
macaddress: 00:00:00:00:02:06 ip: 10.2.0.6
|
||||||
|
macaddress: 00:00:00:00:02:07 ip: 10.2.0.7
|
||||||
|
macaddress: 00:00:00:00:02:08 ip: 10.2.0.8
|
||||||
|
macaddress: 00:00:00:00:03:00 ip: 10.3.0.0
|
||||||
|
macaddress: 00:00:00:00:03:01 ip: 10.3.0.1
|
||||||
|
macaddress: 00:00:00:00:03:02 ip: 10.3.0.2
|
||||||
|
macaddress: 00:00:00:00:03:03 ip: 10.3.0.3
|
||||||
|
macaddress: 00:00:00:00:03:04 ip: 10.3.0.4
|
||||||
|
macaddress: 00:00:00:00:03:05 ip: 10.3.0.5
|
||||||
|
macaddress: 00:00:00:00:03:06 ip: 10.3.0.6
|
||||||
|
macaddress: 00:00:00:00:03:07 ip: 10.3.0.7
|
||||||
|
macaddress: 00:00:00:00:03:08 ip: 10.3.0.8
|
||||||
|
macaddress: 00:00:00:00:03:09 ip: 10.3.0.9
|
||||||
|
macaddress: 00:00:00:00:04:00 ip: 10.4.0.0
|
||||||
|
macaddress: 00:00:00:00:04:01 ip: 10.4.0.1
|
||||||
|
macaddress: 00:00:00:00:04:02 ip: 10.4.0.2
|
||||||
|
macaddress: 00:00:00:00:04:03 ip: 10.4.0.3
|
||||||
|
macaddress: 00:00:00:00:04:04 ip: 10.4.0.4
|
||||||
|
macaddress: 00:00:00:00:04:05 ip: 10.4.0.5
|
||||||
|
macaddress: 00:00:00:00:04:06 ip: 10.4.0.6
|
||||||
|
|
||||||
|
Subnets:
|
||||||
|
location: example1
|
||||||
|
range: 10.0.0.1 - 10.0.0.20
|
||||||
|
first_ip: 10.0.0.1
|
||||||
|
last_ip: 10.0.0.20
|
||||||
|
used: 11
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 9
|
||||||
|
percent: 55
|
||||||
|
touch_count: 11
|
||||||
|
touch_percent: 55
|
||||||
|
backup_count: 1
|
||||||
|
backup_percent: 5
|
||||||
|
status: 1
|
||||||
|
|
||||||
|
location: example1
|
||||||
|
range: 10.1.0.1 - 10.1.0.20
|
||||||
|
first_ip: 10.1.0.1
|
||||||
|
last_ip: 10.1.0.20
|
||||||
|
used: 10
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 10
|
||||||
|
percent: 50
|
||||||
|
touch_count: 10
|
||||||
|
touch_percent: 50
|
||||||
|
backup_count: 0
|
||||||
|
backup_percent: 0
|
||||||
|
status: 1
|
||||||
|
|
||||||
|
location: example2
|
||||||
|
range: 10.3.0.1 - 10.3.0.20
|
||||||
|
first_ip: 10.3.0.1
|
||||||
|
last_ip: 10.3.0.20
|
||||||
|
used: 9
|
||||||
|
touched: 0
|
||||||
|
defined: 20
|
||||||
|
free: 11
|
||||||
|
percent: 45
|
||||||
|
touch_count: 9
|
||||||
|
touch_percent: 45
|
||||||
|
backup_count: 0
|
||||||
|
backup_percent: 0
|
||||||
|
status: 1
|
||||||
|
|
||||||
|
|
||||||
|
Shared-networks:
|
||||||
|
location: example1
|
||||||
|
defined: 40
|
||||||
|
used: 21
|
||||||
|
touched: 0
|
||||||
|
free: 19
|
||||||
|
percent: 52.5
|
||||||
|
touch_count: 21
|
||||||
|
touch_percent: 52.5
|
||||||
|
backup_count: 1
|
||||||
|
backup_percent: 2.5
|
||||||
|
status: 1
|
||||||
|
|
||||||
|
|
||||||
|
Summary:
|
||||||
|
location: All networks
|
||||||
|
defined: 100
|
||||||
|
used: 43
|
||||||
|
touched: 0
|
||||||
|
free: 57
|
||||||
|
percent: 43
|
||||||
|
touch_count: 43
|
||||||
|
touch_percent: 43
|
||||||
|
backup_count: 2
|
||||||
|
backup_percent: 2
|
||||||
|
status: 0
|
||||||
|
|
||||||
|
|
||||||
|
number_of_ranges: 5
|
||||||
|
number_of_shared_networks: 2
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
number_of_ranges_warning: 3
|
||||||
|
number_of_ranges_critical: 0
|
||||||
|
number_of_shared_networks: 2
|
||||||
|
number_of_shared_networks_warning: 1
|
||||||
|
number_of_shared_networks_critical: 0
|
||||||
11
tests/expected/one-line
Normal file
11
tests/expected/one-line
Normal file
|
|
@ -0,0 +1,11 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.0.0.1 - 10.0.0.5 5 5 100.000 0 5 100.000
|
||||||
|
All networks 10.0.0.6 - 10.0.0.10 5 5 100.000 0 5 100.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 10 10 100.000 0 10 100.000
|
||||||
1
tests/expected/parser
Symbolic link
1
tests/expected/parser
Symbolic link
|
|
@ -0,0 +1 @@
|
||||||
|
complete
|
||||||
16
tests/expected/range4
Normal file
16
tests/expected/range4
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
example1 10.0.0.0 - 10.0.0.31 32 12 37.500 0 12 37.500
|
||||||
|
example1 10.1.0.1 - 10.1.0.31 31 10 32.258 0 10 32.258
|
||||||
|
example2 10.2.0.1 - 10.2.0.15 15 8 53.333 0 8 53.333
|
||||||
|
example2 10.3.0.1 - 10.3.0.7 7 7 100.000 0 7 100.000
|
||||||
|
All networks 10.4.0.1 - 10.4.0.1 1 1 100.000 0 1 100.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 63 22 34.921 0 22 34.921
|
||||||
|
example2 22 15 68.182 0 15 68.182
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 86 38 44.186 0 38 44.186
|
||||||
10
tests/expected/range6
Normal file
10
tests/expected/range6
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks dead:abba:1000:: - dead:abba:1000:ff:ffff:ffff:ffff:ffff 4.72237e+21 2 0.000 1 3 0.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 4.72237e+21 2 0.000 1 3 0.000
|
||||||
|
|
@ -1,9 +1,9 @@
|
||||||
{
|
{
|
||||||
"active_leases": [
|
"active_leases": [
|
||||||
{ "ip":"10.0.0.5", "macaddress":"00:00:00:00:00:00" }
|
{ "ip":"10.0.0.5", "macaddress":"00:00:00:00:00:00", "starts":"", "ends":"", "hostname":"" }
|
||||||
],
|
],
|
||||||
"subnets": [
|
"subnets": [
|
||||||
{ "location":"All networks", "range":"10.0.0.1 - 10.0.0.10", "defined":10, "used":1, "free":9 }
|
{ "location":"All networks", "range":"10.0.0.1 - 10.0.0.10", "first_ip":"10.0.0.1", "last_ip":"10.0.0.10", "defined":10, "used":1, "touched":0, "free":9, "percent":10, "touch_count":1, "touch_percent":10, "status":0 }
|
||||||
],
|
],
|
||||||
"shared-networks": [
|
"shared-networks": [
|
||||||
],
|
],
|
||||||
|
|
@ -11,6 +11,13 @@
|
||||||
"location":"All networks",
|
"location":"All networks",
|
||||||
"defined":10,
|
"defined":10,
|
||||||
"used":1,
|
"used":1,
|
||||||
"free":9
|
"touched":0,
|
||||||
|
"free":9,
|
||||||
|
"percent":10,
|
||||||
|
"touch_count":1,
|
||||||
|
"touch_percent":10,
|
||||||
|
"status":0
|
||||||
|
},
|
||||||
|
"trivia": {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
24
tests/expected/same-twice-xml
Normal file
24
tests/expected/same-twice-xml
Normal file
|
|
@ -0,0 +1,24 @@
|
||||||
|
<dhcpstatus>
|
||||||
|
<active_lease>
|
||||||
|
<ip>10.0.0.5</ip>
|
||||||
|
<macaddress>00:00:00:00:00:00</macaddress>
|
||||||
|
<starts></starts>
|
||||||
|
<ends></ends>
|
||||||
|
<hostname></hostname>
|
||||||
|
</active_lease>
|
||||||
|
<subnet>
|
||||||
|
<location>All networks</location>
|
||||||
|
<range>10.0.0.1 - 10.0.0.10</range>
|
||||||
|
<defined>10</defined>
|
||||||
|
<used>1</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>9</free>
|
||||||
|
</subnet>
|
||||||
|
<summary>
|
||||||
|
<location>All networks</location>
|
||||||
|
<defined>10</defined>
|
||||||
|
<used>1</used>
|
||||||
|
<touched>0</touched>
|
||||||
|
<free>9</free>
|
||||||
|
</summary>
|
||||||
|
</dhcpstatus>
|
||||||
3
tests/expected/shnet-alarm
Normal file
3
tests/expected/shnet-alarm
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
CRITICAL: dhcpd-pools: Ranges - crit: 2 warn: 0 ok: 3; | range_crit=2 range_warn=0 range_ok=3
|
||||||
|
Shared nets - crit: 1 warn: 0 ok: 1; | snet_crit=1 snet_warn=0 snet_ok=1
|
||||||
|
2
|
||||||
16
tests/expected/shufled
Normal file
16
tests/expected/shufled
Normal file
|
|
@ -0,0 +1,16 @@
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 3 15.000 0 3 15.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 4 20.000 0 4 20.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 4 20.000 0 4 20.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 3 15.000 0 3 15.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 13 32.500 0 13 32.500
|
||||||
|
example2 40 7 17.500 0 7 17.500
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
All networks 100 23 23.000 0 23 23.000
|
||||||
80
tests/expected/skip
Normal file
80
tests/expected/skip
Normal file
|
|
@ -0,0 +1,80 @@
|
||||||
|
--- skip ok ---
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;33mexample1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000 1 5.000[0m
|
||||||
|
[1;33mexample1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000 0 0.000[0m
|
||||||
|
[1;33mexample2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000 0 0.000[0m
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;33mexample1 40 21 52.500 0 21 52.500 1 2.500[0m
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
All networks 100 43 43.000 0 43 43.000 2 2.000
|
||||||
|
--- skip warning ---
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mexample1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000 1 5.000[0m
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000 0 0.000
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000 1 5.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mexample1 40 21 52.500 0 21 52.500 1 2.500[0m
|
||||||
|
example2 40 17 42.500 0 17 42.500 0 0.000
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
All networks 100 43 43.000 0 43 43.000 2 2.000
|
||||||
|
--- skip critical ok ---
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;33mexample1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000 0 0.000[0m
|
||||||
|
[1;33mexample2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000 0 0.000[0m
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
All networks 100 43 43.000 0 43 43.000 2 2.000
|
||||||
|
--- skip suppressed ---
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mAll networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000 1 5.000[0m
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mexample1 40 21 52.500 0 21 52.500 1 2.500[0m
|
||||||
|
[1;31mexample2 40 17 42.500 0 17 42.500 0 0.000[0m
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mAll networks 100 43 43.000 0 43 43.000 2 2.000[0m
|
||||||
|
--- skip minsize ---
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mexample1 40 21 52.500 0 21 52.500 1 2.500[0m
|
||||||
|
[1;31mexample2 40 17 42.500 0 17 42.500 0 0.000[0m
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mAll networks 100 43 43.000 0 43 43.000 2 2.000[0m
|
||||||
|
--- skip count ok ---
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
[1;31mexample1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000 1 5.000[0m
|
||||||
|
[1;31mexample1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000 0 0.000[0m
|
||||||
|
[1;33mexample2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000 0 0.000[0m
|
||||||
|
[1;31mexample2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000 0 0.000[0m
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
|
||||||
|
Sum of all ranges:
|
||||||
|
name max cur percent touch t+c t+c perc bu bu perc
|
||||||
|
All networks 100 43 43.000 0 43 43.000 2 2.000
|
||||||
168
tests/expected/sorts
Normal file
168
tests/expected/sorts
Normal file
|
|
@ -0,0 +1,168 @@
|
||||||
|
== name ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== ip ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== maximum ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== current ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
1
|
||||||
|
== percent ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
1
|
||||||
|
1
|
||||||
|
== touched ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== t+c ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== t+c perc ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== reverse ==
|
||||||
|
Ranges:
|
||||||
|
shared net name first ip last ip max cur percent touch t+c t+c perc
|
||||||
|
All networks 10.4.0.1 - 10.4.0.20 20 5 25.000 0 5 25.000
|
||||||
|
example2 10.3.0.1 - 10.3.0.20 20 9 45.000 0 9 45.000
|
||||||
|
example2 10.2.0.1 - 10.2.0.20 20 8 40.000 0 8 40.000
|
||||||
|
example1 10.1.0.1 - 10.1.0.20 20 10 50.000 0 10 50.000
|
||||||
|
example1 10.0.0.1 - 10.0.0.20 20 11 55.000 0 11 55.000
|
||||||
|
|
||||||
|
Shared networks:
|
||||||
|
name max cur percent touch t+c t+c perc
|
||||||
|
example1 40 21 52.500 0 21 52.500
|
||||||
|
example2 40 17 42.500 0 17 42.500
|
||||||
|
|
||||||
|
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
|
||||||
|
== broken ==
|
||||||
|
dhcpd-pools: field_selector: unknown sort order: 'x'
|
||||||
|
1
|
||||||
29
tests/expected/statuses
Normal file
29
tests/expected/statuses
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
{
|
||||||
|
"subnets": [
|
||||||
|
{ "location":"example1", "range":"10.0.0.1 - 10.0.0.20", "first_ip":"10.0.0.1", "last_ip":"10.0.0.20", "defined":20, "used":11, "touched":0, "free":9, "percent":55, "touch_count":11, "touch_percent":55, "status":4 },
|
||||||
|
{ "location":"example1", "range":"10.1.0.1 - 10.1.0.20", "first_ip":"10.1.0.1", "last_ip":"10.1.0.20", "defined":20, "used":10, "touched":0, "free":10, "percent":50, "touch_count":10, "touch_percent":50, "status":4 },
|
||||||
|
{ "location":"example2", "range":"10.2.0.1 - 10.2.0.20", "first_ip":"10.2.0.1", "last_ip":"10.2.0.20", "defined":20, "used":8, "touched":0, "free":12, "percent":40, "touch_count":8, "touch_percent":40, "status":4 },
|
||||||
|
{ "location":"example2", "range":"10.3.0.1 - 10.3.0.20", "first_ip":"10.3.0.1", "last_ip":"10.3.0.20", "defined":20, "used":9, "touched":0, "free":11, "percent":45, "touch_count":9, "touch_percent":45, "status":4 },
|
||||||
|
{ "location":"10.4.0.0/24", "range":"10.4.0.1 - 10.4.0.20", "first_ip":"10.4.0.1", "last_ip":"10.4.0.20", "defined":20, "used":5, "touched":0, "free":15, "percent":25, "touch_count":5, "touch_percent":25, "status":4 }
|
||||||
|
],
|
||||||
|
"shared-networks": [
|
||||||
|
{ "location":"example1", "defined":40, "used":21, "touched":0, "free":19, "percent":52.5, "touch_count":21, "touch_percent":52.5, "status":2 },
|
||||||
|
{ "location":"example2", "defined":40, "used":17, "touched":0, "free":23, "percent":42.5, "touch_count":17, "touch_percent":42.5, "status":2 },
|
||||||
|
{ "location":"10.4.0.0/24", "defined":20, "used":5, "touched":0, "free":15, "percent":25, "touch_count":5, "touch_percent":25, "status":1 }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
== minsize
|
||||||
|
{
|
||||||
|
"subnets": [
|
||||||
|
{ "location":"example1", "range":"10.0.0.1 - 10.0.0.20", "first_ip":"10.0.0.1", "last_ip":"10.0.0.20", "defined":20, "used":11, "touched":0, "free":9, "percent":55, "touch_count":11, "touch_percent":55, "status":3 },
|
||||||
|
{ "location":"example1", "range":"10.1.0.1 - 10.1.0.20", "first_ip":"10.1.0.1", "last_ip":"10.1.0.20", "defined":20, "used":10, "touched":0, "free":10, "percent":50, "touch_count":10, "touch_percent":50, "status":3 },
|
||||||
|
{ "location":"example2", "range":"10.2.0.1 - 10.2.0.20", "first_ip":"10.2.0.1", "last_ip":"10.2.0.20", "defined":20, "used":8, "touched":0, "free":12, "percent":40, "touch_count":8, "touch_percent":40, "status":3 },
|
||||||
|
{ "location":"example2", "range":"10.3.0.1 - 10.3.0.20", "first_ip":"10.3.0.1", "last_ip":"10.3.0.20", "defined":20, "used":9, "touched":0, "free":11, "percent":45, "touch_count":9, "touch_percent":45, "status":3 },
|
||||||
|
{ "location":"10.4.0.0/24", "range":"10.4.0.1 - 10.4.0.20", "first_ip":"10.4.0.1", "last_ip":"10.4.0.20", "defined":20, "used":5, "touched":0, "free":15, "percent":25, "touch_count":5, "touch_percent":25, "status":3 }
|
||||||
|
],
|
||||||
|
"shared-networks": [
|
||||||
|
{ "location":"example1", "defined":40, "used":21, "touched":0, "free":19, "percent":52.5, "touch_count":21, "touch_percent":52.5, "status":2 },
|
||||||
|
{ "location":"example2", "defined":40, "used":17, "touched":0, "free":23, "percent":42.5, "touch_count":17, "touch_percent":42.5, "status":2 },
|
||||||
|
{ "location":"10.4.0.0/24", "defined":20, "used":5, "touched":0, "free":15, "percent":25, "touch_count":5, "touch_percent":25, "status":3 }
|
||||||
|
]
|
||||||
|
}
|
||||||
3
tests/expected/v6-perfdata
Normal file
3
tests/expected/v6-perfdata
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
OK: Ranges - crit: 0 warn: 0 ok: 2; | range_crit=0 range_warn=0 range_ok=2 dead:abba:4000::2_r=1;203.2;228.6;0;254 dead:abba:4000::2_rt=0 dead:abba:1000::2_r=2;3.77789e+21;4.25013e+21;0;4.72237e+21 dead:abba:1000::2_rt=1
|
||||||
|
Shared nets - crit: 0 warn: 0 ok: 0; | snet_crit=0 snet_warn=0 snet_ok=0
|
||||||
|
|
||||||
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