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.002232';
3             # ABSTRACT: Extensible DBIx::Class deployment
4              
5 10     10   164933 use Moose;
  10         3975948  
  10         114  
6              
7             has initial_version => (is => 'ro', lazy_build => 1);
8 12     12   90 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
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 17     17 1 334 my $self = shift;
39              
40 17         781 $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 27591 my $self = shift;
47              
48 1   33     61 my $version = (shift||{})->{version} || $self->schema_version;
49              
50 1         34 $self->install_resultsource({
51             result_source => $self->version_storage->version_rs->result_source,
52             version => $version,
53             });
54             }
55              
56             sub prepare_install {
57 16     16 1 12369 $_[0]->prepare_deploy;
58 16         1504 $_[0]->prepare_version_storage_install;
59             }
60              
61             # the following is just a hack so that ->version_storage
62             # won't be lazy
63 24     24 0 1058 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 WHERE IS ALL THE DOC?!
156              
157             To get up and running fast, your best place to start is
158             L<DBIx::Class::DeploymentHandler::Manual::Intro> and then
159             L<DBIx::Class::DeploymentHandler::Manual::CatalystIntro> if your intending on
160             using this with Catalyst.
161              
162             For the full story you should realise that C<DBIx::Class::DeploymentHandler>
163             extends L<DBIx::Class::DeploymentHandler::Dad>, so that's probably the first
164             place to look when you are trying to figure out how everything works.
165              
166             Next would be to look at all the pieces that fill in the blanks that
167             L<DBIx::Class::DeploymentHandler::Dad> expects to be filled. They would be
168             L<DBIx::Class::DeploymentHandler::DeployMethod::SQL::Translator>,
169             L<DBIx::Class::DeploymentHandler::VersionHandler::Monotonic>,
170             L<DBIx::Class::DeploymentHandler::VersionStorage::Standard>, and
171             L<DBIx::Class::DeploymentHandler::WithReasonableDefaults>.
172              
173             =head1 WHY IS THIS SO WEIRD
174              
175             C<DBIx::Class::DeploymentHandler> has a strange structure. The gist is that it
176             delegates to three small objects that are proxied to via interface roles that
177             then create the illusion of one large, monolithic object. Here is a diagram
178             that might help:
179              
180             =begin text
181              
182             Figure 1
183              
184             +------------+
185             | |
186             +------------+ Deployment +-----------+
187             | | Handler | |
188             | | | |
189             | +-----+------+ |
190             | | |
191             | | |
192             : : :
193             v v v
194             /-=-------\ /-=-------\ /-=----------\
195             | | | | | | (interface roles)
196             | Handles | | Handles | | Handles |
197             | Version | | Deploy | | Versioning |
198             | Storage | | | | |
199             | | \-+--+--+-/ \-+---+---+--/
200             \-+--+--+-/ | | | | | |
201             | | | | | | | | |
202             | | | | | | | | |
203             v v v v v v v v v
204             +----------+ +--------+ +-----------+
205             | | | | | | (implementations)
206             | Version | | Deploy | | Version |
207             | Storage | | Method | | Handler |
208             | Standard | | SQLT | | Monotonic |
209             | | | | | |
210             +----------+ +--------+ +-----------+
211              
212             =end text
213              
214             =for html <p><i>Figure 1</i><img src=""></img></p>
215              
216             The nice thing about this is that we have well defined interfaces for the
217             objects that comprise the C<DeploymentHandler>, the smaller objects can be
218             tested in isolation, and the smaller objects can even be swapped in easily. But
219             the real win is that you can subclass the C<DeploymentHandler> without knowing
220             about the underlying delegation; you just treat it like normal Perl and write
221             methods that do what you want.
222              
223             =head1 THIS SUCKS
224              
225             You started your project and weren't using C<DBIx::Class::DeploymentHandler>?
226             Lucky for you I had you in mind when I wrote this doc.
227              
228             First,
229             L<define the version|DBIx::Class::DeploymentHandler::Manual::Intro/Sample_database>
230             in your main schema file (maybe using C<$VERSION>).
231              
232             Then you'll want to just install the version_storage:
233              
234             my $s = My::Schema->connect(...);
235             my $dh = DBIx::Class::DeploymentHandler->new({ schema => $s });
236              
237             $dh->prepare_version_storage_install;
238             $dh->install_version_storage;
239              
240             Then set your database version:
241              
242             $dh->add_database_version({ version => $s->schema_version });
243              
244             Now you should be able to use C<DBIx::Class::DeploymentHandler> like normal!
245              
246             =head1 LOGGING
247              
248             This is a complex tool, and because of that sometimes you'll want to see
249             what exactly is happening. The best way to do that is to use the built in
250             logging functionality. It the standard six log levels; C<fatal>, C<error>,
251             C<warn>, C<info>, C<debug>, and C<trace>. Most of those are pretty self
252             explanatory. Generally a safe level to see what all is going on is debug,
253             which will give you everything except for the exact SQL being run.
254              
255             To enable the various logging levels all you need to do is set an environment
256             variables: C<DBICDH_FATAL>, C<DBICDH_ERROR>, C<DBICDH_WARN>, C<DBICDH_INFO>,
257             C<DBICDH_DEBUG>, and C<DBICDH_TRACE>. Each level can be set on its own,
258             but the default is the first three on and the last three off, and the levels
259             cascade, so if you turn on trace the rest will turn on automatically.
260              
261             =head1 DONATIONS
262              
263             If you'd like to thank me for the work I've done on this module, don't give me
264             a donation. I spend a lot of free time creating free software, but I do it
265             because I love it.
266              
267             Instead, consider donating to someone who might actually need it. Obviously
268             you should do research when donating to a charity, so don't just take my word
269             on this. I like Matthew 25: Ministries:
270             L<http://www.m25m.org/>, but there are a host of other
271             charities that can do much more good than I will with your money.
272             (Third party charity info here:
273             L<http://www.charitynavigator.org/index.cfm?bay=search.summary&orgid=6901>
274              
275             =head1 METHODS
276              
277             This is just a "stub" section to make clear
278             that the bulk of implementation is documented in
279             L<DBIx::Class::DeploymentHandler::Dad>. Since that is implemented using
280             L<Moose> class, see L<DBIx::Class::DeploymentHandler::Dad/ATTRIBUTES>
281             and L<DBIx::Class::DeploymentHandler::Dad/"ORTHODOX METHODS"> for
282             available attributes to pass to C<new>, and methods callable on the
283             resulting object.
284              
285             =head2 new
286              
287             my $s = My::Schema->connect(...);
288             my $dh = DBIx::Class::DeploymentHandler->new({
289             schema => $s,
290             databases => 'SQLite',
291             sql_translator_args => { add_drop_table => 0 },
292             });
293              
294             See L<DBIx::Class::DeploymentHandler::Dad/ATTRIBUTES> and
295             L<DBIx::Class::DeploymentHandler::Dad/"ORTHODOX METHODS"> for available
296             attributes to pass to C<new>.
297              
298             =head2 prepare_version_storage_install
299              
300             $dh->prepare_version_storage_install
301              
302             Creates the needed C<.sql> file to install the version storage and not the rest
303             of the tables
304              
305             =head2 prepare_install
306              
307             $dh->prepare_install
308              
309             First prepare all the tables to be installed and the prepare just the version
310             storage
311              
312             =head2 install_version_storage
313              
314             $dh->install_version_storage
315              
316             Install the version storage and not the rest of the tables
317              
318             =head1 AUTHOR
319              
320             Arthur Axel "fREW" Schmidt <frioux+cpan@gmail.com>
321              
322             =head1 COPYRIGHT AND LICENSE
323              
324             This software is copyright (c) 2019 by Arthur Axel "fREW" Schmidt.
325              
326             This is free software; you can redistribute it and/or modify it under
327             the same terms as the Perl 5 programming language system itself.
328              
329             =cut