PHP (32) - Příklad na BLOG

Dnes ukončíme část seriálu, která se věnuje zacházení se soubory, a dokončíme ji příkladem toho, jak by se v PHP dal napsat BLOG.

2.8.2004 15:00 | Petr Zajíc | přečteno 56094×

Píšeme BLOG

Weblog neboli blog je, zdá se, fenoménem poslední doby. Zatímco před pár lety chtěl každý mít svoje osobní stránky, dneska chce každý člověk šmrnclý internetem mít svůj internetový deníček. Nejsem tomu osobně nijak nakloněn, ale dá se to použít jako dobrý příklad na PHP a soubory, takže s chutí do toho.

Pro účely našeho příkladu budeme předpokládat, že každý den nahrajeme do určité složky na webovém serveru textový soubor s názvem ve formátu yyyymmdd.txt. Úkolem naší aplikace bude tyto soubory seřadit a vypsat nějak pěkně do prohlížeče. Aby toho nebylo málo, bude možné stáhnout si celý obsah blogu jako jeden soubor.

Přenos souborů na server nebudeme řešit. Probírali jsme to v minulém díle. Nicméně, pro testovací účely jsem připravil několik souborů v mojí složce na serveru, takže se můžete pokochat (zčásti) pravdivými útržky z mého života v posledních několika málo dnech. (Připravit alespoň trochu smysluplné texty byla pro mě nejnáročnější část celého článku, takže to tak berte. Samozřejmě by soubory mohly obsahovat cokoli;-))

Vlastní skript bude řešit několik věcí. Zaprvé, projdeme si složku a vypíšeme z ní soubory. Pod každým souborem dole bude ještě zobrazeno datum vypočítané z názvu souboru. Celé to nějak hezky oddělíme. Skript by mohl vypadat zhruba takhle:

<?
$slozka
= dir("./blog");
$licha=true;
while(
$soubor=$slozka->read()) {
  if (
$soubor=="." || $soubor=="..") continue;
  
$barva = $licha ? "#FDF5E6" : "#FFFFFF"; // nebo
  
echo "<div style=\"background-color:$barva\">";
  
readfile ("./blog/".$soubor);
  
$datum=explode(".", basename($soubor));
  echo
"<p align=\"right\">".date("d.m.Y", strtotime($datum[0]))."</p>";
  echo
"<hr></div>";
  
$licha=!$licha;
}
$slozka->close();
?>

Spustit skript

Je to na vysvětlení poměrně jednoduché. Skript vezme obsah složky a načte jej postupně pomocí $slozka->read. Pro lepší čtení budeme u souborů měnit vždy po jednom dni pozadí, k čemuž nám poslouží proměnná $licha. Ta se neustále přepíná mezi TRUE a FALSE a podle její hodnoty nastavujeme pozadí textu souvisejícího s daným dnem.

Vlastní podstatnou část zajišťuje funkce readfile. Hrátky s basename, explode, strtotime a date slouží jen k tomu, abychom z názvu souboru dostali česky formátované datum, které pak pod každý den vypíšeme.

Pozn.: V praxi bychom asi přidali ještě funkci stránkování, protože by to mohlo být časem dlouhé a nepřehledné. Taky by bylo potřeba ošetřit řazení souborů. Obojí jsme vynechali kvůli jednoduchosti. Ke stránkování se ale ještě dostaneme v některém díle o databázích.

Komprese souborů

Jako další věc jsme si slíbili, že budeme chtít celý blog stáhnout jako soubor. Blog se bude zvětšovat. To může vést časem k tomu, že by se stahoval extrémně dlouhý soubor, takže přichází na řadu komprese. PHP podporuje kompresi zlib (soubory gz) a prakticky vždy ponechávají administrátoři tuto volbu zapnutou, takže to lze použít. Skript pro kompresi blogu by mohl vypadat následovně:

<?
ob_start
();
// tohle je stejné jako v předchozím případě...
$contents = ob_get_contents();
ob_end_clean();
header("Content-Description: File Transfer");
header("Content-Type: application/force-download");
header("Content-Disposition: attachment; filename=\"blog.gz\"");
echo
gzencode($contents);
?>

Ukázat celý skript | Spustit skript

Tady bych měl asi vysvětlit několi věcí. Především, správný programátor by měl být líný. A když už jsme si předtím náš deníček tak hezky zformátovali, proč to teď nevyužít? Problém ale spočívá v tom, že předtím jsme psali pomocí příkazu echo přímo do prohlížeče - jak z toho tedy teď dostat soubor? Jednoduše - PHP má sadu funkcí umožňujících bufferování stránky a ukládání do proměnné namísto přímého posílání na výstup.

Ačkoli to normálně nemusí být vždy ten nejlepší nápad, nám se to zrovna hodí. Takže, funkce ob_start celý mechanismus začne a zajistí, že následující příkazy (například echo) se nebudou posílat prohlížeči, ale ukládat do bufferu. Proběhne totéž jako v prvním příkladu jen s tím rozdílem, že to, co bylo původně odesláno do prohlížeče je nyní v proměnné $contents a do prohlížeče se zatím neposlalo nic! Takže si klidně můžeme dovolit odeslat hlavičky a následně cokoli. Proč odesíláme právě tyto hlavičky jste se měli možnost dozvědět v předchozím dílu našeho seriálu.

Celou kompresi pak zajistí funkce gzencode. Ta nejenom provede kompresi, ale rovněž přidá hlavičky souboru podle specifikace, takže vznikne skutečný gz soubor. Můžete si jej uložit na disk, rozbalit a zobrazit. Komprese v našem případě ušetřila asi 50% velikosti souboru, při větších souborech to ale může být ještě lepší.

Pozn.: PHP může podporovat další typy kompresí - například bzip nebo zip. V praxi to ale zdaleka nebývá na serverech povoleno tak často jako gzip.

Soubory v PHP - shrnutí

V několika předchozích dílech jsme hovořili o PHP a souborech. Krátce shrňme, co podstatného jsme se dozvěděli:

Online verze článku: http://www.linuxsoft.cz/article.php?id_article=321