-
Notifications
You must be signed in to change notification settings - Fork 900
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Introduce FirmwareRegistry and FirmwareBinary
One can update physical server's firmware, but the firmware has to be accessible somewhere, usually on some HTTP/FTP/SAMBA link. Furthermore, there is usually some metadata associated with the firmware to tell what kind of hardware (manufacturer, model type) is it meant for. With this commit we introduce a new model, FirmwareRegistry, that supports various kinds of metadata parsing by leveraging STI. User is supposed to create FirmwareRegistry instance via UI feeding it url+authentication and then click "Refresh Relationships" on it. ManageIQ will then invoke ``` registry.sync_fw_binaries_queue ``` which in turn will connect to the metadata repository to list available firmwares and store them as FirmwareBinary (no binary data is really stored, only HTTP/FTP/SAMBA link and some metadata). In followup PRs we will then be able to list available firmware binaries to update physical server's firmware. Signed-off-by: Miha Pleško <[email protected]>
- Loading branch information
1 parent
344a3b3
commit b06a7e1
Showing
21 changed files
with
784 additions
and
1 deletion.
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
class FirmwareBinary < ApplicationRecord | ||
validates :name, :presence => true | ||
has_many :endpoints, :dependent => :destroy, :as => :resource, :inverse_of => :resource | ||
has_many :firmware_binary_firmware_targets, :dependent => :destroy | ||
has_many :firmware_targets, :through => :firmware_binary_firmware_targets | ||
belongs_to :firmware_registry | ||
|
||
def allow_duplicate_endpoint_url? | ||
true | ||
end | ||
|
||
def urls | ||
endpoints.pluck(:url) | ||
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,6 @@ | ||
class FirmwareBinaryFirmwareTarget < ApplicationRecord | ||
self.table_name = 'firmware_binaries_firmware_targets' | ||
|
||
belongs_to :firmware_binary | ||
belongs_to :firmware_target | ||
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,37 @@ | ||
class FirmwareRegistry < ApplicationRecord | ||
include NewWithTypeStiMixin | ||
|
||
has_many :firmware_binaries, :dependent => :destroy | ||
has_one :endpoint, :as => :resource, :dependent => :destroy, :inverse_of => :resource | ||
has_one :authentication, :as => :resource, :dependent => :destroy, :inverse_of => :resource | ||
|
||
validates :name, :presence => true, :uniqueness => true | ||
|
||
def sync_fw_binaries_queue | ||
MiqQueue.put_unless_exists( | ||
:class_name => self.class.name, | ||
:instance_id => id, | ||
:method_name => 'sync_fw_binaries' | ||
) | ||
end | ||
|
||
def sync_fw_binaries | ||
_log.info("Synchronizing FirmwareBinaries from #{self.class.name} [#{id}|#{name}]...") | ||
sync_fw_binaries_raw | ||
self.last_refresh_error = nil | ||
_log.info("Synchronizing FirmwareBinaries from #{self.class.name} [#{id}|#{name}]... Complete") | ||
rescue MiqException::Error => e | ||
self.last_refresh_error = e | ||
ensure | ||
self.last_refresh_on = Time.now.utc | ||
save! | ||
end | ||
|
||
def sync_fw_binaries_raw | ||
raise NotImplementedError, 'Must be implemented in subclass' | ||
end | ||
|
||
def self.display_name(number = 1) | ||
n_('Firmware Registry', 'Firmware Registries', number) | ||
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,62 @@ | ||
class FirmwareRegistry::RestApiDepot < FirmwareRegistry | ||
def sync_fw_binaries_raw | ||
remote_binaries.each do |binary_hash| | ||
binary = FirmwareBinary.find_or_create_by(:firmware_registry => self, :external_ref => binary_hash['id']) | ||
_log.info("Updating FirmwareBinary [#{binary.id} | #{binary.name}]...") | ||
|
||
binary.name = binary_hash['filename'] || binary_hash['id'] | ||
binary.description = binary_hash['description'] | ||
binary.version = binary_hash['version'] | ||
binary.save! | ||
|
||
unless binary.urls.sort == binary_hash['urls'].sort | ||
_log.info("Updating FirmwareBinary [#{binary.id} | #{binary.name}] endpoints...") | ||
endpoints_by_url = binary.endpoints.index_by(&:url) | ||
urls_to_delete = binary.urls - binary_hash['urls'] | ||
urls_to_delete.each { |url| endpoints_by_url[url].destroy } | ||
urls_to_create = binary_hash['urls'] - binary.urls | ||
urls_to_create.each { |url| binary.endpoints.create!(:url => url) } | ||
end | ||
|
||
currents = binary.firmware_targets.map(&:to_hash) | ||
remotes = binary_hash['compatible_server_models'].map { |r| r.symbolize_keys.slice(*FirmwareTarget::MATCH_ATTRIBUTES) } | ||
unless currents.map(&:to_a).sort == remotes.map(&:to_a).sort | ||
_log.info("Updating FirmwareBinary [#{binary.id} | #{binary.name}] targets...") | ||
binary.firmware_targets = remotes.map do |remote| | ||
FirmwareTarget.find_compatible_with(remote, :create => true) | ||
end | ||
end | ||
|
||
_log.info("Updating FirmwareBinary [#{binary.id} | #{binary.name}]... completed.") | ||
end | ||
end | ||
|
||
def remote_binaries | ||
self.class.fetch_from_remote( | ||
endpoint.url, | ||
authentication.userid, | ||
authentication.password, | ||
:verify_ssl => endpoint.security_protocol | ||
) | ||
end | ||
|
||
def self.fetch_from_remote(url, username, password, verify_ssl: OpenSSL::SSL::VERIFY_PEER) | ||
uri = URI.parse(url) | ||
request = Net::HTTP::Get.new(uri.request_uri) | ||
request['Content-Type'] = 'application/json' | ||
request.basic_auth(username, password) | ||
response = Net::HTTP.new(uri.host, uri.port).tap do |http| | ||
http.open_timeout = 5.seconds | ||
http.use_ssl = url.start_with?('https') | ||
http.verify_mode = verify_ssl | ||
end.request(request) | ||
|
||
raise MiqException::Error, "Bad status returned: #{response.code}" if response.code.to_i != 200 | ||
|
||
JSON.parse(response.body) | ||
rescue SocketError => e | ||
raise MiqException::Error, e | ||
rescue JSON::ParserError => e | ||
raise MiqException::Error, e | ||
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,33 @@ | ||
class FirmwareTarget < ApplicationRecord | ||
has_many :firmware_binary_firmware_targets, :dependent => :destroy | ||
has_many :firmware_binaries, :through => :firmware_binary_firmware_targets | ||
|
||
before_create :normalize | ||
before_save :normalize | ||
|
||
# Attributes that need to match for target physical server to be assumed compatible. | ||
MATCH_ATTRIBUTES = %i[manufacturer model].freeze | ||
|
||
def normalize | ||
manufacturer.downcase! | ||
model.downcase! | ||
end | ||
|
||
# Firmware targets are nameless, but we need to show something on the UI. | ||
def name | ||
_('Firmware Target') | ||
end | ||
|
||
def to_hash | ||
attributes.symbolize_keys.slice(*MATCH_ATTRIBUTES) | ||
end | ||
|
||
def self.find_compatible_with(attributes, create: false) | ||
relevant_attrs = attributes.tap do |attrs| | ||
attrs.symbolize_keys! | ||
attrs.slice!(*MATCH_ATTRIBUTES) | ||
attrs.transform_values!(&:downcase) | ||
end | ||
create ? find_or_create_by(relevant_attrs) : find_by(relevant_attrs) | ||
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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
FactoryBot.define do | ||
factory :asset_detail do | ||
sequence(:manufacturer) { |n| "manufacturer_#{seq_padded_for_sorting(n)}" } | ||
sequence(:model) { |n| "model_#{seq_padded_for_sorting(n)}" } | ||
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,14 @@ | ||
FactoryBot.define do | ||
factory :firmware_binary do | ||
sequence(:name) { |n| "firmware_binary_#{seq_padded_for_sorting(n)}" } | ||
sequence(:description) { 'Firmware Binary Description' } | ||
sequence(:version) { 'v1.2.3' } | ||
|
||
trait :with_endpoints do | ||
after(:create) do |binary| | ||
binary.endpoints << FactoryBot.create(:endpoint, :url => 'http://test.binary.1', :resource => binary) | ||
binary.endpoints << FactoryBot.create(:endpoint, :url => 'http://test.binary.2', :resource => binary) | ||
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,11 @@ | ||
FactoryBot.define do | ||
factory :firmware_registry do | ||
sequence(:name) { |n| "firmware_registry_#{seq_padded_for_sorting(n)}" } | ||
endpoint { |n| FactoryBot.create(:endpoint, :url => "http://test.registry.#{n}") } | ||
authentication { FactoryBot.create(:authentication) } | ||
end | ||
|
||
factory :firmware_registry_rest_api_depot, | ||
:parent => :firmware_registry, | ||
:class => 'FirmwareRegistry::RestApiDepot' | ||
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,6 @@ | ||
FactoryBot.define do | ||
factory :firmware_target do | ||
sequence(:manufacturer) { |n| "manufacturer_#{seq_padded_for_sorting(n)}" } | ||
sequence(:model) { |n| "model_#{seq_padded_for_sorting(n)}" } | ||
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 |
---|---|---|
@@ -1,5 +1,11 @@ | ||
FactoryBot.define do | ||
factory :physical_server do | ||
vendor { "lenovo" } | ||
|
||
trait :with_asset_detail do | ||
after :create do |server| | ||
server.asset_detail = FactoryBot.create(:asset_detail) | ||
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,49 @@ | ||
describe FirmwareBinary do | ||
subject { FactoryBot.create(:firmware_binary) } | ||
|
||
describe '#allow_duplicate_endpoint_url?' do | ||
let(:binary1) { FactoryBot.create(:firmware_binary) } | ||
let(:binary2) { FactoryBot.create(:firmware_binary) } | ||
|
||
it 'has the flag set' do | ||
expect(subject.allow_duplicate_endpoint_url?).to eq(true) | ||
end | ||
|
||
it 'two different binaries are allowed to have same url' do | ||
expect do | ||
binary1.endpoints << FactoryBot.create(:endpoint, :url => 'same-url', :resource => binary1) | ||
binary2.endpoints << FactoryBot.create(:endpoint, :url => 'same-url', :resource => binary2) | ||
end.not_to raise_error | ||
end | ||
end | ||
|
||
describe '#urls' do | ||
it 'lists from all endpoints' do | ||
subject.endpoints << FactoryBot.create(:endpoint, :url => 'url1') | ||
subject.endpoints << FactoryBot.create(:endpoint, :url => 'url2') | ||
expect(subject.urls).to eq(%w[url1 url2]) | ||
end | ||
end | ||
|
||
describe '#has_many endpoints' do | ||
before { subject.endpoints = [endpoint1, endpoint2] } | ||
let(:endpoint1) { FactoryBot.create(:endpoint) } | ||
let(:endpoint2) { FactoryBot.create(:endpoint) } | ||
|
||
it 'one to many connection exists' do | ||
expect(subject.endpoints).to match_array([endpoint1, endpoint2]) | ||
end | ||
end | ||
|
||
describe '#has_many firmware_targets' do | ||
before { subject.firmware_targets = [target1, target2] } | ||
let(:target1) { FactoryBot.create(:firmware_target) } | ||
let(:target2) { FactoryBot.create(:firmware_target) } | ||
|
||
it 'many to many connection exists' do | ||
expect(subject.firmware_targets).to match_array([target1, target2]) | ||
expect(target1.firmware_binaries).to match_array([subject]) | ||
expect(target2.firmware_binaries).to match_array([subject]) | ||
end | ||
end | ||
end |
54 changes: 54 additions & 0 deletions
54
spec/models/firmware_registry/data/sample_firmware_binaries.json
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,54 @@ | ||
[ | ||
{ | ||
"description": "DELL BIOS update", | ||
"urls": [ | ||
"http://172.16.93.126:8080/files/dell-bios-2019-03-23.bin", | ||
"https://172.16.93.126:8443/files/dell-bios-2019-03-23.bin" | ||
], | ||
"compatible_server_models": [ | ||
{ | ||
"model": "Common Model", | ||
"manufacturer": "Common Manufacturer" | ||
} | ||
], | ||
"id": 1, | ||
"version": "10.4.3", | ||
"filename": "dell-bios-2019-03-23.bin" | ||
}, | ||
{ | ||
"description": "DELL BIOS update", | ||
"urls": [ | ||
"http://172.16.93.126:8080/files/dell-bios-2019-04-13.bin", | ||
"https://172.16.93.126:8443/files/dell-bios-2019-04-13.bin" | ||
], | ||
"compatible_server_models": [ | ||
{ | ||
"model": "Common Model", | ||
"manufacturer": "Common Manufacturer" | ||
} | ||
], | ||
"id": 2, | ||
"version": "10.4.4", | ||
"filename": "dell-bios-2019-04-13.bin" | ||
}, | ||
{ | ||
"description": "SuperMicro NIC firmware", | ||
"urls": [ | ||
"http://172.16.93.126:8080/files/nic-update-3.exe", | ||
"https://172.16.93.126:8443/files/nic-update-3.exe" | ||
], | ||
"compatible_server_models": [ | ||
{ | ||
"model": "SYS-6033A-GHJ7Z", | ||
"manufacturer": "SuperMicro" | ||
}, | ||
{ | ||
"model": "SYS-5019D-FN8TP", | ||
"manufacturer": "SuperMicro" | ||
} | ||
], | ||
"id": 3, | ||
"version": "4.3.2-101-z", | ||
"filename": "nic-update-3.exe" | ||
} | ||
] |
Oops, something went wrong.