discourse-legacysite-perl/site/glist/lib/GT/Config.pm
2024-06-17 21:49:12 +10:00

928 lines
34 KiB
Perl

# ====================================================================
# Gossamer Threads Module Library - http://gossamer-threads.com/
#
# GT::Config
# Author: Jason Rhinelander
# CVS Info :
# $Id: Config.pm,v 1.45 2005/03/21 05:49:39 jagerman Exp $
#
# Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved.
# ====================================================================
#
# Description:
# A module for handling loading and caching of configuration files.
#
package GT::Config;
# ===============================================================
use strict;
use GT::Base qw/PERSIST/; # Due to the nature of the config file's hash-like interface, we can't inherit from GT::Base - it sets things in $self. We do need GT::Base for its in_eval function though.
use GT::Template::Inheritance;
use GT::AutoLoader;
use constants
DATA => 0,
INHERITED => 1,
FILES => 2,
FILES_MOD => 3,
CODE_STR => 4;
use vars qw(%ATT %ATTRIBS %CACHE %SUB_CACHE $error $ERRORS $VERSION);
# %ATT stores the default attribute values
# %ATTRIBS stores the attributes of each object. Since each object works exactly
# like a hash ref of the data it represents, these attributes cannot be stored
# in $self.
# %CACHE is used to cache any data of objects using the 'cache' option. Each
# file in here has an array ref value - the first value is a hash ref of the
# data, the second a hash ref of inherited keys, the third an array of the
# files inherited from, and the fourth a hash of [size, last modification
# time] pairs of those files.
# %SUB_CACHE is exactly like %CACHE, except that values starting with 'sub {'
# will be compiled into code refs. Each array ref has a fifth value - a hash
# reference list that stores the original value of any code refs that have
# been compiled. %SUB_CACHE is only used when you use 'compile_subs'. Also,
# because different packages can be specified, this stores which package the
# code ref was compiled in.
# $error stores any error that occurs. If a load error happens, you'll need to
# use $error to get the error message (when not using the 'create_ok' option).
# $ERRORS stores all the error codes
# $VERSION - $Id: Config.pm,v 1.45 2005/03/21 05:49:39 jagerman Exp $ - The version.
$VERSION = sprintf "%d.%03d", q$Revision: 1.45 $ =~ /(\d+)\.(\d+)/;
%ATT = (
inheritance => 0, # If set, looks for .tplinfo files for inheritance.
local => 0, # If set, will look for "local" directories containing the file. The file will be saved in a "local" subdirectory of the directory given.
cache => 1, # If set, GT::Config will look in the cache for the object; objects are always stored in the cache, so that ->load(cache => 0) can be used to reload a file.
create_ok => 0, # If set, you'll get a GT::Config object even if the file doesn't exist. You can then save() it to create the file. If not set, a fatal error occurs if the file cannot be located. Note that if the file exists, but has a syntax error, or cannot be read, a fatal error will occur regardless of this option.
empty => 0, # If specified, nothing will be read from disk - can be used to force a new, blank config file
chmod => 0666, # The octal permissions to set on the file immediately after saving
strict => 0, # If true, a fatal error will occur when attempting to access a key that does not exist.
debug => 0, # If true, warnings and debugging will be printing to STDERR
tmpfile => undef, # Possible values: 0, undef, 1. 0 = no tempfile, undef = tempfile if dir writable, 1 = always tempfile
header => '', # Can be set to anything. When saving, this will go before the data. Keep in mind, this has to be correct Perl. [localtime] in here will be replaced with scalar localtime() when saving.
compile_subs => '', # Must be set to a package. If set, any value that starts with 'sub {' will be compiled into a code ref, in the package specified.
sort_order => undef, # Passed to GT::Dumper->dump as 'order' value if set
tab => "\t", # What to use for a "tab" in the config file. Defaults to an actual tab.
);
# Other attributes used internally:
# filename => '', # Whatever you give as the filename
# file => '', # Just the filename (no path)
# path => '', # The path of the filename
# files => {}, # A hash of filename => last_mod_time (may contain multiple entries to support inheritance).
# file_order => [], # The order of the files in 'files'
# data => {}, # The actual data of the config file.
# inherited => {}, # Each base key inherited will have $key => 1 in here. Inherited keys are not saved, unless they are changed between load time and save time.
# compiled => {}, # Any keys that start with 'sub {' will be compiled into code refs if the compile_subs option is on. The code reference is saved here so that recompiling is not necessary
$ERRORS = {
CANT_LOAD => q _Unable to load '%s': %s._,
CANT_COMPILE => q _Unable to compile '%s': %s._,
CANT_FIND => q _Config file '%s' does not exist in directory '%s' or has incorrect permissions set._,
CANT_WRITE => q _Unable to open '%s' for writing: %s._,
CANT_PRINT => q _Unable to write to file '%s': %s._,
CANT_RENAME => q _Unable to move '%s' to '%s': %s._,
WRITE_MISMATCH => q _Unable to save '%s': wrote %d bytes, but file is %d bytes_,
CANT_CREATE_DIR => q _Unable to create directory '%s': %s._,
NOT_HASH => q _Config file '%s' did not return a hash reference._,
BAD_ARGS => q _Bad arguments. Usage: %s_,
NOT_FILE => q _'%s' does not look like a valid filename_,
RECURSION => q _Recursive inheritance detected and interrupted: '%s'_,
UNKNOWN_OPT => q _Unknown option '%s' passed to %s_,
BAD_KEY => q _The key you attempted to access, '%s', does not exist in '%s'_,
CANT_COMPILE_CODE => q _Unable to compile '%s' in file '%s': %s_
};
sub load {
my $class = shift;
my (%attribs, %data);
tie %data, $class, \%attribs;
my $self = bless \%data, ref $class || $class;
$ATTRIBS{$self} = \%attribs; # hehehe ;-)
my $filename = shift or return $self->error(BAD_ARGS => FATAL => 'GT::Config->load("/path/to/config/file", { opts })');
$attribs{filename} = $filename;
$attribs{filename_given} = $filename;
@attribs{'path', 'file'} = ($filename =~ m|^(.*?)[\\/]?([^\\/]+)$|) or return $self->error(NOT_FILE => FATAL => $filename);
$attribs{path} = '.' unless length $attribs{path};
$filename = $attribs{filename} = "$attribs{path}/$attribs{file}"; # _load_data/_load_tree depend on it being like this.
my $opts = shift || {};
ref $opts eq 'HASH' or return $self->error(BAD_ARGS => FATAL => 'GT::Config->load("/path/to/config/file", { opts })');
for (keys %ATT) {
if (/^(?:inheritance|local|cache|create_ok|strict|empty)$/) {
$attribs{$_} = exists $opts->{$_} ? (delete $opts->{$_} ? 1 : 0) : $ATT{$_};
}
elsif ($_ eq 'tmpfile') {
if (exists $opts->{$_}) {
my $tmpfile = delete $opts->{$_};
$attribs{$_} = defined($tmpfile) ? $tmpfile ? 1 : 0 : undef;
}
else {
$attribs{$_} = $ATT{$_};
}
}
else {
$attribs{$_} = exists $opts->{$_} ? delete $opts->{$_} : $ATT{$_};
}
}
$self->debug("Received '$filename' for the file to load", 2) if $attribs{debug} >= 2;
if (keys %$opts) {
$self->error(UNKNOWN_OPT => FATAL => keys %$opts => ref($self) . '->load');
}
$self->debug("Loading '$filename' with options: inheritance => '$attribs{inheritance}', local => '$attribs{local}', cache => '$attribs{cache}', create_ok => '$attribs{create_ok}', empty => '$attribs{empty}', chmod => '$attribs{chmod}', strict => '$attribs{strict}', debug => '$attribs{debug}', compile_subs => '$attribs{compile_subs}'") if $attribs{debug};
$self->debug("Header: '$attribs{header}'", 2) if $attribs{debug} >= 2;
if ($attribs{empty}) {
# An empty config file doesn't get added to the cache
$self->debug("Not loading any data or cache - 'empty' specified") if $attribs{debug};
}
elsif ($attribs{cache} and $attribs{compile_subs} and $SUB_CACHE{$attribs{compile_subs}}->{$filename} and my $debug_unchanged = $self->_is_unchanged(@{$SUB_CACHE{$attribs{compile_subs}}->{$filename}}[FILES, FILES_MOD])) {
$self->debug("Loading '$filename' from compiled sub cache") if $attribs{debug};
@attribs{qw{data inherited file_order files compiled}} = @{$SUB_CACHE{$attribs{compile_subs}}->{$filename}};
$attribs{cache_hit} = 1;
}
elsif ($attribs{cache} and not $attribs{compile_subs} and $CACHE{$filename} and $debug_unchanged = $self->_is_unchanged(@{$CACHE{$filename}}[FILES, FILES_MOD])) {
$self->debug("Loading '$filename' from regular cache") if $attribs{debug};
@attribs{qw{data inherited file_order files}} = @{$CACHE{$filename}};
$attribs{cache_hit} = 1;
}
else {
$self->debug("Not loading '$filename' from cache") if $attribs{debug};
if ($attribs{debug} > 1) { # If the debug level is > 1, display some debugging as to _why_ we aren't loading from cache
$self->debug("Reason: Caching disabled") if not $attribs{cache};
if ($attribs{compile_subs} and not $SUB_CACHE{$attribs{compile_subs}}->{$filename}) { $self->debug("Reason: Not in compiled sub cache") }
elsif (not $attribs{compile_subs} and not $CACHE{$filename}) { $self->debug("Reason: Not in regular cache") }
$self->debug("Reason: File (or inherited files) have changed") if ($attribs{compile_subs} ? $SUB_CACHE{$attribs{compile_subs}}->{$filename} : $CACHE{$filename}) and not $debug_unchanged;
}
$self->_load_data($filename) or return;
if (@{$attribs{file_order}}) { # Don't cache it if it is a new object
if ($attribs{compile_subs}) {
$self->debug("Adding '$filename' (compile package '$attribs{compile_subs}') to the compiled sub cache") if $attribs{debug};
$SUB_CACHE{$attribs{compile_subs}}->{$filename} = [@attribs{qw{data inherited file_order files compiled}}];
}
else {
$self->debug("Adding '$filename' to the regular cache") if $attribs{debug};
$CACHE{$filename} = [@attribs{qw{data inherited file_order files}}];
}
}
}
return $self;
}
$COMPILE{save} = __LINE__ . <<'END_OF_SUB';
sub save {
require GT::Dumper;
my $self = shift;
my $att = $ATTRIBS{$self};
my ($d, $i) = @$att{'data', 'inherited'};
my %data;
for (keys %$d) { # Strip out all inherited data
next if $i->{$_};
$data{$_} = $d->{$_};
}
my $filename = $att->{path};
local $!;
if ($att->{local}) {
$filename .= "/local";
if (!-d $filename) { # $filename is misleading - it's currently a path
# Attempt to create the "local" directory
mkdir($filename, 0777) or return $self->error(CANT_CREATE_DIR => FATAL => $filename => "$!");
CORE::chmod(0777, $filename);
}
}
my $tmpfile = $att->{tmpfile};
if (not defined $tmpfile) {
# Base whether or not we use the tempfile on whether or not we can
# write to the base directory of the file:
$tmpfile = -w $filename;
}
$filename .= "/$att->{file}";
$self->debug("Saving '$filename'") if $att->{debug};
my $localtime = scalar localtime;
my $header = $att->{header};
if ($header) {
$header =~ s/\[localtime\]/$localtime/g;
$header .= "\n" unless $header =~ /\n$/;
}
my $write_filename = $tmpfile ? "$filename.tmp.$$." . time . "." . int rand 10000 : $filename;
my $printed = 0;
my $windows = $^O eq 'MSWin32';
local *FILE;
open FILE, "> $write_filename" or return $self->error(CANT_WRITE => FATAL => $write_filename => "$!");
# Print header, if any:
if ($header) {
$printed += length $header;
$printed += $header =~ y/\n// if $windows; # Windows does \n => \r\n translation on FH output
unless (print FILE $header) {
my $err = "$!";
close FILE;
unlink $write_filename if $tmpfile;
return $self->error(CANT_PRINT => FATAL => $write_filename => $err);
}
}
# Print actual data:
my $dump = GT::Dumper->dump(
var => '',
data => \%data,
sort => 1,
$att->{sort_order} ? (order => $att->{sort_order}) : (),
tab => $att->{tab}
);
$printed += length $dump;
$printed += $dump =~ y/\n// if $windows;
unless (print FILE $dump) {
my $err = "$!";
close FILE;
unlink $write_filename if $tmpfile;
return $self->error(CANT_PRINT => FATAL => $write_filename => $err);
}
# Print the vim info line at the bottom:
my $viminfo = "\n# vim:syn=perl:ts=4:noet\n";
$printed += length $viminfo;
$printed += $viminfo =~ y/\n// if $windows;
unless (print FILE $viminfo) {
my $err = "$!";
close FILE;
unlink $write_filename if $tmpfile;
return $self->error(CANT_PRINT => FATAL => $write_filename => $err);
}
close FILE;
# Check that the file is the right size, because print() returns true if a
# _partial_ print succeeded. Ideally we would check -s on the filehandle after
# each print, but of course that doesn't work on Windows.
unless ((my $actual = -s $write_filename) == $printed) {
unlink $write_filename if $tmpfile;
return $self->error(WRITE_MISMATCH => FATAL => $write_filename => $printed => $actual);
}
if ($tmpfile) {
$self->debug("'$write_filename' saved; renaming to '$filename'") if $att->{debug} > 1;
unless (rename $write_filename, $filename) {
my $err = "$!";
unlink $write_filename;
return $self->error(CANT_RENAME => FATAL => $write_filename => $filename => $err);
}
}
if (defined $att->{chmod}) {
my $mode = (stat $filename)[2] & 07777;
CORE::chmod($att->{chmod}, $filename) unless $att->{chmod} == $mode;
}
$self->debug("'$filename' saved, $printed bytes.") if $att->{debug};
return 1;
}
END_OF_SUB
# Returns true if the current object was loaded from cache, false otherwise.
sub cache_hit { $ATTRIBS{$_[0]}->{cache_hit} }
sub _is_unchanged {
my ($self, $old_order, $old_mod) = @_;
my $att = $ATTRIBS{$self};
$self->debug("Checking for any changes in the file (or inherited files)") if $att->{debug};
my @old_order = @$old_order; # Copy the old file_order and file modification
my %old_mod = %$old_mod; # times. _load_tree will replace them.
my $just_do_ok = not (PERSIST or $att->{inheritance} or $att->{local} or $att->{create_ok});
$self->_load_tree($just_do_ok);
if (@{$att->{file_order}} != @old_order) {
$self->debug("The old order and the new differ: Old: (@old_order) New: (@{$att->{file_order}})") if $att->{debug};
return;
}
for (0 .. $#old_order) {
if ($old_order[$_] ne $att->{file_order}->[$_]) {
$self->debug("The old order and the new differ: Old: (@old_order) New: (@{$att->{file_order}})") if $att->{debug};
return; # The inherited files are not the same as before
}
elsif ($att->{debug} >= 2) {
$self->debug("Old order and new order do not differ. Old: (@old_order) New: (@{$att->{file_order}})");
}
if ($old_mod{$old_order[$_]}->[0] != $att->{files}->{$old_order[$_]}->[0]) {
$self->debug("The file size of $old_order[$_] has changed: Old: $old_mod{$old_order[$_]}->[0], New: $att->{files}->{$old_order[$_]}->[0]") if $att->{debug};
return; # The inherited files have changed in size
}
elsif ($old_mod{$old_order[$_]}->[1] != $att->{files}->{$old_order[$_]}->[1]) {
$self->debug("The modification time of $old_order[$_] has changed: Old: $old_mod{$old_order[$_]}->[1], New: $att->{files}->{$old_order[$_]}->[1]") if $att->{debug};
return; # The inherited files have a changed mtime
}
elsif ($att->{debug} >= 2) {
$self->debug("The file size and modification time of $old_order[$_] has not changed");
}
}
$self->debug("No changes have been made") if $att->{debug};
1; # Here's the prize. Nothing is changed.
}
sub _load_data {
my $self = shift;
my $att = $ATTRIBS{$self};
my $just_do_ok = not (PERSIST or $att->{inheritance} or $att->{local} or $att->{create_ok});
$self->_load_tree($just_do_ok) or return;
if ($just_do_ok and not @{$att->{file_order}}) {
push @{$att->{file_order}}, $att->{filename_given};
}
for my $file (@{$att->{file_order}}) {
local ($@, $!, $^W);
$self->debug("do()ing '$file'") if $att->{debug} >= 2;
my $data = do $file;
if (!$data and $@) {
return $self->error(CANT_LOAD => FATAL => $file => "$@");
}
elsif (!$data and $!) {
return $self->error(CANT_COMPILE => FATAL => $file => "$!");
}
elsif (ref $data ne 'HASH') {
return $self->error(NOT_HASH => FATAL => $file);
}
if ($just_do_ok or $file eq ($att->{local} ? "$att->{path}/local/$att->{file}" : $att->{filename})) {
$att->{data} = $data;
}
else {
for (keys %$data) {
next if exists $att->{data}->{$_};
$att->{data}->{$_} = $data->{$_};
$att->{inherited}->{$_} = 1;
}
}
}
1; # Returning true means loading was successful.
}
sub _load_tree {
my $self = shift;
my $just_do_ok = shift;
my $att = $ATTRIBS{$self};
my $root = $att->{path};
my $file = $att->{file};
if ($att->{inheritance}) {
$att->{file_order} = [GT::Template::Inheritance->get_all_paths(file => $att->{file}, path => $att->{path})];
unless (@{$att->{file_order}} or $att->{create_ok} or $just_do_ok) {
return $self->error('CANT_FIND' => 'FATAL', $att->{file}, $att->{path});
# No files found!
}
for (@{$att->{file_order}}) {
$att->{files}->{$_} = [(stat($_))[7, 9]];
}
}
else {
if (-e "$root/$file") {
$att->{file_order} = ["$root/$file"];
$att->{files}->{"$root/$file"} = [(stat("$root/$file"))[7, 9]];
}
elsif ($att->{create_ok} or $just_do_ok) {
$att->{file_order} = [];
}
else {
return $self->error(CANT_FIND => FATAL => $att->{file}, $att->{path});
}
}
1;
}
$COMPILE{inheritance} = __LINE__ . <<'END_OF_SUB';
sub inheritance {
my $self = shift;
my $att = $ATTRIBS{$self};
$att->{inheritance};
}
END_OF_SUB
$COMPILE{tmpfile} = __LINE__ . <<'END_OF_SUB';
sub tmpfile {
my $self = shift;
my $att = $ATTRIBS{$self};
if (@_) {
my $ret = $att->{tmpfile};
my $tmpfile = shift;
$tmpfile = defined($tmpfile) ? $tmpfile ? 1 : 0 : undef;
$att->{tmpfile} = $tmpfile;
return $ret;
}
$att->{tmpfile};
}
END_OF_SUB
# Must be specified in load() - this only retrieves the value
$COMPILE{create_ok} = __LINE__ . <<'END_OF_SUB';
sub create_ok {
my $self = shift;
my $att = $ATTRIBS{$self};
$att->{create_ok};
}
END_OF_SUB
$COMPILE{chmod} = __LINE__ . <<'END_OF_SUB';
sub chmod {
my $self = shift;
my $att = $ATTRIBS{$self};
if (@_) {
my $ret = $att->{chmod};
$att->{chmod} = shift;
return $ret;
}
$att->{chmod};
}
END_OF_SUB
# Must be specified in load()
$COMPILE{cache} = __LINE__ . <<'END_OF_SUB';
sub cache {
my $self = shift;
my $att = $ATTRIBS{$self};
$att->{cache};
}
END_OF_SUB
$COMPILE{strict} = __LINE__ . <<'END_OF_SUB';
sub strict {
my $self = shift;
my $att = $ATTRIBS{$self};
if (@_) {
my $ret = $att->{strict} ? 1 : 0;
$att->{strict} = shift() ? 1 : 0;
return $ret;
}
$att->{strict};
}
END_OF_SUB
$COMPILE{debug_level} = __LINE__ . <<'END_OF_SUB';
sub debug_level {
my $self = shift;
my $att = $ATTRIBS{$self};
if (@_) {
my $ret = $att->{debug};
$att->{debug} = shift;
return $ret;
}
$att->{debug};
}
END_OF_SUB
$COMPILE{debug} = __LINE__ . <<'END_OF_SUB';
sub debug {
# -------------------------------------------------------
# Displays a debugging message.
#
my ($self, $msg, $min) = @_;
my $att = $ATTRIBS{$self};
$min ||= 1;
return if $att->{debug} < $min;
my $pkg = ref $self || $self;
# Add line numbers if no \n on the debug message
if (substr($msg, -1) ne "\n") {
my ($file, $line) = (caller)[1,2];
$msg .= " at $file line $line.\n";
}
# Remove windows linefeeds (breaks unix terminals).
$msg =~ s/\r//g unless $^O eq 'MSWin32';
print STDERR "$pkg ($$): $msg";
}
END_OF_SUB
$COMPILE{header} = __LINE__ . <<'END_OF_SUB';
sub header {
my $self = shift;
my $att = $ATTRIBS{$self};
if (@_) {
my $ret = $att->{header};
$att->{header} = shift || '';
return $ret;
}
$att->{header};
}
END_OF_SUB
# Be sure to delete the object from %ATTRIBS.
sub DESTROY {
delete $ATTRIBS{$_[0]} if keys %ATTRIBS and exists $ATTRIBS{$_[0]};
}
$COMPILE{error} = __LINE__ . <<'END_OF_SUB';
sub error {
my ($self, $code, $type, @args) = @_;
$type = $type && uc $type eq 'WARN' ? 'WARN' : 'FATAL';
my $pkg = ref $self || $self;
$error = _format_err($pkg, $code, @args);
if ($type eq 'FATAL') {
die $error if GT::Base::in_eval();
if ($SIG{__DIE__}) {
die $error;
}
else {
print STDERR $error;
die "\n";
}
}
elsif ($ATTRIBS{$self}->{debug}) { # A warning, and debugging is on
if ($SIG{__WARN__}) {
CORE::warn $error;
}
else {
print STDERR $error;
}
}
return;
}
END_OF_SUB
sub _format_err {
# -------------------------------------------------------
# Formats an error message for output.
#
my ($pkg, $code, @args) = @_;
my $msg = sprintf($ERRORS->{$code} || $code, @args);
my ($file, $line) = GT::Base::get_file_line($pkg);
return "$pkg ($$): $msg at $file line $line.\n";
}
# Tied hash handling
sub TIEHASH { bless $_[1], $_[0] }
sub STORE {
$_[0]->{data}->{$_[1]} = $_[2];
delete $_[0]->{inherited}->{$_[1]};
delete $_[0]->{compiled}->{$_[1]};
}
sub FETCH {
my $att = shift; # $_[0] is NOT $self - it is the attribute hashref
my $key = shift;
if ($att->{strict} and not exists $att->{data}->{$key}) {
return GT::Config->error(BAD_KEY => FATAL => $key, $att->{filename});
}
elsif ($att->{compile_subs} and not ref $att->{data}->{$key} and substr($att->{data}->{$key}, 0, 5) eq 'sub {') {
return $att->{compiled}->{$key} if exists $att->{compiled}->{$key};
my ($code, $err);
# Perl breaks when the eval below contains a 'use' statement. Somehow, Perl
# thinks it's deeper (in terms of { ... }) than it really is, and so ends up
# either exiting the subroutine prematurely, or, if we try to work around that
# by using another subroutine, or returning early, by jumping back one
# subroutine too many with its return value. So, to get around the whole
# problem, we wrap the code in double-evals if it contains 'use' or 'BEGIN'.
# It won't _break_ anything, but unfortunately it does slow compiled_subs
# globals a little bit slower.
if ($att->{data}->{$key} =~ /\b(use|no)\s+[\w:]/ or $att->{data}->{$key} =~ /\bBEGIN\b/) {
$code = eval "package $att->{compile_subs}; my \$ret = eval qq|\Q$att->{data}->{$key}\E|; die qq|\$\@\n| if \$\@; \$ret;";
}
else {
$code = eval "package $att->{compile_subs}; $att->{data}->{$key};";
}
$err = "$@";
# Perl prior to 5.6.1 breaks on this:
# perl -e 'my $c = eval "package SomePkg; sub bar { use NotThere }"; eval "package OtherPkg; print 1"; die "$@" if $@'
# From that, we die with: syntax error at (eval 2) line 1, near "package OtherPkg"
# This little hack fixes it, but don't ask me why:
eval "package Hack;" if $] < 5.006001;
if (ref $code ne 'CODE') {
GT::Config->error(CANT_COMPILE_CODE => WARN => $key, $att->{filename}, $err);
my $error = "Unable to compile '$key': $err";
$code = sub { $error };
}
return $att->{compiled}->{$key} = $code;
}
$att->{data}->{$key};
}
sub FIRSTKEY { keys %{$_[0]->{data}}; each %{$_[0]->{data}} }
sub NEXTKEY { each %{$_[0]->{data}} }
sub EXISTS { exists $_[0]->{data}->{$_[1]} }
sub DELETE {
my $val;
$val = $_[0]->FETCH($_[1]) if defined wantarray;
delete $_[0]->{inherited}->{$_[1]};
delete $_[0]->{data}->{$_[1]};
delete $_[0]->{compiled}->{$_[1]};
$val;
}
sub CLEAR { %{$_[0]->{data}} = %{$_[0]->{inherited}} = %{$_[0]->{compiled}} = () }
1;
__END__
=head1 NAME
GT::Config - Dumped-hash configuration handler
=head1 SYNOPSIS
use GT::Config;
my $Config = GT::Config->load($config_file);
...
print $Config->{variable};
...
$Config->{othervar} = "something";
...
$Config->save;
=head1 DESCRIPTION
GT::Config provides a simple way to handle loading config files. It can load
and save any config file consisting of a dumped hash. You can then use the
object as if it were the actual hash reference from the config file. It
supports template set inheritance (see L<GT::Template>) and mtime-based
caching.
=head1 METHODS
=head2 load
There is no C<new()> method. To get a new config object you do:
$Config = GT::Config->load("/path/to/config/file", { options });
The first argument is the full path to the file to open to read the
configuration. The file does not necessarily have to exist - see the options
below.
The second argument is a hash reference of options, and is optional. The
possible options are:
=over 4
=item inheritance
If provided as something true, GT::Config will scan for .tplinfo files looking
for inherited template sets. This is typically used for loading globals.txt or
language.txt files from Gossamer Threads products' template sets.
Defaults to off.
=item local
If provided as something true, GT::Config will look for a "local" directory
containing the file. When using inheritance, a "local" directory will also be
looked for in each inherited configuration file. However, regardless of the
C<inheritance> option, "local" configuration files always inherit from their
non-local counterpart.
Additionally, this option causes GT::Config to save the file into a "local"
directory. Also note that the "local" file will _only_ contain keys that were
already in the local file, or were assigned to the config object after loading
the file.
Defaults to off.
=item cache
If provided, will look in the internal cache for a cached copy of the file. If
none is found, a new GT::Config object will be constructed as usual, then saved
in the cache.
Defaults to on. You must pass C<cache =E<gt> 0> to disable cached loading.
Note that new objects are always stored in the cache, allowing you to specify
C<cache =E<gt> 0> to force a reload of a cached file.
=item create_ok
If set, you'll still get back a GT::Config hash even if the file doesn't exist.
You can then save() the object to create a new config file. If this option is
not set, a fatal error will occur when attempting to load a file that does not
exist.
Defaults to off. Pass in C<create_ok =E<gt> 1> if the config file doesn't
necessarily have to exist (i.e. when creating a new config file).
=item empty
The C<empty> option is used to create a new, blank config file - it can be
thought of as a forced version of the C<create_ok> option. It won't read
B<any> files during loading (and as such completely ignores the C<inheritance>
and C<cache> options). This is mainly intended to be used when a complete
replacement of a file is desired, regardless of what is currently on disk.
=item chmod
The C<chmod> option is used to specify the mode of the saved file. It must be
passed in octal form, such as 0644 (but B<not> in string form, such as
C<"0644">). The default is 0666, to allow writing by any users. Though not
terribly secure, this is the sort of environment most CGI scripts require.
Setting a chmod value of undef instructs GT::Config to not perform a chmod.
=item strict
If set, a fatal error will occur when attempting to access a key of the config
file that does not exist. Note, however, that this only covers the first level
data structions - C<$CFG-E<gt>{foo}-E<gt>{bar}> will not fatal if C<foo> is a
hash ref, but C<bar> is not set in that hash reference. C<$CFG-E<gt>{foo}>
(and C<$CFG-E<gt>{foo}-E<gt>{bar}>) will fatal if the key C<foo> does not exist
in the config data.
=item debug
If provided, debugging information will be printed. This will also cause a
warning to occur if L<"fatal"> is disabled and load fails.
Defaults to disabled. Should not be used in production code, except when
debugging.
=item tmpfile
Instructs GT::Config to attempt to use a temporary file when saving. If used,
the contents will be written to a temporary file, then, if successfully
written, the temporary file will be moved to overwrite the real file. This
solves a couple of problems. Firstly, a full disk will never result in a
partial file as if the entire file is not written to the temporary file, it
will not overwrite the file already stored on disk. Secondly, it avoids a
potential problem with multiple processes attempting to write to the file at
the same time.
The following values are accepted:
0 - Do not use a temporary file
undef - Use a temporary file if the base directory is writable
1 - Always use a temporary file
The default is C<undef>, which will attempt to use a temporary file is
possible, but won't fail if the script has permission to modify existing files,
but not to create new ones.
=item header
If provided, when saving a file this header will be written above the data.
Keep in mind that the file must be Perl-compilable, so be careful if you are
doing anything more than comments.
Note that the header may contain the string C<[localtime]>, which will be
replaced with the return value of C<scalar localtime()> when saving, which is
generally a value such as: C<Sun Jan 25 15:12:26 2004>.
=item tab
If provided, this will set what to use for tabs when calling save(). Defaults
to an actual tab, since that cuts down the file size over using multiple
spaces, while leaving the file readable.
=item compile_subs
If provided, any data starting with C<sub {> will be compiled into a
subroutine. This compilation does not happen until the variable is accessed,
at which point a fatal error will occur if the code could not be compiled. The
code referenced will be cached (if using caching), but will be saved as the
original string (starting with C<sub {>) when L<saving|"save">.
B<NOTE:> The argument to compile_subs must be a valid perl package; the code
reference will be compiled in that package. For example,
C<compile_subs =E<gt> 'GForum::Post'> will compile the code ref in the
GForum::Post package. You need to do this to provide access to globals
variables such as $DB, $IN, etc.
=item sort_order
If provided, the option will be passed through as the 'order' option of
GT::Dumper for hash key ordering. See L<GT::Dumper>. GT::Config always sorts
hash keys - this can be used when the default alphanumeric sort is not
sufficient.
=back
=head2 save
To save a config file, simply call C<$object-E<gt>save()>. If the object uses
inheritance, only those keys that were not inherited (or were modified from the
inherited ones) will be saved.
$Config->save();
B<NOTE>: B<ALWAYS SAVE AFTER MAKING ANY CHANGES!!!>. If you do not save after
making changes, the data retrieved from the cache may not be the same as the
data stored in the configuration file on disk. After making ANY changes make
absolutely sure that you either undo the change or save the configuration file.
=head2 cache_hit
Returns whether or not the current object was loaded from cache (1) or loaded
from disk (undef).
=head2 inheritance
Returns the inheritance status (1 or 0) of the object.
=head2 create_ok
Returns the status (1 or 0) of the "create_ok" flag.
=head2 tmpfile
With no arguments, returns whether or not the object will attempt to use a
temporary file when saving. Possible values are:
0 - Do not use a temporary file
undef - Use a temporary file if the base directory is writable
1 - Always use a temporary file
You can pass in a single argument of one of the above values to set whether or
not the object will use a temporary file when saving.
=head2 cache
This method returns whether or not the object is cached. This cannot be
enabled/disabled after loading a config file; you must specify it as an
argument to C<load()> instead.
=head2 debug_level
This method returns the current debug level.
You may provide one argument which sets a new debug level.
0 means no debugging, 1 means basic debugging, 2 means heavy debugging.
If setting a new debug level, the old debug level is returned.
=head2 header
This method returns or sets the header that will be printed when saving.
With no arguments, returns the header.
You may provide one argument which sets a new header. Keep in mind that the
file must be Perl-compilable, so take care if doing anything other than
comments.
If providing a new header, the old header is returned.
Note that the header may contain the value C<[localtime]>, which will be
replaced with the return value of C<scalar localtime()> when saving.
=head2 sort_order
This method returns or sets a code reference to be passed through as the
'order' option of GT::Dumper for hash key ordering. See L<GT::Dumper>.
GT::Config always sorts hash keys - this can be used when the default
alphanumeric sort is not sufficient.
=head1 SEE ALSO
L<GT::Template::Inheritance>
=head1 MAINTAINER
Jason Rhinelander
=head1 COPYRIGHT
Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved.
http://www.gossamer-threads.com/
=head1 VERSION
$Id: Config.pm,v 1.45 2005/03/21 05:49:39 jagerman Exp $
=cut