2008-08-09 04:53:34 +00:00
|
|
|
<?php
|
2013-05-11 18:05:53 +12:00
|
|
|
|
2016-10-14 14:30:05 +13:00
|
|
|
namespace SilverStripe\Dev\Tests;
|
2016-06-15 16:03:16 +12:00
|
|
|
|
2019-08-29 14:54:16 +12:00
|
|
|
use League\Csv\Writer;
|
2023-12-06 16:00:14 +13:00
|
|
|
use SilverStripe\Control\HTTPResponse_Exception;
|
2016-10-14 14:30:05 +13:00
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\CustomLoader;
|
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\Player;
|
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\PlayerContract;
|
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\Team;
|
2016-06-15 16:03:16 +12:00
|
|
|
use SilverStripe\ORM\DataObject;
|
|
|
|
use SilverStripe\ORM\FieldType\DBField;
|
2016-08-19 10:51:35 +12:00
|
|
|
use SilverStripe\Core\Config\Config;
|
|
|
|
use SilverStripe\Dev\CsvBulkLoader;
|
|
|
|
use SilverStripe\Dev\SapphireTest;
|
2023-12-06 16:00:14 +13:00
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\CanModifyModel;
|
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\CantCreateModel;
|
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\CantDeleteModel;
|
|
|
|
use SilverStripe\Dev\Tests\CsvBulkLoaderTest\CantEditModel;
|
2024-09-18 13:53:44 +12:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
2016-08-19 10:51:35 +12:00
|
|
|
|
2016-12-16 17:34:21 +13:00
|
|
|
class CsvBulkLoaderTest extends SapphireTest
|
|
|
|
{
|
|
|
|
|
|
|
|
protected static $fixture_file = 'CsvBulkLoaderTest.yml';
|
|
|
|
|
2020-04-20 18:58:09 +01:00
|
|
|
protected static $extra_dataobjects = [
|
2016-12-16 17:34:21 +13:00
|
|
|
Team::class,
|
|
|
|
Player::class,
|
|
|
|
PlayerContract::class,
|
2023-12-06 16:00:14 +13:00
|
|
|
CanModifyModel::class,
|
|
|
|
CantCreateModel::class,
|
|
|
|
CantEditModel::class,
|
|
|
|
CantDeleteModel::class,
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Name of csv test dir
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $csvPath = null;
|
|
|
|
|
2021-10-27 15:39:47 +13:00
|
|
|
protected function setUp(): void
|
2016-12-16 17:34:21 +13:00
|
|
|
{
|
|
|
|
parent::setUp();
|
|
|
|
$this->csvPath = __DIR__ . '/CsvBulkLoaderTest/csv/';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test plain import with column auto-detection
|
|
|
|
*/
|
|
|
|
public function testLoad()
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithHeader.csv';
|
2022-04-14 13:12:59 +12:00
|
|
|
$file = fopen($filepath ?? '', 'r');
|
2016-12-16 17:34:21 +13:00
|
|
|
$compareCount = $this->getLineCount($file);
|
|
|
|
fgetcsv($file); // pop header row
|
|
|
|
$compareRow = fgetcsv($file);
|
|
|
|
$results = $loader->load($filepath);
|
|
|
|
|
|
|
|
// Test that right amount of columns was imported
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertCount(5, $results, 'Test correct count of imported data');
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
// Test that columns were correctly imported
|
|
|
|
$obj = DataObject::get_one(
|
|
|
|
Player::class,
|
2020-04-20 18:58:09 +01:00
|
|
|
[
|
2016-12-16 17:34:21 +13:00
|
|
|
'"CsvBulkLoaderTest_Player"."FirstName"' => 'John'
|
2020-04-20 18:58:09 +01:00
|
|
|
]
|
2016-12-16 17:34:21 +13:00
|
|
|
);
|
|
|
|
$this->assertNotNull($obj);
|
|
|
|
$this->assertEquals("He's a good guy", $obj->Biography);
|
|
|
|
$this->assertEquals("1988-01-31", $obj->Birthday);
|
|
|
|
$this->assertEquals("1", $obj->IsRegistered);
|
|
|
|
|
|
|
|
fclose($file);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test plain import with clear_table_before_import
|
|
|
|
*/
|
|
|
|
public function testDeleteExistingRecords()
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithHeader.csv';
|
|
|
|
$loader->deleteExistingRecords = true;
|
|
|
|
$results1 = $loader->load($filepath);
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertCount(5, $results1, 'Test correct count of imported data on first load');
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
//delete existing data before doing second CSV import
|
|
|
|
$results2 = $loader->load($filepath);
|
|
|
|
//get all instances of the loaded DataObject from the database and count them
|
|
|
|
$resultDataObject = DataObject::get(Player::class);
|
|
|
|
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertCount(
|
2017-11-29 16:07:30 +13:00
|
|
|
5,
|
2018-02-21 20:22:37 +00:00
|
|
|
$resultDataObject,
|
2016-12-16 17:34:21 +13:00
|
|
|
'Test if existing data is deleted before new data is added'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-02-21 20:22:37 +00:00
|
|
|
public function testLeadingTabs()
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$loader->hasHeaderRow = false;
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->columnMap = [
|
2018-02-21 20:22:37 +00:00
|
|
|
'FirstName',
|
|
|
|
'Biography',
|
|
|
|
null, // ignored column
|
|
|
|
'Birthday',
|
|
|
|
'IsRegistered'
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2018-02-21 20:22:37 +00:00
|
|
|
$filepath = $this->csvPath . 'PlayersWithTabs.csv';
|
|
|
|
$results = $loader->load($filepath);
|
|
|
|
$this->assertCount(5, $results);
|
|
|
|
|
|
|
|
$expectedBios = [
|
|
|
|
"\tHe's a good guy",
|
|
|
|
"=She is awesome.\nSo awesome that she gets multiple rows and \"escaped\" strings in her biography",
|
|
|
|
"-Pretty old\, with an escaped comma",
|
|
|
|
"@Unicode FTW",
|
|
|
|
"+Unicode FTW",
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach (Player::get()->column('Biography') as $bio) {
|
|
|
|
$this->assertContains($bio, $expectedBios);
|
|
|
|
}
|
|
|
|
|
2022-04-14 13:12:59 +12:00
|
|
|
$this->assertEquals(Player::get()->count(), count($expectedBios ?? []));
|
2018-02-21 20:22:37 +00:00
|
|
|
}
|
|
|
|
|
2016-12-16 17:34:21 +13:00
|
|
|
/**
|
|
|
|
* Test import with manual column mapping
|
|
|
|
*/
|
|
|
|
public function testLoadWithColumnMap()
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'Players.csv';
|
2022-04-14 13:12:59 +12:00
|
|
|
$file = fopen($filepath ?? '', 'r');
|
2016-12-16 17:34:21 +13:00
|
|
|
$compareCount = $this->getLineCount($file);
|
|
|
|
$compareRow = fgetcsv($file);
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->columnMap = [
|
2016-12-16 17:34:21 +13:00
|
|
|
'FirstName',
|
|
|
|
'Biography',
|
|
|
|
null, // ignored column
|
|
|
|
'Birthday',
|
|
|
|
'IsRegistered'
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
$loader->hasHeaderRow = false;
|
|
|
|
$results = $loader->load($filepath);
|
|
|
|
|
|
|
|
// Test that right amount of columns was imported
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertCount(4, $results, 'Test correct count of imported data');
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
// Test that columns were correctly imported
|
|
|
|
$obj = DataObject::get_one(
|
|
|
|
Player::class,
|
2020-04-20 18:58:09 +01:00
|
|
|
[
|
2016-12-16 17:34:21 +13:00
|
|
|
'"CsvBulkLoaderTest_Player"."FirstName"' => 'John'
|
2020-04-20 18:58:09 +01:00
|
|
|
]
|
2016-12-16 17:34:21 +13:00
|
|
|
);
|
|
|
|
$this->assertNotNull($obj);
|
|
|
|
$this->assertEquals("He's a good guy", $obj->Biography);
|
|
|
|
$this->assertEquals("1988-01-31", $obj->Birthday);
|
|
|
|
$this->assertEquals("1", $obj->IsRegistered);
|
|
|
|
|
|
|
|
$obj2 = DataObject::get_one(
|
|
|
|
Player::class,
|
2020-04-20 18:58:09 +01:00
|
|
|
[
|
2016-12-16 17:34:21 +13:00
|
|
|
'"CsvBulkLoaderTest_Player"."FirstName"' => 'Jane'
|
2020-04-20 18:58:09 +01:00
|
|
|
]
|
2016-12-16 17:34:21 +13:00
|
|
|
);
|
|
|
|
$this->assertNotNull($obj2);
|
|
|
|
$this->assertEquals('0', $obj2->IsRegistered);
|
|
|
|
|
|
|
|
fclose($file);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test import with manual column mapping and custom column names
|
|
|
|
*/
|
|
|
|
public function testLoadWithCustomHeaderAndRelation()
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithCustomHeaderAndRelation.csv';
|
2022-04-14 13:12:59 +12:00
|
|
|
$file = fopen($filepath ?? '', 'r');
|
2016-12-16 17:34:21 +13:00
|
|
|
$compareCount = $this->getLineCount($file);
|
|
|
|
fgetcsv($file); // pop header row
|
|
|
|
$compareRow = fgetcsv($file);
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->columnMap = [
|
2016-12-16 17:34:21 +13:00
|
|
|
'first name' => 'FirstName',
|
|
|
|
'bio' => 'Biography',
|
|
|
|
'bday' => 'Birthday',
|
|
|
|
'teamtitle' => 'Team.Title', // test existing relation
|
|
|
|
'teamsize' => 'Team.TeamSize', // test existing relation
|
|
|
|
'salary' => 'Contract.Amount' // test relation creation
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
$loader->hasHeaderRow = true;
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->relationCallbacks = [
|
|
|
|
'Team.Title' => [
|
2016-12-16 17:34:21 +13:00
|
|
|
'relationname' => 'Team',
|
|
|
|
'callback' => 'getTeamByTitle'
|
2020-04-20 18:58:09 +01:00
|
|
|
],
|
2016-12-16 17:34:21 +13:00
|
|
|
// contract should be automatically discovered
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
$results = $loader->load($filepath);
|
|
|
|
|
|
|
|
// Test that right amount of columns was imported
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertCount(1, $results, 'Test correct count of imported data');
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
// Test of augumenting existing relation (created by fixture)
|
|
|
|
$testTeam = DataObject::get_one(Team::class, null, null, '"Created" DESC');
|
|
|
|
$this->assertEquals('20', $testTeam->TeamSize, 'Augumenting existing has_one relation works');
|
|
|
|
|
|
|
|
// Test of creating relation
|
|
|
|
$testContract = DataObject::get_one(PlayerContract::class);
|
|
|
|
$testPlayer = DataObject::get_one(
|
|
|
|
Player::class,
|
2020-04-20 18:58:09 +01:00
|
|
|
[
|
2016-12-16 17:34:21 +13:00
|
|
|
'"CsvBulkLoaderTest_Player"."FirstName"' => 'John'
|
2020-04-20 18:58:09 +01:00
|
|
|
]
|
2016-12-16 17:34:21 +13:00
|
|
|
);
|
|
|
|
$this->assertEquals($testPlayer->ContractID, $testContract->ID, 'Creating new has_one relation works');
|
|
|
|
|
|
|
|
// Test nested setting of relation properties
|
|
|
|
$contractAmount = DBField::create_field('Currency', $compareRow[5])->RAW();
|
|
|
|
$this->assertEquals(
|
|
|
|
$testPlayer->Contract()->Amount,
|
|
|
|
$contractAmount,
|
|
|
|
'Setting nested values in a relation works'
|
|
|
|
);
|
|
|
|
|
|
|
|
fclose($file);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test import with custom identifiers by importing the data.
|
|
|
|
*/
|
|
|
|
public function testLoadWithIdentifiers()
|
|
|
|
{
|
|
|
|
// first load
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithId.csv';
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->duplicateChecks = [
|
2016-12-16 17:34:21 +13:00
|
|
|
'ExternalIdentifier' => 'ExternalIdentifier',
|
|
|
|
'NonExistantIdentifier' => 'ExternalIdentifier',
|
|
|
|
'AdditionalIdentifier' => 'ExternalIdentifier'
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
$results = $loader->load($filepath);
|
|
|
|
$createdPlayers = $results->Created();
|
|
|
|
|
|
|
|
$player = $createdPlayers->first();
|
|
|
|
$this->assertEquals($player->FirstName, 'John');
|
|
|
|
$this->assertEquals(
|
|
|
|
$player->Biography,
|
|
|
|
'He\'s a good guy',
|
|
|
|
'test updating of duplicate imports within the same import works'
|
|
|
|
);
|
|
|
|
|
|
|
|
// load with updated data
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithIdUpdated.csv';
|
|
|
|
$results = $loader->load($filepath);
|
|
|
|
|
|
|
|
// HACK need to update the loaded record from the database
|
|
|
|
$player = DataObject::get_by_id(Player::class, $player->ID);
|
|
|
|
$this->assertEquals($player->FirstName, 'JohnUpdated', 'Test updating of existing records works');
|
|
|
|
|
|
|
|
// null values are valid imported
|
|
|
|
// $this->assertEquals($player->Biography, 'He\'s a good guy',
|
2017-12-14 13:50:52 +13:00
|
|
|
// 'Test retaining of previous information on duplicate when overwriting with blank field');
|
2016-12-16 17:34:21 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadWithCustomImportMethods()
|
|
|
|
{
|
|
|
|
$loader = new CustomLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithHeader.csv';
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->columnMap = [
|
2016-12-16 17:34:21 +13:00
|
|
|
'FirstName' => '->importFirstName',
|
|
|
|
'Biography' => 'Biography',
|
|
|
|
'Birthday' => 'Birthday',
|
|
|
|
'IsRegistered' => 'IsRegistered'
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
$results = $loader->load($filepath);
|
|
|
|
$createdPlayers = $results->Created();
|
|
|
|
$player = $createdPlayers->first();
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertEquals('Customized John', $player->FirstName);
|
|
|
|
$this->assertEquals("He's a good guy", $player->Biography);
|
|
|
|
$this->assertEquals("1", $player->IsRegistered);
|
2016-12-16 17:34:21 +13:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testLoadWithCustomImportMethodDuplicateMap()
|
|
|
|
{
|
|
|
|
$loader = new CustomLoader(Player::class);
|
|
|
|
$filepath = $this->csvPath . 'PlayersWithHeader.csv';
|
2020-04-20 18:58:09 +01:00
|
|
|
$loader->columnMap = [
|
2016-12-16 17:34:21 +13:00
|
|
|
'FirstName' => '->updatePlayer',
|
|
|
|
'Biography' => '->updatePlayer',
|
|
|
|
'Birthday' => 'Birthday',
|
|
|
|
'IsRegistered' => 'IsRegistered'
|
2020-04-20 18:58:09 +01:00
|
|
|
];
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
$results = $loader->load($filepath);
|
|
|
|
|
|
|
|
$createdPlayers = $results->Created();
|
|
|
|
$player = $createdPlayers->first();
|
|
|
|
|
|
|
|
$this->assertEquals($player->FirstName, "John. He's a good guy. ");
|
|
|
|
}
|
|
|
|
|
2019-08-29 14:54:16 +12:00
|
|
|
public function testLoadWithByteOrderMark()
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$loader->load($this->csvPath . 'PlayersWithHeaderAndBOM.csv');
|
|
|
|
|
|
|
|
$players = Player::get();
|
|
|
|
|
|
|
|
$this->assertCount(3, $players);
|
|
|
|
$this->assertListContains([
|
|
|
|
['FirstName' => 'Jamie', 'Birthday' => '1882-01-31'],
|
|
|
|
['FirstName' => 'Järg', 'Birthday' => '1982-06-30'],
|
|
|
|
['FirstName' => 'Jacob', 'Birthday' => '2000-04-30'],
|
|
|
|
], $players);
|
|
|
|
}
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
protected function getLineCount(&$file)
|
|
|
|
{
|
|
|
|
$i = 0;
|
|
|
|
while (fgets($file) !== false) {
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
rewind($file);
|
|
|
|
return $i;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testLargeFileSplitIntoSmallerFiles()
|
|
|
|
{
|
2017-07-17 17:59:40 +12:00
|
|
|
Config::modify()->set(CsvBulkLoader::class, 'lines', 3);
|
2016-12-16 17:34:21 +13:00
|
|
|
|
|
|
|
$loader = new CsvBulkLoader(Player::class);
|
|
|
|
$path = $this->csvPath . 'LargeListOfPlayers.csv';
|
|
|
|
|
|
|
|
$results = $loader->load($path);
|
|
|
|
|
2018-02-21 20:22:37 +00:00
|
|
|
$this->assertCount(10, $results);
|
2016-12-16 17:34:21 +13:00
|
|
|
}
|
2023-12-06 16:00:14 +13:00
|
|
|
|
2024-09-18 13:53:44 +12:00
|
|
|
#[DataProvider('provideCheckPermissions')]
|
2023-12-06 16:00:14 +13:00
|
|
|
public function testCheckPermissions(string $class, string $file, bool $respectPerms, string $exceptionMessage)
|
|
|
|
{
|
|
|
|
$loader = new CsvBulkLoader($class);
|
|
|
|
$loader->setCheckPermissions($respectPerms);
|
|
|
|
// Don't delete CantEditModel records, 'cause we need to explicitly edit them
|
|
|
|
$loader->deleteExistingRecords = $class !== CantEditModel::class;
|
|
|
|
// We can't rely on IDs in unit tests, so use Title as the unique field
|
|
|
|
$loader->duplicateChecks['Title'] = 'Title';
|
|
|
|
|
|
|
|
if ($exceptionMessage) {
|
|
|
|
$this->expectException(HTTPResponse_Exception::class);
|
|
|
|
$this->expectExceptionMessage($exceptionMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
$results = $loader->load($this->csvPath . $file);
|
|
|
|
|
|
|
|
// If there's no permission exception, we should get some valid results.
|
|
|
|
if (!$exceptionMessage) {
|
|
|
|
$this->assertCount(3, $results);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-18 13:53:44 +12:00
|
|
|
public static function provideCheckPermissions()
|
2023-12-06 16:00:14 +13:00
|
|
|
{
|
|
|
|
$scenarios = [
|
|
|
|
'Has all permissions' => [
|
|
|
|
'class' => CanModifyModel::class,
|
|
|
|
'file' => 'PermissionCheck.csv',
|
|
|
|
'respectPerms' => true,
|
|
|
|
'exceptionMessage' => '',
|
|
|
|
],
|
|
|
|
'No create permissions' => [
|
|
|
|
'class' => CantCreateModel::class,
|
|
|
|
'file' => 'PermissionCheck.csv',
|
|
|
|
'respectPerms' => true,
|
|
|
|
'exceptionMessage' => "Not allowed to create 'Cant Create Model' records",
|
|
|
|
],
|
|
|
|
'No edit permissions' => [
|
|
|
|
'class' => CantEditModel::class,
|
|
|
|
'file' => 'PermissionCheck.csv',
|
|
|
|
'respectPerms' => true,
|
|
|
|
'exceptionMessage' => "Not allowed to edit 'Cant Edit Model' records",
|
|
|
|
],
|
|
|
|
'No delete permissions' => [
|
|
|
|
'class' => CantDeleteModel::class,
|
|
|
|
'file' => 'PermissionCheck.csv',
|
|
|
|
'respectPerms' => true,
|
|
|
|
'exceptionMessage' => "Not allowed to delete 'Cant Delete Model' records",
|
|
|
|
],
|
|
|
|
];
|
|
|
|
foreach ($scenarios as $name => $scenario) {
|
|
|
|
$scenario['respectPerms'] = false;
|
|
|
|
$scenario['exceptionMessage'] = '';
|
|
|
|
$scenarios[$name . ' but no perm checks'] = $scenario;
|
|
|
|
}
|
|
|
|
return $scenarios;
|
|
|
|
}
|
2008-08-09 04:53:34 +00:00
|
|
|
}
|