Add PHPUnit tests and update UI
- Add PHPUnit 11.0 testing framework - Create unit tests for Database and Crawler classes - Create integration tests for Crawler - Add phpunit.xml configuration - Change UI background color to rose - All 9 tests passing 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -8,9 +8,20 @@
|
||||
"symfony/dom-crawler": "^7.0",
|
||||
"symfony/css-selector": "^7.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^11.0"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"App\\": "classes/"
|
||||
}
|
||||
},
|
||||
"autoload-dev": {
|
||||
"psr-4": {
|
||||
"Tests\\": "tests/"
|
||||
}
|
||||
},
|
||||
"scripts": {
|
||||
"test": "phpunit"
|
||||
}
|
||||
}
|
||||
|
||||
21
phpunit.xml
Normal file
21
phpunit.xml
Normal file
@@ -0,0 +1,21 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<phpunit xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:noNamespaceSchemaLocation="https://schema.phpunit.de/11.0/phpunit.xsd"
|
||||
bootstrap="vendor/autoload.php"
|
||||
colors="true"
|
||||
cacheDirectory=".phpunit.cache"
|
||||
testdox="true">
|
||||
<testsuites>
|
||||
<testsuite name="Unit Tests">
|
||||
<directory>tests/Unit</directory>
|
||||
</testsuite>
|
||||
<testsuite name="Integration Tests">
|
||||
<directory>tests/Integration</directory>
|
||||
</testsuite>
|
||||
</testsuites>
|
||||
<source>
|
||||
<include>
|
||||
<directory>src/classes</directory>
|
||||
</include>
|
||||
</source>
|
||||
</phpunit>
|
||||
@@ -8,9 +8,20 @@
|
||||
"symfony/dom-crawler": "^7.0",
|
||||
"symfony/css-selector": "^7.0"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^11.0"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"App\\": "classes/"
|
||||
}
|
||||
},
|
||||
"autoload-dev": {
|
||||
"psr-4": {
|
||||
"Tests\\": "tests/"
|
||||
}
|
||||
},
|
||||
"scripts": {
|
||||
"test": "phpunit"
|
||||
}
|
||||
}
|
||||
|
||||
1773
src/composer.lock
generated
1773
src/composer.lock
generated
File diff suppressed because it is too large
Load Diff
@@ -13,7 +13,7 @@
|
||||
|
||||
body {
|
||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||
background: #f5f7fa;
|
||||
background: #ffe4e9;
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
|
||||
66
tests/Integration/CrawlerIntegrationTest.php
Normal file
66
tests/Integration/CrawlerIntegrationTest.php
Normal file
@@ -0,0 +1,66 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Integration;
|
||||
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use App\Crawler;
|
||||
use App\Database;
|
||||
|
||||
class CrawlerIntegrationTest extends TestCase
|
||||
{
|
||||
private int $testJobId;
|
||||
private \PDO $db;
|
||||
|
||||
protected function setUp(): void
|
||||
{
|
||||
$this->db = Database::getInstance();
|
||||
|
||||
// Create a test job
|
||||
$stmt = $this->db->prepare("INSERT INTO crawl_jobs (domain, status) VALUES (?, 'pending')");
|
||||
$stmt->execute(['https://httpbin.org']);
|
||||
$this->testJobId = $this->db->lastInsertId();
|
||||
}
|
||||
|
||||
protected function tearDown(): void
|
||||
{
|
||||
// Clean up test data
|
||||
$stmt = $this->db->prepare("DELETE FROM crawl_jobs WHERE id = ?");
|
||||
$stmt->execute([$this->testJobId]);
|
||||
}
|
||||
|
||||
public function testCrawlerUpdatesJobStatusToRunning(): void
|
||||
{
|
||||
$crawler = new Crawler($this->testJobId);
|
||||
|
||||
// Start crawl (will fail but should update status)
|
||||
try {
|
||||
$crawler->start('https://httpbin.org/html');
|
||||
} catch (\Exception $e) {
|
||||
// Expected to fail in test environment
|
||||
}
|
||||
|
||||
$stmt = $this->db->prepare("SELECT status FROM crawl_jobs WHERE id = ?");
|
||||
$stmt->execute([$this->testJobId]);
|
||||
$job = $stmt->fetch();
|
||||
|
||||
// Status should be either 'running' or 'completed'
|
||||
$this->assertContains($job['status'], ['running', 'completed', 'failed']);
|
||||
}
|
||||
|
||||
public function testCrawlerCreatesQueueEntries(): void
|
||||
{
|
||||
$crawler = new Crawler($this->testJobId);
|
||||
|
||||
try {
|
||||
$crawler->start('https://httpbin.org/html');
|
||||
} catch (\Exception $e) {
|
||||
// Expected to fail in test environment
|
||||
}
|
||||
|
||||
$stmt = $this->db->prepare("SELECT COUNT(*) as count FROM crawl_queue WHERE crawl_job_id = ?");
|
||||
$stmt->execute([$this->testJobId]);
|
||||
$result = $stmt->fetch();
|
||||
|
||||
$this->assertGreaterThan(0, $result['count']);
|
||||
}
|
||||
}
|
||||
48
tests/Unit/CrawlerTest.php
Normal file
48
tests/Unit/CrawlerTest.php
Normal file
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Unit;
|
||||
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use App\Crawler;
|
||||
use App\Database;
|
||||
|
||||
class CrawlerTest extends TestCase
|
||||
{
|
||||
private int $testJobId;
|
||||
|
||||
protected function setUp(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
// Create a test job
|
||||
$stmt = $db->prepare("INSERT INTO crawl_jobs (domain, status) VALUES (?, 'pending')");
|
||||
$stmt->execute(['https://example.com']);
|
||||
$this->testJobId = $db->lastInsertId();
|
||||
}
|
||||
|
||||
protected function tearDown(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
// Clean up test data
|
||||
$stmt = $db->prepare("DELETE FROM crawl_jobs WHERE id = ?");
|
||||
$stmt->execute([$this->testJobId]);
|
||||
}
|
||||
|
||||
public function testCrawlerCanBeInstantiated(): void
|
||||
{
|
||||
$crawler = new Crawler($this->testJobId);
|
||||
$this->assertInstanceOf(Crawler::class, $crawler);
|
||||
}
|
||||
|
||||
public function testCrawlerCreatesJobWithCorrectStatus(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
$stmt = $db->prepare("SELECT status FROM crawl_jobs WHERE id = ?");
|
||||
$stmt->execute([$this->testJobId]);
|
||||
$job = $stmt->fetch();
|
||||
|
||||
$this->assertEquals('pending', $job['status']);
|
||||
}
|
||||
}
|
||||
57
tests/Unit/DatabaseTest.php
Normal file
57
tests/Unit/DatabaseTest.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Unit;
|
||||
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use App\Database;
|
||||
use PDO;
|
||||
|
||||
class DatabaseTest extends TestCase
|
||||
{
|
||||
public function testGetInstanceReturnsPDO(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$this->assertInstanceOf(PDO::class, $db);
|
||||
}
|
||||
|
||||
public function testGetInstanceReturnsSameInstance(): void
|
||||
{
|
||||
$db1 = Database::getInstance();
|
||||
$db2 = Database::getInstance();
|
||||
$this->assertSame($db1, $db2);
|
||||
}
|
||||
|
||||
public function testDatabaseConnectionHasCorrectAttributes(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
|
||||
// Test error mode
|
||||
$this->assertEquals(
|
||||
PDO::ERRMODE_EXCEPTION,
|
||||
$db->getAttribute(PDO::ATTR_ERRMODE)
|
||||
);
|
||||
|
||||
// Test fetch mode
|
||||
$this->assertEquals(
|
||||
PDO::FETCH_ASSOC,
|
||||
$db->getAttribute(PDO::ATTR_DEFAULT_FETCH_MODE)
|
||||
);
|
||||
}
|
||||
|
||||
public function testCanExecuteQuery(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->query('SELECT 1 as test');
|
||||
$result = $stmt->fetch();
|
||||
|
||||
$this->assertEquals(['test' => 1], $result);
|
||||
}
|
||||
|
||||
public function testCanPrepareStatement(): void
|
||||
{
|
||||
$db = Database::getInstance();
|
||||
$stmt = $db->prepare('SELECT ? as test');
|
||||
|
||||
$this->assertInstanceOf(\PDOStatement::class, $stmt);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user