-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathutils.pm
247 lines (216 loc) · 6.86 KB
/
utils.pm
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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
package Utils;
use strict;
use warnings;
use List::Util qw( any first);
use Log::Log4perl;
use JSON::PP;
use Net::IDN::Encode qw(:all);
use IO::Socket::SSL::PublicSuffix;
use URI::Escape;
sub new {
my ($class, @arguments) = @_;
my $self = {@arguments};
bless $self, $class;
return $self;
}
# initialize your log4perl object
sub initialize_logger {
my $self = shift;
# This will read form your config. Put a proper path to your log4perl config
Log::Log4perl->init_and_watch($ENV{'SCRIPT_DIR'} . '/log4perl.conf', 10);
my $logger = Log::Log4perl->get_logger;
$Log::Log4perl::DateFormat::GMTIME = 1;
$self->{'logger'} = $logger;
return $self;
}
# Left and right trim the whitspaces
sub trim {
my ($self, $str) = @_;
$str =~ s/^\s+|\s+$//g;
return $str;
}
# Get the index of the value if exist in array
sub get_matching_index {
my ($self, $array, $value) = @_;
my ($index) = grep { $array->[$_] eq $value } (0 .. @$array - 1);
if (defined $index) {
return $index;
}
else {
return -1;
}
}
# Delete the value at a particular index in array
sub delete_item {
my ($self, $array, $index) = @_;
if ($index >= 0) {
# this will remove and make it undef, which we don't want
# return delete $array->[$index];
return splice @$array, $index, 1;
}
else {
# Wrong index value provided for deletion
return 0;
}
}
# Push to array if not exist in array
sub push_to_array {
my ($self, $array, $value) = @_;
if ($self->is_exist($array, $value)) {
return 1;
}
else {
push(@$array, $value);
}
}
# Check whether a value exist in array or not
sub is_exist {
my ($self, $array, $value) = @_;
# grep solution loops through the entire list even if the first element of long list matches.
# 'any' will short-circuit and quit the moment the first match is found, thus it is more efficient
# Also its discourage to use smartmatch
if (any { $_ eq $value } @$array) {
return 1;
}
else {
return 0;
}
}
sub is_file_exist {
my ($self, $filename) = @_;
if (-e $filename) {
return 1;
}
else {
return 0;
}
}
sub is_encoded {
my ($self, $uri) = @_;
$uri = $uri // '';
# There is no foolproof or reliable way to check whether url is encoded or not.
# You'll never know for sure if a string is URL-encoded or if it was supposed to have the sequence %2B in it.
# This logic will work if unescaped string will not contain '%'
if ($uri =~ /(%[\dA-F]{2})+/i) {
return 1;
}
else {
return 0;
}
}
# Punycode is a way to represent International Domain Names (IDNs) with the limited character set (A-Z, 0-9) supported by the domain name system.
# e.g. "münich" would be encoded as "mnich-kva".
sub is_punycode {
my ($self, $uri) = @_;
$uri = $uri // '';
# To prevent non-international domain names containing hyphens from being accidentally interpreted as Punycode,
# international domain name Punycode sequences have a so-called ASCII Compatible Encoding (ACE) prefix, "xn--", prepended.
# An IDN takes the punycode encoding, and adds a "xn--" in front of it.
# So "münich.com" would become "xn--mnich-kva.com".
if ($uri =~ /xn--.+/i) {
return 1;
}
else {
return 0;
}
}
# Get domain name from the given url
sub get_domainname {
my ($self, $hostname) = @_;
my $ps = IO::Socket::SSL::PublicSuffix->default;
my $root_domain = $ps->public_suffix($hostname, 1);
return $root_domain;
}
# Get a value for a like matching key
sub get_value_for_key_like {
my ($self, $hash_ref, $key_to_match) = @_;
my $value = $hash_ref->{(first {m/$key_to_match/} keys %$hash_ref) || ''};
return $value;
}
# Get your local IP
sub get_local_ip {
my ($self) = @_;
$self->{'logger'}->info("Getting the local IP address...");
my $address = eval { Net::Address::IP::Local->public };
if ($@) {
$self->{'logger'}->error("Could not determine IP address : $@");
return 0;
}
else {
$self->{'logger'}->info("IP : $address");
return $address;
}
}
# Get the IP visible to outside world
sub get_public_ip {
my ($self, $ua) = @_;
$self->{'logger'}->info("Getting the public IP address...");
my $url = 'http://whatismyip.akamai.com/';
my $address = eval { $ua->get($url)->res->body };
if ($@) {
$self->{'logger'}->error("Could not determine public IP address : $@");
return 0;
}
else {
$self->{'logger'}->info("IP : $address");
return $address;
}
}
# Read a JSON file
sub read_json_file {
my ($self, $json_file) = @_;
$self->{'logger'}->info("Reading $json_file");
my $json_text = $self->slurp_file($json_file);
my $config_data = decode_json($json_text);
return ($config_data);
}
# Write to a JSON file (pretty print)
sub write_json_file {
my ($self, $output_file, $current_summary) = @_;
$self->{'logger'}->info("Writing output to file: $output_file");
my $out_json = JSON::PP->new->ascii->pretty->allow_nonref;
my $pretty_p_out_json = $out_json->encode($current_summary);
open my $out_json_fh, ">", $output_file
or $self->{'logger'}->error("Can't open $output_file for writing: $!\n");
print $out_json_fh $pretty_p_out_json;
close $out_json_fh or $self->{'logger'}->warn("Unable to close file : $!");
$self->{'logger'}->debug($pretty_p_out_json);
return 1;
}
# Convert file in dos format to unix (needed when you copied file form windows to linux machine)
sub dos2unix {
my ($self, @filename_list) = @_;
foreach my $filename (@filename_list) {
my $exit_code = system('perl', '-p', '-i', '-e' => 's/\r\n/\n/', "$filename");
if ($exit_code != 0) {
$self->{'logger'}->error(
"Unable to convert $filename to unix format. Failed with an exit code of $exit_code."
);
exit($exit_code >> 8);
}
else {
$self->{'logger'}->info("$filename converted to unix format successfully!");
}
}
return 1;
}
# Delete all files present in the given directory
sub del_all_files_from_dir {
my ($self, $dir_name) = @_;
$self->{'logger'}->info("Deleting all files from directory : $dir_name");
while (my $file = glob("$dir_name/*")) {
next if -d $file;
unlink($file) or $self->{'logger'}->warn("Can't remove $file: $!");
}
return 1;
}
# Get all the files present in given directory based on matching criteria
sub get_files_list_in_dir {
my ($self, $files_directory, $files_matching_criteria) = @_;
if (!defined $files_matching_criteria) {
$files_matching_criteria = "*";
}
my @files_with_full_path
= File::Find::Rule->file->name($files_matching_criteria)->in($files_directory);
return (\@files_with_full_path);
}