line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Catalyst::Model::RapidApp::CoreSchema; |
2
|
1
|
|
|
1
|
|
476
|
use Moose; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
7
|
|
3
|
|
|
|
|
|
|
extends 'Catalyst::Model::DBIC::Schema'; |
4
|
|
|
|
|
|
|
|
5
|
1
|
|
|
1
|
|
5866
|
use strict; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
18
|
|
6
|
1
|
|
|
1
|
|
5
|
use warnings; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
40
|
|
7
|
|
|
|
|
|
|
|
8
|
1
|
|
|
1
|
|
6
|
use RapidApp::Util qw(:all); |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
406
|
|
9
|
|
|
|
|
|
|
|
10
|
1
|
|
|
1
|
|
7
|
use Types::Standard qw(:all); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
18
|
|
11
|
|
|
|
|
|
|
|
12
|
1
|
|
|
1
|
|
39679
|
use Catalyst::Utils; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
22
|
|
13
|
1
|
|
|
1
|
|
5
|
use Module::Runtime; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
9
|
|
14
|
1
|
|
|
1
|
|
34
|
use Digest::MD5 qw(md5_hex); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
53
|
|
15
|
1
|
|
|
1
|
|
5
|
use Try::Tiny; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
54
|
|
16
|
1
|
|
|
1
|
|
7
|
use Path::Class qw(file dir); |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
42
|
|
17
|
|
|
|
|
|
|
|
18
|
1
|
|
|
1
|
|
997
|
use DBIx::Class::Schema::Loader; |
|
1
|
|
|
|
|
10262
|
|
|
1
|
|
|
|
|
5
|
|
19
|
1
|
|
|
1
|
|
944
|
use DBIx::Class::Schema::Diff; |
|
1
|
|
|
|
|
480518
|
|
|
1
|
|
|
|
|
74
|
|
20
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
BEGIN { |
22
|
1
|
|
|
1
|
|
9
|
require DBIx::Class::Optional::Dependencies; |
23
|
1
|
50
|
|
|
|
6
|
unless ( DBIx::Class::Optional::Dependencies->req_ok_for('deploy') ) { |
24
|
|
|
|
|
|
|
die( |
25
|
|
|
|
|
|
|
"Required DBIx::Class 'deploy' dependencies missing: " |
26
|
0
|
|
|
|
|
0
|
. ( values %{ DBIx::Class::Optional::Dependencies->req_errorlist_for('deploy') } )[0] |
|
0
|
|
|
|
|
0
|
|
27
|
|
|
|
|
|
|
); |
28
|
|
|
|
|
|
|
} |
29
|
|
|
|
|
|
|
} |
30
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
# -------------------- |
32
|
|
|
|
|
|
|
# User can set their own sqlite file/path by doing this in main app class: |
33
|
|
|
|
|
|
|
# |
34
|
|
|
|
|
|
|
# __PACKAGE__->config( |
35
|
|
|
|
|
|
|
# 'Model::RapidApp::CoreSchema' => { |
36
|
|
|
|
|
|
|
# sqlite_file => '/path/to/coreschema.db' |
37
|
|
|
|
|
|
|
# } |
38
|
|
|
|
|
|
|
# ); |
39
|
|
|
|
|
|
|
# |
40
|
|
|
|
|
|
|
# OR, relative path is local to the app home dir: |
41
|
|
|
|
|
|
|
# __PACKAGE__->config( |
42
|
|
|
|
|
|
|
# 'Model::RapidApp::CoreSchema' => { |
43
|
|
|
|
|
|
|
# sqlite_file => 'my_coreschema.db' |
44
|
|
|
|
|
|
|
# } |
45
|
|
|
|
|
|
|
# ); |
46
|
|
|
|
|
|
|
# |
47
|
|
|
|
|
|
|
# They can also set/override 'schema_class' and 'connect_info' |
48
|
|
|
|
|
|
|
# in the same place |
49
|
|
|
|
|
|
|
# |
50
|
|
|
|
|
|
|
# -------------------- |
51
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
# New: this needs to be outside like this to avoid running afoul of |
53
|
|
|
|
|
|
|
# nasty/complex load order problems with Model::DBIC |
54
|
|
|
|
|
|
|
before 'COMPONENT' => sub { |
55
|
|
|
|
|
|
|
my $class = shift; |
56
|
|
|
|
|
|
|
my $app_class = ref $_[0] || $_[0]; |
57
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
my $home = Catalyst::Utils::home($app_class) || Catalyst::Utils::class2tempdir($app_class,1); |
59
|
|
|
|
|
|
|
my $cust_cnf = try{$app_class->config->{'Model::RapidApp::CoreSchema'}} || {}; |
60
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
$cust_cnf->{sqlite_file} ||= 'rapidapp_coreschema.db'; |
62
|
|
|
|
|
|
|
my $db_file = file($cust_cnf->{sqlite_file}); |
63
|
|
|
|
|
|
|
$db_file = file(dir($home),$db_file) if ($db_file->is_relative); |
64
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
$class->config( |
66
|
|
|
|
|
|
|
schema_class => 'RapidApp::CoreSchema', |
67
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
connect_info => { |
69
|
|
|
|
|
|
|
dsn => join('','dbi:SQLite:dbname=',$db_file), |
70
|
|
|
|
|
|
|
sqlite_unicode => q{1}, |
71
|
|
|
|
|
|
|
on_connect_call => q{use_foreign_keys}, |
72
|
|
|
|
|
|
|
quote_names => q{1}, |
73
|
|
|
|
|
|
|
}, |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
# Allow user-defined config overrides: |
76
|
|
|
|
|
|
|
%$cust_cnf |
77
|
|
|
|
|
|
|
); |
78
|
|
|
|
|
|
|
}; |
79
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
sub BUILD { |
81
|
1
|
|
|
1
|
0
|
23759
|
my $self = shift; |
82
|
1
|
|
|
|
|
42
|
$self->_auto_deploy_schema( $self->schema ); |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
has 'init_admin_password', is => 'ro', isa => Str, default => 'pass'; |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
sub dsn { |
88
|
1
|
|
|
1
|
0
|
3
|
my $self = shift; |
89
|
1
|
|
|
|
|
6
|
return $self->config->{connect_info}{dsn}; |
90
|
|
|
|
|
|
|
} |
91
|
|
|
|
|
|
|
|
92
|
|
|
|
|
|
|
sub connect_user { |
93
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
94
|
0
|
|
|
|
|
0
|
return $self->config->{connect_info}{user}; |
95
|
|
|
|
|
|
|
} |
96
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
sub connect_password { |
98
|
0
|
|
|
0
|
0
|
0
|
my $self = shift; |
99
|
0
|
|
|
|
|
0
|
return $self->config->{connect_info}{password}; |
100
|
|
|
|
|
|
|
} |
101
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
# dsn for the "reference" coreschema database/file. This is used only for the |
103
|
|
|
|
|
|
|
# purposes of schema comparison |
104
|
|
|
|
|
|
|
has 'ref_dsn', is => 'ro', isa => 'Str', lazy => 1, default => sub { |
105
|
|
|
|
|
|
|
my $self = shift; |
106
|
|
|
|
|
|
|
my $path = file( |
107
|
|
|
|
|
|
|
dir(RapidApp->share_dir)->subdir('coreschema'), |
108
|
|
|
|
|
|
|
'ref_sqlite.db' |
109
|
|
|
|
|
|
|
)->resolve; |
110
|
|
|
|
|
|
|
return join('','dbi:SQLite:dbname=',$path); |
111
|
|
|
|
|
|
|
}; |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
sub _auto_deploy_schema { |
114
|
1
|
|
|
1
|
|
15
|
my $self = shift; |
115
|
1
|
|
|
|
|
3
|
my $schema = shift; |
116
|
|
|
|
|
|
|
|
117
|
1
|
|
|
|
|
6
|
my $deploy_statements = $schema->deployment_statements; |
118
|
1
|
|
|
|
|
252241
|
my $md5 = $self->get_clean_md5($deploy_statements); |
119
|
1
|
|
|
|
|
12
|
my $Rs = $schema->resultset('DeployInfo'); |
120
|
1
|
|
|
|
|
786
|
my $Source = $Rs->result_source; |
121
|
1
|
|
|
|
|
162
|
my $table = $schema->class($Source->source_name)->table; |
122
|
1
|
|
33
|
|
|
144
|
$table = (split(/\./,$table,2))[1] || $table; #<-- get 'table' for both 'db.table' and 'table' format |
123
|
1
|
|
|
|
|
3
|
my $deployRow; |
124
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
try { |
126
|
1
|
|
|
1
|
|
42
|
$deployRow = $Rs->find($md5); |
127
|
|
|
|
|
|
|
} |
128
|
|
|
|
|
|
|
catch { |
129
|
|
|
|
|
|
|
# Assume exception is due to not being deployed yet and try to deploy: |
130
|
1
|
|
|
1
|
|
7176
|
$schema->deploy; |
131
|
|
|
|
|
|
|
|
132
|
1
|
|
|
|
|
445089
|
$Rs->create({ |
133
|
|
|
|
|
|
|
md5 => $md5, |
134
|
|
|
|
|
|
|
schema_class => $self->schema_class, |
135
|
|
|
|
|
|
|
schema_version => (eval '$' . $self->schema_class . '::VERSION'), |
136
|
|
|
|
|
|
|
comment => 'DO NOT REMOVE THIS ROW', |
137
|
|
|
|
|
|
|
deployed_ddl => $deploy_statements, |
138
|
|
|
|
|
|
|
deployed_ts => DateTime->now( time_zone => 'local' ), |
139
|
|
|
|
|
|
|
}); |
140
|
|
|
|
|
|
|
|
141
|
1
|
|
|
|
|
7
|
$self->_insert_default_rows($schema); |
142
|
1
|
|
|
|
|
11
|
}; |
143
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
# If we've already been deployed and the ddl checksum matches: |
145
|
1
|
50
|
|
|
|
7
|
return 1 if ($deployRow); |
146
|
|
|
|
|
|
|
|
147
|
1
|
|
|
|
|
183
|
my $count = $Rs->count; |
148
|
1
|
|
|
|
|
3236
|
my $dsn = $self->dsn; |
149
|
|
|
|
|
|
|
|
150
|
1
|
50
|
|
|
|
87
|
die "Database error; deploy_info table ('$table') exists but is empty in CoreSchema database '$dsn'" |
151
|
|
|
|
|
|
|
unless ($count > 0); |
152
|
|
|
|
|
|
|
|
153
|
1
|
50
|
|
|
|
4
|
die "Database error; multiple rows in deploy_info table ('$table') in CoreSchema database '$dsn'" |
154
|
|
|
|
|
|
|
if ($count > 1); |
155
|
|
|
|
|
|
|
|
156
|
1
|
50
|
|
|
|
5
|
my $exist_md5 = $Rs->first->md5 or die "Database error; found deploy_info row in table '$table' " . |
157
|
|
|
|
|
|
|
"in CoreSchema database '$dsn', but it appears to be corrupt (no md5 checksum)."; |
158
|
|
|
|
|
|
|
|
159
|
1
|
50
|
|
|
|
28
|
return 1 if ($md5 eq $exist_md5); |
160
|
|
|
|
|
|
|
|
161
|
|
|
|
|
|
|
# If we're here, it means the md5 of the existing coreschema didn't match, but |
162
|
|
|
|
|
|
|
# that doesn't mean that the differences make it unsafe to use. Any change in |
163
|
|
|
|
|
|
|
# the generated deploy statements, even trivial things like quotes/whitespace, |
164
|
|
|
|
|
|
|
# will come out as a different checksum. So, we'll perform an actual diff to |
165
|
|
|
|
|
|
|
# compare to see what the actual, effective differences are, using the reference |
166
|
|
|
|
|
|
|
# sqlite database file. We're doing this instead of using the actual CoreSchema |
167
|
|
|
|
|
|
|
# classes we already have because we're not interested in differences that can |
168
|
|
|
|
|
|
|
# be caused by specific loaded components, and other code-level changes that |
169
|
|
|
|
|
|
|
# might show up. (See also GitHub Issue #47) |
170
|
|
|
|
|
|
|
|
171
|
0
|
|
|
|
|
0
|
my $Differ = DBIx::Class::Schema::Diff->new( |
172
|
|
|
|
|
|
|
old_schema => $self->_load_connect_schema_ref, |
173
|
|
|
|
|
|
|
new_schema => $self->_load_connect_schema |
174
|
|
|
|
|
|
|
); |
175
|
|
|
|
|
|
|
|
176
|
0
|
0
|
|
|
|
0
|
unless( $Differ->diff ) { |
177
|
|
|
|
|
|
|
# If there are no changes at all, then we're already done... |
178
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
# Future: |
180
|
|
|
|
|
|
|
# |
181
|
|
|
|
|
|
|
# It is fully expected that in later versions of RapidApp there will be |
182
|
|
|
|
|
|
|
# changes to the CoreSchema. Once that happens, these will be handled |
183
|
|
|
|
|
|
|
# either dynamically or via Migrations/DeploymentHandler to alter the |
184
|
|
|
|
|
|
|
# schema from known older versions to the latest. This code isn't present |
185
|
|
|
|
|
|
|
# because it hasn't happened yet -- we're still on Version 1 of the |
186
|
|
|
|
|
|
|
# schema... |
187
|
|
|
|
|
|
|
# |
188
|
|
|
|
|
|
|
# But, the larger plan is to *dynamically* handle schema changes, including |
189
|
|
|
|
|
|
|
# support for merging of user defined schemas with the CoreSchema as well |
190
|
|
|
|
|
|
|
# as user-supplied databases to be used as the CoreSchema. Obviously, |
191
|
|
|
|
|
|
|
# neither checksum or version checks are useful for these dynamic scenarios, |
192
|
|
|
|
|
|
|
# which is why the plan is to define a set of specific rules and tests |
193
|
|
|
|
|
|
|
# that will be required for a schema to be determined as suitable as the |
194
|
|
|
|
|
|
|
# CoreSchema. It is expected that DBIx::Class::Schema::Diff will do the |
195
|
|
|
|
|
|
|
# heavy lifting for this. |
196
|
|
|
|
|
|
|
# |
197
|
|
|
|
|
|
|
# None of this is happening yet, as this is a big subproject on its own, |
198
|
|
|
|
|
|
|
# but the code has been structured with this in mind. For now, the only |
199
|
|
|
|
|
|
|
# check we're doing with Schema::Diff is "all or nothing", but it supports |
200
|
|
|
|
|
|
|
# fine-grained filtering (and, in fact, this planned RapidApp feature is |
201
|
|
|
|
|
|
|
# the entire reason I wrote DBIx::Class::Schema::Diff in the first place, |
202
|
|
|
|
|
|
|
# so in that sense a lot of the work for this has already been done, just |
203
|
|
|
|
|
|
|
# not yet within the RapidApp code base itself). |
204
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
|
206
|
|
|
|
|
|
|
# TODO: for faster startup next time, add/save the new md5 so we can skip |
207
|
|
|
|
|
|
|
# all this diff work... |
208
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
|
210
|
0
|
|
|
|
|
0
|
return 1; |
211
|
|
|
|
|
|
|
} |
212
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
|
214
|
0
|
|
|
|
|
0
|
die join("\n",'','', |
215
|
|
|
|
|
|
|
" The selected CoreSchema database '$dsn' ", |
216
|
|
|
|
|
|
|
" already has a deployed schema but it does not match ", |
217
|
|
|
|
|
|
|
" the current schema.",'', |
218
|
|
|
|
|
|
|
" deployed checksum : $exist_md5", |
219
|
|
|
|
|
|
|
" expected checksum : $md5",'','', |
220
|
|
|
|
|
|
|
" Differences from the reference schema (detected by DBIx::Class::Schema::Diff):", |
221
|
|
|
|
|
|
|
'','', |
222
|
|
|
|
|
|
|
Dumper( $Differ->diff ),'','','' |
223
|
|
|
|
|
|
|
); |
224
|
|
|
|
|
|
|
} |
225
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
|
227
|
|
|
|
|
|
|
# Need to strip out comments and blank lines to make sure the md5s will be consistent |
228
|
|
|
|
|
|
|
sub clean_deploy_statements { |
229
|
1
|
|
|
1
|
0
|
3
|
my ($self, $deploy_statements) = @_; |
230
|
|
|
|
|
|
|
return join("\n", grep { |
231
|
1
|
|
100
|
|
|
75
|
! /^\-\-/ && |
|
160
|
|
|
|
|
423
|
|
232
|
|
|
|
|
|
|
! /^\s*$/ |
233
|
|
|
|
|
|
|
} split(/\r?\n/,$deploy_statements) ); |
234
|
|
|
|
|
|
|
} |
235
|
|
|
|
|
|
|
|
236
|
|
|
|
|
|
|
sub get_clean_md5 { |
237
|
1
|
|
|
1
|
0
|
3
|
my ($self, $deploy_statements) = @_; |
238
|
1
|
|
|
|
|
7
|
my $clean = $self->clean_deploy_statements($deploy_statements); |
239
|
1
|
|
|
|
|
26
|
return md5_hex($clean); |
240
|
|
|
|
|
|
|
} |
241
|
|
|
|
|
|
|
|
242
|
|
|
|
|
|
|
sub _insert_default_rows { |
243
|
1
|
|
|
1
|
|
833
|
my $self = shift; |
244
|
1
|
|
|
|
|
4
|
my $schema = shift; |
245
|
|
|
|
|
|
|
|
246
|
1
|
|
|
|
|
8
|
$schema->resultset('NavtreeNode')->create({ |
247
|
|
|
|
|
|
|
id => 0, |
248
|
|
|
|
|
|
|
pid => undef, |
249
|
|
|
|
|
|
|
text => 'DUMMY ROOT NODE', |
250
|
|
|
|
|
|
|
ordering => 0 |
251
|
|
|
|
|
|
|
}); |
252
|
|
|
|
|
|
|
|
253
|
1
|
|
|
|
|
6
|
$schema->resultset('User')->create({ |
254
|
|
|
|
|
|
|
username => 'admin', |
255
|
|
|
|
|
|
|
set_pw => $self->init_admin_password |
256
|
|
|
|
|
|
|
}); |
257
|
|
|
|
|
|
|
|
258
|
1
|
|
|
|
|
7
|
$schema->resultset('Role')->create({ |
259
|
|
|
|
|
|
|
role => 'administrator', |
260
|
|
|
|
|
|
|
description => 'Full Control' |
261
|
|
|
|
|
|
|
}); |
262
|
|
|
|
|
|
|
|
263
|
1
|
|
|
|
|
6
|
$schema->resultset('UserToRole')->create({ |
264
|
|
|
|
|
|
|
username => 'admin', |
265
|
|
|
|
|
|
|
role => 'administrator', |
266
|
|
|
|
|
|
|
}); |
267
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
} |
269
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
|
271
|
|
|
|
|
|
|
# TODO: clean up these package namespaces after we're done with them... |
272
|
|
|
|
|
|
|
sub _load_connect_schema { |
273
|
0
|
|
|
0
|
|
|
my $self = shift; |
274
|
0
|
|
0
|
|
|
|
my $class = shift || 'RapidApp::CoreSchemaLoad'; |
275
|
0
|
|
0
|
|
|
|
my $dsn = shift || $self->dsn; |
276
|
0
|
|
|
|
|
|
my $user = shift; |
277
|
0
|
|
|
|
|
|
my $password = shift; |
278
|
0
|
0
|
|
|
|
|
$user = $self->connect_user unless defined $user; |
279
|
0
|
0
|
|
|
|
|
$password = $self->connect_password unless defined $password; |
280
|
0
|
|
|
|
|
|
return DBIx::Class::Schema::Loader::make_schema_at( |
281
|
|
|
|
|
|
|
$class => { |
282
|
|
|
|
|
|
|
naming => { ALL => 'v7'}, |
283
|
|
|
|
|
|
|
use_namespaces => 1, |
284
|
|
|
|
|
|
|
use_moose => 1, |
285
|
|
|
|
|
|
|
debug => 0, |
286
|
|
|
|
|
|
|
},[ $dsn, $user, $password ] |
287
|
|
|
|
|
|
|
); |
288
|
|
|
|
|
|
|
} |
289
|
|
|
|
|
|
|
sub _load_connect_schema_ref { |
290
|
0
|
|
|
0
|
|
|
my $self = shift; |
291
|
0
|
|
|
|
|
|
$self->_load_connect_schema('RapidApp::CoreSchemaLoadRef',$self->ref_dsn,'',''); |
292
|
|
|
|
|
|
|
} |
293
|
|
|
|
|
|
|
|
294
|
|
|
|
|
|
|
|
295
|
|
|
|
|
|
|
1; |
296
|
|
|
|
|
|
|
|
297
|
|
|
|
|
|
|
__END__ |
298
|
|
|
|
|
|
|
|
299
|
|
|
|
|
|
|
=head1 NAME |
300
|
|
|
|
|
|
|
|
301
|
|
|
|
|
|
|
Catalyst::Model::RapidApp::CoreSchema - DBIC model for the CoreSchema database |
302
|
|
|
|
|
|
|
|
303
|
|
|
|
|
|
|
=head1 SYNOPSIS |
304
|
|
|
|
|
|
|
|
305
|
|
|
|
|
|
|
package MyApp; |
306
|
|
|
|
|
|
|
|
307
|
|
|
|
|
|
|
use Catalyst qw/ RapidApp::CoreSchema /; |
308
|
|
|
|
|
|
|
|
309
|
|
|
|
|
|
|
=head1 DESCRIPTION |
310
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
This is the Catalyst model which is automatically injected by the |
312
|
|
|
|
|
|
|
L<RapidApp::CoreSchema|Catalyst::Plugin::RapidApp::CoreSchema> plugin and is not |
313
|
|
|
|
|
|
|
intended to be loaded directly. |
314
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
With the default configuration, this model automatically initializes and deploys itself |
316
|
|
|
|
|
|
|
to an SQLite database file in the root of the application named C<rapidapp_coreschema.db>. |
317
|
|
|
|
|
|
|
|
318
|
|
|
|
|
|
|
The CoreSchema database is a common location used for persistence by multiple optional |
319
|
|
|
|
|
|
|
"Core" plugins, such as L<AuthCore|Catalyst::Plugin::RapidApp::AuthCore> and |
320
|
|
|
|
|
|
|
L<NavCore|Catalyst::Plugin::RapidApp::NavCore> |
321
|
|
|
|
|
|
|
|
322
|
|
|
|
|
|
|
=head1 SEE ALSO |
323
|
|
|
|
|
|
|
|
324
|
|
|
|
|
|
|
=over |
325
|
|
|
|
|
|
|
|
326
|
|
|
|
|
|
|
=item * |
327
|
|
|
|
|
|
|
|
328
|
|
|
|
|
|
|
L<RapidApp> |
329
|
|
|
|
|
|
|
|
330
|
|
|
|
|
|
|
=item * |
331
|
|
|
|
|
|
|
|
332
|
|
|
|
|
|
|
L<RapidApp::Manual::Plugins> |
333
|
|
|
|
|
|
|
|
334
|
|
|
|
|
|
|
=item * |
335
|
|
|
|
|
|
|
|
336
|
|
|
|
|
|
|
L<Catalyst::Plugin::RapidApp::CoreSchema> |
337
|
|
|
|
|
|
|
|
338
|
|
|
|
|
|
|
=item * |
339
|
|
|
|
|
|
|
|
340
|
|
|
|
|
|
|
L<Catalyst::Plugin::RapidApp::CoreSchemaAdmin> |
341
|
|
|
|
|
|
|
|
342
|
|
|
|
|
|
|
=item * |
343
|
|
|
|
|
|
|
|
344
|
|
|
|
|
|
|
L<Catalyst> |
345
|
|
|
|
|
|
|
|
346
|
|
|
|
|
|
|
=back |
347
|
|
|
|
|
|
|
|
348
|
|
|
|
|
|
|
=head1 AUTHOR |
349
|
|
|
|
|
|
|
|
350
|
|
|
|
|
|
|
Henry Van Styn <vanstyn@cpan.org> |
351
|
|
|
|
|
|
|
|
352
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE |
353
|
|
|
|
|
|
|
|
354
|
|
|
|
|
|
|
This software is copyright (c) 2013 by IntelliTree Solutions llc. |
355
|
|
|
|
|
|
|
|
356
|
|
|
|
|
|
|
This is free software; you can redistribute it and/or modify it under |
357
|
|
|
|
|
|
|
the same terms as the Perl 5 programming language system itself. |
358
|
|
|
|
|
|
|
|
359
|
|
|
|
|
|
|
=cut |