#--
# Copyright (c) 2010-2012 Engine Yard, Inc.
# Copyright (c) 2007-2009 Sun Microsystems, Inc.
# This source code is available under the MIT license.
# See the file LICENSE.txt for details.
#++

raise "JRuby-Rack must be built with JRuby: try again with `jruby -S rake'" unless defined?(JRUBY_VERSION)

begin
  require 'bundler/setup'
rescue LoadError => e
  require('rubygems') && retry
  puts "Please install Bundler and run `bundle install` to ensure you have all dependencies"
  raise e
end
begin; require 'appraisal'; rescue LoadError; end

desc "Remove target directory"
task :clean do
  rm_r 'target' rescue nil
end

GENERATED = FileList.new

namespace :clean do
  desc "Remove generated files (only)"
  task :generated do
    GENERATED.each { |fn| rm_r fn rescue nil }
  end
  desc "Remove lib/*.jar (test) dependencies"
  task :lib do
    rm_r 'target/lib' rescue nil
  end
end

directory 'target/classes'

desc "Compile classes"
task(:compile => 'target/classes') do
  sh "./mvnw compile #{ENV['JRUBY_VERSION'] ? "-Djruby.version=#{ENV['JRUBY_VERSION']}" : ""}"
end

directory 'target/test-classes'

desc "Compile test classes"
task(:test_prepare => ['target/classes', 'target/test-classes']) do
  sh "./mvnw test-compile #{ENV['JRUBY_VERSION'] ? "-Djruby.version=#{ENV['JRUBY_VERSION']}" : ""}"
end

desc "Unpack the rack gem"
task :unpack_gem => "target" do |t|
  target = File.expand_path(t.prerequisites.first)
  rack_gemfile = Gem.loaded_specs['rack'].cache_file
  unless uptodate?("#{target}/vendor/rack.rb", [__FILE__, rack_gemfile])
    mkdir_p "target/vendor"
    require 'rubygems/installer'
    rack_dir = File.basename(rack_gemfile).sub(/\.gem$/, '')
    Gem::Package.new(rack_gemfile).extract_files("#{target}/#{rack_dir}")
    File.open("#{target}/vendor/rack.rb", "w") do |f|
      f << "dir = File.dirname(__FILE__)\n"
      f << "if dir =~ /.jar!/ && dir !~ /^file:/\n"
      f << "  $LOAD_PATH.unshift 'file:' + dir + '/#{rack_dir}'\n"
      f << "else\n"
      f << "  $LOAD_PATH.unshift dir + '/#{rack_dir}'\n"
      f << "end\n"
      f << "require 'rack'"
    end
  end
end
GENERATED << 'target/vendor/rack.rb'

desc "Generate (ruby) resources"
task :resources => ['target/classes', :unpack_gem] do |t|
  rack_dir = File.basename(FileList["target/rack-*"].first)
  classes_dir = t.prerequisites.first
  { 'target/vendor' => "#{classes_dir}/vendor",
    "target/#{rack_dir}/lib" => "#{classes_dir}/vendor/#{rack_dir}"}.each do |src,dest|
    mkdir_p dest
    FileList["#{src}/*"].each do |f|
      cp_r f, dest
    end
  end
end

task :test_resources => ["target/test-classes"]

namespace :resources do
  desc "Copy (and generate) resources"
  task :copy => :resources do
    sh './mvnw process-resources -Dmdep.skip=true'
  end
  desc "Generate test resources"
  task :test => :test_resources
end

task :speconly => [ :resources, :test_resources ] do
  if ENV['SKIP_SPECS'].to_s == 'true'
    puts "Skipping specs due to SKIP_SPECS=#{ENV['SKIP_SPECS']}"
  else
    opts = ENV['SPEC_OPTS'] ? ENV['SPEC_OPTS'] : %q{ --format documentation --force-color }
    spec = ENV['SPEC'] || File.join(Dir.getwd, "src/spec/ruby/**/*_spec.rb")
    opts = opts.split(' ').push *FileList[spec].to_a
    ruby = ENV['RUBY'] || 'jruby'
    args = ruby.split(' ') + %w{-Isrc/spec/ruby -rbundler/setup -S rspec}
    sh *args, *opts
  end
end

desc "Run specs"
task :spec => [:test_prepare, :speconly]
task :test => :spec

POM_FILE = 'pom.xml'
VERSION_FILE = 'src/main/ruby/jruby/rack/version.rb'

GEM_VERSION =
  if File.read(VERSION_FILE).match(/VERSION =.*?['"](.+)['"].*?$/m)
    $1
  else
    raise "VERSION = ... not matched in #{VERSION_FILE}"
  end

JAR_VERSION = GEM_VERSION.sub(/\.(\D+\w*)/, '-\1') # 1.1.1.SNAPSHOT -> 1.1.1-SNAPSHOT

file (target_jruby_rack = 'target/gem/lib/jruby-rack.rb') do |t|
  mkdir_p File.dirname(t.name)
  File.open(t.name, "wb") do |f|
    f << %Q{
module JRubyJars
  def self.jruby_rack_jar_path
    File.expand_path("../jruby-rack-#{JAR_VERSION}.jar", __FILE__)
  end
  require jruby_rack_jar_path if defined?(JRUBY_VERSION)
end
}
  end
end
GENERATED << target_jruby_rack

file (target_jar = "target/jruby-rack-#{JAR_VERSION}.jar") do |file|
  Rake::Task['compile'].invoke
  Rake::Task['resources'].invoke
  sh "jar cf #{file.name} -C target/classes ."
end

desc "Create the jruby-rack-#{JAR_VERSION}.jar"
task :jar => target_jar

task :default => :jar

file (target_jruby_rack_version = "target/gem/lib/jruby/rack/version.rb") do |file|
  mkdir_p File.dirname(file.name); cp VERSION_FILE, file.name
end

desc "Build the jruby-rack-#{GEM_VERSION}.gem"
task :gem => [target_jar, target_jruby_rack, target_jruby_rack_version] do
  Rake::Task['spec'].invoke unless ENV['SKIP_SPECS'] == 'true'
  cp FileList["CHANGELOG.md", "LICENSE.txt", "README.md"], "target/gem"
  cp target_jar, "target/gem/lib"
  if (jars = FileList["target/gem/lib/*.jar"].to_a).size > 1
    abort "Too many jars! #{jars.map{|j| File.basename(j)}.inspect}\nRun a clean build `rake clean` first"
  end
  require 'date'
  Dir.chdir("target/gem") do
    rm_f 'jruby-rack.gemspec'
    gemspec = Gem::Specification.new do |gem|
      gem.name = %q{jruby-rack}
      gem.version = GEM_VERSION
      gem.authors = ['Nick Sieger', 'Karol Bucek', 'JRuby contributors']
      gem.date = Date.today.to_s
      gem.license = 'MIT'
      gem.description = %{JRuby-Rack is a combined Java and Ruby library that adapts the Java Servlet API to Rack. For JRuby only.}
      gem.summary = %q{Rack adapter for JRuby and Servlet Containers}
      gem.email = ['nick@nicksieger.com']
      gem.files = FileList["./**/*"].exclude("*.gem").map{ |f| f.sub(/^\.\//, '') }
      gem.homepage = %q{http://jruby.org}
      gem.required_ruby_version = '>= 2.6.0' # JRuby >= 9.3
    end

    require 'rubygems/package'
    Gem::Package.build(gemspec)
    File.open('jruby-rack.gemspec', 'w') { |f| f << gemspec.to_ruby }
    mv FileList['*.gem'], '..'
  end
end
GENERATED << 'target/gem/jruby-rack.gemspec'

task :release_checks do
  sh "git diff --exit-code > /dev/null" do |ok,_|
    fail "There are uncommitted changes.\nPlease commit changes or clean workspace before releasing." unless ok
  end

  sh "git rev-parse #{GEM_VERSION} > /dev/null 2>&1" do |ok,_|
    fail "Tag #{GEM_VERSION} already exists.\n" +
      "Please execute these commands to remove it before releasing:\n" +
      "  git tag -d #{GEM_VERSION}\n" +
      "  git push origin :#{GEM_VERSION}" if ok
  end

  pom_version = `./mvnw help:evaluate -Dexpression=project.version`.
    split("\n").reject { |line| line =~ /[INFO]/ }.first.chomp
  if pom_version =~ /dev|SNAPSHOT/
    fail "Can't release a dev/snapshot version.\n" +
      "Please update pom.xml to the final release version, run `mvn install', and commit the result."
  end

  unless pom_version.sub(/\-(\D+\w*)/, '.\1') == GEM_VERSION
    fail "Can't release because pom.xml version (#{pom_version}) is different than " +
      "jruby/rack/version.rb (#{GEM_VERSION}).\n" +
      "Please run `./mvnw install' to bring the two files in sync."
  end

  puts "release #{GEM_VERSION} looks ready to go ..."
end

desc "Release the gem to rubygems and jar to repository.codehaus.org"
task :release => [:release_checks, :clean] do
  args = ''
  args << "-Dgpg.keyname=#{ENV['GPG_KEYNAME']} " if ENV['GPG_KEYNAME']

  sh "./mvnw -Prelease #{args} -DupdateReleaseInfo=true clean deploy"

  sh "git tag #{GEM_VERSION}"

  ENV['SKIP_SPECS'] = 'true'; Rake::Task['gem'].invoke
  sh "gem push target/jruby-rack-#{GEM_VERSION}.gem"

  git_branch = `git branch | sed -n '/\* /s///p'`.chomp
  sh "git push --tags #{ENV['GIT_REMOTE'] || 'origin'} #{git_branch}"

  puts "released JRuby-Rack #{GEM_VERSION} update next SNAPSHOT version using `rake update_version`"
end

desc "Update version to next (1.2.3 -> 1.2.4.SNAPSHOT) or passed VERSION"
task :update_version do
  version = ENV['VERSION'] || ''
  if version.empty? # next version
    gem_version = Gem::Version.create(GEM_VERSION)
    if gem_version.segments.last.is_a?(String)
      version = gem_version.segments[0...-1] # 1.1.1.SNAPSHOT -> 1.1.1
    else  # 1.1.1 -> 1.1.2.SNAPSHOT
      version = gem_version.segments.dup
      version[-1] = version[-1] + 1
      version = version + ['SNAPSHOT']
    end
    version = version.join('.')
  else
    version.sub!('-', '.') # normalize "maven" style VERSION
  end
  if version != GEM_VERSION
    gem_version = Gem::Version.create(version) # validates VERSION string

    lines = File.readlines(VERSION_FILE) # update JRuby::Rack::VERSION
    lines.each {|l| l.sub!(/VERSION =.*$/, %{VERSION = '#{version}'})}
    File.open(VERSION_FILE, "wb") { |f| f.puts *lines }

    pom_version = if gem_version.prerelease?
      segs = gem_version.segments
      "#{segs[0...-1].join('.')}-#{segs[-1]}"
    else
      gem_version.version
    end
    doc = nil # update pom.xml <version>
    File.open(POM_FILE, 'r') do |file|
      require "rexml/document"
      doc = REXML::Document.new file
      doc.root.elements.each('version') { |el| el.text = pom_version }
    end
    File.open(POM_FILE, 'w') do |file|
      file.puts doc.to_s
    end if doc
  end
end
