-
Notifications
You must be signed in to change notification settings - Fork 897
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Introduce Request and Task for firmware update
We're introducing a new operation on physical server: firmware update. We want to use state machine-driven Request for it because it's a long running procedure and we want to be doing it on more than one server with a single click (and then track its progress). That being said - the PR contains a new Request class and a new Task class (with internal state machine). The actual state machine implementation is done in provider codebase - depending on class of the selected physical server. Signed-off-by: Miha Pleško <[email protected]>
- Loading branch information
1 parent
344a3b3
commit 5bd59ad
Showing
11 changed files
with
302 additions
and
16 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
class PhysicalServerFirmwareUpdateRequest < MiqRequest | ||
TASK_DESCRIPTION = 'Physical Server Firmware Update'.freeze | ||
SOURCE_CLASS_NAME = 'PhysicalServer'.freeze | ||
|
||
def description | ||
'Physical Server Firmware Update' | ||
end | ||
|
||
def my_role(_action = nil) | ||
'ems_operations' | ||
end | ||
|
||
def self.request_task_class | ||
PhysicalServerFirmwareUpdateTask | ||
end | ||
|
||
def self.new_request_task(attribs) | ||
affected_ems(attribs).class.firmware_update_class.new(attribs) | ||
end | ||
|
||
def requested_task_idx | ||
[-1] # we are only using one task per request not matter how many servers are affected | ||
end | ||
|
||
def self.affected_physical_servers(attribs) | ||
ids = attribs.dig('options', :src_ids) | ||
raise MiqException::MiqFirmwareUpdateError, 'At least one PhysicalServer is required' if ids&.empty? | ||
|
||
PhysicalServer.where(:id => ids).tap do |servers| | ||
unless servers.size == ids.size | ||
raise MiqException::MiqFirmwareUpdateError, 'At least one PhysicalServer is missing' | ||
end | ||
unless servers.map(&:ems_id).uniq.size == 1 | ||
raise MiqException::MiqFirmwareUpdateError, 'All PhysicalServers need to belong to same EMS' | ||
end | ||
end | ||
end | ||
|
||
def self.affected_ems(attribs) | ||
affected_physical_servers(attribs).first.ext_management_system | ||
end | ||
|
||
def affected_physical_servers | ||
@affected_physical_servers ||= self.class.affected_physical_servers(attributes) | ||
end | ||
|
||
def affected_ems | ||
@affected_ems ||= self.class.affected_ems(attributes) | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
class PhysicalServerFirmwareUpdateTask < MiqRequestTask | ||
include_concern 'StateMachine' | ||
|
||
validates :state, :inclusion => { | ||
:in => %w[pending queued active firmware_updated finished], | ||
:message => 'should be pending, queued, active, firmware_updated or finished' | ||
} | ||
|
||
AUTOMATE_DRIVES = false | ||
|
||
def description | ||
'Physical Server Firmware Update' | ||
end | ||
|
||
def self.base_model | ||
PhysicalServerFirmwareUpdateTask | ||
end | ||
|
||
def do_request | ||
signal :run_firmware_update | ||
end | ||
|
||
def self.request_class | ||
PhysicalServerFirmwareUpdateRequest | ||
end | ||
|
||
def self.display_name(number = 1) | ||
n_('Firmware Update Task', 'Firmware Update Tasks', number) | ||
end | ||
end |
35 changes: 35 additions & 0 deletions
35
app/models/physical_server_firmware_update_task/state_machine.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
module PhysicalServerFirmwareUpdateTask::StateMachine | ||
def run_firmware_update | ||
dump_obj(options, "MIQ(#{self.class.name}##{__method__}) options: ", $log, :info) | ||
signal :start_firmware_update | ||
end | ||
|
||
def start_firmware_update | ||
# Implement firmware update in subclass, user-defined values are stored in options field. | ||
# Affected servers can be accessed via miq_request.affected_physical_servers | ||
raise NotImplementedError, 'Must be implemented in subclass and signal :done_firmware_update when done' | ||
end | ||
|
||
def done_firmware_update | ||
update_and_notify_parent(:message => msg('done updating firmware')) | ||
signal :mark_as_completed | ||
end | ||
|
||
def mark_as_completed | ||
update_and_notify_parent(:state => 'firmware_updated', :message => msg('firmware update completed')) | ||
MiqEvent.raise_evm_event(self, 'generic_task_finish', :message => "Done updating firmware on PhysicalServer(s)") | ||
signal :finish | ||
end | ||
|
||
def finish | ||
if status != 'Error' | ||
_log.info("Executing firmware update task: [#{description}]... Complete") | ||
else | ||
_log.info("Executing firmware update task: [#{description}]... Errored") | ||
end | ||
end | ||
|
||
def msg(txt) | ||
"Updating firmware for PhysicalServer(s): #{txt}" | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
93 changes: 93 additions & 0 deletions
93
spec/models/physical_server_firmware_update_request_spec.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
describe PhysicalServerFirmwareUpdateRequest do | ||
it '.TASK_DESCRIPTION' do | ||
expect(described_class::TASK_DESCRIPTION).to eq('Physical Server Firmware Update') | ||
end | ||
|
||
it '.SOURCE_CLASS_NAME' do | ||
expect(described_class::SOURCE_CLASS_NAME).to eq('PhysicalServer') | ||
end | ||
|
||
it '.request_task_class' do | ||
expect(described_class.request_task_class).to eq(PhysicalServerFirmwareUpdateTask) | ||
end | ||
|
||
it '#description' do | ||
expect(subject.description).to eq('Physical Server Firmware Update') | ||
end | ||
|
||
it '#my_role' do | ||
expect(subject.my_role).to eq('ems_operations') | ||
end | ||
|
||
it '#requested_task_idx' do | ||
expect(subject.requested_task_idx).to eq([-1]) | ||
end | ||
|
||
describe '.new_request_task' do | ||
before { allow(ems.class).to receive(:firmware_update_class).and_return(task) } | ||
|
||
let(:ems) { FactoryBot.create(:ems_physical_infra) } | ||
let(:task) { double('TASK') } | ||
|
||
it 'returns subclassed task' do | ||
expect(described_class).to receive(:affected_ems).and_return(ems) | ||
expect(task).to receive(:new).with('ATTRS') | ||
described_class.new_request_task('ATTRS') | ||
end | ||
end | ||
|
||
describe '.affected_physical_servers' do | ||
let(:attrs) { { 'options' => {:src_ids => src_ids} } } | ||
let(:server1) { FactoryBot.create(:physical_server, :ems_id => 1) } | ||
let(:server2) { FactoryBot.create(:physical_server, :ems_id => 2) } | ||
let(:server3) { FactoryBot.create(:physical_server, :ems_id => 2) } | ||
|
||
context 'when no src_ids are given' do | ||
let(:src_ids) { [] } | ||
it 'handled error is raised' do | ||
expect { described_class.affected_physical_servers(attrs) }.to raise_error(MiqException::MiqFirmwareUpdateError) | ||
end | ||
end | ||
|
||
context 'when invalid src_ids are given' do | ||
let(:src_ids) { ['invalid'] } | ||
it 'handled error is raised' do | ||
expect { described_class.affected_physical_servers(attrs) }.to raise_error(MiqException::MiqFirmwareUpdateError) | ||
end | ||
end | ||
|
||
context 'when servers belong to different ems' do | ||
let(:src_ids) { [server1.id, server2.id] } | ||
it 'handled error is raised' do | ||
expect { described_class.affected_physical_servers(attrs) }.to raise_error(MiqException::MiqFirmwareUpdateError) | ||
end | ||
end | ||
|
||
context 'when all okay' do | ||
let(:src_ids) { [server2.id, server3.id] } | ||
it 'server list is returned' do | ||
expect(described_class.affected_physical_servers(attrs)).to eq([server2, server3]) | ||
end | ||
end | ||
end | ||
|
||
it '#affected_physical_servers' do | ||
expect(described_class).to receive(:affected_physical_servers).and_return('RES') | ||
expect(subject.affected_physical_servers).to eq('RES') | ||
end | ||
|
||
describe '.affected_ems' do | ||
let(:ems) { FactoryBot.create(:ems_physical_infra) } | ||
let(:server) { FactoryBot.create(:physical_server, :ems_id => ems.id) } | ||
|
||
it 'when all okay' do | ||
expect(described_class).to receive(:affected_physical_servers).and_return([server]) | ||
expect(described_class.affected_ems(nil)).to eq(ems) | ||
end | ||
end | ||
|
||
it '#affected_ems' do | ||
expect(described_class).to receive(:affected_ems).and_return('RES') | ||
expect(subject.affected_ems).to eq('RES') | ||
end | ||
end |
50 changes: 50 additions & 0 deletions
50
spec/models/physical_server_firmware_update_task/state_machine_spec.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
describe PhysicalServerFirmwareUpdateTask do | ||
let(:server) { FactoryBot.create(:physical_server) } | ||
let(:src_ids) { [server.id] } | ||
|
||
subject { described_class.new(:options => { :src_ids => src_ids }) } | ||
|
||
describe '#run_firmware_update' do | ||
context 'when ok' do | ||
it do | ||
expect(subject).to receive(:signal).with(:start_firmware_update) | ||
subject.run_firmware_update | ||
end | ||
end | ||
end | ||
|
||
it '#done_firmware_update' do | ||
expect(subject).to receive(:signal).with(:mark_as_completed) | ||
expect(subject).to receive(:update_and_notify_parent) | ||
subject.done_firmware_update | ||
end | ||
|
||
it '#mark_as_completed' do | ||
expect(subject).to receive(:signal).with(:finish) | ||
expect(subject).to receive(:update_and_notify_parent) | ||
expect(MiqAeEvent).to receive(:raise_evm_event) | ||
subject.mark_as_completed | ||
end | ||
|
||
describe '#finish' do | ||
before { allow(subject).to receive(:_log).and_return(log) } | ||
|
||
let(:log) { double('LOG') } | ||
|
||
context 'when task has errored' do | ||
before { subject.update(:status => 'Error') } | ||
it do | ||
expect(log).to receive(:info).with(satisfy { |msg| msg.include?('Errored') }) | ||
subject.finish | ||
end | ||
end | ||
|
||
context 'when task has completed' do | ||
before { subject.update(:status => 'Ok') } | ||
it do | ||
expect(log).to receive(:info).with(satisfy { |msg| msg.include?('... Complete') }) | ||
subject.finish | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
describe PhysicalServerFirmwareUpdateTask do | ||
it '#description' do | ||
expect(subject.description).to eq('Physical Server Firmware Update') | ||
end | ||
|
||
it '.base_model' do | ||
expect(described_class.base_model).to eq(PhysicalServerFirmwareUpdateTask) | ||
end | ||
|
||
it '.request_class' do | ||
expect(described_class.request_class).to eq(PhysicalServerFirmwareUpdateRequest) | ||
end | ||
|
||
it '#do_request' do | ||
expect(subject).to receive(:signal).with(:run_firmware_update) | ||
subject.do_request | ||
end | ||
end |