У меня есть грабли задача, которая должна вставить значение в несколько баз данных.
Я хотел бы передать это значение в задачу rake из командной строки или из другой задачи rake.
Как я могу это сделать?
У меня есть грабли задача, которая должна вставить значение в несколько баз данных.
Я хотел бы передать это значение в задачу rake из командной строки или из другой задачи rake.
Как я могу это сделать?
Ответы:
параметры и зависимости должны быть внутри массивов:
namespace :thing do
desc "it does a thing"
task :work, [:option, :foo, :bar] do |task, args|
puts "work", args
end
task :another, [:option, :foo, :bar] do |task, args|
puts "another #{args}"
Rake::Task["thing:work"].invoke(args[:option], args[:foo], args[:bar])
# or splat the args
# Rake::Task["thing:work"].invoke(*args)
end
end
затем
rake thing:work[1,2,3]
=> work: {:option=>"1", :foo=>"2", :bar=>"3"}
rake thing:another[1,2,3]
=> another {:option=>"1", :foo=>"2", :bar=>"3"}
=> work: {:option=>"1", :foo=>"2", :bar=>"3"}
ПРИМЕЧАНИЕ: переменная
task
является объектом задачи, она не очень полезна, если вы не знаете / не заботитесь о внутренностях Rake.
Рельсы ПРИМЕЧАНИЕ:
Если задача запускается с рельсов, лучше всего предварительно загрузить среду, добавив
=> [:environment]
способ, позволяющий настроить зависимые задачи.
task :work, [:option, :foo, :bar] => [:environment] do |task, args|
puts "work", args
end
rake thing:work[1, 2, 3]
как это не сработает, и вы получите ошибкуDon't know how to build task
rake thing:work'[1,2,3]'
rake thing:work\[1,2,3\]
или thisrake 'thing:work[1,2,3]'
:environment
символ из вашей задачи. У приложений rails есть: задача среды ...
t
средства task
, почему бы не просто использовать в task
качестве имени пар?
Вы можете указать формальные аргументы в rake, добавив символьные аргументы к вызову задачи. Например:
require 'rake'
task :my_task, [:arg1, :arg2] do |t, args|
puts "Args were: #{args} of class #{args.class}"
puts "arg1 was: '#{args[:arg1]}' of class #{args[:arg1].class}"
puts "arg2 was: '#{args[:arg2]}' of class #{args[:arg2].class}"
end
task :invoke_my_task do
Rake.application.invoke_task("my_task[1, 2]")
end
# or if you prefer this syntax...
task :invoke_my_task_2 do
Rake::Task[:my_task].invoke(3, 4)
end
# a task with prerequisites passes its
# arguments to it prerequisites
task :with_prerequisite, [:arg1, :arg2] => :my_task #<- name of prerequisite task
# to specify default values,
# we take advantage of args being a Rake::TaskArguments object
task :with_defaults, :arg1, :arg2 do |t, args|
args.with_defaults(:arg1 => :default_1, :arg2 => :default_2)
puts "Args with defaults were: #{args}"
end
Затем из командной строки:
> rake my_task [1, false] Аргументы были: {: arg1 => "1",: arg2 => "false"} класса Rake :: TaskArguments arg1 было: '1' класса String arg2 был: 'false' класса String > грабли "my_task [1, 2]" Аргументы были: {: arg1 => "1",: arg2 => "2"} > грабли invoke_my_task Аргументы были: {: arg1 => "1",: arg2 => "2"} > грабли invoke_my_task_2 Аргументы были: {: arg1 => 3,: arg2 => 4} > грабли с_предшеством [5,6] Аргументы были: {: arg1 => "5",: arg2 => "6"} > грабли с_дефолтами Аргументы со значениями по умолчанию: {: arg1 =>: default_1,: arg2 =>: default_2} > грабли with_defaults ['x', 'y'] Аргументы со значениями по умолчанию были: {: arg1 => "x",: arg2 => "y"}
Как показано во втором примере, если вы хотите использовать пробелы, кавычки вокруг целевого имени необходимы, чтобы оболочка не разделяла аргументы в пробеле.
Глядя на код в rake.rb , кажется, что rake не анализирует строки задач для извлечения аргументов для предварительных условий, поэтому вы не можете это сделать task :t1 => "dep[1,2]"
. Единственный способ указать разные аргументы для предварительного условия - это явно вызвать его в зависимом действии задачи, как в :invoke_my_task
и :invoke_my_task_2
.
Обратите внимание, что некоторые оболочки (например, zsh) требуют, чтобы вы выходили за скобки: rake my_task\['arg1'\]
WARNING: 'task :t, arg, :needs => [deps]' is deprecated. Please use 'task :t, [args] => [deps]' instead.
zsh: no matches found: ...
), так что вам нужно , чтобы избежать скобки: rake my_task\['arg1'\]
. От robots.thoughtbot.com/post/18129303042/…
В дополнение к ответу по kch (я не нашел, как оставить комментарий к этому, извините):
Вам не нужно указывать переменные как ENV
переменные перед rake
командой. Вы можете просто установить их как обычные параметры командной строки:
rake mytask var=foo
и получить доступ к ним из вашего файла rake как переменные ENV, например:
p ENV['var'] # => "foo"
p
означает?
Если вы хотите передать именованные аргументы (например, со стандартными OptionParser
), вы можете использовать что-то вроде этого:
$ rake user:create -- --user test@example.com --pass 123
обратите внимание --
, что это необходимо для обхода стандартных аргументов Rake. Должно работать с Rake 0.9.x , <= 10.3.x .
Более новый Rake изменил свой синтаксический анализ --
, и теперь вы должны убедиться, что он не передан OptionParser#parse
методу, например сparser.parse!(ARGV[2..-1])
require 'rake'
require 'optparse'
# Rake task for creating an account
namespace :user do |args|
desc 'Creates user account with given credentials: rake user:create'
# environment is required to have access to Rails models
task :create do
options = {}
OptionParser.new(args) do |opts|
opts.banner = "Usage: rake user:create [options]"
opts.on("-u", "--user {username}","User's email address", String) do |user|
options[:user] = user
end
opts.on("-p", "--pass {password}","User's password", String) do |pass|
options[:pass] = pass
end
end.parse!
puts "creating user account..."
u = Hash.new
u[:email] = options[:user]
u[:password] = options[:pass]
# with some DB layer like ActiveRecord:
# user = User.new(u); user.save!
puts "user: " + u.to_s
puts "account created."
exit 0
end
end
exit
в конце убедитесь, что дополнительные аргументы не будут интерпретироваться как задача Rake.
Также должен работать ярлык для аргументов:
rake user:create -- -u test@example.com -p 123
Когда rake-скрипты выглядят так, возможно, пришло время искать другой инструмент, который позволил бы это просто из коробки.
--option-names
. Мое единственное предложение было бы использовать , exit
а не , abort
как abort
оставит вас с кодом возврата 1 к оболочке. Если задача rake является частью скрипта более высокого уровня, более распространенным является предположение, что ненулевой выход - это ошибка определенного типа.
--
? Как передача rake
аргументов в реальную задачу или что-то еще? Нравится task :my_task, :*args do |t, args|
или что?
{username}
здесь. Где это используется? Почему его там нет -u {username}
? Приветствия
10.4.1
и изменен 10.4.2
. github.com/ruby/rake/commit/…
Я нашел ответ на этих двух сайтах: Net Maniac и Aimred .
Вы должны иметь версию> 0.8 рейка, чтобы использовать эту технику
Обычное описание задачи с граблями таково:
desc 'Task Description'
task :task_name => [:depends_on_taskA, :depends_on_taskB] do
#interesting things
end
Чтобы передать аргументы, сделайте три вещи:
Чтобы получить доступ к аргументам в сценарии, используйте args.arg_name
desc 'Takes arguments task'
task :task_name, :display_value, :display_times, :needs => [:depends_on_taskA, :depends_on_taskB] do |t, args|
args.display_times.to_i.times do
puts args.display_value
end
end
Чтобы вызвать эту задачу из командной строки, передайте ей аргументы в [] s
rake task_name['Hello',4]
будет выводить
Hello
Hello
Hello
Hello
и если вы хотите вызвать эту задачу из другой задачи и передать ей аргументы, используйте invoke
task :caller do
puts 'In Caller'
Rake::Task[:task_name].invoke('hi',2)
end
тогда команда
rake caller
будет выводить
In Caller
hi
hi
Я не нашел способ передать аргументы как часть зависимости, так как следующий код ломается:
task :caller => :task_name['hi',2]' do
puts 'In Caller'
end
'task :t, arg, :needs => [deps]' is deprecated. Please use 'task :t, [args] => [deps]' instead.
Другой часто используемый параметр - передавать переменные окружения. В вашем коде вы читаете их через ENV['VAR']
, и можете передать их прямо перед rake
командой, как
$ VAR=foo rake mytask
rake blah -- --these --go --to --a-program
(обратите внимание, --
чтобы сообщить рейку, что его переключатели закончились), см. stackoverflow.com/questions/5086224/…
Я не мог понять, как передавать аргументы, а также окружение, пока не решил это:
namespace :db do
desc 'Export product data'
task :export, [:file_token, :file_path] => :environment do |t, args|
args.with_defaults(:file_token => "products", :file_path => "./lib/data/")
#do stuff [...]
end
end
И тогда я звоню так:
rake db:export['foo, /tmp/']
desc 'an updated version'
task :task_name, [:arg1, :arg2] => [:dependency1, :dependency2] do |t, args|
puts args[:arg1]
end
rake task_name[hello, world]
Я просто хотел иметь возможность бежать:
$ rake some:task arg1 arg2
Просто, правда? (Нет!)
Рейк интерпретирует arg1
и arg2
как задачи, и пытается их запустить. Таким образом, мы просто прерываем, прежде чем это произойдет.
namespace :some do
task task: :environment do
arg1, arg2 = ARGV
# your task...
exit
end
end
Возьми это, скобки!
Отказ от ответственности : я хотел быть в состоянии сделать это в довольно маленьком домашнем проекте. Не предназначен для использования в "реальном мире", так как вы теряете возможность связывать рейковые задачи (т.е. rake task1 task2 task3
). ИМО не стоит. Просто используйте уродливый rake task[arg1,arg2]
.
_, arg1, arg2 = ARGV
как первый аргумент был назван именем рейка. Но exit
это ловкий трюк.
rake task[arg1,arg2] && rake task2 && rake task3
Не уверен, что это менее уродливо, чем rake task[arg1,arg2] task2 task3
. Вероятно, менее эффективно, хотя.
_, *args = ARGV
идеально подходит для захвата всех последующих аргументов! Спасибо, куча!
Я использую обычный аргумент ruby в файле rake:
DB = ARGV[1]
затем я заглушаю задачи rake внизу файла (так как rake будет искать задачу по имени этого аргумента).
task :database_name1
task :database_name2
командная строка:
rake mytask db_name
это кажется мне чище, чем var = foo ENV var и args [blah, blah2].
заглушка немного дрянная, но не слишком плохая, если у вас есть несколько сред, которые являются одноразовыми
dup
в конце: db = ARGV [1] .dup
db = ARGV[1].dup unless ARGV[1].nil?
чтобы исключить исключение дублирования нуля.
Способы передачи аргумента верны в ответе выше. Однако, чтобы выполнить задачу rake с аргументами, в более новой версии rails есть небольшая техническая составляющая.
Он будет работать с граблями "namespace: taskname ['arguments1']"
Обратите внимание на кавычки при запуске задачи из командной строки.
Чтобы передать аргументы задаче по умолчанию, вы можете сделать что-то вроде этого. Например, скажите «версия» - ваш аргумент:
task :default, [:version] => [:build]
task :build, :version do |t,args|
version = args[:version]
puts version ? "version is #{version}" : "no version passed"
end
Тогда вы можете назвать это так:
$ rake
no version passed
или
$ rake default[3.2.1]
version is 3.2.1
или
$ rake build[3.2.1]
version is 3.2.1
Однако я не нашел способа избежать указания имени задачи (по умолчанию или сборка) при передаче аргументов. Хотелось бы услышать, если кто-нибудь знает способ.
Мне нравится синтаксис querystring для передачи аргументов, особенно когда нужно передать много аргументов.
Пример:
rake "mytask[width=10&height=20]"
«Строка запроса»:
width=10&height=20
Предупреждение: обратите внимание, что синтаксис есть rake "mytask[foo=bar]"
и НЕ rake mytask["foo=bar"]
При анализе внутри задачи rake Rack::Utils.parse_nested_query
, мы получаем Hash
:
=> {"width"=>"10", "height"=>"20"}
(Самое интересное, что вы можете передавать хэши и массивы, подробнее ниже)
Вот как это сделать:
require 'rack/utils'
task :mytask, :args_expr do |t,args|
args.with_defaults(:args_expr => "width=10&height=10")
options = Rack::Utils.parse_nested_query(args[:args_expr])
end
Вот более расширенный пример, который я использую с Rails в своем геме delayed_job_active_record_threaded :
bundle exec rake "dj:start[ebooks[workers_number]=16&ebooks[worker_timeout]=60&albums[workers_number]=32&albums[worker_timeout]=120]"
Анализируется так же, как и выше, с зависимостью от среды (для загрузки среды Rails)
namespace :dj do
task :start, [ :args_expr ] => :environment do |t, args|
# defaults here...
options = Rack::Utils.parse_nested_query(args[:args_expr])
end
end
Дает следующее в options
=> {"ebooks"=>{"workers_number"=>"16", "worker_timeout"=>"60"}, "albums"=>{"workers_number"=>"32", "worker_timeout"=>"120"}}
Если вы не можете вспомнить, какая позиция аргумента для чего, и вы хотите сделать что-то вроде хэга для аргумента ruby. Вы можете использовать один аргумент, чтобы передать строку, а затем пересмотреть эту строку в хэш опций.
namespace :dummy_data do
desc "Tests options hash like arguments"
task :test, [:options] => :environment do |t, args|
arg_options = args[:options] || '' # nil catch incase no options are provided
two_d_array = arg_options.scan(/\W*(\w*): (\w*)\W*/)
puts two_d_array.to_s + ' # options are regexed into a 2d array'
string_key_hash = two_d_array.to_h
puts string_key_hash.to_s + ' # options are in a hash with keys as strings'
options = two_d_array.map {|p| [p[0].to_sym, p[1]]}.to_h
puts options.to_s + ' # options are in a hash with symbols'
default_options = {users: '50', friends: '25', colour: 'red', name: 'tom'}
options = default_options.merge(options)
puts options.to_s + ' # default option values are merged into options'
end
end
И в командной строке вы получите.
$ rake dummy_data:test["users: 100 friends: 50 colour: red"]
[["users", "100"], ["friends", "50"], ["colour", "red"]] # options are regexed into a 2d array
{"users"=>"100", "friends"=>"50", "colour"=>"red"} # options are in a hash with keys as strings
{:users=>"100", :friends=>"50", :colour=>"red"} # options are in a hash with symbols
{:users=>"100", :friends=>"50", :colour=>"red", :name=>"tom"} # default option values are merged into options
Большинство методов, описанных выше, не работают для меня, возможно, они устарели в более новых версиях. Актуальное руководство можно найти здесь: http://guides.rubyonrails.org/command_line.html#custom-rake-tasks
Копирование и вставка из руководства здесь:
task :task_name, [:arg_1] => [:pre_1, :pre_2] do |t, args|
# You can use args from here
end
Вызвать это так
bin/rake "task_name[value 1]" # entire argument string should be quoted
Для запуска граблей с традиционным стилем аргументов:
rake task arg1 arg2
А затем используйте:
task :task do |_, args|
puts "This is argument 1: #{args.first}"
end
Добавьте следующий патч Rake Gem:
Rake::Application.class_eval do
alias origin_top_level top_level
def top_level
@top_level_tasks = [top_level_tasks.join(' ')]
origin_top_level
end
def parse_task_string(string) # :nodoc:
parts = string.split ' '
return parts.shift, parts
end
end
Rake::Task.class_eval do
def invoke(*args)
invoke_with_call_chain(args, Rake::InvocationChain::EMPTY)
end
end
При передаче параметров лучше выбрать входной файл, это может быть файл json или что-то еще, что вам нужно, и оттуда прочитайте необходимую вам структуру данных и переменные, включая имя переменной, если это необходимо. Для чтения файла может иметь следующую структуру.
namespace :name_sapace_task do
desc "Description task...."
task :name_task => :environment do
data = ActiveSupport::JSON.decode(File.read(Rails.root+"public/file.json")) if defined?(data)
# and work whit yoour data, example is data["user_id"]
end
end
{
"name_task": "I'm a task",
"user_id": 389,
"users_assigned": [389,672,524],
"task_id": 3
}
rake :name_task