WWW::Webrobot::UserAgentConnection - create and configure a user agent


webrobot documentation Contained in the webrobot distribution.

Index


Code Index:

NAME

Top

WWW::Webrobot::UserAgentConnection - create and configure a user agent

SYNOPSIS

Top

 WWW::Webrobot::UserAgentConnection -> new($cfg, user => $user);

DESCRIPTION

Top

Helper class.

METHODS

Top

WWW::Webrobot::UserAgentConnection -> new ($cfg, %opt)
 $cfg
        Config, see L<WWW::Webrobot::pod::Config>
 %opt
        user => "an id for a user agent"

$conn -> ua

Get the user agent, see LWP::UserAgent

$conn -> cfg

Get the (internal) config data structure

$user -> treat_single_url ($arg, $sym_tbl)

$arg is an entry of a testplan, see WWW::Webrobot::pod::Testplan.

Returns the fail state


webrobot documentation Contained in the webrobot distribution.
package WWW::Webrobot::UserAgentConnection;
use strict;
use warnings;

# Author: Stefan Trcek
# Copyright(c) 2004-2006 ABAS Software AG


use HTTP::Cookies;
use HTTP::Request::Common;
use Time::HiRes;

use WWW::Webrobot::Attributes qw(ua cfg encoding);
use WWW::Webrobot::MyUserAgent;
use WWW::Webrobot::Ext::General::HTTP::Response;
use WWW::Webrobot::AssertConstant;
use WWW::Webrobot::MyEncode qw/has_Encode legacy_mode octet_to_encoding/;


sub new {
    my $proto = shift;
    my $class = ref($proto) || $proto;

    my ($cfg, %opt) = @_;

    my $str_agent = "Webrobot " . $WWW::Webrobot::VERSION;
    $str_agent .= " - $opt{user}" if $opt{user} ne "";

    # setup user agent
    my $cookie_jar = HTTP::Cookies -> new(File => "cookies_.txt", AutoSave => 0);
    my $ua = WWW::Webrobot::MyUserAgent -> new();
    $ua -> cookie_jar($cookie_jar);
    foreach (keys %{$cfg -> {proxy}}) {
        if (/^https$/) {
            # OpenSSL's proxy handling is incompatible to LWP's proxy handling
            my $https_proxy = $cfg->{proxy}->{$_};
            # yet more incompatible: remove trailing slash, bug in Crypt::SSLeay
            $https_proxy =~ s,/$,,;
            $ENV{HTTPS_PROXY} = $https_proxy;
        }
        else {
            $ua -> proxy($_, $cfg -> {proxy} -> {$_});
        }
    }
    $ua -> no_proxy(@{$cfg -> {no_proxy}}) if $cfg -> {no_proxy};
    $ua -> timeout($cfg -> {timeout} || 30);
    $ua -> agent($str_agent);
    $ua -> set_basic_realm($cfg -> {auth_basic}) if $cfg -> {auth_basic};
    $ua -> client_302_bug(1) if $cfg->{client_302_bug};

    my $self = {
        _ua => $ua,
        _cfg => $cfg,
    };
    bless ($self, $class);

    return $self;
}



#privat
sub norm_request {
    my ($self, $r) = @_;
    my $referrer = $self->ua->referrer();
    if ($referrer) {
        if ($self->cfg->{referrer_bug}) {
            $r -> headers() -> header(Referer => $referrer);
        }
        else {
            $r -> headers() -> referer($referrer);
        }
    }
    $self->ua->referrer($r->uri);
    return $r;
}

#static private
sub norm_response {
    my ($r) = @_;
    if (defined $r && ($r->protocol || "") eq 'HTTP/0.9' && ($r->message || "") eq 'EOF') {
        # ??? Dieses Verhalten sollte besser von einer Konfigurationsvariable
        # ??? abhaengig gemacht werden.
        $r->code(500);
        $r->protocol("HTTP/1.0");
        $r->message("Internal Server Error: unexpected EOF");
        $r->headers->{webrobot_orig_response} = "HTTP/0.9 200 EOF";
        $r->headers->{webrobot_message} = "converted to http code 500 by webrobot";
    }
    return $r;
}


sub convert_data {
    my ($uac, $input_data) = @_;
    my $encoding = $uac->encoding;
    return $input_data if !$encoding;
    if (has_Encode || legacy_mode) {
        my %data = ();
        foreach (keys %$input_data) {
            $data{octet_to_encoding($encoding, $_)} =
                octet_to_encoding($encoding, $input_data->{$_});
        }
        return \%data;
    }
    else {
        return $input_data;
    }
}

my %HTTP_ACTION = map {$_=>1} qw/HEAD GET POST PUT DELETE TRACE/;

my %ACTION = (
    NOP => sub {
        return undef;
    },
    HEAD => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        my %header = ( %{$uac->cfg->{http_header}}, %{$arg->{http_header}} );
        return norm_response($uac -> ua -> request($uac->norm_request(HEAD($arg->{url}, %header))));
    },
    GET => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        my %header = ( %{$uac->cfg->{http_header}}, %{$arg->{http_header}} );
        my $data = convert_data($uac, $arg->{data});
        my $url = $arg->{url};
        if ($data && scalar keys %$data) {
            my $tmp_request = POST($arg->{url}, $arg->{data}, %header);
            $url .= "?" . $tmp_request->content() if $tmp_request->content();
        }
        my $request = GET($url, %header);
        return norm_response($uac -> ua -> request($uac->norm_request($request)));
    },
    POST => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        my %header = ( %{$uac->cfg->{http_header}}, %{$arg->{http_header}} );
        my $data = convert_data($uac, $arg->{data});
        return norm_response($uac -> ua -> request($uac->norm_request(POST($arg->{url}, $data, %header))));
    },
    PUT => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        my %header = ( %{$uac->cfg->{http_header}}, %{$arg->{http_header}} );
        return norm_response($uac -> ua -> request($uac->norm_request(PUT($arg->{url}, $arg->{data}, %header))));
    },
    COOKIES => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        my $ua = $uac->ua;
        SWITCH: foreach ($arg->{url}) {
            m/^on$/i and do {
                my $cookie_jar = HTTP::Cookies -> new(File => "cookies.txt", AutoSave => 0);
                $ua->cookie_jar($cookie_jar);
                last;
            };
            m/^off$/i and do {
                $ua->cookie_jar(undef);
                last;
            };
            m/^clear$/i and do {
                if (my $cookies = $ua->cookie_jar()) {
                    $cookies->clear();
                }
                last;
            };
            m/^clear_temporary$/i and do {
                if (my $cookies = $ua->cookie_jar()) {
                    $cookies->clear_temporary_cookies();
                }
                last;
            };
        }
        return undef;
    },
    REFERRER => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        my $ua = $uac->ua;
        SWITCH: foreach ($arg->{url}) {
            m/^clear$/i and do {
                $ua->referrer("");
                last;
            };
            m/^on$/i and do {
                $ua->enable_referrer(1);
                last;
            };
            m/^off$/i and do {
                $ua->enable_referrer(0);
                last;
            };
        }
        return undef;
    },
    BASIC_REALM => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        $uac -> ua -> set_basic_realm($arg->{url});
        return undef;
    },
    CONFIG => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        eval {
            foreach my $tmp (@{$arg->{_mode}}) {
                my ($mode, $parm) = @$tmp;
                SWITCH: foreach ($mode) {
                    /^filename$/ || /^script$/ and do {
                        my $filename = $parm;
                        $filename .= " |" if /^script$/;
                        my $err_msg = /^script$/ ? "Can't start script" : "Can't read file";

                        my $handle = do {local *FH; *FH};
                        { # 'open' produces a warning if the shell script doesn't exist!
                            no warnings;
                            open $handle, "$filename" or die "$err_msg: '$parm'";
                        }
                        my $new_variables = WWW::Webrobot::Properties -> new() -> load_handle($handle) or
                            die "Can't read data from external program '$parm'";
                        my @new_vars = map { [$_, $new_variables->{$_} || ""] } sort keys %$new_variables;
                        $arg->{new_properties} = \@new_vars;
                        foreach (keys %$new_variables) {
                            $sym_tbl->define_symbol($_, $new_variables->{$_});
                        }
                        close $handle;
                        last SWITCH;
                    };
                    die "found $_ in \$arg->{_mode}, expected 'filename', 'script'";
                }
            }
        };
        my $err = $@;
        $arg->{assert} = new WWW::Webrobot::AssertConstant($err, $err ? ["0 $err"] : []);
        return undef;
    },
    SLEEP => sub {
        my ($uac, $arg, $sym_tbl) = @_;
        sleep($arg->{url});
        return undef;
    },
    "GLOBAL-ASSERTION" => sub {
        # This is the definition of the global assertion.
        # It has to be stored which has already been done,
        # so there is nothing to do.

        #my ($uac, $arg, $sym_tbl) = @_;
        return undef;
    },
);

sub check_assertion {
    my ($r, $all_assert) = @_;
    my $all_fail = 0;
    my @tmp = ();
    foreach my $assert (@$all_assert) {
        my ($fail, $fail_str) = $assert -> check($r);
        $all_fail = 1 if $fail;
        push @tmp, @$fail_str;
    }
    return ($all_fail, \@tmp);
}


sub treat_single_url {
    my ($self, $arg, $sym_tbl) = @_;
    #use Data::Dumper; print STDERR Dumper $sym_tbl;

    sleep($self->{_cfg}->{delay}) if $self->{_cfg}->{delay};

    $self -> {_ua} -> clear_redirect_fail();
    my ($r, $fail, $fail_str);
    $self->cfg->{http_header} ||= {}; # ??? really necessary?
    $arg->{data} ||= {}; # ??? really necessary?
    my $METHOD = $ACTION{$arg->{method}} or
        die "'$arg->{method}' is no valid method, expected: ", join ", ", keys %ACTION;

    # do test plan entry (usually HTTP request)
    my ($sec, $usec) = Time::HiRes::gettimeofday();
    eval {
        # NOTE: $r may be undef depending on $METHOD
        $r = $METHOD->($self, $arg, $sym_tbl);
    };
    my $exception = $@;
    my $elaps = Time::HiRes::tv_interval([$sec, $usec], [ Time::HiRes::gettimeofday() ]);
    $r->elapsed_time($elaps) if $r;

    # check result
    if ($self -> {_ua} -> is_redirect_fail()) {
        ($fail, $fail_str) = (0, []);
    }
    elsif ($exception) {
        $r = undef;
        ($fail, $fail_str) = (2, ["2 CALL TO METHOD '$arg->{method}', URL '$arg->{url}' FAILED: $exception"]);
    }
    elsif (! $r || ! defined $arg->{assert}) {
        # Method like COOKIES that don't support assertions
        ($fail, $fail_str) = (undef, []);
    }
    else {
        ($fail, $fail_str) = check_assertion($r, $arg->{assert});
    }

    # set encoding of response
    my $coding = undef;
    if ($HTTP_ACTION{$arg->{method}}) {
        if ($r and my $ct = $r->headers->{'content-type'}) {
            #'content-type' => 'text/plain; charset=utf-8',
            $coding = $1 if ($ct =~ m/.*;\s*charset\s*=\s*(.*)$/);
        }
    }
    $self->encoding($coding);

    if ($arg->{property}) {
        # evaluate new names
        my @new_vars = ();
        foreach (@{$arg->{property}}) {
            my ($mode, $name, $expr) = @$_;
            SWITCH: foreach ($mode) {
                /^value$/ and do {
                    push @new_vars, [$name, $expr];
                    last;
                };
                /^regex$/ and do {
                    next if ! $r;
                    my ($value) = $r->content =~ m/$expr/;
                    push @new_vars, [$name, $value];
                    last;
                };
                /^xpath$/ and do {
                    next if ! $r;
                    push @new_vars, [$name, $r->xpath($expr)];
                    last;
                };
                /^header$/ and do {
                    next if ! $r;
                    push @new_vars, [$name, $r->header($expr)];
                    last;
                };
                /^status$/ and do {
                    next if ! $r;
                    my %val = (
                        code => sub {$_[0]->code},
                        message => sub {$_[0]->message},
                        protocol => sub {$_[0]->protocol},
                    );
                    die("found status='$expr', expected " . join(", ", map {"'$_'"} keys %val)) if ! $val{$expr};
                    push @new_vars, [$name, $val{$expr}->($r)];
                    last;
                };
                /^random$/ and do {
                    next if ! $r;
                    $expr = 0 if $expr < 0;
                    $expr = 15 if $expr > 15;
                    push @new_vars, [$name, int rand 10**$expr];
                    last;
                };
                die "found attribute '$_', expected 'value', 'regex', 'xpath', 'header', 'status', 'random'";
            }
        }
        foreach (@new_vars) {
            $_->[1] ||= "";
            $sym_tbl -> define_symbol($_->[0], $_->[1]);
        }
        $arg->{new_properties} = \@new_vars;
    }

    return ($r, $fail, $fail_str);
}


1;