Runonce: Unterschied zwischen den Versionen

Aus Contao Community Documentation

K (Modulbasierte runonce.php als System runonce.php)
K (Geshi-Anpassung)
Zeile 12: Zeile 12:
 
==Anwendungsbeispiele==
 
==Anwendungsbeispiele==
 
===Löschen einer Datei===
 
===Löschen einer Datei===
<?php @error_reporting(0); @ini_set("display_errors", 0);  
+
 
try {  
+
<source lang="php">
    // Datei relativ zu TL_ROOT
+
<?php @error_reporting(0); @ini_set("display_errors", 0);  
    $file = 'system/modules/demo/delete_me.gif';  
+
try {  
    $objFiles = Files::getInstance();
+
    // Datei relativ zu TL_ROOT
    $objFiles->delete($file);
+
    $file = 'system/modules/demo/delete_me.gif';  
} catch (Exception $e) { $errors[] = $e->getMessage(); }
+
    $objFiles = Files::getInstance();
?>
+
    $objFiles->delete($file);
 +
} catch (Exception $e) { $errors[] = $e->getMessage(); }
 +
?>
 +
</source>
 +
 
 
Die Fehlerausgaben, sollte es welche geben, werden in diesem Beispiel unterdrückt.
 
Die Fehlerausgaben, sollte es welche geben, werden in diesem Beispiel unterdrückt.
  
 
===Datenbank Insert / Update===
 
===Datenbank Insert / Update===
<?php @error_reporting(0); @ini_set("display_errors", 0);  
+
<source lang="php">
$objDatabase = Database::getInstance();
+
<?php @error_reporting(0); @ini_set("display_errors", 0);  
//
+
$objDatabase = Database::getInstance();
// Update database
+
//
try {  
+
// Update database
    $objDatabase->execute("UPDATE `tl_demo_table` SET `demo_counter`=0 WHERE `demo_browser`='Unknown'");
+
try {  
} catch (Exception $e) { $errors[] = $e->getMessage(); }
+
    $objDatabase->execute("UPDATE `tl_demo_table` SET `demo_counter`=0 WHERE demo_browser`='Unknown'");
//
+
} catch (Exception $e) { $errors[] = $e->getMessage(); }
// Insert database
+
//
try {  
+
// Insert database
    $objDatabase->execute("INSERT INTO `tl_demo_table` (`id`, `demo_counter`) VALUES (0, '10')");  
+
try {  
} catch (Exception $e) { $errors[] = $e->getMessage(); }
+
    $objDatabase->execute("INSERT INTO `tl_demo_table` (`id`, `demo_counter`) VALUES (0, '10')");  
?>
+
} catch (Exception $e) { $errors[] = $e->getMessage(); }
 +
?>
 +
</source>
  
 
===Datenbank Insert, OOP Variante===
 
===Datenbank Insert, OOP Variante===
 
Quelle: Forum.
 
Quelle: Forum.
<?php
+
<source lang="php">
class RunonceJob extends Frontend
+
<?php
{
+
class RunonceJob extends Frontend
    public function __construct()
+
{
    {
+
  public function __construct()
        parent::__construct();
+
  {
    }
+
      parent::__construct();
    public function run()
+
  }
    {
+
  public function run()
        $arrInsert=array(
+
  {
            'action'    => 'runonce',
+
      $arrInsert=array(
            'text'      => 'runonce'
+
          'action'    => 'runonce',
        );
+
          'text'      => 'runonce'
        $this->Database->prepare("INSERT INTO tl_log %s")->set($arrInsert)->execute();
+
      );
    }
+
      $this->Database->prepare("INSERT INTO tl_log %s")->set($arrInsert)->execute();
}
+
  }
$objRunonceJob = new RunonceJob();
+
}
$objRunonceJob->run();
+
$objRunonceJob = new RunonceJob();
?>
+
$objRunonceJob->run();
 +
?>
 +
</source>
  
 
===Hinweis===
 
===Hinweis===
Zeile 71: Zeile 79:
 
Diese ''runonce.php'' wird beim nächsten Seitenaufruf einmalig ausgeführt und danach gelöscht.
 
Diese ''runonce.php'' wird beim nächsten Seitenaufruf einmalig ausgeführt und danach gelöscht.
  
$runonceFile = '/system/modules/MODULNAME/config/runonce.php';
+
<source lang="php">
if (file_exists(TL_ROOT . $runonceFile))
+
$runonceFile = '/system/modules/MODULNAME/config/runonce.php';
{
+
if (file_exists(TL_ROOT . $runonceFile))
  include(TL_ROOT . $runonceFile);
+
{
  $objFiles = Files::getInstance();
+
include(TL_ROOT . $runonceFile);
  $objFiles->delete($runonceFile);
+
$objFiles = Files::getInstance();
}
+
$objFiles->delete($runonceFile);
 +
}
 +
</source>
  
 
Nachteil dieser Variante: Die ER Verwaltung meckert, dass das Modul unvollständig sei, da nun eine Datei fehlt.<br />
 
Nachteil dieser Variante: Die ER Verwaltung meckert, dass das Modul unvollständig sei, da nun eine Datei fehlt.<br />
 
Eine Idee wäre, die Datei nicht zu löschen, sondern zu überschreiben:
 
Eine Idee wäre, die Datei nicht zu löschen, sondern zu überschreiben:
  
$runonceFile = 'system/modules/MODULNAME/config/runonce.php'; // relativ zu TL_ROOT
+
<source lang="php">
if (file_exists(TL_ROOT . '/' . $runonceFile))
+
$runonceFile = 'system/modules/MODULNAME/config/runonce.php'; // relativ zu TL_ROOT
{
+
if (file_exists(TL_ROOT . '/' . $runonceFile))
    $GLOBALS['runonce']['MODULNAME'] = false;
+
{
    include(TL_ROOT . '/' . $runonceFile);
+
  $GLOBALS['runonce']['MODULNAME'] = false;
    if ($GLOBALS['runonce']['MODULNAME'] === false)  
+
  include(TL_ROOT . '/' . $runonceFile);
    {
+
  if ($GLOBALS['runonce']['MODULNAME'] === false)  
        $objFiles = Files::getInstance();
+
  {
        $objFiles->delete($runonceFile); // hier wird intern ein "TL_ROOT/" vorgesetzt
+
      $objFiles = Files::getInstance();
        //nun wieder neu anlegen mit neuem Inhalt
+
      $objFiles->delete($runonceFile); // hier wird intern ein "TL_ROOT/" vorgesetzt
        $objFile = new File($runonceFile); // hier wird intern ein "TL_ROOT/" vorgesetzt
+
      //nun wieder neu anlegen mit neuem Inhalt
        $objFile->write("<?php \$GLOBALS['runonce']['MODULNAME'] = true; ?>");
+
      $objFile = new File($runonceFile); // hier wird intern ein "TL_ROOT/" vorgesetzt
        $objFile->close();
+
      $objFile->write("<?php \$GLOBALS['runonce']['MODULNAME'] = true; ?>");
    }
+
      $objFile->close();
}
+
  }
 +
}
 +
</source>
  
 
Wobei man nun die file_exists Prüfung weglassen könnte.<br />
 
Wobei man nun die file_exists Prüfung weglassen könnte.<br />
Zeile 119: Zeile 131:
 
Aus diesem Grund hier beide Dateien für ein einfaches Beispiel.<br />
 
Aus diesem Grund hier beide Dateien für ein einfaches Beispiel.<br />
 
Hier nun der geänderte Code - config.php:<br>
 
Hier nun der geänderte Code - config.php:<br>
<pre>
+
 
 +
<source lang="php">
 
$runonceJob  = 'system/modules/MODULNAME/config/RunonceJob.php'; // eigene runonce
 
$runonceJob  = 'system/modules/MODULNAME/config/RunonceJob.php'; // eigene runonce
 
$runonceFile = 'system/runonce.php'; // system runonce
 
$runonceFile = 'system/runonce.php'; // system runonce
Zeile 136: Zeile 149:
 
$objFile->close();
 
$objFile->close();
 
}
 
}
</pre>
+
</source>
 +
 
 
Hier wird über die Dateigröße geprüft, ob die Datei noch kopiert werden muss oder nicht.<br /><br />
 
Hier wird über die Dateigröße geprüft, ob die Datei noch kopiert werden muss oder nicht.<br /><br />
 
Hier nun der Aufbau der passenden eigenen runonce.php, im obigem Beispiel die RunonceJob.php:
 
Hier nun der Aufbau der passenden eigenen runonce.php, im obigem Beispiel die RunonceJob.php:
<pre>
+
 
 +
<source lang="php">
 
<?php @error_reporting(0); @ini_set("display_errors", 0);   
 
<?php @error_reporting(0); @ini_set("display_errors", 0);   
  
Zeile 159: Zeile 174:
 
}
 
}
 
?>
 
?>
</pre>
+
</source>
 +
 
 
Wichtig ist also, '''jede''' SQL Anweisung in ''<code>try { .... } catch (Exception $e) { $errors[] = $e->getMessage(); }</code>'' zu kapseln.
 
Wichtig ist also, '''jede''' SQL Anweisung in ''<code>try { .... } catch (Exception $e) { $errors[] = $e->getMessage(); }</code>'' zu kapseln.
<br />
 
<br />
 
  
 
==Die Zukunft==
 
==Die Zukunft==

Version vom 3. Oktober 2010, 17:08 Uhr


betrifft
TYPOlight Version ab 2.7
Contao Version ab 2.9

Live Update nutzt diese, die Extensions nutzen diese auch: die Datei TL_ROOT/system/runonce.php

Diese Datei tut genau das, was der Name schon vermuten lässt. Sie wird nur einmal ausgeführt und anschließend gelöscht.
Bei jedem Seitenaufruf wird dazu geprüft, ob diese vorhanden ist und ggf. ausgeführt.

In der Extension legt man diese Datei dann im Pfad wie oben bereits angegeben ab.


Anwendungsbeispiele

Löschen einer Datei

<?php @error_reporting(0); @ini_set("display_errors", 0); 
try { 
    // Datei relativ zu TL_ROOT
    $file = 'system/modules/demo/delete_me.gif'; 
    $objFiles = Files::getInstance();
    $objFiles->delete($file);
} catch (Exception $e) { $errors[] = $e->getMessage(); }
?>

Die Fehlerausgaben, sollte es welche geben, werden in diesem Beispiel unterdrückt.

Datenbank Insert / Update

<?php @error_reporting(0); @ini_set("display_errors", 0); 
$objDatabase = Database::getInstance();
//
// Update database
try { 
    $objDatabase->execute("UPDATE `tl_demo_table` SET `demo_counter`=0 WHERE demo_browser`='Unknown'");
} catch (Exception $e) { $errors[] = $e->getMessage(); }
//
// Insert database
try { 
    $objDatabase->execute("INSERT INTO `tl_demo_table` (`id`, `demo_counter`) VALUES (0, '10')"); 
} catch (Exception $e) { $errors[] = $e->getMessage(); }
?>

Datenbank Insert, OOP Variante

Quelle: Forum.

<?php
class RunonceJob extends Frontend
{
   public function __construct()
   {
       parent::__construct();
   }
   public function run()
   {
       $arrInsert=array(
           'action'    => 'runonce',
           'text'      => 'runonce'
       );
       $this->Database->prepare("INSERT INTO tl_log %s")->set($arrInsert)->execute();
   }
}
$objRunonceJob = new RunonceJob();
$objRunonceJob->run();
?>

Hinweis

Achtung.png Achtung: Vorsicht damit bei Extensions mit Abhängigkeiten zu weiteren Extensions.

Bringen 2 Extensions jeweils eine runonce.php mit, wird nur eine ausgeführt!

Hier ist der Author nicht sicher, ob die erste oder die letzte davon.


Modulbasierte runonce.php

Um den obigen Nachteil aus dem Weg zu gehen, wurde über eine modulbasierte runonce.php nachgedacht.
Den nachfolgenden Code in die config.php des eigenen Modules einfügen. Den Pfad anpassen, "MODULNAME" durch den Verzeichnisnamen des eigenen Moduls ersetzen und eine runonce.php im config Verzeichnis erstellen.
Diese runonce.php wird beim nächsten Seitenaufruf einmalig ausgeführt und danach gelöscht.

$runonceFile = '/system/modules/MODULNAME/config/runonce.php';
if (file_exists(TL_ROOT . $runonceFile))
{
 include(TL_ROOT . $runonceFile);
 $objFiles = Files::getInstance();
 $objFiles->delete($runonceFile);
}

Nachteil dieser Variante: Die ER Verwaltung meckert, dass das Modul unvollständig sei, da nun eine Datei fehlt.
Eine Idee wäre, die Datei nicht zu löschen, sondern zu überschreiben:

$runonceFile = 'system/modules/MODULNAME/config/runonce.php'; // relativ zu TL_ROOT
if (file_exists(TL_ROOT . '/' . $runonceFile))
{
   $GLOBALS['runonce']['MODULNAME'] = false;
   include(TL_ROOT . '/' . $runonceFile);
   if ($GLOBALS['runonce']['MODULNAME'] === false) 
   {
       $objFiles = Files::getInstance();
       $objFiles->delete($runonceFile); // hier wird intern ein "TL_ROOT/" vorgesetzt
       //nun wieder neu anlegen mit neuem Inhalt
       $objFile = new File($runonceFile); // hier wird intern ein "TL_ROOT/" vorgesetzt
       $objFile->write("<?php \$GLOBALS['runonce']['MODULNAME'] = true; ?>");
       $objFile->close();
   }
}

Wobei man nun die file_exists Prüfung weglassen könnte.
Andererseits könnte man diesen Abschnitt immer im Modul lassen und nur durch Setzen der Variablen $runonceFile bestimmen, ob es was zu tun gibt oder nicht.

Probleme

Einen kleinen Nebeneffekt hat die ganze Sache. Es kommt zu einer Fehlermeldung, sobald man die Datenbank Instanz nutzt; egal ob die funktions- oder objektorientierte Variante verwendet wird.
Am Ende der Seite erscheint die Meldung:

Fatal error: Exception thrown without a stack frame in Unknown  on line 0

Das Script selbst wird dabei ohne Probleme abgearbeitet, auch das Backend selbst wird dadurch nicht gestört.

Modulbasierte runonce.php als System runonce.php

Durch Analyse einer runonce.php bei einem Liveupdate wurde klar, wenn diese aus dem System heraus aufgerufen wird und nicht aus einer config Datei, gibt es keine Fehlermeldung.
Durch langes Experimentieren ist der Author auf eine Lösung gekommen, die zumindest bei ihm funktioniert.
Prinzip:

  • in der config.php wird geprüft ob eine Datei /system/runonce.php existiert
  • Ja: Abbruch, beim nächsten Seitenaufruf wird erneut geprüft.
  • Nein: die mitgebrachte RunonceJob.php wird nach /system/runonce.php kopiert und dadurch beim nächsten Seitenaufruf abgearbeitet.
    • die mitgebrachte wird überschrieben, um später zu erkennen, dass diese bereits ausgeführt wurde.

Damit die Fehlermeldung nicht auftritt, muss auch die eigentliche RunonceJob.php einen speziellen Syntax haben.
Aus diesem Grund hier beide Dateien für ein einfaches Beispiel.
Hier nun der geänderte Code - config.php:

$runonceJob  = 'system/modules/MODULNAME/config/RunonceJob.php'; // eigene runonce
$runonceFile = 'system/runonce.php'; // system runonce
 
if ( (file_exists(TL_ROOT . '/' . $runonceJob)) && (!file_exists(TL_ROOT . '/' . $runonceFile)) ) 
{
	//keine /system/runonce.php, let's go
	$objFile = new File($runonceJob); // hier wird intern ein "TL_ROOT/" vorgesetzt
	if ($objFile->filesize > 100) 
	{
		$objFiles = Files::getInstance();
		$objFiles->copy($runonceJob,$runonceFile);
		//
		$objFile->write("<?php // Module Migration Complete ?>");  // Datei muss kleiner 100 Zeichen werden
	}
	$objFile->close();
}

Hier wird über die Dateigröße geprüft, ob die Datei noch kopiert werden muss oder nicht.

Hier nun der Aufbau der passenden eigenen runonce.php, im obigem Beispiel die RunonceJob.php:

<?php @error_reporting(0); @ini_set("display_errors", 0);  
 
if (version_compare(VERSION . '.' . BUILD, '2.8.9', '>'))  // die soll nur ab Contao 2.9 ausgeführt werden
{
    $objDatabase = Database::getInstance();
    $objDatabase->listTables();
 
    // Nur wenn tl_meinmodul.feld_01 existiert aber tl_module.feld_02 nicht, gibt es was zu tun
    if ($objDatabase->fieldExists('feld_01', 'tl_meinmodul') 
    && !$objDatabase->fieldExists('feld_02', 'tl_module'))
    {
        //Migration mit Neufeldanlegung
        //Feld anlegen
        try { $objDatabase->execute("ALTER TABLE `tl_module` ADD `feld_02` varchar(32) NOT NULL default ''"); } 
        catch (Exception $e) { $errors[] = $e->getMessage(); }
       // Nun würden hier weiter Anweisungen folgen, z.B. das neue Feld gleich zu füllen.
    }
}
?>

Wichtig ist also, jede SQL Anweisung in try { .... } catch (Exception $e) { $errors[] = $e->getMessage(); } zu kapseln.

Die Zukunft

Die letzte Lösung funktioniert. Aber es wäre besser, wenn es die Möglichkeit gäbe, dass alle Module ihre eigene runonce.php mitbringen im Verzeichnis /system, ohne das diese gegenseitig überschrieben werden.
Auch dazu gibt es schon eine Überlegung wie das geschafft werden kann. Das ganze kopieren und prüfen fällt dann weg.
Ein Proof of Concept wird es dazu demnächst hier geben. Da es natürlich nicht ohne Änderungen am Core geht, wird dieses dann auch als Feature Request im Ticket System eingestellt.


--BugBuster 22:03, 22. Jul. 2010 (CEST)

Ansichten
Meine Werkzeuge

Contao Community Documentation

Andreas, leg dich da hinten hin und schlaf, dann kommen wir vorwaerts.

Tristan Lins
Navigation
Verstehen
Verwenden
Entwickeln
Verschiedenes
Werkzeuge