1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
|
# Copyright 1999-2025 Gentoo Authors
# Distributed under the terms of the GNU General Public License v2
# @ECLASS: nginx-module.eclass
# @MAINTAINER:
# Zurab Kvachadze <zurabid2016@gmail.com>
# @AUTHOR:
# Zurab Kvachadze <zurabid2016@gmail.com>
# @SUPPORTED_EAPIS: 8
# @BLURB: Provides a common set of functions for building NGINX's dynamic modules
# @DESCRIPTION:
# The nginx-module.eclass automates configuring, building and installing NGINX's
# dynamic modules. Using this eclass is as simple as calling 'inherit nginx-module'.
# This eclass automatically adds dependencies on www-servers/nginx. Henceforth,
# the terms 'package' and 'module' will be used interchangeably to refer to a
# consumer of nginx-module.eclass.
#
# If a part of package's functionality depends on NGINX configuration (e.g. HMAC
# generation support depending on http_ssl module being present), the
# corresponding module's 'config' code should be changed so that the functionality
# in question is either (1) unconditionally enabled/disabled or (2) can be
# toggled with a USE flag. That is, an ebuild author should deduce whether a
# package actually depends on a particular module or on the underlying
# libraries/APIs. In the example HMAC case, the module actually requires
# libcrypto, not the http_ssl module, so the ebuild code reflects this by
# patching the module's 'config' file and depending on dev-libs/openssl directly
# using the ngx_mod_append_libs() function.
#
# If the module makes use of the ngx_devel_kit (NDK) or any other NGINX
# module, there are two approaches.
#
# If these dependencies are not USE-conditional ones, they should be specified
# in the NGINX_MOD_LINK_MODULES array before inheriting the eclass. This way,
# the dependencies added to {,R}DEPEND variables. Additionally, the package is
# linked to shared objects of the specified dependencies. See the variable
# description for details.
#
# If the dependencies are USE-conditional, they should be specified as
# usual in the relevant *DEPEND variable(s). Then, before
# nginx-module_src_configure() is called, the dependencies should be linked to by
# calling the ngx_mod_link_module() function. See the function description for
# more information.
#
# nginx-module.eclass also supports tests provided by the Test::Nginx Perl
# module. To enable them, set NGINX_MOD_OPENRESTY_TESTS to a non-empty value
# prior to inheriting the eclass and, if necessary, populate the
# NGINX_MOD_TEST_LOAD_ORDER variable. All the packages specified in
# NGINX_MOD_TEST_LOAD_ORDER are added to BDEPEND.
#
# The code below presents one of the ways the nginx-module.eclass might be used.
#
# Example usage:
# @CODE
# # This module depends on ngx_devel_kit and ngx-lua-module.
# NGINX_MOD_LINK_MODULES=(
# www-nginx/ngx_devel_kit www-nginx/ngx-lua-module
# )
#
# # Tests utilise Test::Nginx.
# NGINX_MOD_OPENRESTY_TESTS=1
# # We require ngx-lua-module and ngx-echo for tests, but ngx-echo should
# # be loaded first. Otherwise, some tests break.
# NGINX_MOD_TEST_LOAD_ORDER=(
# www-nginx/ngx-echo
# www-nginx/ngx-lua-module
# )
# inherit nginx-module
#
# IUSE="iconv"
#
# DEPEND="iconv? ( www-nginx/ngx-iconv )"
# RDEPEND="${DEPEND}"
#
# src_configure() {
# if use iconv; Then
# ngx_mod_link_module "www-nginx/ngx-iconv"
# ...
# fi
#
# nginx-module_src_configure
# }
# @CODE
case ${EAPI} in
8) ;;
*) die "${ECLASS}: EAPI ${EAPI:-0} not supported" ;;
esac
if [[ -z ${_NGINX_MODULE_ECLASS} ]]; then
_NGINX_MODULE_ECLASS=1
inherit edo flag-o-matic toolchain-funcs
#-----> Generic helper functions <-----
# @FUNCTION: econf_ngx
# @USAGE: [<args>...]
# @DESCRIPTION:
# Call ./configure, passing the supplied arguments.
# The NGINX's build system consists of many helper scripts, which are executed
# relative to the working directory. Therefore, the function only supports
# executing the configure script from the current working directory. This
# function also checks whether the script is executable. If any of the above
# conditions are not satisfied, the function aborts the build process with
# 'die'. It also fails if the script itself exits with a non-zero exit code,
# unless the function is called with 'nonfatal'.
# If running ./configure is required, this function should be called.
econf_ngx() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ ! -x ./configure ]] &&
die "./configure is not present in the current working directory or is not executable"
if [[ $1 == --help ]]; then
# For some reason, NGINX ./configure returns 1 if it is used with the
# '--help' argument.
#
# Executing this without edo gets rid of the "Failed to run" message.
./configure "$@"
return
fi
edo ./configure "$@"
}
# @FUNCTION: ngx_mod_pkg_to_sonames
# @USAGE: <package name>
# @RETURN: Null-delimited list of basenames of shared objects corresponding to the supplied package.
# @DESCRIPTION:
# Takes one argument and prints a null-delimited list of basenames of shared
# objects corresponding to the supplied package.
#
# The mapping between a package and shared objects goes as follows.
#
# 1. The package is stripped of category, yielding a plain
# package name.
#
# 2. The plain package name is then used to lookup into the internal
# associative array NGX_MOD_TO_SONAME. If the lookup fails, the build is
# aborted with 'die'. 'nonfatal' might be used to make the error to find
# shared objects non-fatal.
#
# 3. The obtained shared objects are printed to the stdout as a
# null-delimited list.
#
# Example usage:
# @CODE
# # Obtain shared objects provided by www-nginx/ngx-lua-module.
# mypkg=www-nginx/ngx-lua-module
# mapfile -d '' lua-sonames < <(ngx_mod_pkg_to_sonames "${mypkg}")
# @CODE
ngx_mod_pkg_to_sonames() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ $# -ne 1 ]] && die "${FUNCNAME[0]} must receive exactly one argument"
local pkg="$1"
local dep_sonames
# Strip '${CATEGORY}/' from '${CATEGORY}/${PN}'.
local entry="${pkg#*/}"
# Obtain the name of the shared object of the package with PN '${entry}' by
# looking at the corresponding subscript of the NGX_MOD_TO_SONAME array.
#
# For example, entry=ngx-lua-module yields
# entry="${NGX_MOD_TO_SONAME[ngx-lua-module]}"
# which yields
# entry="ngx_http_lua_module"
entry="${NGX_MOD_TO_SONAME[${entry}]}"
[[ -z ${entry} ]] &&
die -n "No shared objects found for the dependency ${pkg}. Please file a bug"
# Read comma-separated shared object names into the 'dep_sonames' array.
IFS=, read -ra dep_sonames <<< "${entry}"
# Append '.so' to the end of each array member.
dep_sonames=( "${dep_sonames[@]/%/.so}" )
# Print null-delimited list of shared objects' basenames to stdout.
printf "%s\0" "${dep_sonames[@]}"
}
# @FUNCTION: ngx_mod_append_libs
# @USAGE: [<linker flags>...]
# @DESCRIPTION:
# Adds the passed arguments to the list of flags used for the linking the
# module's shared objects. Flags may be of any form accepted by linker.
# See the nginx_src_install() function in nginx.eclass for more details.
#
# This function has no effect after nginx-module_src_configure() has been
# called.
#
# Example usage:
# @CODE
# ngx_mod_append_libs "-L/usr/$(get_libdir)/nginx/modules" \
# "$("$(tc-getPKG_CONFIG)" --libs luajit)"
# @CODE
ngx_mod_append_libs() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ $# -eq 0 ]] && return 0
export _NGINX_GENTOO_MOD_LIBS="${_NGINX_GENTOO_MOD_LIBS} $*"
}
# @FUNCTION: ngx_mod_setup_link_modules
# @DESCRIPTION:
# Adds necessary linker arguments for linking to other NGINX modules' share
# objects installed in /usr/$(get_libdir)/nginx/modules by calling
# ngx_mod_append_libs(). This function takes no arguments.
#
# This function is called internally by the ngx_mod_link_module() function.
# ngx_mod_setup_link_modules() keeps track whether it has already been called,
# doing nothing if it is called again after the first execution.
ngx_mod_setup_link_modules() {
debug-print-function "${FUNCNAME[0]}"
# Check whether this function has already been called.
[[ -n ${_NGX_MOD_SETUP_LINK_CALLED} ]] && return 0
declare -g -r _NGX_MOD_SETUP_LINK_CALLED=1
local moddir
moddir="${EPREFIX}/usr/$(get_libdir)/nginx/modules"
# Add 'moddir' to the list of directories search by linker.
ngx_mod_append_libs "-L${moddir}"
# The string passed to ngx_mod_append_libs undergoes the following
# transformations by NGINX build system (the str variable denotes the
# original string and 'modname' represents the name of the current module):
# 0. Given the value of 'str':
# $ echo "${str}"
# -Wl,-rpath,'\''\$\${ORIGIN}'\''
#
# 1. In auto/module, line 93:
# eval ${modname}_libs=\'$str\'.
# yields
# modname_libs='-Wl,-rpath,'\''\$\${ORIGIN}'\'''
# which can be logically separated into
# (a) '-Wl,-rpath,'
# ^
# |
# The first original single quote (\'$str\')
# ^
# |
# This one
# (b) \' (backslash-escaped semicolon)
# (c) '\$\${ORIGIN}'
# (d) \'
# (e) ''
# ^
# |
# The last original single quote (\'$str\')
# ^
# |
# This one
# To preserve the string we add literal ' and \' so that the
# ORIGIN part does not get expanded.
# - (a) expands to
# -Wl,-rpath
# - (b) expands to
# '
# - (c) expands to
# \$\${ORIGIN}
# - (d) expands to
# '
# - (e) expands to nothing
# Thus, after evaluating, the value of modname_libs is the
# following.
# $ echo "${modname_libs}"
# -Wl,-rpath,'\$\${ORIGIN}'
#
# 2. In auto/make, line 507:
# eval eval ngx_module_libs="\\\"\$${modname}_libs\\\"".
# The expansion of parameters and double quotes produces the
# following.
# \"$modname_libs\"
# The first outermost eval obtains the contents of the
# ${modname}_libs variable and encloses them in double quotes,
# yielding:
# eval ngx_module_libs="-Wl,-rpath,'\$\${ORIGIN}'"
# The second innermost eval expands the double-quoted string,
# produced by the first eval, stripping backslashes from '$'. The
# value of 'ngx_module_libs' is therefore:
# $ echo "${ngx_module_libs}"
# -Wl,-rpath,'$${ORIGIN}'
#
# 3. ngx_module_libs's contents are added to the Makefile. make
# expands $var variable references, double dollar is used to
# suppress the expanding. Thus, the following is passed to the
# shell:
# -Wl,-rpath,'${ORIGIN}'
#
# 4. Finally, shell expands the single quotes, yielding literal:
# -Wl,-rpath,${ORIGIN}
ngx_mod_append_libs "-Wl,-rpath,'\''"'\$\${ORIGIN}'"'\''"
}
# @FUNCTION: ngx_mod_link_module
# @USAGE: <package name>
# @DESCRIPTION:
# Add the required linker flags to link to the shared objects provided by the
# package passed as the argument. This function automatically calls
# ngx_mod_setup_link_modules(), if it has not been called. If the specified
# package provides more than one shared object, all of the shared objects are
# linked to. As ngx_mod_append_libs(), this function has no effect after
# nginx-module_src_configure has been called.
#
# This function uses the ngx_mod_pkg_to_sonames() function under the hood to map
# package names to shared objects. If there are no predefined mappings for the
# selected package, the NGX_MOD_TO_SONAME associative array may be changed
# manually, as presented in the following code excerpt.
#
# @CODE
# NGX_MOD_TO_SONAME+=(
# [www-nginx/ngx-pkg-name]="the_corresponding_soname_without_dot_so_suffix"
# )
# @CODE
#
# See the default value of NGX_MOD_TO_SONAME for examples.
#
# This function might be used to implement USE-conditional dependency on another
# NGINX module. See the code snipped below for an example of such usage.
#
# Example usage:
# @CODE
# inherit nginx-module
#
# DEPEND="iconv? ( www-nginx/ngx-iconv )"
# RDEPEND="${DEPEND}"
#
# src_configure() {
# if use iconv; then
# ngx_mod_link_module "www-nginx/ngx-iconv"
# ...
# fi
#
# nginx-module_src_configure
# }
# @CODE
ngx_mod_link_module() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ $# -ne 1 ]] && die "${FUNCNAME[0]} must receive exactly one argument"
[[ -z ${_NGX_MOD_SETUP_LINK_CALLED} ]] && ngx_mod_setup_link_modules
# Obtain the shared objects names of the module we want to link to (yes,
# there might be more than one shared object for a given NGINX module).
local -a sonames
mapfile -d '' sonames < <(ngx_mod_pkg_to_sonames "$1")
# Prepend '-l:' to each shared object name. The colon instructs the linker
# to link to the given name literally; i.e. '-lmylib' will look for
# 'libmylib.so', while '-l:mylib' will look for 'mylib'.
ngx_mod_append_libs "${sonames[@]/#/-l:}"
}
#-----> ebuild-defined variables <-----
# @ECLASS_VARIABLE: NGINX_MOD_S
# @DESCRIPTION:
# Holds the path to the module sources directory, used in the
# nginx-module_src_configure() phase function. If unset at the time of inherit,
# defaults to ${S}.
: "${NGINX_MOD_S=${S}}"
# The ${S} variable is set to the path of the directory where the actual build
# will be performed. In this directory, symbolic links to NGINX's build system
# and NGINX's headers are created by the nginx-module_src_unpack() phase
# function.
S="${WORKDIR}/nginx"
# @ECLASS_VARIABLE: NGINX_MOD_SHARED_OBJECTS
# @OUTPUT_VARIABLE
# @DESCRIPTION:
# An array containing the basenames of all compiled shared objects (with the
# extension ".so"). For some modules, may consist of more than one shared
# object.
#
# This variable is set in the nginx-module_src_compile() function. Its contents
# are undefined before that.
#
# Example value:
# @CODE
# ngx_http_lua_module.so
# @CODE
# @ECLASS_VARIABLE: NGX_MOD_TO_SONAME
# @DESCRIPTION:
# An associative array that maps NGINX module package names to their shared
# object names. For example, 'ngx-lua-module' is mapped to
# 'ngx_http_lua_module'. The shared objects are specified without the '.so'
# suffix. May be changed/appended to at any time by an ebuild to override/add
# shared object mappings.
declare -g -A NGX_MOD_TO_SONAME+=(
[ngx_devel_kit]=ndk_http_module
[ngx-lua-module]=ngx_http_lua_module
[ngx-xss]=ngx_http_xss_filter_module
[ngx-echo]=ngx_http_echo_module
[ngx-memc]=ngx_http_memc_module
[ngx-eval]=ngx_http_eval_module
[ngx-set-misc]=ngx_http_set_misc_module
[ngx-headers-more]=ngx_http_headers_more_filter_module
[ngx-iconv]=ngx_http_iconv_module
[ngx-srcache]=ngx_http_srcache_filter_module
[ngx-lua-upstream]=ngx_http_lua_upstream_module
)
# @ECLASS_VARIABLE: NGINX_MOD_LINK_MODULES
# @PRE_INHERIT
# @DEFAULT_UNSET
# @DESCRIPTION:
# Set to package names of the NGINX module dependencies of this module. This
# array must be set prior to inheriting the eclass.
#
# All the modules specified in this array are added to DEPEND and RDEPEND. This
# might be disabled by setting NGINX_MOD_OVERRIDE_LINK_DEPEND to a non-empty
# value. Additionally, the listed modules are added to the NEEDED sections of
# the current module's shared objects, i.e. the current module is dynamically
# linked to the shared objects corresponding to packages specified in
# NGINX_MOD_LINK_MODULES.
#
# Each element of the array specifies a dependency of an ebuild. An entry
# consists of a category followed by a package name: ${CATEGORY}/${PN}.
#
# To determine the shared object corresponding to an entry, the eclass looks up
# the respective mapping, specified in the NGX_MOD_TO_SONAME array (see the
# array description for more information). If no match is found, the build is
# aborted with 'die'.
#
# Example usage:
# @CODE
# # This module depends on both NDK and ngx-lua-module.
# NGINX_MOD_LINK_MODULES=(
# www-nginx/ngx_devel_kit
# www-nginx/ngx-lua-module
# )
# inherit nginx-module
# @CODE
# @ECLASS_VARIABLE: NGINX_MOD_OVERRIDE_LINK_DEPEND
# @PRE_INHERIT
# @DEFAULT_UNSET
# @DESCRIPTION:
# Set to a non-empty value prior to inheriting the eclass so that the modules
# listed in NGINX_MOD_LINK_MODULES are not automatically added to DEPEND and
# RDEPEND.
# @ECLASS_VARIABLE: NGINX_MOD_OPENRESTY_TESTS
# @PRE_INHERIT
# @DEFAULT_UNSET
# @DESCRIPTION:
# Set to non-empty value to enable prior to inheriting the eclass to enable the
# tests via the Test::Nginx (dev-perl/Test-Nginx) testing scaffold. See the
# description of the NGINX_MOD_TEST_LOAD_ORDER variable for more details.
# @ECLASS_VARIABLE: NGINX_MOD_TEST_DIR
# @DESCRIPTION:
# Set to directory containing tests relative to NGINX_MOD_S. If
# NGINX_MOD_OPENRESTY_TESTS is not set, has no effect. Defaults to "t".
: "${NGINX_MOD_TEST_DIR:=t}"
# @ECLASS_VARIABLE: NGINX_MOD_TEST_LOAD_ORDER
# @PRE_INHERIT
# @DESCRIPTION:
# If NGINX_MOD_OPENRESTY_TESTS is set to a non-empty value, this array specifies
# simultaneously the test dependencies of the current module and, since NGINX is
# sensitive to the order of module loading, their load order. As a special
# workaround, the current module could also be specified as an entry in order to
# force a specific load order. If the current module is not listed in this
# array, it is loaded first, before its test dependencies.
#
# All the modules specified in this array, barring the current module, are added
# to test BDEPEND. This behaviour may be disabled by setting the
# NGINX_MOD_OVERRIDE_TEST_BDEPEND variable to a non-empty value.
#
# The format of each entry is the same as in the NGINX_MOD_LINK_MODULES
# variable. See its description for details.
#
# The shared object names obtained from each entry are then used to populate the
# TEST_NGINX_LOAD_MODULES environment variable. TEST_NGINX_LOAD_MODULES
# instructs Test::Nginx in what order and which shared objects should be loaded
# during tests.
#
# This array must be set prior to inheriting the eclass. If
# NGINX_MOD_OPENRESTY_TESTS is not set, this variable has no effect.
#
# Example:
# @CODE
# NGINX_MOD_OPENRESTY_TESTS=1
# NGINX_MOD_TEST_LOAD_ORDER=(
# www-nginx/ngx-lua-module www-nginx/ngx-eval
# www-nginx/{my-cool-module,my-another-module}
# )
# @CODE
# @ECLASS_VARIABLE: NGINX_MOD_OVERRIDE_TEST_BDEPEND
# @PRE_INHERIT
# @DEFAULT_UNSET
# @DESCRIPTION:
# Set to a non-empty value prior to inheriting the eclass so that the modules
# listed in NGINX_MOD_TEST_LOAD_ORDER are not automatically added to BDEPEND.
# Has no effect if either NGINX_MOD_OPENRESTY_TESTS or NGINX_MOD_TEST_LOAD_ORDER
# are not set.
#-----> *DEPEND stuff <-----
# As per upstream documentation, modules must be rebuilt with each NGINX
# upgrade.
DEPEND="
www-servers/nginx:=[modules(-)]
"
BDEPEND="${DEPEND}"
RDEPEND="${DEPEND}"
if [[ -z ${NGINX_MOD_OVERRIDE_LINK_DEPEND} &&
${#NGINX_MOD_LINK_MODULES[@]} -gt 0 ]];
then
DEPEND+=" ${NGINX_MOD_LINK_MODULES[*]}"
RDEPEND+=" ${NGINX_MOD_LINK_MODULES[*]}"
fi
#-----> Tests setup <-----
# @FUNCTION: _ngx_mod_set_test_env
# @INTERNAL
# @DESCRIPTION:
# Sets global variables like IUSE and BDEPEND for tests.
_ngx_mod_set_test_env() {
IUSE="test"
RESTRICT="!test? ( test )"
BDEPEND+=" test? (
dev-perl/Test-Nginx
)
"
if [[ -z ${NGINX_MOD_OVERRIDE_TEST_BDEPEND} &&
${#NGINX_MOD_TEST_LOAD_ORDER[@]} -gt 0 ]];
then
local entry
local -a moddep=
for entry in "${NGINX_MOD_TEST_LOAD_ORDER[@]}"; do
# If the current entry is equal to the current package, do not add
# it to BDEPEND.
[[ ${entry} == "${CATEGORY}/${PN}" ]] && continue
moddep+=( "${entry}" )
done
if [[ ${#moddep[@]} -gt 0 ]]; then
BDEPEND+="
test? (
${moddep[*]}
)
"
fi
fi
}
[[ -n ${NGINX_MOD_OPENRESTY_TESTS} ]] && _ngx_mod_set_test_env
unset -f _ngx_mod_set_test_env
#-----> Phase functions <-----
# @FUNCTION: nginx-module_src_unpack
# @DESCRIPTION:
# Unpacks the sources and sets up the build directory in S=${WORKDIR}/nginx.
# Creates the following symbolic links (to not copy the files over):
# - '${S}/src' -> '/usr/include/nginx',
# - '${S}/auto' -> '/usr/src/nginx/auto',
# - '${S}/configure' -> '/usr/src/nginx/configure'.
# For additional information, see the nginx.eclass source, namely the
# nginx_src_install() function.
nginx-module_src_unpack() {
debug-print-function "${FUNCNAME[0]}" "$@"
default_src_unpack
mkdir nginx || die "mkdir failed"
ln -s "${BROOT}/usr/src/nginx/configure" nginx/configure || die "ln failed"
ln -s "${BROOT}/usr/src/nginx/auto" nginx/auto || die "ln failed"
ln -s "${ESYSROOT}/usr/include/nginx" nginx/src || die "ln failed"
}
# @FUNCTION: nginx-module_src_prepare
# @DESCRIPTION:
# Patches module's initialisation code so that any module's preprocessor
# definitions appear in the separate '__ngx_gentoo_mod_config.h' file inside the
# 'build' directory. This function also makes module's "config" script clear
# whatever content build/ngx_auto_config.h may have at the time of invocation.
# Then, default_src_prepare() is called.
nginx-module_src_prepare() {
debug-print-function "${FUNCNAME[0]}" "$@"
pushd "${NGINX_MOD_S}" >/dev/null || die "pushd failed"
# Since NGINX does not guarantee ABI or API stability, we utilise
# preprocessor macros that were used to compile NGINX itself, to build third
# party modules. As such, we do not want for the dummy preprocessor macros
# produced by NGINX build system during module compilation to leak into the
# building environment. However, we do need to "capture" preprocessor macros
# set by the module itself, so we are required to somehow get these
# separately.
#
# To achieve that, the following sed script inserts ': >
# build/ngx_auto_config.h' line at the start of a module's 'config' shell
# script which gets sourced by NGINX build system midway during
# configuration. It has an effect of truncating the file containing NGINX
# preprocessor macros. This results in the file containing only module's
# macros at the end of the module's configuration.
#
# The following command renames the file with module's preprocessor macros
# to __ngx_gentoo_mod_config.h to be later merged with the system NGINX
# header into the actual header used during compilation. Due to the fact
# that executing the config shell script is not the last thing that NGINX
# build system does during configuration, we can not simply rename the
# header after the whole configuration, as it may contain other preprocessor
# macros than only the module's ones.
sed -i -e '1i\' -e ': > build/ngx_auto_config.h' config ||
die "sed failed"
echo 'mv build/ngx_auto_config.h build/__ngx_gentoo_mod_config.h' \
>> config || die "echo failed"
default_src_prepare
popd >/dev/null || die "popd failed"
}
# @FUNCTION: nginx-module_src_configure
# @DESCRIPTION:
# Configures the dynamic module by calling NGINX's ./configure script.
# Custom flags can be supplied as arguments to the function, taking precedence
# over eclass's flags.
# This assembles ngx_auto_config.h from the system ngx_auto_config.h and
# __ngx_gentoo_mod_config.h (see nginx-module_src_prepare()), and
# ngx_auto_headers.h from the system ngx_auto_headers.h.
# Also, sets environment variables and appends necessary libraries if
# NGINX_MOD_LINK_MODULES is set.
nginx-module_src_configure() {
debug-print-function "${FUNCNAME[0]}" "$@"
local ngx_mod_flags
ngx_mod_flags=(
--with-cc="$(tc-getCC)"
--with-cpp="$(tc-getCPP)"
# The '-isystem' flag is used instead of '-I', so as for the installed
# (system) modules' headers to be of lower priority than the headers of
# the currently built module. This only affects the modules that both
# come with and install their own headers, e.g. ngx_devel_kit.
--with-cc-opt="-isystem src/modules"
--with-ld-opt="${LDFLAGS}"
--builddir=build
--add-dynamic-module="${NGINX_MOD_S}"
)
# NGINX build system adds directories under src/ to the include path based
# on the specified configuration flags. Since nginx.eclass does not
# save/restore the configuration flags, we have to add the directories to
# the include path manually.
# The src/os is added automatically by the auto/unix script and the
# src/modules directory is included by the '--with-cc-opt' configuration
# flag.
append-cflags "$(find -H src -mindepth 1 -type d \! \( \( -path 'src/os' -o \
-path 'src/modules' \) -prune \) -printf '-I %p ')"
# Some NGINX modules that depend on ngx_devel_kit (NDK) check whether the
# NDK_SRCS variable is non-empty and error out if it is empty or not
# defined. ngx_devel_kit sets this variable during its build but due to the
# fact that we build modules separately, i.e. the dependant module is not
# build alongside NDK, the variable is not exported in the environment and
# the module halts the build.
#
# For all the modules that I have seen, the ones that inspect this variable
# only check whether NDK_SRCS is non-empty, they do not compare its contents
# nor alter the variable in any way. Therefore, setting NDK_SRCS to a dummy
# value works around the build failures for the plugins that do check the
# variable and, subsequently, has no effect on the modules that do not
# depend on NDK at all or do not check the variable.
#
# This variable is mainly checked by modules developed by OpenResty.
export NDK_SRCS="ndk.c"
# Add the required linking flags required for the modules specified in the
# NGINX_MOD_LINK_MODULES array.
if [[ ${#NGINX_MOD_LINK_MODULES[@]} -gt 0 ]]; then
local mod
for mod in "${NGINX_MOD_LINK_MODULES[@]}"; do
ngx_mod_link_module "${mod}"
done
fi
eval "local -a EXTRA_ECONF=( ${EXTRA_ECONF} )"
# Setting the required environment variable to skip the unnecessary
# execution of certain scripts (see nginx_src_install() in nginx.eclass).
_NGINX_GENTOO_SKIP_PHASES=1 econf_ngx \
"${ngx_mod_flags[@]}" \
"$@" \
"${EXTRA_ECONF[@]}"
cat "${ESYSROOT}/usr/include/nginx/ngx_auto_config.h" \
build/__ngx_gentoo_mod_config.h > build/ngx_auto_config.h ||
die "cat failed"
cp "${ESYSROOT}/usr/include/nginx/ngx_auto_headers.h" build ||
die "cp failed"
}
# @FUNCTION: nginx-module_src_compile
# @DESCRIPTION:
# Compiles the module(s) by calling "make modules" and fills the
# NGINX_MOD_SHARED_OBJECTS array.
nginx-module_src_compile() {
debug-print-function "${FUNCNAME[0]}" "$@"
emake modules
# Save the basenames of all the compiled modules into the
# NGINX_MOD_SHARED_OBJECTS array.
mapfile -d '' NGINX_MOD_SHARED_OBJECTS < \
<(find -H "${S}/build" -maxdepth 1 -type f -name '*.so' -printf '%P\0')
}
# @FUNCTION: nginx-module_src_test
# @DESCRIPTION:
# If NGINX_MOD_OPENRESTY_TESTS is set to a non-empty value, tests the compiled
# module using Test::Nginx (dev-perl/Test-Nginx).
nginx-module_src_test() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ -z ${NGINX_MOD_OPENRESTY_TESTS} ]] && return 0
# The TEST_NGINX_LOAD_MODULES variable holds the space-separated paths of
# modules that should be loaded during testing. The variable is set (in
# order) to the shared object names of the built modules and, then, to
# shared objects of the dependant modules. Doing this the other way around
# introduces some test failures for modules like ngx-eval.
local -x TEST_NGINX_LOAD_MODULES=
local -a dep_sonames pkg_sonames
local cur_pkg_in_load_order=
# The system module directory.
local moddir
moddir="${BROOT}/usr/$(get_libdir)/nginx/modules"
[[ ${#NGINX_MOD_SHARED_OBJECTS[@]} -eq 0 ]] &&
die "No shared objects found for the currently built module"
# Prepend each member of the NGINX_MOD_SHARED_OBJECTS array with
# '${S}/build/' and save the array into pkg_sonames.
pkg_sonames=( "${NGINX_MOD_SHARED_OBJECTS[@]/#/${S}/build/}" )
local pkg
for pkg in "${NGINX_MOD_TEST_LOAD_ORDER[@]}"; do
# If the entry is the current package, use the shared objects saved in
# '${pkg_sonames[@]}' and set the 'cur_pkg_in_load_order' variable.
if [[ ${pkg} == "${CATEGORY}/${PN}" ]]; then
TEST_NGINX_LOAD_MODULES+=" ${pkg_sonames[*]}"
cur_pkg_in_load_order=1
continue
fi
# Save the shared objects names into the dep_sonames array.
mapfile -d '' dep_sonames < <(ngx_mod_pkg_to_sonames "${pkg}")
# Prepend each array member with '${moddir}/' (see above) to obtain the
# absolute path to the shared object.
dep_sonames=( "${dep_sonames[@]/#/${moddir}/}" )
# Add the shared objects' paths to the TEST_NGINX_LOAD_MODULES
# environment variable.
TEST_NGINX_LOAD_MODULES+=" ${dep_sonames[*]}"
done
unset pkg
# If the current package is not specified in NGINX_MOD_TEST_LOAD_ORDER, load
# it before its test dependencies.
if [[ -z ${cur_pkg_in_load_order} ]]; then
TEST_NGINX_LOAD_MODULES="${pkg_sonames[*]} ${TEST_NGINX_LOAD_MODULES}"
fi
pushd "${NGINX_MOD_S}" >/dev/null || die "pushd failed"
# If NGINX_MOD_LINK_MODULES is non-empty, meaning the current module is
# linked to another module in moddir, set LD_LIBRARY_PATH to the module's
# directory so that the dynamic loader can find shared objects we depend on.
[[ ${#NGINX_MOD_LINK_MODULES[@]} -gt 0 ]] &&
local -x LD_LIBRARY_PATH="${moddir}"
# Tests break when run in parallel.
TEST_NGINX_SERVROOT="${T}/servroot" \
edo prove -j1 -I. -r ./"${NGINX_MOD_TEST_DIR}"
popd >/dev/null || die "popd failed"
}
# @FUNCTION: nginx-module_src_install
# @DESCRIPTION:
# Installs the compiled module(s) into /usr/${libdir}/nginx/modules.
nginx-module_src_install() {
debug-print-function "${FUNCNAME[0]}" "$@"
insinto "/usr/$(get_libdir)/nginx/modules"
doins build/*.so
}
# @FUNCTION: nginx-module_pkg_postinst
# @DESCRIPTION:
# Shows the instructions on how to enable and use the just-compiled module.
nginx-module_pkg_postinst() {
debug-print-function "${FUNCNAME[0]}" "$@"
# ngx_devel_kit is an SDK, it does not need to be enabled manually.
[[ ${PN} == ngx_devel_kit ]] && return 0
local mod
elog "${PN} has been compiled."
elog ""
elog "To utilise the module, add the following line(s) to your NGINX"
elog "configuration file, which by default is \"${EROOT}/etc/nginx/nginx.conf\"."
elog ""
for mod in "${NGINX_MOD_SHARED_OBJECTS[@]}"; do
elog " load_module modules/${mod};"
done
}
fi
EXPORT_FUNCTIONS src_unpack src_prepare src_configure src_compile src_test \
src_install pkg_postinst
|