Some more Etemplate tests

This commit is contained in:
nathangray 2017-08-23 11:20:49 -06:00
parent 97543138f8
commit b0cd55f0b6
9 changed files with 533 additions and 13 deletions

View File

@ -344,7 +344,7 @@ class Etemplate extends Etemplate\Widget\Template
{ {
error_log(__METHOD__."(,".array2string($_content).') validation_errors='.array2string(self::$validation_errors)); error_log(__METHOD__."(,".array2string($_content).') validation_errors='.array2string(self::$validation_errors));
self::$response->generic('et2_validation_error', self::$validation_errors); self::$response->generic('et2_validation_error', self::$validation_errors);
exit; return;
} }
// tell request call to remove request, if it is not modified eg. by call to exec in callback // tell request call to remove request, if it is not modified eg. by call to exec in callback

View File

@ -889,7 +889,7 @@ class Widget
public static function set_validation_error($name,$error,$cname=null) public static function set_validation_error($name,$error,$cname=null)
{ {
// not yet used: if (is_null($cname)) $cname = self::$name_vars; // not yet used: if (is_null($cname)) $cname = self::$name_vars;
error_log(__METHOD__."('$name','$error','$cname') ".function_backtrace()); //error_log(__METHOD__."('$name','$error','$cname') ".function_backtrace());
if ($cname) $name = self::form_name($cname,$name); if ($cname) $name = self::form_name($cname,$name);

View File

@ -120,7 +120,7 @@ class Textbox extends Etemplate\Widget
{ {
if (!isset($this->attrs['validator'])) if (!isset($this->attrs['validator']))
{ {
switch($this->type) switch($this->attrs['type'])
{ {
case 'int': case 'int':
case 'integer': case 'integer':
@ -158,7 +158,7 @@ class Textbox extends Etemplate\Widget
} }
if ($this->attrs['validator'] && !preg_match($this->attrs['validator'],$value)) if ($this->attrs['validator'] && !preg_match($this->attrs['validator'],$value))
{ {
switch($this->type) switch($this->attrs['type'])
{ {
case 'integer': case 'integer':
self::set_validation_error($form_name,lang("'%1' is not a valid integer !!!",$value),''); self::set_validation_error($form_name,lang("'%1' is not a valid integer !!!",$value),'');
@ -171,21 +171,21 @@ class Textbox extends Etemplate\Widget
break; break;
} }
} }
elseif ($this->type == 'integer' || $this->type == 'float') // cast int and float and check range elseif ($this->attrs['type'] == 'integer' || $this->attrs['type'] == 'float') // cast int and float and check range
{ {
if ((string)$value !== '' || $this->attrs['needed']) // empty values are Ok if needed is not set if ((string)$value !== '' || $this->attrs['needed']) // empty values are Ok if needed is not set
{ {
$value = $this->type == 'integer' ? (int) $value : (float) str_replace(',','.',$value); // allow for german (and maybe other) format $value = $this->attrs['type'] == 'integer' ? (int) $value : (float) str_replace(',','.',$value); // allow for german (and maybe other) format
if (!empty($this->attrs['min']) && $value < $this->attrs['min']) if (!empty($this->attrs['min']) && $value < $this->attrs['min'])
{ {
self::set_validation_error($form_name,lang("Value has to be at least '%1' !!!",$this->attrs['min']),''); self::set_validation_error($form_name,lang("Value has to be at least '%1' !!!",$this->attrs['min']),'');
$value = $this->type == 'integer' ? (int) $this->attrs['min'] : (float) $this->attrs['min']; $value = $this->attrs['type'] == 'integer' ? (int) $this->attrs['min'] : (float) $this->attrs['min'];
} }
if (!empty($this->attrs['max']) && $value > $this->attrs['max']) if (!empty($this->attrs['max']) && $value > $this->attrs['max'])
{ {
self::set_validation_error($form_name,lang("Value has to be at maximum '%1' !!!",$this->attrs['max']),''); self::set_validation_error($form_name,lang("Value has to be at maximum '%1' !!!",$this->attrs['max']),'');
$value = $this->type == 'integer' ? (int) $this->attrs['max'] : (float) $this->attrs['max']; $value = $this->attrs['type'] == 'integer' ? (int) $this->attrs['max'] : (float) $this->attrs['max'];
} }
} }
} }

View File

@ -0,0 +1,197 @@
<?php
/**
* Test for float textboxes
*
* @link http://www.egroupware.org
* @author Nathan Gray
* @package api
* @copyright (c) 2017 Nathan Gray
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
*/
namespace EGroupware\Api\Etemplate\Widget;
require_once realpath(__DIR__.'/../../test/WidgetBaseTest.php');
use EGroupware\Api\Etemplate;
class FloatTest extends \EGroupware\Api\Etemplate\WidgetBaseTest {
const TEST_TEMPLATE = 'api.float_test';
/**
* Test for validation - floats
*
*
* @dataProvider floatProvider
*/
public function testFloat($value, $expected, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello'
);
$this->validateRoundTrip($etemplate, $content, array('widget' => $value),
$error ? array() : array('widget' => $expected),
$error ? array('widget' => $error) : array()
);
}
/**
* Data provider for float tests
*/
public function floatProvider()
{
return array(
// User value, Expected Error
array('', '', false),
array(1, 1, false),
array(0, 0, false),
array(-1, -1, false),
array(1.5, 1.5, false),
array('1,5', 1.5, false), // Comma as separator is handled
array('one', '', true)
);
}
/**
* Test for float minimum attribute
*
* @param String|numeric $value
* @param float $min Minimum allowed value
* @param boolean $error
*
* @dataProvider minProvider
*/
public function testMin($value, $min, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello',
'widget_readonly' => 'World'
);
$result = $this->mockedExec($etemplate, $content, array(), array(), array());
// Only lowercase
$etemplate->getElementById('widget')->attrs['min'] = $min;
// Check for the load
$data = array();
foreach($result as $command)
{
if($command['type'] == 'et2_load')
{
$data = $command['data'];
break;
}
}
// 'Edit' the data client side
$data['data']['content'] = array('widget' => $value);
// Let it validate
Etemplate::ajax_process_content($data['data']['etemplate_exec_id'], $data['data']['content'], false);
$content = static::$mocked_exec_result;
static::$mocked_exec_result = array();
return $this->validateTest($content,
$error ? array() : array('widget' => $value),
$error ? array('widget' => $error) : array()
);
}
public function minProvider()
{
return Array(
// User value, Min, Error
array('', 0, FALSE),
array(1.0, 0, FALSE),
array(0.0, 0, FALSE),
array(-1.0, 0, TRUE),
array(1.5, 0, FALSE),
array(1, 10, FALSE),
array(10, 10, FALSE),
array(1.5, 1.5, FALSE),
array(2, 1.5, TRUE),
);
}
/**
* Test for float maximum attribute
*
* @param String|numeric $value
* @param float $max Maximum allowed value
* @param boolean $error
*
* @dataProvider maxProvider
*/
public function testMax($value, $max, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello',
'widget_readonly' => 'World'
);
$result = $this->mockedExec($etemplate, $content, array(), array(), array());
// Only lowercase
$etemplate->getElementById('widget')->attrs['max'] = $max;
// Check for the load
$data = array();
foreach($result as $command)
{
if($command['type'] == 'et2_load')
{
$data = $command['data'];
break;
}
}
// 'Edit' the data client side
$data['data']['content'] = array('widget' => $value);
// Let it validate
Etemplate::ajax_process_content($data['data']['etemplate_exec_id'], $data['data']['content'], false);
$content = static::$mocked_exec_result;
static::$mocked_exec_result = array();
return $this->validateTest($content,
$error ? array() : array('widget' => $value),
$error ? array('widget' => $error) : array()
);
}
public function maxProvider()
{
return Array(
// User value, Max, Error
array('', 0, FALSE),
array(1.0, 0, TRUE),
array(0, 0, FALSE),
array(-1.0, 0, FALSE),
array(1.5, 2, FALSE),
array(1, 10, FALSE),
array(10, 10, FALSE),
array(2.5, 2, TRUE),
array(1.5, 2.5, FALSE),
array(3, 2.5, FALSE),
);
}
}

View File

@ -0,0 +1,192 @@
<?php
/**
* Test for integer textboxes
*
* @link http://www.egroupware.org
* @author Nathan Gray
* @package api
* @copyright (c) 2017 Nathan Gray
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
*/
namespace EGroupware\Api\Etemplate\Widget;
require_once realpath(__DIR__.'/../../test/WidgetBaseTest.php');
use EGroupware\Api\Etemplate;
class IntegerTest extends \EGroupware\Api\Etemplate\WidgetBaseTest {
const TEST_TEMPLATE = 'api.integer_test';
/**
* Test for validation - integers
*
*
* @dataProvider integerProvider
*/
public function testInteger($value, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello',
'widget_readonly' => 'World'
);
$this->validateRoundTrip($etemplate, $content, array('widget' => $value),
$error ? array() : array('widget' => $value),
$error ? array('widget' => $error) : array()
);
}
/**
* Data provider for integer tests
*/
public function integerProvider()
{
return array(
// User value, Error
array('', false),
array(1, false),
array(0, false),
array(-1, false),
array(1.5, true),
array('one', true)
);
}
/**
* Test for integer minimum attribute
*
* @param String|numeric $value
* @param int $min Minimum allowed value
* @param boolean $error
*
* @dataProvider minProvider
*/
public function testMin($value, $min, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello',
'widget_readonly' => 'World'
);
$result = $this->mockedExec($etemplate, $content, array(), array(), array());
// Only lowercase
$etemplate->getElementById('widget')->attrs['min'] = $min;
// Check for the load
$data = array();
foreach($result as $command)
{
if($command['type'] == 'et2_load')
{
$data = $command['data'];
break;
}
}
// 'Edit' the data client side
$data['data']['content'] = array('widget' => $value);
// Let it validate
Etemplate::ajax_process_content($data['data']['etemplate_exec_id'], $data['data']['content'], false);
$content = static::$mocked_exec_result;
static::$mocked_exec_result = array();
return $this->validateTest($content,
$error ? array() : array('widget' => $value),
$error ? array('widget' => $error) : array()
);
}
public function minProvider()
{
return Array(
// User value, Min, Error
array('', 0, FALSE),
array(1, 0, FALSE),
array(0, 0, FALSE),
array(-1, 0, TRUE),
array(1.5, 0, TRUE), // Errors because it's not an int
array(1, 10, TRUE),
array(10, 10, FALSE),
);
}
/**
* Test for integer maximum attribute
*
* @param String|numeric $value
* @param int $max Maximum allowed value
* @param boolean $error
*
* @dataProvider maxProvider
*/
public function testMax($value, $max, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello',
'widget_readonly' => 'World'
);
$result = $this->mockedExec($etemplate, $content, array(), array(), array());
// Only lowercase
$etemplate->getElementById('widget')->attrs['max'] = $max;
// Check for the load
$data = array();
foreach($result as $command)
{
if($command['type'] == 'et2_load')
{
$data = $command['data'];
break;
}
}
// 'Edit' the data client side
$data['data']['content'] = array('widget' => $value);
// Let it validate
Etemplate::ajax_process_content($data['data']['etemplate_exec_id'], $data['data']['content'], false);
$content = static::$mocked_exec_result;
static::$mocked_exec_result = array();
return $this->validateTest($content,
$error ? array() : array('widget' => $value),
$error ? array('widget' => $error) : array()
);
}
public function maxProvider()
{
return Array(
// User value, Max, Error
array('', 0, FALSE),
array(1, 0, TRUE),
array(0, 0, FALSE),
array(-1, 0, FALSE),
array(1.5, 2, TRUE), // Errors because it's not an int
array(1, 10, FALSE),
array(10, 10, FALSE),
);
}
}

View File

@ -17,11 +17,6 @@ require_once realpath(__DIR__.'/../../test/WidgetBaseTest.php');
use EGroupware\Api\Etemplate; use EGroupware\Api\Etemplate;
/**
* Description of TextboxTest
*
* @author nathan
*/
class TextboxTest extends \EGroupware\Api\Etemplate\WidgetBaseTest class TextboxTest extends \EGroupware\Api\Etemplate\WidgetBaseTest
{ {
@ -29,6 +24,7 @@ class TextboxTest extends \EGroupware\Api\Etemplate\WidgetBaseTest
/** /**
* Test the widget's basic functionallity - we put data in, it comes back * Test the widget's basic functionallity - we put data in, it comes back
* unchanged.
*/ */
public function testBasic() public function testBasic()
{ {
@ -117,4 +113,60 @@ class TextboxTest extends \EGroupware\Api\Etemplate\WidgetBaseTest
$this->assertEquals(array(), $content); $this->assertEquals(array(), $content);
} }
/**
* Test regex validation
*
* @dataProvider regexProvider
*/
public function testRegex($value, $error)
{
// Instanciate the template
$etemplate = new Etemplate();
$etemplate->read(static::TEST_TEMPLATE, 'test');
// Content - doesn't really matter, we're changing it
$content = array(
'widget' => 'Hello',
'widget_readonly' => 'World'
);
$result = $this->mockedExec($etemplate, $content, array(), array(), array());
// Only lowercase
$etemplate->getElementById('widget')->attrs['validator'] = '/[a-z]*$/';
// Check for the load
$data = array();
foreach($result as $command)
{
if($command['type'] == 'et2_load')
{
$data = $command['data'];
break;
}
}
// 'Edit' the data client side
$data['data']['content'] = array('widget' => $value);
// Let it validate
Etemplate::ajax_process_content($data['data']['etemplate_exec_id'], $data['data']['content'], false);
$content = static::$mocked_exec_result;
static::$mocked_exec_result = array();
return $this->validateTest($content, array('widget' => $value), $error ? array('widget' => $error) : array());
}
public function regexProvider()
{
return array(
// Value Errors
array('', FALSE),
array('Hello', TRUE),
array('hello', FALSE),
array(1234, TRUE),
array('hi1234',TRUE)
);
}
} }

View File

@ -145,4 +145,66 @@ abstract class WidgetBaseTest extends \EGroupware\Api\LoggedInTest {
}); });
return $response; return $response;
} }
/**
* Exec the template with the provided content, change the values according to
* $set_values, then validate against $expected_values
*
* @param \EGroupware\Api\Etemplate $etemplate
* @param array $content
* @param array $set_values
* @param array $expected_values
* @param array $validation_errors
*/
protected function validateRoundTrip(\EGroupware\Api\Etemplate $etemplate, Array $content, Array $set_values, Array $expected_values = null, Array $validation_errors = array())
{
if(is_null($expected_values))
{
$expected_values = $set_values;
}
$result = $this->mockedExec($etemplate, $content, array(), array(), array());
// Check for the load
$data = array();
foreach($result as $command)
{
if($command['type'] == 'et2_load')
{
$data = $command['data'];
break;
}
}
// 'Edit' the data client side
$data['data']['content'] = $set_values;
// Let it validate
Etemplate::ajax_process_content($data['data']['etemplate_exec_id'], $data['data']['content'], false);
$content = static::$mocked_exec_result;
static::$mocked_exec_result = array();
return $this->validateTest($content, $expected_values, $validation_errors);
}
protected function validateTest($content, $expected_values, $validation_errors)
{
// Make validation errors accessible
$ref = new \ReflectionProperty('\\EGroupware\\Api\\Etemplate\\Widget', 'validation_errors');
$ref->setAccessible(true);
$errors = $ref->getValue();
// Test values
foreach($expected_values as $widget_id => $value)
{
$this->assertEquals($value, $content[$widget_id], 'Widget "' . $widget_id . '" did not get expected value');
}
// Check validation errors
foreach($validation_errors as $widget_id => $errored)
{
$this->assertTrue(array_key_exists($widget_id, $validation_errors), "Widget $widget_id caused a validation error");
}
$ref->setValue(array());
}
} }

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE overlay PUBLIC "-//EGroupware GmbH//eTemplate 2//EN" "http://www.egroupware.org/etemplate2.dtd">
<!-- This template is used in automated testing -->
<overlay>
<template id="api.float_test" template="" lang="" group="0" version="16.1">
<float id="widget"/>
</template>
</overlay>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE overlay PUBLIC "-//EGroupware GmbH//eTemplate 2//EN" "http://www.egroupware.org/etemplate2.dtd">
<!-- This template is used in automated testing -->
<overlay>
<template id="api.integer_test" template="" lang="" group="0" version="16.1">
<integer id="widget"/>
<integer id="widget_readonly" readonly="true"/>
</template>
</overlay>