package Net::OpenSSH::ShellQuoter::csh;

use strict;
use warnings;
use Carp;

# Fixme: copied from POSIX

sub new { __PACKAGE__ }

my $noquote_class = q(.\\w/\\-@,:);
my $glob_class    = q(*?\\[\\],{}:!^~);
my $escape_inside_single_quotes_class  = q(\!\n);

sub _single_quote {
    my $arg = shift;
    $arg =~ s/([$escape_inside_single_quotes_class])/\\$1/go;
    "'$arg'"
}

sub quote {
    shift;
    my $quoted = join '',
        map { ( m|\A'\z|                  ? "\\'"             :
                m|\A'|                    ? "\"$_\""          :
                m|\A[$noquote_class]*\z|o ? $_                :
                                            _single_quote($_) )
          } split /(')/o, $_[0];
    length $quoted ? $quoted : "''";
}


sub quote_glob {
    shift;
    my $arg = shift;
    my @parts;
    while ((pos $arg ||0) < length $arg) {
        if ($arg =~ m|\G('+)|gc) {
            push @parts, (length($1) > 1 ? "\"$1\"" : "\\'");
        }
        elsif ($arg =~ m|\G([$noquote_class$glob_class]+)|gco) {
            push @parts, $1;
        }
        elsif ($arg =~ m|\G(\\[$glob_class\\])|gco) {
            push @parts, $1;
        }
        elsif ($arg =~ m|\G([^$glob_class\\']+)|gco) {
            push @parts, _single_quote($1);
        }
        else {
            require Data::Dumper;
            $arg =~ m|\G(.+)|gc;
            die "Internal error: unquotable string:\n". Data::Dumper::Dumper($1) ."\n";
        }
    }
    my $quoted = join('', @parts);
    length $quoted ? $quoted : "''";
}

my %fragments = ( stdin_discard             => '</dev/null',
                  stdout_discard            => '>/dev/null',
                  stdout_and_stderr_discard => '>&/dev/null' );

sub shell_fragments {
    shift;
    my @f = grep defined, @fragments{@_};
    wantarray ? @f : join(' ', @f);
}

1;