###################################################################### Config::Patch 0.06 ###################################################################### NAME Config::Patch - Patch configuration files and unpatch them later SYNOPSIS use Config::Patch; my $patcher = Config::Patch->new( file => "/etc/syslog.conf", key => "mypatch", ); # Append a patch: $patcher->append(q{ # Log my stuff my.* /var/log/my }); # Appends the following to /etc/syslog.conf: *------------------------------------------- | ... | #(Config::Patch-mypatch-append) | # Log my stuff | my.* /var/log/my | #(Config::Patch-mypatch-append) *------------------------------------------- # Prepend a patch: $patcher->prepend(q{ # Log my stuff my.* /var/log/my }); # Prepends the following to /etc/syslog.conf: *------------------------------------------- | #(Config::Patch-mypatch-append) | # Log my stuff | my.* /var/log/my | #(Config::Patch-mypatch-append) | ... *------------------------------------------- # later on, to remove the patch: $patcher->remove(); DESCRIPTION "Config::Patch" helps changing configuration files, remembering the changes, and undoing them if necessary. Every change (patch) is marked by a *key*, which must be unique for the change, in order allow undoing it later on. To facilitate its usage, "Config::Patch" comes with a command line script that performs all functions: # Append a patch echo "my patch text" | config-patch -a -k key -f textfile # Patch a file by search-and-replace echo "none:" | config-patch -s 'all:.*' -k key -f config_file # Comment out sections matched by a regular expression: config-patch -c '(?ms-xi:^all:.*?\n\n)' -k key -f config_file # Remove a previously applied patch config-patch -r -k key -f textfile Note that 'patch' doesn't refer to a patch in the format used by the *patch* program, but to an arbitrary section of text inserted into a file. Patches are line-based, "Config::Patch" always adds/removes entire lines. The only requirement is that lines starting with a # character are comment lines. Other than that, "Config::Patch" is format-agnostic. If you need to pay attention to the syntax of the configuration file to be patched, create a subclass of "Config::Patch" and put the format specific logic there. You can only patch a file *once* with a given key. Note that a single patch might result in multiple patched sections within a file if you're using the "replace()" or "comment_out()" methods. To apply different patches to the same file, use different keys. They can be can rolled back separately. METHODS "$patcher = Config::Patch->new(file => $file, key => $key)" Creates a new patcher object. Optionally, exclusive updates are ensured by flocking if the "flock" parameter is set to 1: my $patcher = Config::Patch->new( file => $file, key => $key, flock => 1, ); "$patcher->append($textstring)" Appends a text string to the config file. "$patcher->prepend($textstring)" Adds a text string to the beginning of the file. "$patcher->remove()" Remove a previously applied patch. The patch key has either been provided with the constructor call previously or can be supplied as "key => $key". "$patcher->patched()" Checks if a patch with the given key was applied to the file already. The patch key has either been provided with the constructor call previously or can be supplied as "key => $key". "$patcher->replace($search, $replace)" Patches by searching for a given pattern $search (regexp) and replacing it by the text string $replace. Example: # Replace the 'all:' target in a Makefile and all # of its production rules by a dummy rule. $patcher->replace(qr(^all:.*?\n\n)sm, "all:\n\techo 'all is gone!'\n"); Note that the replace command will replace *the entire line* if it finds that a regular expression is matching a partial line. CAUTION: Make sure your $search patterns only cover the areas you'd like to replace. Multiple matches within one line are ignored, and so are matches that overlap with areas patched with different keys (*forbidden zones*). "$patcher->comment_out($search)" Patches by commenting out config lines matching the regular expression $search. Example: # Remove the 'all:' target and its production rules # from a makefile $patcher->comment_out(qr(^all:.*?\n\n)sm); Commenting out is just a special case of "replace()". Check its documentation for details. "$patcher->key($key)" Set a new patch key for applying subsequent patches. "($arrayref, $hashref) = $patcher->patches()" Examines the file and locates all patches. It returns two results: $arrayref, a reference to an array, mapping patch keys to the text of the patched sections: $arrayref = [ ['key1', 'patchtext1'], ['key2', 'patchtext2'], ['key2', 'patchtext3'] ]; Note that there can be several patched sections appearing under the same patch key (like the two non-consecutive sections under "key2" above). The second result is a reference $hashref to a hash, holding all patch keys as keys. Its values are the number of patch sections appearing under a given key. LIMITATIONS "Config::Patch" assumes that a hashmark (#) at the beginning of a line in the configuration file marks a comment. COPYRIGHT AND LICENSE Copyright 2005 by Mike Schilli. This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself. AUTHOR 2005, Mike Schilli <cpan@perlmeister.com>