X-Git-Url: http://git.freeside.biz/gitweb/?a=blobdiff_plain;ds=sidebyside;f=rt%2Flib%2FRT%2FUtil.pm;h=fefb0b90c30df9d1f37a164b7a093ceac11ee2a3;hb=44dd00a3ff974a17999e86e64488e996edc71e3c;hp=24efe71f5ead012955e92c44c5c85ab42785b134;hpb=85e677b86fc37c54e6de2b06340351a28f5a5916;p=freeside.git diff --git a/rt/lib/RT/Util.pm b/rt/lib/RT/Util.pm index 24efe71f5..fefb0b90c 100644 --- a/rt/lib/RT/Util.pm +++ b/rt/lib/RT/Util.pm @@ -2,7 +2,7 @@ # # COPYRIGHT: # -# This software is Copyright (c) 1996-2012 Best Practical Solutions, LLC +# This software is Copyright (c) 1996-2019 Best Practical Solutions, LLC # # # (Except where explicitly superseded by other copyright notices) @@ -54,6 +54,8 @@ use warnings; use base 'Exporter'; our @EXPORT = qw/safe_run_child mime_recommended_filename/; +use Encode qw/encode/; + sub safe_run_child (&) { my $our_pid = $$; @@ -65,9 +67,12 @@ sub safe_run_child (&) { # values. Instead we set values, eval code, check pid # on failure and reset values only in our original # process - my $dbh = $RT::Handle->dbh; + my ($oldv_dbh, $oldv_rth); + my $dbh = $RT::Handle ? $RT::Handle->dbh : undef; + $oldv_dbh = $dbh->{'InactiveDestroy'} if $dbh; $dbh->{'InactiveDestroy'} = 1 if $dbh; - $RT::Handle->{'DisconnectHandleOnDestroy'} = 0; + $oldv_rth = $RT::Handle->{'DisconnectHandleOnDestroy'} if $RT::Handle; + $RT::Handle->{'DisconnectHandleOnDestroy'} = 0 if $RT::Handle; my ($reader, $writer); pipe( $reader, $writer ); @@ -90,8 +95,8 @@ sub safe_run_child (&) { my $err = $@; $err =~ s/^Stack:.*$//ms; if ( $our_pid == $$ ) { - $dbh->{'InactiveDestroy'} = 0 if $dbh; - $RT::Handle->{'DisconnectHandleOnDestroy'} = 1; + $dbh->{'InactiveDestroy'} = $oldv_dbh if $dbh; + $RT::Handle->{'DisconnectHandleOnDestroy'} = $oldv_rth if $RT::Handle; die "System Error: $err"; } else { print $writer "System Error: $err"; @@ -104,8 +109,8 @@ sub safe_run_child (&) { my ($response) = $reader->getline; warn $response if $response; - $dbh->{'InactiveDestroy'} = 0 if $dbh; - $RT::Handle->{'DisconnectHandleOnDestroy'} = 1; + $dbh->{'InactiveDestroy'} = $oldv_dbh if $dbh; + $RT::Handle->{'DisconnectHandleOnDestroy'} = $oldv_rth if $RT::Handle; return $want? (@res) : $res[0]; } @@ -122,7 +127,7 @@ sub mime_recommended_filename { $head = $head->head if $head->isa('MIME::Entity'); for my $attr_name (qw( content-disposition.filename content-type.name )) { - my $value = $head->mime_attr($attr_name); + my $value = Encode::decode("UTF-8",$head->mime_attr($attr_name)); if ( defined $value && $value =~ /\S/ ) { return $value; } @@ -130,6 +135,87 @@ sub mime_recommended_filename { return; } +sub assert_bytes { + my $string = shift; + return unless utf8::is_utf8($string); + return unless $string =~ /([^\x00-\x7F])/; + + my $msg; + if (ord($1) > 255) { + $msg = "Expecting a byte string, but was passed characters"; + } else { + $msg = "Expecting a byte string, but was possibly passed charcters;" + ." if the string is actually bytes, please use utf8::downgrade"; + } + $RT::Logger->warn($msg, Carp::longmess()); + +} + + +=head2 C + +Compares two strings for equality in constant-time. Replacement for the C +operator designed to avoid timing side-channel vulnerabilities. Returns zero +or one. + +This is intended for use in cryptographic subsystems for comparing well-formed +data such as hashes - not for direct use with user input or as a general +replacement for the C operator. + +The two string arguments B be of equal length. If the lengths differ, +this function will call C, as proceeding with execution would create +a timing vulnerability. Length is defined by characters, not bytes. + +Strings that should be treated as binary octets rather than Unicode text +should pass a true value for the binary flag. + +This code has been tested to do what it claims. Do not change it without +thorough statistical timing analysis to validate the changes. + +Added to resolve CVE-2017-5361 + +For more on timing attacks, see this Wikipedia article: +B + +=cut + +sub constant_time_eq { + my ($a, $b, $binary) = @_; + + my $result = 0; + + # generic error message avoids potential information leaks + my $generic_error = "Cannot compare values"; + die $generic_error unless defined $a and defined $b; + die $generic_error unless length $a == length $b; + die $generic_error if ref($a) or ref($b); + + for (my $i = 0; $i < length($a); $i++) { + my $a_char = substr($a, $i, 1); + my $b_char = substr($b, $i, 1); + + my (@a_octets, @b_octets); + + if ($binary) { + @a_octets = ord($a_char); + @b_octets = ord($b_char); + } + else { + # encode() is set to die on malformed + @a_octets = unpack("C*", encode('UTF-8', $a_char, Encode::FB_CROAK)); + @b_octets = unpack("C*", encode('UTF-8', $b_char, Encode::FB_CROAK)); + } + + die $generic_error if (scalar @a_octets) != (scalar @b_octets); + + for (my $j = 0; $j < scalar @a_octets; $j++) { + $result |= $a_octets[$j] ^ $b_octets[$j]; + } + } + return 0 + not $result; +} + + RT::Base->_ImportOverlays(); 1;