| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package App::USBKeyCopyCon; |
|
2
|
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
29248
|
use warnings; |
|
|
1
|
|
|
|
|
2
|
|
|
|
1
|
|
|
|
|
36
|
|
|
4
|
1
|
|
|
1
|
|
5
|
use strict; |
|
|
1
|
|
|
|
|
3
|
|
|
|
1
|
|
|
|
|
59
|
|
|
5
|
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
=head1 NAME |
|
7
|
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
App::USBKeyCopyCon - GUI console for bulk copying of USB keys |
|
9
|
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
=cut |
|
11
|
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
our $VERSION = '1.02'; |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
16
|
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
To launch the GUI application that this module implements, simply run the |
|
18
|
|
|
|
|
|
|
supplied wrapper script: |
|
19
|
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
usb-key-copy-con |
|
21
|
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
=head1 DESCRIPTION |
|
23
|
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
This module implements an application for bulk copying USB flash drives |
|
25
|
|
|
|
|
|
|
(storage devices). The application was developed to run on Linux and is |
|
26
|
|
|
|
|
|
|
probably not particularly portable to other platforms. |
|
27
|
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
From a user's perspective the operation is simple: |
|
29
|
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
=over 4 |
|
31
|
|
|
|
|
|
|
|
|
32
|
|
|
|
|
|
|
=item 1 |
|
33
|
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
insert a 'master' USB key when prompted - the contents of the key will be |
|
35
|
|
|
|
|
|
|
copied into a temporary directory on the hard drive, after which the key can be |
|
36
|
|
|
|
|
|
|
removed |
|
37
|
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
=item 2 |
|
39
|
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
insert blank keys into all available USB ports - the app will detect when each |
|
41
|
|
|
|
|
|
|
new key is inserted, start the copy process and alert the user on completion |
|
42
|
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
=item 3 |
|
44
|
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
repeat step 2 as required |
|
46
|
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
=back |
|
48
|
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
The program can write to multiple keys in parallel. It can also use filtering |
|
50
|
|
|
|
|
|
|
on device parameters to only overwrite devices which match the vendor name |
|
51
|
|
|
|
|
|
|
and storage capacity specified - other devices will be ignored. |
|
52
|
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
The specifics of reading the master key, preparing a blank key (formatting |
|
54
|
|
|
|
|
|
|
parameters etc) are implemented in short 'profile' scripts (a reader and a |
|
55
|
|
|
|
|
|
|
writer). You can supply your own profile scripts if your requirements differ |
|
56
|
|
|
|
|
|
|
from those provided. |
|
57
|
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
=head1 DEVELOPER INFORMATION |
|
59
|
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
The remainder of the documentation is targetted at developers who wish to |
|
61
|
|
|
|
|
|
|
modify or customise the application. |
|
62
|
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
The application uses the Gtk2 GUI toolkit. The wrapper script instantiates a |
|
64
|
|
|
|
|
|
|
single application object like this: |
|
65
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
use App::USBKeyCopyCon; |
|
67
|
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
App::USBKeyCopyCon->new->run; |
|
69
|
|
|
|
|
|
|
|
|
70
|
|
|
|
|
|
|
The constructor is responsible for building the user interface and the C<run> |
|
71
|
|
|
|
|
|
|
method invokes the Gtk2 event loop. UI events are dispatched as method calls |
|
72
|
|
|
|
|
|
|
on the application object. |
|
73
|
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
=cut |
|
75
|
|
|
|
|
|
|
|
|
76
|
1
|
|
|
1
|
|
492
|
use Moose; |
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
use Gtk2 -init; |
|
79
|
|
|
|
|
|
|
use Glib qw(TRUE FALSE); |
|
80
|
|
|
|
|
|
|
use Gtk2::SimpleMenu; |
|
81
|
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
use App::USBKeyCopyCon::Chrome; |
|
83
|
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
use Net::DBus; |
|
85
|
|
|
|
|
|
|
use Net::DBus::GLib; |
|
86
|
|
|
|
|
|
|
use Net::DBus::Dumper; |
|
87
|
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
use POSIX qw(:sys_wait_h); |
|
89
|
|
|
|
|
|
|
use IO::Handle qw(); |
|
90
|
|
|
|
|
|
|
use File::Path qw(mkpath rmtree); |
|
91
|
|
|
|
|
|
|
use File::Spec qw(); |
|
92
|
|
|
|
|
|
|
|
|
93
|
|
|
|
|
|
|
use Data::Dumper; |
|
94
|
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
has 'current_state' => ( is => 'rw', isa => 'Str', default => '' ); |
|
96
|
|
|
|
|
|
|
has 'sudo_path' => ( is => 'rw', isa => 'Str', default => '' ); |
|
97
|
|
|
|
|
|
|
has 'master_info' => ( is => 'rw' ); |
|
98
|
|
|
|
|
|
|
has 'options' => ( is => 'rw', default => sub { {} } ); |
|
99
|
|
|
|
|
|
|
has 'profiles' => ( is => 'rw', default => sub { {} } ); |
|
100
|
|
|
|
|
|
|
has 'selected_profile' => ( is => 'rw', isa => 'Str', default => '' ); |
|
101
|
|
|
|
|
|
|
has 'automount_state' => ( is => 'rw', isa => 'Str', default => undef ); |
|
102
|
|
|
|
|
|
|
has 'temp_root' => ( is => 'rw', isa => 'Str', default => undef ); |
|
103
|
|
|
|
|
|
|
has 'master_root' => ( is => 'rw', isa => 'Str', default => undef ); |
|
104
|
|
|
|
|
|
|
has 'mount_dir' => ( is => 'rw', isa => 'Str', default => undef ); |
|
105
|
|
|
|
|
|
|
has 'volume_label' => ( is => 'rw', isa => 'Str', default => '' ); |
|
106
|
|
|
|
|
|
|
has 'selected_sound' => ( is => 'rw', isa => 'Str', default => '' ); |
|
107
|
|
|
|
|
|
|
has 'current_keys' => ( is => 'ro', default => sub { {} } ); |
|
108
|
|
|
|
|
|
|
has 'exit_status' => ( is => 'ro', default => sub { {} } ); |
|
109
|
|
|
|
|
|
|
has 'app_win' => ( is => 'rw', isa => 'Gtk2::Window' ); |
|
110
|
|
|
|
|
|
|
has 'key_rack' => ( is => 'rw', isa => 'Gtk2::Container' ); |
|
111
|
|
|
|
|
|
|
has 'console' => ( is => 'rw', isa => 'Gtk2::TextView' ); |
|
112
|
|
|
|
|
|
|
has 'vendor_combo' => ( is => 'rw', isa => 'Gtk2::ComboBox' ); |
|
113
|
|
|
|
|
|
|
has 'vendor_entry' => ( is => 'rw', isa => 'Gtk2::Entry' ); |
|
114
|
|
|
|
|
|
|
has 'capacity_combo' => ( is => 'rw', isa => 'Gtk2::ComboBox' ); |
|
115
|
|
|
|
|
|
|
has 'capacity_entry' => ( is => 'rw', isa => 'Gtk2::Entry' ); |
|
116
|
|
|
|
|
|
|
has 'hal' => ( is => 'rw', isa => 'Net::DBus::RemoteObject' ); |
|
117
|
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
|
|
120
|
|
|
|
|
|
|
my @menu_entries = ( |
|
121
|
|
|
|
|
|
|
# name, stock id, label |
|
122
|
|
|
|
|
|
|
[ "FileMenu", undef, "_File" ], |
|
123
|
|
|
|
|
|
|
[ "EditMenu", undef, "_Edit" ], |
|
124
|
|
|
|
|
|
|
[ "HelpMenu", undef, "_Help" ], |
|
125
|
|
|
|
|
|
|
# name, stock id, label, accelerator, tooltip, action |
|
126
|
|
|
|
|
|
|
[ "New", 'gtk-new', "_New master key", "<control>N", "Re-read the master key", 'file_new' ], |
|
127
|
|
|
|
|
|
|
[ "Quit", 'gtk-quit', "_Quit", "<control>Q", "Quit", 'file_quit' ], |
|
128
|
|
|
|
|
|
|
[ "Prefs", 'gtk-preferences', "_Preferences", "<control>E", "About", 'edit_preferences' ], |
|
129
|
|
|
|
|
|
|
[ "About", 'gtk-about', "_About", "<control>A", "About", 'help_about' ], |
|
130
|
|
|
|
|
|
|
); |
|
131
|
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
my $menu_ui = "<ui> |
|
133
|
|
|
|
|
|
|
<menubar name='MenuBar'> |
|
134
|
|
|
|
|
|
|
<menu action='FileMenu'> |
|
135
|
|
|
|
|
|
|
<menuitem action='New'/> |
|
136
|
|
|
|
|
|
|
<menuitem action='Quit'/> |
|
137
|
|
|
|
|
|
|
</menu> |
|
138
|
|
|
|
|
|
|
<menu action='EditMenu'> |
|
139
|
|
|
|
|
|
|
<menuitem action='Prefs'/> |
|
140
|
|
|
|
|
|
|
</menu> |
|
141
|
|
|
|
|
|
|
<menu action='HelpMenu'> |
|
142
|
|
|
|
|
|
|
<menuitem action='About'/> |
|
143
|
|
|
|
|
|
|
</menu> |
|
144
|
|
|
|
|
|
|
</menubar> |
|
145
|
|
|
|
|
|
|
</ui>"; |
|
146
|
|
|
|
|
|
|
|
|
147
|
|
|
|
|
|
|
my %hal_key_map = ( |
|
148
|
|
|
|
|
|
|
'info.udi' => 'udi', |
|
149
|
|
|
|
|
|
|
'info.vendor' => 'vendor', |
|
150
|
|
|
|
|
|
|
'info.product' => 'product', |
|
151
|
|
|
|
|
|
|
'block.device' => 'block_device', |
|
152
|
|
|
|
|
|
|
'storage.removable.media_size' => 'media_size', |
|
153
|
|
|
|
|
|
|
'linux.sysfs_path' => 'sysfs_path', |
|
154
|
|
|
|
|
|
|
); |
|
155
|
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
my $gconf_automount_path = '/apps/nautilus/preferences/media_automount'; |
|
157
|
|
|
|
|
|
|
|
|
158
|
|
|
|
|
|
|
use constant VENDOR_EXACT => 0; |
|
159
|
|
|
|
|
|
|
use constant VENDOR_PATTERN => 1; |
|
160
|
|
|
|
|
|
|
use constant VENDOR_ANY => 2; |
|
161
|
|
|
|
|
|
|
use constant CAPACITY_EXACT => 0; |
|
162
|
|
|
|
|
|
|
use constant CAPACITY_MINIMUM => 1; |
|
163
|
|
|
|
|
|
|
use constant CAPACITY_ANY => 2; |
|
164
|
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
|
|
166
|
|
|
|
|
|
|
sub BUILD { |
|
167
|
|
|
|
|
|
|
my $self = shift; |
|
168
|
|
|
|
|
|
|
|
|
169
|
|
|
|
|
|
|
$self->check_for_root_user; |
|
170
|
|
|
|
|
|
|
$self->set_temp_root('/tmp'); |
|
171
|
|
|
|
|
|
|
$self->scan_for_profiles; |
|
172
|
|
|
|
|
|
|
$self->select_profile; |
|
173
|
|
|
|
|
|
|
$self->disable_automount; |
|
174
|
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
my($path) = __FILE__ =~ m{^(.*)[.]pm$}; |
|
176
|
|
|
|
|
|
|
$path = File::Spec->rel2abs($path) . "/copy-complete.wav"; |
|
177
|
|
|
|
|
|
|
$self->selected_sound($path); |
|
178
|
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
$self->build_ui; |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
$self->init_dbus_watcher; |
|
182
|
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
$self->require_master_key; |
|
184
|
|
|
|
|
|
|
} |
|
185
|
|
|
|
|
|
|
|
|
186
|
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
sub sudo_wrap { |
|
188
|
|
|
|
|
|
|
my($self, $command, @env_vars) = @_; |
|
189
|
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
my $sudo = $self->sudo_path or return $command; |
|
191
|
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
if($sudo =~ /gksudo/) { |
|
193
|
|
|
|
|
|
|
my $msg = "The application 'usb-key-copy-con' requires administrative " |
|
194
|
|
|
|
|
|
|
. "privileges to access USB flash drives"; |
|
195
|
|
|
|
|
|
|
return qq{$sudo --preserve-env --message "$msg" "$command"}; |
|
196
|
|
|
|
|
|
|
} |
|
197
|
|
|
|
|
|
|
|
|
198
|
|
|
|
|
|
|
my $env = join '', map { qq($_="$ENV{$_}" ) } @env_vars; |
|
199
|
|
|
|
|
|
|
return qq{$sudo $env $command} |
|
200
|
|
|
|
|
|
|
} |
|
201
|
|
|
|
|
|
|
|
|
202
|
|
|
|
|
|
|
|
|
203
|
|
|
|
|
|
|
sub find_command { |
|
204
|
|
|
|
|
|
|
my($self, $command) = @_; |
|
205
|
|
|
|
|
|
|
|
|
206
|
|
|
|
|
|
|
foreach my $dir (split /:/, $ENV{PATH}) { |
|
207
|
|
|
|
|
|
|
my $path = "$dir/$command"; |
|
208
|
|
|
|
|
|
|
return $path if -x $path; |
|
209
|
|
|
|
|
|
|
} |
|
210
|
|
|
|
|
|
|
return; |
|
211
|
|
|
|
|
|
|
} |
|
212
|
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
|
|
214
|
|
|
|
|
|
|
sub commandline_options { |
|
215
|
|
|
|
|
|
|
my $class = shift; |
|
216
|
|
|
|
|
|
|
return( |
|
217
|
|
|
|
|
|
|
'help|?', |
|
218
|
|
|
|
|
|
|
'--no-root-check|n', |
|
219
|
|
|
|
|
|
|
'--profile|p=s', |
|
220
|
|
|
|
|
|
|
'--profile-dir|d=s' |
|
221
|
|
|
|
|
|
|
); |
|
222
|
|
|
|
|
|
|
} |
|
223
|
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
|
|
225
|
|
|
|
|
|
|
sub scan_for_profiles { |
|
226
|
|
|
|
|
|
|
my $self = shift; |
|
227
|
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
my($path) = File::Spec->rel2abs(__FILE__) =~ m{^(.*)[.]pm$}; |
|
229
|
|
|
|
|
|
|
my @profile_dirs = ($path . "/profiles"); |
|
230
|
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
if(my $custom = $self->options->{'profile-dir'}) { |
|
232
|
|
|
|
|
|
|
push @profile_dirs, File::Spec->rel2abs($custom); |
|
233
|
|
|
|
|
|
|
} |
|
234
|
|
|
|
|
|
|
|
|
235
|
|
|
|
|
|
|
my $result = {}; |
|
236
|
|
|
|
|
|
|
foreach my $dir (@profile_dirs) { |
|
237
|
|
|
|
|
|
|
foreach my $script (glob("$dir/*")) { |
|
238
|
|
|
|
|
|
|
my($profile, $mode) = $script =~ m{^.*/([^/]+)-(reader|writer)[.]\w+$} |
|
239
|
|
|
|
|
|
|
or next; |
|
240
|
|
|
|
|
|
|
$result->{$profile}->{$mode} = $script; |
|
241
|
|
|
|
|
|
|
} |
|
242
|
|
|
|
|
|
|
} |
|
243
|
|
|
|
|
|
|
die "Unable to locate any profile scripts" if not keys %$result; |
|
244
|
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
$self->profiles($result); |
|
246
|
|
|
|
|
|
|
} |
|
247
|
|
|
|
|
|
|
|
|
248
|
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
sub select_profile { |
|
250
|
|
|
|
|
|
|
my($self, $profile) = @_; |
|
251
|
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
$profile ||= $self->options->{profile} || 'copyfiles'; |
|
253
|
|
|
|
|
|
|
if(not $self->profiles->{$profile}) { |
|
254
|
|
|
|
|
|
|
die "Invalid profile name: '$profile'\n" |
|
255
|
|
|
|
|
|
|
. "Known profiles: " |
|
256
|
|
|
|
|
|
|
. join(', ', keys %{$self->profiles}) |
|
257
|
|
|
|
|
|
|
. "\n"; |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
$self->selected_profile($profile); |
|
260
|
|
|
|
|
|
|
my($path) = __FILE__ =~ m{^(.*)[.]pm$}; |
|
261
|
|
|
|
|
|
|
} |
|
262
|
|
|
|
|
|
|
|
|
263
|
|
|
|
|
|
|
sub reader_script { |
|
264
|
|
|
|
|
|
|
my($self) = @_; |
|
265
|
|
|
|
|
|
|
my $profile = $self->profiles->{$self->selected_profile} or return; |
|
266
|
|
|
|
|
|
|
return $profile->{reader}; |
|
267
|
|
|
|
|
|
|
} |
|
268
|
|
|
|
|
|
|
|
|
269
|
|
|
|
|
|
|
sub writer_script { |
|
270
|
|
|
|
|
|
|
my($self) = @_; |
|
271
|
|
|
|
|
|
|
my $profile = $self->profiles->{$self->selected_profile} or return; |
|
272
|
|
|
|
|
|
|
return $profile->{writer}; |
|
273
|
|
|
|
|
|
|
} |
|
274
|
|
|
|
|
|
|
|
|
275
|
|
|
|
|
|
|
|
|
276
|
|
|
|
|
|
|
sub check_for_root_user { |
|
277
|
|
|
|
|
|
|
my $self = shift; |
|
278
|
|
|
|
|
|
|
|
|
279
|
|
|
|
|
|
|
return if $self->options->{'no-root-check'}; |
|
280
|
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
return if $> == 0; |
|
282
|
|
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
my $path = $self->find_command('gksudo') || $self->find_command('sudo'); |
|
284
|
|
|
|
|
|
|
if($path) { |
|
285
|
|
|
|
|
|
|
$self->sudo_path($path); |
|
286
|
|
|
|
|
|
|
return; |
|
287
|
|
|
|
|
|
|
} |
|
288
|
|
|
|
|
|
|
|
|
289
|
|
|
|
|
|
|
die "You must either run this program as root or install sudo\n"; |
|
290
|
|
|
|
|
|
|
} |
|
291
|
|
|
|
|
|
|
|
|
292
|
|
|
|
|
|
|
|
|
293
|
|
|
|
|
|
|
sub disable_automount { |
|
294
|
|
|
|
|
|
|
my $self = shift; |
|
295
|
|
|
|
|
|
|
|
|
296
|
|
|
|
|
|
|
my $state = `gconftool-2 --get $gconf_automount_path 2>/dev/null`; |
|
297
|
|
|
|
|
|
|
return if !defined($state) or $? != 0; |
|
298
|
|
|
|
|
|
|
|
|
299
|
|
|
|
|
|
|
chomp($state); |
|
300
|
|
|
|
|
|
|
$self->automount_state($state); |
|
301
|
|
|
|
|
|
|
system("gconftool-2 --type bool --set $gconf_automount_path false 2>/dev/null"); |
|
302
|
|
|
|
|
|
|
} |
|
303
|
|
|
|
|
|
|
|
|
304
|
|
|
|
|
|
|
|
|
305
|
|
|
|
|
|
|
sub restore_automount { |
|
306
|
|
|
|
|
|
|
my $self = shift; |
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
my $state = $self->automount_state or return; |
|
309
|
|
|
|
|
|
|
system("gconftool-2 --type bool --set $gconf_automount_path $state 2>/dev/null"); |
|
310
|
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
} |
|
312
|
|
|
|
|
|
|
|
|
313
|
|
|
|
|
|
|
|
|
314
|
|
|
|
|
|
|
sub build_ui { |
|
315
|
|
|
|
|
|
|
my $self = shift; |
|
316
|
|
|
|
|
|
|
|
|
317
|
|
|
|
|
|
|
my $window = Gtk2::Window->new; |
|
318
|
|
|
|
|
|
|
$self->app_win($window); |
|
319
|
|
|
|
|
|
|
$window->signal_connect(destroy => sub { Gtk2->main_quit; }); |
|
320
|
|
|
|
|
|
|
$window->set_title('USB Key Copying Console'); |
|
321
|
|
|
|
|
|
|
$window->set_default_size(850, 250); |
|
322
|
|
|
|
|
|
|
|
|
323
|
|
|
|
|
|
|
my $vbox = Gtk2::VBox->new(FALSE); |
|
324
|
|
|
|
|
|
|
$vbox->pack_start($self->build_menu, FALSE, FALSE, 0); |
|
325
|
|
|
|
|
|
|
$vbox->pack_start($self->build_filters, FALSE, FALSE, 0); |
|
326
|
|
|
|
|
|
|
$vbox->pack_start(Gtk2::HSeparator->new, FALSE, TRUE, 0); |
|
327
|
|
|
|
|
|
|
$vbox->pack_start($self->build_key_rack, FALSE, FALSE, 0); |
|
328
|
|
|
|
|
|
|
$vbox->pack_start($self->build_console, TRUE, TRUE, 0); |
|
329
|
|
|
|
|
|
|
$window->add($vbox); |
|
330
|
|
|
|
|
|
|
|
|
331
|
|
|
|
|
|
|
$window->show_all; |
|
332
|
|
|
|
|
|
|
} |
|
333
|
|
|
|
|
|
|
|
|
334
|
|
|
|
|
|
|
|
|
335
|
|
|
|
|
|
|
sub init_dbus_watcher { |
|
336
|
|
|
|
|
|
|
my $self = shift; |
|
337
|
|
|
|
|
|
|
|
|
338
|
|
|
|
|
|
|
my $bus = Net::DBus::GLib->system; |
|
339
|
|
|
|
|
|
|
|
|
340
|
|
|
|
|
|
|
my $hal = $bus->get_service("org.freedesktop.Hal"); |
|
341
|
|
|
|
|
|
|
|
|
342
|
|
|
|
|
|
|
my $manager = $hal->get_object( |
|
343
|
|
|
|
|
|
|
"/org/freedesktop/Hal/Manager", "org.freedesktop.Hal.Manager" |
|
344
|
|
|
|
|
|
|
); |
|
345
|
|
|
|
|
|
|
$self->hal($manager); |
|
346
|
|
|
|
|
|
|
|
|
347
|
|
|
|
|
|
|
$manager->connect_to_signal('DeviceAdded', sub { |
|
348
|
|
|
|
|
|
|
$self->hal_device_added(@_); |
|
349
|
|
|
|
|
|
|
}); |
|
350
|
|
|
|
|
|
|
|
|
351
|
|
|
|
|
|
|
$manager->connect_to_signal('DeviceRemoved', sub { |
|
352
|
|
|
|
|
|
|
$self->hal_device_removed(@_); |
|
353
|
|
|
|
|
|
|
}); |
|
354
|
|
|
|
|
|
|
} |
|
355
|
|
|
|
|
|
|
|
|
356
|
|
|
|
|
|
|
|
|
357
|
|
|
|
|
|
|
sub require_master_key { |
|
358
|
|
|
|
|
|
|
my $self = shift; |
|
359
|
|
|
|
|
|
|
|
|
360
|
|
|
|
|
|
|
if(not $self->reader_script) { |
|
361
|
|
|
|
|
|
|
return $self->ready_to_write; |
|
362
|
|
|
|
|
|
|
} |
|
363
|
|
|
|
|
|
|
$self->current_state('MASTER-WAIT'); |
|
364
|
|
|
|
|
|
|
$self->disable_filter_inputs; |
|
365
|
|
|
|
|
|
|
$self->say("Waiting for USB master key ...\n"); |
|
366
|
|
|
|
|
|
|
} |
|
367
|
|
|
|
|
|
|
|
|
368
|
|
|
|
|
|
|
|
|
369
|
|
|
|
|
|
|
sub hal_device_added { |
|
370
|
|
|
|
|
|
|
my($self, $target_udi) = @_; |
|
371
|
|
|
|
|
|
|
|
|
372
|
|
|
|
|
|
|
return unless $target_udi =~ /storage/; |
|
373
|
|
|
|
|
|
|
my $prop = $self->hal_device_properties($target_udi) or return; |
|
374
|
|
|
|
|
|
|
|
|
375
|
|
|
|
|
|
|
if($self->current_state eq 'MASTER-WAIT') { |
|
376
|
|
|
|
|
|
|
$self->start_master_read($prop); |
|
377
|
|
|
|
|
|
|
return; |
|
378
|
|
|
|
|
|
|
} |
|
379
|
|
|
|
|
|
|
elsif($self->current_state eq 'COPYING') { |
|
380
|
|
|
|
|
|
|
if($self->match_device_filter($prop)) { |
|
381
|
|
|
|
|
|
|
$self->say("Device added: $prop->{block_device}\n"); |
|
382
|
|
|
|
|
|
|
} |
|
383
|
|
|
|
|
|
|
else { |
|
384
|
|
|
|
|
|
|
$self->say(" - device ignored\n"); |
|
385
|
|
|
|
|
|
|
$prop->{ignored} = 1; |
|
386
|
|
|
|
|
|
|
} |
|
387
|
|
|
|
|
|
|
#$self->say(Dumper($prop)); |
|
388
|
|
|
|
|
|
|
$self->add_key_to_rack($prop); |
|
389
|
|
|
|
|
|
|
} |
|
390
|
|
|
|
|
|
|
} |
|
391
|
|
|
|
|
|
|
|
|
392
|
|
|
|
|
|
|
|
|
393
|
|
|
|
|
|
|
sub hal_device_removed { |
|
394
|
|
|
|
|
|
|
my($self, $target_udi) = @_; |
|
395
|
|
|
|
|
|
|
|
|
396
|
|
|
|
|
|
|
return unless $target_udi =~ /storage/; |
|
397
|
|
|
|
|
|
|
|
|
398
|
|
|
|
|
|
|
my $state = $self->current_state; |
|
399
|
|
|
|
|
|
|
if($state eq 'MASTER-COPIED') { |
|
400
|
|
|
|
|
|
|
if($self->master_info->{udi} eq $target_udi) { |
|
401
|
|
|
|
|
|
|
$self->ready_to_write; |
|
402
|
|
|
|
|
|
|
} |
|
403
|
|
|
|
|
|
|
} |
|
404
|
|
|
|
|
|
|
elsif($state eq 'COPYING') { |
|
405
|
|
|
|
|
|
|
if(my $dev = $self->current_keys->{$target_udi}) { |
|
406
|
|
|
|
|
|
|
$self->say("Device removed: $dev->{block_device}\n"); |
|
407
|
|
|
|
|
|
|
} |
|
408
|
|
|
|
|
|
|
$self->remove_key_from_rack($target_udi); |
|
409
|
|
|
|
|
|
|
} |
|
410
|
|
|
|
|
|
|
} |
|
411
|
|
|
|
|
|
|
|
|
412
|
|
|
|
|
|
|
|
|
413
|
|
|
|
|
|
|
sub ready_to_write { |
|
414
|
|
|
|
|
|
|
my($self) = @_; |
|
415
|
|
|
|
|
|
|
|
|
416
|
|
|
|
|
|
|
$self->say("Insert blank keys - copying will start automatically\n"); |
|
417
|
|
|
|
|
|
|
$self->enable_filter_inputs; |
|
418
|
|
|
|
|
|
|
$self->current_state('COPYING'); |
|
419
|
|
|
|
|
|
|
} |
|
420
|
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
|
|
422
|
|
|
|
|
|
|
sub hal_device_properties { |
|
423
|
|
|
|
|
|
|
my($self, $target_udi) = @_; |
|
424
|
|
|
|
|
|
|
|
|
425
|
|
|
|
|
|
|
foreach my $dev ( @{ $self->hal->GetAllDevicesWithProperties } ) { |
|
426
|
|
|
|
|
|
|
my($udi, $prop) = @$dev; |
|
427
|
|
|
|
|
|
|
if($udi eq $target_udi) { |
|
428
|
|
|
|
|
|
|
my $info = {}; |
|
429
|
|
|
|
|
|
|
while(my($hal_key, $key) = each %hal_key_map) { |
|
430
|
|
|
|
|
|
|
$info->{$key} = $prop->{$hal_key} or return; |
|
431
|
|
|
|
|
|
|
} |
|
432
|
|
|
|
|
|
|
($info->{dev}) = $info->{block_device} =~ m{/([^/]+)$}; |
|
433
|
|
|
|
|
|
|
#$self->say(Dumper($prop)); |
|
434
|
|
|
|
|
|
|
return $info; |
|
435
|
|
|
|
|
|
|
} |
|
436
|
|
|
|
|
|
|
} |
|
437
|
|
|
|
|
|
|
} |
|
438
|
|
|
|
|
|
|
|
|
439
|
|
|
|
|
|
|
|
|
440
|
|
|
|
|
|
|
sub match_device_filter { |
|
441
|
|
|
|
|
|
|
my($self, $key_info) = @_; |
|
442
|
|
|
|
|
|
|
|
|
443
|
|
|
|
|
|
|
my $vendor_type = $self->vendor_combo->get_active; |
|
444
|
|
|
|
|
|
|
my $vendor_text = $self->vendor_entry->get_text; |
|
445
|
|
|
|
|
|
|
my $capacity_type = $self->capacity_combo->get_active; |
|
446
|
|
|
|
|
|
|
my $capacity_text = $self->capacity_entry->get_text; |
|
447
|
|
|
|
|
|
|
|
|
448
|
|
|
|
|
|
|
if($vendor_type != VENDOR_ANY) { |
|
449
|
|
|
|
|
|
|
if($vendor_text eq '') { |
|
450
|
|
|
|
|
|
|
$self->say("Vendor filter not set"); |
|
451
|
|
|
|
|
|
|
return FALSE; |
|
452
|
|
|
|
|
|
|
} |
|
453
|
|
|
|
|
|
|
elsif($vendor_type == VENDOR_EXACT) { |
|
454
|
|
|
|
|
|
|
if($key_info->{vendor} ne $vendor_text) { |
|
455
|
|
|
|
|
|
|
$self->say("Vendor '$key_info->{vendor}' does not match"); |
|
456
|
|
|
|
|
|
|
return FALSE; |
|
457
|
|
|
|
|
|
|
} |
|
458
|
|
|
|
|
|
|
} |
|
459
|
|
|
|
|
|
|
elsif($vendor_type == VENDOR_PATTERN) { |
|
460
|
|
|
|
|
|
|
if($key_info->{vendor} !~ /$vendor_text/) { |
|
461
|
|
|
|
|
|
|
$self->say("Vendor '$key_info->{vendor}' does not match"); |
|
462
|
|
|
|
|
|
|
return FALSE; |
|
463
|
|
|
|
|
|
|
} |
|
464
|
|
|
|
|
|
|
} |
|
465
|
|
|
|
|
|
|
} |
|
466
|
|
|
|
|
|
|
|
|
467
|
|
|
|
|
|
|
if($capacity_type != CAPACITY_ANY) { |
|
468
|
|
|
|
|
|
|
if($capacity_text eq '') { |
|
469
|
|
|
|
|
|
|
$self->say("Capacity filter not set"); |
|
470
|
|
|
|
|
|
|
return FALSE; |
|
471
|
|
|
|
|
|
|
} |
|
472
|
|
|
|
|
|
|
elsif($capacity_type == CAPACITY_EXACT) { |
|
473
|
|
|
|
|
|
|
if($key_info->{media_size} != $capacity_text) { |
|
474
|
|
|
|
|
|
|
$self->say("Capacity '$key_info->{media_size}' does not match"); |
|
475
|
|
|
|
|
|
|
return FALSE; |
|
476
|
|
|
|
|
|
|
} |
|
477
|
|
|
|
|
|
|
} |
|
478
|
|
|
|
|
|
|
elsif($capacity_type == CAPACITY_MINIMUM) { |
|
479
|
|
|
|
|
|
|
if($key_info->{media_size} < $capacity_text) { |
|
480
|
|
|
|
|
|
|
$self->say("Capacity '$key_info->{media_size}' too small"); |
|
481
|
|
|
|
|
|
|
return FALSE; |
|
482
|
|
|
|
|
|
|
} |
|
483
|
|
|
|
|
|
|
} |
|
484
|
|
|
|
|
|
|
} |
|
485
|
|
|
|
|
|
|
|
|
486
|
|
|
|
|
|
|
return TRUE; |
|
487
|
|
|
|
|
|
|
} |
|
488
|
|
|
|
|
|
|
|
|
489
|
|
|
|
|
|
|
|
|
490
|
|
|
|
|
|
|
sub start_master_read { |
|
491
|
|
|
|
|
|
|
my($self, $key_info) = @_; |
|
492
|
|
|
|
|
|
|
|
|
493
|
|
|
|
|
|
|
$self->confirm_master_dialog($key_info) or return; |
|
494
|
|
|
|
|
|
|
$self->master_info($key_info); |
|
495
|
|
|
|
|
|
|
$self->vendor_combo->set_active(VENDOR_EXACT); |
|
496
|
|
|
|
|
|
|
$self->vendor_entry->set_text($key_info->{vendor}); |
|
497
|
|
|
|
|
|
|
$self->capacity_combo->set_active(CAPACITY_EXACT); |
|
498
|
|
|
|
|
|
|
$self->capacity_entry->set_text($key_info->{media_size}); |
|
499
|
|
|
|
|
|
|
|
|
500
|
|
|
|
|
|
|
$self->say("Reading master key\n"); |
|
501
|
|
|
|
|
|
|
|
|
502
|
|
|
|
|
|
|
pipe(my $rd, my $wr) or die "pipe(): $!"; |
|
503
|
|
|
|
|
|
|
my $pid = fork(); |
|
504
|
|
|
|
|
|
|
if($pid == 0) { # In the child |
|
505
|
|
|
|
|
|
|
sleep(2); |
|
506
|
|
|
|
|
|
|
$ENV{USB_BLOCK_DEVICE} = $key_info->{block_device}; |
|
507
|
|
|
|
|
|
|
$ENV{USB_MOUNT_DIR} = $self->mount_dir . "/$key_info->{dev}"; |
|
508
|
|
|
|
|
|
|
$ENV{USB_MASTER_ROOT} = $self->master_root; |
|
509
|
|
|
|
|
|
|
mkpath($ENV{USB_MOUNT_DIR}) if not -d $ENV{USB_MOUNT_DIR}; |
|
510
|
|
|
|
|
|
|
close($rd); |
|
511
|
|
|
|
|
|
|
close STDOUT; |
|
512
|
|
|
|
|
|
|
open STDOUT, '>&', $wr or die "error reopening STDOUT: $!"; |
|
513
|
|
|
|
|
|
|
close STDERR; |
|
514
|
|
|
|
|
|
|
open STDERR, '>&', $wr or die "error reopening STDERR: $!"; |
|
515
|
|
|
|
|
|
|
my $command = $self->sudo_wrap( |
|
516
|
|
|
|
|
|
|
$self->reader_script, |
|
517
|
|
|
|
|
|
|
qw(USB_BLOCK_DEVICE USB_MOUNT_DIR USB_MASTER_ROOT), |
|
518
|
|
|
|
|
|
|
); |
|
519
|
|
|
|
|
|
|
exec($command) or die "Error starting reader script: $!"; |
|
520
|
|
|
|
|
|
|
exit; # never reached; |
|
521
|
|
|
|
|
|
|
} |
|
522
|
|
|
|
|
|
|
close($wr); |
|
523
|
|
|
|
|
|
|
$rd->blocking(0); |
|
524
|
|
|
|
|
|
|
Glib::IO->add_watch( |
|
525
|
|
|
|
|
|
|
fileno($rd), ['in', 'err', 'hup'], |
|
526
|
|
|
|
|
|
|
sub { $self->on_master_pipe_read(@_); }, |
|
527
|
|
|
|
|
|
|
); |
|
528
|
|
|
|
|
|
|
$key_info->{pid} = $pid; |
|
529
|
|
|
|
|
|
|
$key_info->{fh} = $rd; |
|
530
|
|
|
|
|
|
|
|
|
531
|
|
|
|
|
|
|
$self->current_state('MASTER-COPYING'); |
|
532
|
|
|
|
|
|
|
} |
|
533
|
|
|
|
|
|
|
|
|
534
|
|
|
|
|
|
|
|
|
535
|
|
|
|
|
|
|
sub on_master_pipe_read { |
|
536
|
|
|
|
|
|
|
my($self, $fd, $cond) = @_; |
|
537
|
|
|
|
|
|
|
|
|
538
|
|
|
|
|
|
|
my $key_info = $self->master_info or return FALSE; |
|
539
|
|
|
|
|
|
|
my $fh = $key_info->{fh}; |
|
540
|
|
|
|
|
|
|
my $buffer; |
|
541
|
|
|
|
|
|
|
if(sysread($fh, $buffer, 100000)) { |
|
542
|
|
|
|
|
|
|
$self->say($buffer); |
|
543
|
|
|
|
|
|
|
return TRUE; |
|
544
|
|
|
|
|
|
|
} |
|
545
|
|
|
|
|
|
|
close($fh); |
|
546
|
|
|
|
|
|
|
delete $key_info->{fh}; |
|
547
|
|
|
|
|
|
|
return FALSE; |
|
548
|
|
|
|
|
|
|
} |
|
549
|
|
|
|
|
|
|
|
|
550
|
|
|
|
|
|
|
|
|
551
|
|
|
|
|
|
|
sub add_key_to_rack { |
|
552
|
|
|
|
|
|
|
my($self, $key_info) = @_; |
|
553
|
|
|
|
|
|
|
|
|
554
|
|
|
|
|
|
|
my $key_widget = Gtk2::VBox->new(FALSE, 0); |
|
555
|
|
|
|
|
|
|
my $pixbuf = $key_info->{ignored} |
|
556
|
|
|
|
|
|
|
? App::USBKeyCopyCon::Chrome::usb_icon('ignored') |
|
557
|
|
|
|
|
|
|
: App::USBKeyCopyCon::Chrome::usb_icon(0); |
|
558
|
|
|
|
|
|
|
my $icon = Gtk2::Image->new_from_pixbuf($pixbuf); |
|
559
|
|
|
|
|
|
|
$key_widget->pack_start($icon, FALSE, FALSE, 2); |
|
560
|
|
|
|
|
|
|
my $label = Gtk2::Label->new($key_info->{dev}); |
|
561
|
|
|
|
|
|
|
$key_widget->pack_start($label, FALSE, FALSE, 2); |
|
562
|
|
|
|
|
|
|
|
|
563
|
|
|
|
|
|
|
$self->key_rack->pack_start($key_widget, FALSE, FALSE, 0); |
|
564
|
|
|
|
|
|
|
$key_widget->show_all; |
|
565
|
|
|
|
|
|
|
$key_info->{widget} = $key_widget; |
|
566
|
|
|
|
|
|
|
$key_info->{icon_widget} = $icon; |
|
567
|
|
|
|
|
|
|
|
|
568
|
|
|
|
|
|
|
$self->current_keys->{ $key_info->{udi} } = $key_info; |
|
569
|
|
|
|
|
|
|
|
|
570
|
|
|
|
|
|
|
if(not $key_info->{ignored}) { |
|
571
|
|
|
|
|
|
|
$self->fork_copier($key_info); |
|
572
|
|
|
|
|
|
|
} |
|
573
|
|
|
|
|
|
|
} |
|
574
|
|
|
|
|
|
|
|
|
575
|
|
|
|
|
|
|
|
|
576
|
|
|
|
|
|
|
sub fork_copier { |
|
577
|
|
|
|
|
|
|
my($self, $key_info) = @_; |
|
578
|
|
|
|
|
|
|
|
|
579
|
|
|
|
|
|
|
pipe(my $rd, my $wr) or die "pipe(): $!"; |
|
580
|
|
|
|
|
|
|
my $pid = fork(); |
|
581
|
|
|
|
|
|
|
if($pid == 0) { # In the child |
|
582
|
|
|
|
|
|
|
sleep(2); |
|
583
|
|
|
|
|
|
|
$ENV{USB_BLOCK_DEVICE} = $key_info->{block_device}; |
|
584
|
|
|
|
|
|
|
$ENV{USB_MOUNT_DIR} = $self->mount_dir . "/$key_info->{dev}"; |
|
585
|
|
|
|
|
|
|
$ENV{USB_MASTER_ROOT} = $self->master_root; |
|
586
|
|
|
|
|
|
|
$ENV{USB_VOLUME_NAME} = $self->volume_label; |
|
587
|
|
|
|
|
|
|
mkpath($ENV{USB_MOUNT_DIR}) if not -d $ENV{USB_MOUNT_DIR}; |
|
588
|
|
|
|
|
|
|
close($rd); |
|
589
|
|
|
|
|
|
|
close STDOUT; |
|
590
|
|
|
|
|
|
|
open STDOUT, '>&', $wr or die "error reopening STDOUT: $!"; |
|
591
|
|
|
|
|
|
|
close STDERR; |
|
592
|
|
|
|
|
|
|
open STDERR, '>&', $wr or die "error reopening STDERR: $!"; |
|
593
|
|
|
|
|
|
|
my $command = $self->sudo_wrap( |
|
594
|
|
|
|
|
|
|
$self->writer_script, |
|
595
|
|
|
|
|
|
|
qw(USB_BLOCK_DEVICE USB_MOUNT_DIR USB_MASTER_ROOT USB_VOLUME_NAME), |
|
596
|
|
|
|
|
|
|
); |
|
597
|
|
|
|
|
|
|
exec($command) or die "Error starting copy script: $!"; |
|
598
|
|
|
|
|
|
|
exit; # never reached; |
|
599
|
|
|
|
|
|
|
} |
|
600
|
|
|
|
|
|
|
close($wr); |
|
601
|
|
|
|
|
|
|
$rd->blocking(0); |
|
602
|
|
|
|
|
|
|
Glib::IO->add_watch( |
|
603
|
|
|
|
|
|
|
fileno($rd), ['in', 'err', 'hup'], |
|
604
|
|
|
|
|
|
|
sub { $self->on_copier_pipe_read(@_); }, |
|
605
|
|
|
|
|
|
|
$key_info->{udi} |
|
606
|
|
|
|
|
|
|
); |
|
607
|
|
|
|
|
|
|
$key_info->{pid} = $pid; |
|
608
|
|
|
|
|
|
|
$key_info->{fh} = $rd; |
|
609
|
|
|
|
|
|
|
$key_info->{output} = ''; |
|
610
|
|
|
|
|
|
|
$key_info->{status} = 0; |
|
611
|
|
|
|
|
|
|
} |
|
612
|
|
|
|
|
|
|
|
|
613
|
|
|
|
|
|
|
|
|
614
|
|
|
|
|
|
|
sub on_copier_pipe_read { |
|
615
|
|
|
|
|
|
|
my($self, $fd, $cond, $udi) = @_; |
|
616
|
|
|
|
|
|
|
|
|
617
|
|
|
|
|
|
|
my $key_info = $self->current_keys->{$udi} or return FALSE; |
|
618
|
|
|
|
|
|
|
my $fh = $key_info->{fh}; |
|
619
|
|
|
|
|
|
|
my $buffer; |
|
620
|
|
|
|
|
|
|
if(sysread($fh, $buffer, 100000)) { |
|
621
|
|
|
|
|
|
|
$key_info->{output} .= $buffer; |
|
622
|
|
|
|
|
|
|
if($key_info->{output} =~ m/\A.*^\{(\d+)\/(\d+)\}/sm) { |
|
623
|
|
|
|
|
|
|
$self->update_key_progress($udi, int(9 * $1 / $2)); |
|
624
|
|
|
|
|
|
|
} |
|
625
|
|
|
|
|
|
|
return TRUE; |
|
626
|
|
|
|
|
|
|
} |
|
627
|
|
|
|
|
|
|
close($fh); |
|
628
|
|
|
|
|
|
|
delete $key_info->{fh}; |
|
629
|
|
|
|
|
|
|
return FALSE; |
|
630
|
|
|
|
|
|
|
} |
|
631
|
|
|
|
|
|
|
|
|
632
|
|
|
|
|
|
|
|
|
633
|
|
|
|
|
|
|
sub remove_key_from_rack { |
|
634
|
|
|
|
|
|
|
my($self, $udi) = @_; |
|
635
|
|
|
|
|
|
|
|
|
636
|
|
|
|
|
|
|
my $key_info = delete $self->current_keys->{$udi} or return; |
|
637
|
|
|
|
|
|
|
$self->key_rack->remove($key_info->{widget}); |
|
638
|
|
|
|
|
|
|
return; |
|
639
|
|
|
|
|
|
|
} |
|
640
|
|
|
|
|
|
|
|
|
641
|
|
|
|
|
|
|
|
|
642
|
|
|
|
|
|
|
sub update_key_progress { |
|
643
|
|
|
|
|
|
|
my($self, $udi, $status) = @_; |
|
644
|
|
|
|
|
|
|
|
|
645
|
|
|
|
|
|
|
$status = -1 if !defined $status or $status < -1 or $status > 10; |
|
646
|
|
|
|
|
|
|
|
|
647
|
|
|
|
|
|
|
my $key_info = $self->current_keys->{$udi} or return; |
|
648
|
|
|
|
|
|
|
$key_info->{status} = $status; |
|
649
|
|
|
|
|
|
|
$key_info->{icon_widget}->set_from_pixbuf( |
|
650
|
|
|
|
|
|
|
App::USBKeyCopyCon::Chrome::usb_icon($status) |
|
651
|
|
|
|
|
|
|
); |
|
652
|
|
|
|
|
|
|
} |
|
653
|
|
|
|
|
|
|
|
|
654
|
|
|
|
|
|
|
|
|
655
|
|
|
|
|
|
|
sub on_menu_file_new { |
|
656
|
|
|
|
|
|
|
my $self = shift; |
|
657
|
|
|
|
|
|
|
$self->require_master_key; |
|
658
|
|
|
|
|
|
|
} |
|
659
|
|
|
|
|
|
|
|
|
660
|
|
|
|
|
|
|
|
|
661
|
|
|
|
|
|
|
sub on_menu_file_quit { |
|
662
|
|
|
|
|
|
|
my $self = shift; |
|
663
|
|
|
|
|
|
|
# TODO: check for work in progress |
|
664
|
|
|
|
|
|
|
# TODO: check if desktop automount should be re-enabled |
|
665
|
|
|
|
|
|
|
Gtk2->main_quit; |
|
666
|
|
|
|
|
|
|
} |
|
667
|
|
|
|
|
|
|
|
|
668
|
|
|
|
|
|
|
|
|
669
|
|
|
|
|
|
|
sub on_menu_edit_preferences { |
|
670
|
|
|
|
|
|
|
my $self = shift; |
|
671
|
|
|
|
|
|
|
$self->say("Edit>Preferences - not implemented\n"); |
|
672
|
|
|
|
|
|
|
} |
|
673
|
|
|
|
|
|
|
|
|
674
|
|
|
|
|
|
|
|
|
675
|
|
|
|
|
|
|
sub on_menu_help_about { |
|
676
|
|
|
|
|
|
|
my $self = shift; |
|
677
|
|
|
|
|
|
|
|
|
678
|
|
|
|
|
|
|
my $dialog = Gtk2::Dialog->new( |
|
679
|
|
|
|
|
|
|
'About: usb-key-copy-con', |
|
680
|
|
|
|
|
|
|
$self->app_win, |
|
681
|
|
|
|
|
|
|
[qw/modal destroy-with-parent/], |
|
682
|
|
|
|
|
|
|
'gtk-close' => 'ok', |
|
683
|
|
|
|
|
|
|
); |
|
684
|
|
|
|
|
|
|
$dialog->set_default_size (90, 80); |
|
685
|
|
|
|
|
|
|
|
|
686
|
|
|
|
|
|
|
my $panel = Gtk2::VBox->new(FALSE, 12); |
|
687
|
|
|
|
|
|
|
|
|
688
|
|
|
|
|
|
|
my $title = Gtk2::Label->new; |
|
689
|
|
|
|
|
|
|
$title->set_markup("<span font_desc='sans 20'> USB Key Copy Console </span>"); |
|
690
|
|
|
|
|
|
|
$title->set_selectable(TRUE); |
|
691
|
|
|
|
|
|
|
$panel->pack_start($title, FALSE, FALSE, 10); |
|
692
|
|
|
|
|
|
|
|
|
693
|
|
|
|
|
|
|
my $version = Gtk2::Label->new; |
|
694
|
|
|
|
|
|
|
$version->set_markup("<span font_desc='sans 16'>Version: $VERSION</span>"); |
|
695
|
|
|
|
|
|
|
$version->set_selectable(TRUE); |
|
696
|
|
|
|
|
|
|
$panel->pack_start($version, FALSE, FALSE, 0); |
|
697
|
|
|
|
|
|
|
|
|
698
|
|
|
|
|
|
|
my $author = Gtk2::Label->new; |
|
699
|
|
|
|
|
|
|
my $detail = '(c) 2009 Grant McLean <grantm@cpan.org>'; |
|
700
|
|
|
|
|
|
|
$author->set_markup(" <span font_desc='sans 10'>$detail</span> "); |
|
701
|
|
|
|
|
|
|
$author->set_selectable(TRUE); |
|
702
|
|
|
|
|
|
|
$panel->pack_start($author, FALSE, FALSE, 10); |
|
703
|
|
|
|
|
|
|
|
|
704
|
|
|
|
|
|
|
$dialog->vbox->pack_start($panel, FALSE, FALSE, 4); |
|
705
|
|
|
|
|
|
|
$dialog->show_all; |
|
706
|
|
|
|
|
|
|
|
|
707
|
|
|
|
|
|
|
$dialog->run; |
|
708
|
|
|
|
|
|
|
|
|
709
|
|
|
|
|
|
|
$dialog->destroy; |
|
710
|
|
|
|
|
|
|
} |
|
711
|
|
|
|
|
|
|
|
|
712
|
|
|
|
|
|
|
|
|
713
|
|
|
|
|
|
|
sub build_menu { |
|
714
|
|
|
|
|
|
|
my $self = shift; |
|
715
|
|
|
|
|
|
|
|
|
716
|
|
|
|
|
|
|
foreach my $item (@menu_entries) { |
|
717
|
|
|
|
|
|
|
if(exists $item->[5]) { |
|
718
|
|
|
|
|
|
|
my $action = 'on_menu_' . $item->[5]; |
|
719
|
|
|
|
|
|
|
$item->[5] = sub { $self->$action(@_) }; |
|
720
|
|
|
|
|
|
|
} |
|
721
|
|
|
|
|
|
|
} |
|
722
|
|
|
|
|
|
|
my $actions = Gtk2::ActionGroup->new("Actions"); |
|
723
|
|
|
|
|
|
|
$actions->add_actions(\@menu_entries, undef); |
|
724
|
|
|
|
|
|
|
|
|
725
|
|
|
|
|
|
|
my $ui = Gtk2::UIManager->new; |
|
726
|
|
|
|
|
|
|
$ui->insert_action_group($actions, 0); |
|
727
|
|
|
|
|
|
|
$self->app_win->add_accel_group($ui->get_accel_group); |
|
728
|
|
|
|
|
|
|
|
|
729
|
|
|
|
|
|
|
$ui->add_ui_from_string ($menu_ui); |
|
730
|
|
|
|
|
|
|
|
|
731
|
|
|
|
|
|
|
return $ui->get_widget('/MenuBar'); |
|
732
|
|
|
|
|
|
|
} |
|
733
|
|
|
|
|
|
|
|
|
734
|
|
|
|
|
|
|
|
|
735
|
|
|
|
|
|
|
sub build_key_rack { |
|
736
|
|
|
|
|
|
|
my $self = shift; |
|
737
|
|
|
|
|
|
|
|
|
738
|
|
|
|
|
|
|
my $box = Gtk2::HBox->new(FALSE, 4); |
|
739
|
|
|
|
|
|
|
|
|
740
|
|
|
|
|
|
|
$self->key_rack($box); |
|
741
|
|
|
|
|
|
|
|
|
742
|
|
|
|
|
|
|
return $box; |
|
743
|
|
|
|
|
|
|
} |
|
744
|
|
|
|
|
|
|
|
|
745
|
|
|
|
|
|
|
|
|
746
|
|
|
|
|
|
|
sub build_filters { |
|
747
|
|
|
|
|
|
|
my $self = shift; |
|
748
|
|
|
|
|
|
|
|
|
749
|
|
|
|
|
|
|
my $box = Gtk2::HBox->new(FALSE, 4); |
|
750
|
|
|
|
|
|
|
|
|
751
|
|
|
|
|
|
|
my $label = Gtk2::Label->new("Filter parameters:"); |
|
752
|
|
|
|
|
|
|
$box->pack_start($label, FALSE, FALSE, 10); |
|
753
|
|
|
|
|
|
|
|
|
754
|
|
|
|
|
|
|
my $vendor_combo = Gtk2::ComboBox->new_text; |
|
755
|
|
|
|
|
|
|
$vendor_combo->append_text('Exactly match vendor'); |
|
756
|
|
|
|
|
|
|
$vendor_combo->append_text('Pattern match vendor'); |
|
757
|
|
|
|
|
|
|
$vendor_combo->append_text('Match any vendor'); |
|
758
|
|
|
|
|
|
|
$vendor_combo->set_active(VENDOR_EXACT); |
|
759
|
|
|
|
|
|
|
#$vendor_combo->signal_connect(changed => sub { $self->apply_filter(); }); |
|
760
|
|
|
|
|
|
|
$box->pack_start($vendor_combo, FALSE, FALSE, 10); |
|
761
|
|
|
|
|
|
|
$self->vendor_combo($vendor_combo); |
|
762
|
|
|
|
|
|
|
|
|
763
|
|
|
|
|
|
|
my $vendor_entry = Gtk2::Entry->new; |
|
764
|
|
|
|
|
|
|
$vendor_entry->set_width_chars(11); |
|
765
|
|
|
|
|
|
|
$vendor_entry->set_text(''); |
|
766
|
|
|
|
|
|
|
$box->pack_start($vendor_entry, FALSE, FALSE, 0); |
|
767
|
|
|
|
|
|
|
$self->vendor_entry($vendor_entry); |
|
768
|
|
|
|
|
|
|
|
|
769
|
|
|
|
|
|
|
my $capacity_combo = Gtk2::ComboBox->new_text; |
|
770
|
|
|
|
|
|
|
$capacity_combo->append_text('Exactly match capacity'); |
|
771
|
|
|
|
|
|
|
$capacity_combo->append_text('Match minimum capacity'); |
|
772
|
|
|
|
|
|
|
$capacity_combo->append_text('Match any capacity'); |
|
773
|
|
|
|
|
|
|
$capacity_combo->set_active(CAPACITY_EXACT); |
|
774
|
|
|
|
|
|
|
#$capacity_combo->signal_connect(changed => sub { $self->apply_filter(); }); |
|
775
|
|
|
|
|
|
|
$box->pack_start($capacity_combo, FALSE, FALSE, 10); |
|
776
|
|
|
|
|
|
|
$self->capacity_combo($capacity_combo); |
|
777
|
|
|
|
|
|
|
|
|
778
|
|
|
|
|
|
|
my $capacity_entry = Gtk2::Entry->new; |
|
779
|
|
|
|
|
|
|
$capacity_entry->set_width_chars(11); |
|
780
|
|
|
|
|
|
|
$capacity_entry->set_text(''); |
|
781
|
|
|
|
|
|
|
$box->pack_start($capacity_entry, FALSE, FALSE, 0); |
|
782
|
|
|
|
|
|
|
$self->capacity_entry($capacity_entry); |
|
783
|
|
|
|
|
|
|
|
|
784
|
|
|
|
|
|
|
return $box; |
|
785
|
|
|
|
|
|
|
} |
|
786
|
|
|
|
|
|
|
|
|
787
|
|
|
|
|
|
|
|
|
788
|
|
|
|
|
|
|
sub disable_filter_inputs { shift->_set_filter_sensitive(FALSE) } |
|
789
|
|
|
|
|
|
|
sub enable_filter_inputs { shift->_set_filter_sensitive(TRUE) } |
|
790
|
|
|
|
|
|
|
|
|
791
|
|
|
|
|
|
|
|
|
792
|
|
|
|
|
|
|
sub _set_filter_sensitive { |
|
793
|
|
|
|
|
|
|
my($self, $state) = @_; |
|
794
|
|
|
|
|
|
|
|
|
795
|
|
|
|
|
|
|
$self->vendor_combo->set_sensitive($state); |
|
796
|
|
|
|
|
|
|
$self->vendor_entry->set_sensitive($state); |
|
797
|
|
|
|
|
|
|
$self->capacity_combo->set_sensitive($state); |
|
798
|
|
|
|
|
|
|
$self->capacity_entry->set_sensitive($state); |
|
799
|
|
|
|
|
|
|
} |
|
800
|
|
|
|
|
|
|
|
|
801
|
|
|
|
|
|
|
|
|
802
|
|
|
|
|
|
|
sub build_console { |
|
803
|
|
|
|
|
|
|
my $self = shift; |
|
804
|
|
|
|
|
|
|
|
|
805
|
|
|
|
|
|
|
my $scrolled_window = Gtk2::ScrolledWindow->new; |
|
806
|
|
|
|
|
|
|
$scrolled_window->set_policy('automatic', 'automatic'); |
|
807
|
|
|
|
|
|
|
$scrolled_window->set_shadow_type('in'); |
|
808
|
|
|
|
|
|
|
|
|
809
|
|
|
|
|
|
|
my $buffer = Gtk2::TextBuffer->new(undef); |
|
810
|
|
|
|
|
|
|
$buffer->delete($buffer->get_bounds); |
|
811
|
|
|
|
|
|
|
|
|
812
|
|
|
|
|
|
|
my $console = Gtk2::TextView->new_with_buffer($buffer); |
|
813
|
|
|
|
|
|
|
$console->set_editable(FALSE); |
|
814
|
|
|
|
|
|
|
$console->set_cursor_visible(FALSE); |
|
815
|
|
|
|
|
|
|
$console->set_wrap_mode('char'); |
|
816
|
|
|
|
|
|
|
|
|
817
|
|
|
|
|
|
|
my $end_mark = $buffer->create_mark( 'end', $buffer->get_end_iter, FALSE); |
|
818
|
|
|
|
|
|
|
$buffer->signal_connect( |
|
819
|
|
|
|
|
|
|
insert_text => sub { |
|
820
|
|
|
|
|
|
|
$console->scroll_to_mark( $end_mark, 0.0, TRUE, 0.0, 0.0 ); |
|
821
|
|
|
|
|
|
|
} |
|
822
|
|
|
|
|
|
|
); |
|
823
|
|
|
|
|
|
|
|
|
824
|
|
|
|
|
|
|
$self->console($console); |
|
825
|
|
|
|
|
|
|
|
|
826
|
|
|
|
|
|
|
$scrolled_window->add($console); |
|
827
|
|
|
|
|
|
|
|
|
828
|
|
|
|
|
|
|
return $scrolled_window; |
|
829
|
|
|
|
|
|
|
} |
|
830
|
|
|
|
|
|
|
|
|
831
|
|
|
|
|
|
|
|
|
832
|
|
|
|
|
|
|
sub say { |
|
833
|
|
|
|
|
|
|
my($self, $msg) = @_; |
|
834
|
|
|
|
|
|
|
|
|
835
|
|
|
|
|
|
|
my $console = $self->console; |
|
836
|
|
|
|
|
|
|
my $buffer = $console->get_buffer; |
|
837
|
|
|
|
|
|
|
my $end = $buffer->get_end_iter; |
|
838
|
|
|
|
|
|
|
$buffer->insert ($end, $msg); |
|
839
|
|
|
|
|
|
|
} |
|
840
|
|
|
|
|
|
|
|
|
841
|
|
|
|
|
|
|
|
|
842
|
|
|
|
|
|
|
sub play_sound_file { |
|
843
|
|
|
|
|
|
|
my($self, $sound_file) = @_; |
|
844
|
|
|
|
|
|
|
|
|
845
|
|
|
|
|
|
|
$sound_file ||= $self->selected_sound; |
|
846
|
|
|
|
|
|
|
|
|
847
|
|
|
|
|
|
|
if(-r $sound_file) { |
|
848
|
|
|
|
|
|
|
system("play $sound_file >/dev/null 2>&1 &"); |
|
849
|
|
|
|
|
|
|
} |
|
850
|
|
|
|
|
|
|
} |
|
851
|
|
|
|
|
|
|
|
|
852
|
|
|
|
|
|
|
|
|
853
|
|
|
|
|
|
|
sub confirm_master_dialog { |
|
854
|
|
|
|
|
|
|
my($self, $key_info) = @_; |
|
855
|
|
|
|
|
|
|
|
|
856
|
|
|
|
|
|
|
my $dialog = Gtk2::Dialog->new( |
|
857
|
|
|
|
|
|
|
"USB Master Key", |
|
858
|
|
|
|
|
|
|
$self->app_win, |
|
859
|
|
|
|
|
|
|
[qw/modal destroy-with-parent/], |
|
860
|
|
|
|
|
|
|
'gtk-cancel' => 'cancel', |
|
861
|
|
|
|
|
|
|
'Read Master Key' => 'ok', |
|
862
|
|
|
|
|
|
|
); |
|
863
|
|
|
|
|
|
|
$dialog->set_default_size (90, 80); |
|
864
|
|
|
|
|
|
|
|
|
865
|
|
|
|
|
|
|
my $table = Gtk2::Table->new(1, 3, FALSE); |
|
866
|
|
|
|
|
|
|
|
|
867
|
|
|
|
|
|
|
my @pack_opts = ( ['expand', 'fill'], ['expand', 'fill'], 4, 2); |
|
868
|
|
|
|
|
|
|
my $row = 0; |
|
869
|
|
|
|
|
|
|
|
|
870
|
|
|
|
|
|
|
my $v_label = Gtk2::Label->new; |
|
871
|
|
|
|
|
|
|
$v_label->set_markup('<b>Vendor:</b>'); |
|
872
|
|
|
|
|
|
|
$v_label->set_alignment(0, 0.5); |
|
873
|
|
|
|
|
|
|
$table->attach($v_label, 0, 1, $row, $row + 1, @pack_opts); |
|
874
|
|
|
|
|
|
|
|
|
875
|
|
|
|
|
|
|
my $v_value = Gtk2::Label->new($key_info->{vendor}); |
|
876
|
|
|
|
|
|
|
$v_value->set_alignment(0, 0.5); |
|
877
|
|
|
|
|
|
|
$table->attach($v_value, 1, 2, $row, $row + 1, @pack_opts); |
|
878
|
|
|
|
|
|
|
$row++; |
|
879
|
|
|
|
|
|
|
|
|
880
|
|
|
|
|
|
|
my $c_label = Gtk2::Label->new; |
|
881
|
|
|
|
|
|
|
$c_label->set_markup('<b>Total Capacity:</b>'); |
|
882
|
|
|
|
|
|
|
$c_label->set_alignment(0, 0.5); |
|
883
|
|
|
|
|
|
|
$table->attach($c_label, 0, 1, $row, $row + 1, @pack_opts); |
|
884
|
|
|
|
|
|
|
|
|
885
|
|
|
|
|
|
|
my $media_size = $key_info->{media_size}; |
|
886
|
|
|
|
|
|
|
1 while $media_size =~ s{^([-+]?\d+)(\d{3})}{$1,$2}; |
|
887
|
|
|
|
|
|
|
my $c_value = Gtk2::Label->new("$media_size bytes"); |
|
888
|
|
|
|
|
|
|
$c_value->set_alignment(0, 0.5); |
|
889
|
|
|
|
|
|
|
$table->attach($c_value, 1, 2, $row, $row + 1, @pack_opts); |
|
890
|
|
|
|
|
|
|
$row++; |
|
891
|
|
|
|
|
|
|
|
|
892
|
|
|
|
|
|
|
my $volume_label = $self->get_volume_label($key_info->{block_device}); |
|
893
|
|
|
|
|
|
|
if($volume_label) { |
|
894
|
|
|
|
|
|
|
my $l_label = Gtk2::Label->new; |
|
895
|
|
|
|
|
|
|
$l_label->set_markup('<b>Volume Label:</b>'); |
|
896
|
|
|
|
|
|
|
$l_label->set_alignment(0, 0.5); |
|
897
|
|
|
|
|
|
|
$table->attach($l_label, 0, 1, $row, $row + 1, @pack_opts); |
|
898
|
|
|
|
|
|
|
|
|
899
|
|
|
|
|
|
|
my $l_value = Gtk2::Label->new($volume_label); |
|
900
|
|
|
|
|
|
|
$l_value->set_alignment(0, 0.5); |
|
901
|
|
|
|
|
|
|
$table->attach($l_value, 1, 2, $row, $row + 1, @pack_opts); |
|
902
|
|
|
|
|
|
|
$row++; |
|
903
|
|
|
|
|
|
|
} |
|
904
|
|
|
|
|
|
|
|
|
905
|
|
|
|
|
|
|
my $t_label = Gtk2::Label->new; |
|
906
|
|
|
|
|
|
|
$t_label->set_markup('<b>Temp Folder:</b>'); |
|
907
|
|
|
|
|
|
|
$t_label->set_alignment(0, 0.5); |
|
908
|
|
|
|
|
|
|
$table->attach($t_label, 0, 1, $row, $row + 1, @pack_opts); |
|
909
|
|
|
|
|
|
|
|
|
910
|
|
|
|
|
|
|
my $t_chooser = Gtk2::FileChooserButton->new( |
|
911
|
|
|
|
|
|
|
'Select a folder', 'select-folder' |
|
912
|
|
|
|
|
|
|
); |
|
913
|
|
|
|
|
|
|
$t_chooser->set_filename('/tmp'); # TODO fixme! |
|
914
|
|
|
|
|
|
|
$table->attach($t_chooser, 1, 2, $row, $row + 1, @pack_opts); |
|
915
|
|
|
|
|
|
|
$row++; |
|
916
|
|
|
|
|
|
|
|
|
917
|
|
|
|
|
|
|
my $p_label = Gtk2::Label->new; |
|
918
|
|
|
|
|
|
|
$p_label->set_markup('<b>Copying Profile:</b>'); |
|
919
|
|
|
|
|
|
|
$p_label->set_alignment(0, 0.5); |
|
920
|
|
|
|
|
|
|
$table->attach($p_label, 0, 1, $row, $row + 1, @pack_opts); |
|
921
|
|
|
|
|
|
|
|
|
922
|
|
|
|
|
|
|
my $profile_combo = Gtk2::ComboBox->new_text; |
|
923
|
|
|
|
|
|
|
my $profiles = $self->profiles; |
|
924
|
|
|
|
|
|
|
my $selected = $self->selected_profile; |
|
925
|
|
|
|
|
|
|
my @profile_names = sort keys %$profiles; |
|
926
|
|
|
|
|
|
|
my $i = 0; |
|
927
|
|
|
|
|
|
|
foreach my $key (@profile_names) { |
|
928
|
|
|
|
|
|
|
next unless $profiles->{$key}->{reader}; |
|
929
|
|
|
|
|
|
|
$profile_combo->append_text($key); |
|
930
|
|
|
|
|
|
|
$profile_combo->set_active($i) if $key eq $selected; |
|
931
|
|
|
|
|
|
|
$i++; |
|
932
|
|
|
|
|
|
|
} |
|
933
|
|
|
|
|
|
|
$table->attach($profile_combo, 1, 2, $row, $row + 1, @pack_opts); |
|
934
|
|
|
|
|
|
|
$row++; |
|
935
|
|
|
|
|
|
|
|
|
936
|
|
|
|
|
|
|
$table->show_all; |
|
937
|
|
|
|
|
|
|
$dialog->vbox->pack_start($table, FALSE, FALSE, 4); |
|
938
|
|
|
|
|
|
|
|
|
939
|
|
|
|
|
|
|
my $result; |
|
940
|
|
|
|
|
|
|
while(!$result or $result eq 'none') { |
|
941
|
|
|
|
|
|
|
$result = $dialog->run; |
|
942
|
|
|
|
|
|
|
} |
|
943
|
|
|
|
|
|
|
my $temp_root = $t_chooser->get_filename; |
|
944
|
|
|
|
|
|
|
|
|
945
|
|
|
|
|
|
|
$dialog->destroy; |
|
946
|
|
|
|
|
|
|
return if $result ne 'ok'; |
|
947
|
|
|
|
|
|
|
|
|
948
|
|
|
|
|
|
|
$self->set_temp_root($temp_root); |
|
949
|
|
|
|
|
|
|
$self->volume_label($volume_label); |
|
950
|
|
|
|
|
|
|
$self->select_profile($profile_names[$profile_combo->get_active]); |
|
951
|
|
|
|
|
|
|
|
|
952
|
|
|
|
|
|
|
return TRUE; |
|
953
|
|
|
|
|
|
|
} |
|
954
|
|
|
|
|
|
|
|
|
955
|
|
|
|
|
|
|
|
|
956
|
|
|
|
|
|
|
sub get_volume_label { |
|
957
|
|
|
|
|
|
|
my($self, $device) = @_; |
|
958
|
|
|
|
|
|
|
|
|
959
|
|
|
|
|
|
|
$device .= '1'; # examine first partition |
|
960
|
|
|
|
|
|
|
my $command = $self->sudo_wrap("dosfslabel $device"); |
|
961
|
|
|
|
|
|
|
my $label = `$command 2>/dev/null`; |
|
962
|
|
|
|
|
|
|
chomp($label) if defined $label; |
|
963
|
|
|
|
|
|
|
return $label; |
|
964
|
|
|
|
|
|
|
} |
|
965
|
|
|
|
|
|
|
|
|
966
|
|
|
|
|
|
|
|
|
967
|
|
|
|
|
|
|
sub tick { |
|
968
|
|
|
|
|
|
|
my $self = shift; |
|
969
|
|
|
|
|
|
|
|
|
970
|
|
|
|
|
|
|
my $exit_status = $self->exit_status; |
|
971
|
|
|
|
|
|
|
return TRUE unless keys %$exit_status; |
|
972
|
|
|
|
|
|
|
|
|
973
|
|
|
|
|
|
|
my $state = $self->current_state; |
|
974
|
|
|
|
|
|
|
if($state eq 'MASTER-COPYING') { |
|
975
|
|
|
|
|
|
|
$self->master_copy_finished($exit_status); |
|
976
|
|
|
|
|
|
|
} |
|
977
|
|
|
|
|
|
|
elsif($state eq 'COPYING') { |
|
978
|
|
|
|
|
|
|
$self->copy_finished($exit_status); |
|
979
|
|
|
|
|
|
|
} |
|
980
|
|
|
|
|
|
|
return TRUE; |
|
981
|
|
|
|
|
|
|
} |
|
982
|
|
|
|
|
|
|
|
|
983
|
|
|
|
|
|
|
|
|
984
|
|
|
|
|
|
|
sub master_copy_finished { |
|
985
|
|
|
|
|
|
|
my($self, $exit_status) = @_; |
|
986
|
|
|
|
|
|
|
|
|
987
|
|
|
|
|
|
|
my $pid = $self->master_info->{pid} or return; |
|
988
|
|
|
|
|
|
|
if(defined($exit_status->{$pid})) { |
|
989
|
|
|
|
|
|
|
if($exit_status->{$pid} == 0) { |
|
990
|
|
|
|
|
|
|
$self->current_state('MASTER-COPIED'); |
|
991
|
|
|
|
|
|
|
$self->say("Remove the master key.\n"); |
|
992
|
|
|
|
|
|
|
} |
|
993
|
|
|
|
|
|
|
else { |
|
994
|
|
|
|
|
|
|
$self->say("Failed to read master key. Please try again.\n"); |
|
995
|
|
|
|
|
|
|
$self->current_state('MASTER-WAIT'); |
|
996
|
|
|
|
|
|
|
} |
|
997
|
|
|
|
|
|
|
} |
|
998
|
|
|
|
|
|
|
} |
|
999
|
|
|
|
|
|
|
|
|
1000
|
|
|
|
|
|
|
|
|
1001
|
|
|
|
|
|
|
sub copy_finished { |
|
1002
|
|
|
|
|
|
|
my($self, $exit_status) = @_; |
|
1003
|
|
|
|
|
|
|
|
|
1004
|
|
|
|
|
|
|
my $current_keys = $self->current_keys; |
|
1005
|
|
|
|
|
|
|
my %pid_to_udi = map { |
|
1006
|
|
|
|
|
|
|
$current_keys->{$_}->{pid} |
|
1007
|
|
|
|
|
|
|
? ($current_keys->{$_}->{pid} => $_) |
|
1008
|
|
|
|
|
|
|
: (); |
|
1009
|
|
|
|
|
|
|
} keys %$current_keys; |
|
1010
|
|
|
|
|
|
|
|
|
1011
|
|
|
|
|
|
|
my $done = 0; |
|
1012
|
|
|
|
|
|
|
foreach my $pid (keys %$exit_status) { |
|
1013
|
|
|
|
|
|
|
my $status = delete $exit_status->{$pid}; |
|
1014
|
|
|
|
|
|
|
my $udi = $pid_to_udi{$pid} or next; |
|
1015
|
|
|
|
|
|
|
if($status == 0) { |
|
1016
|
|
|
|
|
|
|
$self->update_key_progress($udi, 10); |
|
1017
|
|
|
|
|
|
|
} |
|
1018
|
|
|
|
|
|
|
else { |
|
1019
|
|
|
|
|
|
|
$self->update_key_progress($udi, -1); |
|
1020
|
|
|
|
|
|
|
my $key_info = $current_keys->{$udi}; |
|
1021
|
|
|
|
|
|
|
my $output = $key_info->{output}; |
|
1022
|
|
|
|
|
|
|
$output =~ s/^{\d+\/\d+}\n//mg; |
|
1023
|
|
|
|
|
|
|
$self->say("Copy to $key_info->{dev} failed:\n$output\n\n"); |
|
1024
|
|
|
|
|
|
|
} |
|
1025
|
|
|
|
|
|
|
$done++; |
|
1026
|
|
|
|
|
|
|
} |
|
1027
|
|
|
|
|
|
|
|
|
1028
|
|
|
|
|
|
|
if($done) { |
|
1029
|
|
|
|
|
|
|
foreach my $key_info (values %$current_keys) { |
|
1030
|
|
|
|
|
|
|
if($key_info->{status} >= 0 and $key_info->{status} < 10) { |
|
1031
|
|
|
|
|
|
|
$done = 0; |
|
1032
|
|
|
|
|
|
|
last; |
|
1033
|
|
|
|
|
|
|
} |
|
1034
|
|
|
|
|
|
|
} |
|
1035
|
|
|
|
|
|
|
$self->play_sound_file if $done; |
|
1036
|
|
|
|
|
|
|
} |
|
1037
|
|
|
|
|
|
|
return TRUE; |
|
1038
|
|
|
|
|
|
|
} |
|
1039
|
|
|
|
|
|
|
|
|
1040
|
|
|
|
|
|
|
|
|
1041
|
|
|
|
|
|
|
sub set_temp_root { |
|
1042
|
|
|
|
|
|
|
my($self, $new_temp) = @_; |
|
1043
|
|
|
|
|
|
|
|
|
1044
|
|
|
|
|
|
|
$self->clean_temp_dir; |
|
1045
|
|
|
|
|
|
|
|
|
1046
|
|
|
|
|
|
|
$self->temp_root($new_temp); |
|
1047
|
|
|
|
|
|
|
my $temp_dir = "$new_temp/usb-copy.$$"; |
|
1048
|
|
|
|
|
|
|
|
|
1049
|
|
|
|
|
|
|
my $path = "$temp_dir/master"; |
|
1050
|
|
|
|
|
|
|
$self->master_root($path); |
|
1051
|
|
|
|
|
|
|
mkpath($path, { mode => 0700 }) if not -d $path; |
|
1052
|
|
|
|
|
|
|
|
|
1053
|
|
|
|
|
|
|
$path = "$temp_dir/mount"; |
|
1054
|
|
|
|
|
|
|
$self->mount_dir($path); |
|
1055
|
|
|
|
|
|
|
mkpath($path, { mode => 0700 }) if not -d $path; |
|
1056
|
|
|
|
|
|
|
|
|
1057
|
|
|
|
|
|
|
return; |
|
1058
|
|
|
|
|
|
|
} |
|
1059
|
|
|
|
|
|
|
|
|
1060
|
|
|
|
|
|
|
|
|
1061
|
|
|
|
|
|
|
sub clean_temp_dir { |
|
1062
|
|
|
|
|
|
|
my $self = shift; |
|
1063
|
|
|
|
|
|
|
|
|
1064
|
|
|
|
|
|
|
my $path = $self->master_root or return; |
|
1065
|
|
|
|
|
|
|
$path =~ s{/master$}{}; |
|
1066
|
|
|
|
|
|
|
if(-d $path and $self->sudo_path and $self->current_state ne 'MASTER-WAIT') { |
|
1067
|
|
|
|
|
|
|
my $command = $self->sudo_wrap("chown -R $< $path"); |
|
1068
|
|
|
|
|
|
|
system($command); |
|
1069
|
|
|
|
|
|
|
} |
|
1070
|
|
|
|
|
|
|
rmtree($path) if -d $path; |
|
1071
|
|
|
|
|
|
|
} |
|
1072
|
|
|
|
|
|
|
|
|
1073
|
|
|
|
|
|
|
|
|
1074
|
|
|
|
|
|
|
sub run { |
|
1075
|
|
|
|
|
|
|
my $self = shift; |
|
1076
|
|
|
|
|
|
|
|
|
1077
|
|
|
|
|
|
|
# Arrange to catch exit status of child processes |
|
1078
|
|
|
|
|
|
|
my $exit_status = $self->exit_status; |
|
1079
|
|
|
|
|
|
|
$SIG{CHLD} = sub { |
|
1080
|
|
|
|
|
|
|
my $pid; |
|
1081
|
|
|
|
|
|
|
do { |
|
1082
|
|
|
|
|
|
|
$pid = waitpid(-1, WNOHANG); |
|
1083
|
|
|
|
|
|
|
$exit_status->{$pid} = $? if $pid > 0; |
|
1084
|
|
|
|
|
|
|
} while $pid > 0; |
|
1085
|
|
|
|
|
|
|
}; |
|
1086
|
|
|
|
|
|
|
Glib::Timeout->add(500, sub { $self->tick }); |
|
1087
|
|
|
|
|
|
|
|
|
1088
|
|
|
|
|
|
|
Gtk2->main; |
|
1089
|
|
|
|
|
|
|
|
|
1090
|
|
|
|
|
|
|
$self->restore_automount; |
|
1091
|
|
|
|
|
|
|
$self->clean_temp_dir; |
|
1092
|
|
|
|
|
|
|
} |
|
1093
|
|
|
|
|
|
|
|
|
1094
|
|
|
|
|
|
|
|
|
1095
|
|
|
|
|
|
|
1; |
|
1096
|
|
|
|
|
|
|
|
|
1097
|
|
|
|
|
|
|
__END__ |
|
1098
|
|
|
|
|
|
|
|
|
1099
|
|
|
|
|
|
|
=head1 ATTRIBUTES |
|
1100
|
|
|
|
|
|
|
|
|
1101
|
|
|
|
|
|
|
The application object has the following attributes (with correspondingly named |
|
1102
|
|
|
|
|
|
|
accessor methods): |
|
1103
|
|
|
|
|
|
|
|
|
1104
|
|
|
|
|
|
|
=over 4 |
|
1105
|
|
|
|
|
|
|
|
|
1106
|
|
|
|
|
|
|
=item app_win |
|
1107
|
|
|
|
|
|
|
|
|
1108
|
|
|
|
|
|
|
The main Gtk2::Window object. |
|
1109
|
|
|
|
|
|
|
|
|
1110
|
|
|
|
|
|
|
=item automount_state |
|
1111
|
|
|
|
|
|
|
|
|
1112
|
|
|
|
|
|
|
Stores the enabled state ('true' or 'false') of the GNOME/Nautilus media |
|
1113
|
|
|
|
|
|
|
automount option. The function will be disabled on startup and this value will |
|
1114
|
|
|
|
|
|
|
be restored on exit. |
|
1115
|
|
|
|
|
|
|
|
|
1116
|
|
|
|
|
|
|
=item capacity_combo |
|
1117
|
|
|
|
|
|
|
|
|
1118
|
|
|
|
|
|
|
The Gtk2::ComboBox object for the device filter 'Capacity' drop-down menu. |
|
1119
|
|
|
|
|
|
|
|
|
1120
|
|
|
|
|
|
|
=item capacity_entry |
|
1121
|
|
|
|
|
|
|
|
|
1122
|
|
|
|
|
|
|
The Gtk2::Entry object for the device filter 'Capacity' text entry box. |
|
1123
|
|
|
|
|
|
|
|
|
1124
|
|
|
|
|
|
|
=item console |
|
1125
|
|
|
|
|
|
|
|
|
1126
|
|
|
|
|
|
|
The Gtk2::TextView object used for writing output messages. |
|
1127
|
|
|
|
|
|
|
|
|
1128
|
|
|
|
|
|
|
=item current_keys |
|
1129
|
|
|
|
|
|
|
|
|
1130
|
|
|
|
|
|
|
A hash for tracking which (non-master) keys are currently inserted and what |
|
1131
|
|
|
|
|
|
|
stage each copy process is at. The hash key is the device 'UDI' and the value |
|
1132
|
|
|
|
|
|
|
is a hash of device dtails . |
|
1133
|
|
|
|
|
|
|
|
|
1134
|
|
|
|
|
|
|
=item current_state |
|
1135
|
|
|
|
|
|
|
|
|
1136
|
|
|
|
|
|
|
Used to control which mode the application is in: |
|
1137
|
|
|
|
|
|
|
|
|
1138
|
|
|
|
|
|
|
MASTER-WAIT waiting for the user to insert the master key |
|
1139
|
|
|
|
|
|
|
MASTER-COPYING waiting for the master key 'reader' script to complete |
|
1140
|
|
|
|
|
|
|
MASTER-COPIED waiting for the user to remove the master key |
|
1141
|
|
|
|
|
|
|
COPYING waiting for the user to insert blank keys |
|
1142
|
|
|
|
|
|
|
|
|
1143
|
|
|
|
|
|
|
=item exit_status |
|
1144
|
|
|
|
|
|
|
|
|
1145
|
|
|
|
|
|
|
Used by a SIGCHLD handler to track the exit status of the copy scripts. The |
|
1146
|
|
|
|
|
|
|
key is a process ID and the value is the exist status returned by C<wait>. |
|
1147
|
|
|
|
|
|
|
|
|
1148
|
|
|
|
|
|
|
=item hal |
|
1149
|
|
|
|
|
|
|
|
|
1150
|
|
|
|
|
|
|
The DBus object ('org.freedesktop.Hal.Manager') from which device add/remove |
|
1151
|
|
|
|
|
|
|
events are received. |
|
1152
|
|
|
|
|
|
|
|
|
1153
|
|
|
|
|
|
|
=item key_rack |
|
1154
|
|
|
|
|
|
|
|
|
1155
|
|
|
|
|
|
|
The Gtk2::HBox object containing the widgets representing currently inserted |
|
1156
|
|
|
|
|
|
|
keys. |
|
1157
|
|
|
|
|
|
|
|
|
1158
|
|
|
|
|
|
|
=item master_info |
|
1159
|
|
|
|
|
|
|
|
|
1160
|
|
|
|
|
|
|
A hash of device details for the 'master' USB key. |
|
1161
|
|
|
|
|
|
|
|
|
1162
|
|
|
|
|
|
|
=item master_root |
|
1163
|
|
|
|
|
|
|
|
|
1164
|
|
|
|
|
|
|
The path to the temp directory containing the copy of the master key. |
|
1165
|
|
|
|
|
|
|
|
|
1166
|
|
|
|
|
|
|
=item mount_dir |
|
1167
|
|
|
|
|
|
|
|
|
1168
|
|
|
|
|
|
|
The path to the temp directory containing temporary mount points. |
|
1169
|
|
|
|
|
|
|
|
|
1170
|
|
|
|
|
|
|
The volume label read from the master key and to be applied to the copies. |
|
1171
|
|
|
|
|
|
|
|
|
1172
|
|
|
|
|
|
|
=item options |
|
1173
|
|
|
|
|
|
|
|
|
1174
|
|
|
|
|
|
|
A hash of option name/value pairs passed in from comman-line arguments by the |
|
1175
|
|
|
|
|
|
|
wrapper script. |
|
1176
|
|
|
|
|
|
|
|
|
1177
|
|
|
|
|
|
|
=item profiles |
|
1178
|
|
|
|
|
|
|
|
|
1179
|
|
|
|
|
|
|
A hash of details of known profiles. Used to populate the profile drop-down |
|
1180
|
|
|
|
|
|
|
menu on the confirm master key dialog. |
|
1181
|
|
|
|
|
|
|
|
|
1182
|
|
|
|
|
|
|
=item selected_profile |
|
1183
|
|
|
|
|
|
|
|
|
1184
|
|
|
|
|
|
|
The name of the copying profile which will be used to select reader/writer |
|
1185
|
|
|
|
|
|
|
scripts. |
|
1186
|
|
|
|
|
|
|
|
|
1187
|
|
|
|
|
|
|
=item selected_sound |
|
1188
|
|
|
|
|
|
|
|
|
1189
|
|
|
|
|
|
|
Pathname of the currently selected sound file, to be played when copying is |
|
1190
|
|
|
|
|
|
|
complete. |
|
1191
|
|
|
|
|
|
|
|
|
1192
|
|
|
|
|
|
|
=item sudo_path |
|
1193
|
|
|
|
|
|
|
|
|
1194
|
|
|
|
|
|
|
If the script was run by a non-root user and sudo is available, this string |
|
1195
|
|
|
|
|
|
|
will be populated with the pathname of either C<gksudo> or C<sudo>. When |
|
1196
|
|
|
|
|
|
|
running the read/writer scripts the string will be prepended onto the commands. |
|
1197
|
|
|
|
|
|
|
|
|
1198
|
|
|
|
|
|
|
=item temp_root |
|
1199
|
|
|
|
|
|
|
|
|
1200
|
|
|
|
|
|
|
The temp directory selected by the user. The application will create a |
|
1201
|
|
|
|
|
|
|
subdirectory for the copy of the master key and for temporary mount points. |
|
1202
|
|
|
|
|
|
|
|
|
1203
|
|
|
|
|
|
|
=item vendor_combo |
|
1204
|
|
|
|
|
|
|
|
|
1205
|
|
|
|
|
|
|
The Gtk2::ComboBox object for the device filter 'Vendor' drop-down menu. |
|
1206
|
|
|
|
|
|
|
|
|
1207
|
|
|
|
|
|
|
=item vendor_entry |
|
1208
|
|
|
|
|
|
|
|
|
1209
|
|
|
|
|
|
|
The Gtk2::Entry object for the device filter 'Vendor' text entry box. |
|
1210
|
|
|
|
|
|
|
|
|
1211
|
|
|
|
|
|
|
=item volume_label |
|
1212
|
|
|
|
|
|
|
|
|
1213
|
|
|
|
|
|
|
The volume label which will be passed to the writer script. |
|
1214
|
|
|
|
|
|
|
|
|
1215
|
|
|
|
|
|
|
=back |
|
1216
|
|
|
|
|
|
|
|
|
1217
|
|
|
|
|
|
|
=head1 PROFILES |
|
1218
|
|
|
|
|
|
|
|
|
1219
|
|
|
|
|
|
|
The tasks of reading a master key and writing to a blank key are delegated to |
|
1220
|
|
|
|
|
|
|
'reader' and 'writer' scripts. A pair of reader/writer scripts is supplied but |
|
1221
|
|
|
|
|
|
|
the application also supports using different scripts as dictated by a user |
|
1222
|
|
|
|
|
|
|
selection. The supplied scripts assume file-by-file copying and format the |
|
1223
|
|
|
|
|
|
|
blank keys with a VFAT filesystem. An alternate script might for example, use |
|
1224
|
|
|
|
|
|
|
C<dd> to write a complete filesystem image in a single operation. |
|
1225
|
|
|
|
|
|
|
|
|
1226
|
|
|
|
|
|
|
A pair of scripts is referred to as a copying 'profile'. The user can select a |
|
1227
|
|
|
|
|
|
|
profile via a command-line option or from a drop-down list when confirming the |
|
1228
|
|
|
|
|
|
|
master key. |
|
1229
|
|
|
|
|
|
|
|
|
1230
|
|
|
|
|
|
|
The supplied scripts are called: |
|
1231
|
|
|
|
|
|
|
|
|
1232
|
|
|
|
|
|
|
copyfiles-reader.sh |
|
1233
|
|
|
|
|
|
|
copyfiles-writer.sh |
|
1234
|
|
|
|
|
|
|
|
|
1235
|
|
|
|
|
|
|
A profile does not need to include a reader script. If a profile which only |
|
1236
|
|
|
|
|
|
|
includes a writer script is selected (via the command-line options) then the |
|
1237
|
|
|
|
|
|
|
application will go immediately into the mode of waiting for blank keys. |
|
1238
|
|
|
|
|
|
|
|
|
1239
|
|
|
|
|
|
|
=head2 Profile Script API |
|
1240
|
|
|
|
|
|
|
|
|
1241
|
|
|
|
|
|
|
The filename of the reader script must end with C<-reader> (followed by an |
|
1242
|
|
|
|
|
|
|
optional extension) and similarly, the filename of the writer script must end |
|
1243
|
|
|
|
|
|
|
with C<-writer>. |
|
1244
|
|
|
|
|
|
|
|
|
1245
|
|
|
|
|
|
|
The reader/writer scripts do not have to be shell scripts - they merely need to |
|
1246
|
|
|
|
|
|
|
be executable. The application ignores the file extension if it is present. |
|
1247
|
|
|
|
|
|
|
|
|
1248
|
|
|
|
|
|
|
Both reader and writer scripts are assumed to have succeeded if they have an |
|
1249
|
|
|
|
|
|
|
exit status of 0. A non-zero exit status will be considered a failure. |
|
1250
|
|
|
|
|
|
|
|
|
1251
|
|
|
|
|
|
|
When the master key reader script is invoked, the following environment |
|
1252
|
|
|
|
|
|
|
variables will be set: |
|
1253
|
|
|
|
|
|
|
|
|
1254
|
|
|
|
|
|
|
USB_BLOCK_DEVICE e.g.: /dev/sdb |
|
1255
|
|
|
|
|
|
|
USB_MOUNT_DIR e.g.: /tmp/usb-copy.nnnnn/mount/sdb |
|
1256
|
|
|
|
|
|
|
USB_MASTER_ROOT e.g.: /tmp/usb-copy.nnnnn/master |
|
1257
|
|
|
|
|
|
|
|
|
1258
|
|
|
|
|
|
|
The writer script will be passed the same set of variables and one extra: |
|
1259
|
|
|
|
|
|
|
|
|
1260
|
|
|
|
|
|
|
USB_VOLUME_NAME e.g.: FREE-STUFF |
|
1261
|
|
|
|
|
|
|
|
|
1262
|
|
|
|
|
|
|
Be warned that this variable may be empty - depending on what was returned from |
|
1263
|
|
|
|
|
|
|
running C<dosfslabel> against the master key. It is entirely reasonable for a |
|
1264
|
|
|
|
|
|
|
custom writer script to ignore this variable altogether and either use a |
|
1265
|
|
|
|
|
|
|
hardcoded volume label or not use one at all. |
|
1266
|
|
|
|
|
|
|
|
|
1267
|
|
|
|
|
|
|
The writer script can also indicate progress (for updating the progress bar in |
|
1268
|
|
|
|
|
|
|
the icon) by writing lines to STDOUT in the following format: |
|
1269
|
|
|
|
|
|
|
|
|
1270
|
|
|
|
|
|
|
{x/y} |
|
1271
|
|
|
|
|
|
|
|
|
1272
|
|
|
|
|
|
|
Where '{' is the first character on a line; 'x' is an integer indicating the |
|
1273
|
|
|
|
|
|
|
number of steps completed; and 'y' is an integer indicating the total number |
|
1274
|
|
|
|
|
|
|
of steps. For example if the script output this line: |
|
1275
|
|
|
|
|
|
|
|
|
1276
|
|
|
|
|
|
|
{4/8} |
|
1277
|
|
|
|
|
|
|
|
|
1278
|
|
|
|
|
|
|
the status icon would be updated to indicate 50% complete. |
|
1279
|
|
|
|
|
|
|
|
|
1280
|
|
|
|
|
|
|
=head1 METHODS |
|
1281
|
|
|
|
|
|
|
|
|
1282
|
|
|
|
|
|
|
=head2 Constructor |
|
1283
|
|
|
|
|
|
|
|
|
1284
|
|
|
|
|
|
|
The C<new> method is used to create an application object. It in turn calls |
|
1285
|
|
|
|
|
|
|
C<BUILD> to create and populate the application window and hook into HAL (the |
|
1286
|
|
|
|
|
|
|
Hardware Abstraction Layer) via DBus to get notifications of devices been |
|
1287
|
|
|
|
|
|
|
added/removed. |
|
1288
|
|
|
|
|
|
|
|
|
1289
|
|
|
|
|
|
|
=head2 add_key_to_rack ( key_info ) |
|
1290
|
|
|
|
|
|
|
|
|
1291
|
|
|
|
|
|
|
Called from C<hal_device_added> if the newly added device matches the current |
|
1292
|
|
|
|
|
|
|
device filter settings. The C<key_info> parameter supplied is a hashref of |
|
1293
|
|
|
|
|
|
|
device properties as returned by C<hal_device_properties>. A GUI widget |
|
1294
|
|
|
|
|
|
|
representing the new USB key is added to the user interface and a data |
|
1295
|
|
|
|
|
|
|
structure to track the copying process is created. |
|
1296
|
|
|
|
|
|
|
|
|
1297
|
|
|
|
|
|
|
=head2 build_console ( ) |
|
1298
|
|
|
|
|
|
|
|
|
1299
|
|
|
|
|
|
|
Called from C<build_ui> to create the scrolled text window for displaying |
|
1300
|
|
|
|
|
|
|
progress messages. |
|
1301
|
|
|
|
|
|
|
|
|
1302
|
|
|
|
|
|
|
=head2 build_filters ( ) |
|
1303
|
|
|
|
|
|
|
|
|
1304
|
|
|
|
|
|
|
Called from C<build_ui> to create the toolbar of drop-down menus and text |
|
1305
|
|
|
|
|
|
|
entries for the device filter settings. |
|
1306
|
|
|
|
|
|
|
|
|
1307
|
|
|
|
|
|
|
=head2 build_key_rack ( ) |
|
1308
|
|
|
|
|
|
|
|
|
1309
|
|
|
|
|
|
|
Called from C<build_ui> to create the container widget to house the |
|
1310
|
|
|
|
|
|
|
per-key status indicators. |
|
1311
|
|
|
|
|
|
|
|
|
1312
|
|
|
|
|
|
|
=head2 build_menu ( ) |
|
1313
|
|
|
|
|
|
|
|
|
1314
|
|
|
|
|
|
|
Called from C<build_ui> to create the application menu and hook the menu |
|
1315
|
|
|
|
|
|
|
items up to handler methods. |
|
1316
|
|
|
|
|
|
|
|
|
1317
|
|
|
|
|
|
|
=head2 build_ui ( ) |
|
1318
|
|
|
|
|
|
|
|
|
1319
|
|
|
|
|
|
|
Called from the constructor to create the main application window and populate |
|
1320
|
|
|
|
|
|
|
it with Gtk widgets. |
|
1321
|
|
|
|
|
|
|
|
|
1322
|
|
|
|
|
|
|
=head2 check_for_root_user ( ) |
|
1323
|
|
|
|
|
|
|
|
|
1324
|
|
|
|
|
|
|
Called on startup to check that either the script is running as root or that sudo |
|
1325
|
|
|
|
|
|
|
is available. In the latter case, sudo (or gksudo) will be used to invoke the |
|
1326
|
|
|
|
|
|
|
read/writer scripts. |
|
1327
|
|
|
|
|
|
|
|
|
1328
|
|
|
|
|
|
|
If the script is not running with root permissions; and sudo is not available; |
|
1329
|
|
|
|
|
|
|
and the C<--no-root-check> option was not specified, this method will die with |
|
1330
|
|
|
|
|
|
|
an appropriate error message. |
|
1331
|
|
|
|
|
|
|
|
|
1332
|
|
|
|
|
|
|
=head2 clean_temp_dir ( ) |
|
1333
|
|
|
|
|
|
|
|
|
1334
|
|
|
|
|
|
|
Called from the C<run> method immediately before the application exits. This |
|
1335
|
|
|
|
|
|
|
method is responsible for removing the temporary directories containing the |
|
1336
|
|
|
|
|
|
|
master copy of the files and the mount points for the blank keys. |
|
1337
|
|
|
|
|
|
|
|
|
1338
|
|
|
|
|
|
|
When running as a non-root user, this method needs to use sudo in order to |
|
1339
|
|
|
|
|
|
|
remove the files created by the reader script when it was running as root. |
|
1340
|
|
|
|
|
|
|
|
|
1341
|
|
|
|
|
|
|
=head2 commandline_options ( ) |
|
1342
|
|
|
|
|
|
|
|
|
1343
|
|
|
|
|
|
|
This B<class> method returns a list of recognised options in the form expected |
|
1344
|
|
|
|
|
|
|
by L<Getopt::Long>. |
|
1345
|
|
|
|
|
|
|
|
|
1346
|
|
|
|
|
|
|
=head2 confirm_master_dialog ( key_info ) |
|
1347
|
|
|
|
|
|
|
|
|
1348
|
|
|
|
|
|
|
This method is called each time a USB key is inserted when the application is |
|
1349
|
|
|
|
|
|
|
in the C<MASTER-WAIT> state. The C<key_info> parameter supplied is a hashref |
|
1350
|
|
|
|
|
|
|
of device properties as returned by C<hal_device_properties>. this method |
|
1351
|
|
|
|
|
|
|
displays a dialog box to allow the user to confirm that the device should be |
|
1352
|
|
|
|
|
|
|
used as the master key. |
|
1353
|
|
|
|
|
|
|
|
|
1354
|
|
|
|
|
|
|
If the user selects 'Cancel', no further action is taken and the application |
|
1355
|
|
|
|
|
|
|
goes back to waiting for a master key to be inserted. |
|
1356
|
|
|
|
|
|
|
|
|
1357
|
|
|
|
|
|
|
If the user confirms the device should be used as the master, then control is |
|
1358
|
|
|
|
|
|
|
passed to the C<start_master_read> method. |
|
1359
|
|
|
|
|
|
|
|
|
1360
|
|
|
|
|
|
|
=head2 copy_finished ( exit_status ) |
|
1361
|
|
|
|
|
|
|
|
|
1362
|
|
|
|
|
|
|
Called when a 'writer' process exits. Checks the exit status and updates the |
|
1363
|
|
|
|
|
|
|
icon in the key rack (0 = success, non-zero = failure). |
|
1364
|
|
|
|
|
|
|
|
|
1365
|
|
|
|
|
|
|
=head2 disable_automount ( ) |
|
1366
|
|
|
|
|
|
|
|
|
1367
|
|
|
|
|
|
|
This method is called at startup to query GConf for the current GNOME/Nautilus |
|
1368
|
|
|
|
|
|
|
media automount status ('true'/'false' for enabled/disabled). The current |
|
1369
|
|
|
|
|
|
|
state is saved and then the value is set to false. The operation should fail |
|
1370
|
|
|
|
|
|
|
silently in non-GNOME environments. |
|
1371
|
|
|
|
|
|
|
|
|
1372
|
|
|
|
|
|
|
=head2 disable_filter_inputs ( ) |
|
1373
|
|
|
|
|
|
|
|
|
1374
|
|
|
|
|
|
|
This method is called from C<require_master_key> to disable the menu and text |
|
1375
|
|
|
|
|
|
|
entry widgets on the device filter toolbar. |
|
1376
|
|
|
|
|
|
|
|
|
1377
|
|
|
|
|
|
|
=head2 enable_filter_inputs ( ) |
|
1378
|
|
|
|
|
|
|
|
|
1379
|
|
|
|
|
|
|
This method is called from C<require_master_key> to enable the menu and text |
|
1380
|
|
|
|
|
|
|
entry widgets on the device filter toolbar. |
|
1381
|
|
|
|
|
|
|
|
|
1382
|
|
|
|
|
|
|
=head2 find_command ( command ) |
|
1383
|
|
|
|
|
|
|
|
|
1384
|
|
|
|
|
|
|
Takes a command name and returns the path to the first matching executable file |
|
1385
|
|
|
|
|
|
|
found in a directory listed in the $PATH environment variable. Returns |
|
1386
|
|
|
|
|
|
|
C<undef> if no match found. |
|
1387
|
|
|
|
|
|
|
|
|
1388
|
|
|
|
|
|
|
=head2 fork_copier ( key_info ) |
|
1389
|
|
|
|
|
|
|
|
|
1390
|
|
|
|
|
|
|
Called from C<add_key_to_rack>. Forks a 'writer' process and collects its |
|
1391
|
|
|
|
|
|
|
STDOUT+STDERR via a pipe. |
|
1392
|
|
|
|
|
|
|
|
|
1393
|
|
|
|
|
|
|
=head2 get_volume_label ( device ) |
|
1394
|
|
|
|
|
|
|
|
|
1395
|
|
|
|
|
|
|
Called from C<confirm_master_dialog> when collecting information about the key |
|
1396
|
|
|
|
|
|
|
which was just inserted. Current implementation simply runs the C<dosfslabel> |
|
1397
|
|
|
|
|
|
|
command. |
|
1398
|
|
|
|
|
|
|
|
|
1399
|
|
|
|
|
|
|
=head2 hal_device_added ( udi ) |
|
1400
|
|
|
|
|
|
|
|
|
1401
|
|
|
|
|
|
|
Called to handle a 'DeviceAdded' event from HAL via DBus. Delegates to |
|
1402
|
|
|
|
|
|
|
C<start_master_read> if the app is waiting for a master key. Otherwise checks |
|
1403
|
|
|
|
|
|
|
whether the new device parameters match the current filter settings and |
|
1404
|
|
|
|
|
|
|
delegates to C<add_key_to_rack> if they do. |
|
1405
|
|
|
|
|
|
|
|
|
1406
|
|
|
|
|
|
|
=head2 hal_device_properties ( udi ) |
|
1407
|
|
|
|
|
|
|
|
|
1408
|
|
|
|
|
|
|
Called from C<hal_device_added> to query HAL. Returns a hash(ref) of device |
|
1409
|
|
|
|
|
|
|
details. The global variable C<%hal_device_added> defines which attributes |
|
1410
|
|
|
|
|
|
|
returned from HAL will appear in the hash and which keys they will be mapped |
|
1411
|
|
|
|
|
|
|
to. |
|
1412
|
|
|
|
|
|
|
|
|
1413
|
|
|
|
|
|
|
=head2 hal_device_removed ( udi ) |
|
1414
|
|
|
|
|
|
|
|
|
1415
|
|
|
|
|
|
|
Called to handle a 'DeviceRemoved' event from HAL via DBus. Delegates to |
|
1416
|
|
|
|
|
|
|
C<remove_key_from_rack> if the application is in the C<COPYING> state. |
|
1417
|
|
|
|
|
|
|
|
|
1418
|
|
|
|
|
|
|
=head2 init_dbus_watcher ( ) |
|
1419
|
|
|
|
|
|
|
|
|
1420
|
|
|
|
|
|
|
Called from the constructor to hook up device-add events to the |
|
1421
|
|
|
|
|
|
|
C<hal_device_added> method and device-remove events to C<hal_device_removed>. |
|
1422
|
|
|
|
|
|
|
|
|
1423
|
|
|
|
|
|
|
=head2 master_copy_finished ( exit_status ) |
|
1424
|
|
|
|
|
|
|
|
|
1425
|
|
|
|
|
|
|
Called when the 'reader' process exits. Checks the exit status and updates the |
|
1426
|
|
|
|
|
|
|
application state to <MASTER-COPIED> on success or C<MASTER-WAIT> on failure. |
|
1427
|
|
|
|
|
|
|
|
|
1428
|
|
|
|
|
|
|
=head2 match_device_filter ( key_info ) |
|
1429
|
|
|
|
|
|
|
|
|
1430
|
|
|
|
|
|
|
Called from C<hal_device_added> and returns true if the device matches the |
|
1431
|
|
|
|
|
|
|
current filter parameters, or false otherwise. |
|
1432
|
|
|
|
|
|
|
|
|
1433
|
|
|
|
|
|
|
=head2 on_copier_pipe_read ( fileno, condition, udi ) |
|
1434
|
|
|
|
|
|
|
|
|
1435
|
|
|
|
|
|
|
Handler for data received from a 'writer' process. Updates the status icon for |
|
1436
|
|
|
|
|
|
|
the device to indicate progress. |
|
1437
|
|
|
|
|
|
|
|
|
1438
|
|
|
|
|
|
|
=head2 on_master_pipe_read ( fileno, condition, udi ) |
|
1439
|
|
|
|
|
|
|
|
|
1440
|
|
|
|
|
|
|
Handler for data received from the master key 'reader' process. Copies output |
|
1441
|
|
|
|
|
|
|
from the process to the console widget. |
|
1442
|
|
|
|
|
|
|
|
|
1443
|
|
|
|
|
|
|
=head2 on_menu_edit_preferences ( ) |
|
1444
|
|
|
|
|
|
|
|
|
1445
|
|
|
|
|
|
|
Handler for the Edit E<gt> Preferences menu item - not currently implemented. |
|
1446
|
|
|
|
|
|
|
|
|
1447
|
|
|
|
|
|
|
=head2 on_menu_file_new ( ) |
|
1448
|
|
|
|
|
|
|
|
|
1449
|
|
|
|
|
|
|
Handler for the File E<gt> New menu item. Resets the application state via |
|
1450
|
|
|
|
|
|
|
C<require_master_key>. |
|
1451
|
|
|
|
|
|
|
|
|
1452
|
|
|
|
|
|
|
=head2 on_menu_file_quit ( ) |
|
1453
|
|
|
|
|
|
|
|
|
1454
|
|
|
|
|
|
|
Handler for the File E<gt> Quit menu item. Exits the Gtk event loop, which |
|
1455
|
|
|
|
|
|
|
returns control to the C<run> method. |
|
1456
|
|
|
|
|
|
|
|
|
1457
|
|
|
|
|
|
|
=head2 on_menu_help_about ( ) |
|
1458
|
|
|
|
|
|
|
|
|
1459
|
|
|
|
|
|
|
Handler for the Help E<gt> About menu item. Displays 'About' dialog. |
|
1460
|
|
|
|
|
|
|
|
|
1461
|
|
|
|
|
|
|
=head2 play_sound_file ( sound_file ) |
|
1462
|
|
|
|
|
|
|
|
|
1463
|
|
|
|
|
|
|
This method takes a pathname to a sound file (e.g.: a .wav) and plays it. |
|
1464
|
|
|
|
|
|
|
The current implementation simply runs the the SOX C<play> command - it should probably use GStreamer |
|
1465
|
|
|
|
|
|
|
|
|
1466
|
|
|
|
|
|
|
=head2 reader_script ( ) |
|
1467
|
|
|
|
|
|
|
|
|
1468
|
|
|
|
|
|
|
Returns the path to the script from the currently selected profile, which will |
|
1469
|
|
|
|
|
|
|
be used to read the master key. Will return undef if the selected profile does |
|
1470
|
|
|
|
|
|
|
not include a reader script. |
|
1471
|
|
|
|
|
|
|
|
|
1472
|
|
|
|
|
|
|
=head2 ready_to_write ( ) |
|
1473
|
|
|
|
|
|
|
|
|
1474
|
|
|
|
|
|
|
This method is called after the master key has been read (or immediately on |
|
1475
|
|
|
|
|
|
|
startup if the selected profile does not use a reader script) and puts the |
|
1476
|
|
|
|
|
|
|
application into the mode of waiting for blank keys to be inserted. |
|
1477
|
|
|
|
|
|
|
|
|
1478
|
|
|
|
|
|
|
=head2 remove_key_from_rack ( udi ) |
|
1479
|
|
|
|
|
|
|
|
|
1480
|
|
|
|
|
|
|
Called from C<hal_device_removed> to remove the indicator widget corresponding |
|
1481
|
|
|
|
|
|
|
to the USB key which has just been removed. |
|
1482
|
|
|
|
|
|
|
|
|
1483
|
|
|
|
|
|
|
=head2 require_master_key ( ) |
|
1484
|
|
|
|
|
|
|
|
|
1485
|
|
|
|
|
|
|
Called from the constructor to put the app in the C<MASTER-WAIT> mode (waiting |
|
1486
|
|
|
|
|
|
|
for the master key to be inserted). Can also be called from the |
|
1487
|
|
|
|
|
|
|
C<on_menu_file_new> menu event handler. |
|
1488
|
|
|
|
|
|
|
|
|
1489
|
|
|
|
|
|
|
=head2 restore_automount ( ) |
|
1490
|
|
|
|
|
|
|
|
|
1491
|
|
|
|
|
|
|
This method is called at exit time restore the original GConf setting for the |
|
1492
|
|
|
|
|
|
|
GNOME/Nautilus media automount function. |
|
1493
|
|
|
|
|
|
|
|
|
1494
|
|
|
|
|
|
|
=head2 run ( ) |
|
1495
|
|
|
|
|
|
|
|
|
1496
|
|
|
|
|
|
|
This method is called from the wrapper script. It's job is to run the Gtk |
|
1497
|
|
|
|
|
|
|
event loop and when that exits, to call C<clean_temp_dir> and then return. |
|
1498
|
|
|
|
|
|
|
|
|
1499
|
|
|
|
|
|
|
=head2 say ( message ) |
|
1500
|
|
|
|
|
|
|
|
|
1501
|
|
|
|
|
|
|
Appends a message to the console widget. (Note, the caller is responsible |
|
1502
|
|
|
|
|
|
|
for supplying the newline characters). |
|
1503
|
|
|
|
|
|
|
|
|
1504
|
|
|
|
|
|
|
=head2 scan_for_profiles ( ) |
|
1505
|
|
|
|
|
|
|
|
|
1506
|
|
|
|
|
|
|
Populates the hash of profile data in the C<profiles> attribute. |
|
1507
|
|
|
|
|
|
|
|
|
1508
|
|
|
|
|
|
|
=head2 select_profile ( profile_name ) |
|
1509
|
|
|
|
|
|
|
|
|
1510
|
|
|
|
|
|
|
This method is used to select which reader/writer scripts will be used. At |
|
1511
|
|
|
|
|
|
|
present there is one hard-coded call to this method in the constructor. |
|
1512
|
|
|
|
|
|
|
Ideally, the user would select from all available profile scripts in the |
|
1513
|
|
|
|
|
|
|
'confirm master' dialog. |
|
1514
|
|
|
|
|
|
|
|
|
1515
|
|
|
|
|
|
|
=head2 set_temp_root ( pathname ) |
|
1516
|
|
|
|
|
|
|
|
|
1517
|
|
|
|
|
|
|
Called from C<confirm_master_dialog> based on the temp directory selected by |
|
1518
|
|
|
|
|
|
|
the user. |
|
1519
|
|
|
|
|
|
|
|
|
1520
|
|
|
|
|
|
|
=head2 start_master_read ( key_info ) |
|
1521
|
|
|
|
|
|
|
|
|
1522
|
|
|
|
|
|
|
Called from C<hal_device_added> to fork off a 'reader' process to slurp in the |
|
1523
|
|
|
|
|
|
|
contents of the master key. |
|
1524
|
|
|
|
|
|
|
|
|
1525
|
|
|
|
|
|
|
=head2 sudo_wrap ( command env-var-names ) |
|
1526
|
|
|
|
|
|
|
|
|
1527
|
|
|
|
|
|
|
If the script is run by a non-root user and sudo is available and the |
|
1528
|
|
|
|
|
|
|
C<--no-root-check> option was not specified, this method will return a command |
|
1529
|
|
|
|
|
|
|
string which wraps the supplied command in a call to either C<gksudo> or |
|
1530
|
|
|
|
|
|
|
C<sudo>. For all other cases, C<command> is returned unmodified. |
|
1531
|
|
|
|
|
|
|
|
|
1532
|
|
|
|
|
|
|
The C<gksudo> command is preferred since it gives the user a GUI prompt window |
|
1533
|
|
|
|
|
|
|
if it is necessary to prompt for a password. This method handles the different |
|
1534
|
|
|
|
|
|
|
semantics required to pass environment variables through C<gksudo> and C<sudo>. |
|
1535
|
|
|
|
|
|
|
|
|
1536
|
|
|
|
|
|
|
=head2 tick ( ) |
|
1537
|
|
|
|
|
|
|
|
|
1538
|
|
|
|
|
|
|
This timer event handler is used to take the child process exit status values |
|
1539
|
|
|
|
|
|
|
collected by the SIGCHLD handler and pass them to C<master_copy_finished> or |
|
1540
|
|
|
|
|
|
|
C<copy_finished> as appropriate. |
|
1541
|
|
|
|
|
|
|
|
|
1542
|
|
|
|
|
|
|
=head2 update_key_progress ( udi, status ) |
|
1543
|
|
|
|
|
|
|
|
|
1544
|
|
|
|
|
|
|
Called from C<on_copier_pipe_read> to update the status icon for a specified |
|
1545
|
|
|
|
|
|
|
USB key device. The progress parameter is a number in the range 0-10 for |
|
1546
|
|
|
|
|
|
|
copies in progress; -1 for a copy that has failed (non-zero exit status from |
|
1547
|
|
|
|
|
|
|
the 'writer' process); or -2 to indicate a device which did not match the |
|
1548
|
|
|
|
|
|
|
filter settings and is being ignored. |
|
1549
|
|
|
|
|
|
|
|
|
1550
|
|
|
|
|
|
|
=head2 writer_script ( ) |
|
1551
|
|
|
|
|
|
|
|
|
1552
|
|
|
|
|
|
|
Returns the path to the script from the currently selected profile, which will |
|
1553
|
|
|
|
|
|
|
be used to write to the blank keys. |
|
1554
|
|
|
|
|
|
|
|
|
1555
|
|
|
|
|
|
|
=cut |
|
1556
|
|
|
|
|
|
|
|
|
1557
|
|
|
|
|
|
|
=head1 AUTHOR |
|
1558
|
|
|
|
|
|
|
|
|
1559
|
|
|
|
|
|
|
Grant McLean, C<< <grantm at cpan.org> >> |
|
1560
|
|
|
|
|
|
|
|
|
1561
|
|
|
|
|
|
|
=head1 BUGS |
|
1562
|
|
|
|
|
|
|
|
|
1563
|
|
|
|
|
|
|
Please report any bugs or feature requests to C<bug-app-usbkeycopycon at rt.cpan.org>, or through |
|
1564
|
|
|
|
|
|
|
the web interface at L<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=App-USBKeyCopyCon>. I will be notified, and then you'll |
|
1565
|
|
|
|
|
|
|
automatically be notified of progress on your bug as I make changes. |
|
1566
|
|
|
|
|
|
|
|
|
1567
|
|
|
|
|
|
|
|
|
1568
|
|
|
|
|
|
|
=head1 SUPPORT |
|
1569
|
|
|
|
|
|
|
|
|
1570
|
|
|
|
|
|
|
You can find documentation for this module with the perldoc command. |
|
1571
|
|
|
|
|
|
|
|
|
1572
|
|
|
|
|
|
|
perldoc App::USBKeyCopyCon |
|
1573
|
|
|
|
|
|
|
|
|
1574
|
|
|
|
|
|
|
|
|
1575
|
|
|
|
|
|
|
You can also look for information at: |
|
1576
|
|
|
|
|
|
|
|
|
1577
|
|
|
|
|
|
|
=over 4 |
|
1578
|
|
|
|
|
|
|
|
|
1579
|
|
|
|
|
|
|
=item * github: source code repository |
|
1580
|
|
|
|
|
|
|
|
|
1581
|
|
|
|
|
|
|
L<http://github.com/grantm/usb-key-copy-con> |
|
1582
|
|
|
|
|
|
|
|
|
1583
|
|
|
|
|
|
|
=item * RT: CPAN's request tracker |
|
1584
|
|
|
|
|
|
|
|
|
1585
|
|
|
|
|
|
|
L<http://rt.cpan.org/NoAuth/Bugs.html?Dist=App-USBKeyCopyCon> |
|
1586
|
|
|
|
|
|
|
|
|
1587
|
|
|
|
|
|
|
=item * AnnoCPAN: Annotated CPAN documentation |
|
1588
|
|
|
|
|
|
|
|
|
1589
|
|
|
|
|
|
|
L<http://annocpan.org/dist/App-USBKeyCopyCon> |
|
1590
|
|
|
|
|
|
|
|
|
1591
|
|
|
|
|
|
|
=item * CPAN Ratings |
|
1592
|
|
|
|
|
|
|
|
|
1593
|
|
|
|
|
|
|
L<http://cpanratings.perl.org/d/App-USBKeyCopyCon> |
|
1594
|
|
|
|
|
|
|
|
|
1595
|
|
|
|
|
|
|
=item * Search CPAN |
|
1596
|
|
|
|
|
|
|
|
|
1597
|
|
|
|
|
|
|
L<http://search.cpan.org/dist/App-USBKeyCopyCon> |
|
1598
|
|
|
|
|
|
|
|
|
1599
|
|
|
|
|
|
|
=back |
|
1600
|
|
|
|
|
|
|
|
|
1601
|
|
|
|
|
|
|
|
|
1602
|
|
|
|
|
|
|
=head1 COPYRIGHT & LICENSE |
|
1603
|
|
|
|
|
|
|
|
|
1604
|
|
|
|
|
|
|
Copyright 2009 Grant McLean, all rights reserved. |
|
1605
|
|
|
|
|
|
|
|
|
1606
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify it |
|
1607
|
|
|
|
|
|
|
under the same terms as Perl itself. |
|
1608
|
|
|
|
|
|
|
|
|
1609
|
|
|
|
|
|
|
|
|
1610
|
|
|
|
|
|
|
=cut |
|
1611
|
|
|
|
|
|
|
|
|
1612
|
|
|
|
|
|
|
|