/*************************************************************************/
/*  test_string.cpp                                                      */
/*************************************************************************/
/*                       This file is part of:                           */
/*                           GODOT ENGINE                                */
/*                    http://www.godotengine.org                         */
/*************************************************************************/
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                 */
/*                                                                       */
/* Permission is hereby granted, free of charge, to any person obtaining */
/* a copy of this software and associated documentation files (the       */
/* "Software"), to deal in the Software without restriction, including   */
/* without limitation the rights to use, copy, modify, merge, publish,   */
/* distribute, sublicense, and/or sell copies of the Software, and to    */
/* permit persons to whom the Software is furnished to do so, subject to */
/* the following conditions:                                             */
/*                                                                       */
/* The above copyright notice and this permission notice shall be        */
/* included in all copies or substantial portions of the Software.       */
/*                                                                       */
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,       */
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF    */
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY  */
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,  */
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE     */
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                */
/*************************************************************************/
#include "ustring.h"
#include <wchar.h>
//#include "math_funcs.h"
#include <stdio.h>
#include "os/os.h"
#include "drivers/trex/regex.h"

#include "test_string.h"

namespace TestString {

bool test_1() {
	
	OS::get_singleton()->print("\n\nTest 1: Assign from cstr\n");
	
	String s = "Hello";
	
	OS::get_singleton()->print("\tExpected: Hello\n");
	OS::get_singleton()->print("\tResulted: %ls\n",s.c_str());
	
	return (wcscmp(s.c_str(),L"Hello")==0);
	
}

bool test_2() {
	
	OS::get_singleton()->print("\n\nTest 2: Assign from string (operator=)\n");
	
	String s = "Dolly";
	String t = s;
	
	OS::get_singleton()->print("\tExpected: Dolly\n");
	OS::get_singleton()->print("\tResulted: %ls\n",t.c_str());
	
	return (wcscmp(t.c_str(),L"Dolly")==0);
		
}

bool test_3() {
	
	OS::get_singleton()->print("\n\nTest 3: Assign from c-string (copycon)\n");
	
	String s("Sheep");
	String t(s);
	
	OS::get_singleton()->print("\tExpected: Sheep\n");
	OS::get_singleton()->print("\tResulted: %ls\n",t.c_str());
	
	return (wcscmp(t.c_str(),L"Sheep")==0);
		
}

bool test_4() {
	
	OS::get_singleton()->print("\n\nTest 4: Assign from c-widechar (operator=)\n");
	
	String s(L"Give me");
	
	OS::get_singleton()->print("\tExpected: Give me\n");
	OS::get_singleton()->print("\tResulted: %ls\n",s.c_str());
	
	return (wcscmp(s.c_str(),L"Give me")==0);
		
}

bool test_5() {
	
	OS::get_singleton()->print("\n\nTest 5: Assign from c-widechar (copycon)\n");
	
	String s(L"Wool");
	
	OS::get_singleton()->print("\tExpected: Wool\n");
	OS::get_singleton()->print("\tResulted: %ls\n",s.c_str());
	
	return (wcscmp(s.c_str(),L"Wool")==0);
		
}

bool test_6() {
	
	OS::get_singleton()->print("\n\nTest 6: comparisons (equal)\n");
	
	
	String s="Test Compare";
	
	OS::get_singleton()->print("\tComparing to \"Test Compare\"\n");
	
	if (! ( s=="Test Compare" ) )
		return false;
	
	if (! ( s==L"Test Compare" ) )
		return false;
	
	if (! ( s==String("Test Compare") ) )
		return false;
	
	return true;
		
}

bool test_7() {
	
	OS::get_singleton()->print("\n\nTest 7: comparisons (unequal)\n");
	
	
	String s="Test Compare";
	
	OS::get_singleton()->print("\tComparing to \"Test Compare\"\n");
	
	if (! ( s!="Peanut" ) )
		return false;
	
	if (! ( s!=L"Coconut" ) )
		return false;
	
	if (! ( s!=String("Butter") ) )
		return false;
	
	return true;
		
}

bool test_8() {
	
	OS::get_singleton()->print("\n\nTest 8: comparisons (operator<)\n");
	
	
	String s="Bees";
	
	OS::get_singleton()->print("\tComparing to \"Bees\"\n");
	
	if ( ! (s < "Elephant")  )
		return false;
	
	if ( s < L"Amber"  )
		return false;
	
	if ( s < String("Beatrix") )
		return false;
	
	return true;
		
}

bool test_9() {
	
	OS::get_singleton()->print("\n\nTest 9: Concatenation\n");
	
	
	String s;
	
	s+="Have";
	s+=' ';
	s+='a';
	s+=String(" ");
	s = s + L"Nice";
	s = s + " ";
	s = s + String("Day");
	
	OS::get_singleton()->print("\tComparing to \"Have a Nice Day\"\n");
	
	return (s == "Have a Nice Day");
		
}

bool test_10() {
	
	OS::get_singleton()->print("\n\nTest 10: Misc funcs (size/length/empty/etc)\n");
	
	if (! String("").empty())
		return false;
	
	if (String("Mellon").size() != 7)
		return false;

	if (String("Oranges").length() != 7)
		return false;
	
	return true;
		
}


bool test_11() {
	
	OS::get_singleton()->print("\n\nTest 11: Operator[]\n");
	
	String a="Kugar Sane";
	
	a[0]='S';
	a[6]='C';
	
	if (a != "Sugar Cane")
		return false;
	
	if (a[1]!='u')
		return false;
	
	return true;
		
}

bool test_12() {
	
	OS::get_singleton()->print("\n\nTest 12: case functions\n");
	
	
	String a="MoMoNgA";
	
	if (a.to_upper() != "MOMONGA")
		return false;
	
	if (a.nocasecmp_to("momonga")!=0)
		return false;
	
	return true;
		
}

bool test_13() {
	
	OS::get_singleton()->print("\n\nTest 13: UTF8\n");
	
	/* how can i embed UTF in here? */
	
	static const CharType ustr[] = { 0x304A , 0x360F, 0x3088, 0x3046, 0 };
//	static const wchar_t ustr[] = { 'P', 0xCE, 'p',0xD3, 0 };
	String s=ustr;
	
	OS::get_singleton()->print("\tUnicode: %ls\n",ustr);
	s.parse_utf8( s.utf8().get_data() );
	OS::get_singleton()->print("\tConvert/Parse UTF8: %ls\n",s.c_str());
	
	return (s==ustr);
		
}

bool test_14() {
	
	OS::get_singleton()->print("\n\nTest 14: ASCII\n");
	
	String s = L"Primero Leche";
	OS::get_singleton()->print("\tAscii: %s\n",s.ascii().get_data());
	
	String t=s.ascii().get_data();
	return (s==t);
		
}

bool test_15() {
	
	OS::get_singleton()->print("\n\nTest 15: substr\n");
	
	String s="Killer Baby";
	OS::get_singleton()->print("\tsubstr(3,4) of \"%ls\" is \"%ls\"\n",s.c_str(),s.substr(3,4).c_str());
	
	return (s.substr(3,4)=="ler ");
		
}

bool test_16() {
	
	OS::get_singleton()->print("\n\nTest 16: find\n");
	
	String s="Pretty Woman";
	OS::get_singleton()->print("\tString: %ls\n",s.c_str());
	OS::get_singleton()->print("\t\"tty\" is at %i pos.\n",s.find("tty"));
	OS::get_singleton()->print("\t\"Revenge of the Monster Truck\" is at %i pos.\n",s.find("Revenge of the Monster Truck"));
	
	if (s.find("tty")!=3)
		return false;
	
	if (s.find("Revenge of the Monster Truck")!=-1)
		return false;
	
	return true;
		
}

bool test_17() {
	
	OS::get_singleton()->print("\n\nTest 17: find no case\n");
	
	String s="Pretty Whale";
	OS::get_singleton()->print("\tString: %ls\n",s.c_str());
	OS::get_singleton()->print("\t\"WHA\" is at %i pos.\n",s.findn("WHA"));
	OS::get_singleton()->print("\t\"Revenge of the Monster SawFish\" is at %i pos.\n",s.findn("Revenge of the Monster Truck"));
	
	if (s.findn("WHA")!=7)
		return false;
	
	if (s.findn("Revenge of the Monster SawFish")!=-1)
		return false;
	
	return true;
		
}

bool test_18() {
	
	OS::get_singleton()->print("\n\nTest 18: find no case\n");
	
	String s="Pretty Whale";
	OS::get_singleton()->print("\tString: %ls\n",s.c_str());
	OS::get_singleton()->print("\t\"WHA\" is at %i pos.\n",s.findn("WHA"));
	OS::get_singleton()->print("\t\"Revenge of the Monster SawFish\" is at %i pos.\n",s.findn("Revenge of the Monster Truck"));
	
	if (s.findn("WHA")!=7)
		return false;
	
	if (s.findn("Revenge of the Monster SawFish")!=-1)
		return false;
	
	return true;
		
}

bool test_19() {
	
	OS::get_singleton()->print("\n\nTest 19: Search & replace\n");
	
	String s="Happy Birthday, Anna!";
	OS::get_singleton()->print("\tString: %ls\n",s.c_str());
	
	s=s.replace("Birthday","Halloween");
	OS::get_singleton()->print("\tReplaced Birthday/Halloween: %ls.\n",s.c_str());
	
	return (s=="Happy Halloween, Anna!");
		
}

bool test_20() {
	
	OS::get_singleton()->print("\n\nTest 20: Insertion\n");
	
	String s="Who is Frederic?";
	
	OS::get_singleton()->print("\tString: %ls\n",s.c_str());
	s=s.insert( s.find("?")," Chopin" );
	OS::get_singleton()->print("\tInserted Chopin: %ls.\n",s.c_str());
	
	return (s=="Who is Frederic Chopin?");
		
}

bool test_21() {
	
	OS::get_singleton()->print("\n\nTest 21: Number -> String\n");
	
	OS::get_singleton()->print("\tPi is %f\n",33.141593);
	OS::get_singleton()->print("\tPi String is %ls\n",String::num(3.141593).c_str());
	
	return String::num(3.141593)=="3.141593";
		
}

bool test_22() {
	
	OS::get_singleton()->print("\n\nTest 22: String -> Int\n");
	
	static const char* nums[4]={ "1237461283", "- 22", "0", " - 1123412" };
	static const int num[4]={ 1237461283, -22, 0,  -1123412 };
	
	for (int i=0;i<4;i++) {
		OS::get_singleton()->print("\tString: \"%s\" as Int is %i\n",nums[i],String(nums[i]).to_int());
		
		if (String(nums[i]).to_int()!=num[i])
			return false;
	}
	
	return true;
		
}

bool test_23() {
	
	OS::get_singleton()->print("\n\nTest 23: String -> Float\n");
	
	static const char* nums[4]={ "-12348298412.2", "0.05", "2.0002", " -0.0001" };
	static const double num[4]={ -12348298412.2, 0.05, 2.0002, -0.0001 };
	
	for (int i=0;i<4;i++) {
		OS::get_singleton()->print("\tString: \"%s\" as Float is %f\n",nums[i],String(nums[i]).to_double());
		
		if ( ABS(String(nums[i]).to_double()-num[i])>0.00001)
			return false;
	}
	
	return true;
		
}


bool test_24() {
	
	OS::get_singleton()->print("\n\nTest 24: Slicing\n");
	
	String s="Mars,Jupiter,Saturn,Uranus";
	
	const char*slices[4]={"Mars","Jupiter","Saturn","Uranus"};
	
	OS::get_singleton()->print("\tSlicing \"%ls\" by \"%s\"..\n",s.c_str(),",");
	
	for (int i=0;i<s.get_slice_count(",");i++) {
		
		OS::get_singleton()->print("\t\t%i- %ls\n",i+1,s.get_slice(",",i).c_str());
		
		
		if (s.get_slice(",",i)!=slices[i])
			return false;
	}
	
	return true;
		
}

bool test_25() {
	
	OS::get_singleton()->print("\n\nTest 25: Erasing\n");
	
	String s="Josephine is such a cute girl!";
	
	OS::get_singleton()->print("\tString: %ls\n",s.c_str());
	OS::get_singleton()->print("\tRemoving \"cute\"\n");
	
	s.erase(s.find("cute "),String("cute ").length());
	OS::get_singleton()->print("\tResult: %ls\n",s.c_str());
	
	
	return (s=="Josephine is such a girl!");
		
}

bool test_26() {

	OS::get_singleton()->print("\n\nTest 26: RegEx\n");
	RegEx regexp("(.*):(.*)");
	List<String> captures;

	bool match = regexp.match("name:password", &captures);
	printf("\tmatch: %s\n", match?"true":"false");

	printf("\t%i captures:\n", captures.size());
	List<String>::Element *I = captures.front();
	while (I) {

		printf("%ls\n", I->get().c_str());

		I = I->next();
	};
	return captures.size();
};

struct test_27_data {
	char const * data;
	char const * begin;
	bool expected;
};

bool test_27() {

	OS::get_singleton()->print("\n\nTest 27: begins_with\n");
	test_27_data tc[] = {
		{"res://foobar", "res://", true},
		{"res", "res://", false},
		{"abc", "abc", true}
	};
	size_t count = sizeof(tc) / sizeof(tc[0]);
	bool state = true;
	for (size_t i = 0;state && i < count; ++i) {
		String s = tc[i].data;
		state = s.begins_with(tc[i].begin) == tc[i].expected;
		if (state) {
			String sb = tc[i].begin;
			state = s.begins_with(sb) == tc[i].expected;
		}
		if (!state) {
			OS::get_singleton()->print("\n\t Failure on:\n\t\tstring: ", tc[i].data, "\n\t\tbegin: ", tc[i].begin, "\n\t\texpected: ", tc[i].expected ? "true" : "false", "\n");
			break;
		}
	};
	return state;
};


bool test_28() {

	OS::get_singleton()->print("\n\nTest 28: sprintf\n");

	bool success, state = true;
	char output_format[] = "\tTest:\t%ls => %ls (%s)\n";
	String format, output;
	Array args;
	bool error;
	
	// %%
	format = "fish %% frog";
	args.clear();
	output = format.sprintf(args, &error);
	success = (output == String("fish % frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	//////// INTS

	// Int
	format = "fish %d frog";
	args.clear();
	args.push_back(5);
	output = format.sprintf(args, &error);
	success = (output == String("fish 5 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Int left padded with zeroes.
	format = "fish %05d frog";
	args.clear();
	args.push_back(5);
	output = format.sprintf(args, &error);
	success = (output == String("fish 00005 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Int left padded with spaces.
	format = "fish %5d frog";
	args.clear();
	args.push_back(5);
	output = format.sprintf(args, &error);
	success = (output == String("fish     5 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Int right padded with spaces.
	format = "fish %-5d frog";
	args.clear();
	args.push_back(5);
	output = format.sprintf(args, &error);
	success = (output == String("fish 5     frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Int with sign (positive).
	format = "fish %+d frog";
	args.clear();
	args.push_back(5);
	output = format.sprintf(args, &error);
	success = (output == String("fish +5 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Negative int.
	format = "fish %d frog";
	args.clear();
	args.push_back(-5);
	output = format.sprintf(args, &error);
	success = (output == String("fish -5 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Hex (lower)
	format = "fish %x frog";
	args.clear();
	args.push_back(45);
	output = format.sprintf(args, &error);
	success = (output == String("fish 2d frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Hex (upper)
	format = "fish %X frog";
	args.clear();
	args.push_back(45);
	output = format.sprintf(args, &error);
	success = (output == String("fish 2D frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Octal
	format = "fish %o frog";
	args.clear();
	args.push_back(99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 143 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	////// REALS

	// Real
	format = "fish %f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 99.990000 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real left-padded
	format = "fish %11f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish   99.990000 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real right-padded
	format = "fish %-11f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 99.990000   frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real given int.
	format = "fish %f frog";
	args.clear();
	args.push_back(99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 99.000000 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real with sign (positive).
	format = "fish %+f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish +99.990000 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real with 1 decimals.
	format = "fish %.1f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 100.0 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real with 12 decimals.
	format = "fish %.12f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 99.990000000000 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Real with no decimals.
	format = "fish %.f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish 100 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	/////// Strings.

	// String
	format = "fish %s frog";
	args.clear();
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == String("fish cheese frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// String left-padded
	format = "fish %10s frog";
	args.clear();
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == String("fish     cheese frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// String right-padded
	format = "fish %-10s frog";
	args.clear();
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == String("fish cheese     frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	///// Characters

	// Character as string.
	format = "fish %c frog";
	args.clear();
	args.push_back("A");
	output = format.sprintf(args, &error);
	success = (output == String("fish A frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Character as int.
	format = "fish %c frog";
	args.clear();
	args.push_back(65);
	output = format.sprintf(args, &error);
	success = (output == String("fish A frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	///// Dynamic width

	// String dynamic width
	format = "fish %*s frog";
	args.clear();
	args.push_back(10);
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == String("fish     cheese frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Int dynamic width
	format = "fish %*d frog";
	args.clear();
	args.push_back(10);
	args.push_back(99);
	output = format.sprintf(args, &error);
	success = (output == String("fish         99 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Float dynamic width
	format = "fish %*.*f frog";
	args.clear();
	args.push_back(10);
	args.push_back(3);
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == String("fish     99.990 frog") && !error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	///// Errors

	// More formats than arguments.
	format = "fish %s %s frog";
	args.clear();
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == "not enough arguments for format string" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// More arguments than formats.
	format = "fish %s frog";
	args.clear();
	args.push_back("hello");
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == "not all arguments converted during string formatting" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Incomplete format.
	format = "fish %10";
	args.clear();
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == "incomplete format" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Bad character in format string
	format = "fish %&f frog";
	args.clear();
	args.push_back("cheese");
	output = format.sprintf(args, &error);
	success = (output == "unsupported format character" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Too many decimals.
	format = "fish %2.2.2f frog";
	args.clear();
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == "too many decimal points in format" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// * not a number
	format = "fish %*f frog";
	args.clear();
	args.push_back("cheese");
	args.push_back(99.99);
	output = format.sprintf(args, &error);
	success = (output == "* wants number" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Character too long.
	format = "fish %c frog";
	args.clear();
	args.push_back("sc");
	output = format.sprintf(args, &error);
	success = (output == "%c requires number or single-character string" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	// Character bad type.
	format = "fish %c frog";
	args.clear();
	args.push_back(Array());
	output = format.sprintf(args, &error);
	success = (output == "%c requires number or single-character string" && error);
	OS::get_singleton()->print(output_format, format.c_str(), output.c_str(), success ? "OK" : "FAIL");
	if (!success) state = false;

	return state;
}

typedef bool (*TestFunc)(void);

TestFunc test_funcs[] = {
	
	test_1,
	test_2,
	test_3,
	test_4,
	test_5,
	test_6,
	test_7,
	test_8,
	test_9,
	test_10,
	test_11,
	test_12,
	test_13,
	test_14,
	test_15,
	test_16,
	test_17,
	test_18,
	test_19,
	test_20,
	test_21,
	test_22,
	test_23,
	test_24,
	test_25,
	test_26,
	test_27,
	test_28,
	0
	
};

MainLoop* test() {
	
	/** A character length != wchar_t may be forced, so the tests wont work */
	
	ERR_FAIL_COND_V( sizeof(CharType) != sizeof(wchar_t), NULL );
	
	int count=0;
	int passed=0;
	
	while(true) {
		if (!test_funcs[count])
			break;
		bool pass=test_funcs[count]();
		if (pass)
			passed++;
		OS::get_singleton()->print("\t%s\n",pass?"PASS":"FAILED");
		
		count++;
	}
	
	OS::get_singleton()->print("\n\n\n");
	OS::get_singleton()->print("*************\n");
	OS::get_singleton()->print("***TOTALS!***\n");
	OS::get_singleton()->print("*************\n");
	
	OS::get_singleton()->print("Passed %i of %i tests\n",count,passed);
	
	return NULL;
}

}