Original image is transformed only if it's dimentions differ from target size.
New image|images parameter: quality (int). Default is 95.
package Contenido::File;
use strict;
use warnings;
use Contenido::Globals;
use URI;
use Data::Dumper;
use IO::File;
use IO::Scalar;
use Contenido::File::Scheme::HTTP;
use Contenido::File::Scheme::FILE;
use Contenido::DateTime;
use Image::Info qw(image_info dim);
our $IgnoreErrors = 1;
my %translit = (
'а' => 'a', 'б' => 'b', 'в' => 'v', 'г' => 'g', 'д' => 'd', 'е' => 'e', 'ё' => 'e', 'ж' => 'zh', 'з' => 'z', 'и' => 'i', 'й' => 'y',
'к' => 'k', 'л' => 'l', 'м' => 'm', 'н' => 'n', 'о' => 'o', 'п' => 'p', 'р' => 'r', 'с' => 's', 'т' => 't', 'у' => 'u', 'ф' => 'f', 'х' => 'h',
'ц' => 'ts', 'ч' => '4', 'ш' => 'sh', 'щ' => 'sch', 'ъ' => 'y', 'ы' => 'i', 'ь' => 'y', 'э' => 'e', 'ю' => 'u', 'я' => 'a', 'А' => 'A',
'Б' => 'B', 'В' => 'V', 'Г' => 'G', 'Д' => 'D', 'Е' => 'E', 'Ё' => 'E', 'Ж' => 'ZH', 'З' => 'Z', 'И' => 'I', 'Й' => 'Y', 'К' => 'K', 'Л' => 'L',
'М' => 'M', 'Н' => 'N', 'О' => 'O', 'П' => 'P', 'Р' => 'R', 'С' => 'S', 'Т' => 'T', 'У' => 'U', 'Ф' => 'F', 'Х' => 'H', 'Ц' => 'TS', 'Ч' => '4',
'Ш' => 'SH', 'Щ' => 'SCH', 'Ъ' => 'Y', 'Ы' => 'I', 'Ь' => 'Y', 'Э' => 'E', 'Ю' => 'U', 'Я' => 'YA',
);
sub fetch {
my $filename = shift || return;
my $fh;
foreach my $dir (@{$state->{"files_dir"}}) {
my $path = $dir . '/' . $filename;
no strict "refs";
$fh = &{"Contenido::File::Scheme::".uc(scheme($path))."::fetch"}($path);
return $fh if $fh;
}
}
sub store {
my $filename = shift || return;
my $fh = get_fh(shift) || return;
my $dt = Contenido::DateTime->new()->set_locale('en')->set_time_zone("UTC");
my (@successful, @failure);
my %result;
#убираем сдвоенные и более /
$filename =~ s#/+#/#g;
#убираем начальный /
$filename =~ s#^/##;
foreach my $dir (@{$state->{"files_dir"}}) {
seek $fh, 0, 0;
my $path = $dir . '/' . $filename;
my $scheme = uc(scheme($path));
if ( $scheme eq 'FILE' || $state->{file_web_storage} eq 'separate' || ($state->{file_web_storage} eq 'common' && !exists $result{HTTP}{success}) ) {
no strict "refs";
my $return = &{"Contenido::File::Scheme::".$scheme."::store"}($path, $fh, $dt);
if ( $return ) {
push @successful, $path;
push @{$result{$scheme}{success}}, $path;
} else {
push @failure, $path;
push @{$result{$scheme}{fail}}, $path;
}
}
}
if (!$IgnoreErrors && ( exists $result{FILE}{fail} || ($state->{file_web_storage} eq 'separate' && exists $result{HTTP}{fail})
|| ($state->{file_web_storage} eq 'common' && exists $result{HTTP}{fail} && !exists($result{HTTP}{success})) ) ) {
foreach my $path (@successful) {
no strict "refs";
&{"Contenido::File::Scheme::".uc(scheme($path))."::remove"}($path);
}
return;
}
1;
}
sub size {
my $filename = shift;
my %args = @_;
my $size;
#убираем сдвоенные и более /
$filename =~ s#/+#/#g;
#убираем начальный /
$filename =~ s#^/##;
foreach my $dir (@{$state->{"files_dir"}}) {
no strict "refs";
my $response = &{"Contenido::File::Scheme::".uc(scheme($dir))."::size"}($dir."/".$filename, %args);
return unless defined $response;
return if defined $size and $size != $response;
$size = $response;
#TODO
last;
}
return $size;
}
sub remove {
my $filename = shift || return;
#убираем сдвоенные и более /
$filename =~ s#/+#/#g;
#убираем начальный /
$filename =~ s#^/##;
my $success = 0;
foreach my $dir (@{$state->{"files_dir"}}) {
no strict "refs";
my $scheme = uc(scheme($dir));
my $res;
if ( !$success || $scheme eq 'FILE' ) {
$res = &{"Contenido::File::Scheme::".$scheme."::remove"}($dir."/".$filename);
}
if ( $res && $scheme eq 'HTTP' && $state->{file_web_storage} eq 'common' ) {
$success = 1;
}
}
1;
}
sub get_fh {
my $input = shift;
my $fh;
if (not ref $input) {
no strict "refs";
$fh = &{"Contenido::File::Scheme::".uc(scheme($input))."::get_fh"}($input);
} elsif ( ref $input eq 'Apache::Upload' ) {
$fh = $input->fh;
} elsif ((ref $input eq "GLOB") or (ref $input eq 'IO::File')) {
$fh = $input;
} elsif (ref $input eq "SCALAR") {
$fh = IO::Scalar->new($input);
} else {
$log->warning("Path, scalar ref or fh needed");
return;
}
return $fh
}
sub scheme {
my $uri = shift;
my $scheme;
$scheme = URI->new($uri)->scheme() || "file";
$scheme = 'http' if lc($scheme) eq 'https';
return $scheme;
}
# sub files_dir {
# return unless $_[0];
# my $dir;
# for (scheme($_[0])) {
# /http/ && do {
# $dir = URI->new($_[0])->canonical();
# $dir =~ s|/*$|/|;
# last;
# };
# /file/ && do {
# $dir = URI->new($_[0])->path();
# $dir =~ s|/*$|/|;
# last;
# };
# }
# return $dir;
# }
sub store_image {
my $input = shift;
my (%opts) = @_;
my $object = delete $opts{object} || return;
my $attr = delete $opts{attr};
my $no_rename = delete $opts{no_rename};
my ($prop) = exists $opts{prop} && ref $opts{prop} ? ($opts{prop}) : $attr ? grep { $_->{attr} eq $attr } $object->structure : (undef);
return unless ref $prop;
my @preview = exists $prop->{'preview'} && ref $prop->{'preview'} eq 'ARRAY' ? @{$prop->{'preview'}} : exists $prop->{'preview'} && $prop->{'preview'} ? ($prop->{'preview'}) : ();
my @crops = exists $prop->{'crop'} && ref $prop->{'crop'} eq 'ARRAY' ? @{$prop->{'crop'}} : exists $prop->{'crop'} && $prop->{'crop'} ? ($prop->{'crop'}) : ();
my @shrinks = exists $prop->{'shrink'} && ref $prop->{'shrink'} eq 'ARRAY' ? @{$prop->{'shrink'}} : exists $prop->{'shrink'} && $prop->{'shrink'} ? ($prop->{'shrink'}) : ();
my $quality = exists $prop->{quality} && $prop->{quality} =~ /^\d+$/ ? $prop->{quality} : 95;
my $filename = '/images/'.$object->get_file_name() || return;
if ( $no_rename ) {
my $orig_name = '';
if ( ref $input eq 'Apache::Upload' ) {
$orig_name = $input->filename();
} elsif ( !ref $input ) {
$orig_name = $input;
}
if ( $orig_name ) {
if ( $orig_name =~ /\\([^\\]+)$/ ) {
$orig_name = $1;
} elsif ( $orig_name =~ /\/([^\/]+)$/ ) {
$orig_name = $1;
}
$filename =~ s/\/([^\/]+)$//;
my $fname = $1;
$filename .= '/'.($orig_name || $fname);
$filename =~ s/\.([^\.]+)$//;
}
}
my $filename_tmp = $state->{'tmp_dir'}.'/'.join('_', split('/', $filename));
my $fh = get_fh($input);
return unless ref $fh;
my $ext;
my $size = 1073741824;
if ( not ref $input ) {
$ext = $input =~ /(jpe?g|gif|png)$/i ? lc $1 : 'bin';
} elsif ( ref $input eq 'Apache::Upload' ) {
$ext = $input->filename() =~ /(jpe?g|gif|png)$/i ? lc $1 : 'bin';
} elsif ( $opts{filename} ) {
$ext = $opts{filename} =~ /(jpe?g|gif|png)$/i ? lc $1 : 'bin';
}
if ( ref $fh eq 'IO::Scalar' ) {
$size = length("$fh");
} else {
$size = (stat $fh)[7];
}
warn "Size calculated: $size\n" if $DEBUG;
$ext ||= 'bin';
my $fh_tmp = IO::File->new('>'.$filename_tmp.'.'.$ext) || return;
my $buffer;
my $read_count = 0;
while ( my $bytes_read = sysread( $fh, $buffer, 4096 ) ) {
$read_count += $bytes_read;
syswrite $fh_tmp, $buffer, $bytes_read;
}
$size = $read_count;
undef $fh_tmp;
undef $buffer;
undef $fh;
my $image_info = image_info($filename_tmp.'.'.$ext);
if ( !(ref $image_info && $image_info->{width} && $image_info->{height}) || (ref $image_info && $image_info->{error}) ) {
warn "$filename_tmp.$ext has error: ".$image_info->{error}."\n";
# unlink $filename_tmp.'.'.$ext;
return undef;
}
warn "Got image info\n" if $DEBUG;
if ( $image_info->{file_ext} ne $ext ) {
rename $filename_tmp.'.'.$ext, $filename_tmp.'.'.$image_info->{file_ext};
$ext = $image_info->{file_ext};
}
if ( $image_info->{color_type} eq 'CMYK' ) {
rename $filename_tmp.'.'.$ext, $filename_tmp.'.cmyk.'.$ext;
my $c_line = $state->{'convert_binary'}.' -colorspace RGB -quality 100 '.$filename_tmp.'.cmyk.'.$ext.' '.$filename_tmp.'.'.$ext;
`$c_line`;
unlink $filename_tmp.'.cmyk.'.$ext;
}
my $transformed;
if ( exists $prop->{transform} && ref $prop->{transform} eq 'ARRAY' && scalar @{$prop->{transform}} == 2 && $prop->{transform}[0] =~ /(crop|resize|shrink)/ ) {
warn "Need transform\n" if $DEBUG;
my $c_line;
my ($nwidth, $nheight) = $prop->{transform}[1] =~ /(\d+)x(\d+)/i ? ($1, $2) : (0, 0);
unless ( $nwidth && $nheight && $image_info->{width} == $nwidth && $image_info->{height} == $nheight ) {
if ( $prop->{transform}[0] eq 'resize' ) {
$c_line = $state->{'convert_binary'}.' -adaptive-resize \''.$prop->{transform}[1].'>\' -quality '.$quality.' '.$filename_tmp.'.'.$ext.' '.$filename_tmp.'.transformed.'.$ext;
} elsif ( $prop->{transform}[0] eq 'crop' ) {
my $shave_string;
if ( ($image_info->{width} / $image_info->{height}) > ($nwidth / $nheight) ) {
my $shave_pixels = (($image_info->{width} / $image_info->{height}) - ($nwidth / $nheight)) * $image_info->{height};
$shave_string = ' -shave '.int($shave_pixels / 2).'x0';
} elsif ( ($image_info->{height} / $image_info->{width}) > ($nheight / $nwidth) ) {
my $shave_pixels = (($image_info->{height} / $image_info->{width}) - ($nheight / $nwidth)) * $image_info->{width};
$shave_string = ' -shave 0x'.int($shave_pixels / 2);
}
if ( $shave_string ) {
my $c_line = $state->{"convert_binary"}." $shave_string $filename_tmp.$ext $filename_tmp.shaved.$ext";
my $result = `$c_line`;
if (length $result > 0) {
print "Contenido Error: При вызове '$c_line' произошла ошибка '$result' ($@)\n";
return undef;
}
} else {
my $c_line = "cp $filename_tmp.$ext $filename_tmp.shaved.$ext";
my $result = `$c_line`;
if (length $result > 0) {
print "Contenido Error: При вызове '$c_line' произошла ошибка '$result' ($@)\n";
return undef;
}
}
$c_line = $state->{'convert_binary'}.' -adaptive-resize \''.$prop->{transform}[1].'!\' -quality '.$quality.' '.$filename_tmp.'.shaved.'.$ext.' '.$filename_tmp.'.transformed.'.$ext;
} elsif ( $prop->{transform}[0] eq 'shrink' ) {
$c_line = $state->{'convert_binary'}.' -adaptive-resize \''.$prop->{transform}[1].'!\' -quality '.$quality.' '.$filename_tmp.'.'.$ext.' '.$filename_tmp.'.transformed.'.$ext;
}
my $result = `$c_line`;
$transformed = 1;
warn "Transformed\n" if $DEBUG;
unlink $filename_tmp.'.shaved.'.$ext if -e $filename_tmp.'.shaved.'.$ext;
$size = -s $filename_tmp.'.transformed.'.$ext;
}
}
if ( exists $opts{watermark} && $opts{watermark} ) {
warn "Need watermark\n" if $DEBUG;
my $gravity = delete $opts{gravity} || 'Center';
my $source = $transformed ? $filename_tmp.'.transformed.'.$ext : $filename_tmp.'.'.$ext;
my $target = $filename_tmp.'.transformed.'.$ext;
my $offset = delete $opts{offset} || '+0+0';
my $c_line = $state->{'composite_binary'}." -geometry $offset -gravity $gravity -quality 99 $opts{watermark} $source $target";
warn "Watermark: $c_line\n" if $DEBUG;
my $result = `$c_line`;
$transformed = 1;
warn "Watermarked\n" if $DEBUG;
}
my $IMAGE;
my $stored = $transformed ? store($filename.'.'.$ext, $filename_tmp.'.transformed.'.$ext) : store($filename.'.'.$ext, $filename_tmp.'.'.$ext);
if ( $stored ) {
warn "Stored\n" if $DEBUG;
$IMAGE = {};
# hashref slice assigning - жжесть
if ( $transformed && -e $filename_tmp.'.transformed.'.$ext ) {
my ($tw, $th) = Image::Size::imgsize($filename_tmp.'.transformed.'.$ext);
my ($w, $h) = Image::Size::imgsize($filename_tmp.'.'.$ext);
@{$IMAGE}{'filename', 't_width', 't_height', 'width', 'height', 'size'} = (
$filename.'.'.$ext, $tw, $th, $w, $h, $size
);
unlink $filename_tmp.'.transformed.'.$ext;
} else {
@{$IMAGE}{'filename', 'size', 'width', 'height'} = (
$filename.'.'.$ext,
$size,
Image::Size::imgsize($filename_tmp.'.'.$ext),
);
}
warn "Thumbnail generator (preview)\n" if $DEBUG;
foreach my $suffix (@preview) {
my $c_line = $state->{'convert_binary'}.' -resize \''.$suffix.'>\' -quality '.$quality.' '.$filename_tmp.'.'.$ext.' '.$filename_tmp.'.'.$suffix.'.'.$ext;
my $result = `$c_line`;
if (length $result > 0) {
warn 'Contenido Error: При вызове "'.$c_line.'" произошла ошибка "'.$result.'" ('.$@.")\n";
return undef;
}
@{$IMAGE->{'mini'}{$suffix}}{'filename', 'size', 'width', 'height'} = (
$filename.'.'.$suffix.'.'.$ext,
-s $filename_tmp.'.'.$suffix.'.'.$ext,
Image::Size::imgsize($filename_tmp.'.'.$suffix.'.'.$ext),
);
%{$IMAGE->{'resize'}{$suffix}} = %{$IMAGE->{'mini'}{$suffix}};
store($filename.'.'.$suffix.'.'.$ext, $filename_tmp.'.'.$suffix.'.'.$ext);
unlink $filename_tmp.'.'.$suffix.'.'.$ext if -e $filename_tmp.'.'.$suffix.'.'.$ext;
}
if ( @preview ) {
@{$IMAGE->{'mini'}}{'filename', 'width', 'height'} = @{$IMAGE->{'mini'}{$preview[0]}}{'filename', 'width', 'height'};
@{$IMAGE->{'resize'}}{'filename', 'width', 'height'} = @{$IMAGE->{'mini'}{$preview[0]}}{'filename', 'width', 'height'};
}
########## CROPS
warn "Thumbnail generator (crop)\n" if $DEBUG;
foreach my $suffix (@crops) {
my $shave_string;
my ($nwidth, $nheight) = $suffix =~ /(\d+)x(\d+)/i ? ($1, $2) : (0, 0);
if ( ($IMAGE->{width} / $IMAGE->{height}) > ($nwidth / $nheight) ) {
my $shave_pixels = (($IMAGE->{width} / $IMAGE->{height}) - ($nwidth / $nheight)) * $IMAGE->{height};
$shave_string = ' -shave '.int($shave_pixels / 2).'x0';
} elsif ( ($IMAGE->{height} / $IMAGE->{width}) > ($nheight / $nwidth) ) {
my $shave_pixels = (($IMAGE->{height} / $IMAGE->{width}) - ($nheight / $nwidth)) * $IMAGE->{width};
$shave_string = ' -shave 0x'.int($shave_pixels / 2);
}
if ( $shave_string ) {
my $c_line = $state->{"convert_binary"}." $shave_string $filename_tmp.$ext $filename_tmp.shaved.$ext";
my $result = `$c_line`;
if (length $result > 0) {
print "Contenido Error: При вызове '$c_line' произошла ошибка '$result' ($@)\n";
}
} else {
my $c_line = "cp $filename_tmp.$ext $filename_tmp.shaved.$ext";
my $result = `$c_line`;
if (length $result > 0) {
print "Contenido Error: При вызове '$c_line' произошла ошибка '$result' ($@)\n";
}
}
my $c_line = $state->{'convert_binary'}.' -geometry \''.$suffix.'!\' -quality '.$quality.' '.$filename_tmp.'.shaved.'.$ext.' '.$filename_tmp.'.'.$suffix.'.'.$ext;
my $result = `$c_line`;
if (length $result > 0) {
warn 'Contenido Error: При вызове "'.$c_line.'" произошла ошибка "'.$result.'" ('.$@.")\n";
return undef;
}
@{$IMAGE->{'mini'}{$suffix}}{'filename', 'size', 'width', 'height'} = (
$filename.'.'.$suffix.'.'.$ext,
-s $filename_tmp.'.'.$suffix.'.'.$ext,
Image::Size::imgsize($filename_tmp.'.'.$suffix.'.'.$ext),
);
%{$IMAGE->{'crop'}{$suffix}} = %{$IMAGE->{'mini'}{$suffix}};
store($filename.'.'.$suffix.'.'.$ext, $filename_tmp.'.'.$suffix.'.'.$ext);
unlink $filename_tmp.'.shaved.'.$ext if -e $filename_tmp.'.shaved.'.$ext;
unlink $filename_tmp.'.'.$suffix.'.'.$ext if -e $filename_tmp.'.'.$suffix.'.'.$ext;
}
if ( @crops ) {
if ( !exists $IMAGE->{'mini'}{'filename'} ) {
@{$IMAGE->{'mini'}}{'filename', 'width', 'height'} = @{$IMAGE->{'mini'}{$crops[0]}}{'filename', 'width', 'height'};
}
@{$IMAGE->{'crop'}}{'filename', 'width', 'height'} = @{$IMAGE->{'crop'}{$crops[0]}}{'filename', 'width', 'height'};
}
########## SHRINKS
warn "Thumbnail generator (shrink)\n" if $DEBUG;
foreach my $suffix (@shrinks) {
my $c_line = $state->{'convert_binary'}.' -geometry \''.$suffix.'!\' -quality '.$quality.' '.$filename_tmp.'.'.$ext.' '.$filename_tmp.'.'.$suffix.'.'.$ext;
my $result = `$c_line`;
if (length $result > 0) {
warn 'Contenido Error: При вызове "'.$c_line.'" произошла ошибка "'.$result.'" ('.$@.")\n";
return undef;
}
@{$IMAGE->{'mini'}{$suffix}}{'filename', 'size', 'width', 'height'} = (
$filename.'.'.$suffix.'.'.$ext,
-s $filename_tmp.'.'.$suffix.'.'.$ext,
Image::Size::imgsize($filename_tmp.'.'.$suffix.'.'.$ext),
);
%{$IMAGE->{'shrink'}{$suffix}} = %{$IMAGE->{'mini'}{$suffix}};
store($filename.'.'.$suffix.'.'.$ext, $filename_tmp.'.'.$suffix.'.'.$ext);
unlink $filename_tmp.'.'.$suffix.'.'.$ext if -e $filename_tmp.'.'.$suffix.'.'.$ext;
}
if ( @shrinks && !exists $IMAGE->{'mini'}{'filename'} ) {
if ( !exists $IMAGE->{'mini'}{'filename'} ) {
@{$IMAGE->{'mini'}}{'filename', 'width', 'height'} = @{$IMAGE->{'mini'}{$shrinks[0]}}{'filename', 'width', 'height'};
}
@{$IMAGE->{'shrink'}}{'filename', 'width', 'height'} = @{$IMAGE->{'shrink'}{$shrinks[0]}}{'filename', 'width', 'height'};
}
unlink $filename_tmp.'.'.$ext if -e $filename_tmp.'.'.$ext;
$IMAGE->{width} = delete $IMAGE->{t_width} if exists $IMAGE->{t_width};
$IMAGE->{height} = delete $IMAGE->{t_height} if exists $IMAGE->{t_height};
}
warn "That's all: ".Dumper($IMAGE) if $DEBUG;
return $IMAGE;
}
sub remove_image {
my $IMAGE = shift;
if ( ref $IMAGE eq 'HASH' && exists $IMAGE->{filename} ) {
remove($IMAGE->{'filename'}) || return;
}
if ( ref $IMAGE && exists $IMAGE->{mini} && ref $IMAGE->{mini} eq 'HASH' ) {
foreach my $val ( values %{$IMAGE->{mini}} ) {
if ( ref $val && exists $val->{filename} && $val->{filename} ) {
remove($val->{'filename'}) || return;
}
}
}
1;
}
sub store_binary {
my $input = shift;
my (%opts) = @_;
my $object = delete $opts{object} || return;
my $attr = delete $opts{attr};
my ($prop) = exists $opts{prop} && ref $opts{prop} ? ($opts{prop}) : $attr ? grep { $_->{attr} eq $attr } $object->structure : (undef);
return unless ref $prop;
my $filename = '/binary/'.$object->get_file_name() || return;
my $orig_name = '';
if ( ref $input eq 'Apache::Upload' ) {
$orig_name = $input->filename();
} elsif ( !ref $input ) {
$orig_name = $input;
}
if ( $orig_name ) {
if ( $orig_name =~ /\\([^\\]+)$/ ) {
$orig_name = $1;
} elsif ( $orig_name =~ /\/([^\/]+)$/ ) {
$orig_name = $1;
}
}
if ( $prop->{softrename} ) {
my $oid = $object->id || int(rand(10000));
if ( $orig_name ) {
my $set_name = $orig_name;
$set_name =~ s/[\ \t]/_/g;
$set_name = $oid.'_'.$set_name;
$filename =~ s/\/([^\/]+)$//;
my $fname = $1;
unless ( $set_name =~ /^[a-zA-Z_\d\.\-\,]+$/ ) {
$set_name = translit( $set_name );
}
warn "\n\n\n\n\nNew Name: [$set_name]\n\n\n\n\n" if $DEBUG;
unless ( $set_name =~ /^[a-zA-Z_\d\.\-\,]+$/ ) {
$set_name = $fname;
}
$filename .= '/'.$set_name;
$filename =~ s/\.([^\.]+)$//;
}
}
my $filename_tmp = $state->{'tmp_dir'}.'/'.join('_', split('/', $filename));
my $fh = get_fh($input);
return unless ref $fh;
my $ext;
my $size = 1073741824;
if ( not ref $input ) {
$ext = $input =~ /\.([^\.]+)$/ ? lc($1) : 'bin';
} elsif ( ref $input eq 'Apache::Upload' ) {
$ext = $input->filename() =~ /\.([^\.]+)$/ ? lc($1) : 'bin';
$size = (stat $fh)[7];
} elsif ( $opts{filename} ) {
$ext = $opts{filename} =~ /\.([^\.]+)$/ ? lc($1) : 'bin';
}
if ( ref $fh eq 'IO::Scalar' ) {
$size = length("$fh");
} else {
$size = (stat $fh)[7];
}
$ext ||= 'bin';
my $fh_tmp = IO::File->new('>'.$filename_tmp.'.'.$ext) || return;
my $buffer;
my $read_count = 0;
while ( my $bytes_read = sysread( $fh, $buffer, 4096 ) ) {
$read_count += $bytes_read;
syswrite $fh_tmp, $buffer, $bytes_read;
}
$size = $read_count;
undef $fh_tmp;
undef $buffer;
undef $fh;
my $BINARY;
if ( store($filename.'.'.$ext, $filename_tmp.'.'.$ext) ) {
@{$BINARY}{"filename", "ext", "size", "sourcename"} = (
$filename.".".$ext, $ext, $size, $orig_name
);
unlink $filename_tmp.'.'.$ext if -e $filename_tmp.'.'.$ext;
if ( $ext =~ /(rar|7z|zip|arc|lha|arj|cab)/ ) {
$BINARY->{type} = 'archive';
} elsif ( $ext =~ /(doc|docx|rtf)/ ) {
$BINARY->{type} = 'doc';
} elsif ( $ext =~ /(xls|xlsx)/ ) {
$BINARY->{type} = 'xls';
} elsif ( $ext =~ /(mdb|ppt)/ ) {
$BINARY->{type} = 'msoffice';
} elsif ( $ext =~ /(pdf|fb2|djvu)/ ) {
$BINARY->{type} = 'ebook';
} elsif ( $ext =~ /(psd|cdr)/ ) {
$BINARY->{type} = 'graphics';
} elsif ( $ext eq 'ico' ) {
$BINARY->{type} = 'icon';
} elsif ( $ext =~ /(exe|msi|cab)/ ) {
$BINARY->{type} = 'executable';
} else {
$BINARY->{type} = 'unknown';
}
}
return $BINARY;
}
sub remove_binary {
my $BINARY = shift;
if ( ref $BINARY eq 'HASH' && exists $BINARY->{filename} ) {
remove($BINARY->{'filename'}) || return;
}
1;
}
sub translit {
my $str = shift;
my @str = split (//, $str);
my $res = '';
while ( scalar @str ) {
my $alpha = shift @str;
if ( exists $translit{$alpha} ) {
$res .= $translit{$alpha};
} else {
$res .= $alpha;
}
}
return $res;
}
1;
Небольшая справка по веткам
cnddist – контейнер, в котором хранятся все дистрибутивы всех библиотек и программных пакетов, которые использовались при построении различных версий Contenido. Если какой-то библиотеки в данном хранилище нет, инсталлятор сделает попытку "подтянуть" ее с веба (например, с CPAN). Если библиотека слишком старая, есть очень большая вероятность, что ее там уже нет. Поэтому мы храним весь хлам от всех сборок. Если какой-то дистрибутив вдруг отсутствует в cnddist - напишите нам, мы положим его туда.
koi8 – отмирающая ветка, чей код, выдача и все внутренние библиотеки заточены на кодировку KOI8-R. Вносятся только те дополнения, которые касаются внешнего вида и функционала админки, баги ядра, обязательные обновления портов и мелочи, которые легко скопипастить. В дальнейшем планируется полная остановка поддержки по данной ветке.
utf8 – актуальная ветка, заточенная под UTF-8.
Внутри каждой ветки: core – исходники ядра; install – скрипт установки инсталляции; plugins – плагины; samples – "готовые к употреблению" проекты, которые можно поставить, запустить и посмотреть, как они работают.