1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
package FixMyStreet::App::Controller::Contact::Enquiry;
use Moose;
use namespace::autoclean;
use Path::Tiny;
use File::Copy;
use Digest::SHA qw(sha1_hex);
use File::Basename;
BEGIN { extends 'Catalyst::Controller'; }
sub auto : Private {
my ($self, $c) = @_;
unless ( $c->cobrand->call_hook('setup_general_enquiries_stash') ) {
$c->res->redirect( '/' );
$c->detach;
}
}
# This needs to be defined here so /contact/begin doesn't get run instead.
sub begin : Private {
my ($self, $c) = @_;
$c->forward('/begin');
}
sub index : Path : Args(0) {
my ( $self, $c, $preserve_session ) = @_;
# Make sure existing files aren't lost if we're rendering this
# page as a result of validation error.
delete $c->session->{enquiry_files} unless $preserve_session;
$c->stash->{field_errors}->{name} = _("Please enter your full name.") if $c->stash->{field_errors}->{name};
}
sub submit : Path('submit') : Args(0) {
my ( $self, $c ) = @_;
unless ($c->req->method eq 'POST' && $c->forward("/report/new/check_form_submitted") ) {
$c->res->redirect( '/contact/enquiry' );
return;
}
# General enquiries are always private reports, and aren't
# located by the user on the map
$c->set_param('non_public', 1);
$c->set_param('pc', '');
$c->set_param('skipped', 1);
$c->forward('/report/new/non_map_creation', [ [ '/contact/enquiry/handle_uploads' ] ])
or $c->go('index', [ 1 ]);
$c->forward('confirm_report');
$c->stash->{success} = 1;
# Don't want these lingering around for the next time.
delete $c->session->{enquiry_files};
}
sub confirm_report : Private {
my ( $self, $c ) = @_;
my $report = $c->stash->{report};
# We don't ever want to modify an existing user, as general enquiries don't
# require any kind of email confirmation.
$report->user->insert unless $report->user->in_storage;
$report->confirm();
$report->update;
}
sub handle_uploads : Private {
my ( $self, $c ) = @_;
# NB. For simplicity's sake this relies on the UPLOAD_DIR config key provided
# when using the FileSystem PHOTO_STORAGE_BACKEND. Should your FMS site not
# be using this storage backend, you must ensure that UPLOAD_DIR is set
# in order for general enquiries uploads to work.
my $cfg = FixMyStreet->config('PHOTO_STORAGE_OPTIONS');
my $dir = $cfg ? $cfg->{UPLOAD_DIR} : FixMyStreet->config('UPLOAD_DIR');
$dir = path($dir, "enquiry_files")->absolute(FixMyStreet->path_to());
$dir->mkpath;
my $files = $c->session->{enquiry_files} || {};
foreach ($c->req->upload) {
my $upload = $c->req->upload($_);
if ($upload->type !~ /^image/) {
# It's not a photo so remove it before /photo/process_photo rejects it
delete $c->req->uploads->{$_};
# For each file, copy it into place in a subdir of PHOTO_STORAGE_OPTIONS.UPLOAD_DIR
FixMyStreet::PhotoStorage::base64_decode_upload($c, $upload);
# Hash each file to get its filename, but preserve the file extension
# so content-type is correct when POSTing to Open311.
my ($p, $n, $ext) = fileparse($upload->filename, qr/\.[^.]*/);
my $key = sha1_hex($upload->slurp) . $ext;
my $out = path($dir, $key);
unless (copy($upload->tempname, $out)) {
$c->log->info('Couldn\'t copy temp file to destination: ' . $!);
$c->stash->{photo_error} = _("Sorry, we couldn't save your file(s), please try again.");
return;
}
# Then store the file hashes in report->extra along with the original filenames
$files->{$key} = $upload->raw_basename;
}
}
$c->session->{enquiry_files} = $files;
$c->stash->{report}->set_extra_metadata(enquiry_files => $files);
}
1;
|