<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Webboarder &#187; Webworking</title>
	<atom:link href="http://webboarder.de/category/webworking/feed/" rel="self" type="application/rss+xml" />
	<link>http://webboarder.de</link>
	<description>Webworking und Wakeboarding</description>
	<lastBuildDate>Thu, 13 Aug 2009 09:01:03 +0000</lastBuildDate>
	<generator>http://wordpress.org/?v=2.9.2</generator>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
			<item>
		<title>Cron Sandbox (Cronjob verifizieren)</title>
		<link>http://webboarder.de/2009/08/cron-sandbox-cronjob-verifizieren/</link>
		<comments>http://webboarder.de/2009/08/cron-sandbox-cronjob-verifizieren/#comments</comments>
		<pubDate>Thu, 13 Aug 2009 09:00:18 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[Webworking]]></category>
		<category><![CDATA[Linux]]></category>
		<category><![CDATA[Shell]]></category>
		<category><![CDATA[Tool]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=178</guid>
		<description><![CDATA[Hatte gerade das Problem, dass ich gerade die Zeiten eines &#8220;crontab&#8221;-Aufrufs verifizieren musste. Aufgrund der ganzen Parameter-Möglichkeiten gar nicht so einfach. Hab aber gerade eine Seite gefunden, auf der man einfach die Parameter eingeben kann und dann zurückbekommt, wann der Cronjob in der Zukunft aufgerufen wird.
Sehr nett:
http://www.hxpi.com/cron_sandbox.php
]]></description>
			<content:encoded><![CDATA[<p>Hatte gerade das Problem, dass ich gerade die Zeiten eines &#8220;crontab&#8221;-Aufrufs verifizieren musste. Aufgrund der ganzen Parameter-Möglichkeiten gar nicht so einfach. Hab aber gerade eine Seite gefunden, auf der man einfach die Parameter eingeben kann und dann zurückbekommt, wann der Cronjob in der Zukunft aufgerufen wird.</p>
<p>Sehr nett:<br />
<a href="http://www.hxpi.com/cron_sandbox.php">http://www.hxpi.com/cron_sandbox.php</a></p>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/08/cron-sandbox-cronjob-verifizieren/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Update der Serpent Template Engine auf 1.2.3</title>
		<link>http://webboarder.de/2009/07/update-der-serpent-template-engine-auf-1-2-3/</link>
		<comments>http://webboarder.de/2009/07/update-der-serpent-template-engine-auf-1-2-3/#comments</comments>
		<pubDate>Thu, 16 Jul 2009 09:55:53 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[engine]]></category>
		<category><![CDATA[serpent]]></category>
		<category><![CDATA[template]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=174</guid>
		<description><![CDATA[Und wieder mal gabs ein Update auf eine neue Version. Verbessert habe ich unter anderem den Creole-Parser, der jetzt endlich sauber mit Nested Lists umgehen kann.
Außerdem ist es jetzt möglich, die Standard-PHP-Tags (&#8220;&#60;?php&#8221;, &#8220;&#60;?&#8221;, &#8220;&#60;?=&#8221;) als Delimiter zu verwenden. OK, was es vorher auch schon, aber jetzt sind auch die Punkt-Syntax und Mappings kein Problem [...]]]></description>
			<content:encoded><![CDATA[<p>Und wieder mal gabs ein Update auf eine neue Version. Verbessert habe ich unter anderem den Creole-Parser, der jetzt endlich sauber mit Nested Lists umgehen kann.<br />
Außerdem ist es jetzt möglich, die Standard-PHP-Tags (&#8220;&lt;?php&#8221;, &#8220;&lt;?&#8221;, &#8220;&lt;?=&#8221;) als Delimiter zu verwenden. OK, was es vorher auch schon, aber jetzt sind auch die Punkt-Syntax und Mappings kein Problem mehr. Und die Short Tags werden automatisch zu langen Tags umgebaut, was nützlich ist, falls die short tags in der php.ini ausgeschaltet wurden.</p>
<hr />
<ol>
<li>Serpent &#8211; PHP Template Engine<br />
<a href="http://code.google.com/p/serpent-php-template-engine/">http://code.google.com/p/serpent-php-template-engine/</a></li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/07/update-der-serpent-template-engine-auf-1-2-3/feed/</wfw:commentRss>
		<slash:comments>1</slash:comments>
		</item>
		<item>
		<title>Neue Serpent-Version: 1.2.1</title>
		<link>http://webboarder.de/2009/05/neue-serpent-version-121/</link>
		<comments>http://webboarder.de/2009/05/neue-serpent-version-121/#comments</comments>
		<pubDate>Sun, 10 May 2009 08:47:47 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[engine]]></category>
		<category><![CDATA[serpent]]></category>
		<category><![CDATA[template]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=170</guid>
		<description><![CDATA[Eben gerade hab ich die neue Version (Version 1.2.1) der Serpent Template Engine als Download bei Google Code zur Verfügung gestellt.
Dieses ist ein Bugfix, der den Creole-Parser etwas verbessert, welcher mit der Version 1.2 eingeführt wurde. Creole ist eine Wiki-Syntax, die eingeführt wurde, um den Wildwuchs an Syntaxen etwas zu bereinigen. Creole wird mittlerweile von [...]]]></description>
			<content:encoded><![CDATA[<p>Eben gerade hab ich die neue Version (Version 1.2.1) der <a href="http://code.google.com/p/serpent-php-template-engine/downloads/list" target="_blank">Serpent Template Engine als Download</a> bei Google Code zur Verfügung gestellt.</p>
<p>Dieses ist ein Bugfix, der den Creole-Parser etwas verbessert, welcher mit der Version 1.2 eingeführt wurde. Creole ist eine Wiki-Syntax, die eingeführt wurde, um den Wildwuchs an Syntaxen etwas zu bereinigen. Creole wird mittlerweile von einer Vielzahl von Wikis verwendet und sollte beim Bau einer eigenen Wiki-Engine zumindest unterstützt werden. Der User wirds euch danken.</p>
<hr />
<ol>
<li>Serpent &#8211; PHP Template Engine<br />
<a href="http://code.google.com/p/serpent-php-template-engine/">http://code.google.com/p/serpent-php-template-engine/</a></li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/05/neue-serpent-version-121/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>X11-Programme unter Windows mit PuTTY und XMing</title>
		<link>http://webboarder.de/2009/04/x11-programme-unter-windows-mit-putty-und-xming/</link>
		<comments>http://webboarder.de/2009/04/x11-programme-unter-windows-mit-putty-und-xming/#comments</comments>
		<pubDate>Thu, 09 Apr 2009 14:42:53 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[Linux]]></category>
		<category><![CDATA[PuTTY]]></category>
		<category><![CDATA[Shell]]></category>
		<category><![CDATA[Ubuntu]]></category>
		<category><![CDATA[X11]]></category>
		<category><![CDATA[XMing]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=165</guid>
		<description><![CDATA[Ja, die Headline dieses Artikels ist vielleicht etwas gewöhnungsbedürftig. Aber es geht um ein tolles Thema&#8230; fangen wir von vorn an:
Seit längerer Zeit suche ich einen guten Konsolen-Editor, mit dem meine PuTTY-Sessions nicht so nervig wie bisher sind. Mit Vi(m), Nano, Joe usw. konnte ich mich einfach nicht anfreunden, weil sie alle ihre eigenen Tastenkürzel [...]]]></description>
			<content:encoded><![CDATA[<p>Ja, die Headline dieses Artikels ist vielleicht etwas gewöhnungsbedürftig. Aber es geht um ein tolles Thema&#8230; fangen wir von vorn an:</p>
<p>Seit längerer Zeit suche ich einen guten Konsolen-Editor, mit dem meine PuTTY-Sessions nicht so nervig wie bisher sind. Mit Vi(m), Nano, Joe usw. konnte ich mich einfach nicht anfreunden, weil sie alle ihre eigenen Tastenkürzel haben und halt nicht die, die ich aus GUI-Oberflächen (egal ob Windows oder Linux) gewohnt bin. Jetzt bin ich aber gerade über die optimale Lösung gestolpert: XMing zusammen mit PuTTY. </p>
<p>XMing ist ein X-Server, mit dem man auf die X11-Programme eines Servers zugreifen kann. Ich kann also über PuTTY ganz normal eine Konsolen-Session führen, mit dem Unterschied, dass ich jetzt auch Dateien mit z.B. GEdit bearbeiten kann. Ich sehe dann unter Windows ein Fenster mit GEdit drin. Genial. Selbst Gimp und OpenOffice liefen problemlos.</p>
<p>Hier die Anleitung:</p>
<ol>
<li>XMing und XMing-portablePuTTY von der <a href="http://www.straightrunning.com/XmingNotes/" target="_blank">XMing Hauptseite</a> herunterladen und installieren.</li>
<li>Wie gewohnt eine PuTTY-Session starten mit einem kleinen Unterschied: Bei den Einstellungen unter &#8220;Connection > SSH > X11&#8243; muss &#8220;X11 forwarding&#8221; aktiviert sein.</li>
<li>Auf dem Server muss X11 forwarding in &#8220;/etc/ssh/sshd.config&#8221; aktiviert sein. Bei mir (Ubuntu-Server) war es das standardmäßig, musste also nichts ändern. Falls man etwas ändern muss: SSH-Daemon oder Rechner neu starten.</li>
</ol>
<p>Schon klappt es. Auf der XMing-Seite sieht man auch ein paar Screenshots, die Lust auf mehr machen.</p>
<p>Ach ja: Damit eure Konsole nicht blockiert ist, hängt ein &#038; an den Befehl heran, z.B.</p>
<pre class="brush: php">
gedit dummy.txt &amp;
</pre>
<p>Und Nautilus solltet ihr so starten:</p>
<pre class="brush: php">
nautilus --no-desktop &amp;
</pre>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/04/x11-programme-unter-windows-mit-putty-und-xming/feed/</wfw:commentRss>
		<slash:comments>5</slash:comments>
		</item>
		<item>
		<title>phpThumb mit statisch kompiliertem Imagemagick</title>
		<link>http://webboarder.de/2009/04/phpthumb-mit-statisch-kompiliertem-imagemagick/</link>
		<comments>http://webboarder.de/2009/04/phpthumb-mit-statisch-kompiliertem-imagemagick/#comments</comments>
		<pubDate>Thu, 09 Apr 2009 11:18:49 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[ImageMagick]]></category>
		<category><![CDATA[phpThumb]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=161</guid>
		<description><![CDATA[Hatte gerade das Problem, dass ich für einen Kunden eine Bilddatenbank aufsetzen sollte, die mit phpThumb läuft. Der Kunde wollte jetzt auch z.B. PSD-Dateien mit einer Vorschau angezeigt bekommen und somit musste natürlich auch ImageMagick auf dem Server liegen&#8230; tat es aber nicht&#8230; und war auch nicht nachzuinstallieren, weil es um einen Managed Server ging.
Allerdings [...]]]></description>
			<content:encoded><![CDATA[<p>Hatte gerade das Problem, dass ich für einen Kunden eine Bilddatenbank aufsetzen sollte, die mit phpThumb läuft. Der Kunde wollte jetzt auch z.B. PSD-Dateien mit einer Vorschau angezeigt bekommen und somit musste natürlich auch ImageMagick auf dem Server liegen&#8230; tat es aber nicht&#8230; und war auch nicht nachzuinstallieren, weil es um einen Managed Server ging.</p>
<p>Allerdings geistert im Web eine statisch kompilierte Version 4.2.9 (uralt) herum, die eigentlich für Typo3 gedacht ist. Die kann man dann per FTP einfach hochladen, muss noch die Ausführungsrechte setzen und schon gehts&#8230; nur nicht mit PHPThumb. Hier sind ein paar kleine Änderungen nötig.</p>
<p>Zuerst muss der Pfad zum convert-Tool in der &#8220;phpThumb.config.php&#8221; angegeben werden. Einfach folgende Zeile suchen und anpassen:</p>
<pre class="brush: php">
$PHPTHUMB_CONFIG[&#039;imagemagick_path&#039;] = &#039;path/to/imagemagick-4.2.9_i386-static/convert&#039;;
</pre>
<p>Dann muss die &#8220;phpthumb.class.php&#8221; leicht angepasst werden. Zuerst die Funktion &#8220;ImageMagickVersion&#8221;. Da der reguläre Ausdruck in dieser Funktion nicht die Versionsnummer der statisch kompilierten IM-Version matcht, müssen wir da ein wenig nachhelfen. Am einfachsten (und schmutzigsten) ist es, direkt eine eigene Versionsnummer zurückzugeben:</p>
<pre class="brush: php">
// diese Zeile vor dem return am Ende hinzufügen
return &#039;4.2.9 99/09/01&#039;;

return @$versionstring[intval($returnRAW)];
</pre>
<p>Jetzt erkennt phpThumb schon einmal, dass es eine gültige ImageMagick-Version gibt. Allerdings haben wir noch das Problem, dass es in dieser uralten Version noch nicht die &#8220;-resize&#8221; Parameter gab. Hierfür reicht es, in der Funktion &#8220;ImageMagickThumbnailToGD&#8221; die Zeile</p>
<pre class="brush: php">
$IMresizeParameter = &#039;resize&#039;;
</pre>
<p>durch </p>
<pre class="brush: php">
$IMresizeParameter = &#039;size&#039;;
</pre>
<p>zu ersetzen. Und schon läufts.</p>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/04/phpthumb-mit-statisch-kompiliertem-imagemagick/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>InnoDB und große Tabellen</title>
		<link>http://webboarder.de/2009/04/innodb-und-grose-tabellen/</link>
		<comments>http://webboarder.de/2009/04/innodb-und-grose-tabellen/#comments</comments>
		<pubDate>Tue, 07 Apr 2009 09:27:20 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[InnoDB]]></category>
		<category><![CDATA[MySQL]]></category>
		<category><![CDATA[performance]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=145</guid>
		<description><![CDATA[Vor kurzem hatte ich die Aufgabe, ein Gewinnspiel umzusetzen, bei dem User sich pro E-Mail-Adresse einen Gewinncode zurückgeben lassen konnten. Die 7 Millionen Gewinncodes kamen dabei vom Kunden und waren alle unique. Jeder Code durfte dabei nur einmal ausgespielt werden. Um das sicherzustellen kamen nur Transaktionen in Frage. Somit war InnoDB Pflicht.
Ok, als erstes hab [...]]]></description>
			<content:encoded><![CDATA[<p>Vor kurzem hatte ich die Aufgabe, ein Gewinnspiel umzusetzen, bei dem User sich pro E-Mail-Adresse einen Gewinncode zurückgeben lassen konnten. Die 7 Millionen Gewinncodes kamen dabei vom Kunden und waren alle unique. Jeder Code durfte dabei nur einmal ausgespielt werden. Um das sicherzustellen kamen nur Transaktionen in Frage. Somit war InnoDB Pflicht.</p>
<p>Ok, als erstes hab ich die beiden Tabellen erstellt:</p>
<pre class="brush: php">
--
-- Tabellenstruktur für Tabelle `codes`
--

CREATE TABLE `codes` (
	`code` char(6) character set latin1 collate latin1_bin NOT NULL,
	`user_id` int(11) default NULL,
	PRIMARY KEY  (`code`),
	KEY `user_id` (`user_id`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1 ROW_FORMAT=FIXED;

-- --------------------------------------------------------

--
-- Tabellenstruktur für Tabelle `users`
--

CREATE TABLE `users` (
	`id` int(11) NOT NULL auto_increment,
	`salutation` set(&#039;Herr&#039;,&#039;Frau&#039;) NOT NULL,
	`firstname` char(50) NOT NULL,
	`lastname` char(50) NOT NULL,
	`street` char(100) NOT NULL,
	`hnr` char(20) NOT NULL,
	`zip` char(10) NOT NULL,
	`city` char(100) NOT NULL,
	`email` char(100) NOT NULL,
	`created` datetime NOT NULL,
	PRIMARY KEY  (`id`),
	UNIQUE KEY `email` (`email`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=FIXED AUTO_INCREMENT=1;
</pre>
<h2>Problem 1: Wie importiert man 7 Millionen Datensätze möglichst schnell?</h2>
<p>Einzelne Queries sind langsam. Mehrere Queries per Transaktion zu bündeln ist ein wenig schneller. Das schnellste aber scheint LOAD DATA zu sein, welches auch über das Shell-Tool mysqlimport zu verwenden ist. Da man aber bei mysqlimport keinen Fortschritt ausgeworfen bekommt, hab ich fix ein Shell-Skript zusammengeschrieben, welches alle 10.000 Zeilen eine Meldung auswirft.</p>
<pre class="brush: php">
$data_file = &#039;/tmp/codes.txt&#039;;
$db = &#039;otto_paf_mai2009&#039;;
$chunk_file = &#039;/tmp/codes.part&#039;;

# split the file into small chunks to output a status report
$line = 0;
$buffer = &#039;&#039;;
$handle = fopen ($data_file, &#039;r&#039;);
while (!feof($handle))
	{
	$line++;
	$buffer .= fgets($handle);

	if ($line%10000 === 0)
		{
		file_put_contents($chunk_file, $buffer);
		$buffer = &#039;&#039;;

		# the data file has to have the same name as the table
		# it has to be readable by the mysql server (especially the directory)
		# mysqlimport [options] [db_name] [data_file]
		shell_exec(&#039;mysqlimport -uroot --columns=code --verbose &#039;.$db.&#039; &#039;.$chunk_file);

		stdout($result);
		stdout(number_format($line, 0, &#039;,&#039;, &#039;.&#039;));
		}

	}
fclose ($handle);

function stdout($out){fwrite(STDOUT, $out.&amp;quot;\n&amp;quot;);}
function stderr($out){fwrite(STDERR, $out.&amp;quot;\n&amp;quot;);}
</pre>
<p>Interessant war hierbei schon, wie lange MySQL dafür braucht und wie tödlich ein Unique-Key (in diesem Fall der PRIMARY) sein kann. Der Import in MyISAM ohne Unique-Key dauerte eine halbe Minute. Der Import in InnoDB ohne Unique-Key etwa 5 Minuten. Und der Import in InnoDB MIT dem Unique-Key dauerte letztendlich 5,5 Stunden. Autsch! Allerdings war es ganz gut, dass ich den Unique-Key gesetzt habe, denn bei den ersten Daten des Kunden waren tatsächlich Dubletten in den Codes.</p>
<h2>Problem 2: Wie gibt man einen zufälligen Code aus?</h2>
<p>Der offensichtlichste und verbreitetste Ansatz ist:</p>
<pre class="brush: php">
SELECT code
FROM codes
WHERE user_id IS NULL
ORDER BY RAND()
LIMIT 1
</pre>
<p>Dauert aber leider schon bei 600.000 Datensätzen über eine halbe Sekunde. Und das EXPLAIN macht alles klar. In der Spalte &#8220;extra&#8221; steht:</p>
<pre class="brush: php">Using where; Using index; Using temporary; Using filesort</pre>
<p>Das sieht übel aus.</p>
<p>Mein nächster Ansatz war, das ganze in zwei Abfragen auszulagern. Die erste sollte sich die Anzahl der gültigen Codes zurückliefern, die zweite schließlich mit einem zufälligen Offset beim LIMIT arbeiten, der halt zwischen 0 und der Anzahl der gültigen Codes liegen sollte. Das sah etwa so aus:</p>
<pre class="brush: php">
// get count of codes
$sql = $this-&gt;db-&gt;result(&quot;
	SELECT count(*) as count
	FROM codes
	WHERE user_id IS NULL
&quot;);

$offset = rand(0, $sql[&#039;RESULT&#039;][0][&#039;count&#039;]);

// get code from db
$sql = $this-db-&gt;Result(&quot;
	SELECT code
	FROM codes
	WHERE user_id IS NULL
	LIMIT $offset, 1
&quot;);
$code = $sql[&#039;RESULT&#039;][0][&#039;code&#039;];
</pre>
<p>Schon die erste Abfrage dauerte im Schnitt 1,7 Sekunden. Aber warum? Die Antwort findet sich hier: http://www.mysqlperformanceblog.com/2006/12/01/count-for-innodb-tables/<br />
Ein COUNT(*) auf viele Datensätze ist bei InnoDB einfach höllisch langsam. Die nächste Idee war hierbei, die erste Abfrage umzudrehen. Da ich nämlich weiß, wieviele Keys in der DB sind, brauche ich nur die user_ids zählen und von der Gesamtzahl abzuziehen. Das war auch tatsächlich sehr schnell. Selbst die zweite Abfrage&#8230;</p>
<p>ABER HALT&#8230;</p>
<p>&#8230; anscheinend nur bei der ersten Abfrage. Irgendwie schwanken die Zeiten zwischen 0,0003 Sekunden und 2,5 Sekunden. Nach ein bisschen Herumprobieren kam auch hier der Übertäter zum Vorschein: Das LIMIT. Ist der Offset klein, kommt das Ergebnis sehr schnell, ist er groß, dauert es ewig lang. Mist.</p>
<p>Mein derzeitiger Ansatz ist mehr ein Workaround. Da ich weiß, dass die Codes mit einem Klein- oder Großbuchstaben anfangen, wähle ich einfach einen aus und lasse mir nur Codes zurückgeben, die halt diesen am Anfang stehen haben:</p>
<pre class="brush: php">
$values = array_merge( range(&#039;a&#039;, &#039;z&#039;), range(&#039;A&#039;, &#039;Z&#039;) );
$char = $values[ rand(0, count($values)-1 ) ];

$sql = $this-db-&gt;Result(&quot;
	SELECT code
	FROM codes
	FORCE INDEX(PRIMARY)
	WHERE code LIKE &#039;&quot;.$char.&quot;%&#039;
	AND user_id IS NULL
	LIMIT 1
&quot;);
</pre>
<p>Interessant war hierbei das FORCE INDEX. Ohne wollte MySQL lieber den Index für die user_id verwenden als den PRIMARY KEY. Verrückt. Allerdings ist die endgültige Lösung mit 0,007 Sekunden recht schnell. Allerdings bekomme ich nicht wirklich zufällige Ergebnisse heraus.</p>
<p>Wer also eine bessere Lösung kennt oder eine andere Idee hat: Immer her damit!</p>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/04/innodb-und-grose-tabellen/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Neue Serpent Version: 1.1 RC1</title>
		<link>http://webboarder.de/2009/03/neue-serpent-version-11-rc1/</link>
		<comments>http://webboarder.de/2009/03/neue-serpent-version-11-rc1/#comments</comments>
		<pubDate>Sat, 07 Mar 2009 21:08:06 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[engine]]></category>
		<category><![CDATA[serpent]]></category>
		<category><![CDATA[template]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=137</guid>
		<description><![CDATA[Eben gerade hab ich die neue Version (Version 1.1 RC1) der Serpent Template Engine als Download bei Google Code zur Verfügung gestellt.
Wie man an der Versionsnummer sehen kann, ist es ein Release Candidate, der also schon sehr stabil läuft. 
Neue Features sind eine einfachere Initialisation, ein neues Plugin-System (welches es noch einfacher macht, die Template-Engine [...]]]></description>
			<content:encoded><![CDATA[<p>Eben gerade hab ich die neue Version (Version 1.1 RC1) der <a href="http://code.google.com/p/serpent-php-template-engine/" target="_blank">Serpent Template Engine als Download</a> bei Google Code zur Verfügung gestellt.<br />
Wie man an der Versionsnummer sehen kann, ist es ein Release Candidate, der also schon sehr stabil läuft. </p>
<p>Neue Features sind eine einfachere Initialisation, ein neues Plugin-System (welches es noch einfacher macht, die Template-Engine zu erweitern) und die Möglichkeit, auch den Compiler on the fly zu wechseln (weil dieser selbst jetzt auch als Plugin eingebunden ist). Es liegt jetzt auch ein Compiler für die Wiki-ähnliche Markdown-Syntax dabei.</p>
<p>Die <a href="http://code.google.com/p/serpent-php-template-engine/wiki/Overview" target="_blank">Dokumentation </a>ist natürlich schon auf dem neuesten Stand.</p>
<hr />
<ol>
<li>Serpent &#8211; PHP Template Engine<br />
<a href="http://code.google.com/p/serpent-php-template-engine/">http://code.google.com/p/serpent-php-template-engine/</a></li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/03/neue-serpent-version-11-rc1/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Serpent &#8211; PHP Template Engine</title>
		<link>http://webboarder.de/2009/02/serpent-php-template-engine/</link>
		<comments>http://webboarder.de/2009/02/serpent-php-template-engine/#comments</comments>
		<pubDate>Tue, 10 Feb 2009 16:21:32 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[engine]]></category>
		<category><![CDATA[performance]]></category>
		<category><![CDATA[serpent]]></category>
		<category><![CDATA[template]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=108</guid>
		<description><![CDATA[Nach jahrelangem Durchforsten des Webs nach guten Template Engines, die leider nicht das boten, was ich benötige, bin ich jetzt endlich meiner Standard-Engine Smarty abtrünnig geworden und habe in den letzten Wochen meine eigene Template Engine umgesetzt: Serpent.
Und, nein, ich finde Smarty ganz und gar nicht schlecht. Ich werde es auch wieder einsetzen, wenn ich [...]]]></description>
			<content:encoded><![CDATA[<div class="wp-caption alignright" style="width: 210px"><a href="http://code.google.com/p/serpent-php-template-engine/"><img title="Serpent - PHP Template Engine" src="http://snapcasa.com/get.aspx?code=3793&amp;size=m&amp;url=http://code.google.com/p/serpent-php-template-engine/" alt="http://code.google.com/p/serpent-php-template-engine/" width="200" height="150" /></a><p class="wp-caption-text">Serpent - PHP Template Engine</p></div>
<p>Nach jahrelangem Durchforsten des Webs nach guten Template Engines, die leider nicht das boten, was ich benötige, bin ich jetzt endlich meiner Standard-Engine <a href="http://www.smarty.net/" target="_blank">Smarty</a> abtrünnig geworden und habe in den letzten Wochen meine eigene Template Engine umgesetzt: <strong>Serpent</strong>.</p>
<p>Und, nein, ich finde Smarty ganz und gar nicht schlecht. Ich werde es auch wieder einsetzen, wenn ich das Hauptfeature &#8220;template security&#8221; brauchen sollte, aber bis dahin werd ich wohl recht glücklich mit meiner Engine, die genau die für mich sinvoll wirkenden Features integriert.</p>
<p>Soeben habe ich die erste beta-Version veröffentlicht und wäre natürlich glücklich, wenn sie jemand mal ausprobieren und sein Feedback posten bzw. in den &#8220;Issues&#8221; bei Google Code einstellen würde.</p>
<h2>Was unterscheidet Serpent von anderen Template Engines?</h2>
<p>Es integriert zum einen keine Template-Sicherheit, weil ich sie in den letzten 10 Jahren nicht gebraucht habe. Und zum anderen gibt es kein eingebautes Caching-System, weil das Caching des Outputs in vielen Frameworks über das View des <a href="http://de.wikipedia.org/wiki/Model_View_Controller" target="_blank">MVC</a>s geregelt wird. Und da eine Template Engine nur EINE Möglichkeit der Ausgabe darstellt (neben XML, JSON, CSV usw.) ist das auch nicht Aufgabe der Engine.</p>
<p>Auf der Haben-Seite steht, dass die Template-Sprache PHP ist, was es natürlich extrem flexibel macht. Damit ist es ähnlich wie <a href="http://phpsavant.com/" target="_blank">Savant3</a>, hat aber die Vorteile, dass es nicht auf short_tags aufbaut, um eine kurze Syntax hinzukriegen.</p>
<p>Desweiteren untersützt Serpent die von <a href="http://www.djangoproject.com/" target="_blank">Django</a> und <a href="http://dwoo.org/" target="_blank">Dwoo</a> bekannte Template-Vererbung, die von Smarty entliehende Punkt-Syntax für Arrays, Funktionen-Mapping für weniger Schreibarbeit und Resourcen-Handler, weil Templates ja nicht immer aus dem Dateisystem kommen müssen (beim CMS z.B. auch aus einer Datenbank).<br />
E_STRICT-Kompatibilität, gute Objektorientierung und verdammt gute Performance sind selbstverständlich, oder?!</p>
<p>Um ein genaueres Bild von Serpent zu bekommen, empfehle ich, einen Blick auf die Overview-Seite zu werfen:<br />
<a href="http://code.google.com/p/serpent-php-template-engine/wiki/Overview" target="_blank">http://code.google.com/p/serpent-php-template-engine/wiki/Overview</a></p>
<p>Dann viel Spaß beim Testen.</p>
<hr />
<ol>
<li>Serpent &#8211; PHP Template Engine<br />
<a href="http://code.google.com/p/serpent-php-template-engine/">http://code.google.com/p/serpent-php-template-engine/</a></li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/02/serpent-php-template-engine/feed/</wfw:commentRss>
		<slash:comments>7</slash:comments>
		</item>
		<item>
		<title>Reguläre Ausdrücke online testen</title>
		<link>http://webboarder.de/2009/02/regulaere-ausdruecke-online-testen/</link>
		<comments>http://webboarder.de/2009/02/regulaere-ausdruecke-online-testen/#comments</comments>
		<pubDate>Mon, 09 Feb 2009 14:11:33 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[Webworking]]></category>
		<category><![CDATA[Regex]]></category>
		<category><![CDATA[Tool]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=99</guid>
		<description><![CDATA[Wer kennt das nicht? Hin und wieder muss man einen regulären Ausdruck testen, aber wer hat schon Lust, sich dafür etwas zusammenzuschreiben? Dank des genialen Online-Tools &#8220;Regex Tester V3&#8243; auch gar nicht mehr nötig.
Nicht nur, dass man wirklich alle in PHP verbauten Regex-Funktionen komfortabel testen kann. Man kann sogar seine Test-Umgebungen inkl. aller Parameter speichern [...]]]></description>
			<content:encoded><![CDATA[<div class="wp-caption alignright" style="width: 210px"><a href="http://www.regex-tester.de/regex.html" class="broken_link"><img title="Regex Tester V3" src="http://snapcasa.com/get.aspx?code=3793&amp;size=m&amp;url=http://www.regex-tester.de/regex.html" alt="http://www.regex-tester.de/regex.html" width="200" height="150" /></a><p class="wp-caption-text">Regex Tester V3</p></div>
<p>Wer kennt das nicht? Hin und wieder muss man einen regulären Ausdruck testen, aber wer hat schon Lust, sich dafür etwas zusammenzuschreiben? Dank des genialen Online-Tools &#8220;Regex Tester V3&#8243; auch gar nicht mehr nötig.</p>
<p>Nicht nur, dass man wirklich alle in PHP verbauten Regex-Funktionen komfortabel testen kann. Man kann sogar seine Test-Umgebungen inkl. aller Parameter speichern und zusätzlich einen PHP-Beispiel-Code generieren lassen, den man hervorragend z.B. in Foren posten kann.</p>
<p>Ein wirklich starkes Tool.</p>
<hr />
<ol>
<li>Regex Tester V3<br />
<a href="http://www.regex-tester.de/regex.html" class="broken_link">http://www.regex-tester.de/regex.html</a></li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2009/02/regulaere-ausdruecke-online-testen/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>SEO: Doppelten Content vermeiden</title>
		<link>http://webboarder.de/2008/12/seo-doppelten-content-vermeiden/</link>
		<comments>http://webboarder.de/2008/12/seo-doppelten-content-vermeiden/#comments</comments>
		<pubDate>Mon, 01 Dec 2008 19:32:00 +0000</pubDate>
		<dc:creator>Christoph</dc:creator>
				<category><![CDATA[Webworking]]></category>
		<category><![CDATA[htaccess]]></category>
		<category><![CDATA[httpd.conf]]></category>
		<category><![CDATA[SEO]]></category>

		<guid isPermaLink="false">http://webboarder.de/?p=94</guid>
		<description><![CDATA[Alles klar? Wer jetzt noch nicht Bescheid weiß, sollte besser weiterlesen. Doppelter Content sind Seiten, die unter verschiedenen URLs auftauchen, aber ansonsten denselben Inhalt bieten. Google mag das leider gar nicht, und so kann auch schnell mal das Ranking der eigenen Seite schlechter werden, wenn z.B. die Startseite unter http://webboarder.de, http://www.webboarder.de oder http://webboarder.de/index.htm zu finden [...]]]></description>
			<content:encoded><![CDATA[<p>Alles klar? Wer jetzt noch nicht Bescheid weiß, sollte besser weiterlesen. Doppelter Content sind Seiten, die unter verschiedenen URLs auftauchen, aber ansonsten denselben Inhalt bieten. <a href="http://www.google.com/support/webmasters/bin/answer.py?hl=en&amp;answer=66359" target="_blank">Google mag das leider gar nicht</a>, und so kann auch schnell mal das Ranking der eigenen Seite schlechter werden, wenn z.B. die Startseite unter http://webboarder.de, http://www.webboarder.de oder http://webboarder.de/index.htm zu finden ist.</p>
<p>Für mich die wertvollsten Tipps sind hier:</p>
<ol>
<li><strong>Konsistent verlinken:</strong> Nicht hier mal so und auf der nächsten Seite vollkommen anders. Immer dieselbe URL für eine Verlinkung verwenden.</li>
<li><strong>EINE Domain verwenden:</strong> Entweder mit Subdomain &#8220;www&#8221; oder ohne. Aber entscheiden müsst ihr euch.</li>
</ol>
<p>Während Punkt 1 ja noch ganz einfach selbst bewerkstelligt werden kann, ist das bei Punkt 2&#8230; auch nicht schwerer. Einfach folgendes Schnippselchen in die <strong>.htaccess</strong> einfügen und schon solltet ihr keine Probleme mehr haben (nicht vergessen, den Schnippsel anzupassen):</p>
<pre class="brush: css">
&lt;ifmodule mod_rewrite.c&gt;
RewriteEngine On
RewriteCond %{HTTP_HOST} ^www\.webboarder\.de$ [NC]
RewriteRule ^(.*)$ http://webboarder.de/$1 [R=301,L]
&lt;/ifmodule&gt;
</pre>
<hr />
<ol>
<li>Tipps zu dem Thema von Google:<br /><a href="http://www.google.com/support/webmasters/bin/answer.py?hl=en&amp;answer=66359">http://www.google.com/support/webmasters/bin/answer.py?hl=en&amp;answer=66359</a></li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://webboarder.de/2008/12/seo-doppelten-content-vermeiden/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
