File Coverage

blib/lib/DBIx/Class/DeploymentHandler.pm
Criterion Covered Total %
statement 12 12 100.0
branch n/a
condition 1 3 33.3
subroutine 6 6 100.0
pod 3 4 75.0
total 22 25 88.0


line stmt bran cond sub pod time code
1             package DBIx::Class::DeploymentHandler;
2             $DBIx::Class::DeploymentHandler::VERSION = '0.002233';
3             # ABSTRACT: Extensible DBIx::Class deployment
4              
5 11     11   185009 use Moose;
  11         3905242  
  11         106  
6              
7             has initial_version => (is => 'ro', lazy_build => 1);
8 16     16   98 sub _build_initial_version { $_[0]->database_version }
9              
10             extends 'DBIx::Class::DeploymentHandler::Dad';
11             # a single with would be better, but we can't do that
12             # see: http://rt.cpan.org/Public/Bug/Display.html?id=46347
13             with 'DBIx::Class::DeploymentHandler::WithApplicatorDumple' => {
14             interface_role => 'DBIx::Class::DeploymentHandler::HandlesDeploy',
15             class_name => 'DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator',
16             delegate_name => 'deploy_method',
17             attributes_to_assume => [qw(schema schema_version version_source)],
18             attributes_to_copy => [qw(
19             ignore_ddl databases script_directory sql_translator_args force_overwrite txn_prep txn_wrap
20             )],
21             },
22             'DBIx::Class::DeploymentHandler::WithApplicatorDumple' => {
23             interface_role => 'DBIx::Class::DeploymentHandler::HandlesVersioning',
24             class_name => 'DBIx::Class::DeploymentHandler::VersionHandler::Monotonic',
25             delegate_name => 'version_handler',
26             attributes_to_assume => [qw( initial_version schema_version to_version )],
27             },
28             'DBIx::Class::DeploymentHandler::WithApplicatorDumple' => {
29             interface_role => 'DBIx::Class::DeploymentHandler::HandlesVersionStorage',
30             class_name => 'DBIx::Class::DeploymentHandler::VersionStorage::Standard',
31             delegate_name => 'version_storage',
32             attributes_to_assume => ['schema'],
33             attributes_to_copy => [qw(version_source version_class)],
34             };
35             with 'DBIx::Class::DeploymentHandler::WithReasonableDefaults';
36              
37             sub prepare_version_storage_install {
38 20     20 1 333 my $self = shift;
39              
40 20         766 $self->prepare_resultsource_install({
41             result_source => $self->version_storage->version_rs->result_source
42             });
43             }
44              
45             sub install_version_storage {
46 1     1 1 26500 my $self = shift;
47              
48 1   33     46 my $version = (shift||{})->{version} || $self->schema_version;
49              
50 1         44 $self->install_resultsource({
51             result_source => $self->version_storage->version_rs->result_source,
52             version => $version,
53             });
54             }
55              
56             sub prepare_install {
57 19     19 1 8967 $_[0]->prepare_deploy;
58 19         1198 $_[0]->prepare_version_storage_install;
59             }
60              
61             # the following is just a hack so that ->version_storage
62             # won't be lazy
63 29     29 0 1051 sub BUILD { $_[0]->version_storage }
64             __PACKAGE__->meta->make_immutable;
65              
66             1;
67              
68             #vim: ts=2 sw=2 expandtab
69              
70             __END__
71              
72             =pod
73              
74             =head1 NAME
75              
76             DBIx::Class::DeploymentHandler - Extensible DBIx::Class deployment
77              
78             =head1 SYNOPSIS
79              
80             use aliased 'DBIx::Class::DeploymentHandler' => 'DH';
81             my $s = My::Schema->connect(...);
82              
83             my $dh = DH->new({
84             schema => $s,
85             databases => 'SQLite',
86             sql_translator_args => { add_drop_table => 0 },
87             });
88              
89             $dh->prepare_install;
90              
91             $dh->install;
92              
93             or for upgrades:
94              
95             use aliased 'DBIx::Class::DeploymentHandler' => 'DH';
96             my $s = My::Schema->connect(...);
97              
98             my $dh = DH->new({
99             schema => $s,
100             databases => 'SQLite',
101             sql_translator_args => { add_drop_table => 0 },
102             });
103              
104             $dh->prepare_deploy;
105             $dh->prepare_upgrade({
106             from_version => 1,
107             to_version => 2,
108             });
109              
110             $dh->upgrade;
111              
112             =head1 DESCRIPTION
113              
114             C<DBIx::Class::DeploymentHandler> is, as its name suggests, a tool for
115             deploying and upgrading databases with L<DBIx::Class>. It is designed to be
116             much more flexible than L<DBIx::Class::Schema::Versioned>, hence the use of
117             L<Moose> and lots of roles.
118              
119             C<DBIx::Class::DeploymentHandler> itself is just a recommended set of roles
120             that we think will not only work well for everyone, but will also yield the
121             best overall mileage. Each role it uses has its own nuances and
122             documentation, so I won't describe all of them here, but here are a few of the
123             major benefits over how L<DBIx::Class::Schema::Versioned> worked (and
124             L<DBIx::Class::DeploymentHandler::Deprecated> tries to maintain compatibility
125             with):
126              
127             =over
128              
129             =item *
130              
131             Downgrades in addition to upgrades.
132              
133             =item *
134              
135             Multiple sql files files per upgrade/downgrade/install.
136              
137             =item *
138              
139             Perl scripts allowed for upgrade/downgrade/install.
140              
141             =item *
142              
143             Just one set of files needed for upgrade, unlike before where one might need
144             to generate C<factorial(scalar @versions)>, which is just silly.
145              
146             =item *
147              
148             And much, much more!
149              
150             =back
151              
152             That's really just a taste of some of the differences. Check out each role for
153             all the details.
154              
155             =head1 ATTRIBUTES
156              
157             This is just a "stub" section to make clear
158             that the bulk of implementation is documented somewhere else.
159              
160             =head2 Attributes passed to L<DBIx::Class::DeploymentHandler::HandlesDeploy>
161              
162             =over
163              
164             =item *
165              
166             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/ignore_ddl>
167              
168             =item *
169              
170             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/databases>
171              
172             =item *
173              
174             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/script_directory>
175              
176             =item *
177              
178             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/sql_translator_args>
179              
180             =item *
181              
182             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/force_overwrite>
183              
184             =item *
185              
186             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/txn_prep>
187              
188             =item *
189              
190             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator/txn_wrap>
191              
192             =back
193              
194             =head2 Attributes passed to L<DBIx::Class::DeploymentHandler::HandlesVersioning>
195              
196             =over
197              
198             =item *
199              
200             initial_version
201              
202             =item *
203              
204             L<DBIx::Class::DeploymentHandler::Dad/schema_version>
205              
206             =item *
207              
208             L<DBIx::Class::DeploymentHandler::Dad/to_version>
209              
210             =back
211              
212             =head2 Attributes passed to L<DBIx::Class::DeploymentHandler::HandlesVersionStorage>
213              
214             =over
215              
216             =item *
217              
218             version_source
219              
220             =item *
221              
222             version_class
223              
224             =back
225              
226             =head2 Attributes Inherited from Parent Class
227              
228             See L<DBIx::Class::DeploymentHandler::Dad/ATTRIBUTES> and
229             L<DBIx::Class::DeploymentHandler::Dad/"ORTHODOX METHODS"> for the remaining
230             available attributes to pass to C<new>.
231              
232             =head1 WHERE IS ALL THE DOC?!
233              
234             To get up and running fast, your best place to start is
235             L<DBIx::Class::DeploymentHandler::Manual::Intro> and then
236             L<DBIx::Class::DeploymentHandler::Manual::CatalystIntro> if your intending on
237             using this with Catalyst.
238              
239             For the full story you should realise that C<DBIx::Class::DeploymentHandler>
240             extends L<DBIx::Class::DeploymentHandler::Dad>, so that's probably the first
241             place to look when you are trying to figure out how everything works.
242              
243             Next would be to look at all the pieces that fill in the blanks that
244             L<DBIx::Class::DeploymentHandler::Dad> expects to be filled. They would be
245             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator>,
246             L<DBIx::Class::DeploymentHandler::VersionHandler::Monotonic>,
247             L<DBIx::Class::DeploymentHandler::VersionStorage::Standard>, and
248             L<DBIx::Class::DeploymentHandler::WithReasonableDefaults>.
249              
250             =head1 WHY IS THIS SO WEIRD
251              
252             C<DBIx::Class::DeploymentHandler> has a strange structure. The gist is that it
253             delegates to three small objects that are proxied to via interface roles that
254             then create the illusion of one large, monolithic object. Here is a diagram
255             that might help:
256              
257             =begin text
258              
259             Figure 1
260              
261             +------------+
262             | |
263             +------------+ Deployment +-----------+
264             | | Handler | |
265             | | | |
266             | +-----+------+ |
267             | | |
268             | | |
269             : : :
270             v v v
271             /-=-------\ /-=-------\ /-=----------\
272             | | | | | | (interface roles)
273             | Handles | | Handles | | Handles |
274             | Version | | Deploy | | Versioning |
275             | Storage | | | | |
276             | | \-+--+--+-/ \-+---+---+--/
277             \-+--+--+-/ | | | | | |
278             | | | | | | | | |
279             | | | | | | | | |
280             v v v v v v v v v
281             +----------+ +--------+ +-----------+
282             | | | | | | (implementations)
283             | Version | | Deploy | | Version |
284             | Storage | | Method | | Handler |
285             | Standard | | SQLT | | Monotonic |
286             | | | | | |
287             +----------+ +--------+ +-----------+
288              
289             =end text
290              
291             =for html <p><i>Figure 1</i><img src=""></img></p>
292              
293             The nice thing about this is that we have well defined interfaces for the
294             objects that comprise the C<DeploymentHandler>, the smaller objects can be
295             tested in isolation, and the smaller objects can even be swapped in easily. But
296             the real win is that you can subclass the C<DeploymentHandler> without knowing
297             about the underlying delegation; you just treat it like normal Perl and write
298             methods that do what you want.
299              
300             =head1 THIS SUCKS
301              
302             You started your project and weren't using C<DBIx::Class::DeploymentHandler>?
303             Lucky for you I had you in mind when I wrote this doc.
304              
305             First,
306             L<define the version|DBIx::Class::DeploymentHandler::Manual::Intro/Sample_database>
307             in your main schema file (maybe using C<$VERSION>).
308              
309             Then you'll want to just install the version_storage:
310              
311             my $s = My::Schema->connect(...);
312             my $dh = DBIx::Class::DeploymentHandler->new({ schema => $s });
313              
314             $dh->prepare_version_storage_install;
315             $dh->install_version_storage;
316              
317             Then set your database version:
318              
319             $dh->add_database_version({ version => $s->schema_version });
320              
321             Now you should be able to use C<DBIx::Class::DeploymentHandler> like normal!
322              
323             =head1 LOGGING
324              
325             This is a complex tool, and because of that sometimes you'll want to see
326             what exactly is happening. The best way to do that is to use the built in
327             logging functionality. It the standard six log levels; C<fatal>, C<error>,
328             C<warn>, C<info>, C<debug>, and C<trace>. Most of those are pretty self
329             explanatory. Generally a safe level to see what all is going on is debug,
330             which will give you everything except for the exact SQL being run.
331              
332             To enable the various logging levels all you need to do is set an environment
333             variables: C<DBICDH_FATAL>, C<DBICDH_ERROR>, C<DBICDH_WARN>, C<DBICDH_INFO>,
334             C<DBICDH_DEBUG>, and C<DBICDH_TRACE>. Each level can be set on its own,
335             but the default is the first three on and the last three off, and the levels
336             cascade, so if you turn on trace the rest will turn on automatically.
337              
338             =head1 DONATIONS
339              
340             If you'd like to thank me for the work I've done on this module, don't give me
341             a donation. I spend a lot of free time creating free software, but I do it
342             because I love it.
343              
344             Instead, consider donating to someone who might actually need it. Obviously
345             you should do research when donating to a charity, so don't just take my word
346             on this. I like Matthew 25: Ministries:
347             L<http://www.m25m.org/>, but there are a host of other
348             charities that can do much more good than I will with your money.
349             (Third party charity info here:
350             L<http://www.charitynavigator.org/index.cfm?bay=search.summary&orgid=6901>
351              
352             =head1 METHODS
353              
354             This is just a "stub" section to make clear
355             that the bulk of implementation is documented in
356             L<DBIx::Class::DeploymentHandler::Dad>. Since that is implemented using
357             L<Moose> class, see L<DBIx::Class::DeploymentHandler::Dad/ATTRIBUTES>
358             and L<DBIx::Class::DeploymentHandler::Dad/"ORTHODOX METHODS"> for methods
359             callable on the resulting object.
360              
361             =head2 new
362              
363             my $s = My::Schema->connect(...);
364             my $dh = DBIx::Class::DeploymentHandler->new({
365             schema => $s,
366             databases => 'SQLite',
367             sql_translator_args => { add_drop_table => 0 },
368             });
369              
370             =head2 prepare_version_storage_install
371              
372             $dh->prepare_version_storage_install
373              
374             Creates the needed C<.sql> file to install the version storage and not the rest
375             of the tables
376              
377             =head2 prepare_install
378              
379             $dh->prepare_install
380              
381             First prepare all the tables to be installed and the prepare just the version
382             storage
383              
384             =head2 install_version_storage
385              
386             $dh->install_version_storage
387              
388             Install the version storage and not the rest of the tables
389              
390             =head1 AUTHOR
391              
392             Arthur Axel "fREW" Schmidt <frioux+cpan@gmail.com>
393              
394             =head1 COPYRIGHT AND LICENSE
395              
396             This software is copyright (c) 2019 by Arthur Axel "fREW" Schmidt.
397              
398             This is free software; you can redistribute it and/or modify it under
399             the same terms as the Perl 5 programming language system itself.
400              
401             =cut