puts_debug "read " + __FILE__.foreground(:green) require 'semver' require 'rake' UNITTESTS=Array.new module Dev class Project < Hash def initialize(hash,init_defaults=true) hash = Hash.new if hash.nil? hash = defaults.merge(hash) puts_debug "initialize, copying hash values" hash.each { |name,value| self[name]=value } if !hash.nil? update_default_values if init_defaults end def defaults system_call = Dev::SystemCall.new("semver init") if(!File.exist?(".semver")) h={ :src_glob => "**/{*.{rb,feature,spec,cs,c,cpp,h,hpp,csproj,vcproj,snk,vcxproj,sln,filters,xaml,xml,resx,settings,txt,png,bmp,ico},Jamfile,.semver,Gemfile,README,LICENSE}", :name => get_default_value_name, :group => get_default_value_group, :directory => Rake.original_dir(), :type => get_default_value_type, :version => "#{SemVer.find.major}.#{SemVer.find.minor}.#{SemVer.find.patch}", :dev_root => "#{Dev::Environment.dev_root}" } if !Dev::BoostBuild.defaultToolset.nil? h[:toolset] = Dev::BoostBuild.defaultToolset ENV["VisualStudioVersion"]="11.0" if h[:toolset]=="msvc-11.0" end h[:scm_type] = get_default_scm_type if !get_default_scm_type.nil? tmp=get_default_value_paths h[:paths] = tmp if !tmp.nil? return h end def refresh update_default_values end def update_default_values puts_debug "update_default_values" #newValues=Hash.new #self.each{ |k,v| # newValues[k.to_s]=v if !self.has_key?(k.to_s) #} #self.merge(newValues) if !self.has_key?(:dep) || self[:dep].nil? puts_debug "assigning self[:dep] to Hash.new" self[:dep]=Hash.new end self.delete(:setup) if self.has_key?(:setup) ["add","pull","setup","compile","test","features","commit","update" ].each do |k| puts_debug "set_default_value(#{k})" set_default_value(k) end generate_auto_replace_directives generate_tasks(self) end def get_default_value_name dirs=::Rake.original_dir().split('/') return dirs[dirs.length-2] if(dirs.length > 3) return nil end def get_default_value_group dirs=Rake.original_dir().split('/') return dirs[dirs.length-3] if(dirs.length > 3) return "unknown" end def get_default_value_type Dir.glob("**/*.{cpp,hpp}").each { |f| return "c++" } Dir.glob("**/*.{c,h}").each { |f| return "c" } Dir.glob("**/*.{cs}").each { |f| return "C#" } Dir.glob("**/*.{gemspec}").each { |f| return "gem" } return "ruby" end def get_default_scm_type return "svn" if File.exists?(".svn") return "git" if File.exists?(".git") return nil end def get_default_value_paths hash=Hash.new ["v4.0.30319","v3.5","v3.0","v2.0.50727"].each { |v| f="C:/WINDOWS/Microsoft.NET/Framework/#{v}/MSBuild.exe" hash[:msbuild]=f if File.exists?(f) && hash[:msbuild].nil? f="C:/WINDOWS/Microsoft.NET/Framework/#{v}/RegAsm.exe" hash[:regasm]=f if File.exists?(f) && hash[:regasm].nil? } ["2.5.10.11092","2.6.0.12051"].each{ |v| Dir.glob("#{Dev::Environment.dev_root}/dep/*/NUnit/#{v}/bin/nunit-console-x86.exe").each{|f| hash[:nunit]=f if File.exists?(f) && hash[:nunit].nil? } } hash=nil if hash.length < 1 return hash end def update_keys self.each{ |name,value| self[name]=value } end def array_method(name) string_name=name.to_s puts_debug "method_missing name=" + string_name puts " no directives defined for #{string_name}" if self.get_value(string_name).nil? && string_name=="pull" return if(self.get_value(string_name).nil?) puts " no directives defined for #{string_name}" if self.get_value(string_name).length < 1 return if self.get_value(string_name).length < 1 value=self.get_value(string_name) if value.kind_of?(Hash) value.each { |name,value| execute_cmd(value); sleep(0.5) } else value.each { |c| execute_cmd(c); sleep(0.5) } end end def execute_cmd(c) # expand the command here.... command=expand_command(c) puts_debug "command: " + command.to_s + " (in Project.execute_cmd)" if c.include?('<%') && c.include?('%>') #puts "Command: " + c eval(c.gsub("<%","").gsub("%>","")) else # is the command a hash? hash=c if c.kind_of?(Hash) # can the command be converted to a hash? hash=Dev::Environment.s_to_hash(command) if hash.nil? call=nil if(hash.nil?) call=Dev::SystemCall.new(command) else call=Dev::SystemCall.new(hash) end call.puts_summary end end def method_missing( name, *args ); array_method(name); end def expand_command(command); expand_project_variables(command); end def expand_project_variables(str) if str.kind_of?(Hash) expandedHash = Hash.new str.each do |name,value| expandedHash[name]=expand_project_variables(value) end return expandedHash end if str.kind_of?(Array) expandedArray= Array.new str.each{|e|expandedArray << expand_project_variable(e) } return expandedArray end if str.kind_of?(String) result=Dev::Environment.expand_string_variables(str) # expands ruby variables, e.g. '#{name}' to 'widget' if result.include?('<') && result.include?('>') result.scan(/(<[\w,]+>)/).each { |pvar| # expand project variables, e.g. '' to 'widget key = pvar[0].gsub("<","").gsub(">","") # '' to 'C:/wherever/NUnit.exe' result = result.gsub(pvar[0],get_value(key)) } end return result end return str end def loc_cmd cmd="countloc --recurse ." cmd="countloc --recurse --mode ruby ." if self[:type]=="ruby" || self[:type]=="gem" cmd="countloc --recurse --mode csharp ." if self[:type]=="C#" cmd="countloc --recurse --mode cpp ." if self[:type]=="c++" return cmd end def loc system(loc_cmd) end def add unless self[:src_glob].nil? scm = Dev::Scm.new return if scm.scm_type == "none" if self[:src_glob].kind_of?(Hash) self[:src_glob].each do |name,value| puts " adding files " + value.to_s scm.add_file_glob(value) end else puts " adding files " + self[:src_glob].to_s scm.add_file_glob(self[:src_glob]) end end end def loc_total # parse the output for TOTAL LOC call=Dev::SystemCall.new(loc_cmd) words=Array.new words=call.output.split if !call.nil? && !call.output.nil? if(words.length>6) return words[words.length-6] end "?" end def context Dev::Environment.user + "." + Dev::Environment.machine end def check puts " default.taskstamp." + context + " exists." if File.exists?("default.taskstamp."+context) puts " default.taskstamp." + context + " does not exist" unless File.exists?("default.taskstamp."+context) begin hasdiff = has_diff rescue puts "has_diff threw an exception." end puts " no differences detected." unless hasdiff puts " detected differences." if hasdiff if File.exists?("default.taskstamp."+context) && !hasdiff #puts " nothing to do" update exit end #if !has_diff and File.exists?("default.taskstamp."+context) # puts " no differences detected." # puts " default.taskstamp." + context + " exists." # exit #end #puts " detected differences" if has_diff #puts " default.taskstamp." + context + " does not exist" unless File.exists?("default.taskstamp." + context) end def stamp_task(name) File.open("#{name}.taskstamp." + context,"w") { |f| f.write(Time.now.to_s) } end def generate_auto_replace_directives if(Dir.glob("**/*.vcxproj").count > 0 && !self[:dep].nil? && !self[:dep].empty?) self[:replace]=Hash.new if self[:replace].nil? self[:dep].each{ |k,v| if self[:dep][k].has_key?(:rev) && self[:dep][k].has_key?(:dir) && !self[:replace].has_key?(k) group=self[:dep][k][:dir].split('/')[0] searchRegex=Regexp.new "#{group}\\\\#{k}\\\\[\\d]+" self[:replace][k]={ glob: '**/*.vcxproj', search: searchRegex, replace: "#{group}\\#{k}\\#{v[:rev]}" } end } end end def replace generate_auto_replace_directives #puts " " #puts "replace".foreground(:yellow).bright hash=self[:replace] if hash.nil? puts " no replace directives" end unless hash.nil? hash.each do |k,v| unless v.nil? file=v[:file] search=v[:search] replace=v[:replace] glob=v[:glob] unless search.nil? || replace.nil? unless file.nil? Dev::Environment.replace_text_in_file(file,search,replace) end unless glob.nil? Dev::Environment.replace_text_in_glob(glob,search,replace) end end end end end end def features if self[:type]=="C#" && File.exists?("bin/x86/Release/#{self[:name]}.Features.dll") call=Dev::SystemCall.new(expand_project_variables(" /nologo bin/x86/Release/#{self[:name]}.Features.dll")) puts call.command puts call.output raise "exit_code=#{call.status.to_s}" + call.error unless call.status==0 end array_method("features") end def has_build_products end def has_diff call=nil if File.exists?(".git") call=Dev::SystemCall.new('git status') return true if call.output.include?("new file:") return true if call.output.include?("deleted:") return true if call.output.include?("modified:") end call=Dev::SystemCall.new('git diff --name-only') if File.exists?(".git") call=Dev::SystemCall.new('svn diff') if File.exists?(".svn") unless call.nil? || call.output.length==0 puts_debug call.output return true # differences detected else return false # no differences end end def update array_method("update") end def commit scm = Dev::Scm.new return if scm.scm_type == "none" puts " no differences detected" unless has_diff array_method("commit") if has_diff if File.exists?(".svn") call=Dev::SystemCall.new('svn update') call=Dev::SystemCall.new('svn info') url = call.output.match(/URL: ([\d\w\.\:\/-]+)/)[1] rev = call.output.match(/Last Changed Rev: ([\d]+)/)[1] puts " #{url}@#{rev}" end end def rake_working_deps unless dep_hash.nil? dep_hash.each do |key,value| if value.kind_of?(Hash) # potential hash keys: uri, dir, rake # uri is required unless value.get_value("uri").nil? uri=expand_project_variables(value.get_value("uri")) uri_words=uri.split('/') while(uri_words.length > 3) do uri_words.shift end dir=self[:dev_root] + "/wrk/" + uri_words.join('/') unless value.get_value("rake").nil? if(File.exist?(dir)) rake_task=value.get_value("rake") cmd_hash = {:cmd=> "rake #{rake_task}", :dir=> "#{dir}"} end end end end # unless value.get_value("uri").nil? end end # unless dep_hash.nil? end def pull update unless has_diff array_method("pull") filename=Dir.pwd + "/rakefile.rb" puts " updating revision variables for " + filename Dev::Svn::update_revision_variables(filename) end def info #[ "file_count","loc" ].each do |k| # puts_debug "set_default_value(#{k})" # set_default_value(k) # end puts " " Hash.print_hash("",self) puts " " end def set_default_value(key) set_value(key,get_default_value(key)) if get_value(key).nil? && !get_default_value(key).nil? end def get_default_value(key) value=loc_total if key=="loc" array=Array.new if(key=="setup") puts_debug "assigning default values for setup" #array << "bundle install" if File.exists?("Gemfile") array << "{ :cmd=> 'bundle install', :capture_output=> false}" if File.exists?("Gemfile") && "#{RUBY_VERSION}">="1.8.7" # dep,svn directives puts_debug "checking for :dep hash...." dep_hash=self.get_value("dep") puts_debug "dep_hash is nil" if dep_hash.nil? unless dep_hash.nil? puts_debug "dep_hash is not nil" puts_debug "dep_hash is empty" if dep_hash.empty? dep_hash.each do |key,value| puts_debug ":dep[#{key}.to_s]" if value.kind_of?(Hash) dep=Dev::Dep.new(value) dep.setup_commands.each{|cmd| array << cmd} end end end if(!Dir.glob("*.csproj").nil? && Dir.glob("*.csproj").length > 0) array << "<%Dir.mkdir 'bin' unless File.exist?('bin')%>" array << "<%Dir.mkdir 'bin/Debug' unless File.exist?('bin/Debug')%>" array << "<%Dir.mkdir 'bin/Release' unless File.exist?('bin/Release')%>" end end if(key=="pull") # dep,svn directives dep_hash=self.get_value("dep") unless dep_hash.nil? dep_hash.each do |key,value| if value.kind_of?(Hash) dep=Dev::Dep.new(value) dep.pull_commands.each{|cmd| array << cmd} #puts_debug "pull array: " + array.to_s end end end end if(key=="compile") Dir.glob("*.gemspec").each { |gs| #array << "gem build #{gs}" array << "{ :cmd=> 'gem build #{gs}', :capture_output=> false}" } if self[:type]=="gem" Dir.glob("*.csproj").each{|cs| platforms = Dev::Project::extract_platforms(cs) platforms = ["AnyCPU"] if platforms.nil? puts_debug "for project " + cs + " platforms = " + platforms.to_s platforms.each{ |platform| if(platform=="AnyCPU") if(!has_key?(:any_release_compile_flags)) self[:any_release_compile_flags]="/property:Configuration=Release /property:Platform=\"Any CPU\" /p:OutputPath=./bin/Release" end array<<" #{cs} " if RUBY_PLATFORM.include?("w32") array<<"xbuild #{cs} " if !RUBY_PLATFORM.include?("w32") end if(platform=="x86") if(!has_key?(:x86_release_compile_flags)) self[:x86_release_compile_flags]="/property:Configuration=Release /property:Platform=\"x86\" /p:OutputPath=./bin/x86/Release" end array<<" #{cs} " if RUBY_PLATFORM.include?("w32") array<<"xbuild #{cs} " if !RUBY_PLATFORM.include?("w32") end if(platform=="x64") if(!has_key?(:x64_release_compile_flags)) self[:x64_release_compile_flags]="/property:Configuration=Release /property:Platform=\"x64\" /p:OutputPath=./bin/x64/Release" end array<<" #{cs} " if RUBY_PLATFORM.include?("w32") #array<<"xbuild #{cs} " if !RUBY_PLATFORM.include?("w32") end } } if self[:type]=="c++" if(!has_key?(:win32_release_compile_flags)) self[:win32_release_compile_flags]="/property:Configuration=Release /property:Platform=\"win32\"" end Dir.glob("*.sln").each{|sln| slntext=File.read(sln) if(slntext.include?("Format Version 10.00")) array<<" #{sln} " else array<<" #{sln} " end } end end if(key=="test") Dir.glob("*.gemspec").each { |gs| #array << "{cmd: 'gem install ./#{gs.gsub('.gemspec','')}-#{self[:version]}.gem', capture_output: false}" array << "{:cmd=> 'gem install #{gs.gsub('.gemspec','')}-#{self[:version]}.gem', :capture_output=> false}" #array << "gem install ./#{gs.gsub('.gemspec','')}-#{self[:version]}.gem" } if self[:type]=="gem" if self[:type]=="C#" Dir.glob("*.{Test.csproj,Features.csproj}").each { |cs| dll_name="bin/x86/Release/#{File.basename(cs,'.csproj')}.dll" array << "\"\" /nologo #{dll_name} /xml:#{dll_name}.nunit-results.xml" if RUBY_PLATFORM.include?("w32") array << "mono \"\" #{dll_name}" if !RUBY_PLATFORM.include?("w32") } end end if(key=="features") array << "cucumber features" if File.exists?("features") && Dir.glob("features/**/*.rb").length > 0 end if(key=="has_diff") array << 'git diff' if File.exists?(".git") array << "{ :cmd=> 'svn diff', :capture_output=> false}" if File.exists?(".svn") end if(key=="commit") array << 'git commit -a -m "rake commit all"' if File.exists?(".git") array << 'svn commit -m "rake commit all"' if File.exists?(".svn") end if(key=="update") array << 'svn update' if File.exists?(".svn") end value=array if array.length > 0 return value end def self.extract_platforms(filename) results = Array::new text=File.read(filename) text.scan(/(\w+)<\/Platform/).each{ | var_match | results << var_match[0].to_s if !results.include?(var_match[0].to_s)} results end end end # module Dev PROJECT=Dev::Project.new(nil,true) PROJECT.refresh