Only warn if input and output files are equal.
[ssic.git] / src / ssic.pl
index 30f1f255371f92a821cd2ee09feb76dc3608d922..750c77be6b9021a72284ed8cec2d844390c52c5d 100644 (file)
@@ -3,7 +3,7 @@
 use strict;
 use warnings;
 
-use Getopt::Std;
+use Getopt::Long;
 use CGI::SSI;
 
 sub main
@@ -14,8 +14,17 @@ sub main
 
        $SIG{'__WARN__'} = \&warning;
 
-       if (not getopts('o:hV', \%opts)) {
+       Getopt::Long::Configure("no_ignore_case", "bundling", "gnu_compat",
+               "no_getopt_compat");
+       if (not GetOptions(\%opts,
+                       "o=s",
+                       "D=s%",
+                       "I=s",
+                       "h|help",
+                       "V|version",
+               )) {
                usage(*STDERR);
+               exit(4);
        }
 
        if (exists($opts{'h'})) {
@@ -34,12 +43,12 @@ sub main
                if ($#ARGV gt 0) {
                        error(4, "Cannot specify -o with multiple files\n");
                }
-               compile($ARGV[0], $opts{'o'});
+               compile($ARGV[0], $opts{'o'}, $opts{'D'}, $opts{'I'});
        } else {
                for $input (@ARGV) {
                        $output = $input;
                        $output =~ s/\.[^.]+$/.html/;
-                       compile($input, $output);
+                       compile($input, $output, $opts{'D'}, $opts{'I'});
                }
        }
 }
@@ -57,9 +66,11 @@ sub help
 
        usage($fh);
        print("Options:\n");
-       print("  -o <output>  Place the output into <output>\n");
-       print("  -h           Display this information\n");
-       print("  -V           Display compiler version information\n");
+       print("  -D <name>=<value>  Set the variable <name> to <value>\n");
+       print("  -I <directory>     Set the document root to <directory>\n");
+       print("  -o <output>        Place the output into <output>\n");
+       print("  -h, --help         Display this information\n");
+       print("  -V, --version      Display compiler version information\n");
 }
 
 sub version
@@ -77,37 +88,81 @@ sub version
 
 sub warning
 {
-       my ($fmt, $args) = @_;
+       my ($fmt, @args) = @_;
 
-       printf("ssic: Warning: " . $fmt, $args);
+       printf("ssic: Warning: " . $fmt, @args);
 }
 
 sub error
 {
-       my ($status, $fmt, $args) = @_;
+       my ($status, $fmt, @args) = @_;
 
-       printf("ssic: Error: " . $fmt, $args);
+       printf("ssic: Error: " . $fmt, @args);
        exit($status);
 }
 
 sub compile
 {
-       my ($input, $output) = @_;
+       my ($input, $output, $vars, $root) = @_;
        my $input_fh;
+       my $input_abs;
        my $output_fh;
        my $ssi;
+       my $var_name;
+       my $var_value;
 
-       if ($input eq $output) {
-               error(4, "Input and output files are equal\n");
+       if ($input eq $output and $input ne "-") {
+               warning("Input and output files are equal\n");
        }
 
-       open($input_fh, "<", $input);
-       open($output_fh, ">", $output);
+       if ($input eq "-") {
+               $input_fh = *STDIN;
+               $input_abs = File::Spec->rel2abs(".");
+       } else {
+               if (not open($input_fh, "<", $input)) {
+                       error(4, "%s: %s\n", $input, $!);
+               }
+               $input_abs = File::Spec->rel2abs($input);
+       }
+       if ($output eq "-") {
+               $output_fh = *STDOUT;
+       } else {
+               if (not open($output_fh, ">", $output . "~")) {
+                       error(4, "%s: %s\n", $output . "~", $!);
+               }
+       }
+
+       # CGI::SSI uses SCRIPT_FILENAME to determine the value of LAST_MODIFIED.
+       %ENV = (
+               "DOCUMENT_NAME" => $input,
+               "DOCUMENT_URI" => $input,
+               "DOCUMENT_ROOT" => $root,
+               "SCRIPT_FILENAME" => $input_abs,
+       );
 
        $CGI::SSI::DEBUG = 0;
-       $ssi = CGI::SSI->new();
+       $ssi = CGI::SSI->new(
+               "DOCUMENT_NAME" => $input,
+               "DOCUMENT_URI" => $input,
+               "DOCUMENT_ROOT" => $root,
+       );
+
+       while (($var_name, $var_value) = each(%{$vars})) {
+               $ssi->set($var_name => $var_value);
+       }
 
        print($output_fh $ssi->process(<$input_fh>));
+
+       if ($input ne "-") {
+               close($input_fh);
+       }
+       if ($output ne "-") {
+               close($output_fh);
+       }
+
+       if (not rename($output . "~", $output)) {
+               error(4, "%s: %s\n", $output, $!);
+       }
 }
 
 main();