## about:pastiepacker
Initial release of Pastie Packer, released via pastie itself :)

Files for pastiepacker uploaded by pastiepacker.
To unpack files see http://pastiepacker.rubyforge.org

## README.txt
= pastiepacker

* http://pastiepacker.rubyforge.org

== DESCRIPTION:

Prepare to pack or unpack piles of files with the pastiepacker.

== SYNOPSIS:

To pack a folder: +pastiepacker+

Packing options:

-f, --format=FORMAT Possess pasties with a particular persona
Supported formats:
c++, css, diff, html_rails, html,
javascript, php, plain_text, python,
ruby, ruby_on_rails, sql, shell-unix-generic
Default: ruby
-m, --message=MESSAGE Promotional passage for your pastie
-p, --private Posted pasties are private
Ignored for unpacking
-s, --stdout Prints packed pasties instead of posting
-H, --no-header Prevents placing pastiepacker promotion in pasties
That is, no 'about:' section is added to the top of pasties
-h, --help Show this help message.

To only pack a selection of files ending with *txt* you can pass a list of file names via STDIN:

find * | grep "txt$" | pastiepacker

It outputs the url of the prepared pastie, so you can pipe it to xargs:

pastiepacker | xargs open

To unpack a packed pastie: pastiepacker http://pastie.caboo.se/175183
- This unpacks the files into a subfolder 175138/
To unpack a private pastie: pastiepacker http://pastie.caboo.se/private/5hwfheniddqmyasmfcxaw
- This unpacks the files into a subfolder 5hwfheniddqmyasmfcxaw/


== REQUIREMENTS:

* ruby
* rubygems

To test if you have these, the following should work:

gem env

== INSTALL:

* sudo gem install pastiepacker

== LICENSE:

(The MIT License)

Copyright (c) 2008 Dr Nic Williams

Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to
the following conditions:

The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

## History.txt
== 1.0.0 2008-04-03

* 1 major enhancement:
* Initial release


## License.txt
Copyright (c) 2008 Dr Nic Williams

Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject to
the following conditions:

The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

## Manifest.txt
History.txt
License.txt
Manifest.txt
README.txt
Rakefile
bin/pastiepacker
config/hoe.rb
config/requirements.rb
lib/pastiepacker.rb
lib/pastiepacker/command.rb
lib/pastiepacker/conversion.rb
lib/pastiepacker/fetch.rb
lib/pastiepacker/header.rb
lib/pastiepacker/io.rb
lib/pastiepacker/options.rb
lib/pastiepacker/upload.rb
lib/pastiepacker/version.rb
lib/ruby-ext/hash.rb
script/console
script/destroy
script/generate
script/txt2html
setup.rb
tasks/deployment.rake
tasks/environment.rake
tasks/website.rake
test/fixtures/private_pastie.html
test/fixtures/sample_app/History.txt
test/fixtures/sample_app/README.txt
test/fixtures/sample_app/lib/myapp.rb
test/test_app.rb
test/test_conversion.rb
test/test_helper.rb
website/index.html
website/index.txt
website/javascripts/rounded_corners_lite.inc.js
website/stylesheets/screen.css
website/template.html.erb


## Rakefile
require 'config/requirements'
require 'config/hoe' # setup Hoe + all gem configuration

Dir['tasks/**/*.rake'].each { |rake| load rake }

## bin/pastiepacker
#!/usr/bin/env ruby
#
# Created on 2008-4-3.
# Copyright (c) 2008. All rights reserved.

require File.dirname(__FILE__) + "/../lib/pastiepacker"
output = PastiePacker.run(ARGV)
puts output if output

## config/hoe.rb
require 'pastiepacker/version'

AUTHOR = 'Dr Nic Williams' # can also be an array of Authors
EMAIL = "drnicwilliams@gmail.com"
DESCRIPTION = <<-EOS
Prepare to pack or unpack piles of files with the pastiepacker.

To pack a folder: #{File.basename($0)}
To pack some files ending with "txt": find * | grep "txt$" | #{File.basename($0)}
- It outputs the url of the prepared pastie, so you can pipe it to xargs:
- pastiepacker | xargs open

To unpack a packed pastie: #{File.basename($0)} http://pastie.caboo.se/175183
- This unpacks the files into a subfolder 175138/
EOS
GEM_NAME = 'pastiepacker' # what ppl will type to install your gem
RUBYFORGE_PROJECT = 'pastiepacker' # The unix name for your project
HOMEPATH = "http://#{RUBYFORGE_PROJECT}.rubyforge.org"
DOWNLOAD_PATH = "http://rubyforge.org/projects/#{RUBYFORGE_PROJECT}"

@config_file = "~/.rubyforge/user-config.yml"
@config = nil
RUBYFORGE_USERNAME = "unknown"
def rubyforge_username
unless @config
begin
@config = YAML.load(File.read(File.expand_path(@config_file)))
rescue
puts <<-EOS
ERROR: No rubyforge config file found: #{@config_file}
Run 'rubyforge setup' to prepare your env for access to Rubyforge
- See http://newgem.rubyforge.org/rubyforge.html for more details
EOS
exit
end
end
RUBYFORGE_USERNAME.replace @config["username"]
end


REV = nil
# UNCOMMENT IF REQUIRED:
# REV = `svn info`.each {|line| if line =~ /^Revision:/ then k,v = line.split(': '); break v.chomp; else next; end} rescue nil
VERS = PastiePacker::VERSION::STRING + (REV ? ".#{REV}" : "")
RDOC_OPTS = ['--quiet', '--title', 'pastiepacker documentation',
"--opname", "index.html",
"--line-numbers",
"--main", "README",
"--inline-source"]

class Hoe
def extra_deps
@extra_deps.reject! { |x| Array(x).first == 'hoe' }
@extra_deps
end
end

# Generate all the Rake tasks
# Run 'rake -T' to see list of generated tasks (from gem root directory)
$hoe = Hoe.new(GEM_NAME, VERS) do |p|
p.developer(AUTHOR, EMAIL)
p.description = DESCRIPTION
p.summary = DESCRIPTION
p.url = HOMEPATH
p.rubyforge_name = RUBYFORGE_PROJECT if RUBYFORGE_PROJECT
p.test_globs = ["test/**/test_*.rb"]
p.clean_globs |= ['**/.*.sw?', '*.gem', '.config', '**/.DS_Store'] #An array of file patterns to delete on clean.

# == Optional
p.changes = p.paragraphs_of("History.txt", 0..1).join("\n\n")
p.extra_deps = [
['shared-mime-info', '>=0.1'],
['hpricot', '>=0.6']
] # An array of rubygem dependencies [name, version], e.g. [ ['active_support', '>= 1.3.1'] ]

#p.spec_extras = {} # A hash of extra values to set in the gemspec.

end

CHANGES = $hoe.paragraphs_of('History.txt', 0..1).join("\\n\\n")
PATH = (RUBYFORGE_PROJECT == GEM_NAME) ? RUBYFORGE_PROJECT : "#{RUBYFORGE_PROJECT}/#{GEM_NAME}"
$hoe.remote_rdoc_dir = File.join(PATH.gsub(/^#{RUBYFORGE_PROJECT}\/?/,''), 'rdoc')
$hoe.rsync_args = '-av --delete --ignore-errors'

## config/requirements.rb
require 'fileutils'
include FileUtils

require 'rubygems'
%w[rake hoe newgem rubigen].each do |req_gem|
begin
require req_gem
rescue LoadError
puts "This Rakefile requires the '#{req_gem}' RubyGem."
puts "Installation: gem install #{req_gem} -y"
exit
end
end

$:.unshift(File.join(File.dirname(__FILE__), %w[.. lib]))


## lib/pastiepacker.rb
$:.unshift(File.dirname(__FILE__)) unless
$:.include?(File.dirname(__FILE__)) || $:.include?(File.expand_path(File.dirname(__FILE__)))

require "fileutils"
require 'net/http'
require 'timeout'
require 'cgi'
require 'optparse'
require "open-uri"

begin
require 'rubygems'
rescue LoadError
end

require 'shared-mime-info' # shared-mime-info rubygem
require 'hpricot'

require "ruby-ext/hash"
require "pastiepacker/io"
require "pastiepacker/options"
require "pastiepacker/conversion"
require "pastiepacker/upload"
require "pastiepacker/fetch"
require "pastiepacker/header"
require "pastiepacker/command"


## lib/pastiepacker/command.rb
class PastiePacker
attr_accessor :contents

def self.run(args = [])
packer = PastiePacker.new
packer.parse_options(args)
if !(files = packer.input_lines).empty?
packer.do_pack_files(files)
elsif args.empty?
packer.do_pack
else
packer.do_unpack(packer.args || [])
end
end

def do_pack
self.contents = self.path_to_string(FileUtils.pwd)
if contents && !contents.empty?
add_header(File.basename(FileUtils.pwd)) unless no_header?
do_paste
end
end

def do_pack_files(files)
self.contents = self.files_to_string(files)
if contents && !contents.empty?
add_header unless no_header?
do_paste
end
end

def do_unpack(pastie_urls)
pastie_urls.each do |raw_url|
url = process_url(raw_url)
unique_subfolder = url.match(/[\/=]([^\/=]*)$/)[1]
self.contents = fetch_pastie(url)
unpack(File.join(FileUtils.pwd, unique_subfolder))
end
nil # so nothing is printed as a result
end

def process_url(raw_url)
if raw_url =~ /private\/(.*)$/
doc = Hpricot(fetch_pastie(raw_url))
(doc/"a.utility").find { |a| a.inner_html == 'View' }.get_attribute("href")
else
raw_url.gsub(/\.txt$/,'')
end
end
end

## lib/pastiepacker/conversion.rb
class PastiePacker
def path_to_string(path)
output = ""
FileUtils.cd path do
output = files_to_string Dir['**/*'].sort
end
output
end

def files_to_string(files)
readme = files.find { |f| f =~ /^readme/i }
files = [readme] + (files - [readme]) if readme
output = files.inject([]) do |mem, file|
file = file.strip
if File.file?(file)
mem << "## #{file}"
if ascii? file
mem << File.open(file, 'r').read
else
end
mem << ""
end
mem
end.join("\n")
output
end

def unpack(target_path)
FileUtils.rm_rf "#{target_path}/*" # destroys previously unpacked pastie with same id
FileUtils.mkdir_p target_path
FileUtils.cd target_path do
files_contents = contents.split(/^## /)[1..-1] # ignore first ""
files_contents.each do |file_contents|
file_name, contents = file_contents.match(/([^\n]+)\n(.*)/m)[1,2]
if file_name =~ /about:/
# ignoring header
else
contents = contents.gsub(/\n\Z/,'')
FileUtils.mkdir_p File.dirname(file_name)
File.open(file_name, "w") do |f|
f << contents
end
end
end
end
target_path
end

protected
def ascii?(file)
MIME.check(file) == 'text/plain'
end
end

## lib/pastiepacker/fetch.rb
class PastiePacker
def fetch_pastie(url)
txt_url = "#{url}.txt"
# REMOVE: Net::HTTP.get(URI.parse(txt_url))
open(txt_url)
end
end


## lib/pastiepacker/header.rb
class PastiePacker
def add_header(title=nil)
self.contents = header(title) + self.contents
end

protected
def header(title=nil)
message = extra_message ? "#{extra_message}\n\n" : ""
if title
<<-EOS
## about:#{title}
#{message}Files for #{title} uploaded by pastiepacker.
To unpack files see http://pastiepacker.rubyforge.org

EOS
else
<<-EOS
## about:
#{message}Files uploaded by pastiepacker.
To unpack files see http://pastiepacker.rubyforge.org

EOS
end
end
end

## lib/pastiepacker/io.rb
class PastiePacker
def input_stream
$stdin
end

def output_stream
$stdout
end

def input_lines
if !input_stream.tty?
input_stream.readlines
else
[]
end
end
end

## lib/pastiepacker/options.rb
class PastiePacker
attr_accessor :format, :extra_message
attr_accessor :args

def private?; @private; end
def to_stdout?; @to_stdout; end
def no_header?; @no_header; end

def parse_options(args)
@private = false
@format = "ruby"
@extra_message = nil
@to_stdout = false
@no_header = false

OptionParser.new do |opts|
opts.banner = <<BANNER
Prepare to pack or unpack piles of files with the pastiepacker.

To pack a folder: #{File.basename($0)}
To pack some files ending with "txt": find * | grep "txt$" | #{File.basename($0)}
- It outputs the url of the prepared pastie, so you can pipe it to xargs:
- pastiepacker | xargs open

To unpack a packed pastie: #{File.basename($0)} http://pastie.caboo.se/175183
- This unpacks the files into a subfolder 175138/

Options are:
BANNER
opts.separator ""
opts.on("-f", "--format=FORMAT", String,
"Possess pasties with a particular persona",
"Supported formats:",
AVAILABLE_PARSERS.join(', '),
"Ignored for unpacking",
"Default: ruby") { |x| @format = x }
opts.on("-m", "--message=MESSAGE", String,
"Promotional passage for your pastie",
"Default: standard 'about' message") { |x| @extra_message = x }
opts.on("-p", "--private",
"Posted pasties are private",
"Ignored for unpacking",
"Default: false") { |x| @private = x }
opts.on("-s", "--stdout",
"Prints packed pasties instead of posting",
"Default: false") { |x| @to_stdout = x }
opts.on("-H", "--no-header",
"Prevents placing pastiepacker promotion in pasties",
"That is, no 'about:' section is added to the top of pasties",
"Default: false") { |x| @no_header = !x }
opts.on("-h", "--help",
"Show this help message.") { puts opts; exit }
self.args = opts.parse!(args)
end
end
end

## lib/pastiepacker/upload.rb
class PastiePacker
AVAILABLE_PARSERS = %w(
c++ css diff html_rails html javascript
php plain_text python
ruby ruby_on_rails sql
shell-unix-generic
)

def do_paste
if to_stdout?
output_stream.puts contents
else
url = API.new.paste contents, format, private?
end
end

class API
PASTIE_URI = 'pastie.caboo.se'

def paste(body, format='ruby', is_private=false)
raise InvalidParser unless valid_parser?(format)
http = Net::HTTP.new(PASTIE_URI)
query_string = { :paste => {
:body => CGI.escape(body),
:parser => format,
:restricted => is_private,
:authorization => 'burger'
}}.to_query_string
resp, body = http.start { |http|
http.post('/pastes', query_string)
}
if resp.code == '302'
return resp['location']
else
raise Error, "#{resp.code} - #{query_string.inspect} - #{resp.inspect}"
end
end

private
def valid_parser?(format)
PastiePacker::AVAILABLE_PARSERS.include?(format)
end
end

class Error < StandardError; end
class InvalidParser < StandardError; end

end

#
# irb(main):002:0> y = `file ~/Pictures/ryandavis2.jpg README.txt`
# => "/Users/nicwilliams/Pictures/ryandavis2.jpg: JPEG image data, JFIF standard 1.01\nREADME.txt: ASCII English text\n"
# irb(main):003:0> YAML.load(y)
# => {"/Users/nicwilliams/Pictures/ryandavis2.jpg"=>"JPEG image data, JFIF standard 1.01", "README.txt"=>"ASCII English text"}
# irb(main):004:0> YAML.load(y).keys
# => ["/Users/nicwilliams/Pictures/ryandavis2.jpg", "README.txt"]
# file --mime ~/Pictures/ryandavis2.jpg README.txt

## lib/pastiepacker/version.rb
module PastiePacker #:nodoc:
module VERSION #:nodoc:
MAJOR = 1
MINOR = 0
TINY = 0

STRING = [MAJOR, MINOR, TINY].join('.')
end
end


## lib/ruby-ext/hash.rb
class Hash
def to_query_string
map { |k, v|
if v.instance_of?(Hash)
v.map { |sk, sv|
"#{k}[#{sk}]=#{sv}"
}.join('&')
else
"#{k}=#{v}"
end
}.join('&')
end
end


## log/debug.log


## script/console
#!/usr/bin/env ruby
# File: script/console
irb = RUBY_PLATFORM =~ /(:?mswin|mingw)/ ? 'irb.bat' : 'irb'

libs = " -r irb/completion"
# Perhaps use a console_lib to store any extra methods I may want available in the cosole
# libs << " -r #{File.dirname(__FILE__) + '/../lib/console_lib/console_logger.rb'}"
libs << " -r #{File.dirname(__FILE__) + '/../lib/pastiepacker.rb'}"

puts "Loading pastiepacker gem"
exec "#{irb} #{libs} --simple-prompt"

## script/destroy
#!/usr/bin/env ruby
APP_ROOT = File.expand_path(File.join(File.dirname(__FILE__), '..'))

begin
require 'rubigen'
rescue LoadError
require 'rubygems'
require 'rubigen'
end
require 'rubigen/scripts/destroy'

ARGV.shift if ['--help', '-h'].include?(ARGV[0])
RubiGen::Base.use_component_sources! [:rubygems, :newgem, :newgem_theme, :test_unit]
RubiGen::Scripts::Destroy.new.run(ARGV)


## script/generate
#!/usr/bin/env ruby
APP_ROOT = File.expand_path(File.join(File.dirname(__FILE__), '..'))

begin
require 'rubigen'
rescue LoadError
require 'rubygems'
require 'rubigen'
end
require 'rubigen/scripts/generate'

ARGV.shift if ['--help', '-h'].include?(ARGV[0])
RubiGen::Base.use_component_sources! [:rubygems, :newgem, :newgem_theme, :test_unit]
RubiGen::Scripts::Generate.new.run(ARGV)


## script/txt2html
#!/usr/bin/env ruby

require 'rubygems'
begin
require 'newgem'
rescue LoadError
puts "\n\nGenerating the website requires the newgem RubyGem"
puts "Install: gem install newgem\n\n"
exit(1)
end
require 'redcloth'
require 'syntax/convertors/html'
require 'erb'
require File.dirname(__FILE__) + '/../lib/pastiepacker/version.rb'

version = PastiePacker::VERSION::STRING
download = 'http://rubyforge.org/projects/pastiepacker'

class Fixnum
def ordinal
# teens
return 'th' if (10..19).include?(self % 100)
# others
case self % 10
when 1: return 'st'
when 2: return 'nd'
when 3: return 'rd'
else return 'th'
end
end
end

class Time
def pretty
return "#{mday}#{mday.ordinal} #{strftime('%B')} #{year}"
end
end

def convert_syntax(syntax, source)
return Syntax::Convertors::HTML.for_syntax(syntax).convert(source).gsub(%r!^<pre>|</pre>$!,'')
end

if ARGV.length >= 1
src, template = ARGV
template ||= File.join(File.dirname(__FILE__), '/../website/template.html.erb')

else
puts("Usage: #{File.split($0).last} source.txt [template.html.erb] > output.html")
exit!
end

template = ERB.new(File.open(template).read)

title = nil
body = nil
File.open(src) do |fsrc|
title_text = fsrc.readline
body_text = fsrc.read
syntax_items = []
body_text.gsub!(%r!<(pre|code)[^>]*?syntax=['"]([^'"]+)[^>]*>(.*?)</\1>!m){
ident = syntax_items.length
element, syntax, source = $1, $2, $3
syntax_items << "<#{element} class='syntax'>#{convert_syntax(syntax, source)}</#{element}>"
"syntax-temp-#{ident}"
}
title = RedCloth.new(title_text).to_html.gsub(%r!<.*?>!,'').strip
body = RedCloth.new(body_text).to_html
body.gsub!(%r!(?:<pre><code>)?syntax-temp-(\d+)(?:</code></pre>)?!){ syntax_items[$1.to_i] }
end
stat = File.stat(src)
created = stat.ctime
modified = stat.mtime

$stdout << template.result(binding)


## setup.rb
#
# setup.rb
#
# Copyright (c) 2000-2005 Minero Aoki
#
# This program is free software.
# You can distribute/modify this program under the terms of
# the GNU LGPL, Lesser General Public License version 2.1.
#

unless Enumerable.method_defined?(:map) # Ruby 1.4.6
module Enumerable
alias map collect
end
end

unless File.respond_to?(:read) # Ruby 1.6
def File.read(fname)
open(fname) {|f|
return f.read
}
end
end

unless Errno.const_defined?(:ENOTEMPTY) # Windows?
module Errno
class ENOTEMPTY
# We do not raise this exception, implementation is not needed.
end
end
end

def File.binread(fname)
open(fname, 'rb') {|f|
return f.read
}
end

# for corrupted Windows' stat(2)
def File.dir?(path)
File.directory?((path[-1,1] == '/') ? path : path + '/')
end


class ConfigTable

include Enumerable

def initialize(rbconfig)
@rbconfig = rbconfig
@items = []
@table = {}
# options
@install_prefix = nil
@config_opt = nil
@verbose = true
@no_harm = false
end

attr_accessor :install_prefix
attr_accessor :config_opt

attr_writer :verbose

def verbose?
@verbose
end

attr_writer :no_harm

def no_harm?
@no_harm
end

def [](key)
lookup(key).resolve(self)
end

def []=(key, val)
lookup(key).set val
end

def names
@items.map {|i| i.name }
end

def each(&block)
@items.each(&block)
end

def key?(name)
@table.key?(name)
end

def lookup(name)
@table[name] or setup_rb_error "no such config item: #{name}"
end

def add(item)
@items.push item
@table[item.name] = item
end

def remove(name)
item = lookup(name)
@items.delete_if {|i| i.name == name }
@table.delete_if {|name, i| i.name == name }
item
end

def load_script(path, inst = nil)
if File.file?(path)
MetaConfigEnvironment.new(self, inst).instance_eval File.read(path), path
end
end

def savefile
'.config'
end

def load_savefile
begin
File.foreach(savefile()) do |line|
k, v = *line.split(/=/, 2)
self[k] = v.strip
end
rescue Errno::ENOENT
setup_rb_error $!.message + "\n#{File.basename($0)} config first"
end
end

def save
@items.each {|i| i.value }
File.open(savefile(), 'w') {|f|
@items.each do |i|
f.printf "%s=%s\n", i.name, i.value if i.value? and i.value
end
}
end

def load_standard_entries
standard_entries(@rbconfig).each do |ent|
add ent
end
end

def standard_entries(rbconfig)
c = rbconfig

rubypath = File.join(c['bindir'], c['ruby_install_name'] + c['EXEEXT'])

major = c['MAJOR'].to_i
minor = c['MINOR'].to_i
teeny = c['TEENY'].to_i
version = "#{major}.#{minor}"

# ruby ver. >= 1.4.4?
newpath_p = ((major >= 2) or
((major == 1) and
((minor >= 5) or
((minor == 4) and (teeny >= 4)))))

if c['rubylibdir']
# V > 1.6.3
libruby = "#{c['prefix']}/lib/ruby"
librubyver = c['rubylibdir']
librubyverarch = c['archdir']
siteruby = c['sitedir']
siterubyver = c['sitelibdir']
siterubyverarch = c['sitearchdir']
elsif newpath_p
# 1.4.4 <= V <= 1.6.3
libruby = "#{c['prefix']}/lib/ruby"
librubyver = "#{c['prefix']}/lib/ruby/#{version}"
librubyverarch = "#{c['prefix']}/lib/ruby/#{version}/#{c['arch']}"
siteruby = c['sitedir']
siterubyver = "$siteruby/#{version}"
siterubyverarch = "$siterubyver/#{c['arch']}"
else
# V < 1.4.4
libruby = "#{c['prefix']}/lib/ruby"
librubyver = "#{c['prefix']}/lib/ruby/#{version}"
librubyverarch = "#{c['prefix']}/lib/ruby/#{version}/#{c['arch']}"
siteruby = "#{c['prefix']}/lib/ruby/#{version}/site_ruby"
siterubyver = siteruby
siterubyverarch = "$siterubyver/#{c['arch']}"
end
parameterize = lambda {|path|
path.sub(/\A#{Regexp.quote(c['prefix'])}/, '$prefix')
}

if arg = c['configure_args'].split.detect {|arg| /--with-make-prog=/ =~ arg }
makeprog = arg.sub(/'/, '').split(/=/, 2)[1]
else
makeprog = 'make'
end

[
ExecItem.new('installdirs', 'std/site/home',
'std: install under libruby; site: install under site_ruby; home: install under $HOME')\
{|val, table|
case val
when 'std'
table['rbdir'] = '$librubyver'
table['sodir'] = '$librubyverarch'
when 'site'
table['rbdir'] = '$siterubyver'
table['sodir'] = '$siterubyverarch'
when 'home'
setup_rb_error '$HOME was not set' unless ENV['HOME']
table['prefix'] = ENV['HOME']
table['rbdir'] = '$libdir/ruby'
table['sodir'] = '$libdir/ruby'
end
},
PathItem.new('prefix', 'path', c['prefix'],
'path prefix of target environment'),
PathItem.new('bindir', 'path', parameterize.call(c['bindir']),
'the directory for commands'),
PathItem.new('libdir', 'path', parameterize.call(c['libdir']),
'the directory for libraries'),
PathItem.new('datadir', 'path', parameterize.call(c['datadir']),
'the directory for shared data'),
PathItem.new('mandir', 'path', parameterize.call(c['mandir']),
'the directory for man pages'),
PathItem.new('sysconfdir', 'path', parameterize.call(c['sysconfdir']),
'the directory for system configuration files'),
PathItem.new('localstatedir', 'path', parameterize.call(c['localstatedir']),
'the directory for local state data'),
PathItem.new('libruby', 'path', libruby,
'the directory for ruby libraries'),
PathItem.new('librubyver', 'path', librubyver,
'the directory for standard ruby libraries'),
PathItem.new('librubyverarch', 'path', librubyverarch,
'the directory for standard ruby extensions'),
PathItem.new('siteruby', 'path', siteruby,
'the directory for version-independent aux ruby libraries'),
PathItem.new('siterubyver', 'path', siterubyver,
'the directory for aux ruby libraries'),
PathItem.new('siterubyverarch', 'path', siterubyverarch,
'the directory for aux ruby binaries'),
PathItem.new('rbdir', 'path', '$siterubyver',
'the directory for ruby scripts'),
PathItem.new('sodir', 'path', '$siterubyverarch',
'the directory for ruby extentions'),
PathItem.new('rubypath', 'path', rubypath,
'the path to set to #! line'),
ProgramItem.new('rubyprog', 'name', rubypath,
'the ruby program using for installation'),
ProgramItem.new('makeprog', 'name', makeprog,
'the make program to compile ruby extentions'),
SelectItem.new('shebang', 'all/ruby/never', 'ruby',
'shebang line (#!) editing mode'),
BoolItem.new('without-ext', 'yes/no', 'no',
'does not compile/install ruby extentions')
]
end
private :standard_entries

def load_multipackage_entries
multipackage_entries().each do |ent|
add ent
end
end

def multipackage_entries
[
PackageSelectionItem.new('with', 'name,name...', '', 'ALL',
'package names that you want to install'),
PackageSelectionItem.new('without', 'name,name...', '', 'NONE',
'package names that you do not want to install')
]
end
private :multipackage_entries

ALIASES = {
'std-ruby' => 'librubyver',
'stdruby' => 'librubyver',
'rubylibdir' => 'librubyver',
'archdir' => 'librubyverarch',
'site-ruby-common' => 'siteruby', # For backward compatibility
'site-ruby' => 'siterubyver', # For backward compatibility
'bin-dir' => 'bindir',
'bin-dir' => 'bindir',
'rb-dir' => 'rbdir',
'so-dir' => 'sodir',
'data-dir' => 'datadir',
'ruby-path' => 'rubypath',
'ruby-prog' => 'rubyprog',
'ruby' => 'rubyprog',
'make-prog' => 'makeprog',
'make' => 'makeprog'
}

def fixup
ALIASES.each do |ali, name|
@table[ali] = @table[name]
end
@items.freeze
@table.freeze
@options_re = /\A--(#{@table.keys.join('|')})(?:=(.*))?\z/
end

def parse_opt(opt)
m = @options_re.match(opt) or setup_rb_error "config: unknown option #{opt}"
m.to_a[1,2]
end

def dllext
@rbconfig['DLEXT']
end

def value_config?(name)
lookup(name).value?
end

class Item
def initialize(name, template, default, desc)
@name = name.freeze
@template = template
@value = default
@default = default
@description = desc
end

attr_reader :name
attr_reader :description

attr_accessor :default
alias help_default default

def help_opt
"--#{@name}=#{@template}"
end

def value?
true
end

def value
@value
end

def resolve(table)
@value.gsub(%r<\$([^/]+)>) { table[$1] }
end

def set(val)
@value = check(val)
end

private

def check(val)
setup_rb_error "config: --#{name} requires argument" unless val
val
end
end

class BoolItem < Item
def config_type
'bool'
end

def help_opt
"--#{@name}"
end

private

def check(val)
return 'yes' unless val
case val
when /\Ay(es)?\z/i, /\At(rue)?\z/i then 'yes'
when /\An(o)?\z/i, /\Af(alse)\z/i then 'no'
else
setup_rb_error "config: --#{@name} accepts only yes/no for argument"
end
end
end

class PathItem < Item
def config_type
'path'
end

private

def check(path)
setup_rb_error "config: --#{@name} requires argument" unless path
path[0,1] == '$' ? path : File.expand_path(path)
end
end

class ProgramItem < Item
def config_type
'program'
end
end

class SelectItem < Item
def initialize(name, selection, default, desc)
super
@ok = selection.split('/')
end

def config_type
'select'
end

private

def check(val)
unless @ok.include?(val.strip)
setup_rb_error "config: use --#{@name}=#{@template} (#{val})"
end
val.strip
end
end

class ExecItem < Item
def initialize(name, selection, desc, &block)
super name, selection, nil, desc
@ok = selection.split('/')
@action = block
end

def config_type
'exec'
end

def value?
false
end

def resolve(table)
setup_rb_error "$#{name()} wrongly used as option value"
end

undef set

def evaluate(val, table)
v = val.strip.downcase
unless @ok.include?(v)
setup_rb_error "invalid option --#{@name}=#{val} (use #{@template})"
end
@action.call v, table
end
end

class PackageSelectionItem < Item
def initialize(name, template, default, help_default, desc)
super name, template, default, desc
@help_default = help_default
end

attr_reader :help_default

def config_type
'package'
end

private

def check(val)
unless File.dir?("packages/#{val}")
setup_rb_error "config: no such package: #{val}"
end
val
end
end

class MetaConfigEnvironment
def initialize(config, installer)
@config = config
@installer = installer
end

def config_names
@config.names
end

def config?(name)
@config.key?(name)
end

def bool_config?(name)
@config.lookup(name).config_type == 'bool'
end

def path_config?(name)
@config.lookup(name).config_type == 'path'
end

def value_config?(name)
@config.lookup(name).config_type != 'exec'
end

def add_config(item)
@config.add item
end

def add_bool_config(name, default, desc)
@config.add BoolItem.new(name, 'yes/no', default ? 'yes' : 'no', desc)
end

def add_path_config(name, default, desc)
@config.add PathItem.new(name, 'path', default, desc)
end

def set_config_default(name, default)
@config.lookup(name).default = default
end

def remove_config(name)
@config.remove(name)
end

# For only multipackage
def packages
raise '[setup.rb fatal] multi-package metaconfig API packages() called for single-package; contact application package vendor' unless @installer
@installer.packages
end

# For only multipackage
def declare_packages(list)
raise '[setup.rb fatal] multi-package metaconfig API declare_packages() called for single-package; contact application package vendor' unless @installer
@installer.packages = list
end
end

end # class ConfigTable


# This module requires: #verbose?, #no_harm?
module FileOperations

def mkdir_p(dirname, prefix = nil)
dirname = prefix + File.expand_path(dirname) if prefix
$stderr.puts "mkdir -p #{dirname}" if verbose?
return if no_harm?

# Does not check '/', it's too abnormal.
dirs = File.expand_path(dirname).split(%r<(?=/)>)
if /\A[a-z]:\z/i =~ dirs[0]
disk = dirs.shift
dirs[0] = disk + dirs[0]
end
dirs.each_index do |idx|
path = dirs[0..idx].join('')
Dir.mkdir path unless File.dir?(path)
end
end

def rm_f(path)
$stderr.puts "rm -f #{path}" if verbose?
return if no_harm?
force_remove_file path
end

def rm_rf(path)
$stderr.puts "rm -rf #{path}" if verbose?
return if no_harm?
remove_tree path
end

def remove_tree(path)
if File.symlink?(path)
remove_file path
elsif File.dir?(path)
remove_tree0 path
else
force_remove_file path
end
end

def remove_tree0(path)
Dir.foreach(path) do |ent|
next if ent == '.'
next if ent == '..'
entpath = "#{path}/#{ent}"
if File.symlink?(entpath)
remove_file entpath
elsif File.dir?(entpath)
remove_tree0 entpath
else
force_remove_file entpath
end
end
begin
Dir.rmdir path
rescue Errno::ENOTEMPTY
# directory may not be empty
end
end

def move_file(src, dest)
force_remove_file dest
begin
File.rename src, dest
rescue
File.open(dest, 'wb') {|f|
f.write File.binread(src)
}
File.chmod File.stat(src).mode, dest
File.unlink src
end
end

def force_remove_file(path)
begin
remove_file path
rescue
end
end

def remove_file(path)
File.chmod 0777, path
File.unlink path
end

def install(from, dest, mode, prefix = nil)
$stderr.puts "install #{from} #{dest}" if verbose?
return if no_harm?

realdest = prefix ? prefix + File.expand_path(dest) : dest
realdest = File.join(realdest, File.basename(from)) if File.dir?(realdest)
str = File.binread(from)
if diff?(str, realdest)
verbose_off {
rm_f realdest if File.exist?(realdest)
}
File.open(realdest, 'wb') {|f|
f.write str
}
File.chmod mode, realdest

File.open("#{objdir_root()}/InstalledFiles", 'a') {|f|
if prefix
f.puts realdest.sub(prefix, '')
else
f.puts realdest
end
}
end
end

def diff?(new_content, path)
return true unless File.exist?(path)
new_content != File.binread(path)
end

def command(*args)
$stderr.puts args.join(' ') if verbose?
system(*args) or raise RuntimeError,
"system(#{args.map{|a| a.inspect }.join(' ')}) failed"
end

def ruby(*args)
command config('rubyprog'), *args
end

def make(task = nil)
command(*[config('makeprog'), task].compact)
end

def extdir?(dir)
File.exist?("#{dir}/MANIFEST") or File.exist?("#{dir}/extconf.rb")
end

def files_of(dir)
Dir.open(dir) {|d|
return d.select {|ent| File.file?("#{dir}/#{ent}") }
}
end

DIR_REJECT = %w( . .. CVS SCCS RCS CVS.adm .svn )

def directories_of(dir)
Dir.open(dir) {|d|
return d.select {|ent| File.dir?("#{dir}/#{ent}") } - DIR_REJECT
}
end

end


# This module requires: #srcdir_root, #objdir_root, #relpath
module HookScriptAPI

def get_config(key)
@config[key]
end

alias config get_config

# obsolete: use metaconfig to change configuration
def set_config(key, val)
@config[key] = val
end

#
# srcdir/objdir (works only in the package directory)
#

def curr_srcdir
"#{srcdir_root()}/#{relpath()}"
end

def curr_objdir
"#{objdir_root()}/#{relpath()}"
end

def srcfile(path)
"#{curr_srcdir()}/#{path}"
end

def srcexist?(path)
File.exist?(srcfile(path))
end

def srcdirectory?(path)
File.dir?(srcfile(path))
end

def srcfile?(path)
File.file?(srcfile(path))
end

def srcentries(path = '.')
Dir.open("#{curr_srcdir()}/#{path}") {|d|
return d.to_a - %w(. ..)
}
end

def srcfiles(path = '.')
srcentries(path).select {|fname|
File.file?(File.join(curr_srcdir(), path, fname))
}
end

def srcdirectories(path = '.')
srcentries(path).select {|fname|
File.dir?(File.join(curr_srcdir(), path, fname))
}
end

end


class ToplevelInstaller

Version = '3.4.1'
Copyright = 'Copyright (c) 2000-2005 Minero Aoki'

TASKS = [
[ 'all', 'do config, setup, then install' ],
[ 'config', 'saves your configurations' ],
[ 'show', 'shows current configuration' ],
[ 'setup', 'compiles ruby extentions and others' ],
[ 'install', 'installs files' ],
[ 'test', 'run all tests in test/' ],
[ 'clean', "does `make clean' for each extention" ],
[ 'distclean',"does `make distclean' for each extention" ]
]

def ToplevelInstaller.invoke
config = ConfigTable.new(load_rbconfig())
config.load_standard_entries
config.load_multipackage_entries if multipackage?
config.fixup
klass = (multipackage?() ? ToplevelInstallerMulti : ToplevelInstaller)
klass.new(File.dirname($0), config).invoke
end

def ToplevelInstaller.multipackage?
File.dir?(File.dirname($0) + '/packages')
end

def ToplevelInstaller.load_rbconfig
if arg = ARGV.detect {|arg| /\A--rbconfig=/ =~ arg }
ARGV.delete(arg)
load File.expand_path(arg.split(/=/, 2)[1])
$".push 'rbconfig.rb'
else
require 'rbconfig'
end
::Config::CONFIG
end

def initialize(ardir_root, config)
@ardir = File.expand_path(ardir_root)
@config = config
# cache
@valid_task_re = nil
end

def config(key)
@config[key]
end

def inspect
"#<#{self.class} #{__id__()}>"
end

def invoke
run_metaconfigs
case task = parsearg_global()
when nil, 'all'
parsearg_config
init_installers
exec_config
exec_setup
exec_install
else
case task
when 'config', 'test'
;
when 'clean', 'distclean'
@config.load_savefile if File.exist?(@config.savefile)
else
@config.load_savefile
end
__send__ "parsearg_#{task}"
init_installers
__send__ "exec_#{task}"
end
end

def run_metaconfigs
@config.load_script "#{@ardir}/metaconfig"
end

def init_installers
@installer = Installer.new(@config, @ardir, File.expand_path('.'))
end

#
# Hook Script API bases
#

def srcdir_root
@ardir
end

def objdir_root
'.'
end

def relpath
'.'
end

#
# Option Parsing
#

def parsearg_global
while arg = ARGV.shift
case arg
when /\A\w+\z/
setup_rb_error "invalid task: #{arg}" unless valid_task?(arg)
return arg
when '-q', '--quiet'
@config.verbose = false
when '--verbose'
@config.verbose = true
when '--help'
print_usage $stdout
exit 0
when '--version'
puts "#{File.basename($0)} version #{Version}"
exit 0
when '--copyright'
puts Copyright
exit 0
else
setup_rb_error "unknown global option '#{arg}'"
end
end
nil
end

def valid_task?(t)
valid_task_re() =~ t
end

def valid_task_re
@valid_task_re ||= /\A(?:#{TASKS.map {|task,desc| task }.join('|')})\z/
end

def parsearg_no_options
unless ARGV.empty?
task = caller(0).first.slice(%r<`parsearg_(\w+)'>, 1)
setup_rb_error "#{task}: unknown options: #{ARGV.join(' ')}"
end
end

alias parsearg_show parsearg_no_options
alias parsearg_setup parsearg_no_options
alias parsearg_test parsearg_no_options
alias parsearg_clean parsearg_no_options
alias parsearg_distclean parsearg_no_options

def parsearg_config
evalopt = []
set = []
@config.config_opt = []
while i = ARGV.shift
if /\A--?\z/ =~ i
@config.config_opt = ARGV.dup
break
end
name, value = *@config.parse_opt(i)
if @config.value_config?(name)
@config[name] = value
else
evalopt.push [name, value]
end
set.push name
end
evalopt.each do |name, value|
@config.lookup(name).evaluate value, @config
end
# Check if configuration is valid
set.each do |n|
@config[n] if @config.value_config?(n)
end
end

def parsearg_install
@config.no_harm = false
@config.install_prefix = ''
while a = ARGV.shift
case a
when '--no-harm'
@config.no_harm = true
when /\A--prefix=/
path = a.split(/=/, 2)[1]
path = File.expand_path(path) unless path[0,1] == '/'
@config.install_prefix = path
else
setup_rb_error "install: unknown option #{a}"
end
end
end

def print_usage(out)
out.puts 'Typical Installation Procedure:'
out.puts " $ ruby #{File.basename $0} config"
out.puts " $ ruby #{File.basename $0} setup"
out.puts " # ruby #{File.basename $0} install (may require root privilege)"
out.puts
out.puts 'Detailed Usage:'
out.puts " ruby #{File.basename $0} <global option>"
out.puts " ruby #{File.basename $0} [<global options>] <task> [<task options>]"

fmt = " %-24s %s\n"
out.puts
out.puts 'Global options:'
out.printf fmt, '-q,--quiet', 'suppress message outputs'
out.printf fmt, ' --verbose', 'output messages verbosely'
out.printf fmt, ' --help', 'print this message'
out.printf fmt, ' --version', 'print version and quit'
out.printf fmt, ' --copyright', 'print copyright and quit'
out.puts
out.puts 'Tasks:'
TASKS.each do |name, desc|
out.printf fmt, name, desc
end

fmt = " %-24s %s [%s]\n"
out.puts
out.puts 'Options for CONFIG or ALL:'
@config.each do |item|
out.printf fmt, item.help_opt, item.description, item.help_default
end
out.printf fmt, '--rbconfig=path', 'rbconfig.rb to load',"running ruby's"
out.puts
out.puts 'Options for INSTALL:'
out.printf fmt, '--no-harm', 'only display what to do if given', 'off'
out.printf fmt, '--prefix=path', 'install path prefix', ''
out.puts
end

#
# Task Handlers
#

def exec_config
@installer.exec_config
@config.save # must be final
end

def exec_setup
@installer.exec_setup
end

def exec_install
@installer.exec_install
end

def exec_test
@installer.exec_test
end

def exec_show
@config.each do |i|
printf "%-20s %s\n", i.name, i.value if i.value?
end
end

def exec_clean
@installer.exec_clean
end

def exec_distclean
@installer.exec_distclean
end

end # class ToplevelInstaller


class ToplevelInstallerMulti < ToplevelInstaller

include FileOperations

def initialize(ardir_root, config)
super
@packages = directories_of("#{@ardir}/packages")
raise 'no package exists' if @packages.empty?
@root_installer = Installer.new(@config, @ardir, File.expand_path('.'))
end

def run_metaconfigs
@config.load_script "#{@ardir}/metaconfig", self
@packages.each do |name|
@config.load_script "#{@ardir}/packages/#{name}/metaconfig"
end
end

attr_reader :packages

def packages=(list)
raise 'package list is empty' if list.empty?
list.each do |name|
raise "directory packages/#{name} does not exist"\
unless File.dir?("#{@ardir}/packages/#{name}")
end
@packages = list
end

def init_installers
@installers = {}
@packages.each do |pack|
@installers[pack] = Installer.new(@config,
"#{@ardir}/packages/#{pack}",
"packages/#{pack}")
end
with = extract_selection(config('with'))
without = extract_selection(config('without'))
@selected = @installers.keys.select {|name|
(with.empty? or with.include?(name)) \
and not without.include?(name)
}
end

def extract_selection(list)
a = list.split(/,/)
a.each do |name|
setup_rb_error "no such package: #{name}" unless @installers.key?(name)
end
a
end

def print_usage(f)
super
f.puts 'Inluded packages:'
f.puts ' ' + @packages.sort.join(' ')
f.puts
end

#
# Task Handlers
#

def exec_config
run_hook 'pre-config'
each_selected_installers {|inst| inst.exec_config }
run_hook 'post-config'
@config.save # must be final
end

def exec_setup
run_hook 'pre-setup'
each_selected_installers {|inst| inst.exec_setup }
run_hook 'post-setup'
end

def exec_install
run_hook 'pre-install'
each_selected_installers {|inst| inst.exec_install }
run_hook 'post-install'
end

def exec_test
run_hook 'pre-test'
each_selected_installers {|inst| inst.exec_test }
run_hook 'post-test'
end

def exec_clean
rm_f @config.savefile
run_hook 'pre-clean'
each_selected_installers {|inst| inst.exec_clean }
run_hook 'post-clean'
end

def exec_distclean
rm_f @config.savefile
run_hook 'pre-distclean'
each_selected_installers {|inst| inst.exec_distclean }
run_hook 'post-distclean'
end

#
# lib
#

def each_selected_installers
Dir.mkdir 'packages' unless File.dir?('packages')
@selected.each do |pack|
$stderr.puts "Processing the package `#{pack}' ..." if verbose?
Dir.mkdir "packages/#{pack}" unless File.dir?("packages/#{pack}")
Dir.chdir "packages/#{pack}"
yield @installers[pack]
Dir.chdir '../..'
end
end

def run_hook(id)
@root_installer.run_hook id
end

# module FileOperations requires this
def verbose?
@config.verbose?
end

# module FileOperations requires this
def no_harm?
@config.no_harm?
end

end # class ToplevelInstallerMulti


class Installer

FILETYPES = %w( bin lib ext data conf man )

include FileOperations
include HookScriptAPI

def initialize(config, srcroot, objroot)
@config = config
@srcdir = File.expand_path(srcroot)
@objdir = File.expand_path(objroot)
@currdir = '.'
end

def inspect
"#<#{self.class} #{File.basename(@srcdir)}>"
end

def noop(rel)
end

#
# Hook Script API base methods
#

def srcdir_root
@srcdir
end

def objdir_root
@objdir
end

def relpath
@currdir
end

#
# Config Access
#

# module FileOperations requires this
def verbose?
@config.verbose?
end

# module FileOperations requires this
def no_harm?
@config.no_harm?
end

def verbose_off
begin
save, @config.verbose = @config.verbose?, false
yield
ensure
@config.verbose = save
end
end

#
# TASK config
#

def exec_config
exec_task_traverse 'config'
end

alias config_dir_bin noop
alias config_dir_lib noop

def config_dir_ext(rel)
extconf if extdir?(curr_srcdir())
end

alias config_dir_data noop
alias config_dir_conf noop
alias config_dir_man noop

def extconf
ruby "#{curr_srcdir()}/extconf.rb", *@config.config_opt
end

#
# TASK setup
#

def exec_setup
exec_task_traverse 'setup'
end

def setup_dir_bin(rel)
files_of(curr_srcdir()).each do |fname|
update_shebang_line "#{curr_srcdir()}/#{fname}"
end
end

alias setup_dir_lib noop

def setup_dir_ext(rel)
make if extdir?(curr_srcdir())
end

alias setup_dir_data noop
alias setup_dir_conf noop
alias setup_dir_man noop

def update_shebang_line(path)
return if no_harm?
return if config('shebang') == 'never'
old = Shebang.load(path)
if old
$stderr.puts "warning: #{path}: Shebang line includes too many args. It is not portable and your program may not work." if old.args.size > 1
new = new_shebang(old)
return if new.to_s == old.to_s
else
return unless config('shebang') == 'all'
new = Shebang.new(config('rubypath'))
end
$stderr.puts "updating shebang: #{File.basename(path)}" if verbose?
open_atomic_writer(path) {|output|
File.open(path, 'rb') {|f|
f.gets if old # discard
output.puts new.to_s
output.print f.read
}
}
end

def new_shebang(old)
if /\Aruby/ =~ File.basename(old.cmd)
Shebang.new(config('rubypath'), old.args)
elsif File.basename(old.cmd) == 'env' and old.args.first == 'ruby'
Shebang.new(config('rubypath'), old.args[1..-1])
else
return old unless config('shebang') == 'all'
Shebang.new(config('rubypath'))
end
end

def open_atomic_writer(path, &block)
tmpfile = File.basename(path) + '.tmp'
begin
File.open(tmpfile, 'wb', &block)
File.rename tmpfile, File.basename(path)
ensure
File.unlink tmpfile if File.exist?(tmpfile)
end
end

class Shebang
def Shebang.load(path)
line = nil
File.open(path) {|f|
line = f.gets
}
return nil unless /\A#!/ =~ line
parse(line)
end

def Shebang.parse(line)
cmd, *args = *line.strip.sub(/\A\#!/, '').split(' ')
new(cmd, args)
end

def initialize(cmd, args = [])
@cmd = cmd
@args = args
end

attr_reader :cmd
attr_reader :args

def to_s
"#! #{@cmd}" + (@args.empty? ? '' : " #{@args.join(' ')}")
end
end

#
# TASK install
#

def exec_install
rm_f 'InstalledFiles'
exec_task_traverse 'install'
end

def install_dir_bin(rel)
install_files targetfiles(), "#{config('bindir')}/#{rel}", 0755
end

def install_dir_lib(rel)
install_files libfiles(), "#{config('rbdir')}/#{rel}", 0644
end

def install_dir_ext(rel)
return unless extdir?(curr_srcdir())
install_files rubyextentions('.'),
"#{config('sodir')}/#{File.dirname(rel)}",
0555
end

def install_dir_data(rel)
install_files targetfiles(), "#{config('datadir')}/#{rel}", 0644
end

def install_dir_conf(rel)
# FIXME: should not remove current config files
# (rename previous file to .old/.org)
install_files targetfiles(), "#{config('sysconfdir')}/#{rel}", 0644
end

def install_dir_man(rel)
install_files targetfiles(), "#{config('mandir')}/#{rel}", 0644
end

def install_files(list, dest, mode)
mkdir_p dest, @config.install_prefix
list.each do |fname|
install fname, dest, mode, @config.install_prefix
end
end

def libfiles
glob_reject(%w(*.y *.output), targetfiles())
end

def rubyextentions(dir)
ents = glob_select("*.#{@config.dllext}", targetfiles())
if ents.empty?
setup_rb_error "no ruby extention exists: 'ruby #{$0} setup' first"
end
ents
end

def targetfiles
mapdir(existfiles() - hookfiles())
end

def mapdir(ents)
ents.map {|ent|
if File.exist?(ent)
then ent # objdir
else "#{curr_srcdir()}/#{ent}" # srcdir
end
}
end

# picked up many entries from cvs-1.11.1/src/ignore.c
JUNK_FILES = %w(
core RCSLOG tags TAGS .make.state
.nse_depinfo #* .#* cvslog.* ,* .del-* *.olb
*~ *.old *.bak *.BAK *.orig *.rej _$* *$

*.org *.in .*
)

def existfiles
glob_reject(JUNK_FILES, (files_of(curr_srcdir()) | files_of('.')))
end

def hookfiles
%w( pre-%s post-%s pre-%s.rb post-%s.rb ).map {|fmt|
%w( config setup install clean ).map {|t| sprintf(fmt, t) }
}.flatten
end

def glob_select(pat, ents)
re = globs2re([pat])
ents.select {|ent| re =~ ent }
end

def glob_reject(pats, ents)
re = globs2re(pats)
ents.reject {|ent| re =~ ent }
end

GLOB2REGEX = {
'.' => '\.',
'$' => '\$',
'#' => '\#',
'*' => '.*'
}

def globs2re(pats)
/\A(?:#{
pats.map {|pat| pat.gsub(/[\.\$\#\*]/) {|ch| GLOB2REGEX[ch] } }.join('|')
})\z/
end

#
# TASK test
#

TESTDIR = 'test'

def exec_test
unless File.directory?('test')
$stderr.puts 'no test in this package' if verbose?
return
end
$stderr.puts 'Running tests...' if verbose?
begin
require 'test/unit'
rescue LoadError
setup_rb_error 'test/unit cannot loaded. You need Ruby 1.8 or later to invoke this task.'
end
runner = Test::Unit::AutoRunner.new(true)
runner.to_run << TESTDIR
runner.run
end

#
# TASK clean
#

def exec_clean
exec_task_traverse 'clean'
rm_f @config.savefile
rm_f 'InstalledFiles'
end

alias clean_dir_bin noop
alias clean_dir_lib noop
alias clean_dir_data noop
alias clean_dir_conf noop
alias clean_dir_man noop

def clean_dir_ext(rel)
return unless extdir?(curr_srcdir())
make 'clean' if File.file?('Makefile')
end

#
# TASK distclean
#

def exec_distclean
exec_task_traverse 'distclean'
rm_f @config.savefile
rm_f 'InstalledFiles'
end

alias distclean_dir_bin noop
alias distclean_dir_lib noop

def distclean_dir_ext(rel)
return unless extdir?(curr_srcdir())
make 'distclean' if File.file?('Makefile')
end

alias distclean_dir_data noop
alias distclean_dir_conf noop
alias distclean_dir_man noop

#
# Traversing
#

def exec_task_traverse(task)
run_hook "pre-#{task}"
FILETYPES.each do |type|
if type == 'ext' and config('without-ext') == 'yes'
$stderr.puts 'skipping ext/* by user option' if verbose?
next
end
traverse task, type, "#{task}_dir_#{type}"
end
run_hook "post-#{task}"
end

def traverse(task, rel, mid)
dive_into(rel) {
run_hook "pre-#{task}"
__send__ mid, rel.sub(%r[\A.*?(?:/|\z)], '')
directories_of(curr_srcdir()).each do |d|
traverse task, "#{rel}/#{d}", mid
end
run_hook "post-#{task}"
}
end

def dive_into(rel)
return unless File.dir?("#{@srcdir}/#{rel}")

dir = File.basename(rel)
Dir.mkdir dir unless File.dir?(dir)
prevdir = Dir.pwd
Dir.chdir dir
$stderr.puts '---> ' + rel if verbose?
@currdir = rel
yield
Dir.chdir prevdir
$stderr.puts '<--- ' + rel if verbose?
@currdir = File.dirname(rel)
end

def run_hook(id)
path = [ "#{curr_srcdir()}/#{id}",
"#{curr_srcdir()}/#{id}.rb" ].detect {|cand| File.file?(cand) }
return unless path
begin
instance_eval File.read(path), path, 1
rescue
raise if $DEBUG
setup_rb_error "hook #{path} failed:\n" + $!.message
end
end

end # class Installer


class SetupError < StandardError; end

def setup_rb_error(msg)
raise SetupError, msg
end

if $0 == __FILE__
begin
ToplevelInstaller.invoke
rescue SetupError
raise if $DEBUG
$stderr.puts $!.message
$stderr.puts "Try 'ruby #{$0} --help' for detailed usage."
exit 1
end
end


## tasks/deployment.rake
desc 'Release the website and new gem version'
task :deploy => [:check_version, :website, :release] do
puts "Remember to create SVN tag:"
puts "svn copy svn+ssh://#{rubyforge_username}@rubyforge.org/var/svn/#{PATH}/trunk " +
"svn+ssh://#{rubyforge_username}@rubyforge.org/var/svn/#{PATH}/tags/REL-#{VERS} "
puts "Suggested comment:"
puts "Tagging release #{CHANGES}"
end

desc 'Runs tasks website_generate and install_gem as a local deployment of the gem'
task :local_deploy => [:website_generate, :install_gem]

task :check_version do
unless ENV['VERSION']
puts 'Must pass a VERSION=x.y.z release version'
exit
end
unless ENV['VERSION'] == VERS
puts "Please update your version.rb to match the release version, currently #{VERS}"
exit
end
end

desc 'Install the package as a gem, without generating documentation(ri/rdoc)'
task :install_gem_no_doc => [:clean, :package] do
sh "#{'sudo ' unless Hoe::WINDOZE }gem install pkg/*.gem --no-rdoc --no-ri"
end

namespace :manifest do
desc 'Recreate Manifest.txt to include ALL files'
task :refresh do
`rake check_manifest | patch -p0 > Manifest.txt`
end
end

## tasks/environment.rake
task :ruby_env do
RUBY_APP = if RUBY_PLATFORM =~ /java/
"jruby"
else
"ruby"
end unless defined? RUBY_APP
end


## tasks/website.rake
desc 'Generate website files'
task :website_generate => :ruby_env do
(Dir['website/**/*.txt'] - Dir['website/version*.txt']).each do |txt|
sh %{ #{RUBY_APP} script/txt2html #{txt} > #{txt.gsub(/txt$/,'html')} }
end
end

desc 'Upload website files to rubyforge'
task :website_upload do
host = "#{rubyforge_username}@rubyforge.org"
remote_dir = "/var/www/gforge-projects/#{PATH}/"
local_dir = 'website'
sh %{rsync -aCv #{local_dir}/ #{host}:#{remote_dir}}
end

desc 'Generate and upload website files'
task :website => [:website_generate, :website_upload, :publish_docs]


## test/fixtures/private_pastie.html
<html>
<head id="head">
<title>Private Paste - Pastie</title>

</head>
<body>

<div id="topnews" style="display:none;">
<p>
Pastie now <a href="http://pastie.caboo.se/8763" class="subtle">auto-senses</a> if line-wrap is a bad or good idea. <a href="http://beast.caboo.se/forums/4">Feedback?</a>
</p>
</div>

<div id="header">

<ul id="nav">
<li class="hot"><a href="/pastes/new">New</a></li>

<li style="display:none;" class="loggedinonly">
<!--

-->
<a href="" id="mine_nav_link">Mine</a>

<li >
<a href="/pastes">All</a>

</li>

<li><a href="#" onclick="$('search').show(); $('search_box').focus();">Search</a>
</li>
<li id="search" style="display:none;">
<form action="/search/">
<input type="text" style="font-size:0.85em;" size="15" id="search_box" name="q" value="">
</form>
</li>

<li ><a href="/why/">Why</a></li>
<li ><a href="/help/">Help</a></li>

<li class="login loggedinonly" style="display:none;" id="login_nick"></li>

</ul>

<script language="javascript">logged_in_setup();</script>

<h1>

<img src="/images/pastie.gif" />
Pastie
<span class="sub"><a href='http://www.pledgie.com/campaign/show/16'>Show Pastie some love</a></span>
<!--
<span class="sub">What'cha want for Christmas?</span>
<span class="count"> pasties and counting</span>
-->
</h1>


</div>

<!-- <div id="news">
<p style="margin-bottom:0;">Now powered by <a href="http://www.sphinxsearch.com/">Sphinx</a>... Search doesn't suck anymore. :-) Give it a spin.</p>
</div> -->

<div id="container">


<div id="content">







<!--

-->

<!--
<p class="private psmall" id="private">
<img src="/images/icons/mini/locked.gif" />
This paste is private.
</p>
-->







<!-- only for someone who's logged in -->
<p class="smallutils" id="wrapoptions" style="display:none;">

<a href="http://pastie.caboo.se/175214/wrap?key=5hwfheniddqmyasmfcxaw">Wrap text</a>

</p>

<script language="javascript">
logged_in_setup();
if (logged_in_user && logged_in_user=="")
{ $('wrapoptions').show(); }
</script>


<div style="position:absolute; font-size:0.6em; color:#999; z-index:100; right:10px; top:10px;">
<label style="font-size:1em; color:#aaa">Theme:</label>
<select onchange="setActiveStyleSheet(this.value);" id="tm_theme_picker">
<option value="clean">Clean (Pastie)</option>
<option value="all_hallows_eve">All Hallow's Eve</option>
<option value="blackboard">Blackboard</option>

<option value="brilliance_black">Brilliance Black</option>
<option value="cobalt">Cobalt</option>
<option value="expreso_libre">Espresso Libre</option>
<option value="idle">IDLE</option>
<option value="mac_classic">Mac Classic</option>
<option value="magicwb_amiga">MagicWB (Amiga)</option>
<option value="pastels_on_dark">Pastels on Dark</option>
<option value="slate">Slate</option>
<option value="slush_and_poppies">Slush and Poppies</option>

<option value="sunburst">Sunburst</option>
<option value="sunburst_josh">Sunburst (Josh)</option>
<option value="twilight">Twilight</option>
<option value="vibrant_ink">Vibrant Ink</option>
</select>
</div>





<h2>about:pastiepacker </h2>







<div class="allcode">
<table cellpadding="0" cellspacing="0" border="0">
<tr>
<td>

<pre class="textmate-source-numbers">1
2
</pre>

</td>
<td>
<pre class="textmate-source"><pre class="sunburst"><span class='meta meta_paragraph meta_paragraph_text'>Files for pastiepacker uploaded by pastiepacker.
To unpack files see <span class='markup markup_underline markup_underline_link'>http://pastiepacker.rubyforge.org</span>
</pre></pre>
</td>
</tr>
</table>

</div>


<hr />


</div>

<div id="right">





<p>
<label>Pasted</label><br />
<span class="typo_date" title="Fri, 04 Apr 2008 12:09:37 GMT" id='paste_date'>April 04, 2008<br /> 7:09AM EDT</span>
<script language="javascript">
e=$('paste_date');
e.innerHTML=get_local_time_for_date(e.title);
</script>

</p>











<p style="margin-top:2em; font-size:0.75em;">
<a href="/pastes/175214/download?key=5hwfheniddqmyasmfcxaw" class="utility">Download</a>
<span class="pipe">|</span>
<a href="http://pastie.caboo.se/175214.txt?key=5hwfheniddqmyasmfcxaw" class="utility">View</a>
</p>



<p style="margin-top:2em;">
<a href="/pastes/175214/reply?key=5hwfheniddqmyasmfcxaw">Paste again</a>

</p>






<p style="margin-top:2em; display:none;" id="edit_link">
<a href="/pastes/175214/edit?key=5hwfheniddqmyasmfcxaw" class="utility remove">Edit paste</a>
</p>

<script language="javascript">
logged_in_setup();
pasties=readCookie("pasties");
if (logged_in_user && logged_in_user=="")
{ $('edit_link').show(); }
else if (pasties)
{
pasties=pasties.split("&");
if (pasties.include(175214))
{ $('edit_link').show(); }
}
</script>


</div>

<br style="clear:both;" />

</div>

<div id="footer">
<p class="legal">
<a href="/legal/" style="color:#369;">Legal</a>
</p>
<p class="disclaim">
<strong>

Use Pastie in your quest to save humanity,<br /> not in your evil plots to take over the world!
</strong>

</p>
<p class="credit">
Created by <a href="http://www.workingwithrails.com/person/5337-josh-goebel">Josh Goebel</a>
<span class="pipe">|</span>
Hosted by <a href="http://railsmachine.com/">Rails Machine</a>
</p>
<br style="clear:both;" />
</div>

<script src="http://www.google-analytics.com/urchin.js" type="text/javascript">
</script>
<script type="text/javascript">
_uacct = "UA-413537-1";
urchinTracker();
</script>
</body>
</html>


## test/fixtures/sample_app/History.txt
== 1.0.0

* Some history

## test/fixtures/sample_app/README.txt
This is the readme.txt


## test/fixtures/sample_app/lib/myapp.rb
class Myapp

end

## test/test_app.rb
require File.dirname(__FILE__) + '/test_helper'

class TestApp < Test::Unit::TestCase
attr_reader :base_folder, :target_folder

def setup
@base_folder = File.dirname(__FILE__) + "/fixtures/sample_app"
@target_folder = File.dirname(__FILE__) + "/unpack"
FileUtils.mkdir_p target_folder
end

def teardown
FileUtils.rm_rf target_folder
end

def test_pack_current_folder
PastiePacker::API.any_instance.expects(:paste).
with($complete_pastie_and_header, 'ruby', false).
returns("http://pastie.caboo.se/123456")
FileUtils.cd base_folder do
PastiePacker.run
end
end

def test_pack_with_file_names
PastiePacker::API.any_instance.expects(:paste).
with($some_files_and_header, 'ruby', false).
returns("http://pastie.caboo.se/123456")
files = ["README.txt\n", "lib/myapp.rb\n"]
PastiePacker.any_instance.expects(:input_lines).returns(files)
FileUtils.cd base_folder do
PastiePacker.run
end
end

def test_pack_with_extra_comment
pastie = PastiePacker.new
pastie.extra_message = 'This is a bonus comment.'
PastiePacker::API.any_instance.expects(:paste).
with($complete_pastie_and_header_and_comment, nil, nil).
returns("http: