#!/usr/bin/perl -w

use strict;

use lib ('./blib','../blib','../lib','./lib');
use Class::NamedParms;

my @do_tests=(1..9);

my $test_subs = { 
       1 => { -code => \&test_constructors,            -desc => 'constructors           ' },
       2 => { -code => \&test_set,                     -desc => 'set                    ' },
       3 => { -code => \&test_get,                     -desc => 'get                    ' },
       4 => { -code => \&test_clear,                   -desc => 'clear                  ' },
       5 => { -code => \&test_undeclare,               -desc => 'undeclare              ' },
       6 => { -code => \&test_list_declared_parms,     -desc => 'list_declared_parms    ' },
       7 => { -code => \&test_list_initialized_parms,  -desc => 'list_initialized_parms ' },
       8 => { -code => \&test_all_parms,               -desc => 'all_parms              ' },
       9 => { -code => \&test_exists,                  -desc => 'exists                 ' },
};
run_tests($test_subs);

exit;

###########################################################################
###########################################################################

########################################
# constructors                         #
########################################
sub test_constructors {
    eval {
        my $class = Class::NamedParms->new;
    };
    if ($@) {
        return 'direct notation constructor CLASS->new; failed';
    }

    eval {
        my $class = new Class::NamedParms;
    };
    if ($@) {
        return 'indirect direct notation constructor new CLASS; failed';
    }

    eval {
        my $class = Class::NamedParms::new;
    };
    if ($@) {
        return 'Class function constructor CLASS::new; failed';
    }

    eval {
        my $class = Class::NamedParms->new;
        my $nother = $class->new;
    };
    if ($@) {
        return 'Derived instance constructor $instance->new; failed';
    }

    return '';
}

########################################
# set                                  #
########################################
sub test_set {
	eval {
		my $class = Class::NamedParms->new(-testing);
		$class->set;
	};
	if ($@) {
		return "setting parameter using null list failed: $@";
    }

	eval {
		my $class = Class::NamedParms->new(-testing);
		$class->set({ -testing => 1 });
	};
	if ($@) {
		return "setting parameter via anon hash ref failed: $@";
	}

	eval {
		my $class = Class::NamedParms->new(-testing);
		$class->set( -testing => 1 );
	};
	if ($@) {
		return "setting parameter via straight list failed: $@";
	}
	eval {
		my $class = Class::NamedParms->new(-testing);
		$class->set({ -bad_boy => 1 });
	};
	unless ($@) {
		return "setting parameter failed to catch mis-specified parm";
	}
	return '';
}

########################################
# get                                  #
########################################
sub test_get {
	my $class = Class::NamedParms->new(qw(-testing -misc -other));
	my $test_value = '1.0056';
	$class->set({ -testing => $test_value, -misc => 'stuff', '-other' => 'more stuff' });
	eval {
		my $value = $class->get(-testing);
		if ($value ne $test_value) {
			return "value returned was not the same as value set: $@";
		}
        my @list_results = $class->get('-testing','-misc','-other');
        if ($#list_results == 2) {
            unless (($list_results[0] eq $test_value)
                and ($list_results[1] eq 'stuff')
                and ($list_results[2] eq 'more stuff')) {
                return 'returned results are not in correct order';
            }
        } else {
            return 'unexpected number of returned results from get in list context';
        }
        my $item = $class->get('-testing','-other','-misc');
        unless ($item eq 'stuff') {
            return 'failed to return last requested item in a scalar context';
        }
        eval {
            my $value = $class->get;
        };
        unless ($@) {
            return 'failed to catch get with no parameters'
        }
	};
	if ($@) {
		return $@;
	}
	return '';
}

########################################
# exists;                           #
########################################
sub test_exists {
	eval {
        my $test_values = {
            -testing  => 'a',
            -misc     => 'b',
        };
        my @initial_parms = qw(-testing -misc);
		my $class = Class::NamedParms->new(@initial_parms);
		$class->set($test_values);
        foreach my $item (@initial_parms) {
            unless ($class->exists($item)) {
                die("exists failed for $item");
            }
        }
        unless (not $class->exists('nosuchparm')) {
            die("exists gave false positive for undeclared parm");
        }
	};

	if ($@) {
		return $@;
	}
	return '';
}

########################################
# all_parms;                           #
########################################
sub test_all_parms {
	eval {
        my $test_values = {
            -testing  => 'a',
            -misc     => 'b',
        };
        my @initial_parms = keys %$test_values;
		my $class = Class::NamedParms->new(qw(-testing -misc -extra));
		$class->set($test_values);
        my $parms_hash = $class->all_parms;
        my @initialized_parms = keys %$parms_hash;
        if ($#initialized_parms != $#initial_parms) {
            die ("incorrect number of parms returned by all_parms");
        }
        foreach my $item (@initial_parms) {
            unless (exists $parms_hash->{$item}) {
                die("unexpected key returned by all_parms");
            }
        }
	};

	if ($@) {
		return $@;
	}
	return '';
}

########################################
# list_initialized_parms;              #
########################################
sub test_list_initialized_parms {
	eval {
        my $test_values = {
            -testing => 'a',
        };
        my @initial_parms = keys %$test_values;
		my $class = Class::NamedParms->new(qw(-testing -misc));
		$class->set($test_values);
        my @initialized_parms = $class->list_initialized_parms;
        if ($#initialized_parms != $#initial_parms) {
            die ("incorrect number of parms returned by list_initialized_parms");
        }
        my %parms_hash = map { $_ => 1 } @initialized_parms;
        foreach my $item (@initial_parms) {
            unless (exists $parms_hash{$item}) {
                die("unexpected key returned by list_initialized_parms");
            }
        }
	};

	if ($@) {
		return $@;
	}
	return '';
}

########################################
# list_declared_parms;                 #
########################################
sub test_list_declared_parms {
	eval {
        my $test_values = {
            -testing => 'a',
            -misc    => 'b',
        };
        my @initial_parms = keys %$test_values;
		my $class = Class::NamedParms->new(@initial_parms);
        my @declared_parms = $class->list_declared_parms;
        if ($#declared_parms != $#initial_parms) {
            die ("incorrect number of parms returned by list_declared_parms");
        }
        my %parms_hash = map { $_ => 1 } @declared_parms;
        foreach my $item (@initial_parms) {
            eval {
                my $value = $class->get($item);
            };
            unless ($@) {
                die("failed to detect use of declared, but not initialized, parm");
            }
        }
	};

	if ($@) {
		return $@;
	}
	return '';
}

########################################
# undeclare                            #
########################################
sub test_undeclare {
	eval {
		my $class = Class::NamedParms->new(qw(-testing -misc));
		my $test_value0 = '1.0054';
        my $test_value1 = 'a';
		$class->set({ -testing => $test_value0, -misc => $test_value1 });
		my $value = $class->get(-testing);
		if ($value ne $test_value0) {
			return "value returned was not the same as value set\n";
		}
		$class->undeclare(-testing);
        eval {
		    my $new_value = $class->get(-testing);
        };
        unless ($@) {
            die("failed to undeclare key");
		}
        eval {
		    $class->undeclare(-testing);
        };
        unless ($@) {
            die("failed to catch 'undeclare' on a never declared key");
		}
	};

	if ($@) {
		return $@;
	}
	return '';
}

########################################
# clear                                #
########################################
sub test_clear {
	eval {
		my $class = Class::NamedParms->new(-testing);
		my $test_value = '1.0054';
		$class->set({ -testing => $test_value });
		my $value = $class->get(-testing);
		if ($value ne $test_value) {
			return "value returned was not the same as value set\n";
		}
		$class->clear(-testing);
		my $new_value = $class->get(-testing);
		if (defined $new_value) {
			return "failed to clear value\n";
		}
        eval {
		    $class->clear(-other);
        };
        unless ($@) {
            return 'failed to catch clearing of an undeclared parm';
        }
	};
	if ($@) {
		return $@;
	}
	'';
}

##########################################
##########################################

sub run_tests {
    my ($tests) = @_;
    print $do_tests[0],'..',$do_tests[$#do_tests],"\n";
    print STDERR "\n";
    my $n_failures = 0;
    foreach my $test (@do_tests) {
    	my $sub  = $test_subs->{$test}->{-code};
    	my $desc = $test_subs->{$test}->{-desc};
    	my $failure = '';
    	eval { $failure = &$sub; };
    	if ($@) {
    		$failure = $@;
    	}
    	if ($failure ne '') {
    		chomp $failure;
    		print "not ok $test\n";
    		print STDERR "    $desc - $failure\n";
    		$n_failures++;
    	} else {
    		print "ok $test\n";
    		print STDERR "    $desc - ok\n";
    
    	}
    }
    print "END\n";
    return;
}


syntax highlighted by Code2HTML, v. 0.9.1