1 |
8 |
ahitrov@rambler.ru |
package Contenido::Object; |
2 |
|
|
|
3 |
|
|
# ---------------------------------------------------------------------------- |
4 |
|
|
# Класс Объект. |
5 |
|
|
# Родительский класс для всех типов. Раз уж мы используем ООП, то давайте |
6 |
|
|
# действительно его использовать. |
7 |
|
|
# |
8 |
|
|
# От него мы будем наследовать механизмы работы с директорией данных, |
9 |
|
|
# механизмы работы с бинарными файлами и так далее... |
10 |
|
|
# ---------------------------------------------------------------------------- |
11 |
|
|
|
12 |
|
|
use strict; |
13 |
|
|
use warnings; |
14 |
|
|
use locale; |
15 |
|
|
|
16 |
|
|
use vars qw($VERSION $AUTOLOAD); |
17 |
|
|
$VERSION = '4.1'; |
18 |
|
|
|
19 |
|
|
use Utils; |
20 |
|
|
use Contenido::Globals; |
21 |
113 |
ahitrov |
use Contenido::File; |
22 |
8 |
ahitrov@rambler.ru |
use Data::Dumper; |
23 |
217 |
ahitrov |
use Data::Recursive::Encode; |
24 |
|
|
use JSON::XS; |
25 |
8 |
ahitrov@rambler.ru |
|
26 |
|
|
use DBD::Pg; |
27 |
|
|
use Encode; |
28 |
|
|
|
29 |
|
|
use SQL::ProtoTable; |
30 |
|
|
|
31 |
217 |
ahitrov |
our $json_n = JSON::XS->new->utf8(0); |
32 |
|
|
our $json_u = JSON::XS->new->utf8(1); |
33 |
|
|
|
34 |
8 |
ahitrov@rambler.ru |
# required properties теперь берутся из свойств таблицы |
35 |
|
|
sub required_properties { |
36 |
|
|
my $self=shift; |
37 |
|
|
my $class = ref($self) || $self; |
38 |
|
|
if ($class->can('class_table')) { |
39 |
|
|
return $self->class_table->required_properties(); |
40 |
|
|
} else { |
41 |
|
|
$log->error("$class cannot method class_table"); |
42 |
|
|
return (); |
43 |
|
|
} |
44 |
|
|
} |
45 |
|
|
|
46 |
|
|
sub extra_properties { |
47 |
|
|
return (); |
48 |
|
|
} |
49 |
|
|
|
50 |
|
|
sub post_init { |
51 |
|
|
return; |
52 |
|
|
} |
53 |
|
|
|
54 |
|
|
sub pre_store { |
55 |
|
|
return 1; |
56 |
|
|
} |
57 |
|
|
|
58 |
|
|
sub post_store { |
59 |
|
|
return 1; |
60 |
|
|
} |
61 |
|
|
|
62 |
|
|
sub post_finish_store { |
63 |
|
|
return 1; |
64 |
|
|
} |
65 |
|
|
|
66 |
|
|
sub pre_delete { |
67 |
|
|
return 1; |
68 |
|
|
} |
69 |
|
|
|
70 |
|
|
sub post_delete { |
71 |
|
|
return 1; |
72 |
|
|
} |
73 |
|
|
|
74 |
|
|
sub pre_abort { |
75 |
|
|
return 1; |
76 |
|
|
} |
77 |
|
|
|
78 |
|
|
sub t_abort { |
79 |
|
|
my $self = shift; |
80 |
|
|
$self->pre_abort(); |
81 |
|
|
return $self->keeper->t_abort(); |
82 |
|
|
} |
83 |
|
|
|
84 |
|
|
sub new { |
85 |
|
|
$log->error("Method 'new' cannot be called for class Contenido::Object"); |
86 |
|
|
die; |
87 |
|
|
} |
88 |
|
|
|
89 |
|
|
sub class_table { |
90 |
|
|
$log->error("Method 'class_table' cannot be called for Contenido::Object"); |
91 |
|
|
die; |
92 |
|
|
} |
93 |
|
|
|
94 |
|
|
#обьявляем пустой DESTROY чтобы эта зараза в AUTOLOAD не попадала |
95 |
|
|
sub DESTROY {} |
96 |
|
|
|
97 |
|
|
#получение ключа в кеше по $object или по $class/$id |
98 |
|
|
#can be overloaded in class |
99 |
|
|
sub get_object_key { |
100 |
|
|
my $self = shift; |
101 |
|
|
return $self->class_table->_get_object_key($self, @_); |
102 |
|
|
} |
103 |
|
|
|
104 |
|
|
#получение ключа в кеше по $object или по $class/$unique |
105 |
|
|
#can be overloaded in class |
106 |
|
|
sub get_object_unique_key { |
107 |
|
|
my $self = shift; |
108 |
|
|
return $self->class_table->_get_object_unique_key($self, @_); |
109 |
|
|
} |
110 |
|
|
|
111 |
|
|
#скорость тут совершенно не критична... исполняется 1 раз на каждый класс |
112 |
|
|
#??? возможно лучше сделать методы доступа к свойствам на этом этапе |
113 |
|
|
sub class_init { |
114 |
|
|
my $self = shift; |
115 |
|
|
my $class = ref($self) || $self; |
116 |
|
|
|
117 |
|
|
{ |
118 |
|
|
no strict 'refs'; |
119 |
|
|
return 1 if (${$class.'::class_init_done'}); |
120 |
|
|
use strict; |
121 |
|
|
} |
122 |
|
|
|
123 |
|
|
#инициализируем описатель таблицы |
124 |
|
|
if ($class->can('class_table')) { |
125 |
|
|
eval { SQL::ProtoTable->table_init($class->class_table) }; |
126 |
|
|
do { $log->error("Cannot initialise class $class!"); die } if ($@); |
127 |
|
|
} |
128 |
|
|
|
129 |
|
|
#валидация корректности класса (todo) |
130 |
|
|
#$self->class_validate(); |
131 |
|
|
|
132 |
|
|
my $funct; |
133 |
|
|
|
134 |
|
|
#начало текста функции инициализатора класса из базы |
135 |
|
|
my $funct_begin = " |
136 |
|
|
my (\$class, \$row, \$keeper, \$light) = \@_; |
137 |
|
|
"; |
138 |
|
|
|
139 |
|
|
my $funct_start_obj =' return bless({'; |
140 |
|
|
my $funct_end_obj = "}, '$class'); |
141 |
|
|
"; |
142 |
|
|
my $funct_begin_if_light = ' |
143 |
|
|
if ($light) {'; |
144 |
|
|
my $funct_elsif_light = ' |
145 |
|
|
} else {'; |
146 |
|
|
my $funct_endif_light = ' |
147 |
|
|
}'; |
148 |
|
|
|
149 |
|
|
my $func_start_encode = ''; |
150 |
|
|
my $func_end_encode = ''; |
151 |
|
|
|
152 |
|
|
if ($state->db_encode_data) { |
153 |
|
|
$func_start_encode = 'Encode::encode("'.$state->db_encode_data.'", '; |
154 |
|
|
$func_end_encode = ', Encode::FB_HTMLCREF)'; |
155 |
|
|
} |
156 |
|
|
|
157 |
390 |
ahitrov |
my @funct_default_fields = ("class=>'$class'", "keeper=>\$keeper", "__light=>\$light"); |
158 |
8 |
ahitrov@rambler.ru |
my @funct_exra_fields = (); |
159 |
|
|
|
160 |
|
|
#те вещи которые надо заранее подсчитать при инициализации класса |
161 |
|
|
my (%props, %attributes, @extra_fields, %virtual_fields, @structure); |
162 |
|
|
|
163 |
|
|
my $pos = 0; |
164 |
|
|
#последовательность reload: required_properties может быть перекрытым через add_properties который может быть далее перекрыт через extra_properties |
165 |
|
|
|
166 |
|
|
foreach my $prop ($self->required_properties()) { |
167 |
|
|
my $attr = $prop->{attr}; |
168 |
|
|
unless ($attr) { |
169 |
|
|
$log->error("$class have wrong data in required_properties (no attr for field)"); |
170 |
|
|
next; |
171 |
|
|
} |
172 |
|
|
unless ($prop->{db_type} || $prop->{virtual}) { |
173 |
|
|
$log->warning("$class with class table: ".$self->class_table()." property '$attr' missing db_type in table descriptor... can be incompatible with future versions!"); |
174 |
|
|
} |
175 |
|
|
|
176 |
|
|
$props{$attr} = $prop; |
177 |
|
|
|
178 |
|
|
push @structure, $prop; |
179 |
|
|
|
180 |
|
|
#вообще с классом надо подумать... есть идея что для части таблиц класс поле не нужно... только место ест |
181 |
|
|
next if ($attr eq 'class'); |
182 |
|
|
|
183 |
|
|
#поля которые идут в DB могут быть обьявлены ТОЛЬКО в required_properties |
184 |
|
|
if (exists($prop->{db_field}) and $prop->{db_field}) { |
185 |
|
|
$pos++; |
186 |
|
|
#$DBD::Pg versions since 2.0.0 do it automatically |
187 |
|
|
if ($DBD::Pg::VERSION=~/^1\./ and $prop->{db_type} and (($prop->{db_type} eq 'integer[]') or ($prop->{db_type} eq 'integer_ref[]'))) { |
188 |
|
|
push @funct_default_fields, "$attr=>[(\$row->[$pos] and \$row->[$pos]=~/^{(\\d+(?:,\\d+)*)}\$/) ? split(/,/, \$1) : ()]"; |
189 |
|
|
} else { |
190 |
|
|
push @funct_default_fields, "$attr=>$func_start_encode\$row->[$pos]$func_end_encode"; |
191 |
|
|
} |
192 |
|
|
} |
193 |
|
|
|
194 |
|
|
if ($prop->{db_type} and ($prop->{db_type} eq 'integer[]')) { |
195 |
|
|
$attributes{$attr} = 'ARRAY'; |
196 |
|
|
} elsif($prop->{db_type} and ($prop->{db_type} eq 'integer_ref[]')) { |
197 |
|
|
$attributes{$attr} = 'ARRAY_REF'; |
198 |
|
|
} else { |
199 |
|
|
$attributes{$attr} = 'SCALAR'; |
200 |
|
|
} |
201 |
|
|
} |
202 |
|
|
push @funct_default_fields, "attributes=>\$${class}::attributes"; |
203 |
|
|
|
204 |
|
|
my $have_extra = $self->class_table->have_extra; |
205 |
|
|
if ($have_extra) { |
206 |
|
|
my @ap = $self->add_properties() if $self->can('add_properties'); |
207 |
|
|
#последовательность reload: required_properties может быть перекрытым через add_properties который может быть далее перекрыт через extra_properties |
208 |
|
|
foreach my $prop (@ap, $self->extra_properties()) { |
209 |
|
|
my $attr = $prop->{attr}; |
210 |
|
|
if (exists($props{$attr})) { |
211 |
|
|
#reload code |
212 |
|
|
$log->info("Reloaded property $attr for class $class") if ($DEBUG); |
213 |
|
|
while ( my ($field, $value) = each(%$prop)) { |
214 |
|
|
$props{$attr}->{$field} = $value; |
215 |
|
|
} |
216 |
|
|
} else { |
217 |
|
|
$props{$attr} = $prop; |
218 |
|
|
#если это был не overload то это новое extra поле |
219 |
|
|
push @extra_fields, $attr; |
220 |
|
|
push @structure, $prop; |
221 |
|
|
$attributes{$attr} = 'SCALAR'; |
222 |
|
|
if ($prop->{virtual}) { |
223 |
|
|
#выставляем что это вообще виртуальный атрибут |
224 |
|
|
$virtual_fields{$attr} = 1; |
225 |
|
|
} else { |
226 |
|
|
#инициализируем из dump все кроме виртуальных свойств |
227 |
392 |
ahitrov |
push @funct_exra_fields, "$attr=>(\$keeper->serialize_with eq 'json' ? (ref \$dump->{$attr} ? Data::Recursive::Encode->encode_utf8(\$dump->{$attr}) : Encode::encode('utf-8', \$dump->{$attr}, Encode::FB_HTMLCREF) ) : $func_start_encode\$dump->{$attr}$func_end_encode)"; |
228 |
8 |
ahitrov@rambler.ru |
} |
229 |
|
|
} |
230 |
|
|
} |
231 |
|
|
} |
232 |
|
|
|
233 |
|
|
$attributes{class} = 'SCALAR'; |
234 |
|
|
|
235 |
|
|
#если у обьекта есть extra_attributes надо бы вызвать restore_extras если не указан light |
236 |
|
|
#наличие have_extra у таблицы не ведет к обязательному наличию extra_fields |
237 |
|
|
if (@extra_fields) { |
238 |
|
|
# -------------------------------------------------------------------------------------------- |
239 |
|
|
# Чтение из одного дампа в базе данных |
240 |
|
|
# -------------------------------------------------------------------------------------------- |
241 |
|
|
my $funct_eval_dump .= ' |
242 |
217 |
ahitrov |
my $dump = $keeper->serialize_with eq \'json\' ? (Contenido::Object::eval_json(\\$row->[-1]) || {}) : Contenido::Object::eval_dump(\\$row->[-1]); |
243 |
8 |
ahitrov@rambler.ru |
'; |
244 |
|
|
$funct = $funct_begin.$funct_begin_if_light.$funct_start_obj.join(', ', @funct_default_fields).$funct_end_obj.$funct_elsif_light.$funct_eval_dump.$funct_start_obj.join(', ', (@funct_default_fields, @funct_exra_fields)).$funct_end_obj.$funct_endif_light; |
245 |
|
|
} else { |
246 |
|
|
$funct = $funct_begin.$funct_start_obj.join(', ', @funct_default_fields).$funct_end_obj; |
247 |
|
|
} |
248 |
217 |
ahitrov |
# warn "Restore function: [$funct]\n"; |
249 |
8 |
ahitrov@rambler.ru |
|
250 |
|
|
create_method($class, 'init_from_db', $funct); |
251 |
|
|
|
252 |
|
|
{ |
253 |
|
|
no strict 'refs'; |
254 |
|
|
${$class.'::structure'} = \@structure; |
255 |
|
|
${$class.'::attributes'} = \%attributes; |
256 |
|
|
${$class.'::extra_fields'} = \@extra_fields; |
257 |
|
|
${$class.'::virtual_fields'} = \%virtual_fields; |
258 |
|
|
${$class.'::class_init_done'} = 1; |
259 |
|
|
} |
260 |
|
|
return 1; |
261 |
|
|
} |
262 |
|
|
|
263 |
|
|
# ------------------------------------------------------------------------------------------- |
264 |
|
|
# Сохраняет внешние свойства объекта в зависимости от выбранного способа... |
265 |
|
|
# ------------------------------------------------------------------------------------------- |
266 |
|
|
sub store_extras { |
267 |
|
|
my $self = shift; |
268 |
|
|
my %opts = @_; |
269 |
|
|
do {$log->error("Метод store_extras() можно вызывать только у объектов, но не классов\n"); die } unless ref($self); |
270 |
|
|
|
271 |
|
|
do { $log->error("В объекте не определена ссылка на базу данных"); die } unless ref($self->keeper); |
272 |
|
|
do { $log->error("Не задан режим сохранения данных (insert/update)"); die } if (($opts{mode} ne 'insert') && ($opts{mode} ne 'update')); |
273 |
|
|
do { $log->error("Не задан идентификатор объекта (а должен быть задан в обязательном порядке)"); die } unless($self->id()); |
274 |
|
|
|
275 |
|
|
if ($self->keeper->store_method() eq 'sqldump') { |
276 |
|
|
my $extra_table=$self->class_table->extra_table; |
277 |
|
|
do { $log->error("No extra table for class $self->{class}"); die } unless ($extra_table); |
278 |
|
|
if ($opts{mode} eq 'insert') { |
279 |
|
|
$self->keeper->TSQL->do("INSERT INTO $extra_table (id, data) VALUES (?, ?)", {}, $self->id(), $self->_create_extra_dump()) || $self->t_abort(); |
280 |
|
|
} else { |
281 |
|
|
$self->keeper->TSQL->do("UPDATE $extra_table SET data=? WHERE id=?", {}, $self->_create_extra_dump(), $self->id()) || $self->t_abort(); |
282 |
|
|
} |
283 |
|
|
|
284 |
|
|
} elsif ($self->keeper->store_method() eq 'toast') { |
285 |
|
|
my $table = $self->class_table->db_table; |
286 |
|
|
do { $log->error("There no db_table for class $self->{class}"); die } unless ($table); |
287 |
|
|
$self->keeper->TSQL->do("UPDATE $table SET data=? WHERE id=?", {}, $self->_create_extra_dump(), $self->id()) || $self->t_abort(); |
288 |
|
|
|
289 |
|
|
} else { |
290 |
|
|
$log->error("Метод сохранения объектов задан неверно! Возможные значения - TOAST, SQLDUMP"); |
291 |
|
|
die; |
292 |
|
|
} |
293 |
|
|
|
294 |
|
|
return 1; |
295 |
|
|
} |
296 |
|
|
|
297 |
|
|
|
298 |
|
|
sub _create_extra_dump { |
299 |
|
|
my $self = shift; |
300 |
|
|
|
301 |
|
|
do { $log->error("Метод _create_extra_dump можно вызывать только у объектов, но не классов"); die } unless ref($self); |
302 |
|
|
|
303 |
|
|
my $class_table = $self->class_table; |
304 |
|
|
return undef unless ($class_table and $class_table->have_extra); |
305 |
|
|
|
306 |
|
|
my $extra_fields = []; |
307 |
|
|
my $virtual_fields = {}; |
308 |
217 |
ahitrov |
|
309 |
|
|
if ( $self->keeper->serialize_with eq 'json' ) { |
310 |
8 |
ahitrov@rambler.ru |
no strict 'refs'; |
311 |
|
|
#пропускаем virtual attributes |
312 |
|
|
#да я знаю что так писать нельзя но блин крута как смотрится |
313 |
|
|
$extra_fields = ${$self->{class}.'::extra_fields'}; |
314 |
|
|
$virtual_fields = ${$self->{class}.'::virtual_fields'}; |
315 |
|
|
#надо использовать все extra поля кроме тех что находятся в virtual_fields списке |
316 |
|
|
if ($state->db_encode_data) { |
317 |
217 |
ahitrov |
return Encode::decode('utf-8', $json_n->encode ({map { $_=> Encode::decode($state->db_encode_data, $self->{$_}, Encode::FB_HTMLCREF) } grep { !$virtual_fields->{$_} && (defined $self->{$_}) } @$extra_fields})); |
318 |
|
|
} else { |
319 |
|
|
my $content = Encode::decode('utf-8', $json_n->encode ({map { $_=>$self->{$_} } grep { !$virtual_fields->{$_} && (defined $self->{$_}) } @$extra_fields})); |
320 |
|
|
# warn "Store content is [$content]. UTF-8 Flag [".Encode::is_utf8($content)."]\n"; |
321 |
|
|
return $content; |
322 |
|
|
} |
323 |
|
|
} else { |
324 |
|
|
no strict 'refs'; |
325 |
|
|
#пропускаем virtual attributes |
326 |
|
|
#да я знаю что так писать нельзя но блин крута как смотрится |
327 |
|
|
$extra_fields = ${$self->{class}.'::extra_fields'}; |
328 |
|
|
$virtual_fields = ${$self->{class}.'::virtual_fields'}; |
329 |
|
|
#надо использовать все extra поля кроме тех что находятся в virtual_fields списке |
330 |
|
|
local $Data::Dumper::Indent = 0; |
331 |
|
|
if ($state->db_encode_data) { |
332 |
8 |
ahitrov@rambler.ru |
return Data::Dumper::Dumper({map { $_=> Encode::decode($state->db_encode_data, $self->{$_}, Encode::FB_HTMLCREF) } grep { !$virtual_fields->{$_} && (defined $self->{$_}) } @$extra_fields}); |
333 |
|
|
} else { |
334 |
|
|
return Data::Dumper::Dumper({map { $_=>$self->{$_} } grep { !$virtual_fields->{$_} && (defined $self->{$_}) } @$extra_fields}); |
335 |
|
|
} |
336 |
|
|
} |
337 |
|
|
} |
338 |
|
|
|
339 |
|
|
# ------------------------------------------------------------------------------------------- |
340 |
|
|
# Считывает внешние свойства объекта в зависимости от выбранного способа... |
341 |
|
|
# ------------------------------------------------------------------------------------------- |
342 |
|
|
sub restore_extras { |
343 |
|
|
my ($self, $row) = @_; |
344 |
|
|
|
345 |
|
|
do { $log->error("Метод restore_extras() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
346 |
|
|
|
347 |
|
|
my $extra_fields; |
348 |
|
|
{ |
349 |
|
|
no strict 'refs'; |
350 |
|
|
$extra_fields = ${$self->{class}.'::extra_fields'}; |
351 |
|
|
} |
352 |
|
|
|
353 |
|
|
if (@$extra_fields) { |
354 |
|
|
if (($Contenido::Globals::store_method eq 'toast') or ($Contenido::Globals::store_method eq 'sqldump')) { |
355 |
|
|
# -------------------------------------------------------------------------------------------- |
356 |
|
|
# Чтение из одного дампа в базе данных |
357 |
|
|
# -------------------------------------------------------------------------------------------- |
358 |
217 |
ahitrov |
my $dump_ = $self->keeper->serialize_with eq 'json' ? eval_json(\$row->[-1]) : eval_dump(\$row->[-1]); |
359 |
8 |
ahitrov@rambler.ru |
if ($dump_) { |
360 |
|
|
foreach (@$extra_fields) { |
361 |
|
|
$self->{$_} = $dump_->{$_}; |
362 |
|
|
} |
363 |
|
|
} |
364 |
|
|
} else { |
365 |
|
|
$log->error("Метод сохранения объектов задан неверно! Возможные значения - TOAST, SQLDUMP, SINGLE, DUMP"); |
366 |
|
|
die; |
367 |
|
|
} |
368 |
|
|
} |
369 |
|
|
} |
370 |
|
|
|
371 |
217 |
ahitrov |
sub _serialize { |
372 |
|
|
my $self = shift; |
373 |
|
|
my $data = shift; |
374 |
|
|
if ( $self->keeper->serialize_with eq 'json' ) { |
375 |
476 |
ahitrov |
return $json_n->encode(ref $data ? $data : {}); |
376 |
217 |
ahitrov |
} else { |
377 |
|
|
local $Data::Dumper::Indent = 0; |
378 |
|
|
return Data::Dumper::Dumper($data); |
379 |
|
|
} |
380 |
|
|
} |
381 |
|
|
|
382 |
8 |
ahitrov@rambler.ru |
# ---------------------------------------------------------------------------- |
383 |
|
|
# Выбирает хеш из перл-дампа по атрибуту |
384 |
|
|
# Пример: |
385 |
|
|
# my $pics_hashe = $doc->get_data('images'); |
386 |
|
|
# ---------------------------------------------------------------------------- |
387 |
|
|
sub get_data { |
388 |
|
|
my $self = shift; |
389 |
|
|
my $attr = shift; |
390 |
217 |
ahitrov |
my $encode = shift; |
391 |
|
|
my $data = $self->keeper->serialize_with eq 'json' ? ( $encode ? Data::Recursive::Encode->encode_utf8(eval_json(\$self->{$attr})) : eval_json(\$self->{$attr}) ) : eval_dump(\$self->{$attr}); |
392 |
8 |
ahitrov@rambler.ru |
return ($data || {}); |
393 |
|
|
} |
394 |
|
|
|
395 |
|
|
# ---------------------------------------------------------------------------- |
396 |
|
|
# Выбирает картинку из обьекта по ее атрибуту |
397 |
|
|
# Возвращает обьект типа Contenido::Image |
398 |
|
|
# |
399 |
|
|
# Пример: |
400 |
|
|
# my $pic = $doc->get_pic('top_image'); |
401 |
|
|
# |
402 |
|
|
# ---------------------------------------------------------------------------- |
403 |
|
|
sub get_pic { |
404 |
|
|
my $self = shift; |
405 |
|
|
my $attr = shift; |
406 |
|
|
|
407 |
|
|
Contenido::Image->new ( |
408 |
217 |
ahitrov |
img => $self->get_data($attr, 'encode'), |
409 |
8 |
ahitrov@rambler.ru |
attr => $attr, |
410 |
|
|
); |
411 |
|
|
} |
412 |
|
|
|
413 |
|
|
# ---------------------------------------------------------------------------- |
414 |
|
|
# Выбирает картинки из обьекта по атрибуту их хранилища |
415 |
|
|
# Возвращает массив обьектов типа Contenido::Image |
416 |
|
|
# |
417 |
|
|
# Пример: |
418 |
|
|
# my @pics = $doc->get_pics('images', { |
419 |
|
|
# order => 'reverse', # порядок сортировки по ключам ('reverse' ,'asis', по умолчанию - 'direct') |
420 |
|
|
# keys => [3..12, 1..2], # диапазон ключей |
421 |
|
|
# }); |
422 |
|
|
# |
423 |
|
|
# ---------------------------------------------------------------------------- |
424 |
|
|
sub get_pics { |
425 |
|
|
my $self = shift; |
426 |
|
|
my $attr = shift; |
427 |
|
|
my %args = ref $_[0] ? %{$_[0]} : @_; |
428 |
217 |
ahitrov |
my $pics = $self->get_data($attr, 'encode'); |
429 |
8 |
ahitrov@rambler.ru |
|
430 |
|
|
# Дефолты |
431 |
|
|
$args{order} ||= 'direct'; |
432 |
|
|
|
433 |
|
|
# выясняем ключики нужных нам картинок... |
434 |
|
|
my @keys = ref $args{keys} ne 'ARRAY' ? grep {s/\D+//, /^\d+$/} keys %{$pics} : @{$args{keys}}; |
435 |
|
|
|
436 |
|
|
my $prefix = 'image_'; # а надо бы, чтоб так: my $prefix = $attr.'_'; |
437 |
|
|
|
438 |
|
|
map { Contenido::Image->new ( |
439 |
|
|
img => $pics->{$prefix.$_}, |
440 |
|
|
attr => $prefix.$_, |
441 |
|
|
group => $attr, |
442 |
|
|
key => $_, |
443 |
|
|
)} sort { $args{order} eq 'asis' ? 1 : $args{order} ne 'reverse' ? $a <=> $b : $b <=> $a } @keys; |
444 |
|
|
} |
445 |
|
|
|
446 |
|
|
sub _get_sql { |
447 |
|
|
my ($self,%opts)=@_; |
448 |
|
|
|
449 |
|
|
#детект класса таблицы по которой работаем |
450 |
|
|
my $table = $self->_get_table(%opts); |
451 |
|
|
unless ($table) { |
452 |
|
|
$log->error("Не могу получить таблицу запроса"); |
453 |
|
|
return; |
454 |
|
|
} |
455 |
|
|
|
456 |
|
|
my ($query, $binds) = $table->generate_sql(%opts); |
457 |
|
|
my @binds = (); |
458 |
|
|
|
459 |
|
|
if ($state->db_encode_data) { |
460 |
|
|
foreach my $i (0..$#{$binds}) { |
461 |
|
|
$binds->[$i] = Encode::decode($state->db_encode_data, $binds->[$i], Encode::FB_HTMLCREF); |
462 |
|
|
} |
463 |
|
|
} |
464 |
|
|
|
465 |
|
|
return $query, $binds; |
466 |
|
|
} |
467 |
|
|
|
468 |
|
|
# Формат использования: |
469 |
|
|
# $document->store() |
470 |
|
|
# |
471 |
|
|
# Если $id задан то мы считаем, что этот объект в базе есть. Если |
472 |
|
|
# не задан, то мы считаем, что этого объекта в базе нет и создаем его. |
473 |
|
|
# ---------------------------------------------------------------------------- |
474 |
|
|
sub store { |
475 |
|
|
my $self = shift; |
476 |
|
|
do { $log->error("Метод store() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
477 |
|
|
|
478 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($self->keeper); |
479 |
|
|
|
480 |
|
|
return undef if ($self->keeper->state->readonly()); |
481 |
|
|
|
482 |
|
|
$self->keeper->t_connect() || do { $self->keeper->error(); return undef; }; |
483 |
|
|
$self->{status} ||= 0; # Значение статуса по умолчанию = 0 |
484 |
|
|
|
485 |
|
|
unless ($self->pre_store()) { |
486 |
|
|
$log->notice("pre_store call failed!"); |
487 |
|
|
return undef; |
488 |
|
|
} |
489 |
|
|
|
490 |
|
|
my (@fields, @values, @default_pairs, @default_fields, @default_values, @binary_fields); |
491 |
|
|
|
492 |
|
|
foreach ($self->required_properties()) { |
493 |
|
|
|
494 |
|
|
my $value = $self->{$_->{attr}}; |
495 |
217 |
ahitrov |
if ( exists $_->{db_field} && $_->{db_field} ) { |
496 |
|
|
$value = undef if (defined($value) and $value eq '') and (lc($_->{db_type}) eq 'float' or lc($_->{db_type} eq 'integer')); |
497 |
|
|
$value = undef if lc $_->{db_type} eq 'integer[]' && ref $value ne 'ARRAY'; |
498 |
|
|
$value = undef if lc $_->{db_type} eq 'integer_ref[]' && ref $value ne 'ARRAY'; |
499 |
|
|
} |
500 |
8 |
ahitrov@rambler.ru |
|
501 |
|
|
#пропустить readonly если у документа уже есть id |
502 |
|
|
if ($self->id() and $_->{readonly}) { |
503 |
|
|
|
504 |
|
|
#нет поля в базе у атрибута |
505 |
|
|
} elsif (!$_->{db_field}) { |
506 |
|
|
|
507 |
|
|
#установка default если оно есть и стоит авто или нет значения у поля |
508 |
|
|
} elsif (defined($_->{default}) and ($_->{auto} or !defined($value))) { |
509 |
|
|
push @default_fields, $_->{db_field}; |
510 |
|
|
push @default_values, $_->{default}; |
511 |
|
|
push @default_pairs, "$_->{db_field}=$_->{default}"; |
512 |
|
|
|
513 |
|
|
#пропустить auto без default |
514 |
|
|
} elsif ($_->{auto}) { |
515 |
|
|
|
516 |
|
|
#установка валидных полей |
517 |
|
|
} elsif (defined($value)) { |
518 |
|
|
push @fields, $_->{db_field}; |
519 |
|
|
if ($_->{db_type} eq 'integer[]') { |
520 |
|
|
push @values, '{'.join(',', grep { $_ } @$value).'}'; |
521 |
|
|
} elsif ($_->{db_type} eq 'integer_ref[]') { |
522 |
|
|
push @values, '{'.join(',', grep { $_ } @$value).'}'; |
523 |
|
|
} else { |
524 |
|
|
#some special work for bytea column type |
525 |
|
|
push @binary_fields, scalar(@fields) if ($_->{db_type} eq 'bytea'); |
526 |
|
|
if ($state->db_encode_data) { |
527 |
|
|
push @values, Encode::decode($state->db_encode_data, $value, Encode::FB_HTMLCREF); |
528 |
|
|
} else { |
529 |
|
|
push @values, $value; |
530 |
|
|
} |
531 |
|
|
} |
532 |
|
|
|
533 |
|
|
#undef to NULL or empty array |
534 |
|
|
} else { |
535 |
|
|
push @default_fields, $_->{db_field}; |
536 |
|
|
push @default_values, 'NULL'; |
537 |
|
|
push @default_pairs, "$_->{db_field}=NULL"; |
538 |
|
|
} |
539 |
|
|
} |
540 |
|
|
|
541 |
|
|
#если использется toast то загоняется за 1 sql запрос и extra тоже |
542 |
390 |
ahitrov |
if (($self->keeper->store_method() eq 'toast') and $self->class_table->have_extra and !$self->{__light}) { |
543 |
8 |
ahitrov@rambler.ru |
push @fields, 'data'; |
544 |
|
|
push @values, $self->_create_extra_dump(); |
545 |
|
|
} |
546 |
|
|
|
547 |
|
|
|
548 |
|
|
my $values_string = ''; |
549 |
|
|
my $mode = 'update'; |
550 |
|
|
if ($self->id()) { |
551 |
|
|
if (@fields) { |
552 |
|
|
$values_string = join(' = ?, ', @fields).' = ?'; |
553 |
|
|
$values_string .= ', '.join(', ', @default_pairs) if (@default_pairs); |
554 |
|
|
#нет обычных значений работаем только по @default_pairs |
555 |
|
|
} else { |
556 |
|
|
$values_string = join(', ', @default_pairs) if (@default_pairs); |
557 |
|
|
} |
558 |
|
|
my $sql='UPDATE '.$self->class_table->db_table.' SET '.$values_string." WHERE ".$self->class_table()->id_field()." = ?"; |
559 |
|
|
|
560 |
|
|
my $sth=$self->keeper->TSQL->prepare_cached($sql, {}, 1) || return $self->t_abort(); |
561 |
|
|
#settin special escape for bytea column type!!! |
562 |
|
|
foreach (@binary_fields) { |
563 |
|
|
$sth->bind_param($_, undef, {pg_type => DBD::Pg::PG_BYTEA}); |
564 |
|
|
} |
565 |
|
|
unless ($sth->execute(@values, $self->{id})) { |
566 |
|
|
$log->error("DBI execute error on $sql\n".Data::Dumper::Dumper(\@values)); |
567 |
|
|
$sth->finish(); |
568 |
|
|
return $self->t_abort(); |
569 |
|
|
} |
570 |
|
|
$sth->finish(); |
571 |
|
|
|
572 |
390 |
ahitrov |
if (($self->keeper->store_method() ne 'toast') and $self->class_table->have_extra and !$self->{__light}) { |
573 |
8 |
ahitrov@rambler.ru |
$self->store_extras(mode => $mode) || return $self->t_abort(); |
574 |
|
|
} |
575 |
|
|
|
576 |
|
|
} else { |
577 |
|
|
$mode = 'insert'; |
578 |
|
|
if (@fields) { |
579 |
|
|
$values_string = '?, 'x(scalar (@fields)-1).'?'; |
580 |
|
|
$values_string .= ', '.join(', ', @default_values) if (@default_values); |
581 |
|
|
#нет обычных значений работаем только по @default_pairs |
582 |
|
|
} else { |
583 |
|
|
$values_string = join(', ', @default_values) if (@default_values); |
584 |
|
|
} |
585 |
|
|
my $sql='INSERT INTO '.$self->class_table->db_table.' ('.join(', ', (@fields, @default_fields)).') VALUES ('.$values_string.')'; |
586 |
|
|
|
587 |
|
|
my $sth=$self->keeper->TSQL->prepare_cached($sql, {}, 1) || return $self->t_abort(); |
588 |
|
|
#settin special escape for bytea column type!!! |
589 |
|
|
foreach (@binary_fields) { |
590 |
|
|
$sth->bind_param($_, undef, {pg_type => DBD::Pg::PG_BYTEA}); |
591 |
|
|
} |
592 |
|
|
unless ($sth->execute(@values)) { |
593 |
|
|
$log->error("DBI execute error on $sql\n".Data::Dumper::Dumper(\@values)); |
594 |
|
|
$sth->finish(); |
595 |
|
|
return $self->t_abort(); |
596 |
|
|
} |
597 |
|
|
$sth->finish(); |
598 |
|
|
|
599 |
388 |
ahitrov |
my $id = $self->keeper->TSQL->selectrow_array("SELECT currval('".$self->class_table->db_id_sequence()."')"); |
600 |
8 |
ahitrov@rambler.ru |
$self->id($id); |
601 |
|
|
return $self->t_abort("Документу присвоен неверный идентификатор") if (! defined($self->{id}) || ($self->{id} <= 0)); |
602 |
|
|
|
603 |
390 |
ahitrov |
if (($self->keeper->store_method() ne 'toast') and $self->class_table->have_extra and !$self->{__light}) { |
604 |
8 |
ahitrov@rambler.ru |
$self->store_extras(mode => $mode) || return $self->t_abort(); |
605 |
|
|
} |
606 |
|
|
|
607 |
|
|
} |
608 |
|
|
|
609 |
|
|
$self->post_store(mode => $mode); |
610 |
|
|
|
611 |
|
|
$self->keeper->t_finish(); |
612 |
|
|
|
613 |
|
|
$self->post_finish_store(); |
614 |
|
|
|
615 |
|
|
$self->drop_cache($mode) if ($self->keeper->state()->memcached_enable()); |
616 |
|
|
|
617 |
|
|
return 1; |
618 |
|
|
} |
619 |
|
|
|
620 |
|
|
# ---------------------------------------------------------------------------- |
621 |
|
|
# Метод delete() для удаления объекта из базы данных. |
622 |
|
|
# |
623 |
|
|
# Формат использования: |
624 |
|
|
# $document->delete() |
625 |
|
|
# ---------------------------------------------------------------------------- |
626 |
|
|
sub delete { |
627 |
|
|
my $self = shift; |
628 |
|
|
my (%opts) = @_; |
629 |
|
|
do { $log->error("Метод delete() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
630 |
|
|
|
631 |
|
|
return undef if ($self->keeper->state->readonly()); |
632 |
|
|
|
633 |
|
|
unless ($self->pre_delete()) { |
634 |
|
|
$log->error("pre_delete call failed!"); |
635 |
|
|
return undef; |
636 |
|
|
} |
637 |
|
|
|
638 |
|
|
my $keeper = $self->keeper; |
639 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($keeper); |
640 |
|
|
|
641 |
390 |
ahitrov |
my $delete_attachments = exists $opts{attachments} ? $opts{attachments} : 1; |
642 |
|
|
if ( $delete_attachments ) { |
643 |
8 |
ahitrov@rambler.ru |
my @props = $self->structure(); |
644 |
|
|
if ( @props ) { |
645 |
347 |
ahitrov |
@props = grep { $_->{type} =~ /^(image|images|multimedia_new|multimedia_multi)$/ } @props; |
646 |
8 |
ahitrov@rambler.ru |
foreach my $prop ( @props ) { |
647 |
|
|
my $att = $self->get_image($prop->{attr}); |
648 |
|
|
if ( $prop->{type} eq 'image' ) { |
649 |
|
|
if ( ref $att && exists $att->{filename} && $att->{filename} ) { |
650 |
|
|
Contenido::File::remove( $att->{filename} ); |
651 |
|
|
} |
652 |
55 |
ahitrov@rambler.ru |
if ( exists $att->{mini} && ref $att->{mini} eq 'HASH' ) { |
653 |
|
|
Contenido::File::remove( $att->{mini}{filename} ) if exists $att->{mini}{filename}; |
654 |
|
|
foreach my $val ( values %{ $att->{mini} } ) { |
655 |
|
|
if ( ref $val && exists $val->{filename} && $val->{filename} && ($val->{filename} ne $att->{mini}{filename}) ) { |
656 |
|
|
Contenido::File::remove( $val->{filename} ); |
657 |
8 |
ahitrov@rambler.ru |
} |
658 |
|
|
} |
659 |
|
|
} |
660 |
|
|
|
661 |
|
|
} elsif ( $prop->{type} eq 'images' ) { |
662 |
|
|
for ( 1..100 ) { |
663 |
|
|
next unless exists $att->{"image_$_"}; |
664 |
|
|
my $img = $att->{"image_$_"}; |
665 |
|
|
if ( ref $img && exists $img->{filename} && $img->{filename} ) { |
666 |
|
|
Contenido::File::remove( $img->{filename} ); |
667 |
|
|
} |
668 |
55 |
ahitrov@rambler.ru |
if ( exists $img->{mini} && ref $img->{mini} eq 'HASH' ) { |
669 |
|
|
Contenido::File::remove( $img->{mini}{filename} ) if exists $img->{mini}{filename}; |
670 |
|
|
foreach my $val ( values %{ $img->{mini} } ) { |
671 |
|
|
if ( ref $val && exists $val->{filename} && $val->{filename} && ($val->{filename} ne $img->{mini}{filename}) ) { |
672 |
|
|
Contenido::File::remove( $val->{filename} ); |
673 |
8 |
ahitrov@rambler.ru |
} |
674 |
|
|
} |
675 |
|
|
} |
676 |
|
|
} |
677 |
440 |
ahitrov |
} elsif ( $prop->{type} eq 'multimedia_multi' ) { |
678 |
|
|
for ( 1..100 ) { |
679 |
|
|
next unless exists $att->{"file_$_"}; |
680 |
|
|
my $file = $att->{"file_$_"}; |
681 |
|
|
if ( ref $file && exists $file->{filename} && $file->{filename} ) { |
682 |
|
|
Contenido::File::remove( $file->{filename} ); |
683 |
|
|
} |
684 |
|
|
} |
685 |
8 |
ahitrov@rambler.ru |
} elsif ( $prop->{type} eq 'multimedia_new' ) { |
686 |
|
|
if ( ref $att && exists $att->{filename} && $att->{filename} ) { |
687 |
|
|
Contenido::File::remove( $att->{filename} ); |
688 |
|
|
} |
689 |
347 |
ahitrov |
} elsif ( $prop->{type} eq 'multimedia_multi' ) { |
690 |
|
|
for ( 1..100 ) { |
691 |
|
|
next unless exists $att->{"file_$_"}; |
692 |
|
|
my $file = $att->{"file_$_"}; |
693 |
|
|
if ( ref $file && exists $file->{filename} && $file->{filename} ) { |
694 |
|
|
Contenido::File::remove( $file->{filename} ); |
695 |
|
|
} |
696 |
|
|
} |
697 |
8 |
ahitrov@rambler.ru |
} |
698 |
|
|
} |
699 |
|
|
} |
700 |
|
|
} |
701 |
45 |
ahitrov@rambler.ru |
do { $log->warning("Вызов метода delete() без указания идентификатора для удаления"); return undef } |
702 |
55 |
ahitrov@rambler.ru |
unless ($self->{id}); |
703 |
8 |
ahitrov@rambler.ru |
$keeper->t_connect() || do { $keeper->error(); return undef; }; |
704 |
|
|
$keeper->TSQL->do("DELETE FROM ".$self->class_table->db_table." WHERE id = ?", {}, $self->id) || return $self->t_abort(); |
705 |
|
|
|
706 |
|
|
# Удаление связей этого документа с другими документами... |
707 |
|
|
my %document_links; |
708 |
|
|
if ( $keeper->state->{available_links} && ref $keeper->state->{available_links} eq 'ARRAY' ) { |
709 |
|
|
foreach my $classlink ( @{ $keeper->state->{available_links} } ) { |
710 |
|
|
my $sources = $classlink->available_sources; |
711 |
|
|
if ( ref $sources eq 'ARRAY' && @$sources ) { |
712 |
|
|
$document_links{$classlink->class_table->db_table}{source} = 1 if grep { $self->class eq $_ } @$sources; |
713 |
|
|
} |
714 |
|
|
my $dests = $classlink->available_destinations; |
715 |
|
|
if ( ref $dests eq 'ARRAY' && @$dests ) { |
716 |
45 |
ahitrov@rambler.ru |
$document_links{$classlink->class_table->db_table}{dest} = 1 if grep { $self->class eq $_ } @$dests; |
717 |
8 |
ahitrov@rambler.ru |
} |
718 |
|
|
} |
719 |
|
|
foreach my $tablename ( keys %document_links ) { |
720 |
|
|
my (@wheres, @values); |
721 |
|
|
if ( exists $document_links{$tablename}{source} ) { |
722 |
|
|
push @wheres, "(source_id = ? AND source_class = ?)"; |
723 |
|
|
push @values, ( $self->id, $self->class ); |
724 |
|
|
} |
725 |
|
|
if ( exists $document_links{$tablename}{dest} ) { |
726 |
|
|
push @wheres, "(dest_id = ? AND dest_class = ?)"; |
727 |
|
|
push @values, ( $self->id, $self->class ); |
728 |
|
|
} |
729 |
|
|
my $request = "DELETE FROM $tablename WHERE ".join (' OR ', @wheres); |
730 |
|
|
warn "DELETE LINKS. Request: [$request]\n" if $DEBUG; |
731 |
|
|
warn "Values: [".join(', ', @values)."]\n" if $DEBUG; |
732 |
|
|
$keeper->TSQL->do($request, {}, @values) || return $self->t_abort(); |
733 |
|
|
} |
734 |
|
|
} else { |
735 |
|
|
$keeper->TSQL->do("DELETE FROM links WHERE source_id = ? AND source_class = ? ", {}, $self->id, $self->class) || return $self->t_abort(); |
736 |
|
|
$keeper->TSQL->do("DELETE FROM links WHERE dest_id = ? AND dest_class = ? ", {}, $self->id, $self->class) || return $self->t_abort(); |
737 |
|
|
} |
738 |
|
|
$keeper->t_finish(); |
739 |
|
|
|
740 |
|
|
$self->post_delete(); |
741 |
|
|
|
742 |
|
|
$self->drop_cache('delete') if ($keeper->state()->memcached_enable()); |
743 |
|
|
|
744 |
|
|
return 1; |
745 |
|
|
} |
746 |
|
|
|
747 |
|
|
# ---------------------------------------------------------------------------- |
748 |
|
|
# Метод links() возвращает массив объектов типа Contenido::Link |
749 |
|
|
# |
750 |
|
|
# Формат использования: |
751 |
|
|
# $document->links([класс]) |
752 |
|
|
# ---------------------------------------------------------------------------- |
753 |
|
|
sub links { |
754 |
|
|
my ($self, $lclass, $direction, %opts) = @_; |
755 |
|
|
do { $log->error("Метод ->links() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
756 |
|
|
|
757 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($self->keeper); |
758 |
|
|
|
759 |
|
|
do { $log->warning("Вызов метода ->links() без указания идентификатора сообщения-источника"); return () } unless ($self->id() > 0); |
760 |
|
|
|
761 |
|
|
my $check = defined $direction ? 'dest_id' : 'source_id'; |
762 |
|
|
|
763 |
|
|
$opts{$check} = $self->id(); |
764 |
|
|
|
765 |
|
|
if (defined($lclass) && (length($lclass) > 0)) { |
766 |
55 |
ahitrov@rambler.ru |
$opts{class} = $lclass; |
767 |
8 |
ahitrov@rambler.ru |
} |
768 |
|
|
|
769 |
|
|
my @links = $self->keeper->get_links(%opts); |
770 |
|
|
|
771 |
|
|
$self->{links} = \@links; |
772 |
|
|
return @links; |
773 |
|
|
} |
774 |
|
|
|
775 |
|
|
|
776 |
|
|
sub linked_to { |
777 |
|
|
my ($self, $lclass) = @_; |
778 |
|
|
$self->links($lclass, 1); |
779 |
|
|
} |
780 |
|
|
|
781 |
|
|
|
782 |
|
|
# ---------------------------------------------------------------------------- |
783 |
|
|
# Установка связи. Должен быть обязательно задан класс... |
784 |
|
|
# В качестве source_id выступает идентификатор объекта, в качестве $dest_id - |
785 |
|
|
# заданный. |
786 |
|
|
# |
787 |
|
|
# Формат использования: |
788 |
|
|
# $document->set_link($lclass, $dest_id) |
789 |
|
|
# |
790 |
|
|
# Проверки не производится - у сообщения может быть несколько одинаковых |
791 |
|
|
# связей. |
792 |
|
|
# ---------------------------------------------------------------------------- |
793 |
|
|
sub set_link { |
794 |
|
|
my ($self, $lclass, $dest_id, $dest_class, @opts) = @_; |
795 |
|
|
do { $log->error("Метод ->set_link() вызван с неправильным кол-вом агрументов"); die } if @opts % 2; |
796 |
|
|
do { $log->error("Метод ->set_link() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
797 |
|
|
my %opts = @opts; |
798 |
|
|
|
799 |
|
|
return undef if ($self->keeper->state->readonly()); |
800 |
|
|
|
801 |
|
|
do { $log->warning("Вызов метода ->set_link() без указания идентификатора сообщения-источника"); return undef } unless ($self->id() > 0); |
802 |
|
|
do { $log->warning("Вызов метода ->set_link() без указания идентификатора сообщения-цели"); return undef } unless ($dest_id >= 0); |
803 |
115 |
ahitrov |
do { $log->warning("Вызов метода ->set_link() без указания класса связи"); } unless defined($lclass) && length($lclass); |
804 |
8 |
ahitrov@rambler.ru |
|
805 |
|
|
# Создаем объект связи... |
806 |
|
|
my $link = $lclass->new($self->keeper); |
807 |
|
|
|
808 |
|
|
$link->dest_id($dest_id); |
809 |
|
|
$link->dest_class($dest_class); |
810 |
|
|
|
811 |
|
|
$link->status(1); |
812 |
|
|
|
813 |
|
|
$link->source_id($self->id()); |
814 |
|
|
$link->source_class($self->class()); |
815 |
|
|
|
816 |
|
|
while (my ($k,$v) = each %opts) { |
817 |
55 |
ahitrov@rambler.ru |
$link->{$k} = $v; |
818 |
8 |
ahitrov@rambler.ru |
} |
819 |
|
|
|
820 |
|
|
if ($link->store()) |
821 |
|
|
{ |
822 |
55 |
ahitrov@rambler.ru |
return $link->id; |
823 |
8 |
ahitrov@rambler.ru |
} else { |
824 |
55 |
ahitrov@rambler.ru |
return undef; |
825 |
8 |
ahitrov@rambler.ru |
} |
826 |
|
|
} |
827 |
|
|
|
828 |
|
|
# ------------------------------------------------------------------- |
829 |
|
|
# Превращает объект в проблессенный хэш. |
830 |
|
|
# |
831 |
|
|
sub prepare_for_cache { |
832 |
|
|
my $self = shift; |
833 |
|
|
|
834 |
|
|
do { $log->error("Метод ->prepare_for_cache() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
835 |
|
|
|
836 |
|
|
my $hash = {}; |
837 |
|
|
|
838 |
|
|
foreach ( $self->structure() ) { |
839 |
55 |
ahitrov@rambler.ru |
$hash->{$_->{attr}} = $self->{$_->{attr}} if defined $self->{$_->{attr}}; |
840 |
8 |
ahitrov@rambler.ru |
} |
841 |
|
|
bless $hash, $self->class(); |
842 |
|
|
return $hash; |
843 |
|
|
} |
844 |
|
|
|
845 |
|
|
# ------------------------------------------------------------------- |
846 |
|
|
# Восстанавливает полноценный объект по проблессенному хэшу. |
847 |
|
|
# Хэш при этом превращается в полноценный объект. |
848 |
|
|
# ------------------------------------------------------------------- |
849 |
|
|
sub recover_from_cache { |
850 |
|
|
my $self = shift; |
851 |
|
|
my $keeper = shift; |
852 |
|
|
|
853 |
|
|
do { $log->error("Метод ->recover_from_cache() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
854 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($keeper); |
855 |
|
|
|
856 |
|
|
#не нужен тут bless очередной... 100% если уж попали в обьектный метод то он явно имеет класс нужный |
857 |
|
|
$self->init(); |
858 |
|
|
$self->keeper($keeper); |
859 |
|
|
|
860 |
|
|
return 1; |
861 |
|
|
} |
862 |
|
|
|
863 |
|
|
# ------------------------------------------------------------------- |
864 |
|
|
# Возвращает хэш: |
865 |
|
|
# {действие1 => [кэш1, кэш2, ...], действие2 => [кэш1, кэш2, ...], ...} |
866 |
|
|
# Т.е. для каждого действия задается список имен ключей в кэше, |
867 |
|
|
# которые надо удалить. |
868 |
|
|
# Дефолтные значени действий: insert, update, delete |
869 |
|
|
# Для более сложной логики работы этот метод должен быть переопределен |
870 |
|
|
# в классе самого объекта |
871 |
|
|
# |
872 |
|
|
sub dependencies { |
873 |
|
|
my ($self, $mode) = @_; |
874 |
|
|
|
875 |
|
|
my @keys = ($self->get_object_key,); |
876 |
|
|
my $object_unique_key = $self->get_object_unique_key; |
877 |
|
|
push @keys, $object_unique_key if defined $object_unique_key; |
878 |
|
|
|
879 |
|
|
return |
880 |
|
|
($mode eq 'delete') || ($mode eq 'insert') || ($mode eq 'update') |
881 |
|
|
? \@keys |
882 |
|
|
: []; |
883 |
|
|
} |
884 |
|
|
|
885 |
|
|
# ------------------------------------------------------------------- |
886 |
|
|
# Удаляет из кэша ключи, заданные при помощи dependencies(). |
887 |
|
|
# Пример вызова: |
888 |
|
|
# $group->drop_cache('update'); |
889 |
|
|
# |
890 |
|
|
sub drop_cache { |
891 |
|
|
my $self = shift; |
892 |
|
|
my $mode = shift; |
893 |
|
|
|
894 |
|
|
do { $log->error("Метод ->drop_cache() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
895 |
|
|
|
896 |
|
|
my $keeper = $self->keeper; |
897 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($keeper); |
898 |
|
|
|
899 |
|
|
my $dependencies = $self->dependencies($mode, @_); |
900 |
|
|
|
901 |
|
|
my @not_deleted = (); |
902 |
|
|
if ( defined($dependencies) && (ref($dependencies) eq 'ARRAY') ) { |
903 |
|
|
for (@$dependencies) { |
904 |
|
|
my $res = $self->keeper->MEMD ? $self->keeper->MEMD->delete($_) : undef; |
905 |
|
|
push @not_deleted, $_ unless $res; |
906 |
|
|
$keeper->MEMD->delete($_) if ($keeper->MEMD); |
907 |
|
|
} |
908 |
|
|
} |
909 |
|
|
return @not_deleted; |
910 |
|
|
} |
911 |
|
|
|
912 |
|
|
|
913 |
|
|
sub keeper { |
914 |
|
|
my $self = shift; |
915 |
|
|
my $project_keeper = shift; |
916 |
|
|
|
917 |
|
|
do { $log->error("Метод keeper() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
918 |
|
|
|
919 |
|
|
if ($project_keeper && ref $project_keeper) { |
920 |
|
|
$self->{keeper} = $project_keeper; |
921 |
|
|
return $project_keeper; |
922 |
|
|
} |
923 |
|
|
return $self->{keeper} && ref $self->{keeper} ? $self->{keeper} : $keeper; |
924 |
|
|
} |
925 |
|
|
|
926 |
|
|
#делаем затычку для init_from_db чтобы проинициализировать класс если надо |
927 |
|
|
sub init_from_db { |
928 |
|
|
my $self = shift; |
929 |
|
|
|
930 |
|
|
my $class = ref($self) || $self; |
931 |
|
|
|
932 |
|
|
#защита от бесконечной рекурсии если class_init не срабатывает |
933 |
|
|
if (defined($_[-1]) and ($_[-1] eq 'RECURSIVE CALL FLAG!')) { |
934 |
|
|
do { $log->error("$class cannot be initialized (->class_init dont work) (recursive call) !!!"); die }; |
935 |
|
|
} |
936 |
|
|
|
937 |
|
|
#если клас каким то странным образом все еще не проинициализирован то попробовать проинициализировать |
938 |
|
|
#только инициализация метода init_from_db допускает не ref на входе |
939 |
|
|
if ($class and $class->isa('Contenido::Object')) { |
940 |
|
|
no strict 'refs'; |
941 |
|
|
if (${$class.'::class_init_done'}) { |
942 |
|
|
do { $log->error("$class already initialized but DONT HAVE init_from_db method!!!"); die }; |
943 |
|
|
} else { |
944 |
|
|
if ($self->class_init()) { |
945 |
|
|
return $self->init_from_db(@_, 'RECURSIVE CALL FLAG!'); |
946 |
|
|
} else { |
947 |
|
|
do { $log->error("$class cannot be initialized (->class_init dont work) !!!"); die }; |
948 |
|
|
} |
949 |
|
|
} |
950 |
|
|
} else { |
951 |
|
|
do { $log->error("$class cannot be initialized (not Contenido::Object child class) !!!"); die }; |
952 |
|
|
} |
953 |
|
|
} |
954 |
|
|
|
955 |
|
|
# ---------------------------------------------------------------------------- |
956 |
|
|
# Это умный AUTOLOAD. Ловит методов для установки/чтения полей... |
957 |
|
|
# Версия 1.0 |
958 |
|
|
# теперь он герерирует необходимый метод доступу если надо |
959 |
|
|
# ---------------------------------------------------------------------------- |
960 |
|
|
sub AUTOLOAD { |
961 |
|
|
my $self = shift; |
962 |
|
|
my $attribute = $AUTOLOAD; |
963 |
|
|
|
964 |
|
|
$log->info("$self calling AUTOLOAD method: $attribute") if ($DEBUG_CORE); |
965 |
|
|
|
966 |
|
|
$attribute=~s/^.*:://; |
967 |
|
|
|
968 |
|
|
my $class = ref($self); |
969 |
|
|
unless ($class and $class->isa('Contenido::Object')) { |
970 |
|
|
|
971 |
|
|
my $mason_comp = ref($HTML::Mason::Commands::m) ? $HTML::Mason::Commands::m->current_comp() : undef; |
972 |
|
|
my $mason_file = ref($mason_comp) ? $mason_comp->path : undef; |
973 |
|
|
my ($package, $filename, $line) = caller; |
974 |
|
|
|
975 |
|
|
$log->warning("Wrong AUTOLOAD call with self='$self'/class='$class' and method '$attribute' called from '$package/$filename/$line' ".($mason_file ? "called from $mason_file" : '')."\n".Data::Dumper::Dumper($self)); |
976 |
|
|
if (wantarray) { return (); } else { return undef; } |
977 |
|
|
} |
978 |
|
|
|
979 |
|
|
#вообщето сюда было бы не плохо засунуть инициализацию класса если уж мы каким то хреном сюда попали для неинициализированного класса |
980 |
|
|
{ |
981 |
|
|
no strict 'refs'; |
982 |
|
|
unless (${$class.'::class_init_done'}) { |
983 |
|
|
my ($package, $filename, $line) = caller; |
984 |
|
|
$log->error("AUTOLOAD called method '$attribute' for not initialised class ($class) from '$package/$filename/$line'"); |
985 |
|
|
if (wantarray) { return (); } else { return undef; } |
986 |
|
|
} |
987 |
|
|
} |
988 |
|
|
|
989 |
|
|
if (! exists($self->{attributes}->{$attribute})) { |
990 |
|
|
|
991 |
|
|
my $mason_comp = ref($HTML::Mason::Commands::m) ? $HTML::Mason::Commands::m->current_comp() : undef; |
992 |
|
|
my $mason_file = ref($mason_comp) ? $mason_comp->path : undef; |
993 |
|
|
my ($package, $filename, $line) = caller; |
994 |
|
|
|
995 |
|
|
$log->error(ref($self)."): Вызов метода, для которого не существует обрабатываемого свойства: ->$attribute() called from $package/$filename/$line ".($mason_file ? "called from $mason_file" : '')."\n".Data::Dumper::Dumper($self)); |
996 |
|
|
if (wantarray) { return (); } else { return undef; } |
997 |
|
|
#special work with ARRAY types |
998 |
|
|
} elsif ($self->{attributes}->{$attribute} eq 'ARRAY') { |
999 |
|
|
my $funct = " |
1000 |
|
|
use Contenido::Globals; |
1001 |
|
|
my \$self = shift; |
1002 |
|
|
unless (ref(\$self->{$attribute}) eq 'ARRAY') { |
1003 |
|
|
my (\$package, \$filename, \$line) = caller; |
1004 |
|
|
\$log->error(\"Wrong structure in field $attribute called from \$package/\$filename/\$line \\n\".Data::Dumper::Dumper(\$self)) if (\$self->{$attribute});; |
1005 |
|
|
\$self->{$attribute} = []; |
1006 |
|
|
} |
1007 |
|
|
\$self->{$attribute} = [\@_] if (\@_); |
1008 |
|
|
return \@{\$self->{$attribute}};"; |
1009 |
|
|
|
1010 |
|
|
if (create_method($class, $attribute, $funct)) { |
1011 |
|
|
return $self->$attribute(@_); |
1012 |
|
|
} else { |
1013 |
|
|
$log->error("Cannot create method $attribute for class $self->{class}"); |
1014 |
|
|
#fallback to old autoload method if create method fail |
1015 |
|
|
unless (ref($self->{$attribute}) eq 'ARRAY') { |
1016 |
|
|
my ($package, $filename, $line) = caller; |
1017 |
|
|
$log->error("Wrong structure in field $attribute called from $package/$filename/$line \n".Data::Dumper::Dumper($self)); |
1018 |
|
|
$self->{$attribute} = []; |
1019 |
|
|
} |
1020 |
|
|
$self->{$attribute} = [@_] if (@_); |
1021 |
|
|
return @{$self->{$attribute}}; |
1022 |
|
|
} |
1023 |
|
|
#todo: добавить работу с images Нормальную когда она будет готова |
1024 |
|
|
} else { |
1025 |
|
|
#todo: валидация формата полей |
1026 |
|
|
my $funct = " |
1027 |
|
|
my \$self = shift; |
1028 |
|
|
\$self->{$attribute} = shift if (\@_); |
1029 |
|
|
return \$self->{$attribute};"; |
1030 |
|
|
|
1031 |
|
|
if (create_method($class, $attribute, $funct)) { |
1032 |
|
|
return $self->$attribute(@_); |
1033 |
|
|
} else { |
1034 |
|
|
$log->error("Cannot create method $attribute for class $self->{class}"); |
1035 |
|
|
#fallback to old autoload method if create method fail |
1036 |
|
|
$self->{$attribute} = shift if (@_); |
1037 |
|
|
return $self->{$attribute}; |
1038 |
|
|
} |
1039 |
|
|
} |
1040 |
|
|
} |
1041 |
|
|
|
1042 |
|
|
sub eval_dump { |
1043 |
|
|
no strict 'vars'; |
1044 |
|
|
return {} unless ${$_[0]}; |
1045 |
|
|
return eval ${$_[0]}; |
1046 |
|
|
} |
1047 |
|
|
|
1048 |
217 |
ahitrov |
sub eval_json { |
1049 |
|
|
return undef unless ${$_[0]}; |
1050 |
232 |
ahitrov |
my $str = ${$_[0]}; |
1051 |
|
|
my $chr = substr($str, 0, 1); return $str unless $chr eq '{' || $chr eq '['; |
1052 |
|
|
my $value = $json_u->decode( $str ); |
1053 |
217 |
ahitrov |
# map { $_ = Encode::encode(\'utf-8\', $_) unless ref $_; } values %$value; |
1054 |
|
|
return $value; |
1055 |
|
|
} |
1056 |
|
|
|
1057 |
8 |
ahitrov@rambler.ru |
sub create_method { |
1058 |
|
|
my ($class, $sub_name, $code) = @_; |
1059 |
|
|
|
1060 |
|
|
unless ($class and $sub_name and $code) { |
1061 |
|
|
$log->error("Wrong call create_method $class/$sub_name/$code"); |
1062 |
|
|
return 0; |
1063 |
|
|
} |
1064 |
|
|
|
1065 |
|
|
my $string = "package $class;\n\nsub $sub_name {\n$code\n}\n\n1;"; |
1066 |
|
|
eval $string; |
1067 |
|
|
|
1068 |
|
|
if ($@) { |
1069 |
|
|
$log->error("Cannot create method $sub_name for class $class because $@ (method code:\n$string\n)"); |
1070 |
|
|
return 0; |
1071 |
|
|
} else { |
1072 |
|
|
$log->info("Method '$sub_name' for class '$class' (method code:\n$string\n) created ok") if ($DEBUG_CORE); |
1073 |
|
|
return 1; |
1074 |
|
|
} |
1075 |
|
|
} |
1076 |
|
|
|
1077 |
|
|
######################################## ONLY FOR INTERNAL USE!!!! ################################################# |
1078 |
|
|
#todo добавить проверку что если классов список то проверить что у них 1 table а не 5 разных |
1079 |
|
|
sub _get_table { |
1080 |
|
|
my ($self, %opts) = @_; |
1081 |
|
|
|
1082 |
|
|
my $class_table; |
1083 |
|
|
|
1084 |
|
|
my $table=$opts{table}; |
1085 |
|
|
my $class=$opts{class} || ref $self || $self; |
1086 |
|
|
|
1087 |
|
|
#пришла таблица в %opts |
1088 |
|
|
if ($table and $table->can('new')) { |
1089 |
|
|
$class_table=$table; |
1090 |
|
|
#иначе пробуем по классу |
1091 |
|
|
} elsif ($class and !ref($class)) { |
1092 |
|
|
unless ($class->can('class_table')) { |
1093 |
|
|
$log->error("$class cannot class_table"); |
1094 |
|
|
return undef; |
1095 |
|
|
} |
1096 |
|
|
$class_table=$class->class_table(); |
1097 |
|
|
#иначе пробуем по первому классу в списке |
1098 |
|
|
} elsif ($class and ref($class) eq 'ARRAY' and @$class) { |
1099 |
|
|
unless ($class->[0]->can('class_table')) { |
1100 |
|
|
$log->error("$class->[0] cannot class_table"); |
1101 |
|
|
return undef; |
1102 |
|
|
} |
1103 |
|
|
$class_table=$class->[0]->class_table(); |
1104 |
|
|
#иначе умолчательную |
1105 |
|
|
} else { |
1106 |
|
|
$class_table='SQL::DocumentTable'; |
1107 |
|
|
} |
1108 |
|
|
|
1109 |
|
|
if ($class_table->can('new')) { |
1110 |
|
|
return $class_table->new(); |
1111 |
|
|
} else { |
1112 |
|
|
$log->error("$class_table cannot new!!!!"); |
1113 |
|
|
return undef; |
1114 |
|
|
} |
1115 |
|
|
} |
1116 |
|
|
|
1117 |
|
|
####################################################################################################################### |
1118 |
|
|
########## OLD CODE FOR COMPATIBILITY ################################################################################# |
1119 |
|
|
####################################################################################################################### |
1120 |
|
|
sub structure { |
1121 |
|
|
my $self = shift; |
1122 |
|
|
my $class = ref($self); |
1123 |
|
|
{ |
1124 |
|
|
no strict 'refs'; |
1125 |
|
|
return @${$class.'::structure'}; |
1126 |
|
|
} |
1127 |
|
|
} |
1128 |
|
|
|
1129 |
|
|
|
1130 |
|
|
# оставлена для обратной совместимости... |
1131 |
|
|
sub get_image { |
1132 |
|
|
my $self = shift; |
1133 |
217 |
ahitrov |
if ( $self->keeper->serialize_with eq 'json' ) { |
1134 |
|
|
return $self->get_data(shift, 'encode'); |
1135 |
|
|
} else { |
1136 |
|
|
return $self->get_data(shift); |
1137 |
|
|
} |
1138 |
8 |
ahitrov@rambler.ru |
} |
1139 |
|
|
|
1140 |
|
|
sub raw_restore { |
1141 |
|
|
my $self = shift; |
1142 |
|
|
do { $log->error("Метод restore() можно вызывать только у объектов, но не классов"); die } unless ref $self; |
1143 |
|
|
do { $log->warning("Вызов метода Contenido\:\:Object\:\:raw_restore() без указания идентификатора для чтения"); return undef } unless $self->id; |
1144 |
|
|
$self->restore_extras(); |
1145 |
|
|
} |
1146 |
|
|
|
1147 |
|
|
sub init { |
1148 |
|
|
my $self = shift; |
1149 |
|
|
my $class = ref($self) || $self; |
1150 |
|
|
$self->class_init(); |
1151 |
|
|
{ |
1152 |
|
|
no strict 'refs'; |
1153 |
|
|
$self->{attributes} = ${$class.'::attributes'}; |
1154 |
|
|
} |
1155 |
|
|
return 1; |
1156 |
|
|
} |
1157 |
|
|
|
1158 |
|
|
sub get_file_name { |
1159 |
|
|
my $self = shift; |
1160 |
|
|
|
1161 |
|
|
do { $log->error("Метод get_file_name можно вызывать только у объектов, но не классов"); die } unless ref $self; |
1162 |
|
|
|
1163 |
|
|
my @date; |
1164 |
|
|
my $time = time; |
1165 |
|
|
|
1166 |
|
|
if ($self->{"dtime"} and $self->{"dtime"} =~ /^(\d{4})-(\d{2})-(\d{2})/) { |
1167 |
|
|
@date = ($1, $2, $3); |
1168 |
|
|
} else { |
1169 |
|
|
@date = (localtime $time)[5, 4, 3]; $date[0] += 1900; $date[1] += 1; |
1170 |
|
|
} |
1171 |
|
|
|
1172 |
|
|
my $component_class = lc((reverse split "::", ref $self)[0]); |
1173 |
|
|
my $component_date = sprintf "%04d/%02d/%02d", @date; |
1174 |
|
|
my $component_time_rand = sprintf "%010d_%05d", $time, int rand 99999; |
1175 |
|
|
|
1176 |
|
|
return join "/", $component_class, $component_date, $component_time_rand; |
1177 |
|
|
} |
1178 |
|
|
|
1179 |
|
|
sub get { |
1180 |
|
|
my ( $self, %opts ) = @_; |
1181 |
|
|
my $class = ref $self || $self; |
1182 |
|
|
my $local_keeper = (ref($self) and ref($self->keeper)) ? $self->keeper : $keeper; |
1183 |
|
|
delete $opts{class}; |
1184 |
|
|
return $keeper->get_documents( class => $class, %opts ); |
1185 |
|
|
} |
1186 |
|
|
|
1187 |
|
|
sub contenido_is_available { |
1188 |
|
|
return 1; |
1189 |
|
|
} |
1190 |
|
|
|
1191 |
|
|
sub contenido_status_style { |
1192 |
|
|
return; |
1193 |
|
|
} |
1194 |
|
|
|
1195 |
|
|
sub memcached_expire { |
1196 |
|
|
return $_[0]->keeper->state->memcached_object_expire; |
1197 |
|
|
} |
1198 |
|
|
|
1199 |
113 |
ahitrov |
# ---------------------------------------------------------------------------- |
1200 |
125 |
ahitrov |
# Метод _store_image() сохраняет графику, привязанную к полю image или images |
1201 |
113 |
ahitrov |
# |
1202 |
|
|
# Формат использования: |
1203 |
125 |
ahitrov |
# $document->_store_image( INPUT, attr => 'fieldname' ) |
1204 |
365 |
ahitrov |
# $document->_store_image( INPUT, prop => $prophash ) |
1205 |
113 |
ahitrov |
# ---------------------------------------------------------------------------- |
1206 |
|
|
sub _store_image { |
1207 |
|
|
my $self = shift; |
1208 |
|
|
do { $log->error("Метод delete() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
1209 |
|
|
|
1210 |
|
|
my $input = shift; |
1211 |
|
|
my (%opts) = @_; |
1212 |
|
|
|
1213 |
365 |
ahitrov |
return Contenido::File::store_image( $input, object => $self, %opts ); |
1214 |
113 |
ahitrov |
} |
1215 |
|
|
|
1216 |
125 |
ahitrov |
# ---------------------------------------------------------------------------- |
1217 |
|
|
# Метод _delete_image() удаляет файлы, связанные с полем image или images. |
1218 |
|
|
# Вычищает все мини-копии |
1219 |
|
|
# |
1220 |
|
|
# Формат использования: |
1221 |
347 |
ahitrov |
# $document->_delete_image( $image_attr_structure ) |
1222 |
125 |
ahitrov |
# ---------------------------------------------------------------------------- |
1223 |
113 |
ahitrov |
sub _delete_image { |
1224 |
|
|
my $self = shift; |
1225 |
|
|
my $IMAGE = shift; |
1226 |
|
|
|
1227 |
|
|
return Contenido::File::remove_image( $IMAGE ); |
1228 |
|
|
} |
1229 |
|
|
|
1230 |
125 |
ahitrov |
# ---------------------------------------------------------------------------- |
1231 |
347 |
ahitrov |
# Метод _store_binary() сохраняет произвольный бинарный файл, привязанную к полю multimedia_multi или multimedia_new |
1232 |
125 |
ahitrov |
# |
1233 |
|
|
# Формат использования: |
1234 |
|
|
# $document->_store_binary( INPUT, attr => 'fieldname' ) |
1235 |
|
|
# ---------------------------------------------------------------------------- |
1236 |
|
|
sub _store_binary { |
1237 |
|
|
my $self = shift; |
1238 |
|
|
do { $log->error("Метод delete() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
1239 |
|
|
|
1240 |
|
|
my $input = shift; |
1241 |
|
|
my (%opts) = @_; |
1242 |
|
|
|
1243 |
|
|
return Contenido::File::store_binary( $input, object => $self, attr => $opts{attr} ); |
1244 |
|
|
} |
1245 |
|
|
|
1246 |
|
|
# ---------------------------------------------------------------------------- |
1247 |
|
|
# Метод _delete_binary() удаляет файлы, связанные с полем multimedia или multimedia_new. |
1248 |
|
|
# Не пытается искать мини-копии |
1249 |
|
|
# |
1250 |
|
|
# Формат использования: |
1251 |
|
|
# $document->_delete_binary( $binary_attr_structure ) |
1252 |
|
|
# ---------------------------------------------------------------------------- |
1253 |
|
|
sub _delete_binary { |
1254 |
|
|
my $self = shift; |
1255 |
|
|
my $BINARY = shift; |
1256 |
|
|
|
1257 |
|
|
return Contenido::File::remove_binary( $BINARY ); |
1258 |
|
|
} |
1259 |
|
|
|
1260 |
8 |
ahitrov@rambler.ru |
1; |
1261 |
|
|
|