Bug 17600: Standardize our EXPORT_OK
[srvgit] / t / db_dependent / selenium / regressions.t
old mode 100644 (file)
new mode 100755 (executable)
index 4c6cd99..220345f
 # along with Koha; if not, see <http://www.gnu.org/licenses>.
 
 use Modern::Perl;
+use utf8;
 
 use C4::Context;
 
-use Test::More tests => 3;
+use Test::More tests => 7;
+use Test::MockModule;
 
 use C4::Context;
+use C4::Biblio qw( AddBiblio );
+use C4::Circulation qw( AddIssue );
 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 $@;
+skip "Selenium::Remote::Driver is needed for selenium tests.", 7 if $@;
 
 my $s = t::lib::Selenium->new;
 
@@ -44,27 +50,54 @@ my $AudioAlerts_value = C4::Context->preference('AudioAlerts');
 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
@@ -78,21 +111,24 @@ subtest 'OPAC - Remove from cart' => sub {
     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 {
@@ -113,8 +149,220 @@ 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(&|$)}, 'The second page should display category without the invalid value' );
+
+    push @cleanup, $patron, $patron->category, $patron->library;
+
+};
+
+subtest 'Encoding in session variables' => sub {
+    plan tests => 18;
+
+    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 $biblio = $builder->build_sample_biblio;
+    my $item = $builder->build_sample_item(
+        {
+            biblionumber => $biblio->biblionumber,
+            library      => $library->branchcode,
+        }
+    );
+
+    my $original_SessionStorage = C4::Context->preference('SessionStorage');
+    for my $SessionStorage ( qw( memcached mysql tmp ) ) {
+        C4::Context->set_preference( 'SessionStorage', $SessionStorage );
+        for my $branchname (qw( Test1 Test2❤️ Test3ä )) {
+            my $library =
+              Koha::Libraries->find($branchname) || $builder->build_object(
+                {
+                    class => 'Koha::Libraries',
+                    value => {
+                        branchcode => $branchname,
+                        branchname => $branchname,
+                    }
+                }
+              );
+            # Make sure we are logged in
+            $driver->get( $base_url . q|mainpage.pl?logout.x=1| );
+            $s->auth;
+            # Switch to the new library
+            $driver->get( $base_url . 'circ/set-library.pl' );
+            $s->fill_form( { branch => $branchname } );
+            $s->submit_form;
+            # Check an item out
+            $driver->get( $base_url
+                  . 'circ/circulation.pl?borrowernumber='
+                  . $patron->borrowernumber );
+            # We must have the logged-in-branch-name displayed, or we got a 500
+            is(
+                $driver->find_element( '//span[@class="logged-in-branch-name"]')->get_text(),
+                $branchname,
+                sprintf( "logged-in-branch-name set - SessionStorage=%s, branchname=%s", $SessionStorage, $branchname
+                )
+            );
+
+            $driver->find_element('//input[@id="barcode"]')->send_keys( $item->barcode );
+            $driver->find_element('//fieldset[@id="circ_circulation_issue"]/button[@type="submit"]')->click;
+
+            # Display the table clicking on the "Show checkouts" button
+            $driver->find_element('//a[@id="issues-table-load-now-button"]')
+              ->click;
+
+            my @tds = $driver->find_elements(
+                '//table[@id="issues-table"]/tbody/tr[2]/td');
+
+            # Select the td for "Checked out from" (FIXME this is not robust and could be improved
+            my $td_checked_out_from = $tds[8];
+            is(
+                $td_checked_out_from->get_text(),
+                $branchname,
+                sprintf( "'Checked out from' column should contain the branchname - SessionStorage=%s, branchname=%s", $SessionStorage, $branchname )
+            );
+
+            # Remove the check in
+            Koha::Checkouts->find({ itemnumber => $item->itemnumber })->delete;
+        }
+    }
+
+    C4::Context->set_preference('SessionStorage', $original_SessionStorage);
+    push @cleanup, $item, $biblio, $patron, $patron->category, $patron->library;
+    push @cleanup, Koha::Libraries->find($_) for qw( Test1 Test2❤️ Test3ä );
+
+};
+
+$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, '');
+}