blob: f7b9cd964289c7bae6682b6479a1581d1e11b840 [file] [log] [blame]
Alexander Afanasyev2aa39622014-01-22 11:51:11 -08001# -*- Mode: python; py-indent-offset: 4; indent-tabs-mode: nil; coding: utf-8; -*-
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -07002
3"""
Yumin Xiaab497452016-05-10 20:23:24 +08004Copyright (c) 2014-2016, Regents of the University of California,
taylorchu5ae28772015-03-08 17:45:52 -07005 Arizona Board of Regents,
6 Colorado State University,
7 University Pierre & Marie Curie, Sorbonne University,
8 Washington University in St. Louis,
9 Beijing Institute of Technology,
10 The University of Memphis.
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -070011
12This file is part of NFD (Named Data Networking Forwarding Daemon).
13See AUTHORS.md for complete list of NFD authors and contributors.
14
15NFD is free software: you can redistribute it and/or modify it under the terms
16of the GNU General Public License as published by the Free Software Foundation,
17either version 3 of the License, or (at your option) any later version.
18
19NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
20without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
21PURPOSE. See the GNU General Public License for more details.
22
23You should have received a copy of the GNU General Public License along with
24NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
25"""
Alexander Afanasyev2aa39622014-01-22 11:51:11 -080026
Alexander Afanasyev58d479c2016-03-24 13:56:55 -070027VERSION = "0.4.1"
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -070028APPNAME = "nfd"
29BUGREPORT = "http://redmine.named-data.net/projects/nfd"
Alexander Afanasyevb47d5382014-05-05 14:35:03 -070030URL = "http://named-data.net/doc/NFD/"
Alexander Afanasyev48f5a3c2014-08-22 22:33:01 -070031GIT_TAG_PREFIX = "NFD-"
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -070032
Alexander Afanasyevb47d5382014-05-05 14:35:03 -070033from waflib import Logs, Utils, Context
Alexander Afanasyev20757882014-08-25 22:39:08 -070034import os
Alexander Afanasyev2aa39622014-01-22 11:51:11 -080035
36def options(opt):
Alexander Afanasyev5cda2e02014-04-11 13:58:44 -070037 opt.load(['compiler_cxx', 'gnu_dirs'])
Wentao Shang53df1632014-04-21 12:01:32 -070038 opt.load(['boost', 'unix-socket', 'dependency-checker', 'websocket',
Alexander Afanasyeve9186212014-08-23 20:15:48 -070039 'default-compiler-flags', 'coverage', 'pch', 'boost-kqueue',
Junxiao Shi1c93cae2014-12-09 22:52:17 -070040 'doxygen', 'sphinx_build', 'type_traits', 'compiler-features'],
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -070041 tooldir=['.waf-tools'])
Alexander Afanasyev2aa39622014-01-22 11:51:11 -080042
43 nfdopt = opt.add_option_group('NFD Options')
Alexander Afanasyev885a85b2014-04-12 21:01:13 -070044 opt.addUnixOptions(nfdopt)
Wentao Shang53df1632014-04-21 12:01:32 -070045 opt.addWebsocketOptions(nfdopt)
Alexander Afanasyev885a85b2014-04-12 21:01:13 -070046 opt.addDependencyOptions(nfdopt, 'libpcap')
47 nfdopt.add_option('--without-libpcap', action='store_true', default=False,
48 dest='without_libpcap',
49 help='''Disable libpcap (Ethernet face support will be disabled)''')
50
Alexander Afanasyev5cda2e02014-04-11 13:58:44 -070051 opt.addDependencyOptions(nfdopt, 'librt', '(optional)')
52 opt.addDependencyOptions(nfdopt, 'libresolv', '(optional)')
53
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -070054 nfdopt.add_option('--with-tests', action='store_true', default=False,
55 dest='with_tests', help='''Build unit tests''')
Ilya Moiseenko96b80bb2014-04-05 20:53:27 -040056 nfdopt.add_option('--with-other-tests', action='store_true', default=False,
57 dest='with_other_tests', help='''Build other tests''')
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -060058
Alexander Afanasyev8269a052015-02-09 16:25:36 -080059 nfdopt.add_option('--with-custom-logger', type='string', default=None,
60 dest='with_custom_logger',
61 help='''Path to custom-logger.hpp and custom-logger-factory.hpp '''
62 '''implementing Logger and LoggerFactory interfaces''')
63
Alexander Afanasyev2aa39622014-01-22 11:51:11 -080064def configure(conf):
Beichuan Zhang55b8ed42014-04-26 22:25:44 -070065 conf.load(['compiler_cxx', 'gnu_dirs',
Alexander Afanasyeve9186212014-08-23 20:15:48 -070066 'default-compiler-flags', 'pch', 'boost-kqueue',
Wentao Shang53df1632014-04-21 12:01:32 -070067 'boost', 'dependency-checker', 'websocket',
Junxiao Shi1c93cae2014-12-09 22:52:17 -070068 'doxygen', 'sphinx_build', 'type_traits', 'compiler-features'])
Alexander Afanasyevefea8fe2014-03-23 00:00:35 -070069
Davide Pesaventof0ae4422014-05-05 16:32:12 +020070 conf.find_program('bash', var='BASH')
71
taylorchu5ae28772015-03-08 17:45:52 -070072 if 'PKG_CONFIG_PATH' not in os.environ:
Alexander Afanasyevcdb8f362015-03-30 10:52:54 -070073 os.environ['PKG_CONFIG_PATH'] = Utils.subst_vars('${LIBDIR}/pkgconfig', conf.env)
Alexander Afanasyev4a771362014-04-24 21:29:33 -070074 conf.check_cfg(package='libndn-cxx', args=['--cflags', '--libs'],
75 uselib_store='NDN_CXX', mandatory=True)
Davide Pesavento44deacc2014-02-19 10:48:07 +010076
Davide Pesaventob499a602014-11-18 22:36:56 +010077 conf.checkDependency(name='librt', lib='rt', mandatory=False)
78 conf.checkDependency(name='libresolv', lib='resolv', mandatory=False)
79
Alexander Afanasyev748e9892015-01-28 15:07:41 -080080 if not conf.check_cxx(msg='Checking if privilege drop/elevation is supported', mandatory=False,
81 define_name='HAVE_PRIVILEGE_DROP_AND_ELEVATE', fragment='''
82#include <unistd.h>
83#include <pwd.h>
84#include <grp.h>
85int
86main(int, char**)
87{
Alexander Afanasyeve0d71b02016-05-25 11:39:13 -070088 char buffer[100];
Alexander Afanasyev748e9892015-01-28 15:07:41 -080089 ::sysconf(_SC_GETGR_R_SIZE_MAX);
90 group grp;
Alexander Afanasyeve0d71b02016-05-25 11:39:13 -070091 group* grpRes;
92 getgrnam_r("nogroup", &grp, buffer, 100, &grpRes);
Alexander Afanasyev748e9892015-01-28 15:07:41 -080093 passwd pwd;
Alexander Afanasyeve0d71b02016-05-25 11:39:13 -070094 passwd* pwdRes;
95 getpwnam_r("nobody", &pwd, buffer, 100, &pwdRes);
Alexander Afanasyev748e9892015-01-28 15:07:41 -080096
97 int ret = setegid(grp.gr_gid);
98 ret = seteuid(pwd.pw_uid);
99 (void)(ret);
Weiwei Liuf5aee942016-03-19 07:00:42 +0000100
101 getegid();
102 geteuid();
Alexander Afanasyev748e9892015-01-28 15:07:41 -0800103 return 0;
104}
105'''):
106 Logs.warn('Dropping privileges is not supported on this platform')
107
Alexander Afanasyev49343f62015-01-26 21:58:07 -0800108 conf.check_cxx(header_name='ifaddrs.h', mandatory=False)
Alexander Afanasyev49343f62015-01-26 21:58:07 -0800109
Yumin Xiaab497452016-05-10 20:23:24 +0800110 boost_libs = 'system chrono program_options random thread log log_setup'
Alexander Afanasyev2aa39622014-01-22 11:51:11 -0800111 if conf.options.with_tests:
112 conf.env['WITH_TESTS'] = 1
Junxiao Shi88884492014-02-15 15:57:43 -0700113 conf.define('WITH_TESTS', 1);
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -0700114 boost_libs += ' unit_test_framework'
Alexander Afanasyev2aa39622014-01-22 11:51:11 -0800115
Ilya Moiseenko96b80bb2014-04-05 20:53:27 -0400116 if conf.options.with_other_tests:
117 conf.env['WITH_OTHER_TESTS'] = 1
118
Yumin Xiaab497452016-05-10 20:23:24 +0800119 conf.check_boost(lib=boost_libs, mt=True)
Davide Pesaventocafae242016-04-22 02:21:35 +0200120 if conf.env.BOOST_VERSION_NUMBER < 105400:
121 Logs.error("Minimum required boost version is 1.54.0")
Junxiao Shiea48d8b2014-03-16 13:53:47 -0700122 Logs.error("Please upgrade your distribution or install custom boost libraries" +
123 " (http://redmine.named-data.net/projects/nfd/wiki/Boost_FAQ)")
Alexander Afanasyev2aa39622014-01-22 11:51:11 -0800124 return
Alexander Afanasyevc78b1412014-02-19 14:08:26 -0800125
Alexander Afanasyev6602b3b2016-03-15 14:14:11 -0700126 if conf.env['CXX_NAME'] == 'clang' and conf.env.BOOST_VERSION_NUMBER < 105800:
127 conf.define('BOOST_ASIO_HAS_STD_ARRAY', 1) # Workaround for http://redmine.named-data.net/issues/3360#note-14
Eric Newberrycb27ea82016-07-19 12:40:52 -0700128 conf.define('BOOST_ASIO_HAS_STD_CHRONO', 1) # Solution documented at https://redmine.named-data.net/issues/3588#note-10
Alexander Afanasyev6602b3b2016-03-15 14:14:11 -0700129
Alexander Afanasyevc78b1412014-02-19 14:08:26 -0800130 conf.load('unix-socket')
Wentao Shang53df1632014-04-21 12:01:32 -0700131 conf.checkWebsocket(mandatory=True)
Davide Pesavento44deacc2014-02-19 10:48:07 +0100132
Alexander Afanasyev885a85b2014-04-12 21:01:13 -0700133 if not conf.options.without_libpcap:
Alexander Afanasyeve9186212014-08-23 20:15:48 -0700134 conf.check_asio_pcap_support()
135 if conf.env['HAVE_ASIO_PCAP_SUPPORT']:
136 conf.checkDependency(name='libpcap', lib='pcap', mandatory=True,
137 errmsg='not found, but required for Ethernet face support. '
138 'Specify --without-libpcap to disable Ethernet face support.')
139 else:
140 Logs.warn('Warning: Ethernet face support is not supported on this platform with Boost libraries version 1.56. '
141 'See http://redmine.named-data.net/issues/1877 for more details')
Davide Pesaventof0ae4422014-05-05 16:32:12 +0200142 if conf.env['HAVE_LIBPCAP']:
143 conf.check_cxx(function_name='pcap_set_immediate_mode', header_name='pcap/pcap.h',
144 cxxflags='-Wno-error', use='LIBPCAP', mandatory=False)
Alexander Afanasyevefea8fe2014-03-23 00:00:35 -0700145
Alexander Afanasyev8269a052015-02-09 16:25:36 -0800146 if conf.options.with_custom_logger:
147 conf.define('HAVE_CUSTOM_LOGGER', 1)
148 conf.env['INCLUDES_CUSTOM_LOGGER'] = [conf.options.with_custom_logger]
149 conf.env['HAVE_CUSTOM_LOGGER'] = 1
150
Alexander Afanasyev689569b2014-02-16 20:20:07 -0800151 conf.load('coverage')
152
Steve DiBenedettofbb40a82014-03-11 19:40:15 -0600153 conf.define('DEFAULT_CONFIG_FILE', '%s/ndn/nfd.conf' % conf.env['SYSCONFDIR'])
Steve DiBenedetto2c2b8892014-02-27 11:46:48 -0700154
Junxiao Shi759f7062014-11-29 10:19:22 -0700155 # disable assertions in release builds
156 if not conf.options.debug:
157 conf.define('NDEBUG', 1)
158
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700159 conf.write_config_header('config.hpp')
Alexander Afanasyev2aa39622014-01-22 11:51:11 -0800160
Junxiao Shi09bf7c42014-01-31 11:10:25 -0700161def build(bld):
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700162 version(bld)
163
164 bld(features="subst",
165 name='version',
166 source='version.hpp.in',
167 target='version.hpp',
168 install_path=None,
169 VERSION_STRING=VERSION_BASE,
170 VERSION_BUILD=VERSION,
171 VERSION=int(VERSION_SPLIT[0]) * 1000000 +
172 int(VERSION_SPLIT[1]) * 1000 +
173 int(VERSION_SPLIT[2]),
174 VERSION_MAJOR=VERSION_SPLIT[0],
175 VERSION_MINOR=VERSION_SPLIT[1],
176 VERSION_PATCH=VERSION_SPLIT[2],
177 )
178
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700179 core = bld(
180 target='core-objects',
181 name='core-objects',
Alexander Afanasyev58ea4582014-05-30 08:38:56 +0300182 features='cxx pch',
Alexander Afanasyev8269a052015-02-09 16:25:36 -0800183 source=bld.path.ant_glob(['core/**/*.cpp'],
184 excl=['core/logger*.cpp']),
Alexander Afanasyev3d1874a2016-06-15 12:25:01 -0700185 use='version NDN_CXX BOOST LIBRT',
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700186 includes='. core',
187 export_includes='. core',
Alexander Afanasyev58ea4582014-05-30 08:38:56 +0300188 headers='common.hpp',
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700189 )
190
Alexander Afanasyev8269a052015-02-09 16:25:36 -0800191 if bld.env['HAVE_CUSTOM_LOGGER']:
192 core.use += " CUSTOM_LOGGER"
193 else:
194 core.source += bld.path.ant_glob(['core/logger*.cpp'])
195
Alexander Afanasyevc78b1412014-02-19 14:08:26 -0800196 nfd_objects = bld(
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700197 target='daemon-objects',
198 name='daemon-objects',
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -0700199 features='cxx',
200 source=bld.path.ant_glob(['daemon/**/*.cpp'],
201 excl=['daemon/face/ethernet-*.cpp',
202 'daemon/face/unix-*.cpp',
Wentao Shang53df1632014-04-21 12:01:32 -0700203 'daemon/face/websocket-*.cpp',
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -0700204 'daemon/main.cpp']),
Steve DiBenedettoef04f272014-06-04 14:28:31 -0600205 use='core-objects WEBSOCKET',
206 includes='daemon',
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700207 export_includes='daemon',
Junxiao Shi0fcb41e2014-01-24 10:29:43 -0700208 )
209
Alexander Afanasyev885a85b2014-04-12 21:01:13 -0700210 if bld.env['HAVE_LIBPCAP']:
Davide Pesavento44deacc2014-02-19 10:48:07 +0100211 nfd_objects.source += bld.path.ant_glob('daemon/face/ethernet-*.cpp')
Alexander Afanasyev5cda2e02014-04-11 13:58:44 -0700212 nfd_objects.use += ' LIBPCAP'
Davide Pesavento44deacc2014-02-19 10:48:07 +0100213
Alexander Afanasyevc78b1412014-02-19 14:08:26 -0800214 if bld.env['HAVE_UNIX_SOCKETS']:
Davide Pesavento44deacc2014-02-19 10:48:07 +0100215 nfd_objects.source += bld.path.ant_glob('daemon/face/unix-*.cpp')
Alexander Afanasyevc78b1412014-02-19 14:08:26 -0800216
Wentao Shang53df1632014-04-21 12:01:32 -0700217 if bld.env['HAVE_WEBSOCKET']:
218 nfd_objects.source += bld.path.ant_glob('daemon/face/websocket-*.cpp')
219
Alexander Afanasyev3ecec502014-04-16 13:42:44 -0700220 rib_objects = bld(
221 target='rib-objects',
222 name='rib-objects',
223 features='cxx',
Alexander Afanasyevf08a7372015-02-09 21:28:19 -0800224 source=bld.path.ant_glob(['rib/**/*.cpp']),
Alexander Afanasyev3ecec502014-04-16 13:42:44 -0700225 use='core-objects',
226 )
227
Alexander Afanasyevf08a7372015-02-09 21:28:19 -0800228 bld(target='bin/nfd',
Alexander Afanasyev3ecec502014-04-16 13:42:44 -0700229 features='cxx cxxprogram',
Alexander Afanasyevf08a7372015-02-09 21:28:19 -0800230 source='daemon/main.cpp',
231 use='daemon-objects rib-objects',
Alexander Afanasyev3ecec502014-04-16 13:42:44 -0700232 )
233
Alexander Afanasyev262203b2015-01-26 16:39:59 -0800234 bld.recurse("tools")
Alexander Afanasyev613e2a92014-04-15 13:36:58 -0700235 bld.recurse("tests")
Ilya Moiseenko96b80bb2014-04-05 20:53:27 -0400236
Alexander Afanasyev97e4cac2014-03-28 10:55:11 -0700237 bld(features="subst",
238 source='nfd.conf.sample.in',
239 target='nfd.conf.sample',
Alexander Afanasyev885a85b2014-04-12 21:01:13 -0700240 install_path="${SYSCONFDIR}/ndn",
Wentao Shang53df1632014-04-21 12:01:32 -0700241 IF_HAVE_LIBPCAP="" if bld.env['HAVE_LIBPCAP'] else "; ",
242 IF_HAVE_WEBSOCKET="" if bld.env['HAVE_WEBSOCKET'] else "; ")
Alexander Afanasyev35fc2b72014-02-13 16:56:21 -0800243
Alexander Afanasyev49272f72014-04-06 21:49:46 -0700244 if bld.env['SPHINX_BUILD']:
245 bld(features="sphinx",
246 builder="man",
247 outdir="docs/manpages",
248 config="docs/conf.py",
249 source=bld.path.ant_glob('docs/manpages/**/*.rst'),
Beichuan Zhang55b8ed42014-04-26 22:25:44 -0700250 install_path="${MANDIR}/",
251 VERSION=VERSION)
Alexander Afanasyev49272f72014-04-06 21:49:46 -0700252
Alexander Afanasyev5c475972015-12-20 16:16:56 -0800253 bld.install_files("${SYSCONFDIR}/ndn", "autoconfig.conf.sample")
254
Alexander Afanasyev284257b2014-04-11 14:16:51 -0700255def docs(bld):
256 from waflib import Options
257 Options.commands = ['doxygen', 'sphinx'] + Options.commands
258
Alexander Afanasyev35fc2b72014-02-13 16:56:21 -0800259def doxygen(bld):
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700260 version(bld)
261
Alexander Afanasyev35fc2b72014-02-13 16:56:21 -0800262 if not bld.env.DOXYGEN:
Beichuan Zhang55b8ed42014-04-26 22:25:44 -0700263 Logs.error("ERROR: cannot build documentation (`doxygen' is not found in $PATH)")
264 else:
265 bld(features="subst",
266 name="doxygen-conf",
Alexander Afanasyev20757882014-08-25 22:39:08 -0700267 source=["docs/doxygen.conf.in",
268 "docs/named_data_theme/named_data_footer-with-analytics.html.in"],
269 target=["docs/doxygen.conf",
270 "docs/named_data_theme/named_data_footer-with-analytics.html"],
Alexander Afanasyevabb307c2015-10-23 17:00:13 -0700271 VERSION=VERSION,
Alexander Afanasyev20757882014-08-25 22:39:08 -0700272 HTML_FOOTER="../build/docs/named_data_theme/named_data_footer-with-analytics.html" \
273 if os.getenv('GOOGLE_ANALYTICS', None) \
274 else "../docs/named_data_theme/named_data_footer.html",
275 GOOGLE_ANALYTICS=os.getenv('GOOGLE_ANALYTICS', ""),
Beichuan Zhang55b8ed42014-04-26 22:25:44 -0700276 )
277
278 bld(features="doxygen",
279 doxyfile='docs/doxygen.conf',
280 use="doxygen-conf")
Alexander Afanasyev49272f72014-04-06 21:49:46 -0700281
282def sphinx(bld):
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700283 version(bld)
284
Beichuan Zhang55b8ed42014-04-26 22:25:44 -0700285 if not bld.env.SPHINX_BUILD:
286 bld.fatal("ERROR: cannot build documentation (`sphinx-build' is not found in $PATH)")
287 else:
288 bld(features="sphinx",
289 outdir="docs",
290 source=bld.path.ant_glob('docs/**/*.rst'),
291 config="docs/conf.py",
Alexander Afanasyevabb307c2015-10-23 17:00:13 -0700292 VERSION=VERSION)
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700293
294def version(ctx):
Alexander Afanasyev26181532014-05-07 23:38:51 -0700295 if getattr(Context.g_module, 'VERSION_BASE', None):
296 return
297
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700298 Context.g_module.VERSION_BASE = Context.g_module.VERSION
299 Context.g_module.VERSION_SPLIT = [v for v in VERSION_BASE.split('.')]
300
Alexander Afanasyev48f5a3c2014-08-22 22:33:01 -0700301 didGetVersion = False
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700302 try:
Alexander Afanasyev48f5a3c2014-08-22 22:33:01 -0700303 cmd = ['git', 'describe', '--always', '--match', '%s*' % GIT_TAG_PREFIX]
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700304 p = Utils.subprocess.Popen(cmd, stdout=Utils.subprocess.PIPE,
305 stderr=None, stdin=None)
Alexander Afanasyev3b21fa32014-09-01 13:25:20 -0700306 out = str(p.communicate()[0].strip())
Alexander Afanasyev48f5a3c2014-08-22 22:33:01 -0700307 didGetVersion = (p.returncode == 0 and out != "")
308 if didGetVersion:
309 if out.startswith(GIT_TAG_PREFIX):
310 Context.g_module.VERSION = out[len(GIT_TAG_PREFIX):]
311 else:
312 Context.g_module.VERSION = "%s-commit-%s" % (Context.g_module.VERSION_BASE, out)
313 except OSError:
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700314 pass
315
Alexander Afanasyev48f5a3c2014-08-22 22:33:01 -0700316 versionFile = ctx.path.find_node('VERSION')
317
318 if not didGetVersion and versionFile is not None:
319 try:
320 Context.g_module.VERSION = versionFile.read()
321 return
322 except (OSError, IOError):
323 pass
324
325 # version was obtained from git, update VERSION file if necessary
326 if versionFile is not None:
327 try:
328 version = versionFile.read()
329 if version == Context.g_module.VERSION:
330 return # no need to update
331 except (OSError, IOError):
332 Logs.warn("VERSION file exists, but not readable")
333 else:
334 versionFile = ctx.path.make_node('VERSION')
335
336 if versionFile is None:
337 return
338
339 try:
340 versionFile.write(Context.g_module.VERSION)
341 except (OSError, IOError):
342 Logs.warn("VERSION file is not writeable")
343
Alexander Afanasyevb47d5382014-05-05 14:35:03 -0700344def dist(ctx):
345 version(ctx)
346
347def distcheck(ctx):
348 version(ctx)