Class: Proxy::RemoteExecution::Ssh::Actions::PullScript

Inherits:
Dynflow::Action::Runner
  • Object
show all
Defined in:
lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb

Constant Summary collapse

JobDelivered =
Class.new

Instance Method Summary collapse

Instance Method Details

#cleanup(_plan = nil) ⇒ Object



37
38
39
40
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 37

def cleanup(_plan = nil)
  job_storage.drop_job(execution_plan_id, run_step_id)
  Proxy::Dynflow::OtpManager.passwords.delete(execution_plan_id)
end

#host_nameObject



138
139
140
141
142
143
144
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 138

def host_name
  alternative_names = input.fetch(:alternative_names, {})

  alternative_names[:consumer_uuid] ||
    alternative_names[:fqdn] ||
    input[:hostname]
end

#init_runObject



25
26
27
28
29
30
31
32
33
34
35
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 25

def init_run
  otp_password = if input[:with_mqtt]
                   ::Proxy::Dynflow::OtpManager.generate_otp(execution_plan_id)
                 end

  input[:job_uuid] = job_storage.store_job(host_name, execution_plan_id, run_step_id, input[:script].tr("\r", ''))
  output[:state] = :ready_for_pickup
  output[:result] = []
  mqtt_start(otp_password) if input[:with_mqtt]
  suspend
end

#job_storageObject



154
155
156
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 154

def job_storage
  Proxy::RemoteExecution::Ssh.job_storage
end

#kill_runObject



82
83
84
85
86
87
88
89
90
91
92
93
94
95
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 82

def kill_run
  case output[:state]
  when :ready_for_pickup
    # If the job is not running yet on the client, wipe it from storage
    cleanup
    # TODO: Stop the action
  when :notified, :running
    # Client was notified or is already running, dealing with this situation
    # is only supported if mqtt is available
    # Otherwise we have to wait it out
    mqtt_cancel if input[:with_mqtt]
  end
  suspend
end

#mqtt_cancelObject



112
113
114
115
116
117
118
119
120
121
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 112

def mqtt_cancel
  cleanup
  payload = mqtt_payload_base.merge(
    metadata: {
      'event': 'cancel',
      'job_uuid': input[:job_uuid]
    }
  )
  mqtt_notify payload
end

#mqtt_notify(payload) ⇒ Object



123
124
125
126
127
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 123

def mqtt_notify(payload)
  with_mqtt_client do |c|
    c.publish(mqtt_topic, JSON.dump(payload), false, 1)
  end
end

#mqtt_payload_baseObject



158
159
160
161
162
163
164
165
166
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 158

def mqtt_payload_base
  {
    type: 'data',
    message_id: SecureRandom.uuid,
    version: 1,
    sent: DateTime.now.iso8601,
    directive: 'foreman'
  }
end

#mqtt_start(otp_password) ⇒ Object



97
98
99
100
101
102
103
104
105
106
107
108
109
110
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 97

def mqtt_start(otp_password)
  payload = mqtt_payload_base.merge(
    content: "#{input[:proxy_url]}/ssh/jobs/#{input[:job_uuid]}",
    metadata: {
      'event': 'start',
      'job_uuid': input[:job_uuid],
      'username': execution_plan_id,
      'password': otp_password,
      'return_url': "#{input[:proxy_url]}/ssh/jobs/#{input[:job_uuid]}/update",
    },
  )
  mqtt_notify payload
  output[:state] = :notified
end

#mqtt_topicObject



146
147
148
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 146

def mqtt_topic
  "yggdrasil/#{host_name}/data/in"
end

#plan(action_input, mqtt: false) ⇒ Object



11
12
13
14
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 11

def plan(action_input, mqtt: false)
  super(action_input)
  input[:with_mqtt] = mqtt
end

#process_external_event(event) ⇒ Object



42
43
44
45
46
47
48
49
50
51
52
53
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 42

def process_external_event(event)
  output[:state] = :running
  data = event.data
  case data['version']
  when nil
    process_external_unversioned(data)
  when '1'
    process_external_v1(data)
  else
    raise "Unexpected update message version '#{data['version']}'"
  end
end

#process_external_unversioned(payload) ⇒ Object



55
56
57
58
59
60
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 55

def process_external_unversioned(payload)
  continuous_output = Proxy::Dynflow::ContinuousOutput.new
  Array(payload['output']).each { |line| continuous_output.add_output(line, payload['type']) } if payload.key?('output')
  exit_code = payload['exit_code'].to_i if payload['exit_code']
  process_update(Proxy::Dynflow::Runner::Update.new(continuous_output, exit_code))
end

#process_external_v1(payload) ⇒ Object



62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 62

def process_external_v1(payload)
  continuous_output = Proxy::Dynflow::ContinuousOutput.new
  exit_code = nil

  payload['updates'].each do |update|
    time = Time.parse update['timestamp']
    type = update['type']
    case type
    when 'output'
      continuous_output.add_output(update['content'], update['stream'], time)
    when 'exit'
      exit_code = update['exit_code'].to_i
    else
      raise "Unexpected update type '#{update['type']}'"
    end
  end

  process_update(Proxy::Dynflow::Runner::Update.new(continuous_output, exit_code))
end

#run(event = nil) ⇒ Object



16
17
18
19
20
21
22
23
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 16

def run(event = nil)
  if event == JobDelivered
    output[:state] = :delivered
    suspend
  else
    super
  end
end

#settingsObject



150
151
152
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 150

def settings
  Proxy::RemoteExecution::Ssh::Plugin.settings
end

#with_mqtt_client(&block) ⇒ Object



129
130
131
132
133
134
135
136
# File 'lib/smart_proxy_remote_execution_ssh/actions/pull_script.rb', line 129

def with_mqtt_client(&block)
  MQTT::Client.connect(settings.mqtt_broker, settings.mqtt_port,
                       :ssl => settings.mqtt_tls,
                       :cert_file => ::Proxy::SETTINGS.foreman_ssl_cert || ::Proxy::SETTINGS.ssl_certificate,
                       :key_file => ::Proxy::SETTINGS.foreman_ssl_key || ::Proxy::SETTINGS.ssl_private_key,
                       :ca_file => ::Proxy::SETTINGS.foreman_ssl_ca || ::Proxy::SETTINGS.ssl_ca_file,
                       &block)
end