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 |
|
|
return $content; |
321 |
|
|
} |
322 |
|
|
} else { |
323 |
|
|
no strict 'refs'; |
324 |
|
|
#пропускаем virtual attributes |
325 |
|
|
#да я знаю что так писать нельзя но блин крута как смотрится |
326 |
|
|
$extra_fields = ${$self->{class}.'::extra_fields'}; |
327 |
|
|
$virtual_fields = ${$self->{class}.'::virtual_fields'}; |
328 |
|
|
#надо использовать все extra поля кроме тех что находятся в virtual_fields списке |
329 |
|
|
local $Data::Dumper::Indent = 0; |
330 |
|
|
if ($state->db_encode_data) { |
331 |
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}); |
332 |
|
|
} else { |
333 |
|
|
return Data::Dumper::Dumper({map { $_=>$self->{$_} } grep { !$virtual_fields->{$_} && (defined $self->{$_}) } @$extra_fields}); |
334 |
|
|
} |
335 |
|
|
} |
336 |
|
|
} |
337 |
|
|
|
338 |
|
|
# ------------------------------------------------------------------------------------------- |
339 |
|
|
# Считывает внешние свойства объекта в зависимости от выбранного способа... |
340 |
|
|
# ------------------------------------------------------------------------------------------- |
341 |
|
|
sub restore_extras { |
342 |
|
|
my ($self, $row) = @_; |
343 |
|
|
|
344 |
|
|
do { $log->error("Метод restore_extras() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
345 |
|
|
|
346 |
|
|
my $extra_fields; |
347 |
|
|
{ |
348 |
|
|
no strict 'refs'; |
349 |
|
|
$extra_fields = ${$self->{class}.'::extra_fields'}; |
350 |
|
|
} |
351 |
|
|
|
352 |
|
|
if (@$extra_fields) { |
353 |
|
|
if (($Contenido::Globals::store_method eq 'toast') or ($Contenido::Globals::store_method eq 'sqldump')) { |
354 |
|
|
# -------------------------------------------------------------------------------------------- |
355 |
|
|
# Чтение из одного дампа в базе данных |
356 |
|
|
# -------------------------------------------------------------------------------------------- |
357 |
217 |
ahitrov |
my $dump_ = $self->keeper->serialize_with eq 'json' ? eval_json(\$row->[-1]) : eval_dump(\$row->[-1]); |
358 |
8 |
ahitrov@rambler.ru |
if ($dump_) { |
359 |
|
|
foreach (@$extra_fields) { |
360 |
|
|
$self->{$_} = $dump_->{$_}; |
361 |
|
|
} |
362 |
|
|
} |
363 |
|
|
} else { |
364 |
|
|
$log->error("Метод сохранения объектов задан неверно! Возможные значения - TOAST, SQLDUMP, SINGLE, DUMP"); |
365 |
|
|
die; |
366 |
|
|
} |
367 |
|
|
} |
368 |
|
|
} |
369 |
|
|
|
370 |
217 |
ahitrov |
sub _serialize { |
371 |
|
|
my $self = shift; |
372 |
|
|
my $data = shift; |
373 |
|
|
if ( $self->keeper->serialize_with eq 'json' ) { |
374 |
476 |
ahitrov |
return $json_n->encode(ref $data ? $data : {}); |
375 |
217 |
ahitrov |
} else { |
376 |
|
|
local $Data::Dumper::Indent = 0; |
377 |
|
|
return Data::Dumper::Dumper($data); |
378 |
|
|
} |
379 |
|
|
} |
380 |
|
|
|
381 |
8 |
ahitrov@rambler.ru |
# ---------------------------------------------------------------------------- |
382 |
|
|
# Выбирает хеш из перл-дампа по атрибуту |
383 |
|
|
# Пример: |
384 |
|
|
# my $pics_hashe = $doc->get_data('images'); |
385 |
|
|
# ---------------------------------------------------------------------------- |
386 |
|
|
sub get_data { |
387 |
|
|
my $self = shift; |
388 |
|
|
my $attr = shift; |
389 |
217 |
ahitrov |
my $encode = shift; |
390 |
|
|
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}); |
391 |
8 |
ahitrov@rambler.ru |
return ($data || {}); |
392 |
|
|
} |
393 |
|
|
|
394 |
|
|
# ---------------------------------------------------------------------------- |
395 |
|
|
# Выбирает картинку из обьекта по ее атрибуту |
396 |
|
|
# Возвращает обьект типа Contenido::Image |
397 |
|
|
# |
398 |
|
|
# Пример: |
399 |
|
|
# my $pic = $doc->get_pic('top_image'); |
400 |
|
|
# |
401 |
|
|
# ---------------------------------------------------------------------------- |
402 |
|
|
sub get_pic { |
403 |
|
|
my $self = shift; |
404 |
|
|
my $attr = shift; |
405 |
|
|
|
406 |
|
|
Contenido::Image->new ( |
407 |
217 |
ahitrov |
img => $self->get_data($attr, 'encode'), |
408 |
8 |
ahitrov@rambler.ru |
attr => $attr, |
409 |
|
|
); |
410 |
|
|
} |
411 |
|
|
|
412 |
|
|
# ---------------------------------------------------------------------------- |
413 |
|
|
# Выбирает картинки из обьекта по атрибуту их хранилища |
414 |
|
|
# Возвращает массив обьектов типа Contenido::Image |
415 |
|
|
# |
416 |
|
|
# Пример: |
417 |
|
|
# my @pics = $doc->get_pics('images', { |
418 |
|
|
# order => 'reverse', # порядок сортировки по ключам ('reverse' ,'asis', по умолчанию - 'direct') |
419 |
|
|
# keys => [3..12, 1..2], # диапазон ключей |
420 |
|
|
# }); |
421 |
|
|
# |
422 |
|
|
# ---------------------------------------------------------------------------- |
423 |
|
|
sub get_pics { |
424 |
|
|
my $self = shift; |
425 |
|
|
my $attr = shift; |
426 |
|
|
my %args = ref $_[0] ? %{$_[0]} : @_; |
427 |
217 |
ahitrov |
my $pics = $self->get_data($attr, 'encode'); |
428 |
8 |
ahitrov@rambler.ru |
|
429 |
|
|
# Дефолты |
430 |
|
|
$args{order} ||= 'direct'; |
431 |
|
|
|
432 |
|
|
# выясняем ключики нужных нам картинок... |
433 |
|
|
my @keys = ref $args{keys} ne 'ARRAY' ? grep {s/\D+//, /^\d+$/} keys %{$pics} : @{$args{keys}}; |
434 |
|
|
|
435 |
|
|
my $prefix = 'image_'; # а надо бы, чтоб так: my $prefix = $attr.'_'; |
436 |
|
|
|
437 |
|
|
map { Contenido::Image->new ( |
438 |
|
|
img => $pics->{$prefix.$_}, |
439 |
|
|
attr => $prefix.$_, |
440 |
|
|
group => $attr, |
441 |
|
|
key => $_, |
442 |
|
|
)} sort { $args{order} eq 'asis' ? 1 : $args{order} ne 'reverse' ? $a <=> $b : $b <=> $a } @keys; |
443 |
|
|
} |
444 |
|
|
|
445 |
|
|
sub _get_sql { |
446 |
|
|
my ($self,%opts)=@_; |
447 |
|
|
|
448 |
|
|
#детект класса таблицы по которой работаем |
449 |
|
|
my $table = $self->_get_table(%opts); |
450 |
|
|
unless ($table) { |
451 |
|
|
$log->error("Не могу получить таблицу запроса"); |
452 |
|
|
return; |
453 |
|
|
} |
454 |
|
|
|
455 |
|
|
my ($query, $binds) = $table->generate_sql(%opts); |
456 |
|
|
my @binds = (); |
457 |
|
|
|
458 |
|
|
if ($state->db_encode_data) { |
459 |
|
|
foreach my $i (0..$#{$binds}) { |
460 |
582 |
ahitrov |
if ( ref $binds->[$i] ) { |
461 |
|
|
$binds->[$i] = Data::Recursive::Encode->decode($state->db_encode_data, $binds->[$i], Encode::FB_HTMLCREF); |
462 |
|
|
} else { |
463 |
|
|
$binds->[$i] = Encode::decode($state->db_encode_data, $binds->[$i], Encode::FB_HTMLCREF); |
464 |
|
|
} |
465 |
8 |
ahitrov@rambler.ru |
} |
466 |
567 |
ahitrov |
} elsif ( $DBD::Pg::VERSION >= '3' ) { |
467 |
|
|
foreach my $i (0..$#{$binds}) { |
468 |
582 |
ahitrov |
if ( ref $binds->[$i] ) { |
469 |
|
|
$binds->[$i] = Data::Recursive::Encode->decode_utf8($binds->[$i]); |
470 |
|
|
} else { |
471 |
|
|
$binds->[$i] = Encode::decode('utf-8', $binds->[$i], Encode::FB_HTMLCREF); |
472 |
|
|
} |
473 |
567 |
ahitrov |
} |
474 |
8 |
ahitrov@rambler.ru |
} |
475 |
|
|
|
476 |
|
|
return $query, $binds; |
477 |
|
|
} |
478 |
|
|
|
479 |
|
|
# Формат использования: |
480 |
|
|
# $document->store() |
481 |
|
|
# |
482 |
|
|
# Если $id задан то мы считаем, что этот объект в базе есть. Если |
483 |
|
|
# не задан, то мы считаем, что этого объекта в базе нет и создаем его. |
484 |
|
|
# ---------------------------------------------------------------------------- |
485 |
|
|
sub store { |
486 |
|
|
my $self = shift; |
487 |
|
|
do { $log->error("Метод store() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
488 |
|
|
|
489 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($self->keeper); |
490 |
|
|
|
491 |
|
|
return undef if ($self->keeper->state->readonly()); |
492 |
|
|
|
493 |
|
|
$self->keeper->t_connect() || do { $self->keeper->error(); return undef; }; |
494 |
|
|
$self->{status} ||= 0; # Значение статуса по умолчанию = 0 |
495 |
|
|
|
496 |
|
|
unless ($self->pre_store()) { |
497 |
|
|
$log->notice("pre_store call failed!"); |
498 |
|
|
return undef; |
499 |
|
|
} |
500 |
|
|
|
501 |
|
|
my (@fields, @values, @default_pairs, @default_fields, @default_values, @binary_fields); |
502 |
|
|
|
503 |
|
|
foreach ($self->required_properties()) { |
504 |
|
|
|
505 |
|
|
my $value = $self->{$_->{attr}}; |
506 |
217 |
ahitrov |
if ( exists $_->{db_field} && $_->{db_field} ) { |
507 |
|
|
$value = undef if (defined($value) and $value eq '') and (lc($_->{db_type}) eq 'float' or lc($_->{db_type} eq 'integer')); |
508 |
|
|
$value = undef if lc $_->{db_type} eq 'integer[]' && ref $value ne 'ARRAY'; |
509 |
|
|
$value = undef if lc $_->{db_type} eq 'integer_ref[]' && ref $value ne 'ARRAY'; |
510 |
|
|
} |
511 |
8 |
ahitrov@rambler.ru |
|
512 |
|
|
#пропустить readonly если у документа уже есть id |
513 |
|
|
if ($self->id() and $_->{readonly}) { |
514 |
|
|
|
515 |
|
|
#нет поля в базе у атрибута |
516 |
|
|
} elsif (!$_->{db_field}) { |
517 |
|
|
|
518 |
|
|
#установка default если оно есть и стоит авто или нет значения у поля |
519 |
|
|
} elsif (defined($_->{default}) and ($_->{auto} or !defined($value))) { |
520 |
|
|
push @default_fields, $_->{db_field}; |
521 |
|
|
push @default_values, $_->{default}; |
522 |
|
|
push @default_pairs, "$_->{db_field}=$_->{default}"; |
523 |
|
|
|
524 |
|
|
#пропустить auto без default |
525 |
|
|
} elsif ($_->{auto}) { |
526 |
|
|
|
527 |
|
|
#установка валидных полей |
528 |
|
|
} elsif (defined($value)) { |
529 |
|
|
push @fields, $_->{db_field}; |
530 |
|
|
if ($_->{db_type} eq 'integer[]') { |
531 |
|
|
push @values, '{'.join(',', grep { $_ } @$value).'}'; |
532 |
|
|
} elsif ($_->{db_type} eq 'integer_ref[]') { |
533 |
|
|
push @values, '{'.join(',', grep { $_ } @$value).'}'; |
534 |
|
|
} else { |
535 |
|
|
#some special work for bytea column type |
536 |
|
|
push @binary_fields, scalar(@fields) if ($_->{db_type} eq 'bytea'); |
537 |
|
|
if ($state->db_encode_data) { |
538 |
|
|
push @values, Encode::decode($state->db_encode_data, $value, Encode::FB_HTMLCREF); |
539 |
567 |
ahitrov |
} elsif ($DBD::Pg::VERSION >= '3') { |
540 |
|
|
push @values, Encode::decode('utf-8', $value, Encode::FB_HTMLCREF); |
541 |
8 |
ahitrov@rambler.ru |
} else { |
542 |
|
|
push @values, $value; |
543 |
|
|
} |
544 |
|
|
} |
545 |
|
|
|
546 |
|
|
#undef to NULL or empty array |
547 |
|
|
} else { |
548 |
|
|
push @default_fields, $_->{db_field}; |
549 |
|
|
push @default_values, 'NULL'; |
550 |
|
|
push @default_pairs, "$_->{db_field}=NULL"; |
551 |
|
|
} |
552 |
|
|
} |
553 |
|
|
|
554 |
|
|
#если использется toast то загоняется за 1 sql запрос и extra тоже |
555 |
390 |
ahitrov |
if (($self->keeper->store_method() eq 'toast') and $self->class_table->have_extra and !$self->{__light}) { |
556 |
8 |
ahitrov@rambler.ru |
push @fields, 'data'; |
557 |
|
|
push @values, $self->_create_extra_dump(); |
558 |
|
|
} |
559 |
|
|
|
560 |
|
|
|
561 |
|
|
my $values_string = ''; |
562 |
|
|
my $mode = 'update'; |
563 |
|
|
if ($self->id()) { |
564 |
|
|
if (@fields) { |
565 |
|
|
$values_string = join(' = ?, ', @fields).' = ?'; |
566 |
|
|
$values_string .= ', '.join(', ', @default_pairs) if (@default_pairs); |
567 |
|
|
#нет обычных значений работаем только по @default_pairs |
568 |
|
|
} else { |
569 |
|
|
$values_string = join(', ', @default_pairs) if (@default_pairs); |
570 |
|
|
} |
571 |
|
|
my $sql='UPDATE '.$self->class_table->db_table.' SET '.$values_string." WHERE ".$self->class_table()->id_field()." = ?"; |
572 |
|
|
|
573 |
|
|
my $sth=$self->keeper->TSQL->prepare_cached($sql, {}, 1) || return $self->t_abort(); |
574 |
|
|
#settin special escape for bytea column type!!! |
575 |
|
|
foreach (@binary_fields) { |
576 |
|
|
$sth->bind_param($_, undef, {pg_type => DBD::Pg::PG_BYTEA}); |
577 |
|
|
} |
578 |
|
|
unless ($sth->execute(@values, $self->{id})) { |
579 |
567 |
ahitrov |
$log->error("DBI execute error on $sql\n".Data::Dumper::Dumper( $DBD::Pg::VERSION >= '3' ? Data::Recursive::Encode->encode_utf8( \@values ) : \@values )); |
580 |
8 |
ahitrov@rambler.ru |
$sth->finish(); |
581 |
|
|
return $self->t_abort(); |
582 |
|
|
} |
583 |
|
|
$sth->finish(); |
584 |
|
|
|
585 |
390 |
ahitrov |
if (($self->keeper->store_method() ne 'toast') and $self->class_table->have_extra and !$self->{__light}) { |
586 |
8 |
ahitrov@rambler.ru |
$self->store_extras(mode => $mode) || return $self->t_abort(); |
587 |
|
|
} |
588 |
|
|
|
589 |
|
|
} else { |
590 |
|
|
$mode = 'insert'; |
591 |
|
|
if (@fields) { |
592 |
|
|
$values_string = '?, 'x(scalar (@fields)-1).'?'; |
593 |
|
|
$values_string .= ', '.join(', ', @default_values) if (@default_values); |
594 |
|
|
#нет обычных значений работаем только по @default_pairs |
595 |
|
|
} else { |
596 |
|
|
$values_string = join(', ', @default_values) if (@default_values); |
597 |
|
|
} |
598 |
|
|
my $sql='INSERT INTO '.$self->class_table->db_table.' ('.join(', ', (@fields, @default_fields)).') VALUES ('.$values_string.')'; |
599 |
|
|
|
600 |
|
|
my $sth=$self->keeper->TSQL->prepare_cached($sql, {}, 1) || return $self->t_abort(); |
601 |
|
|
#settin special escape for bytea column type!!! |
602 |
|
|
foreach (@binary_fields) { |
603 |
|
|
$sth->bind_param($_, undef, {pg_type => DBD::Pg::PG_BYTEA}); |
604 |
|
|
} |
605 |
|
|
unless ($sth->execute(@values)) { |
606 |
567 |
ahitrov |
$log->error("DBI execute error on $sql\n".Data::Dumper::Dumper( $DBD::Pg::VERSION >= '3' ? Data::Recursive::Encode->encode_utf8( \@values ) : \@values )); |
607 |
8 |
ahitrov@rambler.ru |
$sth->finish(); |
608 |
|
|
return $self->t_abort(); |
609 |
|
|
} |
610 |
|
|
$sth->finish(); |
611 |
|
|
|
612 |
388 |
ahitrov |
my $id = $self->keeper->TSQL->selectrow_array("SELECT currval('".$self->class_table->db_id_sequence()."')"); |
613 |
8 |
ahitrov@rambler.ru |
$self->id($id); |
614 |
|
|
return $self->t_abort("Документу присвоен неверный идентификатор") if (! defined($self->{id}) || ($self->{id} <= 0)); |
615 |
|
|
|
616 |
390 |
ahitrov |
if (($self->keeper->store_method() ne 'toast') and $self->class_table->have_extra and !$self->{__light}) { |
617 |
8 |
ahitrov@rambler.ru |
$self->store_extras(mode => $mode) || return $self->t_abort(); |
618 |
|
|
} |
619 |
|
|
|
620 |
|
|
} |
621 |
|
|
|
622 |
|
|
$self->post_store(mode => $mode); |
623 |
|
|
|
624 |
|
|
$self->keeper->t_finish(); |
625 |
|
|
|
626 |
|
|
$self->post_finish_store(); |
627 |
|
|
|
628 |
|
|
$self->drop_cache($mode) if ($self->keeper->state()->memcached_enable()); |
629 |
|
|
|
630 |
|
|
return 1; |
631 |
|
|
} |
632 |
|
|
|
633 |
|
|
# ---------------------------------------------------------------------------- |
634 |
|
|
# Метод delete() для удаления объекта из базы данных. |
635 |
|
|
# |
636 |
|
|
# Формат использования: |
637 |
|
|
# $document->delete() |
638 |
|
|
# ---------------------------------------------------------------------------- |
639 |
|
|
sub delete { |
640 |
|
|
my $self = shift; |
641 |
|
|
my (%opts) = @_; |
642 |
|
|
do { $log->error("Метод delete() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
643 |
|
|
|
644 |
|
|
return undef if ($self->keeper->state->readonly()); |
645 |
|
|
|
646 |
|
|
unless ($self->pre_delete()) { |
647 |
|
|
$log->error("pre_delete call failed!"); |
648 |
|
|
return undef; |
649 |
|
|
} |
650 |
|
|
|
651 |
|
|
my $keeper = $self->keeper; |
652 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($keeper); |
653 |
|
|
|
654 |
390 |
ahitrov |
my $delete_attachments = exists $opts{attachments} ? $opts{attachments} : 1; |
655 |
|
|
if ( $delete_attachments ) { |
656 |
8 |
ahitrov@rambler.ru |
my @props = $self->structure(); |
657 |
|
|
if ( @props ) { |
658 |
347 |
ahitrov |
@props = grep { $_->{type} =~ /^(image|images|multimedia_new|multimedia_multi)$/ } @props; |
659 |
8 |
ahitrov@rambler.ru |
foreach my $prop ( @props ) { |
660 |
|
|
my $att = $self->get_image($prop->{attr}); |
661 |
|
|
if ( $prop->{type} eq 'image' ) { |
662 |
|
|
if ( ref $att && exists $att->{filename} && $att->{filename} ) { |
663 |
|
|
Contenido::File::remove( $att->{filename} ); |
664 |
|
|
} |
665 |
55 |
ahitrov@rambler.ru |
if ( exists $att->{mini} && ref $att->{mini} eq 'HASH' ) { |
666 |
|
|
Contenido::File::remove( $att->{mini}{filename} ) if exists $att->{mini}{filename}; |
667 |
|
|
foreach my $val ( values %{ $att->{mini} } ) { |
668 |
|
|
if ( ref $val && exists $val->{filename} && $val->{filename} && ($val->{filename} ne $att->{mini}{filename}) ) { |
669 |
|
|
Contenido::File::remove( $val->{filename} ); |
670 |
8 |
ahitrov@rambler.ru |
} |
671 |
|
|
} |
672 |
|
|
} |
673 |
|
|
|
674 |
|
|
} elsif ( $prop->{type} eq 'images' ) { |
675 |
|
|
for ( 1..100 ) { |
676 |
|
|
next unless exists $att->{"image_$_"}; |
677 |
|
|
my $img = $att->{"image_$_"}; |
678 |
|
|
if ( ref $img && exists $img->{filename} && $img->{filename} ) { |
679 |
|
|
Contenido::File::remove( $img->{filename} ); |
680 |
|
|
} |
681 |
55 |
ahitrov@rambler.ru |
if ( exists $img->{mini} && ref $img->{mini} eq 'HASH' ) { |
682 |
|
|
Contenido::File::remove( $img->{mini}{filename} ) if exists $img->{mini}{filename}; |
683 |
|
|
foreach my $val ( values %{ $img->{mini} } ) { |
684 |
|
|
if ( ref $val && exists $val->{filename} && $val->{filename} && ($val->{filename} ne $img->{mini}{filename}) ) { |
685 |
|
|
Contenido::File::remove( $val->{filename} ); |
686 |
8 |
ahitrov@rambler.ru |
} |
687 |
|
|
} |
688 |
|
|
} |
689 |
|
|
} |
690 |
|
|
} elsif ( $prop->{type} eq 'multimedia_new' ) { |
691 |
|
|
if ( ref $att && exists $att->{filename} && $att->{filename} ) { |
692 |
|
|
Contenido::File::remove( $att->{filename} ); |
693 |
|
|
} |
694 |
347 |
ahitrov |
} elsif ( $prop->{type} eq 'multimedia_multi' ) { |
695 |
|
|
for ( 1..100 ) { |
696 |
|
|
next unless exists $att->{"file_$_"}; |
697 |
|
|
my $file = $att->{"file_$_"}; |
698 |
|
|
if ( ref $file && exists $file->{filename} && $file->{filename} ) { |
699 |
|
|
Contenido::File::remove( $file->{filename} ); |
700 |
|
|
} |
701 |
|
|
} |
702 |
8 |
ahitrov@rambler.ru |
} |
703 |
|
|
} |
704 |
|
|
} |
705 |
|
|
} |
706 |
45 |
ahitrov@rambler.ru |
do { $log->warning("Вызов метода delete() без указания идентификатора для удаления"); return undef } |
707 |
55 |
ahitrov@rambler.ru |
unless ($self->{id}); |
708 |
8 |
ahitrov@rambler.ru |
$keeper->t_connect() || do { $keeper->error(); return undef; }; |
709 |
|
|
$keeper->TSQL->do("DELETE FROM ".$self->class_table->db_table." WHERE id = ?", {}, $self->id) || return $self->t_abort(); |
710 |
|
|
|
711 |
|
|
# Удаление связей этого документа с другими документами... |
712 |
|
|
my %document_links; |
713 |
|
|
if ( $keeper->state->{available_links} && ref $keeper->state->{available_links} eq 'ARRAY' ) { |
714 |
|
|
foreach my $classlink ( @{ $keeper->state->{available_links} } ) { |
715 |
|
|
my $sources = $classlink->available_sources; |
716 |
|
|
if ( ref $sources eq 'ARRAY' && @$sources ) { |
717 |
|
|
$document_links{$classlink->class_table->db_table}{source} = 1 if grep { $self->class eq $_ } @$sources; |
718 |
|
|
} |
719 |
|
|
my $dests = $classlink->available_destinations; |
720 |
|
|
if ( ref $dests eq 'ARRAY' && @$dests ) { |
721 |
45 |
ahitrov@rambler.ru |
$document_links{$classlink->class_table->db_table}{dest} = 1 if grep { $self->class eq $_ } @$dests; |
722 |
8 |
ahitrov@rambler.ru |
} |
723 |
|
|
} |
724 |
|
|
foreach my $tablename ( keys %document_links ) { |
725 |
|
|
my (@wheres, @values); |
726 |
|
|
if ( exists $document_links{$tablename}{source} ) { |
727 |
|
|
push @wheres, "(source_id = ? AND source_class = ?)"; |
728 |
|
|
push @values, ( $self->id, $self->class ); |
729 |
|
|
} |
730 |
|
|
if ( exists $document_links{$tablename}{dest} ) { |
731 |
|
|
push @wheres, "(dest_id = ? AND dest_class = ?)"; |
732 |
|
|
push @values, ( $self->id, $self->class ); |
733 |
|
|
} |
734 |
|
|
my $request = "DELETE FROM $tablename WHERE ".join (' OR ', @wheres); |
735 |
|
|
warn "DELETE LINKS. Request: [$request]\n" if $DEBUG; |
736 |
|
|
warn "Values: [".join(', ', @values)."]\n" if $DEBUG; |
737 |
|
|
$keeper->TSQL->do($request, {}, @values) || return $self->t_abort(); |
738 |
|
|
} |
739 |
|
|
} else { |
740 |
|
|
$keeper->TSQL->do("DELETE FROM links WHERE source_id = ? AND source_class = ? ", {}, $self->id, $self->class) || return $self->t_abort(); |
741 |
|
|
$keeper->TSQL->do("DELETE FROM links WHERE dest_id = ? AND dest_class = ? ", {}, $self->id, $self->class) || return $self->t_abort(); |
742 |
|
|
} |
743 |
|
|
$keeper->t_finish(); |
744 |
|
|
|
745 |
|
|
$self->post_delete(); |
746 |
|
|
|
747 |
|
|
$self->drop_cache('delete') if ($keeper->state()->memcached_enable()); |
748 |
|
|
|
749 |
|
|
return 1; |
750 |
|
|
} |
751 |
|
|
|
752 |
|
|
# ---------------------------------------------------------------------------- |
753 |
|
|
# Метод links() возвращает массив объектов типа Contenido::Link |
754 |
|
|
# |
755 |
|
|
# Формат использования: |
756 |
|
|
# $document->links([класс]) |
757 |
|
|
# ---------------------------------------------------------------------------- |
758 |
|
|
sub links { |
759 |
|
|
my ($self, $lclass, $direction, %opts) = @_; |
760 |
|
|
do { $log->error("Метод ->links() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
761 |
|
|
|
762 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($self->keeper); |
763 |
|
|
|
764 |
|
|
do { $log->warning("Вызов метода ->links() без указания идентификатора сообщения-источника"); return () } unless ($self->id() > 0); |
765 |
|
|
|
766 |
|
|
my $check = defined $direction ? 'dest_id' : 'source_id'; |
767 |
|
|
|
768 |
|
|
$opts{$check} = $self->id(); |
769 |
|
|
|
770 |
|
|
if (defined($lclass) && (length($lclass) > 0)) { |
771 |
55 |
ahitrov@rambler.ru |
$opts{class} = $lclass; |
772 |
8 |
ahitrov@rambler.ru |
} |
773 |
|
|
|
774 |
|
|
my @links = $self->keeper->get_links(%opts); |
775 |
|
|
|
776 |
|
|
$self->{links} = \@links; |
777 |
|
|
return @links; |
778 |
|
|
} |
779 |
|
|
|
780 |
|
|
|
781 |
|
|
sub linked_to { |
782 |
|
|
my ($self, $lclass) = @_; |
783 |
|
|
$self->links($lclass, 1); |
784 |
|
|
} |
785 |
|
|
|
786 |
|
|
|
787 |
|
|
# ---------------------------------------------------------------------------- |
788 |
|
|
# Установка связи. Должен быть обязательно задан класс... |
789 |
|
|
# В качестве source_id выступает идентификатор объекта, в качестве $dest_id - |
790 |
|
|
# заданный. |
791 |
|
|
# |
792 |
|
|
# Формат использования: |
793 |
|
|
# $document->set_link($lclass, $dest_id) |
794 |
|
|
# |
795 |
|
|
# Проверки не производится - у сообщения может быть несколько одинаковых |
796 |
|
|
# связей. |
797 |
|
|
# ---------------------------------------------------------------------------- |
798 |
|
|
sub set_link { |
799 |
|
|
my ($self, $lclass, $dest_id, $dest_class, @opts) = @_; |
800 |
|
|
do { $log->error("Метод ->set_link() вызван с неправильным кол-вом агрументов"); die } if @opts % 2; |
801 |
|
|
do { $log->error("Метод ->set_link() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
802 |
|
|
my %opts = @opts; |
803 |
|
|
|
804 |
|
|
return undef if ($self->keeper->state->readonly()); |
805 |
|
|
|
806 |
|
|
do { $log->warning("Вызов метода ->set_link() без указания идентификатора сообщения-источника"); return undef } unless ($self->id() > 0); |
807 |
|
|
do { $log->warning("Вызов метода ->set_link() без указания идентификатора сообщения-цели"); return undef } unless ($dest_id >= 0); |
808 |
115 |
ahitrov |
do { $log->warning("Вызов метода ->set_link() без указания класса связи"); } unless defined($lclass) && length($lclass); |
809 |
8 |
ahitrov@rambler.ru |
|
810 |
|
|
# Создаем объект связи... |
811 |
|
|
my $link = $lclass->new($self->keeper); |
812 |
|
|
|
813 |
|
|
$link->dest_id($dest_id); |
814 |
|
|
$link->dest_class($dest_class); |
815 |
|
|
|
816 |
|
|
$link->status(1); |
817 |
|
|
|
818 |
|
|
$link->source_id($self->id()); |
819 |
|
|
$link->source_class($self->class()); |
820 |
|
|
|
821 |
|
|
while (my ($k,$v) = each %opts) { |
822 |
55 |
ahitrov@rambler.ru |
$link->{$k} = $v; |
823 |
8 |
ahitrov@rambler.ru |
} |
824 |
|
|
|
825 |
|
|
if ($link->store()) |
826 |
|
|
{ |
827 |
55 |
ahitrov@rambler.ru |
return $link->id; |
828 |
8 |
ahitrov@rambler.ru |
} else { |
829 |
55 |
ahitrov@rambler.ru |
return undef; |
830 |
8 |
ahitrov@rambler.ru |
} |
831 |
|
|
} |
832 |
|
|
|
833 |
|
|
# ------------------------------------------------------------------- |
834 |
|
|
# Превращает объект в проблессенный хэш. |
835 |
|
|
# |
836 |
|
|
sub prepare_for_cache { |
837 |
|
|
my $self = shift; |
838 |
|
|
|
839 |
|
|
do { $log->error("Метод ->prepare_for_cache() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
840 |
|
|
|
841 |
|
|
my $hash = {}; |
842 |
|
|
|
843 |
|
|
foreach ( $self->structure() ) { |
844 |
55 |
ahitrov@rambler.ru |
$hash->{$_->{attr}} = $self->{$_->{attr}} if defined $self->{$_->{attr}}; |
845 |
8 |
ahitrov@rambler.ru |
} |
846 |
|
|
bless $hash, $self->class(); |
847 |
|
|
return $hash; |
848 |
|
|
} |
849 |
|
|
|
850 |
|
|
# ------------------------------------------------------------------- |
851 |
|
|
# Восстанавливает полноценный объект по проблессенному хэшу. |
852 |
|
|
# Хэш при этом превращается в полноценный объект. |
853 |
|
|
# ------------------------------------------------------------------- |
854 |
|
|
sub recover_from_cache { |
855 |
|
|
my $self = shift; |
856 |
|
|
my $keeper = shift; |
857 |
|
|
|
858 |
|
|
do { $log->error("Метод ->recover_from_cache() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
859 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($keeper); |
860 |
|
|
|
861 |
|
|
#не нужен тут bless очередной... 100% если уж попали в обьектный метод то он явно имеет класс нужный |
862 |
|
|
$self->init(); |
863 |
|
|
$self->keeper($keeper); |
864 |
|
|
|
865 |
|
|
return 1; |
866 |
|
|
} |
867 |
|
|
|
868 |
|
|
# ------------------------------------------------------------------- |
869 |
|
|
# Возвращает хэш: |
870 |
|
|
# {действие1 => [кэш1, кэш2, ...], действие2 => [кэш1, кэш2, ...], ...} |
871 |
|
|
# Т.е. для каждого действия задается список имен ключей в кэше, |
872 |
|
|
# которые надо удалить. |
873 |
|
|
# Дефолтные значени действий: insert, update, delete |
874 |
|
|
# Для более сложной логики работы этот метод должен быть переопределен |
875 |
|
|
# в классе самого объекта |
876 |
|
|
# |
877 |
|
|
sub dependencies { |
878 |
|
|
my ($self, $mode) = @_; |
879 |
|
|
|
880 |
|
|
my @keys = ($self->get_object_key,); |
881 |
|
|
my $object_unique_key = $self->get_object_unique_key; |
882 |
|
|
push @keys, $object_unique_key if defined $object_unique_key; |
883 |
|
|
|
884 |
|
|
return |
885 |
|
|
($mode eq 'delete') || ($mode eq 'insert') || ($mode eq 'update') |
886 |
|
|
? \@keys |
887 |
|
|
: []; |
888 |
|
|
} |
889 |
|
|
|
890 |
|
|
# ------------------------------------------------------------------- |
891 |
|
|
# Удаляет из кэша ключи, заданные при помощи dependencies(). |
892 |
|
|
# Пример вызова: |
893 |
|
|
# $group->drop_cache('update'); |
894 |
|
|
# |
895 |
|
|
sub drop_cache { |
896 |
|
|
my $self = shift; |
897 |
|
|
my $mode = shift; |
898 |
|
|
|
899 |
|
|
do { $log->error("Метод ->drop_cache() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
900 |
|
|
|
901 |
|
|
my $keeper = $self->keeper; |
902 |
|
|
do { $log->error("В объекте документа не определена ссылка на базу данных"); die } unless ref($keeper); |
903 |
|
|
|
904 |
|
|
my $dependencies = $self->dependencies($mode, @_); |
905 |
|
|
|
906 |
|
|
my @not_deleted = (); |
907 |
|
|
if ( defined($dependencies) && (ref($dependencies) eq 'ARRAY') ) { |
908 |
|
|
for (@$dependencies) { |
909 |
|
|
my $res = $self->keeper->MEMD ? $self->keeper->MEMD->delete($_) : undef; |
910 |
|
|
push @not_deleted, $_ unless $res; |
911 |
|
|
$keeper->MEMD->delete($_) if ($keeper->MEMD); |
912 |
|
|
} |
913 |
|
|
} |
914 |
|
|
return @not_deleted; |
915 |
|
|
} |
916 |
|
|
|
917 |
|
|
|
918 |
|
|
sub keeper { |
919 |
|
|
my $self = shift; |
920 |
|
|
my $project_keeper = shift; |
921 |
|
|
|
922 |
|
|
do { $log->error("Метод keeper() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
923 |
|
|
|
924 |
|
|
if ($project_keeper && ref $project_keeper) { |
925 |
|
|
$self->{keeper} = $project_keeper; |
926 |
|
|
return $project_keeper; |
927 |
|
|
} |
928 |
|
|
return $self->{keeper} && ref $self->{keeper} ? $self->{keeper} : $keeper; |
929 |
|
|
} |
930 |
|
|
|
931 |
|
|
#делаем затычку для init_from_db чтобы проинициализировать класс если надо |
932 |
|
|
sub init_from_db { |
933 |
|
|
my $self = shift; |
934 |
|
|
|
935 |
|
|
my $class = ref($self) || $self; |
936 |
|
|
|
937 |
|
|
#защита от бесконечной рекурсии если class_init не срабатывает |
938 |
|
|
if (defined($_[-1]) and ($_[-1] eq 'RECURSIVE CALL FLAG!')) { |
939 |
|
|
do { $log->error("$class cannot be initialized (->class_init dont work) (recursive call) !!!"); die }; |
940 |
|
|
} |
941 |
|
|
|
942 |
|
|
#если клас каким то странным образом все еще не проинициализирован то попробовать проинициализировать |
943 |
|
|
#только инициализация метода init_from_db допускает не ref на входе |
944 |
|
|
if ($class and $class->isa('Contenido::Object')) { |
945 |
|
|
no strict 'refs'; |
946 |
|
|
if (${$class.'::class_init_done'}) { |
947 |
|
|
do { $log->error("$class already initialized but DONT HAVE init_from_db method!!!"); die }; |
948 |
|
|
} else { |
949 |
|
|
if ($self->class_init()) { |
950 |
|
|
return $self->init_from_db(@_, 'RECURSIVE CALL FLAG!'); |
951 |
|
|
} else { |
952 |
|
|
do { $log->error("$class cannot be initialized (->class_init dont work) !!!"); die }; |
953 |
|
|
} |
954 |
|
|
} |
955 |
|
|
} else { |
956 |
|
|
do { $log->error("$class cannot be initialized (not Contenido::Object child class) !!!"); die }; |
957 |
|
|
} |
958 |
|
|
} |
959 |
|
|
|
960 |
|
|
# ---------------------------------------------------------------------------- |
961 |
|
|
# Это умный AUTOLOAD. Ловит методов для установки/чтения полей... |
962 |
|
|
# Версия 1.0 |
963 |
|
|
# теперь он герерирует необходимый метод доступу если надо |
964 |
|
|
# ---------------------------------------------------------------------------- |
965 |
|
|
sub AUTOLOAD { |
966 |
|
|
my $self = shift; |
967 |
|
|
my $attribute = $AUTOLOAD; |
968 |
|
|
|
969 |
|
|
$log->info("$self calling AUTOLOAD method: $attribute") if ($DEBUG_CORE); |
970 |
|
|
|
971 |
|
|
$attribute=~s/^.*:://; |
972 |
|
|
|
973 |
|
|
my $class = ref($self); |
974 |
|
|
unless ($class and $class->isa('Contenido::Object')) { |
975 |
|
|
|
976 |
|
|
my $mason_comp = ref($HTML::Mason::Commands::m) ? $HTML::Mason::Commands::m->current_comp() : undef; |
977 |
|
|
my $mason_file = ref($mason_comp) ? $mason_comp->path : undef; |
978 |
|
|
my ($package, $filename, $line) = caller; |
979 |
|
|
|
980 |
|
|
$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)); |
981 |
|
|
if (wantarray) { return (); } else { return undef; } |
982 |
|
|
} |
983 |
|
|
|
984 |
|
|
#вообщето сюда было бы не плохо засунуть инициализацию класса если уж мы каким то хреном сюда попали для неинициализированного класса |
985 |
|
|
{ |
986 |
|
|
no strict 'refs'; |
987 |
|
|
unless (${$class.'::class_init_done'}) { |
988 |
|
|
my ($package, $filename, $line) = caller; |
989 |
|
|
$log->error("AUTOLOAD called method '$attribute' for not initialised class ($class) from '$package/$filename/$line'"); |
990 |
|
|
if (wantarray) { return (); } else { return undef; } |
991 |
|
|
} |
992 |
|
|
} |
993 |
|
|
|
994 |
|
|
if (! exists($self->{attributes}->{$attribute})) { |
995 |
|
|
|
996 |
|
|
my $mason_comp = ref($HTML::Mason::Commands::m) ? $HTML::Mason::Commands::m->current_comp() : undef; |
997 |
|
|
my $mason_file = ref($mason_comp) ? $mason_comp->path : undef; |
998 |
|
|
my ($package, $filename, $line) = caller; |
999 |
|
|
|
1000 |
|
|
$log->error(ref($self)."): Вызов метода, для которого не существует обрабатываемого свойства: ->$attribute() called from $package/$filename/$line ".($mason_file ? "called from $mason_file" : '')."\n".Data::Dumper::Dumper($self)); |
1001 |
|
|
if (wantarray) { return (); } else { return undef; } |
1002 |
|
|
#special work with ARRAY types |
1003 |
|
|
} elsif ($self->{attributes}->{$attribute} eq 'ARRAY') { |
1004 |
|
|
my $funct = " |
1005 |
|
|
use Contenido::Globals; |
1006 |
|
|
my \$self = shift; |
1007 |
|
|
unless (ref(\$self->{$attribute}) eq 'ARRAY') { |
1008 |
|
|
my (\$package, \$filename, \$line) = caller; |
1009 |
|
|
\$log->error(\"Wrong structure in field $attribute called from \$package/\$filename/\$line \\n\".Data::Dumper::Dumper(\$self)) if (\$self->{$attribute});; |
1010 |
|
|
\$self->{$attribute} = []; |
1011 |
|
|
} |
1012 |
|
|
\$self->{$attribute} = [\@_] if (\@_); |
1013 |
|
|
return \@{\$self->{$attribute}};"; |
1014 |
|
|
|
1015 |
|
|
if (create_method($class, $attribute, $funct)) { |
1016 |
|
|
return $self->$attribute(@_); |
1017 |
|
|
} else { |
1018 |
|
|
$log->error("Cannot create method $attribute for class $self->{class}"); |
1019 |
|
|
#fallback to old autoload method if create method fail |
1020 |
|
|
unless (ref($self->{$attribute}) eq 'ARRAY') { |
1021 |
|
|
my ($package, $filename, $line) = caller; |
1022 |
|
|
$log->error("Wrong structure in field $attribute called from $package/$filename/$line \n".Data::Dumper::Dumper($self)); |
1023 |
|
|
$self->{$attribute} = []; |
1024 |
|
|
} |
1025 |
|
|
$self->{$attribute} = [@_] if (@_); |
1026 |
|
|
return @{$self->{$attribute}}; |
1027 |
|
|
} |
1028 |
|
|
#todo: добавить работу с images Нормальную когда она будет готова |
1029 |
|
|
} else { |
1030 |
|
|
#todo: валидация формата полей |
1031 |
|
|
my $funct = " |
1032 |
|
|
my \$self = shift; |
1033 |
|
|
\$self->{$attribute} = shift if (\@_); |
1034 |
|
|
return \$self->{$attribute};"; |
1035 |
|
|
|
1036 |
|
|
if (create_method($class, $attribute, $funct)) { |
1037 |
|
|
return $self->$attribute(@_); |
1038 |
|
|
} else { |
1039 |
|
|
$log->error("Cannot create method $attribute for class $self->{class}"); |
1040 |
|
|
#fallback to old autoload method if create method fail |
1041 |
|
|
$self->{$attribute} = shift if (@_); |
1042 |
|
|
return $self->{$attribute}; |
1043 |
|
|
} |
1044 |
|
|
} |
1045 |
|
|
} |
1046 |
|
|
|
1047 |
|
|
sub eval_dump { |
1048 |
|
|
no strict 'vars'; |
1049 |
|
|
return {} unless ${$_[0]}; |
1050 |
|
|
return eval ${$_[0]}; |
1051 |
|
|
} |
1052 |
|
|
|
1053 |
217 |
ahitrov |
sub eval_json { |
1054 |
|
|
return undef unless ${$_[0]}; |
1055 |
232 |
ahitrov |
my $str = ${$_[0]}; |
1056 |
|
|
my $chr = substr($str, 0, 1); return $str unless $chr eq '{' || $chr eq '['; |
1057 |
|
|
my $value = $json_u->decode( $str ); |
1058 |
217 |
ahitrov |
# map { $_ = Encode::encode(\'utf-8\', $_) unless ref $_; } values %$value; |
1059 |
|
|
return $value; |
1060 |
|
|
} |
1061 |
|
|
|
1062 |
8 |
ahitrov@rambler.ru |
sub create_method { |
1063 |
|
|
my ($class, $sub_name, $code) = @_; |
1064 |
|
|
|
1065 |
|
|
unless ($class and $sub_name and $code) { |
1066 |
|
|
$log->error("Wrong call create_method $class/$sub_name/$code"); |
1067 |
|
|
return 0; |
1068 |
|
|
} |
1069 |
|
|
|
1070 |
|
|
my $string = "package $class;\n\nsub $sub_name {\n$code\n}\n\n1;"; |
1071 |
|
|
eval $string; |
1072 |
|
|
|
1073 |
|
|
if ($@) { |
1074 |
|
|
$log->error("Cannot create method $sub_name for class $class because $@ (method code:\n$string\n)"); |
1075 |
|
|
return 0; |
1076 |
|
|
} else { |
1077 |
|
|
$log->info("Method '$sub_name' for class '$class' (method code:\n$string\n) created ok") if ($DEBUG_CORE); |
1078 |
|
|
return 1; |
1079 |
|
|
} |
1080 |
|
|
} |
1081 |
|
|
|
1082 |
|
|
######################################## ONLY FOR INTERNAL USE!!!! ################################################# |
1083 |
|
|
#todo добавить проверку что если классов список то проверить что у них 1 table а не 5 разных |
1084 |
|
|
sub _get_table { |
1085 |
|
|
my ($self, %opts) = @_; |
1086 |
|
|
|
1087 |
|
|
my $class_table; |
1088 |
|
|
|
1089 |
|
|
my $table=$opts{table}; |
1090 |
|
|
my $class=$opts{class} || ref $self || $self; |
1091 |
|
|
|
1092 |
|
|
#пришла таблица в %opts |
1093 |
|
|
if ($table and $table->can('new')) { |
1094 |
|
|
$class_table=$table; |
1095 |
|
|
#иначе пробуем по классу |
1096 |
|
|
} elsif ($class and !ref($class)) { |
1097 |
|
|
unless ($class->can('class_table')) { |
1098 |
|
|
$log->error("$class cannot class_table"); |
1099 |
|
|
return undef; |
1100 |
|
|
} |
1101 |
|
|
$class_table=$class->class_table(); |
1102 |
|
|
#иначе пробуем по первому классу в списке |
1103 |
|
|
} elsif ($class and ref($class) eq 'ARRAY' and @$class) { |
1104 |
|
|
unless ($class->[0]->can('class_table')) { |
1105 |
|
|
$log->error("$class->[0] cannot class_table"); |
1106 |
|
|
return undef; |
1107 |
|
|
} |
1108 |
|
|
$class_table=$class->[0]->class_table(); |
1109 |
|
|
#иначе умолчательную |
1110 |
|
|
} else { |
1111 |
|
|
$class_table='SQL::DocumentTable'; |
1112 |
|
|
} |
1113 |
|
|
|
1114 |
|
|
if ($class_table->can('new')) { |
1115 |
|
|
return $class_table->new(); |
1116 |
|
|
} else { |
1117 |
|
|
$log->error("$class_table cannot new!!!!"); |
1118 |
|
|
return undef; |
1119 |
|
|
} |
1120 |
|
|
} |
1121 |
|
|
|
1122 |
|
|
####################################################################################################################### |
1123 |
|
|
########## OLD CODE FOR COMPATIBILITY ################################################################################# |
1124 |
|
|
####################################################################################################################### |
1125 |
|
|
sub structure { |
1126 |
|
|
my $self = shift; |
1127 |
|
|
my $class = ref($self); |
1128 |
|
|
{ |
1129 |
|
|
no strict 'refs'; |
1130 |
|
|
return @${$class.'::structure'}; |
1131 |
|
|
} |
1132 |
|
|
} |
1133 |
|
|
|
1134 |
|
|
|
1135 |
|
|
# оставлена для обратной совместимости... |
1136 |
|
|
sub get_image { |
1137 |
|
|
my $self = shift; |
1138 |
217 |
ahitrov |
if ( $self->keeper->serialize_with eq 'json' ) { |
1139 |
|
|
return $self->get_data(shift, 'encode'); |
1140 |
|
|
} else { |
1141 |
|
|
return $self->get_data(shift); |
1142 |
|
|
} |
1143 |
8 |
ahitrov@rambler.ru |
} |
1144 |
|
|
|
1145 |
|
|
sub raw_restore { |
1146 |
|
|
my $self = shift; |
1147 |
|
|
do { $log->error("Метод restore() можно вызывать только у объектов, но не классов"); die } unless ref $self; |
1148 |
|
|
do { $log->warning("Вызов метода Contenido\:\:Object\:\:raw_restore() без указания идентификатора для чтения"); return undef } unless $self->id; |
1149 |
|
|
$self->restore_extras(); |
1150 |
|
|
} |
1151 |
|
|
|
1152 |
|
|
sub init { |
1153 |
|
|
my $self = shift; |
1154 |
|
|
my $class = ref($self) || $self; |
1155 |
|
|
$self->class_init(); |
1156 |
|
|
{ |
1157 |
|
|
no strict 'refs'; |
1158 |
|
|
$self->{attributes} = ${$class.'::attributes'}; |
1159 |
|
|
} |
1160 |
|
|
return 1; |
1161 |
|
|
} |
1162 |
|
|
|
1163 |
|
|
sub get_file_name { |
1164 |
|
|
my $self = shift; |
1165 |
|
|
|
1166 |
|
|
do { $log->error("Метод get_file_name можно вызывать только у объектов, но не классов"); die } unless ref $self; |
1167 |
|
|
|
1168 |
|
|
my @date; |
1169 |
|
|
my $time = time; |
1170 |
|
|
|
1171 |
|
|
if ($self->{"dtime"} and $self->{"dtime"} =~ /^(\d{4})-(\d{2})-(\d{2})/) { |
1172 |
|
|
@date = ($1, $2, $3); |
1173 |
|
|
} else { |
1174 |
|
|
@date = (localtime $time)[5, 4, 3]; $date[0] += 1900; $date[1] += 1; |
1175 |
|
|
} |
1176 |
|
|
|
1177 |
|
|
my $component_class = lc((reverse split "::", ref $self)[0]); |
1178 |
|
|
my $component_date = sprintf "%04d/%02d/%02d", @date; |
1179 |
|
|
my $component_time_rand = sprintf "%010d_%05d", $time, int rand 99999; |
1180 |
|
|
|
1181 |
|
|
return join "/", $component_class, $component_date, $component_time_rand; |
1182 |
|
|
} |
1183 |
|
|
|
1184 |
|
|
sub get { |
1185 |
|
|
my ( $self, %opts ) = @_; |
1186 |
|
|
my $class = ref $self || $self; |
1187 |
|
|
my $local_keeper = (ref($self) and ref($self->keeper)) ? $self->keeper : $keeper; |
1188 |
|
|
delete $opts{class}; |
1189 |
|
|
return $keeper->get_documents( class => $class, %opts ); |
1190 |
|
|
} |
1191 |
|
|
|
1192 |
|
|
sub contenido_is_available { |
1193 |
|
|
return 1; |
1194 |
|
|
} |
1195 |
|
|
|
1196 |
|
|
sub contenido_status_style { |
1197 |
|
|
return; |
1198 |
|
|
} |
1199 |
|
|
|
1200 |
|
|
sub memcached_expire { |
1201 |
|
|
return $_[0]->keeper->state->memcached_object_expire; |
1202 |
|
|
} |
1203 |
|
|
|
1204 |
113 |
ahitrov |
# ---------------------------------------------------------------------------- |
1205 |
125 |
ahitrov |
# Метод _store_image() сохраняет графику, привязанную к полю image или images |
1206 |
113 |
ahitrov |
# |
1207 |
|
|
# Формат использования: |
1208 |
125 |
ahitrov |
# $document->_store_image( INPUT, attr => 'fieldname' ) |
1209 |
365 |
ahitrov |
# $document->_store_image( INPUT, prop => $prophash ) |
1210 |
113 |
ahitrov |
# ---------------------------------------------------------------------------- |
1211 |
|
|
sub _store_image { |
1212 |
|
|
my $self = shift; |
1213 |
|
|
do { $log->error("Метод delete() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
1214 |
|
|
|
1215 |
|
|
my $input = shift; |
1216 |
|
|
my (%opts) = @_; |
1217 |
|
|
|
1218 |
365 |
ahitrov |
return Contenido::File::store_image( $input, object => $self, %opts ); |
1219 |
113 |
ahitrov |
} |
1220 |
|
|
|
1221 |
125 |
ahitrov |
# ---------------------------------------------------------------------------- |
1222 |
|
|
# Метод _delete_image() удаляет файлы, связанные с полем image или images. |
1223 |
|
|
# Вычищает все мини-копии |
1224 |
|
|
# |
1225 |
|
|
# Формат использования: |
1226 |
347 |
ahitrov |
# $document->_delete_image( $image_attr_structure ) |
1227 |
125 |
ahitrov |
# ---------------------------------------------------------------------------- |
1228 |
113 |
ahitrov |
sub _delete_image { |
1229 |
|
|
my $self = shift; |
1230 |
|
|
my $IMAGE = shift; |
1231 |
|
|
|
1232 |
|
|
return Contenido::File::remove_image( $IMAGE ); |
1233 |
|
|
} |
1234 |
|
|
|
1235 |
125 |
ahitrov |
# ---------------------------------------------------------------------------- |
1236 |
347 |
ahitrov |
# Метод _store_binary() сохраняет произвольный бинарный файл, привязанную к полю multimedia_multi или multimedia_new |
1237 |
125 |
ahitrov |
# |
1238 |
|
|
# Формат использования: |
1239 |
|
|
# $document->_store_binary( INPUT, attr => 'fieldname' ) |
1240 |
|
|
# ---------------------------------------------------------------------------- |
1241 |
|
|
sub _store_binary { |
1242 |
|
|
my $self = shift; |
1243 |
|
|
do { $log->error("Метод delete() можно вызывать только у объектов, но не классов"); die } unless ref($self); |
1244 |
|
|
|
1245 |
|
|
my $input = shift; |
1246 |
|
|
my (%opts) = @_; |
1247 |
|
|
|
1248 |
|
|
return Contenido::File::store_binary( $input, object => $self, attr => $opts{attr} ); |
1249 |
|
|
} |
1250 |
|
|
|
1251 |
|
|
# ---------------------------------------------------------------------------- |
1252 |
|
|
# Метод _delete_binary() удаляет файлы, связанные с полем multimedia или multimedia_new. |
1253 |
|
|
# Не пытается искать мини-копии |
1254 |
|
|
# |
1255 |
|
|
# Формат использования: |
1256 |
|
|
# $document->_delete_binary( $binary_attr_structure ) |
1257 |
|
|
# ---------------------------------------------------------------------------- |
1258 |
|
|
sub _delete_binary { |
1259 |
|
|
my $self = shift; |
1260 |
|
|
my $BINARY = shift; |
1261 |
|
|
|
1262 |
|
|
return Contenido::File::remove_binary( $BINARY ); |
1263 |
|
|
} |
1264 |
|
|
|
1265 |
8 |
ahitrov@rambler.ru |
1; |
1266 |
|
|
|