I’m writing a PHP app that has a ‘control panel’ that writes a prefs file with certain variables. On every POST, if the file doesn’t exist, it is created. If it does exist, it is unlinked and a new file is touched with the same filename and new variables. This file is then included on another page with displays content based on the variables inside it.
$file = "phpsettings.php";
if (!file_exists($file)) {
touch($file);
$handle = fopen ($file, 'r+');
$str = "<?php \$pref1 = \"$mypref\"; ?>";
} else {
unlink($file);
touch($file);
$handle = fopen ($file, 'r+');
$str = "<?php \$pref1 = \"$mypref\"; ?>";
}
fwrite ($handle, $str);
fclose ($handle);
Is this a safe way of writing preferences, provided this file will be overwritten many times per day? What is a good way of both alerting the user of this control panel if the file wasn’t saved correctly, and in that case, what would be a good contingency plan to avoid breaking the page this prefs file is included on short of defining a default set of variables to fill if !(file_exists)?
If you store your settings in an array, you can serialize() them and write to a text file, rather than writing raw php to a php file and including it.
If you’re not sanitising your input for those preferences, and say $mypref1 represents someone’s name, there’s nothing stopping them from filling this out in the form field:
and your resulting PHP will become
So firstly, storing your preferences in an array and using serialize() is much safer:
In your question, you also mention that if the file does exist, it is unlinked. You can simply truncate it to zero length by passing “w” as the second argument to fopen – you don’t need to manually delete it. This should set the mtime anyway, negating the need for the call to touch().
If the values being written to the file are preferences, surely each preference could have a default, unless there are hundreds? array_merge will allow you to overwrite on a per-key basis, so if you do something like this:
If the issue is that there are heaps, and you don’t want to have to initialise them all, you could have a get_preference method which just wraps an isset call to the prefs array.
Beyond all of the questions this raises though, the reality is that with your app, in the unlikely event that something does go wrong with the fopen, it should be regarded as a serious failure anyway, and the handful of users you’re likely to have making use of this feature are going to be contacting you pretty darn quick if something goes wrong.