2 # - Indent with tabs only.
3 # - Keep files sorted; one line per file.
4 # - Directories in lib/ must have their own *dir variable (see hypervisor).
5 # - All directories must be listed DIRS.
6 # - Use autogen.sh to generate Makefile.in and configure script.
8 # Automake doesn't export these variables before version 1.10.
9 abs_top_builddir = @abs_top_builddir@
10 abs_top_srcdir = @abs_top_srcdir@
12 ACLOCAL_AMFLAGS = -I autotools
13 DOCBOOK_WRAPPER = $(top_srcdir)/autotools/docbook-wrapper
14 BUILD_BASH_COMPLETION = $(top_srcdir)/autotools/build-bash-completion
15 RUN_IN_TEMPDIR = $(top_srcdir)/autotools/run-in-tempdir
16 CHECK_PYTHON_CODE = $(top_srcdir)/autotools/check-python-code
17 CHECK_MAN = $(top_srcdir)/autotools/check-man
18 REPLACE_VARS_SED = autotools/replace_vars.sed
20 clientdir = $(pkgpythondir)/client
21 hypervisordir = $(pkgpythondir)/hypervisor
22 httpdir = $(pkgpythondir)/http
23 masterddir = $(pkgpythondir)/masterd
24 confddir = $(pkgpythondir)/confd
25 rapidir = $(pkgpythondir)/rapi
26 serverdir = $(pkgpythondir)/server
27 watcherdir = $(pkgpythondir)/watcher
28 impexpddir = $(pkgpythondir)/impexpd
29 toolsdir = $(pkglibdir)/tools
30 docdir = $(datadir)/doc/$(PACKAGE)
32 # Delete output file if an error occurred while building it
42 doc/examples/gnt-debug \
68 ganeti-[0-9]*.[0-9]*.[0-9]* \
71 all_dirfiles = $(addsuffix /.dir,$(DIRS) $(BUILDTIME_DIRS))
73 MAINTAINERCLEANFILES = \
76 doc/install-quick.rst \
80 maintainer-clean-local:
81 rm -rf $(BUILDTIME_DIRS)
84 $(addsuffix /*.py[co],$(DIRS)) \
87 autotools/replace_vars.sed \
90 daemons/ganeti-cleaner \
92 doc/examples/bash_completion \
93 doc/examples/ganeti.initd \
94 doc/examples/ganeti.cron \
95 doc/examples/gnt-config-backup \
96 doc/examples/hooks/ipsec \
100 $(nodist_pkgpython_PYTHON)
102 # BUILT_SOURCES should only be used as a dependency on phony targets. Otherwise
103 # it'll cause the target to rebuild every time.
111 nodist_pkgpython_PYTHON = \
115 lib/build/__init__.py
119 lib/asyncnotifier.py \
150 lib/client/__init__.py \
151 lib/client/gnt_backup.py \
152 lib/client/gnt_cluster.py \
153 lib/client/gnt_debug.py \
154 lib/client/gnt_instance.py \
155 lib/client/gnt_job.py \
156 lib/client/gnt_node.py \
159 hypervisor_PYTHON = \
160 lib/hypervisor/__init__.py \
161 lib/hypervisor/hv_base.py \
162 lib/hypervisor/hv_chroot.py \
163 lib/hypervisor/hv_fake.py \
164 lib/hypervisor/hv_kvm.py \
165 lib/hypervisor/hv_lxc.py \
166 lib/hypervisor/hv_xen.py
169 lib/rapi/__init__.py \
170 lib/rapi/baserlib.py \
172 lib/rapi/client_utils.py \
173 lib/rapi/connector.py \
177 lib/http/__init__.py \
183 lib/confd/__init__.py \
184 lib/confd/client.py \
185 lib/confd/querylib.py \
189 lib/masterd/__init__.py \
190 lib/masterd/instance.py
193 lib/impexpd/__init__.py
196 lib/watcher/__init__.py
199 lib/server/__init__.py \
200 lib/server/confd.py \
201 lib/server/masterd.py \
210 doc/cluster-merge.rst \
216 doc/install-quick.rst \
219 doc/move-instance.rst \
225 $(RUN_IN_TEMPDIR): | $(all_dirfiles)
227 doc/html/index.html: $(docrst) $(docpng) doc/conf.py configure.ac \
228 $(RUN_IN_TEMPDIR) | doc/html/.dir
229 @test -n "$(SPHINX)" || \
230 { echo 'sphinx-build' not found during configure; exit 1; }
231 PYTHONPATH=. $(RUN_IN_TEMPDIR) $(SPHINX) -q -W -b html \
233 -D version="$(VERSION_MAJOR).$(VERSION_MINOR)" \
234 -D release="$(PACKAGE_VERSION)" \
235 $(abs_top_srcdir)/doc $(CURDIR)/doc/html
236 rm -f doc/html/.buildinfo doc/html/objects.inv
239 doc/html: doc/html/index.html
243 { echo '.. This file is automatically updated at build time from $<.'; \
244 echo '.. Do not edit.'; \
249 doc/install-quick.rst: INSTALL
251 { echo '.. This file is automatically updated at build time from $<.'; \
252 echo '.. Do not edit.'; \
259 doc/design-2.1-lock-acquire.dot \
260 doc/design-2.1-lock-release.dot
262 docpng = $(patsubst %.dot,%.png,$(docdot))
264 # Things to build but not to install (add it to EXTRA_DIST if it should be
269 doc/examples/bash_completion \
270 doc/examples/ganeti.cron \
271 doc/examples/ganeti.initd \
272 doc/examples/gnt-config-backup \
273 doc/examples/hooks/ipsec \
278 scripts/gnt-cluster \
280 scripts/gnt-instance \
286 daemons/ganeti-confd \
287 daemons/ganeti-masterd \
288 daemons/ganeti-noded \
289 daemons/ganeti-watcher \
291 scripts/gnt-cluster \
293 scripts/gnt-instance \
298 dist_sbin_SCRIPTS = \
301 nodist_sbin_SCRIPTS = \
302 $(PYTHON_BOOTSTRAP) \
303 daemons/ganeti-cleaner
305 dist_tools_SCRIPTS = \
310 tools/cluster-merge \
312 tools/move-instance \
314 tools/sanitize-config
316 pkglib_python_scripts = \
317 daemons/import-export \
318 tools/check-cert-expired
321 daemons/daemon-util \
322 daemons/ensure-dirs \
323 $(pkglib_python_scripts)
328 autotools/build-bash-completion \
329 autotools/check-python-code \
330 autotools/check-man \
331 autotools/docbook-wrapper \
332 autotools/gen-coverage \
333 autotools/testrunner \
335 daemons/daemon-util.in \
336 daemons/ensure-dirs.in \
337 daemons/ganeti-cleaner.in \
338 $(pkglib_python_scripts) \
345 doc/examples/ganeti.initd.in \
346 doc/examples/ganeti.cron.in \
347 doc/examples/gnt-config-backup.in \
348 doc/examples/dumb-allocator \
349 doc/examples/ganeti.default \
350 doc/examples/ganeti.default-debug \
351 doc/examples/hooks/ethers \
352 doc/examples/hooks/ipsec.in \
353 doc/examples/gnt-debug/README \
354 doc/examples/gnt-debug/delay0.json \
355 doc/examples/gnt-debug/delay50.json \
379 man/ganeti-cleaner.8 \
381 man/ganeti-masterd.8 \
383 man/ganeti-os-interface.7 \
385 man/ganeti-watcher.8 \
394 mansgml = $(patsubst %.7,%.sgml,$(patsubst %.8,%.sgml,$(man_MANS)))
395 manhtml = $(patsubst %.sgml,%.html,$(mansgml))
397 $(patsubst %.7,%.7.in,$(patsubst %.8,%.8.in,$(man_MANS))) \
398 $(patsubst %.html,%.html.in,$(manhtml))
401 test/data/bdev-drbd-8.0.txt \
402 test/data/bdev-drbd-8.3.txt \
403 test/data/bdev-drbd-disk.txt \
404 test/data/bdev-drbd-net-ip4.txt \
405 test/data/bdev-drbd-net-ip6.txt \
406 test/data/cert1.pem \
407 test/data/proc_drbd8.txt \
408 test/data/proc_drbd80-emptyline.txt \
409 test/data/proc_drbd83.txt \
410 test/data/sys_drbd_usermode_helper.txt \
411 test/import-export_unittest-helper
414 test/ganeti.asyncnotifier_unittest.py \
415 test/ganeti.backend_unittest.py \
416 test/ganeti.bdev_unittest.py \
417 test/ganeti.cli_unittest.py \
418 test/ganeti.daemon_unittest.py \
419 test/ganeti.cmdlib_unittest.py \
420 test/ganeti.compat_unittest.py \
421 test/ganeti.confd.client_unittest.py \
422 test/ganeti.config_unittest.py \
423 test/ganeti.constants_unittest.py \
424 test/ganeti.errors_unittest.py \
425 test/ganeti.hooks_unittest.py \
426 test/ganeti.http_unittest.py \
427 test/ganeti.hypervisor.hv_kvm_unittest.py \
428 test/ganeti.impexpd_unittest.py \
429 test/ganeti.jqueue_unittest.py \
430 test/ganeti.locking_unittest.py \
431 test/ganeti.luxi_unittest.py \
432 test/ganeti.masterd.instance_unittest.py \
433 test/ganeti.mcpu_unittest.py \
434 test/ganeti.netutils_unittest.py \
435 test/ganeti.objects_unittest.py \
436 test/ganeti.opcodes_unittest.py \
437 test/ganeti.rapi.client_unittest.py \
438 test/ganeti.rapi.resources_unittest.py \
439 test/ganeti.rapi.rlib2_unittest.py \
440 test/ganeti.rpc_unittest.py \
441 test/ganeti.runtime_unittest.py \
442 test/ganeti.serializer_unittest.py \
443 test/ganeti.ssh_unittest.py \
444 test/ganeti.uidpool_unittest.py \
445 test/ganeti.utils_unittest.py \
446 test/ganeti.utils_mlockall_unittest.py \
447 test/ganeti.workerpool_unittest.py \
448 test/cfgupgrade_unittest.py \
449 test/docs_unittest.py \
450 test/tempfile_fork_unittest.py
453 test/check-cert-expired_unittest.bash \
454 test/daemon-util_unittest.bash \
455 test/ganeti-cleaner_unittest.bash \
456 test/import-export_unittest.bash \
461 TESTS = $(dist_TESTS) $(nodist_TESTS)
463 # Environment for all tests
464 PLAIN_TESTS_ENVIRONMENT = \
465 PYTHONPATH=. TOP_SRCDIR=$(abs_top_srcdir) PYTHON=$(PYTHON) $(RUN_IN_TEMPDIR)
467 # Environment for tests run by automake
468 TESTS_ENVIRONMENT = \
469 $(PLAIN_TESTS_ENVIRONMENT) $(abs_top_srcdir)/autotools/testrunner
472 $(dist_sbin_SCRIPTS) \
473 $(dist_tools_SCRIPTS) \
474 $(pkglib_python_scripts) \
476 $(pkgpython_PYTHON) \
478 $(hypervisor_PYTHON) \
490 test/check-cert-expired_unittest.bash \
491 test/daemon-util_unittest.bash \
492 test/ganeti-cleaner_unittest.bash \
493 test/import-export_unittest.bash \
496 check_python_code = \
497 $(BUILD_BASH_COMPLETION) \
502 ganeti/http/server.py \
503 $(dist_sbin_SCRIPTS) \
504 $(dist_tools_SCRIPTS) \
505 $(pkglib_python_scripts) \
506 $(BUILD_BASH_COMPLETION)
508 test/daemon-util_unittest.bash: daemons/daemon-util
510 test/ganeti-cleaner_unittest.bash: daemons/ganeti-cleaner
512 devel/upload: devel/upload.in $(REPLACE_VARS_SED)
513 sed -f $(REPLACE_VARS_SED) < $< > $@
516 daemons/%: daemons/%.in $(REPLACE_VARS_SED)
517 sed -f $(REPLACE_VARS_SED) < $< > $@
520 doc/examples/%: doc/examples/%.in $(REPLACE_VARS_SED)
521 sed -f $(REPLACE_VARS_SED) < $< > $@
523 doc/examples/hooks/%: doc/examples/hooks/%.in $(REPLACE_VARS_SED)
524 sed -f $(REPLACE_VARS_SED) < $< > $@
526 doc/examples/bash_completion: $(BUILD_BASH_COMPLETION) $(RUN_IN_TEMPDIR) \
527 lib/cli.py $(gnt_scripts) $(client_PYTHON) tools/burnin
528 PYTHONPATH=. $(RUN_IN_TEMPDIR) $(CURDIR)/$(BUILD_BASH_COMPLETION) > $@
531 @test -n "$(DOT)" || { echo 'dot' not found during configure; exit 1; }
532 $(DOT) -Tpng -o $@ $<
534 man/%.7.in man/%.8.in: man/%.sgml man/footer.sgml $(DOCBOOK_WRAPPER)
535 @test -n "$(DOCBOOK2MAN)" || \
536 { echo 'docbook2man' not found during configure; exit 1; }
537 $(DOCBOOK_WRAPPER) "$(DOCBOOK2MAN)" $< $(notdir $(@:.in=)) $@
538 if test -n "$(MAN_HAS_WARNINGS)"; then $(CHECK_MAN) $@; fi
540 man/%.html.in: man/%.sgml man/footer.sgml $(DOCBOOK_WRAPPER)
541 @test -n "$(DOCBOOK2HTML)" || \
542 { echo 'docbook2html' not found during configure; exit 1; }
543 $(DOCBOOK_WRAPPER) "$(DOCBOOK2HTML) --nochunks" $< $(notdir $(@:.in=)) $@
545 man/%.7: man/%.7.in $(REPLACE_VARS_SED)
546 sed -f $(REPLACE_VARS_SED) < $< > $@
548 man/%.8: man/%.8.in $(REPLACE_VARS_SED)
549 sed -f $(REPLACE_VARS_SED) < $< > $@
551 man/%.html: man/%.html.in $(REPLACE_VARS_SED)
552 sed -f $(REPLACE_VARS_SED) < $< > $@
555 if test -d .git; then \
557 elif test ! -f $@ ; then \
558 echo "Cannot auto-generate $@ file"; exit 1; \
561 .PHONY: regen-vcs-version
565 if test -d .git; then \
567 $(MAKE) vcs-version; \
570 lib/_autoconf.py: Makefile vcs-version | lib/.dir
572 VCSVER=`cat $(abs_top_srcdir)/vcs-version`; \
573 { echo '# This file is automatically generated, do not edit!'; \
576 echo '"""Build-time configuration for Ganeti.'; \
578 echo 'This file is autogenerated by the build process.'; \
579 echo 'For any changes you need to re-run ./configure (and'; \
580 echo 'not edit by hand).'; \
584 echo '# pylint: disable-msg=C0301,C0324'; \
585 echo '# because this is autogenerated, we do not want'; \
586 echo '# style warnings' ; \
588 echo "PACKAGE_VERSION = '$(PACKAGE_VERSION)'"; \
589 echo "VERSION_MAJOR = '$(VERSION_MAJOR)'"; \
590 echo "VERSION_MINOR = '$(VERSION_MINOR)'"; \
591 echo "VERSION_REVISION = '$(VERSION_REVISION)'"; \
592 echo "VERSION_SUFFIX = '$(VERSION_SUFFIX)'"; \
593 echo "VERSION_FULL = '$(VERSION_FULL)'"; \
594 echo "LOCALSTATEDIR = '$(localstatedir)'"; \
595 echo "SYSCONFDIR = '$(sysconfdir)'"; \
596 echo "SSH_CONFIG_DIR = '$(SSH_CONFIG_DIR)'"; \
597 echo "EXPORT_DIR = '$(EXPORT_DIR)'"; \
598 echo "OS_SEARCH_PATH = [$(OS_SEARCH_PATH)]"; \
599 echo "XEN_BOOTLOADER = '$(XEN_BOOTLOADER)'"; \
600 echo "XEN_KERNEL = '$(XEN_KERNEL)'"; \
601 echo "XEN_INITRD = '$(XEN_INITRD)'"; \
602 echo "FILE_STORAGE_DIR = '$(FILE_STORAGE_DIR)'"; \
603 echo "ENABLE_FILE_STORAGE = $(ENABLE_FILE_STORAGE)"; \
604 echo "IALLOCATOR_SEARCH_PATH = [$(IALLOCATOR_SEARCH_PATH)]"; \
605 echo "KVM_PATH = '$(KVM_PATH)'"; \
606 echo "SOCAT_PATH = '$(SOCAT)'"; \
607 echo "SOCAT_USE_ESCAPE = $(SOCAT_USE_ESCAPE)"; \
608 echo "LVM_STRIPECOUNT = $(LVM_STRIPECOUNT)"; \
609 echo "TOOLSDIR = '$(toolsdir)'"; \
610 echo "GNT_SCRIPTS = [$(foreach i,$(notdir $(gnt_scripts)),'$(i)',)]"; \
611 echo "PKGLIBDIR = '$(pkglibdir)'"; \
612 echo "DRBD_BARRIERS = $(DRBD_BARRIERS)"; \
613 echo "SYSLOG_USAGE = '$(SYSLOG_USAGE)'"; \
614 echo "DAEMONS_GROUP = '$(DAEMONS_GROUP)'"; \
615 echo "ADMIN_GROUP = '$(ADMIN_GROUP)'"; \
616 echo "MASTERD_USER = '$(MASTERD_USER)'"; \
617 echo "MASTERD_GROUP = '$(MASTERD_GROUP)'"; \
618 echo "RAPI_USER = '$(RAPI_USER)'"; \
619 echo "RAPI_GROUP = '$(RAPI_GROUP)'"; \
620 echo "CONFD_USER = '$(CONFD_USER)'"; \
621 echo "CONFD_GROUP = '$(CONFD_GROUP)'"; \
622 echo "NODED_USER = '$(NODED_USER)'"; \
623 echo "VCS_VERSION = '$$VCSVER'"; \
626 $(REPLACE_VARS_SED): Makefile
628 { echo 's#@PREFIX@#$(prefix)#g'; \
629 echo 's#@SYSCONFDIR@#$(sysconfdir)#g'; \
630 echo 's#@LOCALSTATEDIR@#$(localstatedir)#g'; \
631 echo 's#@BINDIR@#$(bindir)#g'; \
632 echo 's#@SBINDIR@#$(sbindir)#g'; \
633 echo 's#@GANETI_VERSION@#$(PACKAGE_VERSION)#g'; \
634 echo 's#@CUSTOM_XEN_BOOTLOADER@#$(XEN_BOOTLOADER)#g'; \
635 echo 's#@CUSTOM_XEN_KERNEL@#$(XEN_KERNEL)#g'; \
636 echo 's#@CUSTOM_XEN_INITRD@#$(XEN_INITRD)#g'; \
637 echo 's#@RPL_FILE_STORAGE_DIR@#$(FILE_STORAGE_DIR)#g'; \
638 echo 's#@RPL_SSH_INITD_SCRIPT@#$(SSH_INITD_SCRIPT)#g'; \
639 echo 's#@PKGLIBDIR@#$(pkglibdir)#g'; \
640 echo 's#@GNTMASTERUSER@#$(MASTERD_USER)#g'; \
641 echo 's#@GNTRAPIUSER@#$(RAPI_USER)#g'; \
642 echo 's#@GNTCONFDUSER@#$(CONFD_USER)#g'; \
643 echo 's#@GNTNODEDUSER@#$(NODED_USER)#g'; \
644 echo 's#@GNTRAPIGROUP@#$(RAPI_GROUP)#g'; \
645 echo 's#@GNTADMINGROUP@#$(ADMIN_GROUP)#g'; \
646 echo 's#@GNTCONFDGROUP@#$(CONFD_GROUP)#g'; \
647 echo 's#@GNTMASTERDGROUP@#$(MASTERD_GROUP)#g'; \
648 echo 's#@GNTDAEMONSGROUP@#$(DAEMONS_GROUP)#g'; \
651 # Using deferred evaluation
652 daemons/ganeti-%: MODULE = ganeti.server.$(patsubst ganeti-%,%,$(notdir $@))
653 daemons/ganeti-watcher: MODULE = ganeti.watcher
654 scripts/%: MODULE = ganeti.client.$(subst -,_,$(notdir $@))
656 $(PYTHON_BOOTSTRAP): Makefile | $(all_dirfiles)
657 test -n "$(MODULE)" || { echo Missing module; exit 1; }
659 { echo '#!/usr/bin/python'; \
660 echo '# This file is automatically generated, do not edit!'; \
661 echo "# Edit $(MODULE) instead."; \
663 echo '"""Bootstrap script for L{$(MODULE)}"""'; \
665 echo '# pylint: disable-msg=C0103'; \
666 echo '# C0103: Invalid name'; \
669 echo 'import $(MODULE) as main'; \
671 echo '# Temporarily alias commands until bash completion'; \
672 echo '# generator is changed'; \
673 echo 'if hasattr(main, "commands"):'; \
674 echo ' commands = main.commands'; \
676 echo 'if __name__ == "__main__":'; \
677 echo ' sys.exit(main.Main())'; \
681 # We need to create symlinks because "make distcheck" will not install Python
682 # files when building.
683 stamp-srclinks: Makefile | $(all_dirfiles)
685 for i in $(srclink_files); do \
686 if test ! -f $$i -a -f $(abs_top_srcdir)/$$i; then \
687 $(LN_S) $(abs_top_srcdir)/$$i $$i; \
694 cd $(top_builddir) && test -h "$@" || { rm -f $@ && $(LN_S) lib $@; }
697 check-dirs: $(BUILT_SOURCES)
699 find . -type d \( \( -name . \) -o \( \
701 -name autom4te.cache \
702 \) -prune -o -print \) | { \
706 $(strip $(patsubst %,(./%) ;;,$(DIRCHECK_EXCLUDE) $(DIRS))) \
707 *) error=1; echo "Directory $$dir not listed in Makefile" >&2 ;; \
710 for dir in $(DIRS); do \
711 if ! test -d "$$dir"; then \
712 echo "Directory $$dir listed in DIRS does not exist" >&2; \
716 if test -n "$$error"; then exit 1; else exit 0; fi; \
719 check-local: check-dirs
720 $(CHECK_PYTHON_CODE) $(check_python_code)
723 lint: $(BUILT_SOURCES)
724 @test -n "$(PYLINT)" || { echo 'pylint' not found during configure; exit 1; }
725 $(PYLINT) $(LINT_OPTS) $(lint_python_code)
727 # a dist hook rule for updating the vcs-version file; this is
728 # hardcoded due to where it needs to build the file...
730 $(MAKE) regen-vcs-version && \
731 rm -f $(top_distdir)/vcs-version && \
732 cp -p $(srcdir)/vcs-version $(top_distdir)
734 # a distcheck hook rule for catching revision control directories
736 if find $(top_distdir) -name .svn -or -name .git | grep .; then \
737 echo "Found revision control files in final archive." 1>&2; \
740 if find $(top_distdir) -name '*.py[co]' | grep .; then \
741 echo "Found Python byte code in final archive." 1>&2; \
744 if find $(top_distdir) -name '*~' | grep .; then \
745 echo "Found backup files in final archive." 1>&2; \
748 if test -n "$(BUILD_RELEASE)" && \
749 grep -n -H -E '^\*.*unreleased' $(top_distdir)/NEWS; then \
750 echo "Found unreleased version in NEWS." >&2; \
754 # When building a release, stricter checks should be used
755 distcheck-release: export BUILD_RELEASE = 1
756 distcheck-release: distcheck
759 @mkdir_p@ "$(DESTDIR)${localstatedir}/lib/ganeti" \
760 "$(DESTDIR)${localstatedir}/log/ganeti" \
761 "$(DESTDIR)${localstatedir}/run/ganeti"
763 # To avoid conflicts between directory names and other targets, a file inside
764 # the directory is used to ensure its existence.
766 @mkdir_p@ $* && touch $@
769 apidoc: epydoc.conf $(RUN_IN_TEMPDIR) $(BUILT_SOURCES)
770 $(RUN_IN_TEMPDIR) epydoc -v \
771 --conf $(CURDIR)/epydoc.conf \
772 --output $(CURDIR)/doc/api
775 TAGS: $(BUILT_SOURCES)
777 find . -path './lib/*.py' -o -path './scripts/gnt-*' -o \
778 -path './daemons/ganeti-*' -o -path './tools/*' -o \
779 -path './qa/*.py' | \
783 coverage: $(BUILT_SOURCES) $(python_tests)
785 COVERAGE_FILE=$(CURDIR)/doc/coverage/data \
786 TEXT_COVERAGE=$(CURDIR)/doc/coverage/report.txt \
787 HTML_COVERAGE=$(CURDIR)/doc/coverage \
788 $(PLAIN_TESTS_ENVIRONMENT) $(abs_top_srcdir)/autotools/gen-coverage \
791 commit-check: distcheck lint apidoc
793 -include ./Makefile.local