💾 Archived View for h2903872.stratoserver.net › GGI › GGI.pm captured on 2022-04-29 at 11:25:55.

View Raw

More Information

⬅️ Previous capture (2022-04-28)

-=-=-=-=-=-=-

package GGI;
#require Exporter;
#@ISA=qw(Exporter);
#@EXPORT=qw();

use strict;
use warnings;
no warnings qw(uninitialized); # avoid warnings for undefined/NULL values!
use utf8::all;
#use URL::Encode;
use Encode;
use URI::Escape;

=encoding utf8

=head1 GGI

GGI.pm - Gemini Gateway Interface: Support perl-programs on Gemini servers.

=head1 SUMMARY

GGI handles the accumulation of multiple parameters in the path-component of the URL through
automatic redirects and offers a simple method to access those parameters from the perl program.

GGI supports the generation of all Gemini line types: responses, headers, text, quotes, links
and pre-formatted, as well as some convenient methods to generate menus and to print tables with
and without ASCII or UniCode frames.

More about the Gemini protocol: L<https://gemini.circumlunar.space/>

=head1 VERSION

GGI.pm Version from 2021-03-26

=head1 AUTHORS

GGI by Frank Jüdes, L<http://www.linux4specialists.com>

This pod text by Frank Jüdes, Perl by Larry Wall and the C<perl5-porters>.

=head1 COPYRIGHT

The GGI module is Copyright (c) 2021 Frank Jüdes, West Virginia, U.S.A.
All rights reserved.

You may distribute under the terms of either the GNU General Public
License or the Artistic License, as specified in the Perl 5.10.0 README file.

=head1 SUPPORT and WARRANTY

The GGI is free Open Source software. IT COMES WITHOUT WARRANTY OF ANY KIND.

=head1 EXAMPLES

=head2 Make the output of the I<ps> utility available through a Gemini server

  #!/bin/perl
  use warnings;
  use strict;
  use utf8::all;
  use GGI;

  my $ggi = new GGI;    # Create a new GGI object and print SUCCESS header
  $ggi->H1('ps test');  # Print a header
  my @output=`ps -fA`;  # Run ps and catch all the output
  chomp @output;        # Remove EOLs
  $ggi->Pre(@output);   # Print the output as preformatted text

=head2 Collect Parameters into the PATH_INFO variable

  #!/bin/perl
  use warnings;
  use strict;
  use utf8::all;
  use GGI;

  my $ggi = GGI->new(1);         # New GGI object without writing any header
  if (not defined $ggi->Param) { # No parameter present: Get the amount of parameters wanted
    $ggi->ResponseHeader('INPUT',qq(GGI-Test, ParamTest: How many parameters do you want to test?));
  } # END if

  my $Current = $ggi->Param+1;   # How many parameters are there
  my $Wanted  = $ggi->Param(0);  # How many parameters are wanted?
  if ($Current <= $Wanted) {     # Get the next parameter
    $ggi->ResponseHeader('INPUT',qq(GGI-Test, Paramtest: Please enter a value for Parameter $Current of $Wanted));
  } # END if

  # Done with parameters! - Main program
  $ggi->H1("Hello World!");

  if (defined $ggi->Param) {
    $ggi->H2("The parameters");
    $ggi->List(@{$ggi->Param('*')});
  } else {
    $ggi->H1("No Parameters.");
  } # END if

  $ggi->Link(qq(gemini://h2903872.stratoserver.net/ Go back to the server's home.));

=head2 Implemented methods

The following paragraphs explain the implemented methods of this package in great detail. 

=cut


#----------------------------------------------------------------------------------------------------
=head3 new([NoPrintHeader]

Create a new GGI object:

=over 2

=item * A new object is created, some internal variables initialized.

=item * If there is a parameter at the end of the URL, the value is added as a path-component and
a re-direct to the new URL is sent to the client.

=item * If there is no parameter attached to the URL, all parameters are being copied
from the PATH_INFO into the internal PARAMS array.

=item * If called without any parameters an OK response-header will be printed automatically and the function
will return to the main program. If any TRUE value as parameter is passed, no header will be printed at
all and it is the duty of the calling program to print a Gemini response-header if other than SUCCESS
should be printed.

=back

=cut

sub new {
  my ($Class,$NoPrintHeader) = @_;

  my $self = {};
  bless $self,$Class;

  $self->{HeaderHasBeenSent} = 0;
  $self->{ResponseStatus} = {'INPUT'                       => 10,
                             'SENSITIVE_INPUT'             => 11,
                             'SUCCESS'                     => 20,
                             'REDIRECT_TEMPORARY'          => 30,
                             'REDIRECT_PERMANENT'          => 31,
                             'FAILURE_TEMPORARY'           => 40,
                             'SERVER_UNAVAILABLE'          => 41,
                             'CGI_ERROR'                   => 42,
                             'PROXY_ERROR'                 => 43,
                             'SLOW_DOWN'                   => 44,
                             'FAILURE_PERMANENT'           => 50,
                             'NOT_FOUND'                   => 51,
                             'PROXY_REQUEST_REFUSED'       => 52,
                             'BAD_REQUEST'                 => 53,
                             'CLIENT_CERTIFICATE_REQUIRED' => 60,
                             'CERTIFICATE_NOT_AUTHORIZED'  => 61,
                             'CERTIFICATE_NOT_VALID'       => 62
                             };
                             
  # ********** This is the gemini-server dependend HACKING SECTION **********
  
  # First: Some servers store the PATH_INFO with a leading "/", some other's don't. 
  # So let's remove the leading "/" if it exists:
  if ((length($ENV{PATH_INFO}) > 0) and (substr($ENV{PATH_INFO},0,1) eq '/')) {
    $ENV{PATH_INFO} = substr($ENV{PATH_INFO},1);
  } # END if

  # Second: Some servers store the QUERY_STRING with a leading "?", some other's don't. 
  # So let's remove the leading "?" if it exists:
  if ((length($ENV{QUERY_STRING}) > 0) and (substr($ENV{QUERY_STRING},0,1) eq '?')) {
    $ENV{QUERY_STRING} = substr($ENV{QUERY_STRING},1);
  } # END if

  # Third: Missing CGI variables in the Molly-Brown server!
  # The jetforce-server sets the variables SCRIPT_NAME and GEMINI_URL correctly, the Molly-Brown server does not.
  # This makes the awful hack below necessary‼ 
  # WARNING: This will only work if the CGI-program is located in /cgi-bin ‼
  # Unless the CGI-variable SCRIPT_NAME is provided by the Molly-Brown-server, this is the only way to 
  # automate the parameter redirection and provide something like the GEMINI_URL
  if ((not defined $ENV{SCRIPT_NAME}) and (defined $ENV{SCRIPT_PATH})) {
    my ($ScriptName) = $ENV{SCRIPT_PATH} =~ m!(/cgi-bin/.*)!;
    $ENV{SCRIPT_NAME} = $ScriptName;
    $ENV{GEMINI_URL} = "gemini://$ENV{SERVER_NAME}${ScriptName}";
    $ENV{GEMINI_URL} .= "/" . $ENV{PATH_INFO} if (length($ENV{PATH_INFO}) > 0);
    $ENV{GEMINI_URL} .= "?" . $ENV{QUERY_STRING} if (length($ENV{QUERY_STRING}) > 0);
  } # END if

  # Handle parameters
  if (length($ENV{QUERY_STRING}) > 0) { 
    # There is a parameter in the URL, so add parameter-value to the path, print redirect header and end the program
    my $URL = $ENV{GEMINI_URL};
    $URL =~ s/\?/\//;
    $self->ResponseHeader('REDIRECT_TEMPORARY',$URL);
    exit 0; # END program and let the clinet execute re-direct.
  } else {  # analyze PATH_INFO and extract parameters
    if (length($ENV{PATH_INFO}) > 0) {
      my $PathInfo = $ENV{PATH_INFO};
#      $PathInfo =~ s/%([0-9,A-F]{2})/'chr(0x'. $1. ')'/eeg;   # URL-Decoding
#      $self->{PARAMS} = [split('/',$PathInfo)];               # split the parameters
#      my $DecodedPathInfo = URL::Encode::url_decode_utf8($PathInfo);
#      $self->{PARAMS} = [split('/',$DecodedPathInfo)];         # split the parameters
      my $DecodedPathInfo = Encode::decode('utf8',uri_unescape($PathInfo));
      $self->{PARAMS} = [split('/',$DecodedPathInfo)];         # split the parameters
    } # END if
  } # END if

  $self->ResponseHeader('SUCCESS') unless (defined $NoPrintHeader); # print an OK header by default
  return $self;
} # END new


#----------------------------------------------------------------------------------------------------
=head3 BaseURL

Returns the current request URL without the parameters
=cut
sub BaseURL {
  return substr($ENV{GEMINI_URL},0,-length($ENV{PATH_INFO}));
} # END BaseURL

#----------------------------------------------------------------------------------------------------
=head3 MyURL

Returns the current request URL - Basically a shortcut for $ENV{GEMINI_URL}.
=cut

sub MyURL {
  return $ENV{GEMINI_URL};
} # END MyURL

#----------------------------------------------------------------------------------------------------
=head3 ServerName

Returns the current request's servername - Basically a shortcut for $ENV{SERVER_NAME}.
=cut

sub ServerName {
  return $ENV{SERVER_NAME};
} # END ServerName

#----------------------------------------------------------------------------------------------------
=head3 ScriptName

Returns the current request's script-name - Basically a shortcut for $ENV{SCRIPT_NAME}.
=cut

sub ScriptName {
  return $ENV{SCRIPT_NAME};
} # END ScriptName

#----------------------------------------------------------------------------------------------------
=head3 WashParameter

Removes potentialy dangerous stuff from the parameters, currently for Linux

=over 2

=item * Everything attached with a ";" at the end, including the ";" is removed

=item * Everything between backticks "`", including the back-ticks themselves, is removed

=item * Everything in '$( )' is removed, including the $-sign and the parentheses

=back

=cut

sub WashParameter {
  my $self = shift @_;
  
  return unless (defined $self->{PARAMS});
  
  map {
    $_ =~ s/;.*$//;       # no command attachment
    $_ =~ s/\`.*\`//;     # no command insertion
    $_ =~ s/\$\(.*?\)//;  # no bash exec
  } @{$self->{PARAMS}};     
} # END WashParameter

#----------------------------------------------------------------------------------------------------
=head3 param([Name])

Access to the GGI-parameters:

=over 2

=item * If there are no GGI parameters present, undef is returned.

=item * Called without an index-value in array-mode, a reference to the parameter-array is returned.

=item * Called without an index-value in scalar-mode the highes index into the parameter array is returned.

=item * Called with the index-value of "#", the numer of parameters is returned.

=item * Called with the index-value of "*", a reference to the parameter-array is returned.

=item * Called with an index-value the value of the indexed parameter is returned if it exists.
If no parameter with the given index exists, I<undef> is returned.

=back

=cut

sub Param {
  my ($self,$Index) = @_;

  return undef unless (defined $self->{PARAMS});
  if (not defined $Index) {
    return $self->{PARAMS} if (wantarray);
    return $#{$self->{PARAMS}};
  } # END if
  return scalar @{$self->{PARAMS}} if ($Index eq '#');
  return $self->{PARAMS} if ($Index eq '*');
  return $self->{PARAMS}->[$Index];
} # END param


#----------------------------------------------------------------------------------------------------
# pr(@Lines) (internal)
#
# Print an array of lines with CR/LF at the end
#----------------------------------------------------------------------------------------------------
sub pr {
  my ($self,@Lines) = @_;
  map { print "$_\r\n"; } @Lines # causing double spacing in some clients
  #map { print "$_\n"; } @Lines
} # END pr

#----------------------------------------------------------------------------------------------------
=head3 print(@Lines)

Print an array of lines with CR/LF at the end
=cut

sub print {
  my ($self,@Lines) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  $self->pr(@Lines);
} # END print

#----------------------------------------------------------------------------------------------------
=head3 ResponseHeader(StatusCode[,Meta])

Print the response header with the meta-information, if present. The following response status-codes
are defined, according to the Gemini protocol documentation:

  - INPUT, SENSITIVE_INPUT
  - SUCCESS
  - REDIRECT_TEMPORARY, REDIRECT_PERMANENT
  - FAILURE_TEMPORARY, SERVER_UNAVAILABLE, CGI_ERROR, PROXY_ERROR, SLOW_DOWN
  - FAILURE_PERMANENT, NOT_FOUND, PROXY_REQUEST_REFUSED, BAD_REQUEST
  - CLIENT_CERTIFICATE_REQUIRED, CERTIFICATE_NOT_AUTHORIZED, CERTIFICATE_NOT_VALID
=cut

sub ResponseHeader {
  my ($self,$StatusCode,$Message) = @_;
  return if ($self->{HeaderHasBeenSent});
  $StatusCode = 'SUCCESS' unless (defined $StatusCode);
  if ($StatusCode eq 'SUCCESS') {
    $self->pr("20 text/gemini");
    $self->{HeaderHasBeenSent} = 1;
  } else {
    if (exists $self->{ResponseStatus}->{$StatusCode}) {
      $self->pr($self->{ResponseStatus}->{$StatusCode} . " $Message");
    } else {
      $self->pr($self->{ResponseStatus}->{CGI_ERROR} . " Unknown Response code '$StatusCode'.");
    } # END if
    exit(0);
  } # END if
} # END ResponseHeader

#----------------------------------------------------------------------------------------------------
=head3 Input(HeaderText)

Print an Input response header with prompt. This is basically a shortcut for I<ResponseHeader('INPUT',[Prompttext]);>
=cut

sub Input {
  my ($self,$Prompt) = @_;
  $self->ResponseHeader('INPUT',$Prompt);
} # END Input

#----------------------------------------------------------------------------------------------------
=head3 H[1-3](HeaderText)

Print headerline 1-3
=cut

sub H1 {
  my ($self,$Line) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  $self->pr("# $Line");
} # END H1

sub H2 {
  my ($self,$Line) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  $self->pr("## $Line");
} # END H2

sub H3 {
  my ($self,$Line) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  $self->pr("### $Line");
} # END H3

#----------------------------------------------------------------------------------------------------
=head3 Pre(@Lines)

Print an array of lines as a preformatted block
=cut

sub Pre {
  my ($self,@Lines) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  $self->pr('```');
  map { print $_,"\n"; } @Lines;
  $self->pr('```');
} # END Pre

#----------------------------------------------------------------------------------------------------


#----------------------------------------------------------------------------------------------------
=head3 LabeledPre(Label,@Lines)

Print an array of lines as a labeled preformatted block
=cut

sub LabeledPre {
  my ($self,$Label,@Lines) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  $self->pr('``` ' . $Label);
  map { print $_,"\n"; } @Lines;
  $self->pr('```');
} # END Pre

#----------------------------------------------------------------------------------------------------

=head3 List(@Lines)

Print an array of lines as an unsorted list.
=cut

sub List {
  my ($self,@Lines) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  map { $self->pr('* ' . $_) } @Lines;
} # END List

#----------------------------------------------------------------------------------------------------
=head3 Link(@Lines)

Print an array of lines as links
=cut

sub Link {
  my ($self,@Lines) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  map { $self->pr('=> ' . $_) } @Lines;
} # END Link

#----------------------------------------------------------------------------------------------------
=head3 Quote(@Lines)

Print an array of lines as quotes
=cut

sub Quote {
  my ($self,@Lines) = @_;
  $self->ResponseHeader('SUCCESS') unless ($self->{HeaderHasBeenSent}); # send header if necessary
  map { $self->pr('> ' . $_) } @Lines;
} # END Link

#----------------------------------------------------------------------------------------------------
=head3 PrintMenu(Menu,BaseURL,Title,Description)

Print the value of I<Title> as an H1-title if defined. Then print
the content of the I<Description> below if defined, followed by the link-menu itself.

I<Menu> is a simple perl-hash with the prompts as keys and the values as content.
These values will be appended to the I<BaseURL> as link-targets and the keys
will become the link d