#!/usr/bin/perl
#EDIT THE ABOVE LINE TO POINT TO THE LOCATION OF Perl ON YOUR SERVER

##### DO NOT EDIT ANYTHING BELOW THIS LINE #####
use strict;use utf8;use JSON;use MIME::Base64;my$iv='1.01';my$fname='SAMLtest.pl';my$cwd=$ENV{SCRIPT_FILENAME};if($cwd){$cwd=~s!(?:[^/\\]*)$!!}else{($_=$0)=~s![\\/][^\\/]+$!!;unshift@INC,$_;$cwd=$_};$cwd=~s/\\/\//g;use lib qw(.);my%in=parse();my$act=$in{act};my$nolog=$in{nolog};require CGI::Carp;CGI::Carp->import('fatalsToBrowser','set_message');set_message('For help, please refer to the <a href="https://mid.as/kb" target=_blank>MIDAS Knowledge Base</a>');my$midas;eval{$midas=decode_json(SSI_INCLUDE("$cwd/midas.dat"))};if($@){}my$certprefix=lc(substr($midas->{midasid},9,3).substr($midas->{midasid},-1,1));$certprefix='MIDAS' if !$certprefix;if(!$act){if($in{SAMLResponse}){samlresponse()}else{splash()}}elsif($act eq 'start'){samlsetup()}elsif($act eq 'gen_cert_key'){gen_cert_key()}elsif($act eq 'samlsetup'){samlsetup()}elsif($act eq 'samlsave'){samlsave()}elsif($act eq 'test'){samltest()}sub pagehead {print "Content-type: text/html; charset=UTF-8\n\n";print qq~<!doctype html><html><head> <title>SAML (SSO) Test v$iv</title> <LINK REL="icon" href=""> <style> :root{--theme-background-color:#FFF;--theme-border-color:#264666;--theme-box-color-dark:rgba(38,70,102,.5);--theme-button-color-1:rgba(137,176,215,.5);--theme-button-color-2:rgba(38,70,102,.7);--theme-carat-color:#3B99FC;--theme-color-1:#FFF;--theme-color-2:#264666;--theme-color-h2:#264666;--theme-gradient-color-1:#FFF;--theme-gradient-color-2:rgba(38,70,102,1); } ::-webkit-scrollbar{width:16px} ::-webkit-scrollbar-corner{background:rgba(38,70,102,.4);border-radius:20px} ::-webkit-scrollbar-thumb{background:rgba(38,70,102,.5);border-radius:20px;cursor:pointer} html{height:100%} body{margin:0;padding:10px;font-family:'Open Sans',Calibri,Tahoma,sans-serif;font-size:16px;text-align:center;cursor:default;background:var(--theme-background-color);background:radial-gradient(at bottom right,var(--theme-gradient-color-1) 70%,var(--theme-gradient-color-2) 100%);background-attachment:fixed} form{padding:0;margin:0} fieldset{border-radius:10px;border:1px solid #FFF;padding:0 5px 5px} input[type=text],input[type=email],input[type=password]{font-size:1em;border:none;box-shadow:5px 5px 10px 3px var(--theme-box-color-dark);border-radius:20px;padding:6px 10px;caret-color:var(--theme-carat-color);margin:0 5px 5px 5px;width:-webkit-fill-available} select{cursor:pointer;font-size:1em;display:inline-block;padding:0.6em 1.4em 0.5em 0.8em;box-sizing:border-box;margin:0 5px 2px 5px;border:none;box-shadow:5px 5px 10px 3px var(--theme-box-color-dark);border-radius:20px;appearance:none;-moz-appearance:none;-webkit-appearance:none;background-color:#fff;background-image:url(data:image/svg+xml;charset=UTF-8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%22292.4%22%20height%3D%22292.4%22%3E%3Cpath%20fill%3D%22%2389b0d7%22%20d%3D%22M287%2069.4a17.6%2017.6%200%200%200-13-5.4H18.4c-5%200-9.3%201.8-12.9%205.4A17.6%2017.6%200%200%200%200%2082.2c0%205%201.8%209.3%205.4%2012.9l128%20127.9c3.6%203.6%207.8%205.4%2012.8%205.4s9.2-1.8%2012.8-5.4L287%2095c3.5-3.5%205.4-7.8%205.4-12.8%200-5-1.9-9.2-5.5-12.8z%22%2F%3E%3C%2Fsvg%3E);background-repeat:no-repeat, repeat;background-position:right 0.5em top 50%, 0 0;background-size:0.65em auto, 100%} input[type=checkbox]{background-color:#FFF;cursor:pointer;vertical-align:middle;width:24px;appearance:none;-webkit-appearance:none;-moz-appearance:none;height:24px;box-shadow:5px 5px 10px 3px var(--theme-box-color-dark);margin-right:10px;margin-bottom:5px} input[type=checkbox]:checked:after{margin-left:7px;margin-top:1px;width:6px;height:14px;border:solid white;border-width:0 4px 4px 0;-webkit-transform:rotate(45deg);-moz-transform:rotate(45deg);-ms-transform:rotate(45deg);transform:rotate(45deg);content:'';display:inline-block} input[type=radio]{background-color:#FFF;cursor:pointer;vertical-align:middle;width:24px;appearance:none;-webkit-appearance:none;-moz-appearance:none;height:24px;border-radius:12px;box-shadow:5px 5px 10px 3px var(--theme-box-color-dark);margin-right:10px;margin-bottom:5px} input[type=radio]:checked:after{margin-left:5px;margin-top:5px;width:0;height:0;border:solid white;border-width:7px;border-radius:12px;content:'';display:inline-block} input[type=checkbox]:checked,input[type=radio]:checked{background-color:var(--theme-color-2)} h2{color:var(--theme-color-h2);font-weight:400;font-size:1.5em;margin:5px} button{cursor:pointer;color:var(--theme-color-1);font-size:1.2em;margin:5px;padding:10px 15px;background:linear-gradient(to bottom,var(--theme-button-color-1) 0%,var(--theme-button-color-2) 100%);border:1px outset #1C2E40;border-radius:1.5em;box-shadow:5px 5px 10px rgba(0,0,0,.4)} button:hover{border:1px inset #1C2E40;box-shadow:0 0 0 rgba(0,0,0,0)} button[disabled]{opacity:.5}button[disabled]:hover{cursor:not-allowed;border:1px outset #1C2E40;box-shadow:2px 2px 6px rgba(0,0,0,0.6)}footer{color:var(--theme-color2)}.arrow_left{background-image:URL('');display:inline-block;width:24px;height:24px;background-size:cover;vertical-align:middle}.arrow_right{background-image:URL('');display:inline-block;width:24px;height:24px;background-size:cover;vertical-align:middle}.box{background:rgba(38,70,102,.5);border:1px solid #264666;border-radius:10px;box-shadow: 1px 1px 8px 1px rgb(134,161,213) inset;padding: 5px 10px 5px 10px;margin-left:auto;margin-right:auto;color:#FFF}.box_red{background:rgba(255,0,0,0.3)}.l{text-align:left}.c{text-align:center}.r{text-align:right}.b{font-weight:bold}.dn{display:none}.cp{cursor:pointer}label{cursor:pointer}a{color:#eabe0e;text-decoration:none}a:hover{text-decoration:underline}fieldset{margin-top:20px}.width100{width:100%;width:-webkit-fill-available;width:-moz-available;width:fill-available}legend{font-size:1.5em;color:var(--theme-color-2);position:relative;top:-10px}hr{color:var(--theme-border-color);background-color:var(--theme-border-color);border:0;margin:5px;height:2px} h3{color:gold;margin:5px;font-weight:400} .logo{width:180px;height:110px;margin-left:auto;margin-right:auto;background-image:URL('');background-size:180px 110px;background-repeat:no-repeat} .logo div{text-align:right;padding-top:6px;padding-right:5px;font-style:italic;font-size:10px} .color1{color:#FFFFFF} .color2{color:#264666} .color3{color:#264666} .color4{color:#FF0000} .color5{color:#66FF00} .color6{color:gold} .color7{color:#000000}.cross{height:22px;width:22px;background-image:URL('');display:inline-block;vertical-align:middle}.tick{height:20px;width:22px;background-image:URL('');display:inline-block;vertical-align:middle}.warn{height:22px;width:22px;background-image:URL('');display:inline-block;vertical-align:middle}.p5{padding:5px}</style></head><body><div class="logo color3"><div>SAML Test v$iv</div></div>~}sub pagefoot{print qq~<hr><footer>For assistance, please visit <a href="https://mid.as/saml-integration" target="_blank">mid.as/saml-integration</a></footer><hr></body></html>~}sub splash{my$nonetsaml2;eval{require Net::SAML2};if($@){$nonetsaml2=1}pagehead();print qq~<form name=f method="post" action="$fname"><input type=hidden name="act" value="start"><input type=hidden name="nolog" value="$nolog"><h2>IS YOUR SERVER READY FOR SEAMLESS SSO INTEGRATION?</h2><div class="box color1" style="width:100%;max-width:600px;margin-left:auto;margin-right:auto"><div style="width:100%;height:300px;overflow:auto;font-family:Arial;font-size:1em;text-align:justify;padding-right:5px">This tool will allow you to test your server setup to determine whether you're able to integrate MIDAS with your SAML service, to allow users to be authenticated and seamlessly logged into your MIDAS system upon each access.<br><br>THIS TOOL IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING, BUT NOT LIMITED TO WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT WILL THE AUTHOR BE LIABLE TO YOU FOR INDIRECT, SPECIAL, OR CONSEQUENTIAL DAMAGES, ARISING OUT OF ANY USE THEREOF OR BREACH OF ANY WARRANTY, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.<br><br>Click "Continue" to begin.</div>~;if($nonetsaml2){print qq~<div class=color4>The Perl module Net::SAML2 was not detected on this server<br>Please <a href="https://mid.as/kb/00129/how-to-install-perl-modules" target=_blank>install this module</a> then reload this page to continue</div>~}else{print q~<button>Continue <span class=arrow_right></span></button>~}print q~</div></form>~;pagefoot()}sub samlsetup{pagehead();my$idp_meta=SSI_INCLUDE("$cwd/SAML/$certprefix-idp-meta.dat")||SSI_INCLUDE("$cwd/SAML/MIDAS-idp-meta.dat");my$idp_cert=SSI_INCLUDE("$cwd/SAML/$certprefix-idp-cert.dat")||SSI_INCLUDE("$cwd/SAML/MIDAS-idp-cert.dat");my$sp_key=SSI_INCLUDE("$cwd/SAML/$certprefix-sp-key.dat")||SSI_INCLUDE("$cwd/SAML/MIDAS-sp-key.dat");my$sp_key_ph=($sp_key)?'[Not Shown]':q~Paste Service Provider's Private Key Here...~;my$sp_cert=SSI_INCLUDE("$cwd/SAML/$certprefix-sp-cert.dat")||SSI_INCLUDE("$cwd/SAML/MIDAS-sp-cert.dat");my$acsurl=($midas->{paths}{aliasurl})?"https://$midas->{paths}{aliasurl}$fname":"https://$midas->{paths}{weburl}$fname";$acsurl='https://'.$ENV{SERVER_NAME}.$ENV{REQUEST_URI} if !$acsurl;print qq~<script>document.addEventListener('DOMContentLoaded', ()=>{const genCertKeyButton=document.getElementById('gen_cert_key');const saveButton=document.getElementById('saml_save');genCertKeyButton.addEventListener('click', (event)=>{event.preventDefault();genCertKeyButton.innerHTML='Generating...';fetch('$fname', {method: 'POST',headers: {'Content-Type': 'application/x-www-form-urlencoded',},body: 'act=gen_cert_key',}).then(response=>response.text()).then(data=>{const r=data.split('|');document.getElementById('saml_sp_key').textContent=r[0];document.getElementById('saml_sp_cert').textContent=r[1];genCertKeyButton.innerHTML='Generate'}).catch(error=>{console.error('Error:', error);genCertKeyButton.innerHTML='Error Generating'})});saveButton.addEventListener('click', (event)=>{event.preventDefault();saveButton.innerHTML='Saving...';formData=new FormData(document.getElementById('saml_info'));formData.append('act','samlsave');const params=new URLSearchParams();for (const [key, value] of formData.entries()){params.append(key, value)}fetch('$fname', {method: 'POST',headers: {'Content-Type': 'application/x-www-form-urlencoded'},body: params.toString()}).then(response=>response.text()).then(data=>{saveButton.innerHTML='Saved';if(confirm("Settings Saved - Would you like to perform a SAML test now?")){self.location.href="$fname?act=test"}}).catch(error=>{console.error('Error:', error);saveButton.innerHTML='Error Saving'})})});</script><form id=saml_info><fieldset class=box><legend>Identity Provider (IdP) Settings</legend><div style="display:grid;grid-template-columns:100px 1fr;grid-gap:5px;align-items:center"><div class=r>Metadata:</div><div><textarea name=saml_idp_meta class="width100 font08" style="word-wrap:normal;height:80px" spellcheck=false placeholder="Paste URL or Raw Metadata Here...">$idp_meta</textarea></div><div class=r>Certificate:</div><div><textarea name=saml_idp_cert class="width100 font08" style="word-wrap:normal;height:80px" spellcheck=false placeholder="Paste Identify Provider's Certificate Here...">$idp_cert</textarea></div></div></fieldset><fieldset class=box><legend>Service Provider (SP) Settings</legend><div style="display:grid;grid-template-columns:100px 1fr 140px;grid-gap:5px;align-items:center"><div class=r style="grid-column:1 / 1;grid-row:1 / 1">Assertion Consumer Service (ACS) URL:</div><div class="l color6" style="grid-column:2 / 3;grid-row:1 / 1">$acsurl</div><div class=r style="grid-column:1 / 1;grid-row:2 / 2">Private Key:</div><div style="grid-column:2 / 2;grid-row:2 / 2"><textarea name=saml_sp_key class="width100 font08" style="word-wrap:normal;height:80px" spellcheck=false id=saml_sp_key placeholder="$sp_key_ph"></textarea></div><div class=r style="grid-column:1 / 1;grid-row:3 / 3">Certificate:</div><div style="grid-column:2 / 2;grid-row:3 / 3"><textarea name=saml_sp_cert class="width100 font08" style="word-wrap:normal;height:80px" spellcheck=false id=saml_sp_cert placeholder="Paste Service Provider's Certificate Here...">$sp_cert</textarea></div><div style="grid-column:3 / 3;grid-row:2 / 4">~;print q~<button id=gen_cert_key>Generate</button>~ if $midas->{paths}{openssl};print q~</div></div></fieldset><button id=saml_save>Save <span class=arrow_right></span></button></form>~;pagefoot()}sub gen_cert_key{unlink "$cwd/SAML/$certprefix-sp-key-temp.dat";unlink "$cwd/SAML/$certprefix-sp-cert-temp.dat";my$dataname;my$dbs;eval{$dbs=@{$midas->{database}{db}}};if($dbs){foreach my$k(0..@{$midas->{database}{db}}-1){if($midas->{database}{default} eq $midas->{database}{db}[$k]{r}){$dataname=$midas->{database}{db}[$k]{display};last}}}$dataname='MIDAS' if !$dataname;my$country=$midas->{languages}{default};$country=~s/\w\w-//;my$aliasdomain=$midas->{paths}{aliasurl};$aliasdomain=~s!/(.*)!!;my$subject='/C='.$country.'/O='.$dataname.'/CN='.$aliasdomain;my$sp_key;my$sp_cert;if(-e "$cwd/SAML/$certprefix-sp-key-temp.dat"){$sp_key=SSI_INCLUDE("$cwd/SAML/$certprefix-sp-key-temp.dat");$sp_cert=SSI_INCLUDE("$cwd/SAML/$certprefix-sp-cert-temp.dat");unlink "$cwd/SAML/$certprefix-sp-key-temp.dat";unlink "$cwd/SAML/$certprefix-sp-cert-temp.dat"}else{require LWP::UserAgent;LWP::UserAgent->import();eval{require IO::Socket::SSL};if($@){$ENV{PERL_LWP_SSL_VERIFY_HOSTNAME}=0}my$ua=LWP::UserAgent->new(env_proxy=>0,keep_alive=>0,timeout=>30,agent =>"SAML TEST v$iv");if($midas->{paths}{proxy}){$ENV{HTTPS_PROXY}=$midas->{paths}{proxy};$ua->proxy(['http','https'],$midas->{paths}{proxy})}my$r=$ua->post('https://u.mid.as/SAMLcertgen.pl',[dataname=>$dataname,prefix=>$certprefix,country=>$country,alias=>$aliasdomain]);my@response=($r->is_success)?($r->code,$r->status_line,$r->content):($r->code,$r->status_line);if($response[0]==200){if($response[2]=~m/^-----BEGIN PRIVATE KEY-----/){my@spdata=split(/\|/,$response[2]);$sp_key=$spdata[0];$sp_cert=$spdata[1]}else{$sp_key=$sp_cert='Unable to generate'}}}print "Content-type: text/plain; charset=UTF-8\n\n$sp_key|$sp_cert"}sub samlsave{if(!-e "$cwd/SAML"){mkdir "$cwd/SAML/"}if($in{saml_idp_meta}){$in{saml_idp_meta}=~s/\r//g;$in{saml_idp_meta}=~s/&(?!amp;)/&amp;/g;open(my$TMPF,'>:encoding(UTF-8)',"$cwd/SAML/$certprefix-idp-meta.dat");flock($TMPF,2);print $TMPF $in{saml_idp_meta};flock($TMPF,8);close($TMPF)}if($in{saml_idp_cert}){$in{saml_idp_cert}=~s/\r//g;open(my$TMPF,'>:encoding(UTF-8)',"$cwd/SAML/$certprefix-idp-cert.dat");flock($TMPF,2);print $TMPF $in{saml_idp_cert};flock($TMPF,8);close($TMPF)}if($in{saml_sp_key}){$in{saml_sp_key}=~s/\r//g;open(my$TMPF,'>:encoding(UTF-8)',"$cwd/SAML/$certprefix-sp-key.dat");flock($TMPF,2);print $TMPF $in{saml_sp_key};flock($TMPF,8);close($TMPF)}if($in{saml_sp_cert}){$in{saml_sp_cert}=~s/\r//g;open(my$TMPF,'>:encoding(UTF-8)',"$cwd/SAML/$certprefix-sp-cert.dat");flock($TMPF,2);print $TMPF $in{saml_sp_cert};flock($TMPF,8);close($TMPF)}print "Content-type: text/plain; charset=UTF-8\n\n1";exit}sub samltest{eval{require Net::SAML2};if($@){ssoerr('Net::SAML2 not found')}my$issuer='midas';my$provider='midas';my$idp_meta="$cwd/SAML/$certprefix-idp-meta.dat";my$idp_cert="$cwd/SAML/$certprefix-idp-cert.dat";my$sp_key="$cwd/SAML/$certprefix-sp-key.dat";my$idp_metadata=SSI_INCLUDE($idp_meta);if(!$idp_meta){ssoerr('Identity Provider metadata not found')}my$xml_doc;if($idp_metadata=~m/</){eval{$xml_doc=XML::LibXML->load_xml(location=>$idp_meta)};if($@){ssoerr('Invalid Identity Provider metadata')}}else{require LWP::UserAgent;LWP::UserAgent->import();$ENV{PERL_LWP_SSL_VERIFY_HOSTNAME}=0;my$ua=LWP::UserAgent->new;$ua->timeout(10);my$response=$ua->get("$idp_metadata");$response=$response->content;eval{$xml_doc=XML::LibXML->load_xml(string=>$response)};if($@){ssoerr('Invalid Identity Provider metadata')}}$idp_metadata=$xml_doc->documentElement;my $idp=Net::SAML2::IdP->new_from_xml(xml=>$idp_metadata,cacert=>$idp_cert,);my $authnreq=Net::SAML2::Protocol::AuthnRequest->new(issuer=> $issuer,destination=>$idp->sso_url('urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Redirect'),provider_name=>$provider,);my$saml_request_id=$authnreq->id;if(!$saml_request_id){ssoerr('SAML Request ID not found')}print "Set-Cookie:SAMLID=$saml_request_id; HttpOnly; SameSite=Strict; Secure\n";my $redirect=Net::SAML2::Binding::Redirect->new(key=>$sp_key,cert=>$idp->cert('signing'),param=>'SAMLRequest',url=>$idp->sso_url('urn:oasis:names:tc:SAML:2.0:bindings:HTTP-Redirect'),);my$url=$redirect->sign($authnreq->as_xml);if(!$url){ssoerr('Redirect URL not found')}print "Location:$url\n\n"}sub samlresponse{my$saml_response=$in{SAMLResponse};my$op;$op.=qq~<fieldset calss=box><legend>SAML OUTPUT</legend><textarea class=width100 style="height:200px">~;my$tryagain;if($in{action} eq 'verify'){eval{require Net::SAML2};if($@){pagehead();print qq~<fieldset calss=box><legend>SAML OUTPUT</legend><textarea class=width100 style="height:200px">Net::SAML2 Not Found</textarea></fieldset>~;pagefoot();exit}require Net::SAML2::Binding::POST;require Net::SAML2::Protocol::Assertion;$op.=qq~Got SAML response and action is "verify"\n~;my$issuer='midas';my$idp_cert="$cwd/SAML/$certprefix-idp-cert.dat";my%cookies=getCookies();my$saml_request_id=$cookies{'SAMLID'};$op.=qq~Got SAML request id: $saml_request_id\n~;my$post=Net::SAML2::Binding::POST->new(cacert=>$idp_cert );my$ret=$post->handle_response($saml_response);my$assertion=Net::SAML2::Protocol::Assertion->new_from_xml(xml=>decode_base64($saml_response));my$valid=$assertion->valid($issuer,$saml_request_id);if($valid){my$dn=determine_name($assertion);my$em=determine_email($assertion);if(($dn)&&($em)){$op.=qq~========================================\nSUCCESS! I received the user/email: $dn <$em>\n========================================\n~}else{require Data::Dumper;my$raw=Data::Dumper::Dumper($assertion);$op.=qq~VALID RESPONSE - but unable to determine name/email\n$raw~}}}else{$op.=qq~Got SAML response but action is NOT "verify":\n$saml_response\n~;$tryagain=1}$op.=q~</textarea></fieldset>~;if($tryagain){$op.=qq~<form method=post action="$fname" style="display:none"><input type=hidden name=action value='verify'><textarea name=SAMLResponse>$saml_response</textarea><script>document.forms[0].submit()</script>~}else{print "Set-Cookie:SAMLID=; HttpOnly; SameSite=Strict; Secure\n"}pagehead();print $op;pagefoot()}sub determine_email{my$assertion=shift;my$email=$assertion->attributes->{email}[0]||$assertion->attributes->{Email}[0]||$assertion->attributes->{EmailAddress}[0]||$assertion->attributes->{mail}[0]||$assertion->attributes->{'http://schemas.xmlsoap.org/ws/2005/05/identity/claims/emailaddress'}[0];return $email}sub determine_name{my$assertion=shift;my$name;my$firstname;my$surname;$firstname=$assertion->attributes->{firstname}[0]||$assertion->attributes->{firstName}[0]||$assertion->attributes->{FirstName}[0]||$assertion->attributes->{givenname}[0]||$assertion->attributes->{GivenName}[0]||$assertion->attributes->{givenName}[0];$surname=$assertion->attributes->{lastname}[0]||$assertion->attributes->{lastName}[0]||$assertion->attributes->{LastName}[0]||$assertion->attributes->{surname}[0]||$assertion->attributes->{Surname}[0]||$assertion->attributes->{sn}[0];$name=($surname)?$firstname.' '.$surname:$firstname;$name=$assertion->attributes->{'http://schemas.xmlsoap.org/ws/2005/05/identity/claims/name'}[0] if !$name;return $name}sub ssoerr{my$err=shift;pagehead();print qq~<h1>err</h1>~;pagefoot();exit}sub getCookies{my%cooks;foreach(split(/; /,$ENV{HTTP_COOKIE})){my($key,$val)=split(/=/,$_,2);$cooks{$key}=$val if defined $key}return %cooks}sub SSI_INCLUDE{local$/;open(my$F,'<',shift);my$c=<$F>;close($F);return $c}sub parse{my(@pairs,%in);my($buffer,$pair,$name,$value);if($ENV{REQUEST_METHOD}eq'GET'){@pairs=split(/&/,$ENV{QUERY_STRING})}elsif($ENV{REQUEST_METHOD}eq'POST'){read(STDIN,$buffer,$ENV{CONTENT_LENGTH});@pairs=split(/&/,$buffer)}PAIR:foreach my$pair(@pairs){($name,$value)=split(/=/,$pair);$name=~tr/+/ /;$name=~s/%([a-fA-F0-9][a-fA-F0-9])/pack("C",hex($1))/eg;$value=~tr/+/ /;$value=~s/%([a-fA-F0-9][a-fA-F0-9])/pack("C",hex($1))/eg;($value eq "---") and next PAIR;exists $in{$name}?($in{$name}.="~~$value"):($in{$name}=$value)}return %in}