Zend_File_Transfer-Introduction.xml 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- EN-Revision: 17134 -->
  3. <!-- Reviewed: no -->
  4. <sect1 id="zend.file.transfer.introduction">
  5. <title>Zend_File_Transfer</title>
  6. <para>
  7. <classname>Zend_File_Transfer</classname> bietet exzessiven Support für Datei Uploads und
  8. Downloads. Es kommt mit eingebauten Prüfungen für Dateien und Funktionslitäten um Dateien
  9. mit Filtern zu verändern. Protokoll-Adapter erlauben
  10. <classname>Zend_File_Transfer</classname> die selbe API für Transportprotokolle wie HTTP,
  11. FTP, WEBDAV und andere zu verwenden.
  12. </para>
  13. <note>
  14. <title>Einschränkungen</title>
  15. <para>
  16. Die aktuelle Implementation von <classname>Zend_File_Transfer</classname>
  17. ist auf HTTP Post Uploads limitiert. Andere Adapter die Downloads und andere Protokolle
  18. unterstützen werden in zukünftigen Releases hinzugefügt. Aktuell sollte
  19. <classname>Zend_File_Transfer_Adapter_Http</classname> direkt verwendet werden. Sobald
  20. andere Adapter vorhanden sind, kann ein gemeinsames Interface verwendet werden.
  21. </para>
  22. </note>
  23. <note>
  24. <title>Formulare</title>
  25. <para>
  26. Wenn man <classname>Zend_Form</classname> verwendet sollte man die APIs die von
  27. <classname>Zend_Form</classname> zur Verfügung gestellt werden, und
  28. <classname>Zend_File_Transfer</classname> nicht direkt, verwenden. Der Dateitransfer
  29. Support von <classname>Zend_Form</classname> ist in
  30. <classname>Zend_File_Transfer</classname> implementiert, weshalb die Informationen in
  31. diesem Kapitel für fortgeschrittene Benutzer von <classname>Zend_Form</classname>
  32. interessant sind.
  33. </para>
  34. </note>
  35. <para>
  36. Die Verwendung von <classname>Zend_File_Transfer</classname> ist relativ einfach. Es besteht
  37. aus zwei Teilen. Dem HTTP Formular, wärend <classname>Zend_File_Transfer</classname> die
  38. hochgeladenen Dateien behandelt. Siehe das folgende Beispiel:
  39. </para>
  40. <example id="zend.file.transfer.introduction.example">
  41. <title>Einfaches Formular für File-Uploads</title>
  42. <para>
  43. Dieses Beispiel zeigt einen einfachen Dateiupload. Das erste Teil ist das Dateiformular.
  44. In unserem Beispiel gibt es nur eine Datei welche wir hochladen wollen.
  45. </para>
  46. <programlisting language="xml"><![CDATA[
  47. <form enctype="multipart/form-data" action="/file/upload" method="POST">
  48. <input type="hidden" name="MAX_FILE_SIZE" value="100000" />
  49. Choose a file to upload: <input name="uploadedfile" type="file" />
  50. <br />
  51. <input type="submit" value="Upload File" />
  52. </form>
  53. ]]></programlisting>
  54. <para>
  55. Der Bequemlichkeit halber kann
  56. <link linkend="zend.form.standardElements.file">Zend_Form_Element_File</link> verwendet
  57. werden statt das HTML manuell zu erstellen.
  58. </para>
  59. <para>
  60. Der nächste Schritt ist die Erstellung des Empfängers des Uploads. In unserem Beispiel
  61. ist der Empfänger bei <code>/file/upload</code> zu finden. Als nächstes erstellen wir
  62. also den <code>file</code> Controller mit der <code>upload</code> Aktion.
  63. </para>
  64. <programlisting language="php"><![CDATA[
  65. $adapter = new Zend_File_Transfer_Adapter_Http();
  66. $adapter->setDestination('C:\temp');
  67. if (!$adapter->receive()) {
  68. $messages = $adapter->getMessages();
  69. echo implode("\n", $messages);
  70. }
  71. ]]></programlisting>
  72. <para>
  73. Dieses Codebeispiel demonstriert die einfachste Verwendung von
  74. <classname>Zend_File_Transfer</classname>. Ein lokales Ziel wird mit der
  75. <code>setDestination</code> Methode definiert, und anschließend die
  76. <methodname>receive()</methodname> Methode aufgerufen. Wenn irgendwelche Uploadfehler
  77. auftreten werden diese als Ausnahme zurückgegeben.
  78. </para>
  79. </example>
  80. <note>
  81. <title>Achtung</title>
  82. <para>
  83. Dieses Beispiel ist nur für die Demonstration der grundsätzlichen API von
  84. <classname>Zend_File_Transfer</classname>. Man sollte dieses Code Beispiel
  85. <emphasis>niemals</emphasis> in einer Produktivumgebung einsetzen da es massive
  86. Sicherheitslücken aufweisst. Man sollte immer Prüfungen verwenden um die Sicherheit
  87. zu erhöhen.
  88. </para>
  89. </note>
  90. <sect2 id="zend.file.transfer.introduction.adapters">
  91. <title>Von Zend_File_Transfer unterstützte Adapter</title>
  92. <para>
  93. <classname>Zend_File_Transfer</classname> wurde designt um verschiedenste Adapter und
  94. auch Richtungen zu unterstützen. Mit <classname>Zend_File_Transfer</classname> kann man
  95. Dateien Hochladen, Herunterladen und sogar Weiterleiten (Hochladen mit einem Adapter und
  96. Herunterladen mit einem anderen Adapter zur gleichen Zeit).
  97. </para>
  98. </sect2>
  99. <sect2 id="zend.file.transfer.introduction.options">
  100. <title>Optionen für Zend_File_Transfer</title>
  101. <para>
  102. <classname>Zend_File_Transfer</classname> und seine Adapter unterstützen verschiedene
  103. Optionen. Alle Optionen können gesetzt werden indem Sie entweder dem Constructor
  104. übergeben werden, oder durch Aufruf der <methodname>setOptions($options)</methodname>.
  105. <methodname>getOptions()</methodname> gibt die Optionen zurück die aktuell gesetzt
  106. sind. Nachfolgend ist eine Liste aller unterstützten Optionen:
  107. </para>
  108. <itemizedlist>
  109. <listitem>
  110. <para>
  111. <emphasis>ignoreNoFile</emphasis>: Wenn diese Option auf true gesetzt ist,
  112. ignorieren alle Prüfer Dateien die nicht vom Formular hochgeladen wurde. Der
  113. Standardwert ist false, was einen Fehler verursacht wenn die Datei nicht
  114. spezifiziert wurde.
  115. </para>
  116. </listitem>
  117. </itemizedlist>
  118. </sect2>
  119. <sect2 id="zend.file.transfer.introduction.checking">
  120. <title>Dateien prüfen</title>
  121. <para>
  122. <classname>Zend_File_Transfer</classname> hat verschiedene Methoden die auf
  123. verschiedenste Stati von spezifizierten Dateien prüfen. Diese sind nützlich wenn man
  124. Dateien bearbeiten will nachdem Sie empfangen wurden. Diese Methoden beinhalten:
  125. </para>
  126. <itemizedlist>
  127. <listitem>
  128. <para>
  129. <emphasis>isValid($files = null)</emphasis>: Diese Methode prüft ob die
  130. angegebene Datei gültig ist, basierend auf den Prüfungen welche dieser Datei
  131. angehängt sind. Wenn keine Dateien spezifiziert wurden, werden alle Dateien
  132. geprüft. Man kann <methodname>isValid()</methodname> aufrufen bevor
  133. <methodname>receive()</methodname> aufgerufen wird; in diesem Fall ruft
  134. <methodname>receive()</methodname> intern <code>isValid()</code> nicht mehr
  135. auf.
  136. </para>
  137. </listitem>
  138. <listitem>
  139. <para>
  140. <emphasis>isUploaded($files = null)</emphasis>: Diese Methode prüft ob die
  141. spezifizierte Datei vom Benutzer hochgeladen wurde. Das ist nützlich wenn man
  142. eine oder mehrere Dateien definiert hat. Wenn keine Dateien spezifiziert wurden,
  143. werden alle Dateien geprüft.
  144. </para>
  145. </listitem>
  146. <listitem>
  147. <para>
  148. <emphasis>isReceived($files = null)</emphasis>: Diese Methode prüft ob die
  149. spezifizierte Datei bereits empfangen wurde. Wenn keine Dateien angegeben
  150. wurden, werden alle Dateien geprüft.
  151. </para>
  152. </listitem>
  153. </itemizedlist>
  154. <example id="zend.file.transfer.introduction.checking.example">
  155. <title>Dateien prüfen</title>
  156. <programlisting language="php"><![CDATA[
  157. $upload = new Zend_File_Transfer();
  158. // Gibt alle bekannten internen Datei Informationen zurück
  159. $files = $upload->getFileInfo();
  160. foreach ($files as $file => $info) {
  161. // Datei hochgeladen ?
  162. if (!$upload->isUploaded($file)) {
  163. print "Warum hast Du die Datei nicht hochgeladen ?";
  164. continue;
  165. }
  166. // Prüfungen sind ok ?
  167. if (!$upload->isValid($file)) {
  168. print "Sorry, aber die Datei ist nicht das was wir wollten";
  169. continue;
  170. }
  171. }
  172. $upload->receive();
  173. ]]></programlisting>
  174. </example>
  175. </sect2>
  176. <sect2 id="zend.file.transfer.introduction.informations">
  177. <title>Zusätzliche Dateiinformationen</title>
  178. <para>
  179. <classname>Zend_File_Transfer</classname> kann zusätzliche Informationen über Dateien
  180. zurückgeben. Die folgenden Methoden sind vorhanden:
  181. </para>
  182. <itemizedlist>
  183. <listitem>
  184. <para>
  185. <emphasis>getFileName($file = null, $path = true)</emphasis>: Diese Methode
  186. gibt den wirklichen Namen der übertragenen Datei zurück.
  187. </para>
  188. </listitem>
  189. <listitem>
  190. <para>
  191. <emphasis>getFileInfo($file = null)</emphasis>: Diese Methode gibt die
  192. internen Informationen für die angegebene übertragene Datei zurück.
  193. </para>
  194. </listitem>
  195. <listitem>
  196. <para>
  197. <emphasis>getFileSize($file = null)</emphasis>: Diese Methode gibt die
  198. echte Dateigröße für die angegebene Datei zurück.
  199. </para>
  200. </listitem>
  201. <listitem>
  202. <para>
  203. <emphasis>getHash($hash = 'crc32', $files = null)</emphasis>: Diese Methode
  204. gibt einen Hash des Inhalts einer angegebenen übertragenen Datei zurück.
  205. </para>
  206. </listitem>
  207. <listitem>
  208. <para>
  209. <emphasis>getMimeType($files = null)</emphasis>: Diese Methode gibt den
  210. Mimetyp der angegebenen übertragenen Datei zurück.
  211. </para>
  212. </listitem>
  213. </itemizedlist>
  214. <para>
  215. <methodname>getFileName()</methodname> akzeptiert den Namen des Elements als ersten
  216. Parameter. Wenn kein Name angegeben wird, werden alle bekannten Dateinamen in einem
  217. Array zurückgegeben. Wenn die Datei eine MultiDatei ist, wird auch ein Array
  218. zurückgegeben. Wenn nur eine einzelne Datei vorhanden ist wird nur ein String
  219. zurückgegeben.
  220. </para>
  221. <para>
  222. Standardmäßig werden Dateinamen mit dem kompletten Pfad zurückgegeben. Wenn man nur den
  223. Dateinamen ohne Pfad benötigt, kann der zweite Parameter <varname>$path</varname>
  224. gesetzt werden, welcher den Dateinamen ausschneidet wenn er auf false gesetzt wird.
  225. </para>
  226. <example id="zend.file.transfer.introduction.informations.example1">
  227. <title>Den Dateinamen bekommen</title>
  228. <programlisting language="php"><![CDATA[
  229. $upload = new Zend_File_Transfer();
  230. $upload->receive();
  231. // Gibt die Dateinamen aller Dateien zurück
  232. $names = $upload->getFileName();
  233. // Gibt den Dateinamen des Formularelements 'foo' zurück
  234. $names = $upload->getFileName('foo');
  235. ]]></programlisting>
  236. </example>
  237. <note>
  238. <para>
  239. Es ist zu beachten das sich der Dateinamen ändern kann nachdem die Datei empfangen
  240. wurde (receive) weil alle Filter angewendet werden, sobald die Datei empfangen
  241. wurde. Deswegen sollte man <methodname>getFileName()</methodname> immer ausführen
  242. nachdem die Dateien empfangen wurden.
  243. </para>
  244. </note>
  245. <para>
  246. <methodname>getFileSize()</methodname> gibt standardmäßig die echte Dateigröße in SI
  247. Schreibweise zurück was bedeutet das man <code>2kB</code> statt <code>2048</code>
  248. erhält. Wenn man die reine Größe benötigt muß man die <code>useByteString</code> Option
  249. auf false setzen.
  250. </para>
  251. <example id="zend.file.transfer.introduction.informations.example.getfilesize">
  252. <title>Die Größe einer Datei erhalten</title>
  253. <programlisting language="php"><![CDATA[
  254. $upload = new Zend_File_Transfer();
  255. $upload->receive();
  256. // Gibt die Größen aller Dateien als Array zurück
  257. // wenn mehr als eine Datei hochgeladen wurde
  258. $size = $upload->getFileSize();
  259. // Wechsle die SI Schreibweise damit reine Nummern zurückgegeben werden
  260. $upload->setOption(array('useByteString' => false));
  261. $size = $upload->getFileSize();
  262. ]]></programlisting>
  263. </example>
  264. <para>
  265. <methodname>getHash()</methodname> akzeptiert den Namen eines Hash Algorithmus als
  266. ersten Parameter. Für eine Liste bekannter Algorithmen kann in
  267. <ulink url="http://php.net/hash_algos">PHP's hash_algos Methode</ulink> gesehen werden.
  268. Wenn kein Algorithmus spezifiziert wird, wird <code>crc32</code> als Standardalgorithmus
  269. verwendet.
  270. </para>
  271. <example id="zend.file.transfer.introduction.informations.example2">
  272. <title>Den Hash einer Datei erhalten</title>
  273. <programlisting language="php"><![CDATA[
  274. $upload = new Zend_File_Transfer();
  275. $upload->receive();
  276. // Gibt die Hashes von allen Dateien als Array zurück
  277. // wenn mehr als eine Datei hochgeladen wurde
  278. $hash = $upload->getHash('md5');
  279. // Gibt den Has für das 'foo' Formularelement zurück
  280. $names = $upload->getHash('crc32', 'foo');
  281. ]]></programlisting>
  282. </example>
  283. <note>
  284. <para>
  285. Es ist zu beachten das der zurückgegebene Wert ein Array ist, wenn die Datei oder
  286. der Formularname mehr als eine Datei enthält.
  287. </para>
  288. </note>
  289. <para>
  290. <methodname>getMimeType()</methodname> gibt den Mimetyp einer Datei zurück. Wenn mehr
  291. als eine Datei hochgeladen wurde wird ein Array zurückgegeben, andernfalls ein String.
  292. </para>
  293. <example id="zend.file.transfer.introduction.informations.getmimetype">
  294. <title>Den Mimetyp einer Datei bekommen</title>
  295. <programlisting language="php"><![CDATA[
  296. $upload = new Zend_File_Transfer();
  297. $upload->receive();
  298. $mime = $upload->getMimeType();
  299. // Gibt den Mimetyp des 'foo' Form Elements zurück
  300. $names = $upload->getMimeType('foo');
  301. ]]></programlisting>
  302. </example>
  303. <note>
  304. <para>
  305. Beachte das diese Methode die fileinfo Erweiterung verwendet wenn Sie vorhanden ist.
  306. Wenn diese Erweiterung nicht gefunden werden kann wird die mimemagic Erweiterung
  307. verwendet. Wenn keine Erweiterung gefunden wird, dann wird der Mimetyp verwendet der
  308. vom Dateiserver wärend des Hochladens der Datei angegeben wurde.
  309. </para>
  310. </note>
  311. </sect2>
  312. <sect2 id="zend.file.transfer.introduction.uploadprogress">
  313. <title>Fortschritt für Datei Uploads</title>
  314. <para>
  315. <classname>Zend_File_Transfer</classname> kann den aktuellen Status eines gerade
  316. stattfindenden Datei Uploads erheben. Um dieses Feature zu verwenden muß man entweder
  317. die <code>APC</code> Erweiterung verwenden, die mit den meisten standardmäßigen PHP
  318. Installationen vorhanden ist, oder die <code>uploadprogress</code> Erweiterung. Beide
  319. Erweiterungen werden erkannt und automatisch verwendet. Um den Fortschritt zu erhalten
  320. muß man einige Voraussetzungen erfüllen.
  321. </para>
  322. <para>
  323. Erstens, muß man entweder <code>APC</code> oder <code>uploadprogress</code> aktiviert
  324. haben. Es ist zu beachten das dieses Feature von <code>APC</code> in der eigenen php.ini
  325. ausgeschaltet werden kann.
  326. </para>
  327. <para>
  328. Zweitens, muß man die richtigen unsichtbaren Felder im Formular hinzugefügt haben das
  329. die Dateien sendet. Wenn man <classname>Zend_Form_Element_File</classname> verwendet
  330. werden diese unsichtbaren Felder automatisch von <classname>Zend_Form</classname>
  331. hinzugefügt.
  332. </para>
  333. <para>
  334. Wenn die oberen zwei Punkte vorhanden sind dann ist man in der Lage den aktuellen
  335. Fortschritt des Datei uploads zu erhalten indem man die <code>getProgress</code> Methode
  336. verwendet. Aktuell gibt es 2 offizielle Wege um das hand zu haben.
  337. </para>
  338. <sect3 id="zend.file.transfer.introduction.uploadprogress.progressadapter">
  339. <title>Verwenden eines Progressbar Adapters</title>
  340. <para>
  341. Man kann einen bequemen <emphasis>Zend_ProgressBar</emphasis> verwenden um den
  342. aktuellen Fortschritt zu erhalten und kann Ihn dann auf einfachem Wege dem Benutzer
  343. zeigen.
  344. </para>
  345. <para>
  346. Um das zu ermöglichen, muß man den gewünschten
  347. <emphasis>Zend_ProgressBar_Adapter</emphasis> bei
  348. <methodname>getProgress()</methodname> hinzufügen wenn es das erste Mal aufgerufen
  349. wird. Für Details über den zu verwendenden Adapter, bitte im Kapitel <link
  350. linkend="zend.progressbar.adapters">Zend_ProgressBar Standard Adapters</link>
  351. nachsehen.
  352. </para>
  353. <example id="zend.file.transfer.introduction.uploadprogress.progressadapter.example1">
  354. <title>
  355. Verwenden eines Progressbar Adapters um den aktuellen Status zu erhalten
  356. </title>
  357. <programlisting language="php"><![CDATA[
  358. $adapter = new Zend_ProgressBar_Adapter_Console();
  359. $upload = Zend_File_Transfer_Adapter_Http::getProgress($adapter);
  360. $upload = null;
  361. while (!$upload['done']) {
  362. $upload = Zend_File_Transfer_Adapter_Http:getProgress($upload);
  363. }
  364. ]]></programlisting>
  365. </example>
  366. <para>
  367. Die komplette Handhabung wird von <methodname>getProgress()</methodname> im
  368. Hintergrund durchgeführt.
  369. </para>
  370. </sect3>
  371. <sect3 id="zend.file.transfer.introduction.uploadprogress.manually">
  372. <title>getProgress() händisch verwenden</title>
  373. <para>
  374. Man kann mit <methodname>getProgress()</methodname> auch händisch arbeiten, also
  375. ohne die Verwendung von <classname>Zend_ProgressBar</classname>.
  376. </para>
  377. <para>
  378. <methodname>getProgress()</methodname> muß ohne Einstellungen aufgerufen werden. Es
  379. gibt anschließend ein Array mit verschiedenen Schlüssel zurück. Sie unterscheiden
  380. sich, abhängig von der verwendeten PHP Extension. Aber die folgenden Schlüssel
  381. werden unabhängig von der Extension zurück gegeben:
  382. </para>
  383. <itemizedlist>
  384. <listitem>
  385. <para>
  386. <emphasis>id</emphasis>: Die ID dieses Uploads. Die ID identifiziert den
  387. Upload in der Extension. Sie wird automatisch geschrieben. Man sollte Sie
  388. nie ändern oder den Wert selbst setzen.
  389. </para>
  390. </listitem>
  391. <listitem>
  392. <para>
  393. <emphasis>total</emphasis>: Die komplette Größe der Datei die hochgeladen
  394. wird in Bytes als Integer.
  395. </para>
  396. </listitem>
  397. <listitem>
  398. <para>
  399. <emphasis>current</emphasis>: Die aktuelle hochgeladene Größe der
  400. Datei in Bytes als Integer.
  401. </para>
  402. </listitem>
  403. <listitem>
  404. <para>
  405. <emphasis>rate</emphasis>: Die durchschnittliche Geschwindigkeit des
  406. Uploads in Bytes pro Sekunde als Integer.
  407. </para>
  408. </listitem>
  409. <listitem>
  410. <para>
  411. <emphasis>done</emphasis>: Gibt true zurück wenn der Upload abgeschlossen
  412. wurde, andernfalls false.
  413. </para>
  414. </listitem>
  415. <listitem>
  416. <para>
  417. <emphasis>message</emphasis>: Die aktuelle Meldung. Entweder der
  418. Fortschritt als Text in der Form <emphasis>10kB / 200kB</emphasis>, oder
  419. eine hilfreiche Nachricht im Fall eines Problems. Probleme könnten sein,
  420. das kein Upload durchgeführt wird, das ein Fehler wärend des Empfangens der
  421. Daten, für den Fortschritt, aufgetreten ist, oder das der Upload abgebrochen
  422. wurde.
  423. </para>
  424. </listitem>
  425. <listitem>
  426. <para>
  427. <emphasis>progress</emphasis>: Dieser optionale Schlüssel nimmt eine
  428. Instanz von Zend_ProgressBar_Adapter oder Zend_ProgressBar, und erlaubt es,
  429. den aktuellen Status des Uploads, in einer Progressbar zu erhalten
  430. </para>
  431. </listitem>
  432. <listitem>
  433. <para>
  434. <emphasis>session</emphasis>: Dieser optionale Schlüssel nimmt den Namen
  435. eines Session Namespaces entgegen der in Zend_ProgressBar verwendet wird.
  436. Wenn dieser Schlüssel nicht angegeben wird, ist er standardmäßig
  437. <classname>Zend_File_Transfer_Adapter_Http_ProgressBar</classname>.
  438. </para>
  439. </listitem>
  440. </itemizedlist>
  441. <para>
  442. Alle anderen zurückgegebenen Schlüssel werden direkt von den Extensions übernommen
  443. und werden nicht geprüft.
  444. </para>
  445. <para>
  446. Das folgende Beispiel zeigt eine mögliche händische Verwendung:
  447. </para>
  448. <example id="zend.file.transfer.introduction.uploadprogress.manually.example1">
  449. <title>Händische Verwendung des Datei Fortschritts</title>
  450. <programlisting language="php"><![CDATA[
  451. $upload = Zend_File_Transfer_Adapter_Http::getProgress();
  452. while (!$upload['done']) {
  453. $upload = Zend_File_Transfer_Adapter_Http:getProgress($upload);
  454. print "\nAktueller Fortschritt:".$upload['message'];
  455. // Tu was zu tun ist
  456. }
  457. ]]></programlisting>
  458. </example>
  459. </sect3>
  460. </sect2>
  461. </sect1>
  462. <!--
  463. vim:se ts=4 sw=4 tw=80 et:
  464. -->