line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Rapi::Blog; |
2
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
623
|
use strict; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
22
|
|
4
|
1
|
|
|
1
|
|
4
|
use warnings; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
23
|
|
5
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
# ABSTRACT: RapidApp-powered blog |
7
|
|
|
|
|
|
|
|
8
|
1
|
|
|
1
|
|
414
|
use RapidApp 1.3200; |
|
1
|
|
|
|
|
22590
|
|
|
1
|
|
|
|
|
24
|
|
9
|
|
|
|
|
|
|
|
10
|
1
|
|
|
1
|
|
454
|
use Moose; |
|
1
|
|
|
|
|
410418
|
|
|
1
|
|
|
|
|
7
|
|
11
|
|
|
|
|
|
|
extends 'RapidApp::Builder'; |
12
|
|
|
|
|
|
|
|
13
|
1
|
|
|
1
|
|
6695
|
use Types::Standard qw(:all); |
|
1
|
|
|
|
|
57162
|
|
|
1
|
|
|
|
|
10
|
|
14
|
|
|
|
|
|
|
|
15
|
1
|
|
|
1
|
|
40487
|
use RapidApp::Util ':all'; |
|
1
|
|
|
|
|
1056900
|
|
|
1
|
|
|
|
|
478
|
|
16
|
1
|
|
|
1
|
|
8
|
use File::ShareDir qw(dist_dir); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
53
|
|
17
|
1
|
|
|
1
|
|
546
|
use FindBin; |
|
1
|
|
|
|
|
842
|
|
|
1
|
|
|
|
|
45
|
|
18
|
|
|
|
|
|
|
require Module::Locate; |
19
|
1
|
|
|
1
|
|
6
|
use Path::Class qw/file dir/; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
46
|
|
20
|
1
|
|
|
1
|
|
371
|
use YAML::XS 0.64 'LoadFile'; |
|
1
|
|
|
|
|
2358
|
|
|
1
|
|
|
|
|
46
|
|
21
|
|
|
|
|
|
|
|
22
|
1
|
|
|
1
|
|
443
|
use Rapi::Blog::Scaffold; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
33
|
|
23
|
1
|
|
|
1
|
|
363
|
use Rapi::Blog::Scaffold::Set; |
|
1
|
|
|
|
|
4
|
|
|
1
|
|
|
|
|
2013
|
|
24
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
our $VERSION = '1.1301'; |
26
|
|
|
|
|
|
|
our $TITLE = "Rapi::Blog v" . $VERSION; |
27
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
has 'site_path', is => 'ro', required => 1; |
29
|
|
|
|
|
|
|
has 'scaffold_path', is => 'ro', isa => Maybe[Str], default => sub { undef }; |
30
|
|
|
|
|
|
|
has 'builtin_scaffold', is => 'ro', isa => Maybe[Str], default => sub { undef }; |
31
|
|
|
|
|
|
|
has 'scaffold_config', is => 'ro', isa => HashRef, default => sub {{}}; |
32
|
|
|
|
|
|
|
has 'fallback_builtin_scaffold', is => 'ro', isa => Bool, default => sub {0}; |
33
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
has 'enable_password_reset', is => 'ro', isa => Bool, default => sub {1}; |
35
|
|
|
|
|
|
|
has 'enable_user_sign_up', is => 'ro', isa => Bool, default => sub {1}; |
36
|
|
|
|
|
|
|
has 'enable_email_login', is => 'ro', isa => Bool, default => sub {1}; |
37
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
has 'underlay_scaffolds', is => 'ro', isa => ArrayRef[Str], default => sub {[]}; |
39
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
has 'smtp_config', is => 'ro', isa => Maybe[HashRef], default => sub { undef }; |
41
|
|
|
|
|
|
|
has 'override_email_recipient', is => 'ro', isa => Maybe[Str], default => sub { undef }; |
42
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
has 'recaptcha_config', is => 'ro', isa => Maybe[HashRef[Str]], default => sub { undef }; |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
has '+base_appname', default => sub { 'Rapi::Blog::App' }; |
47
|
|
|
|
|
|
|
has '+debug', default => sub {1}; |
48
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
sub BUILD { |
50
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
51
|
0
|
0
|
|
|
|
|
print STDERR join('',' -- ',(blessed $self),' v',$self->VERSION,' -- ',"\n") if ($self->debug); |
52
|
|
|
|
|
|
|
} |
53
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
has 'share_dir', is => 'ro', isa => Str, lazy => 1, default => sub { |
55
|
|
|
|
|
|
|
my $self = shift; |
56
|
|
|
|
|
|
|
$self->_get_share_dir; |
57
|
|
|
|
|
|
|
}; |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
sub _get_share_dir { |
60
|
0
|
|
0
|
0
|
|
|
my $self = shift || __PACKAGE__; |
61
|
|
|
|
|
|
|
$ENV{RAPI_BLOG_SHARE_DIR} || ( |
62
|
0
|
|
|
0
|
|
|
try{dist_dir('Rapi-Blog')} || ( |
63
|
0
|
0
|
0
|
|
|
|
-d "$FindBin::Bin/share" ? "$FindBin::Bin/share" : |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
64
|
|
|
|
|
|
|
-d "$FindBin::Bin/../share" ? "$FindBin::Bin/../share" : |
65
|
|
|
|
|
|
|
join('',$self->_module_locate_dir,'/../../share') |
66
|
|
|
|
|
|
|
) |
67
|
|
|
|
|
|
|
) |
68
|
|
|
|
|
|
|
} |
69
|
|
|
|
|
|
|
|
70
|
|
|
|
|
|
|
sub _module_locate_dir { |
71
|
0
|
|
|
0
|
|
|
my $self = shift; |
72
|
0
|
0
|
|
|
|
|
my $pm_path = Module::Locate::locate('Rapi::Blog') or die "Failed to locate Rapi::Blog?!"; |
73
|
0
|
|
|
|
|
|
file($pm_path)->parent->stringify |
74
|
|
|
|
|
|
|
} |
75
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
has '+inject_components', default => sub { |
77
|
|
|
|
|
|
|
my $self = shift; |
78
|
|
|
|
|
|
|
my $model = 'Rapi::Blog::Model::DB'; |
79
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
my $db = $self->site_dir->file('rapi_blog.db'); |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
Module::Runtime::require_module($model); |
83
|
|
|
|
|
|
|
$model->config->{connect_info}{dsn} = "dbi:SQLite:$db"; |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
return [ |
86
|
|
|
|
|
|
|
[ $model => 'Model::DB' ], |
87
|
|
|
|
|
|
|
[ 'Rapi::Blog::Model::Mailer' => 'Model::Mailer' ], |
88
|
|
|
|
|
|
|
[ 'Rapi::Blog::Controller::Remote' => 'Controller::Remote' ], |
89
|
|
|
|
|
|
|
[ 'Rapi::Blog::Controller::Remote::PreauthAction' => 'Controller::Remote::PreauthAction' ] |
90
|
|
|
|
|
|
|
] |
91
|
|
|
|
|
|
|
}; |
92
|
|
|
|
|
|
|
|
93
|
|
|
|
|
|
|
|
94
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
has 'site_dir', is => 'ro', init_arg => undef, lazy => 1, default => sub { |
96
|
|
|
|
|
|
|
my $self = shift; |
97
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
my $Dir = dir( $self->site_path )->absolute; |
99
|
|
|
|
|
|
|
-d $Dir or die "Scaffold directory '$Dir' not found.\n"; |
100
|
|
|
|
|
|
|
|
101
|
|
|
|
|
|
|
return $Dir |
102
|
|
|
|
|
|
|
}, isa => InstanceOf['Path::Class::Dir']; |
103
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
has 'scaffolds', is => 'ro', lazy => 1, default => sub { undef }; |
106
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
has 'ScaffoldSet', is => 'ro', init_arg => undef, lazy => 1, default => sub { |
108
|
|
|
|
|
|
|
my $self = shift; |
109
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
my $scafs = $self->scaffolds || []; |
111
|
|
|
|
|
|
|
$scafs = [ $scafs ] unless (ref($scafs)||'' eq 'ARRAY'); |
112
|
|
|
|
|
|
|
$scafs = [ $self->scaffold_dir ] unless (scalar(@$scafs) > 0); |
113
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
my @list = map { |
115
|
|
|
|
|
|
|
Rapi::Blog::Scaffold->factory( $_ ) |
116
|
|
|
|
|
|
|
} @$scafs, @{ $self->_get_underlay_scaffold_dirs }; |
117
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
my $Set = Rapi::Blog::Scaffold::Set->new( Scaffolds => \@list ); |
119
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
# Apply any custom configs to the *first* scaffold: |
121
|
|
|
|
|
|
|
$self->scaffold_config and $Set->first->config->_apply_params( $self->scaffold_config ); |
122
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
$Set |
124
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
}, isa => InstanceOf['Rapi::Blog::Scaffold::Set']; |
126
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
# This exists to be able to provide access to the running Blog config, including within |
129
|
|
|
|
|
|
|
# templates, without the risk associated with providing direct access to the Rapi::Blog |
130
|
|
|
|
|
|
|
# instance outright: |
131
|
|
|
|
|
|
|
has 'BlogCfg', is => 'ro', init_arg => undef, lazy => 1, default => sub { |
132
|
|
|
|
|
|
|
my $self = shift; |
133
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
my @keys = grep { |
135
|
|
|
|
|
|
|
my $Attr = $self->meta->get_attribute($_); |
136
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
!($_ =~ /^_/) # ignore attrs with private names (i.e. start with "_") |
138
|
|
|
|
|
|
|
&& ($Attr->reader||'') eq $_ # only consider attributes with normal accessor names |
139
|
|
|
|
|
|
|
&& $Attr->has_value($self) # and already have a value |
140
|
|
|
|
|
|
|
|
141
|
|
|
|
|
|
|
} $self->meta->get_attribute_list; |
142
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
# If any normal methods are desired in the future, add them to keys here |
144
|
|
|
|
|
|
|
|
145
|
|
|
|
|
|
|
my $cfg = { map { $_ => $self->$_ } @keys }; |
146
|
|
|
|
|
|
|
|
147
|
|
|
|
|
|
|
$cfg |
148
|
|
|
|
|
|
|
}, isa => HashRef; |
149
|
|
|
|
|
|
|
|
150
|
|
|
|
|
|
|
|
151
|
|
|
|
|
|
|
|
152
|
|
|
|
|
|
|
# Single merged config object which considers, prioritizes and flattens the configs of all scaffolds |
153
|
|
|
|
|
|
|
has 'scaffold_cfg', is => 'ro', init_arg => undef, lazy => 1, default => sub { |
154
|
|
|
|
|
|
|
my $self = shift; |
155
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
my %merged = ( |
157
|
|
|
|
|
|
|
map { |
158
|
|
|
|
|
|
|
%{ $_->config->_all_as_hash } |
159
|
|
|
|
|
|
|
} reverse ($self->ScaffoldSet->all) |
160
|
|
|
|
|
|
|
); |
161
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
Rapi::Blog::Scaffold::Config->new( %merged ) |
163
|
|
|
|
|
|
|
|
164
|
|
|
|
|
|
|
}, isa => InstanceOf['Rapi::Blog::Scaffold::Config']; |
165
|
|
|
|
|
|
|
|
166
|
|
|
|
|
|
|
|
167
|
|
|
|
|
|
|
|
168
|
|
|
|
|
|
|
has 'scaffold_dir', is => 'ro', init_arg => undef, lazy => 1, default => sub { |
169
|
|
|
|
|
|
|
my $self = shift; |
170
|
|
|
|
|
|
|
|
171
|
|
|
|
|
|
|
my $path; |
172
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
if(my $scaffold_name = $self->builtin_scaffold) { |
174
|
|
|
|
|
|
|
die join('', |
175
|
|
|
|
|
|
|
" Error: don't use both 'builtin_scaffold' and 'scaffold_path' options" |
176
|
|
|
|
|
|
|
) if ($self->scaffold_path); |
177
|
|
|
|
|
|
|
my $Dir = $self->_get_builtin_scaffold_dir($scaffold_name)->absolute; |
178
|
|
|
|
|
|
|
-d $Dir or die "builtin scaffold '$scaffold_name' not found\n"; |
179
|
|
|
|
|
|
|
|
180
|
|
|
|
|
|
|
$path = $Dir->stringify; |
181
|
|
|
|
|
|
|
} |
182
|
|
|
|
|
|
|
else { |
183
|
|
|
|
|
|
|
$path = $self->scaffold_path || $self->site_dir->subdir('scaffold'); |
184
|
|
|
|
|
|
|
} |
185
|
|
|
|
|
|
|
|
186
|
|
|
|
|
|
|
my $Dir = dir( $path ); |
187
|
|
|
|
|
|
|
if(! -d $Dir) { |
188
|
|
|
|
|
|
|
if($self->fallback_builtin_scaffold) { |
189
|
|
|
|
|
|
|
my $scaffold_name = 'bootstrap-blog'; |
190
|
|
|
|
|
|
|
warn join('', |
191
|
|
|
|
|
|
|
"\n ** WARNING: local scaffold directory not found;\n --> using builtin ", |
192
|
|
|
|
|
|
|
"scaffold '$scaffold_name' (fallback_builtin_scaffold is set to true)\n\n" |
193
|
|
|
|
|
|
|
); |
194
|
|
|
|
|
|
|
$Dir = $self->_get_builtin_scaffold_dir($scaffold_name); |
195
|
|
|
|
|
|
|
-d $Dir or die join('', |
196
|
|
|
|
|
|
|
" Fatal error: fallback scaffold not found (this could indicate a ", |
197
|
|
|
|
|
|
|
"problem with your Rapi::Blog installation)\n\n" |
198
|
|
|
|
|
|
|
); |
199
|
|
|
|
|
|
|
} |
200
|
|
|
|
|
|
|
else { |
201
|
|
|
|
|
|
|
die "Scaffold directory '$Dir' not found.\n"; |
202
|
|
|
|
|
|
|
} |
203
|
|
|
|
|
|
|
} |
204
|
|
|
|
|
|
|
return $Dir |
205
|
|
|
|
|
|
|
}, isa => InstanceOf['Path::Class::Dir']; |
206
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
sub _get_builtin_scaffold_dir { |
208
|
0
|
|
|
0
|
|
|
my ($self, $scaffold_name) = @_; |
209
|
0
|
|
0
|
|
|
|
$scaffold_name ||= 'bootstrap-blog'; |
210
|
|
|
|
|
|
|
|
211
|
0
|
|
|
|
|
|
my $Scaffolds = dir( $self->share_dir )->subdir('scaffolds')->absolute; |
212
|
0
|
0
|
|
|
|
|
-d $Scaffolds or die join('', |
213
|
|
|
|
|
|
|
" Fatal error: Unable to locate scaffold share dir (this could indicate a ", |
214
|
|
|
|
|
|
|
"problem with your Rapi::Blog installation)\n\n" |
215
|
|
|
|
|
|
|
); |
216
|
|
|
|
|
|
|
|
217
|
0
|
|
|
|
|
|
$Scaffolds->subdir($scaffold_name) |
218
|
|
|
|
|
|
|
} |
219
|
|
|
|
|
|
|
|
220
|
|
|
|
|
|
|
|
221
|
|
|
|
|
|
|
after 'bootstrap' => sub { |
222
|
|
|
|
|
|
|
my $self = shift; |
223
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
my $c = $self->appname; |
225
|
|
|
|
|
|
|
$c->setup_plugins(['+Rapi::Blog::CatalystApp']); |
226
|
|
|
|
|
|
|
|
227
|
|
|
|
|
|
|
}; |
228
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
sub _get_underlay_scaffold_dirs { |
231
|
0
|
|
|
0
|
|
|
my $self = shift; |
232
|
|
|
|
|
|
|
|
233
|
0
|
|
|
|
|
|
my $CommonUnderlay = dir( $self->share_dir )->subdir('common_underlay')->absolute; |
234
|
0
|
0
|
|
|
|
|
-d $CommonUnderlay or die join('', |
235
|
|
|
|
|
|
|
" Fatal error: Unable to locate common underlay scaffold dir (this could ", |
236
|
|
|
|
|
|
|
"indicate a problem with your Rapi::Blog installation)\n\n" |
237
|
|
|
|
|
|
|
); |
238
|
|
|
|
|
|
|
|
239
|
|
|
|
|
|
|
return [ |
240
|
0
|
|
|
|
|
|
@{$self->underlay_scaffolds}, |
|
0
|
|
|
|
|
|
|
241
|
|
|
|
|
|
|
$CommonUnderlay |
242
|
|
|
|
|
|
|
] |
243
|
|
|
|
|
|
|
} |
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
|
246
|
|
|
|
|
|
|
sub _enforce_valid_recaptcha_config { |
247
|
0
|
|
|
0
|
|
|
my $self = shift; |
248
|
0
|
0
|
|
|
|
|
my $cfg = $self->recaptcha_config or return 1; # No config at all is valid |
249
|
|
|
|
|
|
|
|
250
|
0
|
|
|
|
|
|
my @valid_keys = qw/public_key private_key verify_url strict_mode/; |
251
|
0
|
|
|
|
|
|
my %keys = map {$_=>1} @valid_keys; |
|
0
|
|
|
|
|
|
|
252
|
0
|
|
|
|
|
|
for my $k (keys %$cfg) { |
253
|
0
|
0
|
|
|
|
|
$keys{$k} or die join('', |
254
|
|
|
|
|
|
|
"Unknown recaptcha_config param '$k' - ", |
255
|
|
|
|
|
|
|
"only valid params are: ",join(', ',@valid_keys) |
256
|
|
|
|
|
|
|
) |
257
|
|
|
|
|
|
|
} |
258
|
|
|
|
|
|
|
|
259
|
|
|
|
|
|
|
die "Invalid recaptcha_config - both 'public_key' and 'private_key' params are required" |
260
|
0
|
0
|
0
|
|
|
|
unless ($cfg->{public_key} && $cfg->{private_key}); |
261
|
|
|
|
|
|
|
|
262
|
0
|
0
|
|
|
|
|
if(exists $cfg->{strict_mode}) { |
263
|
0
|
|
|
|
|
|
my $v = $cfg->{strict_mode}; |
264
|
0
|
0
|
|
|
|
|
my $disp = defined $v ? "'$v'" : 'undef'; |
265
|
0
|
0
|
0
|
|
|
|
die "Bad value $disp for 'strict_mode' in recaptcha_config - must be either 1 (true) or 0 (false)\n" |
266
|
|
|
|
|
|
|
unless ("$v" eq '0' || "$v" eq '1') |
267
|
|
|
|
|
|
|
} |
268
|
|
|
|
|
|
|
} |
269
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
|
271
|
0
|
|
|
0
|
|
|
sub _build_version { $VERSION } |
272
|
0
|
|
|
0
|
|
|
sub _build_plugins { [qw/ |
273
|
|
|
|
|
|
|
RapidApp::RapidDbic |
274
|
|
|
|
|
|
|
RapidApp::AuthCore |
275
|
|
|
|
|
|
|
RapidApp::NavCore |
276
|
|
|
|
|
|
|
RapidApp::CoreSchemaAdmin |
277
|
|
|
|
|
|
|
/]} |
278
|
|
|
|
|
|
|
|
279
|
|
|
|
|
|
|
sub _build_base_config { |
280
|
0
|
|
|
0
|
|
|
my $self = shift; |
281
|
|
|
|
|
|
|
|
282
|
0
|
|
|
|
|
|
$self->_enforce_valid_recaptcha_config; |
283
|
|
|
|
|
|
|
|
284
|
0
|
|
|
|
|
|
my $tpl_dir = join('/',$self->share_dir,'templates'); |
285
|
0
|
0
|
|
|
|
|
-d $tpl_dir or die join('', |
286
|
|
|
|
|
|
|
"template dir ($tpl_dir) not found; ", |
287
|
|
|
|
|
|
|
__PACKAGE__, " may not be installed properly.\n" |
288
|
|
|
|
|
|
|
); |
289
|
|
|
|
|
|
|
|
290
|
0
|
|
|
|
|
|
my $loc_assets_dir = join('/',$self->share_dir,'assets'); |
291
|
0
|
0
|
|
|
|
|
-d $loc_assets_dir or die join('', |
292
|
|
|
|
|
|
|
"assets dir ($loc_assets_dir) not found; ", |
293
|
|
|
|
|
|
|
__PACKAGE__, " may not be installed properly.\n" |
294
|
|
|
|
|
|
|
); |
295
|
|
|
|
|
|
|
|
296
|
|
|
|
|
|
|
my $config = { |
297
|
|
|
|
|
|
|
|
298
|
|
|
|
|
|
|
'RapidApp' => { |
299
|
|
|
|
|
|
|
module_root_namespace => 'adm', |
300
|
|
|
|
|
|
|
local_assets_dir => $loc_assets_dir, |
301
|
|
|
|
|
|
|
|
302
|
|
|
|
|
|
|
load_modules => { |
303
|
|
|
|
|
|
|
sections => { |
304
|
|
|
|
|
|
|
class => 'Rapi::Blog::Module::SectionTree', |
305
|
|
|
|
|
|
|
params => {} |
306
|
|
|
|
|
|
|
} |
307
|
|
|
|
|
|
|
}, |
308
|
|
|
|
|
|
|
|
309
|
|
|
|
|
|
|
}, |
310
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
'Plugin::RapidApp::NavCore' => { |
312
|
|
|
|
|
|
|
navtree_class => 'Rapi::Blog::Module::NavTree', |
313
|
|
|
|
|
|
|
}, |
314
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
'Model::RapidApp::CoreSchema' => { |
316
|
|
|
|
|
|
|
sqlite_file => $self->site_dir->file('rapidapp_coreschema.db')->stringify |
317
|
|
|
|
|
|
|
}, |
318
|
|
|
|
|
|
|
|
319
|
|
|
|
|
|
|
'Plugin::RapidApp::AuthCore' => { |
320
|
|
|
|
|
|
|
linked_user_model => 'DB::User' |
321
|
|
|
|
|
|
|
}, |
322
|
|
|
|
|
|
|
|
323
|
|
|
|
|
|
|
'Controller::SimpleCAS' => { |
324
|
|
|
|
|
|
|
store_path => $self->site_dir->subdir('cas_store')->stringify |
325
|
|
|
|
|
|
|
}, |
326
|
|
|
|
|
|
|
|
327
|
|
|
|
|
|
|
'Plugin::RapidApp::TabGui' => { |
328
|
|
|
|
|
|
|
title => $TITLE, |
329
|
|
|
|
|
|
|
nav_title => 'Administration', |
330
|
|
|
|
|
|
|
banner_template => file($tpl_dir,'banner.html')->stringify, |
331
|
|
|
|
|
|
|
dashboard_url => '/tpl/dashboard.md', |
332
|
|
|
|
|
|
|
navtree_init_width => 190, |
333
|
|
|
|
|
|
|
}, |
334
|
|
|
|
|
|
|
|
335
|
|
|
|
|
|
|
'Controller::RapidApp::Template' => { |
336
|
|
|
|
|
|
|
root_template_prefix => '/', |
337
|
|
|
|
|
|
|
root_template => $self->scaffold_cfg->landing_page, |
338
|
|
|
|
|
|
|
read_alias_path => '/tpl', #<-- already the default |
339
|
|
|
|
|
|
|
edit_alias_path => '/tple', #<-- already the default |
340
|
|
|
|
|
|
|
default_template_extension => undef, |
341
|
|
|
|
|
|
|
include_paths => [ $tpl_dir ], |
342
|
|
|
|
|
|
|
access_class => 'Rapi::Blog::Template::AccessStore', |
343
|
|
|
|
|
|
|
access_params => { |
344
|
|
|
|
|
|
|
|
345
|
|
|
|
|
|
|
BlogCfg => $self->BlogCfg, |
346
|
|
|
|
|
|
|
ScaffoldSet => $self->ScaffoldSet, |
347
|
|
|
|
|
|
|
scaffold_cfg => $self->scaffold_cfg, |
348
|
|
|
|
|
|
|
|
349
|
|
|
|
|
|
|
#internal_post_path => $self->scaffold_cfg->internal_post_path, |
350
|
|
|
|
|
|
|
#default_view_path => $self->scaffold_cfg->default_view_path, |
351
|
|
|
|
|
|
|
|
352
|
|
|
|
|
|
|
|
353
|
|
|
|
|
|
|
#scaffold_dir => $self->scaffold_dir, |
354
|
|
|
|
|
|
|
#scaffold_cnf => $self->scaffold_cnf, |
355
|
|
|
|
|
|
|
#static_paths => $self->scaffold_cnf->{static_paths}, |
356
|
|
|
|
|
|
|
#private_paths => $self->scaffold_cnf->{private_paths}, |
357
|
|
|
|
|
|
|
#default_ext => $self->scaffold_cnf->{default_ext}, |
358
|
|
|
|
|
|
|
# |
359
|
|
|
|
|
|
|
#internal_post_path => $self->scaffold_cnf->{internal_post_path}, |
360
|
|
|
|
|
|
|
#view_wrappers => $self->scaffold_cnf->{view_wrappers}, |
361
|
|
|
|
|
|
|
#default_view_path => $self->default_view_path, |
362
|
|
|
|
|
|
|
#preview_path => $self->preview_path, |
363
|
|
|
|
|
|
|
# |
364
|
|
|
|
|
|
|
#underlay_scaffold_dirs => $self->_get_underlay_scaffold_dirs, |
365
|
|
|
|
|
|
|
|
366
|
0
|
|
|
0
|
|
|
get_Model => sub { $self->base_appname->model('DB') } |
367
|
|
|
|
|
|
|
} |
368
|
|
|
|
|
|
|
}, |
369
|
|
|
|
|
|
|
|
370
|
0
|
0
|
|
|
|
|
'Model::Mailer' => { |
371
|
|
|
|
|
|
|
smtp_config => $self->smtp_config, |
372
|
|
|
|
|
|
|
( $self->override_email_recipient ? (envelope_to => $self->override_email_recipient) : () ) |
373
|
|
|
|
|
|
|
} |
374
|
|
|
|
|
|
|
|
375
|
|
|
|
|
|
|
}; |
376
|
|
|
|
|
|
|
|
377
|
0
|
0
|
|
|
|
|
if(my $faviconPath = $self->ScaffoldSet->first_config_value_filepath('favicon')) { |
378
|
0
|
|
|
|
|
|
$config->{RapidApp}{default_favicon_url} = $faviconPath; |
379
|
|
|
|
|
|
|
} |
380
|
|
|
|
|
|
|
|
381
|
0
|
0
|
|
|
|
|
if(my $loginTpl = $self->ScaffoldSet->first_config_value_file('login')) { |
382
|
0
|
|
|
|
|
|
$config->{'Plugin::RapidApp::AuthCore'}{login_template} = $loginTpl; |
383
|
|
|
|
|
|
|
} |
384
|
|
|
|
|
|
|
|
385
|
0
|
0
|
|
|
|
|
if(my $errorTpl = $self->ScaffoldSet->first_config_value_file('error')) { |
386
|
0
|
|
|
|
|
|
$config->{'RapidApp'}{error_template} = $errorTpl; |
387
|
|
|
|
|
|
|
} |
388
|
|
|
|
|
|
|
|
389
|
0
|
|
|
|
|
|
return $config |
390
|
|
|
|
|
|
|
} |
391
|
|
|
|
|
|
|
|
392
|
|
|
|
|
|
|
1; |
393
|
|
|
|
|
|
|
|
394
|
|
|
|
|
|
|
__END__ |
395
|
|
|
|
|
|
|
|
396
|
|
|
|
|
|
|
=head1 NAME |
397
|
|
|
|
|
|
|
|
398
|
|
|
|
|
|
|
Rapi::Blog - Plack-compatible, RapidApp-based blog engine |
399
|
|
|
|
|
|
|
|
400
|
|
|
|
|
|
|
=head1 SYNOPSIS |
401
|
|
|
|
|
|
|
|
402
|
|
|
|
|
|
|
use Rapi::Blog; |
403
|
|
|
|
|
|
|
|
404
|
|
|
|
|
|
|
my $app = Rapi::Blog->new({ |
405
|
|
|
|
|
|
|
site_path => '/path/to/some-site', |
406
|
|
|
|
|
|
|
scaffold_path => '/path/to/some-site/scaffold', # default |
407
|
|
|
|
|
|
|
}); |
408
|
|
|
|
|
|
|
|
409
|
|
|
|
|
|
|
# Plack/PSGI app: |
410
|
|
|
|
|
|
|
$app->to_app |
411
|
|
|
|
|
|
|
|
412
|
|
|
|
|
|
|
Create a new site from scratch using the L<rabl.pl> utility script: |
413
|
|
|
|
|
|
|
|
414
|
|
|
|
|
|
|
rabl.pl create /path/to/some-site |
415
|
|
|
|
|
|
|
cd /path/to/some-site && plackup |
416
|
|
|
|
|
|
|
|
417
|
|
|
|
|
|
|
=head1 DESCRIPTION |
418
|
|
|
|
|
|
|
|
419
|
|
|
|
|
|
|
This is a L<Plack>-compatible blogging platform written using L<RapidApp>. This module was first |
420
|
|
|
|
|
|
|
released during The Perl Conference 2017 in Washington D.C. where a talk/demo was given on the |
421
|
|
|
|
|
|
|
platform: |
422
|
|
|
|
|
|
|
|
423
|
|
|
|
|
|
|
=begin HTML |
424
|
|
|
|
|
|
|
|
425
|
|
|
|
|
|
|
<p><a href="http://rapi.io/tpc2017"><img |
426
|
|
|
|
|
|
|
src="https://raw.githubusercontent.com/vanstyn/Rapi-Blog/master/share/tpc2017-video-preview.png" |
427
|
|
|
|
|
|
|
width="800" |
428
|
|
|
|
|
|
|
alt="Rapi::Blog talk/video" |
429
|
|
|
|
|
|
|
title="Rapi::Blog talk/video" |
430
|
|
|
|
|
|
|
/></a></p> |
431
|
|
|
|
|
|
|
|
432
|
|
|
|
|
|
|
=end HTML |
433
|
|
|
|
|
|
|
|
434
|
|
|
|
|
|
|
L<rapi.io/tpc2017|http://rapi.io/tpc2017> |
435
|
|
|
|
|
|
|
|
436
|
|
|
|
|
|
|
See L<Rapi::Blog::Manual> for more information and usage. |
437
|
|
|
|
|
|
|
|
438
|
|
|
|
|
|
|
=head1 CONFIGURATION |
439
|
|
|
|
|
|
|
|
440
|
|
|
|
|
|
|
C<Rapi::Blog> extends L<RapidApp::Builder> and supports all of its options, as well as the following |
441
|
|
|
|
|
|
|
params specific to this module: |
442
|
|
|
|
|
|
|
|
443
|
|
|
|
|
|
|
=head2 site_path |
444
|
|
|
|
|
|
|
|
445
|
|
|
|
|
|
|
Only required param - path to the directory containing the site. |
446
|
|
|
|
|
|
|
|
447
|
|
|
|
|
|
|
=head2 scaffold_path |
448
|
|
|
|
|
|
|
|
449
|
|
|
|
|
|
|
Path to the directory containing the "scaffold" of the site. This is like a document root with |
450
|
|
|
|
|
|
|
some extra functionality. |
451
|
|
|
|
|
|
|
|
452
|
|
|
|
|
|
|
If not supplied, defaults to C<'scaffold/'> within the C<site_path> directory. |
453
|
|
|
|
|
|
|
|
454
|
|
|
|
|
|
|
=head2 builtin_scaffold |
455
|
|
|
|
|
|
|
|
456
|
|
|
|
|
|
|
Alternative to C<scaffold_path>, the name of one of the builtin skeleton scaffolds to use as the |
457
|
|
|
|
|
|
|
live scaffold. This is mainly useful for dev and content-only testing. As of version C<1.0000>) there |
458
|
|
|
|
|
|
|
are two built-in scaffolds: |
459
|
|
|
|
|
|
|
|
460
|
|
|
|
|
|
|
=head3 bootstrap-blog |
461
|
|
|
|
|
|
|
|
462
|
|
|
|
|
|
|
This is the default out-of-the-box scaffold which is based on the "Blog" example from the Twitter |
463
|
|
|
|
|
|
|
Bootstrap HTML/CSS framework (v3.3.7): L<http://getbootstrap.com/examples/blog/>. This mainly exists |
464
|
|
|
|
|
|
|
to serve as a useful reference implementation of the basic features/directives provided by the |
465
|
|
|
|
|
|
|
Template API. |
466
|
|
|
|
|
|
|
|
467
|
|
|
|
|
|
|
=head3 keep-it-simple |
468
|
|
|
|
|
|
|
|
469
|
|
|
|
|
|
|
Based on the "Keep It Simple" website template by L<http://www.Styleshout.com> |
470
|
|
|
|
|
|
|
|
471
|
|
|
|
|
|
|
=head2 fallback_builtin_scaffold |
472
|
|
|
|
|
|
|
|
473
|
|
|
|
|
|
|
If set to true and the local scaffold directory doesn't exist, the default builtin skeleton scaffold |
474
|
|
|
|
|
|
|
'bootstrap-blog' will be used instead. Useful for testing and content-only scenarios. |
475
|
|
|
|
|
|
|
|
476
|
|
|
|
|
|
|
Defaults to false. |
477
|
|
|
|
|
|
|
|
478
|
|
|
|
|
|
|
=head2 smtp_config |
479
|
|
|
|
|
|
|
|
480
|
|
|
|
|
|
|
Optional HashRef of L<Email::Sender::Transport::SMTP> params which will be used by the app for |
481
|
|
|
|
|
|
|
sending E-Mails, such as password resets and other notifications. The options are passed directly |
482
|
|
|
|
|
|
|
to C<Email::Sender::Transport::SMTP->new()>. If the special param C<transport_class> is included, |
483
|
|
|
|
|
|
|
it will be used as the transport class instead of C<Email::Sender::Transport::SMTP>. If this is |
484
|
|
|
|
|
|
|
supplied, it should still be a valid L<Email::Sender::Transport> class. |
485
|
|
|
|
|
|
|
|
486
|
|
|
|
|
|
|
If this option is not supplied, E-Mails will be sent via the localhost using C<sendmail> via |
487
|
|
|
|
|
|
|
the default L<Email::Sender::Transport::Sendmail> options. |
488
|
|
|
|
|
|
|
|
489
|
|
|
|
|
|
|
=head2 override_email_recipient |
490
|
|
|
|
|
|
|
|
491
|
|
|
|
|
|
|
If set, all e-mails generated by the system will be sent to the specified address instead of normal |
492
|
|
|
|
|
|
|
recipients. |
493
|
|
|
|
|
|
|
|
494
|
|
|
|
|
|
|
=head2 recaptcha_config |
495
|
|
|
|
|
|
|
|
496
|
|
|
|
|
|
|
Optional HashRef config to enable Google reCAPTCHA v2 validation on supported forms. An account and API |
497
|
|
|
|
|
|
|
key pair must be setup with Google first. This config supports the following params: |
498
|
|
|
|
|
|
|
|
499
|
|
|
|
|
|
|
=head3 public_key |
500
|
|
|
|
|
|
|
|
501
|
|
|
|
|
|
|
Required. The public, or "SITE KEY" provided by the Google reCAPTCHA settings, after being setup in the |
502
|
|
|
|
|
|
|
Google reCAPTCHA system L<www.google.com/recaptcha/admin|http://www.google.com/recaptcha/admin> |
503
|
|
|
|
|
|
|
|
504
|
|
|
|
|
|
|
=head3 private_key |
505
|
|
|
|
|
|
|
|
506
|
|
|
|
|
|
|
Required. The private, or "SECRET KEY" provided by the Google reCAPTCHA settings, after being setup in the |
507
|
|
|
|
|
|
|
Google reCAPTCHA system L<www.google.com/recaptcha/admin|http://www.google.com/recaptcha/admin>. Both the |
508
|
|
|
|
|
|
|
C<public_key> and the C<private_key> are provided as a pair and both are required. |
509
|
|
|
|
|
|
|
|
510
|
|
|
|
|
|
|
=head3 verify_url |
511
|
|
|
|
|
|
|
|
512
|
|
|
|
|
|
|
Optional URL to use when performing the actual reCAPCTHA validation with Google. Defaults to |
513
|
|
|
|
|
|
|
C<https://www.google.com/recaptcha/api/siteverify> which should probably never need to be changed. |
514
|
|
|
|
|
|
|
|
515
|
|
|
|
|
|
|
=head3 strict_mode |
516
|
|
|
|
|
|
|
|
517
|
|
|
|
|
|
|
Optional mode (turned off by default) which can be enabled to tighten the enforcement reCAPTCHA, |
518
|
|
|
|
|
|
|
requiring it in all locations which is is setup on the server side, regardless of whether or not the |
519
|
|
|
|
|
|
|
client form is actually prompting the user with the appropriate reCAPTCHA "I am not a robot" checkbox |
520
|
|
|
|
|
|
|
dialog. In those cases of client-side forms not properly setup, they will never be able to submit |
521
|
|
|
|
|
|
|
because they will always fail reCAPTCHA validation. |
522
|
|
|
|
|
|
|
|
523
|
|
|
|
|
|
|
When this mode is off (the default) reCAPTCHA validation is only performed when both the client and |
524
|
|
|
|
|
|
|
the server are properly setup. The downside of this is that it leaves open the scenario of a spammer |
525
|
|
|
|
|
|
|
direct posting to the server instead of using the actual form. Whether or not this mode should be |
526
|
|
|
|
|
|
|
used should be based on need -- if spammers exploit this, turn it on. Otherwise, it is best to leave |
527
|
|
|
|
|
|
|
it off as it turning it on has the potential to make the site less resilient and reliable. |
528
|
|
|
|
|
|
|
|
529
|
|
|
|
|
|
|
This param accepts only 2 possible values: 1 (enabled) or 0 (disabled, which is the default). |
530
|
|
|
|
|
|
|
|
531
|
|
|
|
|
|
|
|
532
|
|
|
|
|
|
|
=head1 METHODS |
533
|
|
|
|
|
|
|
|
534
|
|
|
|
|
|
|
=head2 to_app |
535
|
|
|
|
|
|
|
|
536
|
|
|
|
|
|
|
PSGI C<$app> CodeRef. Derives from L<Plack::Component> |
537
|
|
|
|
|
|
|
|
538
|
|
|
|
|
|
|
=head1 SEE ALSO |
539
|
|
|
|
|
|
|
|
540
|
|
|
|
|
|
|
=over |
541
|
|
|
|
|
|
|
|
542
|
|
|
|
|
|
|
=item * |
543
|
|
|
|
|
|
|
|
544
|
|
|
|
|
|
|
L<rabl.pl> |
545
|
|
|
|
|
|
|
|
546
|
|
|
|
|
|
|
=item * |
547
|
|
|
|
|
|
|
|
548
|
|
|
|
|
|
|
L<Rapi::Blog::Manual> |
549
|
|
|
|
|
|
|
|
550
|
|
|
|
|
|
|
=item * |
551
|
|
|
|
|
|
|
|
552
|
|
|
|
|
|
|
L<RapidApp> |
553
|
|
|
|
|
|
|
|
554
|
|
|
|
|
|
|
=item * |
555
|
|
|
|
|
|
|
|
556
|
|
|
|
|
|
|
L<RapidApp::Builder> |
557
|
|
|
|
|
|
|
|
558
|
|
|
|
|
|
|
=item * |
559
|
|
|
|
|
|
|
|
560
|
|
|
|
|
|
|
L<Plack> |
561
|
|
|
|
|
|
|
|
562
|
|
|
|
|
|
|
=item * |
563
|
|
|
|
|
|
|
|
564
|
|
|
|
|
|
|
L<http://rapi.io/blog> |
565
|
|
|
|
|
|
|
|
566
|
|
|
|
|
|
|
=back |
567
|
|
|
|
|
|
|
|
568
|
|
|
|
|
|
|
|
569
|
|
|
|
|
|
|
=head1 AUTHOR |
570
|
|
|
|
|
|
|
|
571
|
|
|
|
|
|
|
Henry Van Styn <vanstyn@cpan.org> |
572
|
|
|
|
|
|
|
|
573
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE |
574
|
|
|
|
|
|
|
|
575
|
|
|
|
|
|
|
This software is copyright (c) 2017 by IntelliTree Solutions llc. |
576
|
|
|
|
|
|
|
|
577
|
|
|
|
|
|
|
This is free software; you can redistribute it and/or modify it under |
578
|
|
|
|
|
|
|
the same terms as the Perl 5 programming language system itself. |
579
|
|
|
|
|
|
|
|
580
|
|
|
|
|
|
|
=cut |
581
|
|
|
|
|
|
|
|
582
|
|
|
|
|
|
|
|