Port ::Replicated from Moose to Moo+Type::Tiny
[dbsrgits/DBIx-Class.git] / lib / DBIx / Class / Optional / Dependencies.pm
CommitLineData
8057ed01 1package DBIx::Class::Optional::Dependencies;
2
cb551b07 3### This may look crazy, but it in fact tangibly ( by 50(!)% ) shortens
4# the skip-test time when everything requested is unavailable
bd510251 5BEGIN {
6 if ( $ENV{RELEASE_TESTING} ) {
7 require warnings and warnings->import;
8 require strict and strict->import;
9 }
10}
cb551b07 11
12sub croak {
13 require Carp;
14 Carp::croak(@_);
15};
16###
fb39747c 17
18# NO EXTERNAL NON-5.8.1 CORE DEPENDENCIES EVER (e.g. C::A::G)
8057ed01 19# This module is to be loaded by Makefile.PM on a pristine system
20
f6b26571 21# POD is generated automatically by calling _gen_pod from the
22# Makefile.PL in $AUTHOR mode
23
60256113 24# *DELIBERATELY* not making a group for these - they must disappear
25# forever as optdeps in the first place
2b48ebff 26my $moose_basic = {
68de9438 27 'Moose' => '0.98',
28 'MooseX::Types' => '0.21',
d401ab6b 29 'MooseX::Types::LoadableClass' => '0.011',
68de9438 30};
31
31c31b8d 32my $dbic_reqs = {
60256113 33
34 # NOTE: the rationale for 2 JSON::Any versions is that
35 # we need the newer only to work around JSON::XS, which
36 # itself is an optional dep
37 _json_any => {
38 req => {
39 'JSON::Any' => '1.23',
40 },
41 },
42
43 _json_xs_compatible_json_any => {
44 req => {
45 'JSON::Any' => '1.31',
46 },
47 },
48
1e1cc55e 49 _types_common => {
50 req => {
51 'Type::Utils' => '0',
52 'Type::Library' => '0',
53 'Types::Standard' => '0',
54 'Types::LoadableClass' => '0',
55 },
56 },
57
7d2772a4 58 # a common placeholder for engines with IC::DT support based off DT::F::S
2baba3d9 59 _icdt_strptime_based => {
60 augment => {
61 icdt => {
62 req => {
63 'DateTime::Format::Strptime' => '1.2',
64 },
65 },
66 }
67 },
7d2772a4 68
60256113 69 _rdbms_generic_odbc => {
70 req => {
71 'DBD::ODBC' => 0,
72 }
73 },
74
75 _rdbms_generic_ado => {
76 req => {
77 'DBD::ADO' => 0,
78 }
79 },
80
461e818a 81 # must list any dep used by adhoc testing
82 # this prevents the "skips due to forgotten deps" issue
83 test_adhoc => {
84 req => {
83eef562 85 'Class::DBI::Plugin::DeepAbstractSearch' => '0',
86 'Class::DBI' => '3.000005',
461e818a 87 'Date::Simple' => '3.03',
88 'YAML' => '0',
89 'Class::Unload' => '0.07',
83eef562 90 'Time::Piece' => '0',
91 'Time::Piece::MySQL' => '0',
461e818a 92 },
93 },
94
8057ed01 95 replicated => {
1e1cc55e 96 include => '_types_common',
f6b26571 97 pod => {
98 title => 'Storage::Replicated',
99 desc => 'Modules required for L<DBIx::Class::Storage::DBI::Replicated>',
100 },
8057ed01 101 },
102
68de9438 103 test_replicated => {
e163f484 104 include => 'replicated',
68de9438 105 },
106
8057ed01 107 admin => {
60256113 108 include => '_json_any',
2b48ebff 109 req => {
e163f484 110 %$moose_basic,
e163f484 111 'MooseX::Types::Path::Class' => '0.05',
112 'MooseX::Types::JSON' => '0.02',
ebcd0e4f 113 },
114 pod => {
115 title => 'DBIx::Class::Admin',
116 desc => 'Modules required for the DBIx::Class administrative library',
117 },
118 },
119
a4a02f15 120 admin_script => {
e163f484 121 include => 'admin',
ebcd0e4f 122 req => {
e163f484 123 'Getopt::Long::Descriptive' => '0.081',
124 'Text::CSV' => '1.16',
2b48ebff 125 },
e144415f 126 pod => {
127 title => 'dbicadmin',
128 desc => 'Modules required for the CLI DBIx::Class interface dbicadmin',
129 },
8057ed01 130 },
131
132 deploy => {
f6b26571 133 req => {
08ac7648 134 'SQL::Translator' => '0.11018',
f6b26571 135 },
136 pod => {
137 title => 'Storage::DBI::deploy()',
5529838f 138 desc => 'Modules required for L<DBIx::Class::Storage::DBI/deployment_statements> and L<DBIx::Class::Schema/deploy>',
f6b26571 139 },
8057ed01 140 },
141
2baba3d9 142 icdt => {
143 req => {
144 'DateTime' => '0.55',
145 },
146 pod => {
147 title => 'InflateColumn::DateTime support',
148 desc =>
149 'Modules required for L<DBIx::Class::InflateColumn::DateTime>. '
150 . 'Note that this group does not require much on its own, but '
151 . 'instead is augmented by various RDBMS-specific groups. See the '
152 . 'documentation of each C<rbms_*> group for details',
153 },
154 },
155
c7d50a7d 156 id_shortener => {
e163f484 157 req => {
158 'Math::BigInt' => '1.80',
159 'Math::Base36' => '0.07',
160 },
c7d50a7d 161 },
162
83eef562 163 cdbicompat => {
164 req => {
165 'Class::Data::Inheritable' => '0',
166 'Class::Trigger' => '0',
167 'DBIx::ContextualFetch' => '0',
168 'Clone' => '0.32',
169 },
170 pod => {
171 title => 'DBIx::Class::CDBICompat support',
172 desc => 'Modules required for L<DBIx::Class::CDBICompat>'
173 },
174 },
175
a109c954 176 test_pod => {
f6b26571 177 req => {
cbb19edf 178 'Test::Pod' => '1.42',
a109c954 179 },
cb551b07 180 release_testing_mandatory => 1,
a109c954 181 },
182
183 test_podcoverage => {
184 req => {
f6b26571 185 'Test::Pod::Coverage' => '1.08',
186 'Pod::Coverage' => '0.20',
a109c954 187 },
cb551b07 188 release_testing_mandatory => 1,
a109c954 189 },
190
ffce4b65 191 test_whitespace => {
a109c954 192 req => {
8273e845 193 'Test::EOL' => '1.0',
ffce4b65 194 'Test::NoTabs' => '0.9',
f6b26571 195 },
cb551b07 196 release_testing_mandatory => 1,
8057ed01 197 },
198
4a233f30 199 test_strictures => {
200 req => {
1e1cc55e 201 'Test::Strict' => '0.24',
4a233f30 202 },
cb551b07 203 release_testing_mandatory => 1,
4a233f30 204 },
205
2a2a7b23 206 test_prettydebug => {
60256113 207 include => '_json_any',
2a2a7b23 208 },
209
99503754 210 test_admin_script => {
60256113 211 include => [qw( admin_script _json_xs_compatible_json_any )],
99503754 212 req => {
99503754 213 'JSON' => 0,
be855469 214 'JSON::PP' => 0,
215 'Cpanel::JSON::XS' => 0,
99503754 216 'JSON::XS' => 0,
217 $^O eq 'MSWin32'
218 # for t/admin/10script.t
219 ? ('Win32::ShellQuote' => 0)
220 # DWIW does not compile (./configure even) on win32
221 : ('JSON::DWIW' => 0 )
222 ,
223 }
224 },
225
556c4fe6 226 test_leaks_heavy => {
f6b26571 227 req => {
556c4fe6 228 'Class::MethodCache' => '0.02',
229 'PadWalker' => '1.06',
a109c954 230 },
231 },
f6b26571 232
8057ed01 233
be68095d 234 # this is just for completeness as SQLite
235 # is a core dep of DBIC for testing
236 rdbms_sqlite => {
237 req => {
e163f484 238 'DBD::SQLite' => 0,
be68095d 239 },
240 pod => {
241 title => 'SQLite support',
242 desc => 'Modules required to connect to SQLite',
243 },
2baba3d9 244 augment => {
245 icdt => {
246 req => {
247 'DateTime::Format::SQLite' => '0',
248 },
249 },
250 },
251 },
252
253 # centralize the specification, as we have ICDT tests which can
254 # test the full behavior of RDBMS-specific ICDT on top of bare SQLite
255 # not _-prefixed so that it will show up under req_group_list
256 icdt_pg => {
257 augment => {
258 icdt => {
259 req => {
260 'DateTime::Format::Pg' => '0.16004',
261 },
262 },
263 },
be68095d 264 },
265
266 rdbms_pg => {
2baba3d9 267 include => 'icdt_pg',
be68095d 268 req => {
a4fc1239 269 # when changing this list make sure to adjust xt/optional_deps.t
e163f484 270 'DBD::Pg' => 0,
be68095d 271 },
272 pod => {
273 title => 'PostgreSQL support',
274 desc => 'Modules required to connect to PostgreSQL',
275 },
276 },
277
7d2772a4 278 _rdbms_mssql_common => {
279 include => '_icdt_strptime_based',
280 },
281
be68095d 282 rdbms_mssql_odbc => {
7d2772a4 283 include => [qw( _rdbms_generic_odbc _rdbms_mssql_common )],
be68095d 284 pod => {
285 title => 'MSSQL support via DBD::ODBC',
286 desc => 'Modules required to connect to MSSQL via DBD::ODBC',
287 },
288 },
289
290 rdbms_mssql_sybase => {
7d2772a4 291 include => '_rdbms_mssql_common',
be68095d 292 req => {
e163f484 293 'DBD::Sybase' => 0,
be68095d 294 },
295 pod => {
296 title => 'MSSQL support via DBD::Sybase',
56dca25f 297 desc => 'Modules required to connect to MSSQL via DBD::Sybase',
298 },
299 },
300
301 rdbms_mssql_ado => {
7d2772a4 302 include => [qw( _rdbms_generic_ado _rdbms_mssql_common )],
56dca25f 303 pod => {
304 title => 'MSSQL support via DBD::ADO (Windows only)',
305 desc => 'Modules required to connect to MSSQL via DBD::ADO. This particular DBD is available on Windows only',
be68095d 306 },
307 },
308
7d2772a4 309 _rdbms_msaccess_common => {
310 include => '_icdt_strptime_based',
311 },
312
726c8f65 313 rdbms_msaccess_odbc => {
7d2772a4 314 include => [qw( _rdbms_generic_odbc _rdbms_msaccess_common )],
726c8f65 315 pod => {
316 title => 'MS Access support via DBD::ODBC',
317 desc => 'Modules required to connect to MS Access via DBD::ODBC',
318 },
319 },
320
321 rdbms_msaccess_ado => {
7d2772a4 322 include => [qw( _rdbms_generic_ado _rdbms_msaccess_common )],
726c8f65 323 pod => {
324 title => 'MS Access support via DBD::ADO (Windows only)',
325 desc => 'Modules required to connect to MS Access via DBD::ADO. This particular DBD is available on Windows only',
326 },
327 },
328
2baba3d9 329 # centralize the specification, as we have ICDT tests which can
330 # test the full behavior of RDBMS-specific ICDT on top of bare SQLite
331 # not _-prefixed so that it will show up under req_group_list
332 icdt_mysql => {
333 augment => {
334 icdt => {
335 req => {
336 'DateTime::Format::MySQL' => '0',
337 },
338 },
339 },
340 },
341
be68095d 342 rdbms_mysql => {
2baba3d9 343 include => 'icdt_mysql',
be68095d 344 req => {
e163f484 345 'DBD::mysql' => 0,
be68095d 346 },
347 pod => {
348 title => 'MySQL support',
349 desc => 'Modules required to connect to MySQL',
350 },
351 },
352
353 rdbms_oracle => {
e163f484 354 include => 'id_shortener',
be68095d 355 req => {
e163f484 356 'DBD::Oracle' => 0,
be68095d 357 },
358 pod => {
359 title => 'Oracle support',
360 desc => 'Modules required to connect to Oracle',
361 },
2baba3d9 362 augment => {
363 icdt => {
364 req => {
365 'DateTime::Format::Oracle' => '0',
366 },
367 },
368 },
be68095d 369 },
370
371 rdbms_ase => {
7d2772a4 372 include => '_icdt_strptime_based',
be68095d 373 req => {
e163f484 374 'DBD::Sybase' => 0,
be68095d 375 },
376 pod => {
377 title => 'Sybase ASE support',
378 desc => 'Modules required to connect to Sybase ASE',
379 },
380 },
381
7d2772a4 382 _rdbms_db2_common => {
2baba3d9 383 augment => {
384 icdt => {
385 req => {
386 'DateTime::Format::DB2' => '0',
387 },
388 },
389 },
7d2772a4 390 },
391
be68095d 392 rdbms_db2 => {
7d2772a4 393 include => '_rdbms_db2_common',
be68095d 394 req => {
e163f484 395 'DBD::DB2' => 0,
be68095d 396 },
397 pod => {
398 title => 'DB2 support',
399 desc => 'Modules required to connect to DB2',
400 },
401 },
402
199fbc45 403 rdbms_db2_400 => {
7d2772a4 404 include => [qw( _rdbms_generic_odbc _rdbms_db2_common )],
199fbc45 405 pod => {
406 title => 'DB2 on AS/400 support',
407 desc => 'Modules required to connect to DB2 on AS/400',
408 },
409 },
410
411 rdbms_informix => {
7d2772a4 412 include => '_icdt_strptime_based',
199fbc45 413 req => {
e163f484 414 'DBD::Informix' => 0,
199fbc45 415 },
416 pod => {
417 title => 'Informix support',
418 desc => 'Modules required to connect to Informix',
419 },
8273e845 420 },
199fbc45 421
7d2772a4 422 _rdbms_sqlanywhere_common => {
423 inclide => '_icdt_strptime_based',
424 },
425
199fbc45 426 rdbms_sqlanywhere => {
7d2772a4 427 include => '_rdbms_sqlanywhere_common',
199fbc45 428 req => {
e163f484 429 'DBD::SQLAnywhere' => 0,
199fbc45 430 },
431 pod => {
432 title => 'SQLAnywhere support',
433 desc => 'Modules required to connect to SQLAnywhere',
434 },
8273e845 435 },
199fbc45 436
437 rdbms_sqlanywhere_odbc => {
7d2772a4 438 include => [qw( _rdbms_generic_odbc _rdbms_sqlanywhere_common )],
199fbc45 439 pod => {
440 title => 'SQLAnywhere support via DBD::ODBC',
441 desc => 'Modules required to connect to SQLAnywhere via DBD::ODBC',
442 },
8273e845 443 },
199fbc45 444
7d2772a4 445 _rdbms_firebird_common => {
446 include => '_icdt_strptime_based',
447 },
448
199fbc45 449 rdbms_firebird => {
7d2772a4 450 include => '_rdbms_firebird_common',
199fbc45 451 req => {
e163f484 452 'DBD::Firebird' => 0,
199fbc45 453 },
454 pod => {
455 title => 'Firebird support',
456 desc => 'Modules required to connect to Firebird',
457 },
8273e845 458 },
199fbc45 459
460 rdbms_firebird_interbase => {
7d2772a4 461 include => '_rdbms_firebird_common',
199fbc45 462 req => {
e163f484 463 'DBD::InterBase' => 0,
199fbc45 464 },
465 pod => {
466 title => 'Firebird support via DBD::InterBase',
467 desc => 'Modules required to connect to Firebird via DBD::InterBase',
468 },
8273e845 469 },
199fbc45 470
471 rdbms_firebird_odbc => {
7d2772a4 472 include => [qw( _rdbms_generic_odbc _rdbms_firebird_common )],
199fbc45 473 pod => {
474 title => 'Firebird support via DBD::ODBC',
475 desc => 'Modules required to connect to Firebird via DBD::ODBC',
476 },
8273e845 477 },
199fbc45 478
7975645b 479 test_rdbms_sqlite => {
480 include => 'rdbms_sqlite',
481 req => {
482 ###
483 ### IMPORTANT - do not raise this dependency
484 ### even though many bugfixes are present in newer versions, the general DBIC
485 ### rule is to bend over backwards for available DBDs (given upgrading them is
486 ### often *not* easy or even possible)
487 ###
488 'DBD::SQLite' => '1.29',
489 },
490 },
491
68de9438 492 test_rdbms_pg => {
e163f484 493 include => 'rdbms_pg',
e3a7746c 494 env => [
495 DBICTEST_PG_DSN => 1,
496 DBICTEST_PG_USER => 0,
497 DBICTEST_PG_PASS => 0,
498 ],
f6b26571 499 req => {
e3a7746c 500 # the order does matter because the rdbms support group might require
501 # a different version that the test group
502 #
503 # when changing this list make sure to adjust xt/optional_deps.t
e3a7746c 504 'DBD::Pg' => '2.009002', # specific version to test bytea
f6b26571 505 },
8057ed01 506 },
507
afae8507 508 test_rdbms_mssql_odbc => {
e163f484 509 include => 'rdbms_mssql_odbc',
e3a7746c 510 env => [
511 DBICTEST_MSSQL_ODBC_DSN => 1,
512 DBICTEST_MSSQL_ODBC_USER => 0,
513 DBICTEST_MSSQL_ODBC_PASS => 0,
514 ],
afae8507 515 },
516
56dca25f 517 test_rdbms_mssql_ado => {
e163f484 518 include => 'rdbms_mssql_ado',
e3a7746c 519 env => [
520 DBICTEST_MSSQL_ADO_DSN => 1,
521 DBICTEST_MSSQL_ADO_USER => 0,
522 DBICTEST_MSSQL_ADO_PASS => 0,
523 ],
56dca25f 524 },
525
afae8507 526 test_rdbms_mssql_sybase => {
e163f484 527 include => 'rdbms_mssql_sybase',
e3a7746c 528 env => [
529 DBICTEST_MSSQL_DSN => 1,
530 DBICTEST_MSSQL_USER => 0,
531 DBICTEST_MSSQL_PASS => 0,
532 ],
afae8507 533 },
534
726c8f65 535 test_rdbms_msaccess_odbc => {
4b9c54b5 536 include => 'rdbms_msaccess_odbc',
e3a7746c 537 env => [
538 DBICTEST_MSACCESS_ODBC_DSN => 1,
539 DBICTEST_MSACCESS_ODBC_USER => 0,
540 DBICTEST_MSACCESS_ODBC_PASS => 0,
541 ],
726c8f65 542 req => {
e3a7746c 543 'Data::GUID' => '0',
726c8f65 544 },
545 },
546
547 test_rdbms_msaccess_ado => {
4b9c54b5 548 include => 'rdbms_msaccess_ado',
e3a7746c 549 env => [
550 DBICTEST_MSACCESS_ADO_DSN => 1,
551 DBICTEST_MSACCESS_ADO_USER => 0,
552 DBICTEST_MSACCESS_ADO_PASS => 0,
553 ],
726c8f65 554 req => {
e3a7746c 555 'Data::GUID' => 0,
726c8f65 556 },
557 },
558
68de9438 559 test_rdbms_mysql => {
e163f484 560 include => 'rdbms_mysql',
e3a7746c 561 env => [
562 DBICTEST_MYSQL_DSN => 1,
563 DBICTEST_MYSQL_USER => 0,
564 DBICTEST_MYSQL_PASS => 0,
565 ],
8057ed01 566 },
567
68de9438 568 test_rdbms_oracle => {
e163f484 569 include => 'rdbms_oracle',
e3a7746c 570 env => [
571 DBICTEST_ORA_DSN => 1,
572 DBICTEST_ORA_USER => 0,
573 DBICTEST_ORA_PASS => 0,
574 ],
f6b26571 575 req => {
e3a7746c 576 'DBD::Oracle' => '1.24',
f6b26571 577 },
8057ed01 578 },
579
68de9438 580 test_rdbms_ase => {
e163f484 581 include => 'rdbms_ase',
e3a7746c 582 env => [
583 DBICTEST_SYBASE_DSN => 1,
584 DBICTEST_SYBASE_USER => 0,
585 DBICTEST_SYBASE_PASS => 0,
586 ],
8057ed01 587 },
588
68de9438 589 test_rdbms_db2 => {
e163f484 590 include => 'rdbms_db2',
e3a7746c 591 env => [
592 DBICTEST_DB2_DSN => 1,
593 DBICTEST_DB2_USER => 0,
594 DBICTEST_DB2_PASS => 0,
595 ],
f58a165c 596 },
597
199fbc45 598 test_rdbms_db2_400 => {
e163f484 599 include => 'rdbms_db2_400',
e3a7746c 600 env => [
601 DBICTEST_DB2_400_DSN => 1,
602 DBICTEST_DB2_400_USER => 0,
603 DBICTEST_DB2_400_PASS => 0,
604 ],
199fbc45 605 },
606
607 test_rdbms_informix => {
e163f484 608 include => 'rdbms_informix',
e3a7746c 609 env => [
610 DBICTEST_INFORMIX_DSN => 1,
611 DBICTEST_INFORMIX_USER => 0,
612 DBICTEST_INFORMIX_PASS => 0,
613 ],
199fbc45 614 },
615
616 test_rdbms_sqlanywhere => {
e163f484 617 include => 'rdbms_sqlanywhere',
e3a7746c 618 env => [
619 DBICTEST_SQLANYWHERE_DSN => 1,
620 DBICTEST_SQLANYWHERE_USER => 0,
621 DBICTEST_SQLANYWHERE_PASS => 0,
622 ],
199fbc45 623 },
624
625 test_rdbms_sqlanywhere_odbc => {
e163f484 626 include => 'rdbms_sqlanywhere_odbc',
e3a7746c 627 env => [
628 DBICTEST_SQLANYWHERE_ODBC_DSN => 1,
629 DBICTEST_SQLANYWHERE_ODBC_USER => 0,
630 DBICTEST_SQLANYWHERE_ODBC_PASS => 0,
631 ],
199fbc45 632 },
633
634 test_rdbms_firebird => {
e163f484 635 include => 'rdbms_firebird',
e3a7746c 636 env => [
637 DBICTEST_FIREBIRD_DSN => 1,
638 DBICTEST_FIREBIRD_USER => 0,
639 DBICTEST_FIREBIRD_PASS => 0,
640 ],
199fbc45 641 },
642
643 test_rdbms_firebird_interbase => {
e163f484 644 include => 'rdbms_firebird_interbase',
e3a7746c 645 env => [
646 DBICTEST_FIREBIRD_INTERBASE_DSN => 1,
647 DBICTEST_FIREBIRD_INTERBASE_USER => 0,
648 DBICTEST_FIREBIRD_INTERBASE_PASS => 0,
649 ],
199fbc45 650 },
651
e02b39b4 652 test_rdbms_firebird_odbc => {
e163f484 653 include => 'rdbms_firebird_odbc',
e3a7746c 654 env => [
655 DBICTEST_FIREBIRD_ODBC_DSN => 1,
656 DBICTEST_FIREBIRD_ODBC_USER => 0,
657 DBICTEST_FIREBIRD_ODBC_PASS => 0,
658 ],
e02b39b4 659 },
660
42168332 661 test_memcached => {
e3a7746c 662 env => [
663 DBICTEST_MEMCACHED => 1,
664 ],
42168332 665 req => {
e3a7746c 666 'Cache::Memcached' => 0,
42168332 667 },
668 },
669
344f1f52 670 dist_dir => {
60256113 671 # we need to run the dbicadmin so we can self-generate its POD
672 # also we do not want surprises in case JSON::XS is in the path
673 # so make sure we get an always-working JSON::Any
674 include => [qw( admin_script _json_xs_compatible_json_any )],
344f1f52 675 req => {
b81d8515 676 'ExtUtils::MakeMaker' => '6.64',
06fc872a 677 'Pod::Inherit' => '0.91',
678 },
344f1f52 679 },
680
681 dist_upload => {
682 req => {
683 'CPAN::Uploader' => '0.103001',
684 },
685 },
8057ed01 686};
687
f6b26571 688
3ca0dcf0 689
690### Public API
691
afb8fc52 692sub import {
693 my $class = shift;
694
695 if (@_) {
696
697 my $action = shift;
698
699 if ($action eq '-die_without') {
700 my $err;
701 {
702 local $@;
703 eval { $class->die_unless_req_ok_for(\@_); 1 }
704 or $err = $@;
705 }
706 die "\n$err\n" if $err;
707 }
708 elsif ($action eq '-list_missing') {
709 print $class->modreq_missing_for(\@_);
710 print "\n";
711 exit 0;
712 }
cb551b07 713 elsif ($action eq '-skip_all_without') {
714
715 # sanity check - make sure ->current_test is 0 and no plan has been declared
716 do {
717 local $@;
718 defined eval {
719 Test::Builder->new->current_test
720 or
721 Test::Builder->new->has_plan
722 };
723 } and croak("Unable to invoke -skip_all_without after testing has started");
724
725 if ( my $missing = $class->req_missing_for(\@_) ) {
726
727 die ("\nMandatory requirements not satisfied during release-testing: $missing\n\n")
728 if $ENV{RELEASE_TESTING} and $class->_groups_to_reqs(\@_)->{release_testing_mandatory};
729
730 print "1..0 # SKIP requirements not satisfied: $missing\n";
731 exit 0;
732 }
733 }
afb8fc52 734 elsif ($action =~ /^-/) {
735 croak "Unknown import-time action '$action'";
736 }
737 else {
738 croak "$class is not an exporter, unable to import '$action'";
739 }
740 }
741
742 1;
743}
744
745sub unimport {
746 croak( __PACKAGE__ . " does not implement unimport" );
747}
748
3ca0dcf0 749# OO for (mistakenly considered) ease of extensibility, not due to any need to
750# carry state of any sort. This API is currently used outside, so leave as-is.
751# FIXME - make sure to not propagate this further if module is extracted as a
752# standalone library - keep the stupidity to a DBIC-secific shim!
753#
fb39747c 754sub req_list_for {
d1bd396f 755 shift->_groups_to_reqs(shift)->{effective_modreqs};
e3a7746c 756}
757
758sub modreq_list_for {
d1bd396f 759 shift->_groups_to_reqs(shift)->{modreqs};
fb39747c 760}
761
3ca0dcf0 762sub req_group_list {
31c31b8d 763 +{ map
764 { $_ => $_[0]->_groups_to_reqs($_) }
60256113 765 grep { $_ !~ /^_/ } keys %$dbic_reqs
31c31b8d 766 }
3ca0dcf0 767}
fb39747c 768
d1bd396f 769sub req_errorlist_for { shift->modreq_errorlist_for(shift) } # deprecated
5ffa39c7 770sub modreq_errorlist_for {
d1bd396f 771 my ($self, $groups) = @_;
772 $self->_errorlist_for_modreqs( $self->_groups_to_reqs($groups)->{modreqs} );
344f1f52 773}
774
fb39747c 775sub req_ok_for {
d1bd396f 776 shift->req_missing_for(shift) ? 0 : 1;
e3a7746c 777}
778
779sub req_missing_for {
d1bd396f 780 my ($self, $groups) = @_;
e3a7746c 781
d1bd396f 782 my $reqs = $self->_groups_to_reqs($groups);
250d9e55 783
784 my $mods_missing = $reqs->{missing_envvars}
785 ? $self->_list_physically_missing_modules( $reqs->{modreqs} )
786 : $self->modreq_missing_for($groups)
787 ;
e3a7746c 788
789 return '' if
790 ! $mods_missing
791 and
792 ! $reqs->{missing_envvars}
31c31b8d 793 ;
e3a7746c 794
795 my @res = $mods_missing || ();
796
d1f653cf 797 push @res, 'the following group(s) of environment variables: ' . join ' and ', sort map
e3a7746c 798 { __envvar_group_desc($_) }
799 @{$reqs->{missing_envvars}}
800 if $reqs->{missing_envvars};
801
802 return (
803 ( join ' as well as ', @res )
804 .
805 ( $reqs->{modreqs_fully_documented} ? " (see @{[ ref $self || $self ]} documentation for details)" : '' ),
806 );
fb39747c 807}
808
e3a7746c 809sub modreq_missing_for {
d1bd396f 810 my ($self, $groups) = @_;
fb39747c 811
d1bd396f 812 my $reqs = $self->_groups_to_reqs($groups);
e3a7746c 813 my $modreq_errors = $self->_errorlist_for_modreqs($reqs->{modreqs})
814 or return '';
fb39747c 815
e3a7746c 816 join ' ', map
817 { $reqs->{modreqs}{$_} ? qq("$_~>=$reqs->{modreqs}{$_}") : $_ }
818 sort { lc($a) cmp lc($b) } keys %$modreq_errors
31c31b8d 819 ;
fb39747c 820}
821
461e818a 822my $tb;
823sub skip_without {
824 my ($self, $groups) = @_;
825
826 $tb ||= do { local $@; eval { Test::Builder->new } }
827 or croak "Calling skip_without() before loading Test::Builder makes no sense";
828
829 if ( my $err = $self->req_missing_for($groups) ) {
830 my ($fn, $ln) = (caller(0))[1,2];
831 $tb->skip("block in $fn around line $ln requires $err");
832 local $^W = 0;
833 last SKIP;
834 }
835
836 1;
837}
838
3ca0dcf0 839sub die_unless_req_ok_for {
d1bd396f 840 if (my $err = shift->req_missing_for(shift) ) {
e3a7746c 841 die "Unable to continue due to missing requirements: $err\n";
31c31b8d 842 }
fb39747c 843}
844
3ca0dcf0 845
846
e3a7746c 847### Private functions
848
849# potentially shorten group desc
850sub __envvar_group_desc {
851 my @envs = @{$_[0]};
852
853 my (@res, $last_prefix);
854 while (my $ev = shift @envs) {
855 my ($pref, $sep, $suff) = split / ([\_\-]) (?= [^\_\-]+ \z )/x, $ev;
856
857 if ( defined $sep and ($last_prefix||'') eq $pref ) {
858 push @res, "...${sep}${suff}"
859 }
860 else {
861 push @res, $ev;
862 }
863
864 $last_prefix = $pref if $sep;
865 }
866
867 join '/', @res;
868}
869
461e818a 870my $groupname_re = qr/ [A-Z_a-z][0-9A-Z_a-z]* /x;
871my $modname_re = qr/ [A-Z_a-z] [0-9A-Z_a-z]* (?:::[0-9A-Z_a-z]+)* /x;
872my $modver_re = qr/ [0-9]+ (?: \. [0-9]+ )? /x;
873
a3f8bd01 874# Expand includes from a random group in a specific order:
875# nonvariable groups first, then their includes, then the variable groups,
876# then their includes.
877# This allows reliably marking the rest of the mod reqs as variable (this is
878# also why variable includes are currently not allowed)
879sub __expand_includes {
880 my ($groups, $seen) = @_;
881
882 # !! DIFFERENT !! behavior and return depending on invocation mode
883 # (easier to recurse this way)
884 my $is_toplevel = $seen
885 ? 0
886 : !! ($seen = {})
887 ;
3ca0dcf0 888
a3f8bd01 889 my ($res_per_type, $missing_envvars);
d8799bab 890
a3f8bd01 891 # breadth-first evaluation, with non-variable includes on top
892 for my $g (@$groups) {
d8799bab 893
a3f8bd01 894 croak "Invalid requirement group name '$g': only ascii alphanumerics and _ are allowed"
461e818a 895 if $g !~ qr/ \A $groupname_re \z/x;
e163f484 896
a3f8bd01 897 my $r = $dbic_reqs->{$g}
898 or croak "Requirement group '$g' is not defined";
d8799bab 899
a3f8bd01 900 # always do this check *before* the $seen check
901 croak "Group '$g' with variable effective_modreqs can not be specified as an 'include'"
902 if ( $r->{env} and ! $is_toplevel );
d8799bab 903
a3f8bd01 904 next if $seen->{$g}++;
31c31b8d 905
a3f8bd01 906 my $req_type = 'static';
d8799bab 907
a3f8bd01 908 if ( my @e = @{$r->{env}||[]} ) {
e3a7746c 909
a3f8bd01 910 croak "Unexpected 'env' attribute under group '$g' (only allowed in test_* groups)"
911 unless $g =~ /^test_/;
e3a7746c 912
a3f8bd01 913 croak "Unexpected *odd* list in 'env' under group '$g'"
e3a7746c 914 if @e % 2;
915
e3a7746c 916 # deconstruct the whole thing
a3f8bd01 917 my (@group_envnames_list, $some_envs_required, $some_required_missing);
e3a7746c 918 while (@e) {
919 push @group_envnames_list, my $envname = shift @e;
920
921 # env required or not
922 next unless shift @e;
923
924 $some_envs_required ||= 1;
925
a3f8bd01 926 $some_required_missing ||= (
e3a7746c 927 ! defined $ENV{$envname}
928 or
929 ! length $ENV{$envname}
930 );
931 }
932
a3f8bd01 933 croak "None of the envvars in group '$g' declared as required, making the requirement moot"
e3a7746c 934 unless $some_envs_required;
935
a3f8bd01 936 if ($some_required_missing) {
937 push @{$missing_envvars->{$g}}, \@group_envnames_list;
938 $req_type = 'variable';
939 }
940 }
941
942 push @{$res_per_type->{"base_${req_type}"}}, $g;
943
944 if (my $i = $dbic_reqs->{$g}{include}) {
945 $i = [ $i ] unless ref $i eq 'ARRAY';
946
947 croak "Malformed 'include' for group '$g': must be another existing group name or arrayref of existing group names"
948 unless @$i;
949
950 push @{$res_per_type->{"incs_${req_type}"}}, @$i;
e3a7746c 951 }
a3f8bd01 952 }
953
954 my @ret = map {
955 @{ $res_per_type->{"base_${_}"} || [] },
956 ( $res_per_type->{"incs_${_}"} ? __expand_includes( $res_per_type->{"incs_${_}"}, $seen ) : () ),
957 } qw(static variable);
958
959 return ! $is_toplevel ? @ret : do {
960 my $rv = {};
961 $rv->{$_} = {
962 idx => 1 + keys %$rv,
963 missing_envvars => $missing_envvars->{$_},
964 } for @ret;
2baba3d9 965 $rv->{$_}{user_requested} = 1 for @$groups;
a3f8bd01 966 $rv;
967 };
968}
969
970### Private OO API
971our %req_unavailability_cache;
972
973# this method is just a lister and envvar/metadata checker - it does not try to load anything
974sub _groups_to_reqs {
461e818a 975 my ($self, $want) = @_;
a3f8bd01 976
461e818a 977 $want = [ $want || () ]
978 unless ref $want eq 'ARRAY';
a3f8bd01 979
980 croak "@{[ (caller(1))[3] ]}() expects a requirement group name or arrayref of group names"
461e818a 981 unless @$want;
e3a7746c 982
a3f8bd01 983 my $ret = {
984 modreqs => {},
985 modreqs_fully_documented => 1,
986 };
e163f484 987
461e818a 988 my $groups;
989 for my $piece (@$want) {
990 if ($piece =~ qr/ \A $groupname_re \z /x) {
991 push @$groups, $piece;
992 }
993 elsif ( my ($mod, $ver) = $piece =~ qr/ \A ($modname_re) \>\= ($modver_re) \z /x ) {
994 croak "Ad hoc module specification lists '$mod' twice"
995 if exists $ret->{modreqs}{$mod};
996
997 croak "Ad hoc module specification '${mod} >= $ver' (or greater) not listed in the test_adhoc optdep group" if (
998 ! defined $dbic_reqs->{test_adhoc}{req}{$mod}
999 or
1000 $dbic_reqs->{test_adhoc}{req}{$mod} < $ver
1001 );
1002
1003 $ret->{modreqs}{$mod} = $ver;
1004 $ret->{modreqs_fully_documented} = 0;
1005 }
1006 else {
1007 croak "Unsupported argument '$piece' supplied to @{[ (caller(1))[3] ]}()"
1008 }
1009 }
1010
a3f8bd01 1011 my $all_groups = __expand_includes($groups);
e163f484 1012
2baba3d9 1013 # pre-assemble list of augmentations, perform basic sanity checks
1014 # Note that below we *DO NOT* respect the source/target reationship, but
1015 # instead always default to augment the "later" group
1016 # This is done so that the "stable/variable" boundary keeps working as
1017 # expected
1018 my $augmentations;
1019 for my $requesting_group (keys %$all_groups) {
1020 if (my $ag = $dbic_reqs->{$requesting_group}{augment}) {
1021 for my $target_group (keys %$ag) {
1022
1023 croak "Group '$requesting_group' claims to augment a non-existent group '$target_group'"
1024 unless $dbic_reqs->{$target_group};
1025
1026 croak "Augmentation combined with variable effective_modreqs currently unsupported for group '$requesting_group'"
1027 if $dbic_reqs->{$requesting_group}{env};
1028
1029 croak "Augmentation of group '$target_group' with variable effective_modreqs unsupported (requested by '$requesting_group')"
1030 if $dbic_reqs->{$target_group}{env};
1031
1032 if (my @foreign = grep { $_ ne 'req' } keys %{$ag->{$target_group}} ) {
1033 croak "Only 'req' augmentations are currently supported (group '$requesting_group' attempts to alter '$foreign[0]' of group '$target_group'";
1034 }
1035
1036 $ret->{augments}{$target_group} = 1;
1037
1038 # no augmentation for stuff that hasn't been selected
1039 if ( $all_groups->{$target_group} and my $ar = $ag->{$target_group}{req} ) {
1040 push @{$augmentations->{
1041 ( $all_groups->{$requesting_group}{idx} < $all_groups->{$target_group}{idx} )
1042 ? $target_group
1043 : $requesting_group
1044 }}, $ar;
1045 }
1046 }
1047 }
1048 }
1049
a3f8bd01 1050 for my $group (sort { $all_groups->{$a}{idx} <=> $all_groups->{$b}{idx} } keys %$all_groups ) {
e163f484 1051
a3f8bd01 1052 my $group_reqs = $dbic_reqs->{$group}{req};
e163f484 1053
a3f8bd01 1054 # sanity-check
2baba3d9 1055 for my $req_bag ($group_reqs, @{ $augmentations->{$group} || [] } ) {
1056 for (keys %$req_bag) {
e163f484 1057
461e818a 1058 $_ =~ / \A $modname_re \z /x
2baba3d9 1059 or croak "Requirement '$_' in group '$group' is not a valid module name";
a3f8bd01 1060
2baba3d9 1061 # !!!DO NOT CHANGE!!!
1062 # remember - version.pm may not be available on the system
1063 croak "Requirement '$_' in group '$group' specifies an invalid version '$req_bag->{$_}' (only plain non-underscored floating point decimals are supported)"
461e818a 1064 if ( ($req_bag->{$_}||0) !~ qr/ \A $modver_re \z /x );
2baba3d9 1065 }
a3f8bd01 1066 }
e163f484 1067
a3f8bd01 1068 if (my $e = $all_groups->{$group}{missing_envvars}) {
1069 push @{$ret->{missing_envvars}}, @$e;
e163f484 1070 }
1071
31c31b8d 1072 # assemble into the final ret
e3a7746c 1073 for my $type (
1074 'modreqs',
a3f8bd01 1075 ( $ret->{missing_envvars} ? () : 'effective_modreqs' ),
e3a7746c 1076 ) {
2baba3d9 1077 for my $req_bag ($group_reqs, @{ $augmentations->{$group} || [] } ) {
e163f484 1078 for my $mod (keys %$req_bag) {
31c31b8d 1079
e163f484 1080 $ret->{$type}{$mod} = $req_bag->{$mod}||0 if (
31c31b8d 1081
e163f484 1082 ! exists $ret->{$type}{$mod}
1083 or
1084 # we sanitized the version to be numeric above - we can just -gt it
1085 ($req_bag->{$mod}||0) > $ret->{$type}{$mod}
fb39747c 1086
e163f484 1087 );
1088 }
e3a7746c 1089 }
fb39747c 1090 }
31c31b8d 1091
2baba3d9 1092 $ret->{modreqs_fully_documented} &&= !!$dbic_reqs->{$group}{pod}
1093 if $all_groups->{$group}{user_requested};
cb551b07 1094
1095 $ret->{release_testing_mandatory} ||= !!$dbic_reqs->{$group}{release_testing_mandatory};
31c31b8d 1096 }
1097
1098 return $ret;
1099}
1100
1101
250d9e55 1102# this method tries to find/load specified modreqs and returns a hashref of
31c31b8d 1103# module/loaderror pairs for anything that failed
1104sub _errorlist_for_modreqs {
1105 # args supposedly already went through _groups_to_reqs and are therefore sanitized
1106 # safe to eval at will
1107 my ($self, $reqs) = @_;
1108
1109 my $ret;
1110
1111 for my $m ( keys %$reqs ) {
1112 my $v = $reqs->{$m};
1113
1114 if (! exists $req_unavailability_cache{$m}{$v} ) {
1115 local $@;
1116 eval( "require $m;" . ( $v ? "$m->VERSION(q($v))" : '' ) );
1117 $req_unavailability_cache{$m}{$v} = $@;
fb39747c 1118 }
1119
31c31b8d 1120 $ret->{$m} = $req_unavailability_cache{$m}{$v}
1121 if $req_unavailability_cache{$m}{$v};
1122 }
1123
1124 $ret;
fb39747c 1125}
1126
250d9e55 1127# Unlike the above DO NOT try to load anything
1128# This is executed when some needed envvars are not available
1129# which in turn means a module load will never be reached anyway
1130# This is important because some modules (especially DBDs) can be
1131# *really* fickle when a require() is attempted, with pretty confusing
1132# side-effects (especially on windows)
1133sub _list_physically_missing_modules {
1134 my ($self, $modreqs) = @_;
1135
1136 # in case there is a coderef in @INC there is nothing we can definitively prove
1137 # so short circuit directly
1138 return '' if grep { length ref $_ } @INC;
1139
1140 my @definitely_missing;
1141 for my $mod (keys %$modreqs) {
1142 (my $fn = $mod . '.pm') =~ s|::|/|g;
1143
1144 push @definitely_missing, $mod unless grep
1145 # this should work on any combination of slashes
1146 { $_ and -d $_ and -f "$_/$fn" and -r "$_/$fn" }
1147 @INC
1148 ;
1149 }
1150
1151 join ' ', map
1152 { $modreqs->{$_} ? qq("$_~>=$modreqs->{$_}") : $_ }
1153 sort { lc($a) cmp lc($b) } @definitely_missing
1154 ;
1155}
1156
31c31b8d 1157
e3fc11e1 1158# This is to be called by the author only (automatically in Makefile.PL)
f6b26571 1159sub _gen_pod {
47589465 1160 my ($class, $distver, $pod_dir) = @_;
31fa1764 1161
47589465 1162 die "No POD root dir supplied" unless $pod_dir;
31fa1764 1163
ccebe1f1 1164 $distver ||=
31fa1764 1165 eval { require DBIx::Class; DBIx::Class->VERSION; }
1166 ||
ccebe1f1 1167 die
31fa1764 1168"\n\n---------------------------------------------------------------------\n" .
1169'Unable to load core DBIx::Class module to determine current version, '.
1170'possibly due to missing dependencies. Author-mode autodocumentation ' .
1171"halted\n\n" . $@ .
1172"\n\n---------------------------------------------------------------------\n"
31fa1764 1173 ;
1174
3d4c5a84 1175 # do not ask for a recent version, use 1.x API calls
47589465 1176 # this *may* execute on a smoker with old perl or whatnot
1177 require File::Path;
1178
1179 (my $modfn = __PACKAGE__ . '.pm') =~ s|::|/|g;
1180
1181 (my $podfn = "$pod_dir/$modfn") =~ s/\.pm$/\.pod/;
1182 (my $dir = $podfn) =~ s|/[^/]+$||;
1183
1184 File::Path::mkpath([$dir]);
1185
31c31b8d 1186 my $sqltver = $class->req_list_for('deploy')->{'SQL::Translator'}
e3fc11e1 1187 or die "Hrmm? No sqlt dep?";
7e3dc46f 1188
3ca0dcf0 1189
1190 my @chunks;
1191
1192#@@
1193#@@ HEADER
1194#@@
79b1bf0a 1195 push @chunks, <<"EOC";
af4ac504 1196#########################################################################
1197##################### A U T O G E N E R A T E D ########################
1198#########################################################################
1199#
1200# The contents of this POD file are auto-generated. Any changes you make
1201# will be lost. If you need to change the generated text edit _gen_pod()
1202# at the end of $modfn
1203#
79b1bf0a 1204
1205=head1 NAME
1206
1207$class - Optional module dependency specifications (for module authors)
af4ac504 1208EOC
3ca0dcf0 1209
1210
1211#@@
1212#@@ SYNOPSIS HEADING
1213#@@
79b1bf0a 1214 push @chunks, <<"EOC";
1215=head1 SYNOPSIS
1216
1217Somewhere in your build-file (e.g. L<ExtUtils::MakeMaker>'s F<Makefile.PL>):
7e3dc46f 1218
1219 ...
1220
79b1bf0a 1221 \$EUMM_ARGS{CONFIGURE_REQUIRES} = {
1222 \%{ \$EUMM_ARGS{CONFIGURE_REQUIRES} || {} },
1223 'DBIx::Class' => '$distver',
1224 };
7e3dc46f 1225
79b1bf0a 1226 ...
7e3dc46f 1227
2baba3d9 1228 my %DBIC_DEPLOY_AND_ORACLE_DEPS = %{ eval {
79b1bf0a 1229 require $class;
2baba3d9 1230 $class->req_list_for([qw( deploy rdbms_oracle icdt )]);
79b1bf0a 1231 } || {} };
7e3dc46f 1232
79b1bf0a 1233 \$EUMM_ARGS{PREREQ_PM} = {
2baba3d9 1234 \%DBIC_DEPLOY_AND_ORACLE_DEPS,
79b1bf0a 1235 \%{ \$EUMM_ARGS{PREREQ_PM} || {} },
1236 };
7e3dc46f 1237
1238 ...
1239
79b1bf0a 1240 ExtUtils::MakeMaker::WriteMakefile(\%EUMM_ARGS);
1241
1242B<Note>: The C<eval> protection within the example is due to support for
1243requirements during L<the C<configure> build phase|CPAN::Meta::Spec/Phases>
1244not being available on a sufficient portion of production installations of
1245Perl. Robust support for such dependency requirements is available in the
1246L<CPAN> installer only since version C<1.94_56> first made available for
1247production with perl version C<5.12>. It is the belief of the current
1248maintainer that support for requirements during the C<configure> build phase
1249will not be sufficiently ubiquitous until the B<year 2020> at the earliest,
1250hence the extra care demonstrated above. It should also be noted that some
12513rd party installers (e.g. L<cpanminus|App::cpanminus>) do the right thing
1252with configure requirements independent from the versions of perl and CPAN
1253available.
1254EOC
3ca0dcf0 1255
1256
1257#@@
1258#@@ DESCRIPTION HEADING
1259#@@
79b1bf0a 1260 push @chunks, <<'EOC';
1261=head1 DESCRIPTION
1262
f6b26571 1263Some of the less-frequently used features of L<DBIx::Class> have external
1264module dependencies on their own. In order not to burden the average user
79b1bf0a 1265with modules they will never use, these optional dependencies are not included
f6b26571 1266in the base Makefile.PL. Instead an exception with a descriptive message is
79b1bf0a 1267thrown when a specific feature can't find one or several modules required for
1268its operation. This module is the central holding place for the current list
7e3dc46f 1269of such dependencies, for DBIx::Class core authors, and DBIx::Class extension
1270authors alike.
79b1bf0a 1271
1272Dependencies are organized in L<groups|/CURRENT REQUIREMENT GROUPS> where each
1273group can list one or more required modules, with an optional minimum version
e3a7746c 1274(or 0 for any version). In addition groups prefixed with C<test_> can specify
1275a set of environment variables, some (or all) of which are marked as required
1276for the group to be considered by L</req_list_for>
1277
1278Each group name (or a combination thereof) can be used in the
1279L<public methods|/METHODS> as described below.
79b1bf0a 1280EOC
3ca0dcf0 1281
1282
1283#@@
1284#@@ REQUIREMENT GROUPLIST HEADING
1285#@@
79b1bf0a 1286 push @chunks, '=head1 CURRENT REQUIREMENT GROUPS';
f6b26571 1287
2baba3d9 1288 my $standalone_info;
1289
31c31b8d 1290 for my $group (sort keys %$dbic_reqs) {
f6b26571 1291
2baba3d9 1292 my $info = $standalone_info->{$group} ||= $class->_groups_to_reqs($group);
f6b26571 1293
2baba3d9 1294 next unless (
1295 $info->{modreqs_fully_documented}
1296 and
1297 ( $info->{augments} or $info->{modreqs} )
1298 );
1299
1300 my $p = $dbic_reqs->{$group}{pod};
f6b26571 1301
1302 push @chunks, (
1303 "=head2 $p->{title}",
2baba3d9 1304 "=head3 $group",
1305 $p->{desc},
f6b26571 1306 '=over',
f6b26571 1307 );
2baba3d9 1308
1309 if ( keys %{ $info->{modreqs}||{} } ) {
1310 push @chunks, map
1311 { "=item * $_" . ($info->{modreqs}{$_} ? " >= $info->{modreqs}{$_}" : '') }
1312 ( sort keys %{ $info->{modreqs} } )
1313 ;
1314 }
1315 else {
1316 push @chunks, '=item * No standalone requirements',
1317 }
1318
1319 push @chunks, '=back';
1320
1321 for my $ag ( sort keys %{ $info->{augments} || {} } ) {
1322 my $ag_info = $standalone_info->{$ag} ||= $class->_groups_to_reqs($ag);
1323
1324 my $newreqs = $class->modreq_list_for([ $group, $ag ]);
1325 for (keys %$newreqs) {
1326 delete $newreqs->{$_} if (
1327 ( defined $info->{modreqs}{$_} and $info->{modreqs}{$_} == $newreqs->{$_} )
1328 or
1329 ( defined $ag_info->{modreqs}{$_} and $ag_info->{modreqs}{$_} == $newreqs->{$_} )
1330 );
1331 }
1332
1333 if (keys %$newreqs) {
1334 push @chunks, (
1335 "Combined with L</$ag> additionally requires:",
1336 '=over',
1337 ( map
1338 { "=item * $_" . ($newreqs->{$_} ? " >= $newreqs->{$_}" : '') }
1339 ( sort keys %$newreqs )
1340 ),
1341 '=back',
1342 );
1343 }
1344 }
f6b26571 1345 }
1346
3ca0dcf0 1347
1348#@@
1349#@@ API DOCUMENTATION HEADING
1350#@@
79b1bf0a 1351 push @chunks, <<'EOC';
1352
afb8fc52 1353=head1 IMPORT-LIKE ACTIONS
1354
1355Even though this module is not an L<Exporter>, it recognizes several C<actions>
1356supplied to its C<import> method.
1357
cb551b07 1358=head2 -skip_all_without
1359
1360=over
1361
1362=item Arguments: @group_names
1363
1364=back
1365
1366A convenience wrapper for use during testing:
1367EOC
1368
1369 push @chunks, " use $class -skip_all_without => qw(admin test_rdbms_mysql);";
1370
1371 push @chunks, 'Roughly equivalent to the following code:';
1372
1373 push @chunks, sprintf <<'EOS', ($class) x 2;
1374
1375 BEGIN {
1376 require %s;
1377 if ( my $missing = %s->req_missing_for(\@group_names_) ) {
1378 print "1..0 # SKIP requirements not satisfied: $missing\n";
1379 exit 0;
1380 }
1381 }
1382EOS
1383
1384 push @chunks, <<'EOC';
1385
1386It also takes into account the C<RELEASE_TESTING> environment variable and
1387behaves like L</-die_without> for any requirement groups marked as
1388C<release_testing_mandatory>.
1389
afb8fc52 1390=head2 -die_without
1391
1392=over
1393
1394=item Arguments: @group_names
1395
1396=back
1397
1398A convenience wrapper around L</die_unless_req_ok_for>:
1399EOC
1400
1401 push @chunks, " use $class -die_without => qw(deploy admin);";
1402
1403 push @chunks, <<'EOC';
1404
1405=head2 -list_missing
1406
1407=over
1408
1409=item Arguments: @group_names
1410
1411=back
1412
1413A convenience wrapper around L</modreq_missing_for>:
1414
1415 perl -Ilib -MDBIx::Class::Optional::Dependencies=-list_missing,deploy,admin | cpanm
1416
79b1bf0a 1417=head1 METHODS
1418
1419=head2 req_group_list
1420
1421=over
1422
1423=item Arguments: none
1424
1425=item Return Value: \%list_of_requirement_groups
1426
1427=back
1428
e3fc11e1 1429This method should be used by DBIx::Class packagers, to get a hashref of all
31c31b8d 1430dependencies B<keyed> by dependency group. Each key (group name), or a combination
1431thereof (as an arrayref) can be supplied to the methods below.
79b1bf0a 1432The B<values> of the returned hash are currently a set of options B<without a
1433well defined structure>. If you have use for any of the contents - contact the
1434maintainers, instead of treating this as public (left alone stable) API.
1435
1436=head2 req_list_for
1437
1438=over
1439
31c31b8d 1440=item Arguments: $group_name | \@group_names
79b1bf0a 1441
31c31b8d 1442=item Return Value: \%set_of_module_version_pairs
79b1bf0a 1443
1444=back
1445
f6b26571 1446This method should be used by DBIx::Class extension authors, to determine the
31c31b8d 1447version of modules a specific set of features requires for this version of
e3a7746c 1448DBIx::Class (regardless of their availability on the system).
1449See the L</SYNOPSIS> for a real-world example.
1450
1451When handling C<test_*> groups this method behaves B<differently> from
1452L</modreq_list_for> below (and is the only such inconsistency among the
1453C<req_*> methods). If a particular group declares as requirements some
1454C<environment variables> and these requirements are not satisfied (the envvars
1455are unset) - then the C<module requirements> of this group are not included in
1456the returned list.
1457
1458=head2 modreq_list_for
1459
1460=over
1461
1462=item Arguments: $group_name | \@group_names
1463
1464=item Return Value: \%set_of_module_version_pairs
1465
1466=back
1467
1468Same as L</req_list_for> but does not take into consideration any
1469C<environment variable requirements> - returns just the list of required
1470modules.
79b1bf0a 1471
1472=head2 req_ok_for
1473
1474=over
1475
31c31b8d 1476=item Arguments: $group_name | \@group_names
79b1bf0a 1477
1478=item Return Value: 1|0
1479
1480=back
1481
e3a7746c 1482Returns true or false depending on whether all modules/envvars required by
1483the group(s) are loadable/set on the system.
79b1bf0a 1484
1485=head2 req_missing_for
1486
1487=over
1488
31c31b8d 1489=item Arguments: $group_name | \@group_names
79b1bf0a 1490
1491=item Return Value: $error_message_string
1492
1493=back
1494
31c31b8d 1495Returns a single-line string suitable for inclusion in larger error messages.
e3a7746c 1496This method would normally be used by DBIx::Class core features, to indicate to
1497the user that they need to install specific modules and/or set specific
1498environment variables before being able to use a specific feature set.
f6b26571 1499
e3fc11e1 1500For example if some of the requirements for C<deploy> are not available,
1501the returned string could look like:
79b1bf0a 1502EOC
f6b26571 1503
79b1bf0a 1504 push @chunks, qq{ "SQL::Translator~>=$sqltver" (see $class documentation for details)};
f6b26571 1505
79b1bf0a 1506 push @chunks, <<'EOC';
f6b26571 1507The author is expected to prepend the necessary text to this message before
e3a7746c 1508returning the actual error seen by the user. See also L</modreq_missing_for>
1509
1510=head2 modreq_missing_for
f6b26571 1511
e3a7746c 1512=over
1513
1514=item Arguments: $group_name | \@group_names
1515
1516=item Return Value: $error_message_string
1517
1518=back
1519
1520Same as L</req_missing_for> except that the error string is guaranteed to be
1521either empty, or contain a set of module requirement specifications suitable
1522for piping to e.g. L<cpanminus|App::cpanminus>. The method explicitly does not
1523attempt to validate the state of required environment variables (if any).
1524
1525For instance if some of the requirements for C<deploy> are not available,
1526the returned string could look like:
1527EOC
1528
1529 push @chunks, qq{ "SQL::Translator~>=$sqltver"};
1530
1531 push @chunks, <<'EOC';
afb8fc52 1532
1533See also L</-list_missing>.
1534
461e818a 1535=head2 skip_without
1536
1537=over
1538
1539=item Arguments: $group_name | \@group_names
1540
1541=back
1542
1543A convenience wrapper around L<skip|Test::More/SKIP>. It does not take neither
1544a reason (it is generated by L</req_missing_for>) nor an amount of skipped tests
1545(it is always C<1>, thus mandating unconditional use of
1546L<done_testing|Test::More/done_testing>). Most useful in combination with ad hoc
1547requirement specifications:
1548EOC
1549
1550 push @chunks, <<EOC;
1551 SKIP: {
1552 $class->skip_without([ deploy YAML>=0.90 ]);
1553
1554 ...
1555 }
1556EOC
1557
1558 push @chunks, <<'EOC';
1559
79b1bf0a 1560=head2 die_unless_req_ok_for
1561
1562=over
1563
31c31b8d 1564=item Arguments: $group_name | \@group_names
79b1bf0a 1565
1566=back
1567
31c31b8d 1568Checks if L</req_ok_for> passes for the supplied group(s), and
344f1f52 1569in case of failure throws an exception including the information
afb8fc52 1570from L</req_missing_for>. See also L</-die_without>.
79b1bf0a 1571
5ffa39c7 1572=head2 modreq_errorlist_for
79b1bf0a 1573
1574=over
1575
31c31b8d 1576=item Arguments: $group_name | \@group_names
79b1bf0a 1577
31c31b8d 1578=item Return Value: \%set_of_loaderrors_per_module
79b1bf0a 1579
1580=back
1581
4a0eed52 1582Returns a hashref containing the actual errors that occurred while attempting
31c31b8d 1583to load each module in the requirement group(s).
5ffa39c7 1584
1585=head2 req_errorlist_for
1586
1587Deprecated method name, equivalent (via proxy) to L</modreq_errorlist_for>.
1588
79b1bf0a 1589EOC
3ca0dcf0 1590
1591#@@
1592#@@ FOOTER
1593#@@
79b1bf0a 1594 push @chunks, <<'EOC';
1595=head1 FURTHER QUESTIONS?
1596
1597Check the list of L<additional DBIC resources|DBIx::Class/GETTING HELP/SUPPORT>.
1598
1599=head1 COPYRIGHT AND LICENSE
1600
a2bd3796 1601This module is free software L<copyright|DBIx::Class/COPYRIGHT AND LICENSE>
1602by the L<DBIx::Class (DBIC) authors|DBIx::Class/AUTHORS>. You can
1603redistribute it and/or modify it under the same terms as the
1604L<DBIx::Class library|DBIx::Class/COPYRIGHT AND LICENSE>.
79b1bf0a 1605EOC
3ca0dcf0 1606
3f5e367a 1607 eval {
1608 open (my $fh, '>', $podfn) or die;
1609 print $fh join ("\n\n", @chunks) or die;
1610 print $fh "\n" or die;
1611 close ($fh) or die;
1612 } or croak( "Unable to write $podfn: " . ( $! || $@ || 'unknown error') );
f6b26571 1613}
1614
8057ed01 16151;