3 # InspIRCd -- Internet Relay Chat Daemon
5 # Copyright (C) 2019 Matt Schatz <genius3000@g3k.solutions>
6 # Copyright (C) 2019 Anatole Denis <natolumin@rezel.net>
7 # Copyright (C) 2017 emerson <github@emersonveenstra.net>
8 # Copyright (C) 2013-2020 Sadie Powell <sadie@witchery.services>
9 # Copyright (C) 2012, 2019 Robby <robby@chatbelgie.be>
10 # Copyright (C) 2012 ChrisTX <xpipe@hotmail.de>
11 # Copyright (C) 2010 Daniel De Graaf <danieldg@inspircd.org>
12 # Copyright (C) 2008 Thomas Stagner <aquanight@inspircd.org>
13 # Copyright (C) 2008 Robin Burchell <robin+git@viroteck.net>
14 # Copyright (C) 2007 Dennis Friis <peavey@inspircd.org>
15 # Copyright (C) 2006-2008 Craig Edwards <brain@inspircd.org>
17 # This file is part of InspIRCd. InspIRCd is free software: you can
18 # redistribute it and/or modify it under the terms of the GNU General Public
19 # License as published by the Free Software Foundation, version 2.
21 # This program is distributed in the hope that it will be useful, but WITHOUT
22 # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
23 # FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
26 # You should have received a copy of the GNU General Public License
27 # along with this program. If not, see <http://www.gnu.org/licenses/>.
37 use warnings FATAL => qw(all);
40 use File::Basename qw(basename);
42 use File::Spec::Functions qw(abs2rel catfile catdir rel2abs);
43 use FindBin qw($RealDir);
44 use Getopt::Long qw(GetOptions);
45 use POSIX qw(getgid getuid);
57 $opt_disable_auto_extras,
58 $opt_disable_interactive,
59 $opt_distribution_label,
74 sub enable_extras (@);
76 sub disable_extras (@);
79 my @opt_disableextras;
81 exit 1 unless GetOptions(
82 'clean' => \&cmd_clean,
84 'update' => \&cmd_update,
86 'binary-dir=s' => \$opt_binary_dir,
87 'config-dir=s' => \$opt_config_dir,
88 'data-dir=s' => \$opt_data_dir,
89 'development' => \$opt_development,
90 'disable-auto-extras' => \$opt_disable_auto_extras,
91 'disable-interactive' => \$opt_disable_interactive,
92 'distribution-label=s' => \$opt_distribution_label,
93 'example-dir=s' => \$opt_example_dir,
95 'log-dir=s' => \$opt_log_dir,
96 'manual-dir=s' => \$opt_manual_dir,
97 'module-dir=s' => \$opt_module_dir,
98 'portable' => \$opt_portable,
99 'prefix=s' => \$opt_prefix,
100 'script-dir=s' => \$opt_script_dir,
101 'socketengine=s' => \$opt_socketengine,
102 'system' => \$opt_system,
103 'uid=s' => \$opt_uid,
105 # TODO: when the modulemanager rewrite is done these should be removed.
106 'disable-extras=s@' => \@opt_disableextras,
107 'enable-extras=s@' => \@opt_enableextras,
108 'list-extras' => sub { list_extras; exit 0; },
111 if (scalar(@opt_enableextras) + scalar(@opt_disableextras) > 0) {
112 @opt_enableextras = split /[, ]+/, join(',', @opt_enableextras);
113 @opt_disableextras = split /[, ]+/, join(',', @opt_disableextras);
114 enable_extras(@opt_enableextras);
115 disable_extras(@opt_disableextras);
117 print "Remember: YOU are responsible for making sure any libraries needed have been installed!\n";
121 our $interactive = !(
124 defined $opt_binary_dir ||
125 defined $opt_config_dir ||
126 defined $opt_data_dir ||
127 defined $opt_development ||
128 defined $opt_disable_auto_extras ||
129 defined $opt_disable_interactive ||
130 defined $opt_distribution_label ||
131 defined $opt_example_dir ||
133 defined $opt_log_dir ||
134 defined $opt_manual_dir ||
135 defined $opt_module_dir ||
136 defined $opt_portable ||
137 defined $opt_prefix ||
138 defined $opt_script_dir ||
139 defined $opt_socketengine ||
140 defined $opt_system ||
144 my %version = get_version $opt_distribution_label;
145 print_format "<|BOLD Configuring InspIRCd $version{FULL} on $^O.|>\n";
149 %config = read_config_file(CONFIGURE_CACHE_FILE);
150 run_test abs2rel(CONFIGURE_CACHE_FILE, $RealDir), %config;
151 if (!defined $config{VERSION}) {
152 $config{VERSION} = CONFIGURE_CACHE_VERSION;
153 } elsif ($config{VERSION} != CONFIGURE_CACHE_VERSION) {
154 print_warning "ignoring contents of ${\CONFIGURE_CACHE_FILE} as it was generated by an incompatible version of $0!";
155 %config = ('VERSION', CONFIGURE_CACHE_VERSION);
159 $config{CXX} = find_compiler($config{CXX} // $ENV{CXX});
160 unless ($config{CXX}) {
161 say 'A suitable C++ compiler could not be detected on your system!';
162 unless ($interactive) {
163 say 'Set the CXX environment variable to the path to a C++ compiler binary if this is incorrect.';
166 until ($config{CXX}) {
167 my $compiler_path = prompt_string 1, 'Please enter the path to a C++ compiler binary:', 'c++';
168 $config{CXX} = find_compiler $compiler_path;
171 my %compiler = get_compiler_info($config{CXX});
173 $config{HAS_ARC4RANDOM_BUF} = run_test 'arc4random_buf()', test_file($config{CXX}, 'arc4random_buf.cpp');
174 $config{HAS_CLOCK_GETTIME} = run_test 'clock_gettime()', test_file($config{CXX}, 'clock_gettime.cpp', $^O eq 'darwin' ? undef : '-lrt');
175 $config{HAS_EVENTFD} = run_test 'eventfd()', test_file($config{CXX}, 'eventfd.cpp');
178 push @socketengines, 'epoll' if run_test 'epoll', test_header $config{CXX}, 'sys/epoll.h';
179 push @socketengines, 'kqueue' if run_test 'kqueue', test_file $config{CXX}, 'kqueue.cpp';
180 push @socketengines, 'poll' if run_test 'poll', test_header $config{CXX}, 'poll.h';
181 push @socketengines, 'select';
183 if (defined $opt_socketengine) {
184 unless (grep { $_ eq $opt_socketengine } @socketengines) {
185 my $reason = -f "src/socketengines/socketengine_$opt_socketengine.cpp" ? 'is not available on this platform' : 'does not exist';
186 print_error "The socket engine you requested ($opt_socketengine) $reason!",
187 'Available socket engines are:',
188 map { " * $_" } @socketengines;
191 $config{SOCKETENGINE} = $opt_socketengine // $socketengines[0];
193 if (defined $opt_portable) {
194 print_error '--portable and --system can not be used together!' if defined $opt_system;
195 $config{DESTDIR} = catfile $RealDir, 'run', '';
196 $config{BASE_DIR} = $opt_prefix // '';
197 $config{BINARY_DIR} = $opt_binary_dir // 'bin';
198 $config{CONFIG_DIR} = $opt_config_dir // 'conf';
199 $config{DATA_DIR} = $opt_data_dir // 'data';
200 $config{EXAMPLE_DIR} = $opt_example_dir // catdir $config{CONFIG_DIR}, 'examples';
201 $config{LOG_DIR} = $opt_log_dir // 'logs';
202 $config{MANUAL_DIR} = $opt_manual_dir // 'manuals';
203 $config{MODULE_DIR} = $opt_module_dir // 'modules';
204 $config{SCRIPT_DIR} = $opt_script_dir // $config{BASE_DIR};
205 } elsif (defined $opt_system) {
206 $config{BASE_DIR} = $opt_prefix // '/var/lib/inspircd';
207 $config{BINARY_DIR} = $opt_binary_dir // '/usr/sbin';
208 $config{CONFIG_DIR} = $opt_config_dir // '/etc/inspircd';
209 $config{DATA_DIR} = $opt_data_dir // '/var/inspircd';
210 $config{EXAMPLE_DIR} = $opt_example_dir // '/usr/share/doc/inspircd';
211 $config{LOG_DIR} = $opt_log_dir // '/var/log/inspircd';
212 $config{MANUAL_DIR} = $opt_manual_dir // '/usr/share/man/man1';
213 $config{MODULE_DIR} = $opt_module_dir // '/usr/lib/inspircd';
214 $config{SCRIPT_DIR} = $opt_script_dir // '/usr/share/inspircd'
216 $config{BASE_DIR} = rel2abs $opt_prefix // $config{BASE_DIR} // catdir $RealDir, 'run';
217 $config{BINARY_DIR} = $opt_binary_dir // $config{BINARY_DIR} // catdir $config{BASE_DIR}, 'bin';
218 $config{CONFIG_DIR} = $opt_config_dir // $config{CONFIG_DIR} // catdir $config{BASE_DIR}, 'conf';
219 $config{DATA_DIR} = $opt_data_dir // $config{DATA_DIR} // catdir $config{BASE_DIR}, 'data';
220 $config{EXAMPLE_DIR} = $opt_example_dir // $config{EXAMPLE_DIR} // catdir $config{CONFIG_DIR}, 'examples';
221 $config{LOG_DIR} = $opt_log_dir // $config{LOG_DIR} // catdir $config{BASE_DIR}, 'logs';
222 $config{MANUAL_DIR} = $opt_manual_dir // $config{MANUAL_DIR} // catdir $config{BASE_DIR}, 'manuals';
223 $config{MODULE_DIR} = $opt_module_dir // $config{MODULE_DIR} // catdir $config{BASE_DIR}, 'modules';
224 $config{SCRIPT_DIR} = $opt_script_dir // $config{SCRIPT_DIR} // $config{BASE_DIR};
227 # Parse --gid=123 or --gid=foo and extract the group id.
229 if (defined $opt_gid) {
230 @group = $opt_gid =~ /^\d+$/ ? getgrgid($opt_gid) : getgrnam($opt_gid);
231 print_error "there is no '$opt_gid' group on this system!" unless @group;
233 @group = $opt_system ? getgrnam('irc') : getgrgid($config{GID} // getgid());
234 print_error "you need to specify a group to run as using '--gid [id|name]'!" unless @group;
236 print_warning <<"EOW";
237 You are building as the privileged $group[0] group and have not specified
238 an unprivileged group to run InspIRCd as.
240 This is almost never what you should do. You should probably either create a new
241 unprivileged user/group to build and run as or pass the '--gid [id|name]' flag
242 to specify an unprivileged group to run as.
244 if (!prompt_bool $interactive, "Are you sure you want to build as the $group[0] group?", 0) {
245 # PACKAGERS: You do not need to delete this check. Use `--gid $(id -g)` or `--gid 0` instead.
246 say STDERR "If you are sure you want to build as the $group[0] group pass the --gid $group[2] flag." unless $interactive;
251 $config{GROUP} = $group[0];
252 $config{GID} = $group[2];
254 # Parse --uid=123 or --uid=foo and extract the user id.
256 if (defined $opt_uid) {
257 @user = $opt_uid =~ /^\d+$/ ? getpwuid($opt_uid) : getpwnam($opt_uid);
258 print_error "there is no '$opt_uid' user on this system!" unless @user;
260 @user = $opt_system ? getpwnam('irc') : getpwuid($config{UID} // getuid());
261 print_error "you need to specify a user to run as using '--uid [id|name]'!" unless @user;
263 print_warning <<"EOW";
264 You are building as the privileged $user[0] user and have not specified
265 an unprivileged user to run InspIRCd as.
267 This is almost never what you should do. You should probably either create a new
268 unprivileged user/group to build and run as or pass the '--uid [id|name]' flag
269 to specify an unprivileged user to run as.
271 if (!prompt_bool $interactive, "Are you sure you want to build as the $user[0] user?", 0) {
272 # PACKAGERS: You do not need to delete this check. Use `--uid $(id -u)` or `--uid 0` instead.
273 say STDERR "If you are sure you want to build as the $user[0] user pass the --uid $user[2] flag." unless $interactive;
278 $config{USER} = $user[0];
279 $config{UID} = $user[2];
281 # Warn the user about clock drifting when running on OpenVZ.
282 if (-e '/proc/user_beancounters' || -e '/proc/vz/vzaquota') {
283 print_warning <<'EOW';
284 You are building InspIRCd inside of an OpenVZ container. If you
285 plan to use InspIRCd in this container then you should make sure that NTP is
286 configured on the Hardware Node. Failure to do so may result in clock drifting!
290 # Warn the user about OpenBSD shipping incredibly broken compilers/linkers.
291 if ($^O eq 'openbsd') {
292 print_warning <<'EOW';
293 You are building InspIRCd on OpenBSD. The C++ compilers and linkers
294 that OpenBSD ship are incredibly broken. You may have strange linker errors
295 and crashes. Please consider using a different OS like FreeBSD/NetBSD instead.
299 # Check that the user actually wants this version.
300 if (defined $version{REAL_LABEL}) {
301 print_warning <<'EOW';
302 You are building a development version. This contains code which has
303 not been tested as heavily and may contain various faults which could seriously
304 affect the running of your server. It is recommended that you use a stable
307 You can obtain the latest stable version from https://www.inspircd.org or by
308 running `<|GREEN git checkout $(git describe --abbrev=0 --tags insp3)|>` if you are
311 if (!prompt_bool $interactive, 'I understand this warning and want to continue anyway.', $opt_development // 0) {
312 say STDERR 'If you understand this warning and still want to continue pass the --development flag.' unless $interactive;
317 # Configure directory settings.
318 my $question = <<"EOQ";
319 Currently, InspIRCd is configured with the following paths:
321 <|BOLD Base:|> $config{BASE_DIR}
322 <|BOLD Binary:|> $config{BINARY_DIR}
323 <|BOLD Config:|> $config{CONFIG_DIR}
324 <|BOLD Data:|> $config{DATA_DIR}
325 <|BOLD Log:|> $config{LOG_DIR}
326 <|BOLD Manual:|> $config{MANUAL_DIR}
327 <|BOLD Module:|> $config{MODULE_DIR}
328 <|BOLD Script:|> $config{SCRIPT_DIR}
330 Do you want to change these settings?
332 if (prompt_bool $interactive, $question, 0) {
333 my $original_base_dir = $config{BASE_DIR};
334 $config{BASE_DIR} = prompt_dir $interactive, 'In what directory do you wish to install the InspIRCd base?', $config{BASE_DIR};
335 foreach my $key (qw(BINARY_DIR CONFIG_DIR DATA_DIR LOG_DIR MANUAL_DIR MODULE_DIR SCRIPT_DIR)) {
336 $config{$key} =~ s/^\Q$original_base_dir\E/$config{BASE_DIR}/;
338 $config{BINARY_DIR} = prompt_dir $interactive, 'In what directory should the InspIRCd binary be placed?', $config{BINARY_DIR};
339 $config{CONFIG_DIR} = prompt_dir $interactive, 'In what directory are configuration files to be stored?', $config{CONFIG_DIR};
340 $config{DATA_DIR} = prompt_dir $interactive, 'In what directory are variable data files to be stored?', $config{DATA_DIR};
341 $config{LOG_DIR} = prompt_dir $interactive, 'In what directory are log files to be stored?', $config{LOG_DIR};
342 $config{MANUAL_DIR} = prompt_dir $interactive, 'In what directory are manual pages to be placed?', $config{MANUAL_DIR};
343 $config{MODULE_DIR} = prompt_dir $interactive, 'In what directory are modules to be placed?', $config{MODULE_DIR};
344 $config{SCRIPT_DIR} = prompt_dir $interactive, 'In what directory are scripts to be placed?', $config{SCRIPT_DIR};
345 $config{EXAMPLE_DIR} = $config{CONFIG_DIR} . '/examples';
348 # Configure module settings.
350 Currently, InspIRCd is configured to automatically enable all available extra modules.
352 Would you like to enable extra modules manually?
354 if (prompt_bool $interactive, $question, 0) {
355 foreach my $extra (<$RealDir/src/modules/extra/m_*.cpp>) {
356 my $module_name = basename $extra, '.cpp';
357 if (prompt_bool $interactive, "Would you like to enable $module_name?", 0) {
358 enable_extras $module_name;
361 } elsif (!defined $opt_disable_auto_extras) {
362 # TODO: finish modulemanager rewrite and replace this code with:
363 # system './modulemanager', 'enable', '--auto';
365 # Missing: m_ldap, m_regex_stdlib, m_ssl_mbedtls
366 'm_argon2.cpp' => 'pkg-config --exists libargon2',
367 'm_geo_maxmind.cpp' => 'pkg-config --exists libmaxminddb',
368 'm_mysql.cpp' => 'mysql_config --version',
369 'm_pgsql.cpp' => 'pg_config --version',
370 'm_regex_pcre.cpp' => 'pcre-config --version',
371 'm_regex_posix.cpp' => undef,
372 'm_regex_re2.cpp' => 'pkg-config --exists re2',
373 'm_regex_tre.cpp' => 'pkg-config --exists tre',
374 'm_sqlite3.cpp' => 'pkg-config --exists sqlite3',
375 'm_ssl_gnutls.cpp' => 'pkg-config --exists gnutls',
376 'm_ssl_openssl.cpp' => 'pkg-config --exists openssl',
377 'm_sslrehashsignal.cpp' => undef,
379 while (my ($module, $command) = each %modules) {
380 unless (defined $command && system "$command 1>/dev/null 2>/dev/null") {
381 enable_extras $module;
386 # Generate SSL certificates.
388 Would you like to generate a self-signed SSL certificate now? This certificate
389 can be used for testing but <|BOLD should not|> be used on a production network.
391 Note: you can get a <|BOLD free|> CA-signed certificate from Let's Encrypt. See
392 https://letsencrypt.org/getting-started/ for more details.
395 if (<$RealDir/src/modules/m_ssl_*.cpp>) {
396 if (prompt_bool $interactive, $question, $interactive) {
397 create_directory CONFIGURE_DIRECTORY, 0750 or print_error "unable to create ${\CONFIGURE_DIRECTORY}: $!";
398 system './tools/genssl', 'auto', CONFIGURE_DIRECTORY;
400 my @pems = <${\CONFIGURE_DIRECTORY}/{cert,csr,dhparams,key}.pem>;
402 The following self-signed files were previously generated and will be installed
403 when you run Make. Do you want to delete them?
405 * ${\join "\n * ", @pems}
407 if (@pems && prompt_bool $interactive, $question, 0) {
411 } elsif (!defined $opt_disable_auto_extras) {
412 print_warning <<"EOM";
413 You are building without enabling any SSL modules. This is not
414 recommended as SSL greatly enhances the security and privacy of your IRC server
415 and in a future version will be <|BOLD required|> for linking servers.
417 Please read the following documentation pages on how to enable SSL support:
419 GnuTLS (recommended): https://docs.inspircd.org/3/modules/ssl_gnutls
420 mbedTLS: https://docs.inspircd.org/3/modules/ssl_mbedtls
421 OpenSSL: https://docs.inspircd.org/3/modules/ssl_openssl
425 # Cache the distribution label so that its not lost when --update is run.
426 $config{DISTRIBUTION} = $opt_distribution_label if $opt_distribution_label;
428 write_configure_cache %config;
429 parse_templates \%config, \%compiler, \%version;
431 print_format <<"EOM";
433 Configuration is complete! You have chosen to build with the following settings:
436 <|GREEN Binary:|> $config{CXX}
437 <|GREEN Name:|> $compiler{NAME}
438 <|GREEN Version:|> $compiler{VERSION}
440 <|GREEN Extra Modules:|>
443 for my $file (<$RealDir/src/modules/m_*>) {
444 my $module = basename $file, '.cpp';
445 say " * $module" if -l $file;
449 push @makeargs, "-C${\abs2rel $RealDir}" unless getcwd eq $RealDir;
450 push @makeargs, "-j${\(get_cpu_count() + 1)}";
452 print_format <<"EOM";
455 <|GREEN Base:|> $config{BASE_DIR}
456 <|GREEN Binary:|> $config{BINARY_DIR}
457 <|GREEN Config:|> $config{CONFIG_DIR}
458 <|GREEN Data:|> $config{DATA_DIR}
459 <|GREEN Example:|> $config{EXAMPLE_DIR}
460 <|GREEN Log:|> $config{LOG_DIR}
461 <|GREEN Manual:|> $config{MANUAL_DIR}
462 <|GREEN Module:|> $config{MODULE_DIR}
463 <|GREEN Script:|> $config{SCRIPT_DIR}
465 <|GREEN Execution Group:|> $config{GROUP} ($config{GID})
466 <|GREEN Execution User:|> $config{USER} ($config{UID})
467 <|GREEN Socket Engine:|> $config{SOCKETENGINE}
469 To build with these settings run '<|GREEN make ${\join ' ', @makeargs} install|>' now.
473 # Routine to list out the extra/ modules that have been enabled.
474 # Note: when getting any filenames out and comparing, it's important to lc it if the
475 # file system is not case-sensitive (== Epoc, MacOS, OS/2 (incl DOS/DJGPP), VMS, Win32
476 # (incl NetWare, Symbian)). Cygwin may or may not be case-sensitive, depending on
477 # configuration, however, File::Spec does not currently tell us (it assumes Unix behavior).
481 my $srcdir = File::Spec->catdir("src", "modules");
482 my $abs_srcdir = File::Spec->rel2abs($srcdir);
485 opendir $dd, File::Spec->catdir($abs_srcdir, "extra") or die (File::Spec->catdir($abs_srcdir, "extra") . ": $!\n");
486 my @extras = map { File::Spec->case_tolerant() ? lc($_) : $_ } (readdir($dd));
489 opendir $dd, $abs_srcdir or die "$abs_srcdir: $!\n";
490 my @sources = map { File::Spec->case_tolerant() ? lc($_) : $_ } (readdir($dd));
493 my $maxlen = (sort { $b <=> $a } (map {length($_)} (@extras)))[0];
495 EXTRA: for my $extra (@extras) {
496 next if (File::Spec->curdir() eq $extra || File::Spec->updir() eq $extra);
497 my $abs_extra = File::Spec->catfile($abs_srcdir, "extra", $extra);
498 my $abs_source = File::Spec->catfile($abs_srcdir, $extra);
499 next unless ($extra =~ m/\.(cpp|h)$/ || (-d $abs_extra)); # C++ Source/Header, or directory
500 if (-l $abs_source) {
501 # Symlink, is it in the right place?
502 my $targ = readlink($abs_source);
503 my $abs_targ = File::Spec->rel2abs($targ, $abs_srcdir);
504 if ($abs_targ eq $abs_extra) {
505 $extras{$extra} = "\e[32;1menabled\e[0m";
507 $extras{$extra} = sprintf("\e[31;1mwrong symlink target (%s)\e[0m", $abs_targ);
509 } elsif (-e $abs_source) {
510 my ($devext, $inoext) = stat($abs_extra);
511 my ($devsrc, $inosrc, undef, $lnksrc) = stat($abs_source);
513 if ($devsrc == $devext && $inosrc == $inoext) {
514 $extras{$extra} = "\e[32;1menabled\e[0m";
516 $extras{$extra} = sprintf("\e[31;1mwrong hardlink target (%d:%d)\e[0m", $devsrc, $inosrc);
519 open my $extfd, "<", $abs_extra;
520 open my $srcfd, "<", $abs_source;
522 if (scalar(<$extfd>) eq scalar(<$srcfd>)) {
523 $extras{$extra} = "\e[32;1menabled\e[0m";
525 $extras{$extra} = sprintf("\e[31;1mout of synch (re-copy)\e[0m");
529 $extras{$extra} = "\e[33;1mdisabled\e[0m";
532 # Now let's add dependency info
533 for my $extra (keys(%extras)) {
534 next unless $extras{$extra} =~ m/enabled/; # only process enabled extras.
535 my $abs_extra = File::Spec->catfile($abs_srcdir, "extra", $extra);
536 my @deps = split /\s+/, get_directive($abs_extra, 'ModDep', '');
537 for my $dep (@deps) {
538 if (exists($extras{$dep})) {
539 my $ref = \$extras{$dep}; # Take reference.
540 if ($$ref !~ m/needed by/) {
541 # First dependency found.
542 if ($$ref =~ m/enabled/) {
543 $$ref .= " (needed by \e[32;1m$extra\e[0m";
545 $$ref =~ s/\e\[.*?m//g; # Strip out previous coloring. Will be set in bold+red+blink later.
546 $$ref .= " (needed by \e[0;32;1;5m$extra\e[0;31;1;5m";
549 if ($$ref =~ m/enabled/) {
550 $$ref .= ", \e[32;1m$extra\e[0m";
552 $$ref .= ", \e[0;32;1;5m$extra\e[0;31;1;5m";
558 for my $extra (sort {$a cmp $b} keys(%extras)) {
559 my $text = $extras{$extra};
560 if ($text =~ m/needed by/ && $text !~ m/enabled/) {
561 printf "\e[31;1;5m%-*s = %s%s\e[0m\n", $maxlen, $extra, $text, ($text =~ m/needed by/ ? ")" : "");
563 printf "%-*s = %s%s\n", $maxlen, $extra, $text, ($text =~ m/needed by/ ? "\e[0m)" : "");
566 return keys(%extras) if wantarray; # Can be used by manage_extras.
569 sub enable_extras(@) {
570 my $moduledir = catdir $RealDir, 'src', 'modules';
571 my $extradir = catdir $moduledir, 'extra';
574 my $shortname = $extra =~ s/(?:^m_|\.cpp$)//gr;
575 my $extrafile = "m_$shortname.cpp";
577 my $extrapath = catfile $extradir, $extrafile;
578 if (!-f $extrapath) {
579 print_error "<|GREEN $extra|> is not an extra module!";
582 my $modulepath = catfile $moduledir, $extrafile;
583 if (-l $modulepath) {
584 if (readlink($modulepath) ne $extrapath) {
585 unlink $modulepath; # Remove the dead symlink.
587 next; # Module is already enabled.
591 if (-e $modulepath) {
592 print_error "unable to symlink <|GREEN ${\abs2rel $modulepath}|> to <|GREEN ${\abs2rel $extrapath}|>: the target exists and is not a symlink.";
594 print_format "Enabling the <|GREEN $shortname|> module ...\n";
595 symlink $extrapath, $modulepath or print_error "unable to symlink <|GREEN ${\abs2rel $modulepath}|> to <|GREEN ${\abs2rel $extrapath}|>: $!";
600 sub disable_extras(@) {
601 my $moduledir = catdir $RealDir, 'src', 'modules';
602 my $extradir = catdir $moduledir, 'extra';
605 my $shortname = $extra =~ s/(?:^m_|\.cpp$)//gr;
606 my $extrafile = "m_$shortname.cpp";
608 my $modulepath = catfile $moduledir, $extrafile;
609 my $extrapath = catfile $extradir, $extrafile;
610 if (!-e $modulepath && !-e $extrapath) {
611 print_error "the <|GREEN $shortname|> module does not exist!";
612 } elsif (!-e $modulepath && -e $extrapath) {
613 print_error "the <|GREEN $shortname|> module is not currently enabled!";
614 } elsif ((-e $modulepath && !-e $extrapath) || !-l $modulepath) {
615 print_error "the <|GREEN $shortname|> module is not an extra module!";
617 print_format "Disabling the <|GREEN $shortname|> module ...\n";
618 unlink $modulepath or print_error "unable to unlink <|GREEN $extrapath|>: $!";