use C4::Context;
-use Test::More tests => 3;
+use Test::More tests => 6;
+use Test::MockModule;
use C4::Context;
+use C4::Biblio qw( AddBiblio );
+use C4::Circulation;
use Koha::AuthUtils;
+use t::lib::Mocks;
use t::lib::Selenium;
use t::lib::TestBuilder;
+use t::lib::Mocks;
eval { require Selenium::Remote::Driver; };
skip "Selenium::Remote::Driver is needed for selenium tests.", 1 if $@;
C4::Context->set_preference('AudioAlerts', '1');
our @cleanup;
-subtest 'OPAC - borrowernumber and branchcode as html attributes' => sub {
- plan tests => 2;
+subtest 'OPAC - borrowernumber, branchcode and categorycode as html attributes' => sub {
+ plan tests => 3;
my $patron = $builder->build_object(
{ class => 'Koha::Patrons', value => { flags => 1 } } );
- my $password = Koha::AuthUtils::generate_password();
- $patron->update_password( $patron->userid, $password );
+ my $password = Koha::AuthUtils::generate_password($patron->category);
+ t::lib::Mocks::mock_preference( 'RequireStrongPassword', 0 );
+ $patron->set_password({ password => $password });
$s->opac_auth( $patron->userid, $password );
my $elt = $driver->find_element('//span[@class="loggedinusername"]');
- is( $elt->get_attribute('data-branchcode'), $patron->library->branchcode,
+ is( $elt->get_attribute('data-branchcode', 1), $patron->library->branchcode,
"Since bug 20921 span.loggedinusername should contain data-branchcode"
+ # No idea why we need the second param of get_attribute(). As
+ # data-branchcode is still there after page finished loading.
);
- is( $elt->get_attribute('data-borrowernumber'), $patron->borrowernumber,
+ is( $elt->get_attribute('data-borrowernumber', 1), $patron->borrowernumber,
"Since bug 20921 span.loggedinusername should contain data-borrowernumber"
);
+ is( $elt->get_attribute('data-categorycode', 1), $patron->categorycode,
+"Since bug 26847 span.loggedinusername should contain data-categorycode"
+ );
push @cleanup, $patron, $patron->category, $patron->library;
};
+subtest 'OPAC - Bibliographic record detail page must contain the data-biblionumber' => sub {
+ plan tests => 1;
+
+ my $builder = t::lib::TestBuilder->new;
+
+ my ( $biblionumber, $biblioitemnumber ) = add_biblio();
+ my $biblio = Koha::Biblios->find($biblionumber);
+
+ $driver->get( $opac_base_url . "opac-detail.pl?biblionumber=$biblionumber" );
+
+ my $elt = $driver->find_element('//div[@id="catalogue_detail_biblio"]');
+ is( $elt->get_attribute( 'data-biblionumber', 1 ),
+ $biblionumber, "#catalogue_detail_biblio contains data-biblionumber" );
+
+ push @cleanup, $biblio;
+ };
+
subtest 'OPAC - Remove from cart' => sub {
plan tests => 4;
+ # We need to prevent scrolling to prevent the floating toolbar from overlapping buttons we are testing
+ my $window_size = $driver->get_window_size();
+ $driver->set_window_size(1920,10800);
+
$driver->get( $opac_base_url . "opac-search.pl?q=d" );
# A better way to do that would be to modify the way we display the basket count
my $biblionumber3 = $checkboxes[2]->get_value();
my $biblionumber5 = $checkboxes[4]->get_value();
- $driver->find_element('//a[@class="addtocart cart'.$biblionumber1.'"]')->click;
+ $driver->find_element('//a[@class="btn btn-link btn-sm addtocart cart cart'.$biblionumber1.'"]')->click;
my $basket_count_elt = $driver->find_element('//span[@id="basketcount"]/span');
is( $basket_count_elt->get_text(),
1, 'One element should have been added to the cart' );
- $driver->find_element('//a[@class="addtocart cart'.$biblionumber3.'"]')->click;
- $driver->find_element('//a[@class="addtocart cart'.$biblionumber5.'"]')->click;
+ $driver->find_element('//a[@class="btn btn-link btn-sm addtocart cart cart'.$biblionumber3.'"]')->click;
+ $driver->find_element('//a[@class="btn btn-link btn-sm addtocart cart cart'.$biblionumber5.'"]')->click;
$basket_count_elt = $driver->find_element('//span[@id="basketcount"]/span');
is( $basket_count_elt->get_text(),
3, '3 elements should have been added to the cart' );
- $driver->find_element('//a[@class="cartRemove cartR'.$biblionumber3.'"]')->click;
+ $driver->find_element('//a[@class="btn btn-link btn-sm remove cartRemove cartR'.$biblionumber3.'"]')->click;
$basket_count_elt = $driver->find_element('//span[@id="basketcount"]/span');
is( $basket_count_elt->get_text(),
2, '1 element should have been removed from the cart' );
+
+ # Reset window size
+ $driver->set_window_size($window_size->{'height'}, $window_size->{'width'});
};
subtest 'Play sound on the circulation page' => sub {
push @cleanup, $patron, $patron->category, $patron->library;
};
+subtest 'Display circulation table correctly' => sub {
+ plan tests => 1;
+
+ my $builder = t::lib::TestBuilder->new;
+ my $library = $builder->build_object( { class => 'Koha::Libraries' } );
+ my $patron = $builder->build_object(
+ {
+ class => 'Koha::Patrons',
+ value => { branchcode => $library->branchcode, flags => 0 }
+ }
+ );
+
+ my ( $biblionumber, $biblioitemnumber ) = add_biblio();
+ my $item = $builder->build_sample_item(
+ {
+ biblionumber => $biblionumber,
+ library => $library->branchcode,
+ }
+ );
+ my $context = Test::MockModule->new('C4::Context');
+ $context->mock(
+ 'userenv',
+ sub {
+ return { branch => $library->branchcode };
+ }
+ );
+
+ C4::Circulation::AddIssue( $patron->unblessed, $item->barcode );
+
+ my $mainpage = $s->base_url . q|mainpage.pl|;
+ $driver->get($mainpage . q|?logout.x=1|);
+ $s->auth;
+
+ $driver->get( $base_url
+ . "/circ/circulation.pl?borrowernumber="
+ . $patron->borrowernumber );
+
+ # Display the table clicking on the "Show checkouts" button
+ $driver->find_element('//a[@id="issues-table-load-now-button"]')->click;
+
+ my @thead_th = $driver->find_elements('//table[@id="issues-table"]/thead/tr/th');
+ my $thead_length = 0;
+ $thead_length += $_->get_attribute('colspan', 1) || 0 for @thead_th;
+
+ my @tfoot_td = $driver->find_elements('//table[@id="issues-table"]/tfoot/tr/td');
+ my $tfoot_length = 0;
+ $tfoot_length += $_->get_attribute('colspan', 1) || 0 for @tfoot_td;
+
+ my @tbody_td = $driver->find_elements('//table[@id="issues-table"]/tbody/tr[2]/td');
+ my $tbody_length = 0;
+ $tbody_length += 1 for @tbody_td;
+
+ is( $thead_length == $tfoot_length && $tfoot_length == $tbody_length,
+ 1, "Checkouts table must be correctly aligned" )
+ or diag(
+ "thead: $thead_length ; tfoot: $tfoot_length ; tbody: $tbody_length");
+
+ push @cleanup, $patron->checkouts, $item, $item->biblio, $patron,
+ $patron->category, $library;
+};
+
+subtest 'XSS vulnerabilities in pagination' => sub {
+ plan tests => 3;
+
+ my $patron = $builder->build_object({ class => 'Koha::Patrons' });
+ for ( 1 .. 30 ) { # We want the pagination to be displayed
+ push @cleanup, $builder->build_object(
+ {
+ class => 'Koha::Virtualshelves',
+ value => {
+ category => 2,
+ allow_change_from_owner => 1,
+ allow_change_from_others => 0,
+ owner => $patron->borrowernumber
+ }
+ }
+ );
+ }
+
+ my $password = Koha::AuthUtils::generate_password($patron->category);
+ t::lib::Mocks::mock_preference( 'RequireStrongPassword', 0 );
+ $patron->set_password({ password => $password });
+ $s->opac_auth( $patron->userid, $password );
+
+ my $public_lists = $s->opac_base_url . q|opac-shelves.pl?op=list&category=2|;
+ $driver->get($public_lists);
+
+ $s->remove_error_handler;
+ my $alert_text = eval { $driver->get_alert_text() };
+ $s->add_error_handler;
+ is( $alert_text, undef, 'No alert box displayed' );
+
+ my $booh_alert = 'booh!';
+ $public_lists = $s->opac_base_url . qq|opac-shelves.pl?op=list&category=2"><script>alert('$booh_alert')</script>|;
+ $driver->get($public_lists);
+
+ $s->remove_error_handler;
+ $alert_text = eval { $driver->get_alert_text() };
+ $s->add_error_handler;
+ is( $alert_text, undef, 'No alert box displayed, even if evil intent' );
+
+ my $second_page = $driver->find_element('//div[@class="pages"]/span[@class="currentPage"]/following-sibling::a');
+ like( $second_page->get_attribute('href'), qr{category=2%22%3E%3Cscript%3Ealert%28%27booh%21%27%29%3C%2Fscript%3E}, 'The second page should display the variables and attributes correctly URI escaped' );
+
+ push @cleanup, $patron, $patron->category, $patron->library;
+
+ $driver->quit();
+};
+
END {
- C4::Context->preference('SearchEngine', $SearchEngine_value);
- C4::Context->preference('AudioAlerts', $AudioAlerts_value);
+ C4::Context->set_preference('SearchEngine', $SearchEngine_value);
+ C4::Context->set_preference('AudioAlerts', $AudioAlerts_value);
$_->delete for @cleanup;
};
+
+sub add_biblio {
+ my ($title, $author) = @_;
+
+ my $marcflavour = C4::Context->preference('marcflavour');
+
+ my $biblio = MARC::Record->new();
+ my ( $tag, $code );
+ $tag = $marcflavour eq 'UNIMARC' ? '200' : '245';
+ $biblio->append_fields(
+ MARC::Field->new($tag, ' ', ' ', a => $title || 'a title'),
+ );
+
+ ($tag, $code) = $marcflavour eq 'UNIMARC' ? (200, 'f') : (100, 'a');
+ $biblio->append_fields(
+ MARC::Field->new($tag, ' ', ' ', $code => $author || 'an author'),
+ );
+
+ return C4::Biblio::AddBiblio($biblio, '');
+}