| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676 |
- use strict;
- use warnings;
- package Test::Selenium::Remote::Driver;
- use parent 'Selenium::Remote::Driver';
- # ABSTRACT: Useful testing subclass for Selenium::Remote::Driver
- use Test::Selenium::Remote::WebElement;
- use Test::More;
- use Test::Builder;
- use Test::LongString;
- use IO::Socket;
- our $AUTOLOAD;
- my $Test = Test::Builder->new;
- $Test->exported_to(__PACKAGE__);
- my %comparator = (
- is => 'is_eq',
- isnt => 'isnt_eq',
- like => 'like',
- unlike => 'unlike',
- );
- my $comparator_keys = join '|', keys %comparator;
- # These commands don't require a locator
- my %no_locator = map { $_ => 1 }
- qw( alert_text current_window_handle current_url
- title page_source body location path);
- sub no_locator {
- my $self = shift;
- my $method = shift;
- return $no_locator{$method};
- }
- sub AUTOLOAD {
- my $name = $AUTOLOAD;
- $name =~ s/.*:://;
- return if $name eq 'DESTROY';
- my $self = $_[0];
- my $sub;
- if ($name =~ /(\w+)_($comparator_keys)$/i) {
- my $getter = "get_$1";
- my $comparator = $comparator{lc $2};
- # make a subroutine that will call Test::Builder's test methods
- # with driver data from the getter
- if ($self->no_locator($1)) {
- $sub = sub {
- my( $self, $str, $name ) = @_;
- diag "Test::Selenium::Remote::Driver running no_locator $getter (@_[1..$#_])"
- if $self->{verbose};
- $name = "$getter, '$str'"
- if $self->{default_names} and !defined $name;
- no strict 'refs';
- my $rc = $Test->$comparator( $self->$getter, $str, $name );
- if (!$rc && $self->error_callback) {
- &{$self->error_callback}($name);
- }
- return $rc;
- };
- }
- else {
- $sub = sub {
- my( $self, $locator, $str, $name ) = @_;
- diag "Test::Selenium::Remote::Driver running with locator $getter (@_[1..$#_])"
- if $self->{verbose};
- $name = "$getter, $locator, '$str'"
- if $self->{default_names} and !defined $name;
- no strict 'refs';
- no strict 'refs';
- my $rc = $Test->$comparator( $self->$getter($locator), $str, $name );
- if (!$rc && $self->error_callback) {
- &{$self->error_callback}($name);
- }
- return $rc;
- };
- }
- }
- elsif ($name =~ /(\w+?)_?ok$/i) {
- my $cmd = $1;
- # make a subroutine for ok() around the selenium command
- $sub = sub {
- my $self = shift;
- my $name = pop;
- my ($arg1, $arg2) = @_;
- if ($self->{default_names} and !defined $name) {
- $name = $cmd;
- $name .= ", $arg1" if defined $arg1;
- $name .= ", $arg2" if defined $arg2;
- }
- diag "Test::Selenium::Remote::Driver running _ok $cmd (@_[1..$#_])"
- if $self->{verbose};
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $rc = '';
- eval { $rc = $self->$cmd( $arg1, $arg2 ) };
- die $@ if $@ and $@ =~ /Can't locate object method/;
- diag($@) if $@;
- $rc = ok( $rc, $name );
- if (!$rc && $self->error_callback) {
- &{$self->error_callback}($name);
- }
- return $rc;
- };
- }
- # jump directly to the new subroutine, avoiding an extra frame stack
- if ($sub) {
- no strict 'refs';
- *{$AUTOLOAD} = $sub;
- goto &$AUTOLOAD;
- }
- else {
- # try to pass through to Selenium::Remote::Driver
- my $sel = 'Selenium::Remote::Driver';
- my $sub = "${sel}::${name}";
- goto &$sub if exists &$sub;
- my ($package, $filename, $line) = caller;
- die qq(Can't locate object method "$name" via package ")
- . __PACKAGE__
- . qq(" (also tried "$sel") at $filename line $line\n);
- }
- }
- sub error_callback {
- my ($self, $cb) = @_;
- if (defined($cb)) {
- $self->{error_callback} = $cb;
- }
- return $self->{error_callback};
- }
- =head1 NAME
- Test::Selenium::Remote::Driver
- =head1 DESCRIPTION
- A subclass of L<Selenium::Remote::Driver>. which provides useful testing
- functions.
- This is an I<experimental> addition to the Selenium::Remote::Driver
- distribution, and some interfaces may change.
- =head1 Methods
- =head2 new ( %opts )
- This will create a new Test::Selenium::Remote::Driver object, which subclasses
- L<Selenium::Remote::Driver>. This subclass provides useful testing
- functions. It is modeled on L<Test::WWW::Selenium>.
- Environment vars can be used to specify options to pass to
- L<Selenium::Remote::Driver>. ENV vars are prefixed with C<TWD_>.
- ( After the old fork name, "Test::WebDriver" )
- Set the Selenium server address with C<$TWD_HOST> and C<$TWD_PORT>.
- Pick which browser is used using the C<$TWD_BROWSER>, C<$TWD_VERSION>,
- C<$TWD_PLATFORM>, C<$TWD_JAVASCRIPT>, C<$TWD_EXTRA_CAPABILITIES>.
- See L<Selenium::Driver::Remote> for the meanings of these options.
- =cut
- sub new {
- my ($class, %p) = @_;
- for my $opt (qw/remote_server_addr port browser_name version platform
- javascript auto_close extra_capabilities/) {
- $p{$opt} ||= $ENV{ 'TWD_' . uc($opt) };
- }
- $p{browser_name} ||= $ENV{TWD_BROWSER}; # ykwim
- $p{remote_server_addr} ||= $ENV{TWD_HOST}; # ykwim
- $p{webelement_class} ||= 'Test::Selenium::Remote::WebElement';
- my $self = $class->SUPER::new(%p);
- $self->{verbose} = $p{verbose};
- return $self;
- }
- =head2 server_is_running( $host, $port )
- Returns true if a Selenium server is running. The host and port
- parameters are optional, and default to C<localhost:4444>.
- Environment vars C<TWD_HOST> and C<TWD_PORT> can also be used to
- determine the server to check.
- =cut
- sub server_is_running {
- my $class_or_self = shift;
- my $host = $ENV{TWD_HOST} || shift || 'localhost';
- my $port = $ENV{TWD_PORT} || shift || 4444;
- return ($host, $port) if IO::Socket::INET->new(
- PeerAddr => $host,
- PeerPort => $port,
- );
- return;
- }
- =head1 Testing Methods
- The following testing methods are available. For
- more documentation, see the related test methods in L<Selenium::Remote::Driver>
- (And feel free to submit a patch to flesh out the documentation for these here).
- alert_text_is
- alert_text_isnt
- alert_text_like
- alert_text_unlike
- current_window_handle_is
- current_window_handle_isnt
- current_window_handle_like
- current_window_handle_unlike
- window_handles_is
- window_handles_isnt
- window_handles_like
- window_handles_unlike
- window_size_is
- window_size_isnt
- window_size_like
- window_size_unlike
- window_position_is
- window_position_isnt
- window_position_like
- window_position_unlike
- current_url_is
- current_url_isnt
- current_url_like
- current_url_unlike
- title_is
- title_isnt
- title_like
- title_unlike
- active_element_is
- active_element_isnt
- active_element_like
- active_element_unlike
- # Basically the same as 'content_like()', but content_like() supports multiple regex's.
- page_source_is
- page_source_isnt
- page_source_like
- page_source_unlike
- send_keys_to_active_element_ok
- send_keys_to_alert_ok
- send_keys_to_prompt_ok
- send_modifier_ok
- accept_alert_ok
- dismiss_alert_ok
- move_mouse_to_location_ok
- move_to_ok
- get_ok
- go_back_ok
- go_forward_ok
- add_cookie_ok
- get_page_source_ok
- find_element_ok($search_target)
- find_element_ok($search_target)
- find_elements_ok
- find_child_element_ok
- find_child_elements_ok
- compare_elements_ok
- click_ok
- double_click_ok
- =head2 $twd->type_ok($search_target, $keys, [, $desc ]);
- $twd->type_ok( $search_target, $keys [, $desc ] );
- Use L<Selenium::Remote::Driver/find_element> to resolve the C<$search_target>
- to a web element, and then type C<$keys> into it, providing an optional test
- label.
- Currently, other finders besides the default are not supported for C<type_ok()>.
- =cut
- sub type_ok {
- my $self = shift;
- my $locator = shift;
- my $keys = shift;
- my $desc = shift;
- return $self->find_element($locator)->send_keys_ok($keys,$desc);
- }
- =head2 $twd->find_element_ok($search_target [, $desc ]);
- $twd->find_element_ok( $search_target [, $desc ] );
- Returns true if C<$search_target> is successfully found on the page. L<$search_target>
- is passed to L<Selenium::Remote::Driver/find_element> using the C<default_finder>. See
- there for more details on the format. Currently, other finders besides the default are not supported
- for C<find_element_ok()>.
- =cut
- # Eventually, it would be nice to support other finds like Test::WWW::Selenium does, like this:
- # 'xpath=//foo', or 'css=.foo', etc.
- =head2 $twd->find_no_element_ok($search_target [, $desc ]);
- $twd->find_no_element_ok( $search_target [, $desc ] );
- Returns true if C<$search_target> is I<not> found on the page. L<$search_target>
- is passed to L<Selenium::Remote::Driver/find_element> using the C<default_finder>. See
- there for more details on the format. Currently, other finders besides the default are not supported
- for C<find_no_element_ok()>.
- =cut
- sub find_no_element_ok {
- my $self = shift;
- my $search_target = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level +1;
- eval { $self->find_element($search_target) };
- ok((defined $@),$desc);
- }
- =head2 $twd->content_like( $regex [, $desc ] )
- $twd->content_like( $regex [, $desc ] )
- $twd->content_like( [$regex_1, $regex_2] [, $desc ] )
- Tells if the content of the page matches I<$regex>. If an arrayref of regex's
- are provided, one 'test' is run for each regex against the content of the
- current page.
- A default description of 'Content is like "$regex"' will be provided if there
- is no description.
- =cut
- sub content_like {
- my $self = shift;
- my $regex = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $content = $self->get_page_source();
- if (not ref $regex eq 'ARRAY') {
- my $desc = qq{Content is like "$regex"} if (not defined $desc);
- return like_string($content , $regex, $desc );
- }
- elsif (ref $regex eq 'ARRAY') {
- for my $re (@$regex) {
- my $desc = qq{Content is like "$re"} if (not defined $desc);
- like_string($content , $re, $desc );
- }
- }
- }
- =head2 $twd->content_unlike( $regex [, $desc ] )
- $twd->content_unlike( $regex [, $desc ] )
- $twd->content_unlike( [$regex_1, $regex_2] [, $desc ] )
- Tells if the content of the page does NOT match I<$regex>. If an arrayref of regex's
- are provided, one 'test' is run for each regex against the content of the
- current page.
- A default description of 'Content is unlike "$regex"' will be provided if there
- is no description.
- =cut
- sub content_unlike {
- my $self = shift;
- my $regex = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $content = $self->get_page_source();
- if (not ref $regex eq 'ARRAY') {
- my $desc = qq{Content is unlike "$regex"} if (not defined $desc);
- return unlike_string($content , $regex, $desc );
- }
- elsif (ref $regex eq 'ARRAY') {
- for my $re (@$regex) {
- my $desc = qq{Content is unlike "$re"} if (not defined $desc);
- unlike_string($content , $re, $desc );
- }
- }
- }
- =head2 $twd->text_like( $regex [, $desc ] )
- $twd->text_like( $regex [, $desc ] )
- $twd->text_like( [$regex_1, $regex_2] [, $desc ] )
- Tells if the text of the page (as returned by C<< get_body() >>) matches
- I<$regex>. If an arrayref of regex's are provided, one 'test' is run for each
- regex against the content of the current page.
- A default description of 'Content is like "$regex"' will be provided if there
- is no description.
- To also match the HTML see, C<< content_unlike() >>.
- =cut
- sub text_like {
- my $self = shift;
- my $regex = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $text = $self->get_body();
- if (not ref $regex eq 'ARRAY') {
- my $desc = qq{Text is like "$regex"} if (not defined $desc);
- return like_string($text , $regex, $desc );
- }
- elsif (ref $regex eq 'ARRAY') {
- for my $re (@$regex) {
- my $desc = qq{Text is like "$re"} if (not defined $desc);
- like_string($text , $re, $desc );
- }
- }
- }
- =head2 $twd->text_unlike( $regex [, $desc ] )
- $twd->text_unlike( $regex [, $desc ] )
- $twd->text_unlike( [$regex_1, $regex_2] [, $desc ] )
- Tells if the text of the page (as returned by C<< get_body() >>)
- does NOT match I<$regex>. If an arrayref of regex's
- are provided, one 'test' is run for each regex against the content of the
- current page.
- A default description of 'Text is unlike "$regex"' will be provided if there
- is no description.
- To also match the HTML see, C<< content_unlike() >>.
- =cut
- sub text_unlike {
- my $self = shift;
- my $regex = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $text = $self->get_body();
- if (not ref $regex eq 'ARRAY') {
- my $desc = qq{Text is unlike "$regex"} if (not defined $desc);
- return unlike_string($text , $regex, $desc );
- }
- elsif (ref $regex eq 'ARRAY') {
- for my $re (@$regex) {
- my $desc = qq{Text is unlike "$re"} if (not defined $desc);
- unlike_string($text , $re, $desc );
- }
- }
- }
- #####
- =head2 $twd->content_contains( $str [, $desc ] )
- $twd->content_contains( $str [, $desc ] )
- $twd->content_contains( [$str_1, $str_2] [, $desc ] )
- Tells if the content of the page contains I<$str>. If an arrayref of strngs's
- are provided, one 'test' is run for each string against the content of the
- current page.
- A default description of 'Content contains "$str"' will be provided if there
- is no description.
- =cut
- sub content_contains {
- my $self = shift;
- my $str = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $content = $self->get_page_source();
- if (not ref $str eq 'ARRAY') {
- my $desc = qq{Content contains "$str"} if (not defined $desc);
- return contains_string($content , $str, $desc );
- }
- elsif (ref $str eq 'ARRAY') {
- for my $s (@$str) {
- my $desc = qq{Content contains "$s"} if (not defined $desc);
- contains_string($content , $s, $desc );
- }
- }
- }
- =head2 $twd->content_lacks( $str [, $desc ] )
- $twd->content_lacks( $str [, $desc ] )
- $twd->content_lacks( [$str_1, $str_2] [, $desc ] )
- Tells if the content of the page does NOT contain I<$str>. If an arrayref of strings
- are provided, one 'test' is run for each string against the content of the
- current page.
- A default description of 'Content lacks "$str"' will be provided if there
- is no description.
- =cut
- sub content_lacks {
- my $self = shift;
- my $str = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $content = $self->get_page_source();
- if (not ref $str eq 'ARRAY') {
- my $desc = qq{Content lacks "$str"} if (not defined $desc);
- return lacks_string($content , $str, $desc );
- }
- elsif (ref $str eq 'ARRAY') {
- for my $s (@$str) {
- my $desc = qq{Content lacks "$s"} if (not defined $desc);
- lacks_string($content , $s, $desc );
- }
- }
- }
- =head2 $twd->text_contains( $str [, $desc ] )
- $twd->text_contains( $str [, $desc ] )
- $twd->text_contains( [$str_1, $str_2] [, $desc ] )
- Tells if the text of the page (as returned by C<< get_body() >>) contains
- I<$str>. If an arrayref of strings are provided, one 'test' is run for each
- regex against the content of the current page.
- A default description of 'Text contains "$str"' will be provided if there
- is no description.
- To also match the HTML see, C<< content_uncontains() >>.
- =cut
- sub text_contains {
- my $self = shift;
- my $str = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $text = $self->get_body();
- if (not ref $str eq 'ARRAY') {
- my $desc = qq{Text contains "$str"} if (not defined $desc);
- return contains_string($text , $str, $desc );
- }
- elsif (ref $str eq 'ARRAY') {
- for my $s (@$str) {
- my $desc = qq{Text contains "$s"} if (not defined $desc);
- contains_string($text , $s, $desc );
- }
- }
- }
- =head2 $twd->text_lacks( $str [, $desc ] )
- $twd->text_lacks( $str [, $desc ] )
- $twd->text_lacks( [$str_1, $str_2] [, $desc ] )
- Tells if the text of the page (as returned by C<< get_body() >>)
- does NOT contain I<$str>. If an arrayref of strings
- are provided, one 'test' is run for each regex against the content of the
- current page.
- A default description of 'Text is lacks "$str"' will be provided if there
- is no description.
- To also match the HTML see, C<< content_lacks() >>.
- =cut
- sub text_lacks {
- my $self = shift;
- my $str = shift;
- my $desc = shift;
- local $Test::Builder::Level = $Test::Builder::Level + 1;
- my $text = $self->get_body();
- if (not ref $str eq 'ARRAY') {
- my $desc = qq{Text is lacks "$str"} if (not defined $desc);
- return lacks_string($text , $str, $desc );
- }
- elsif (ref $str eq 'ARRAY') {
- for my $s (@$str) {
- my $desc = qq{Text is lacks "$s"} if (not defined $desc);
- lacks_string($text , $s, $desc );
- }
- }
- }
- 1;
- __END__
- =head1 NOTES
- This module was forked from Test::WebDriver 0.01.
- For Best Practice - I recommend subclassing Test::Selenium::Remote::Driver for your application,
- and then refactoring common or app specific methods into MyApp::WebDriver so that
- your test files do not have much duplication. As your app changes, you can update
- MyApp::WebDriver rather than all the individual test files.
- =head1 AUTHORS
- =over 4
- =item *
- Created by: Luke Closs <lukec@cpan.org>, but inspired by
- L<Test::WWW::Selenium> and its authors.
- =back
- =head1 CONTRIBUTORS
- This work was sponsored by Prime Radiant, Inc. Mark Stosberg <mark@stosberg.com> also contributed.
- =head1 COPYRIGHT AND LICENSE
- Copyright (c) 2012 Prime Radiant, Inc.
- This program is free software; you can redistribute it and/or
- modify it under the same terms as Perl itself.
|