Class: Promiscuous::CLI

Inherits:
Object
  • Object
show all
Defined in:
lib/promiscuous/cli.rb

Instance Method Summary collapse

Instance Method Details

#force_backend(backend) ⇒ Object



16
17
18
19
20
# File 'lib/promiscuous/cli.rb', line 16

def force_backend(backend)
  Promiscuous::AMQP.disconnect
  Promiscuous::Config.backend = backend
  Promiscuous::AMQP.connect
end

#load_app(options = {}) ⇒ Object



141
142
143
144
145
146
147
148
149
# File 'lib/promiscuous/cli.rb', line 141

def load_app(options={})
  if options[:require]
    require options[:require]
  else
    require 'rails'
    require File.expand_path("./config/environment.rb")
    ::Rails.application.eager_load!
  end
end

#maybe_warn_bareback(options) ⇒ Object



160
161
162
163
164
165
166
# File 'lib/promiscuous/cli.rb', line 160

def maybe_warn_bareback(options)
  if options[:bareback]
    print_status "WARNING: --- BAREBACK MODE ----"
    print_status "WARNING: You are replicating without protection, you can get corrupted in no time"
    print_status "WARNING: --- BAREBACK MODE ----"
  end
end

#parse_args(args) ⇒ Object



84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
# File 'lib/promiscuous/cli.rb', line 84

def parse_args(args)
  options = {}

  require 'optparse'
  parser = OptionParser.new do |opts|
    opts.banner = "Usage: promiscuous [options] action"

    opts.separator ""
    opts.separator "Actions:"
    opts.separator "    publish"
    opts.separator "    subscribe"
    opts.separator ""
    opts.separator "Options:"

    opts.on "-s", "--sync", "Use a separate queue for sychronizing databases" do
      options[:personality] = :sync
    end

    opts.on "-c", "--criteria CRITERIA", "Published criteria in sync mode. e.g. Member.where(:created_at.gt => 1.day.ago)" do |criteria|
      options[:criteria] = criteria
    end

    opts.on "-b", "--bareback", "Bareback mode aka continue on error. Use with extreme caution" do
      options[:bareback] = true
    end

    opts.on "-r", "--require FILE", "File to require to load your app. Don't worry about it with rails" do |file|
      options[:require] = file
    end

    opts.on("-h", "--help", "Show this message") do
      puts opts
      exit
    end

    opts.on("-V", "--version", "Show version") do
      puts "Promiscuous #{Promiscuous::VERSION}"
      puts "License MIT"
      exit
    end
  end

  args = args.dup
  parser.parse!(args)

  options[:action] = args.shift.try(:to_sym)
  raise "Please specify an action (publish or subscribe)" unless options[:action].in? [:publish, :subscribe]

  if options[:action] == :publish && options[:personality] == :sync
    raise "Please specify a criteria" unless options[:criteria]
  else
    raise "Why are you specifying a criteria?" if options[:criteria]
  end

  options
end


168
169
170
171
# File 'lib/promiscuous/cli.rb', line 168

def print_status(msg)
  Promiscuous.info msg
  $stderr.puts msg
end

#publish(options = {}) ⇒ Object



44
45
46
47
48
49
# File 'lib/promiscuous/cli.rb', line 44

def publish(options={})
  replicate do
    Promiscuous::Worker.replicate(options)
    print_status "Replicating with #{Promiscuous::Publisher::Mongoid::Defer.klasses.count} publishers"
  end
end

#publish_sync(options = {}) ⇒ Object



58
59
60
61
62
63
64
65
66
67
68
69
# File 'lib/promiscuous/cli.rb', line 58

def publish_sync(options={})
  print_status "Replicating #{options[:criteria]}..."
  criteria = eval(options[:criteria])

  bar = ProgressBar.create(:format => '%t |%b>%i| %c/%C %e', :title => 'Publishing', :total => criteria.count)
  criteria.each do |doc|
    doc.promiscuous_sync(options)
    bar.increment
  end

  print_status "Done. You may switch your subscriber worker back to regular mode, and delete the sync queues"
end

#replicate(config_options = {}, &block) ⇒ Object



4
5
6
7
8
9
10
11
12
13
14
# File 'lib/promiscuous/cli.rb', line 4

def replicate(config_options={}, &block)
  require 'eventmachine'
  require 'em-synchrony'

  EM.synchrony do
    trap_signals
    Promiscuous::Loader.load_descriptors if defined?(Rails)
    force_backend :rubyamqp
    block.call
  end
end

#runObject



151
152
153
154
155
156
157
158
# File 'lib/promiscuous/cli.rb', line 151

def run
  options = parse_args(ARGV)
  load_app(options)
  maybe_warn_bareback(options)

  # calls publish, publish_sync, subscribe, subscribe_sync
  __send__([options[:action], options[:personality]].compact.join('_'), options)
end

#subscribe(options = {}) ⇒ Object



51
52
53
54
55
56
# File 'lib/promiscuous/cli.rb', line 51

def subscribe(options={})
  replicate do
    Promiscuous::Worker.replicate(options)
    print_status "Replicating with #{Promiscuous::Subscriber::AMQP.subscribers.count} subscribers"
  end
end

#subscribe_sync(options = {}) ⇒ Object



71
72
73
74
75
76
77
78
79
80
81
82
# File 'lib/promiscuous/cli.rb', line 71

def subscribe_sync(options={})
  replicate do
    # Create the regular queue if needed, so we don't lose messages.
    Promiscuous::AMQP.open_queue(Promiscuous::Subscriber::Worker.new.queue_bindings)

    print_status "WARNING: --- SYNC MODE ----"
    print_status "WARNING: Make sure you are not running the regular subscriber worker (it's racy)"
    print_status "WARNING: --- SYNC MODE ----"
    Promiscuous::Worker.replicate(options)
    print_status "Replicating with #{Promiscuous::Subscriber::AMQP.subscribers.count} subscribers"
  end
end

#trap_signalsObject



22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
# File 'lib/promiscuous/cli.rb', line 22

def trap_signals
  %w(SIGTERM SIGINT).each do |signal|
    Signal.trap(signal) do
      print_status "Exiting..."
      Promiscuous::Worker.stop
      EM.stop
    end
  end

  Signal.trap 'SIGUSR2' do
    Thread.list.each do |thread|
      print_status '-' * 80
      if thread.backtrace
        print_status "Thread #{thread} #{thread['label']}"
        print_status thread.backtrace.join("\n")
      else
        print_status "Thread #{thread} #{thread['label']} -- no backtrace"
      end
    end
  end
end