#
#--
# Ronin Web - A Ruby library for Ronin that provides support for web
# scraping and spidering functionality.
#
# Copyright (c) 2006-2009 Hal Brodigan (postmodern.mod3 at gmail.com)
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
#++
#
require 'ronin/network/http'
require 'uri/http'
require 'nokogiri'
require 'mechanize'
require 'open-uri'
module Ronin
module Web
#
# Returns a Nokogiri::HTML::Document object for the specified _body_
# of html.
#
def Web.html(body,&block)
doc = Nokogiri::HTML(body)
block.call(doc) if block
return doc
end
#
# Creates a new Nokogiri::HTML::Builder with the given _block_.
#
# Web.build_html do
# body {
# div(:style => 'display:none;') {
# object(:classid => 'blabla')
# }
# }
# end
#
def Web.build_html(&block)
Nokogiri::HTML::Builder.new(&block)
end
#
# Returns a Nokogiri::XML::Document object for the specified _body_
# of xml and the given _block_.
#
def Web.xml(body,&block)
doc = Nokogiri::XML(body)
block.call(doc) if block
return doc
end
#
# Creates a new Nokogiri::XML::Builder with the given _block_.
#
# Web.build_xml do
# post(:id => 2) {
# title { text("some example) }
# body { text("this is one contrived example.") }
# }
# end
#
def Web.build_xml(&block)
Nokogiri::XML::Builder.new(&block)
end
#
# Returns the default Ronin Web proxy port.
#
def Web.default_proxy_port
Network::HTTP.default_proxy_port
end
#
# Sets the default Ronin Web proxy port to the specified _port_.
#
def Web.default_proxy_port=(port)
Network::HTTP.default_proxy_port = port
end
#
# Returns the +Hash+ of the Ronin Web proxy information.
#
def Web.proxy
Network::HTTP.proxy
end
#
# Resets the Web proxy settings.
#
def Web.disable_proxy
Network::HTTP.disable_proxy
end
#
# Creates a HTTP URI based from the given _proxy_info_ hash. The
# _proxy_info_ hash defaults to Web.proxy, if not given.
#
def Web.proxy_url(proxy_info=Web.proxy)
if Web.proxy[:host]
userinfo = nil
if (Web.proxy[:user] || Web.proxy[:password])
userinfo = "#{Web.proxy[:user]}:#{Web.proxy[:password]}"
end
return URI::HTTP.build(
:host => Web.proxy[:host],
:port => Web.proxy[:port],
:userinfo => userinfo,
:path => '/'
)
end
end
#
# Returns the supported Web User-Agent Aliases.
#
def Web.user_agent_aliases
WWW::Mechanize::AGENT_ALIASES
end
#
# Returns the Ronin Web User-Agent
#
def Web.user_agent
Network::HTTP.user_agent
end
#
# Sets the Ronin Web User-Agent to the specified _new_agent_.
#
def Web.user_agent=(new_agent)
Network::HTTP.user_agent = new_agent
end
#
# Sets the Ronin Web User-Agent to the specified user agent alias
# _name_.
#
def Web.user_agent_alias=(name)
Network::HTTP.user_agent = Web.user_agent_aliases[name.to_s]
end
#
# Opens the _url_ with the given _options_. The contents of the _url_
# will be returned.
#
# _options_ may contain the following keys:
# :user_agent_alias:: The User-Agent Alias to use.
# :user_agent:: The User-Agent string to use.
# :proxy:: A +Hash+ of the proxy information to use.
# :user:: The HTTP Basic Authentication user name.
# :password:: The HTTP Basic Authentication password.
# :content_length_proc:: A callback which will be passed the
# content-length of the HTTP response.
# :progress_proc:: A callback which will be passed the size
# of each fragment, once received from the
# server.
#
# Web.open('http://www.hackety.org/')
#
# Web.open('http://tenderlovemaking.com/',
# :user_agent_alias => 'Linux Mozilla')
#
# Web.open('http://www.wired.com/', :user_agent => 'the future')
#
def Web.open(url,options={})
user_agent_alias = options.delete(:user_agent_alias)
proxy = (options.delete(:proxy) || Web.proxy)
user = options.delete(:user)
password = options.delete(:password)
content_length_proc = options.delete(:content_length_proc)
progress_proc = options.delete(:progress_proc)
headers = Network::HTTP.headers(options)
if user_agent_alias
headers['User-Agent'] = Web.user_agent_aliases[user_agent_alias]
end
if proxy[:host]
headers[:proxy] = Web.proxy_url(proxy)
end
if user
headers[:http_basic_authentication] = [user, password]
end
if content_length_proc
headers[:content_length_proc] = content_length_proc
end
if progress_proc
headers[:progress_proc] = progress_proc
end
return Kernel.open(url,headers)
end
#
# Creates a new Mechanize agent with the given _options_.
#
# _options_ may contain the following keys:
# :user_agent_alias:: The User-Agent Alias to use.
# :user_agent:: The User-Agent string to use.
# :proxy:: A +Hash+ of the proxy information to use.
#
# Web.agent
# Web.agent(:user_agent_alias => 'Linux Mozilla')
# Web.agent(:user_agent => 'wooden pants')
#
def Web.agent(options={},&block)
agent = WWW::Mechanize.new
if options[:user_agent_alias]
agent.user_agent_alias = options[:user_agent_alias]
elsif options[:user_agent]
agent.user_agent = options[:user_agent]
elsif Web.user_agent
agent.user_agent = Web.user_agent
end
proxy = (options[:proxy] || Web.proxy)
if proxy[:host]
agent.set_proxy(proxy[:host],proxy[:port],proxy[:user],proxy[:password])
end
block.call(agent) if block
return agent
end
#
# Gets the specified _url_ with the given _options_. If a _block_ is
# given, it will be passed the retrieved page.
#
# _options_ may contain the following keys:
# :user_agent_alias:: The User-Agent Alias to use.
# :user_agent:: The User-Agent string to use.
# :proxy:: A +Hash+ of the proxy information to use.
#
# Web.get('http://www.0x000000.com') # => WWW::Mechanize::Page
#
# Web.get('http://www.rubyinside.com') do |page|
# page.search('div.post/h2/a').each do |title|
# puts title.inner_text
# end
# end
#
def Web.get(url,options={},&block)
page = Web.agent(options).get(url)
block.call(page) if block
return page
end
#
# Gets the specified _url_ with the given _options_, returning the body
# of the requested page. If a _block_ is given, it will be passed the
# body of the retrieved page.
#
# _options_ may contain the following keys:
# :user_agent_alias:: The User-Agent Alias to use.
# :user_agent:: The User-Agent string to use.
# :proxy:: A +Hash+ of the proxy information to use.
#
# Web.get_body('http://www.rubyinside.com') # => String
#
# Web.get_body('http://www.rubyinside.com') do |body|
# puts body
# end
#
def Web.get_body(url,options={},&block)
body = Web.get(url,options).body
block.call(body) if block
return body
end
#
# Posts the specified _url_ with the given _options_. If a _block_ is
# given, it will be passed the posted page.
#
# _options_ may contain the following keys:
# :query:: The query parameters to post to the specified _url_.
# :user_agent_alias:: The User-Agent Alias to use.
# :user_agent:: The User-Agent string to use.
# :proxy:: A +Hash+ of the proxy information to use.
#
# Web.post('http://www.rubyinside.com') # => WWW::Mechanize::Page
#
def Web.post(url,options={},&block)
query = {}
query.merge!(options[:query]) if options[:query]
page = Web.agent(options).post(url,query)
block.call(page) if block
return page
end
#
# Poststhe specified _url_ with the given _options_, returning the body
# of the posted page. If a _block_ is given, it will be passed the
# body of the posted page.
#
# _options_ may contain the following keys:
# :user_agent_alias:: The User-Agent Alias to use.
# :user_agent:: The User-Agent string to use.
# :proxy:: A +Hash+ of the proxy information to use.
#
# Web.post_body('http://www.rubyinside.com') # => String
#
# Web.post_body('http://www.rubyinside.com') do |body|
# puts body
# end
#
def Web.post_body(url,options={},&block)
body = Web.post(url,options).body
block.call(body) if block
return body
end
end
end