# rubocop:disable Metrics/ModuleLength $LOAD_PATH.unshift(File.dirname(__FILE__)) require 'annotate/version' require 'annotate/annotate_models' require 'annotate/annotate_routes' require 'annotate/constants' begin # ActiveSupport 3.x... require 'active_support/hash_with_indifferent_access' require 'active_support/core_ext/object/blank' rescue StandardError # ActiveSupport 2.x... require 'active_support/core_ext/hash/indifferent_access' require 'active_support/core_ext/blank' end module Annotate ## # The set of available options to customize the behavior of Annotate. # POSITION_OPTIONS = [ :position_in_routes, :position_in_class, :position_in_test, :position_in_fixture, :position_in_factory, :position, :position_in_serializer ].freeze FLAG_OPTIONS = [ :show_indexes, :simple_indexes, :include_version, :exclude_tests, :exclude_fixtures, :exclude_factories, :ignore_model_sub_dir, :format_bare, :format_rdoc, :format_markdown, :sort, :force, :frozen, :trace, :timestamp, :exclude_serializers, :classified_sort, :show_foreign_keys, :show_complete_foreign_keys, :exclude_scaffolds, :exclude_controllers, :exclude_helpers, :exclude_sti_subclasses, :ignore_unknown_models, :with_comment ].freeze OTHER_OPTIONS = [ :additional_file_patterns, :ignore_columns, :skip_on_db_migrate, :wrapper_open, :wrapper_close, :wrapper, :routes, :models, :hide_limit_column_types, :hide_default_column_types, :ignore_routes, :active_admin ].freeze PATH_OPTIONS = [ :require, :model_dir, :root_dir ].freeze def self.all_options [POSITION_OPTIONS, FLAG_OPTIONS, PATH_OPTIONS, OTHER_OPTIONS] end ## # Set default values that can be overridden via environment variables. # def self.set_defaults(options = {}) return if @has_set_defaults @has_set_defaults = true options = HashWithIndifferentAccess.new(options) all_options.flatten.each do |key| if options.key?(key) default_value = if options[key].is_a?(Array) options[key].join(',') else options[key] end end default_value = ENV[key.to_s] unless ENV[key.to_s].blank? ENV[key.to_s] = default_value.nil? ? nil : default_value.to_s end end ## # TODO: what is the difference between this and set_defaults? # def self.setup_options(options = {}) POSITION_OPTIONS.each do |key| options[key] = fallback(ENV[key.to_s], ENV['position'], 'before') end FLAG_OPTIONS.each do |key| options[key] = true?(ENV[key.to_s]) end OTHER_OPTIONS.each do |key| options[key] = !ENV[key.to_s].blank? ? ENV[key.to_s] : nil end PATH_OPTIONS.each do |key| options[key] = !ENV[key.to_s].blank? ? ENV[key.to_s].split(',') : [] end options[:additional_file_patterns] ||= [] options[:model_dir] = ['app/models'] if options[:model_dir].empty? options[:wrapper_open] ||= options[:wrapper] options[:wrapper_close] ||= options[:wrapper] # These were added in 2.7.0 but so this is to revert to old behavior by default options[:exclude_scaffolds] = Annotate.true?(ENV.fetch('exclude_scaffolds', 'true')) options[:exclude_controllers] = Annotate.true?(ENV.fetch('exclude_controllers', 'true')) options[:exclude_helpers] = Annotate.true?(ENV.fetch('exclude_helpers', 'true')) options end def self.reset_options all_options.flatten.each { |key| ENV[key.to_s] = nil } end def self.skip_on_migration? ENV['ANNOTATE_SKIP_ON_DB_MIGRATE'] =~ Constants::TRUE_RE || ENV['skip_on_db_migrate'] =~ Constants::TRUE_RE end def self.include_routes? ENV['routes'] =~ Constants::TRUE_RE end def self.include_models? ENV['models'] =~ Constants::TRUE_RE end def self.loaded_tasks=(val) @loaded_tasks = val end def self.loaded_tasks @loaded_tasks end def self.load_tasks return if loaded_tasks self.loaded_tasks = true Dir[File.join(File.dirname(__FILE__), 'tasks', '**/*.rake')].each do |rake| load rake end end def self.load_requires(options) options[:require].count > 0 && options[:require].each { |path| require path } end def self.eager_load(options) load_requires(options) require 'annotate/active_record_patch' if defined?(Rails::Application) if Rails.version.split('.').first.to_i < 3 Rails.configuration.eager_load_paths.each do |load_path| matcher = /\A#{Regexp.escape(load_path)}(.*)\.rb\Z/ Dir.glob("#{load_path}/**/*.rb").sort.each do |file| require_dependency file.sub(matcher, '\1') end end else klass = Rails::Application.send(:subclasses).first klass.eager_load! end else options[:model_dir].each do |dir| FileList["#{dir}/**/*.rb"].each do |fname| require File.expand_path(fname) end end end end def self.bootstrap_rake begin require 'rake/dsl_definition' rescue StandardError => e # We might just be on an old version of Rake... $stderr.puts e.message exit e.status_code end require 'rake' load './Rakefile' if File.exist?('./Rakefile') begin Rake::Task[:environment].invoke rescue nil end unless defined?(Rails) # Not in a Rails project, so time to load up the parts of # ActiveSupport we need. require 'active_support' require 'active_support/core_ext/class/subclasses' require 'active_support/core_ext/string/inflections' end load_tasks Rake::Task[:set_annotation_options].invoke end def self.fallback(*args) args.detect { |arg| !arg.blank? } end def self.true?(val) return false if val.blank? return false unless val =~ Constants::TRUE_RE true end end