1 package t
::lib
::TestBuilder
;
10 use Koha
::DateUtils
qw( dt_from_string );
12 use Bytes
::Random
::Secure
;
18 SIZE_BARCODE
=> 20, # Not perfect but avoid to fetch the value when creating a new item
24 bless( $self, $class );
26 $self->schema( Koha
::Database
->new()->schema );
27 $self->schema->storage->sql_maker->quote_char('`');
29 $self->{gen_type
} = _gen_type
();
30 $self->{default_values
} = _gen_default_values
();
35 my ($self, $schema) = @_;
37 if( defined( $schema ) ) {
38 $self->{schema
} = $schema;
40 return $self->{schema
};
43 # sub clear has been obsoleted; use delete_all from the schema resultset
46 my ( $self, $params ) = @_;
47 my $source = $params->{source
} || return;
48 my @recs = ref( $params->{records
} ) eq 'ARRAY'?
49 @
{$params->{records
}}: ( $params->{records
} // () );
50 # tables without PK are not supported
51 my @pk = $self->schema->source( $source )->primary_columns;
54 foreach my $rec ( @recs ) {
55 # delete only works when you supply full primary key values
56 # $cond does not include searches for undef (not allowed in PK)
57 my $cond = { map { defined $rec->{$_}?
($_, $rec->{$_}): (); } @pk };
58 next if keys %$cond < @pk;
59 $self->schema->resultset( $source )->search( $cond )->delete;
60 # we clear the pk columns in the supplied hash
61 # this indirectly signals at least an attempt to delete
62 map { delete $rec->{$_}; } @pk;
69 my ( $self, $params ) = @_;
71 my $class = $params->{class};
72 my $value = $params->{value
};
74 if ( not defined $class ) {
75 carp
"Missing class param";
79 my @unknowns = grep( !/^(class|value)$/, keys %{ $params });
80 carp
"Unknown parameter(s): ", join( ', ', @unknowns ) if scalar @unknowns;
83 my $source = $class->_type;
85 my $hashref = $self->build({ source
=> $source, value
=> $value });
87 if ( $class eq 'Koha::Old::Patrons' ) {
88 $object = $class->search({ borrowernumber
=> $hashref->{borrowernumber
} })->next;
89 } elsif ( $class eq 'Koha::Statistics' ) {
90 $object = $class->search({ datetime
=> $hashref->{datetime
} })->next;
93 my @pks = $self->schema->source( $class->_type )->primary_columns;
94 foreach my $pk ( @pks ) {
95 push @ids, $hashref->{ $pk };
98 $object = $class->find( @ids );
105 # build returns a hash of column values for a created record, or undef
106 # build does NOT update a record, or pass back values of an existing record
107 my ($self, $params) = @_;
108 my $source = $params->{source
};
110 carp
"Source parameter not specified!";
113 my $value = $params->{value
};
115 my @unknowns = grep( !/^(source|value)$/, keys %{ $params });
116 carp
"Unknown parameter(s): ", join( ', ', @unknowns ) if scalar @unknowns;
118 my $col_values = $self->_buildColumnValues({
122 return if !$col_values; # did not meet unique constraints?
124 # loop thru all fk and create linked records if needed
125 # fills remaining entries in $col_values
126 my $foreign_keys = $self->_getForeignKeys( { source
=> $source } );
127 for my $fk ( @
$foreign_keys ) {
128 # skip when FK points to itself: e.g. borrowers:guarantorid
129 next if $fk->{source
} eq $source;
130 my $keys = $fk->{keys};
131 my $tbl = $fk->{source
};
132 my $res = $self->_create_links( $tbl, $keys, $col_values, $value );
133 return if !$res; # failed: no need to go further
134 foreach( keys %$res ) { # save new values
135 $col_values->{$_} = $res->{$_};
139 # store this record and return hashref
140 return $self->_storeColumnValues({
142 values => $col_values,
146 sub build_sample_biblio
{
147 my ( $self, $args ) = @_;
149 my $title = $args->{title
} || 'Some boring read';
150 my $author = $args->{author
} || 'Some boring author';
151 my $frameworkcode = $args->{frameworkcode
} || '';
152 my $itemtype = $args->{itemtype
}
153 || $self->build_object( { class => 'Koha::ItemTypes' } )->itemtype;
155 my $marcflavour = C4
::Context
->preference('marcflavour');
157 my $record = MARC
::Record
->new();
158 my ( $tag, $subfield ) = $marcflavour eq 'UNIMARC' ?
( 200, 'a' ) : ( 245, 'a' );
159 $record->append_fields(
160 MARC
::Field
->new( $tag, ' ', ' ', $subfield => $title ),
163 ( $tag, $subfield ) = $marcflavour eq 'UNIMARC' ?
( 200, 'f' ) : ( 100, 'a' );
164 $record->append_fields(
165 MARC
::Field
->new( $tag, ' ', ' ', $subfield => $author ),
168 ( $tag, $subfield ) = $marcflavour eq 'UNIMARC' ?
( 995, 'r' ) : ( 942, 'c' );
169 $record->append_fields(
170 MARC
::Field
->new( $tag, ' ', ' ', $subfield => $itemtype )
173 my ($biblio_id) = C4
::Biblio
::AddBiblio
( $record, $frameworkcode );
174 return Koha
::Biblios
->find($biblio_id);
177 sub build_sample_item
{
178 my ( $self, $args ) = @_;
181 delete $args->{biblionumber
} || $self->build_sample_biblio->biblionumber;
182 my $library = delete $args->{library
}
183 || $self->build_object( { class => 'Koha::Libraries' } )->branchcode;
185 # If itype is not passed it will be picked from the biblio (see Koha::Item->store)
187 my $barcode = delete $args->{barcode
}
188 || $self->_gen_text( { info
=> { size
=> SIZE_BARCODE
} } );
190 return Koha
::Item
->new(
192 biblionumber
=> $biblionumber,
193 homebranch
=> $library,
194 holdingbranch
=> $library,
198 )->store->get_from_storage;
201 # ------------------------------------------------------------------------------
202 # Internal helper routines
205 # returns undef for failure to create linked records
206 # otherwise returns hashref containing new column values for parent record
207 my ( $self, $linked_tbl, $keys, $col_values, $value ) = @_;
210 my ( $cnt_scalar, $cnt_null ) = ( 0, 0 );
212 # First, collect all values for creating a linked record (if needed)
213 foreach my $fk ( @
$keys ) {
214 my ( $col, $destcol ) = ( $fk->{col_name
}, $fk->{col_fk_name
} );
215 if( ref( $value->{$col} ) eq 'HASH' ) {
216 # add all keys from the FK hash
217 $fk_value = { %{ $value->{$col} }, %$fk_value };
219 if( exists $col_values->{$col} ) {
220 # add specific value (this does not necessarily exclude some
221 # values from the hash in the preceding if)
222 $fk_value->{ $destcol } = $col_values->{ $col };
224 $cnt_null++ if !defined( $col_values->{$col} );
228 # If we saw all FK columns, first run the following checks
229 if( $cnt_scalar == @
$keys ) {
230 # if one or more fk cols are null, the FK constraint will not be forced
231 return {} if $cnt_null > 0;
232 # does the record exist already?
233 return {} if $self->schema->resultset($linked_tbl)->find( $fk_value );
235 # create record with a recursive build call
236 my $row = $self->build({ source
=> $linked_tbl, value
=> $fk_value });
237 return if !$row; # failure
239 # Finally, only return the new values
241 foreach my $fk ( @
$keys ) {
242 my ( $col, $destcol ) = ( $fk->{col_name
}, $fk->{col_fk_name
} );
243 next if exists $col_values->{ $col };
244 $rv->{ $col } = $row->{ $destcol };
246 return $rv; # success
251 my $source = $params->{source
} || return;
252 $source =~ s
|(\w
+)$|$1|;
256 sub _buildColumnValues
{
257 my ($self, $params) = @_;
258 my $source = _formatSource
( $params ) || return;
259 my $original_value = $params->{value
};
262 my @columns = $self->schema->source($source)->columns;
263 my %unique_constraints = $self->schema->source($source)->unique_constraints();
266 # we try max $build_value times if there are unique constraints
267 BUILD_VALUE
: while ( $build_value ) {
268 # generate random values for all columns
269 for my $col_name( @columns ) {
270 my $valref = $self->_buildColumnValue({
272 column_name
=> $col_name,
273 value
=> $original_value,
275 return if !$valref; # failure
276 if( @
$valref ) { # could be empty
277 # there will be only one value, but it could be undef
278 $col_values->{$col_name} = $valref->[0];
282 # verify the data would respect each unique constraint
283 # note that this is INCOMPLETE since not all col_values are filled
284 CONSTRAINTS
: foreach my $constraint (keys %unique_constraints) {
287 my $constraint_columns = $unique_constraints{$constraint};
288 # loop through all constraint columns and build the condition
289 foreach my $constraint_column ( @
$constraint_columns ) {
291 # if one column does not exist or is undef, skip it
292 # an insert with a null will not trigger the constraint
294 if !exists $col_values->{ $constraint_column } ||
295 !defined $col_values->{ $constraint_column };
296 $condition->{ $constraint_column } =
297 $col_values->{ $constraint_column };
299 my $count = $self->schema
300 ->resultset( $source )
301 ->search( $condition )
304 # no point checking more stuff, exit the loop
309 last; # you passed all tests
311 return $col_values if $build_value > 0;
313 # if you get here, we have a problem
314 warn "Violation of unique constraint in $source";
318 sub _getForeignKeys
{
320 # Returns the following arrayref
321 # [ [ source => name, keys => [ col_name => A, col_fk_name => B ] ], ... ]
322 # The array gives source name and keys for each FK constraint
324 my ($self, $params) = @_;
325 my $source = $self->schema->source( $params->{source
} );
327 my ( @foreign_keys, $check_dupl );
328 my @relationships = $source->relationships;
329 for my $rel_name( @relationships ) {
330 my $rel_info = $source->relationship_info($rel_name);
331 if( $rel_info->{attrs
}->{is_foreign_key_constraint
} ) {
332 $rel_info->{source
} =~ s/^.*:://g;
333 my $rel = { source
=> $rel_info->{source
} };
336 while( my ($col_fk_name, $col_name) = each(%{$rel_info->{cond
}}) ) {
337 $col_name =~ s
|self
.(\w
+)|$1|;
338 $col_fk_name =~ s
|foreign
.(\w
+)|$1|;
340 col_name
=> $col_name,
341 col_fk_name
=> $col_fk_name,
344 # check if the combination table and keys is unique
345 # so skip double belongs_to relations (as in Biblioitem)
346 my $tag = $rel->{source
}. ':'.
347 join ',', sort map { $_->{col_name
} } @keys;
348 next if $check_dupl->{$tag};
349 $check_dupl->{$tag} = 1;
350 $rel->{keys} = \
@keys;
351 push @foreign_keys, $rel;
354 return \
@foreign_keys;
357 sub _storeColumnValues
{
358 my ($self, $params) = @_;
359 my $source = $params->{source
};
360 my $col_values = $params->{values};
361 my $new_row = $self->schema->resultset( $source )->create( $col_values );
362 return $new_row?
{ $new_row->get_columns }: {};
365 sub _buildColumnValue
{
366 # returns an arrayref if all goes well
367 # an empty arrayref typically means: auto_incr column or fk column
368 # undef means failure
369 my ($self, $params) = @_;
370 my $source = $params->{source
};
371 my $value = $params->{value
};
372 my $col_name = $params->{column_name
};
374 my $col_info = $self->schema->source($source)->column_info($col_name);
377 if( $col_info->{is_auto_increment
} ) {
378 if( exists $value->{$col_name} ) {
379 warn "Value not allowed for auto_incr $col_name in $source";
382 # otherwise: no need to assign a value
383 } elsif( $col_info->{is_foreign_key
} || _should_be_fk
($source,$col_name) ) {
384 if( exists $value->{$col_name} ) {
385 if( !defined $value->{$col_name} && !$col_info->{is_nullable
} ) {
386 # This explicit undef is not allowed
387 warn "Null value for $col_name in $source not allowed";
390 if( ref( $value->{$col_name} ) ne 'HASH' ) {
391 push @
$retvalue, $value->{$col_name};
393 # sub build will handle a passed hash value later on
395 } elsif( ref( $value->{$col_name} ) eq 'HASH' ) {
396 # this is not allowed for a column that is not a FK
397 warn "Hash not allowed for $col_name in $source";
399 } elsif( exists $value->{$col_name} ) {
400 if( !defined $value->{$col_name} && !$col_info->{is_nullable
} ) {
401 # This explicit undef is not allowed
402 warn "Null value for $col_name in $source not allowed";
405 push @
$retvalue, $value->{$col_name};
406 } elsif( exists $self->{default_values
}{$source}{$col_name} ) {
407 my $v = $self->{default_values
}{$source}{$col_name};
408 $v = &$v() if ref($v) eq 'CODE';
411 my $data_type = $col_info->{data_type
};
412 $data_type =~ s
| |_
|;
413 if( my $hdlr = $self->{gen_type
}->{$data_type} ) {
414 push @
$retvalue, &$hdlr( $self, { info
=> $col_info } );
416 warn "Unknown type $data_type for $col_name in $source";
424 # This sub is only needed for inconsistencies in the schema
425 # A column is not marked as FK, but a belongs_to relation is defined
426 my ( $source, $column ) = @_;
427 my $inconsistencies = {
428 'Item.biblionumber' => 1, #FIXME: Please remove me when I become FK
430 return $inconsistencies->{ "$source.$column" };
435 tinyint
=> \
&_gen_int
,
436 smallint
=> \
&_gen_int
,
437 mediumint
=> \
&_gen_int
,
438 integer
=> \
&_gen_int
,
439 bigint
=> \
&_gen_int
,
441 float
=> \
&_gen_real
,
442 decimal
=> \
&_gen_real
,
443 double_precision
=> \
&_gen_real
,
445 timestamp
=> \
&_gen_datetime
,
446 datetime
=> \
&_gen_datetime
,
450 varchar
=> \
&_gen_text
,
451 tinytext
=> \
&_gen_text
,
453 mediumtext
=> \
&_gen_text
,
454 longtext
=> \
&_gen_text
,
456 set
=> \
&_gen_set_enum
,
457 enum
=> \
&_gen_set_enum
,
459 tinyblob
=> \
&_gen_blob
,
460 mediumblob
=> \
&_gen_blob
,
462 longblob
=> \
&_gen_blob
,
467 my ($self, $params) = @_;
468 my $data_type = $params->{info
}->{data_type
};
471 if( $data_type eq 'tinyint' ) {
474 elsif( $data_type eq 'smallint' ) {
477 elsif( $data_type eq 'mediumint' ) {
480 elsif( $data_type eq 'integer' ) {
483 elsif( $data_type eq 'bigint' ) {
484 $max = 9223372036854775807;
486 return int( rand($max+1) );
490 my ($self, $params) = @_;
492 if( defined( $params->{info
}->{size
} ) ) {
493 $max = 10 ** ($params->{info
}->{size
}->[0] - $params->{info
}->{size
}->[1]);
495 $max = 10 ** 5 if $max > 10 ** 5;
496 return sprintf("%.2f", rand($max-0.1));
500 my ($self, $params) = @_;
501 return $self->schema->storage->datetime_parser->format_date(dt_from_string
)
505 my ($self, $params) = @_;
506 return $self->schema->storage->datetime_parser->format_datetime(dt_from_string
);
510 my ($self, $params) = @_;
511 # From perldoc String::Random
512 my $size = $params->{info
}{size
} // 10;
513 $size -= alt_rand
(0.5 * $size);
514 my $regex = $size > 1
515 ?
'[A-Za-z][A-Za-z0-9_]{'.($size-1).'}'
517 my $random = String
::Random
->new( rand_gen
=> \
&alt_rand
);
518 # rand_gen is only supported from 0.27 onward
519 return $random->randregex($regex);
522 sub alt_rand
{ #Alternative randomizer
524 my $random = Bytes
::Random
::Secure
->new( NonBlocking
=> 1 );
525 my $r = $random->irand / 2**32;
526 return int( $r * $max );
530 my ($self, $params) = @_;
531 return $params->{info
}->{extra
}->{list
}->[0];
535 my ($self, $params) = @_;;
539 sub _gen_default_values
{
544 gonenoaddress
=> undef,
555 more_subfields_xml
=> undef,
560 # Not X, used for statistics
561 category_type
=> sub { return [ qw( A C S I P ) ]->[int(rand(5))] },
564 pickup_location
=> 0,
568 rentalcharge_daily
=> 0,
569 rentalcharge_hourly
=> 0,
570 defaultreplacecost
=> 0,
581 RefundLostItemFeeRules
=> {
582 rule_name
=> 'refund',
589 t::lib::TestBuilder.pm - Koha module to create test records
593 use t::lib::TestBuilder;
594 my $builder = t::lib::TestBuilder->new;
596 # The following call creates a patron, linked to branch CPL.
597 # Surname is provided, other columns are randomly generated.
598 # Branch CPL is created if it does not exist.
599 my $patron = $builder->build({
600 source => 'Borrower',
601 value => { surname => 'Jansen', branchcode => 'CPL' },
606 This module automatically creates database records for you.
607 If needed, records for foreign keys are created too.
608 Values will be randomly generated if not passed to TestBuilder.
609 Note that you should wrap these actions in a transaction yourself.
615 my $builder = t::lib::TestBuilder->new;
617 Constructor - Returns the object TestBuilder
621 my $schema = $builder->schema;
623 Getter - Returns the schema of DBIx::Class
629 records => $patron, # OR: records => [ $patron, ... ],
632 Delete individual records, created by builder.
633 Returns the number of delete attempts, or undef.
637 $builder->build({ source => $source_name, value => $value });
639 Create a test record in the table, represented by $source_name.
640 The name is required and must conform to the DBIx::Class schema.
641 Values may be specified by the optional $value hashref. Will be
642 randomized otherwise.
643 If needed, TestBuilder creates linked records for foreign keys.
644 Returns the values of the new record as a hashref, or undef if
645 the record could not be created.
647 Note that build also supports recursive hash references inside the
648 value hash for foreign key columns, like:
650 column1 => 'some_value',
652 columnA => 'another_value',
655 The hash for fk_col2 here means: create a linked record with build
656 where columnA has this value. In case of a composite FK the hashes
659 Realize that passing primary key values to build may result in undef
660 if a record with that primary key already exists.
664 Given a plural Koha::Object-derived class, it creates a random element, and
665 returns the corresponding Koha::Object.
667 my $patron = $builder->build_object({ class => 'Koha::Patrons' [, value => { ... }] });
671 Yohann Dufour <yohann.dufour@biblibre.com>
673 Koha Development Team
677 Copyright 2014 - Biblibre SARL
681 This file is part of Koha.
683 Koha is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by
684 the Free Software Foundation; either version 3 of the License, or (at your option) any later version.
686 Koha is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
688 You should have received a copy of the GNU General Public License along with Koha; if not, see <http://www.gnu.org/licenses>.