From 3e8d11d0bae207645793541f0027efea07832ff1 Mon Sep 17 00:00:00 2001 From: Kamil Szubrycht Date: Tue, 29 Jun 2021 12:28:39 +0200 Subject: [PATCH] e2e tests --- .env.production | 2 + .github/workflows/e2e.yml | 113 ++++++++++++++++++++ e2e/.gitignore | 2 + e2e/Gemfile | 8 ++ e2e/Gemfile.lock | 28 +++++ e2e/Rakefile | 7 ++ e2e/foreman/config/initializers/mock-fog.rb | 51 +++++++++ e2e/foreman/config/settings.yaml | 4 + e2e/foreman/db/seeds.d/200-lisa.rb | 50 +++++++++ e2e/production.json.with_puppet_plugin | 73 +++++++++++++ e2e/production.json.without_puppet_plugin | 73 +++++++++++++ e2e/smart-proxy/app.rb | 42 ++++++++ e2e/test_case.rb | 51 +++++++++ e2e/tests/create_host_test.rb | 39 +++++++ src/settings/example.json | 4 +- src/settings/test.json | 10 +- 16 files changed, 550 insertions(+), 7 deletions(-) create mode 100644 .github/workflows/e2e.yml create mode 100644 e2e/.gitignore create mode 100644 e2e/Gemfile create mode 100644 e2e/Gemfile.lock create mode 100644 e2e/Rakefile create mode 100644 e2e/foreman/config/initializers/mock-fog.rb create mode 100644 e2e/foreman/config/settings.yaml create mode 100644 e2e/foreman/db/seeds.d/200-lisa.rb create mode 100644 e2e/production.json.with_puppet_plugin create mode 100644 e2e/production.json.without_puppet_plugin create mode 100644 e2e/smart-proxy/app.rb create mode 100644 e2e/test_case.rb create mode 100644 e2e/tests/create_host_test.rb diff --git a/.env.production b/.env.production index a4ad21cd..be65607a 100644 --- a/.env.production +++ b/.env.production @@ -1 +1,3 @@ +REACT_APP_FOREMAN_URL=http://localhost:5000 +REACT_APP_GRAPHQL_API_URL="${REACT_APP_FOREMAN_URL}/api/graphql" REACT_APP_SETTINGS_FILE=production diff --git a/.github/workflows/e2e.yml b/.github/workflows/e2e.yml new file mode 100644 index 00000000..c1e4cbaa --- /dev/null +++ b/.github/workflows/e2e.yml @@ -0,0 +1,113 @@ +--- +name: CI +on: [push, pull_request] +env: + SEED_ADMIN_PASSWORD: password + SEED_LOCATION: "LAN" + SEED_ORGANIZATION: "Organization 1" + RAILS_ENV: development + DATABASE_URL: postgresql://postgres:@localhost/test + DATABASE_CLEANER_ALLOW_REMOTE_DATABASE_URL: true +jobs: + test: + runs-on: ubuntu-latest + services: + postgres: + image: postgres:12.1 + ports: ['5432:5432'] + options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 + strategy: + fail-fast: true + max-parallel: 2 + matrix: + foreman-core-branch: [2.5-stable, develop] + foreman-puppet-branch: [null, master] + ruby-version: [2.7] + node-version: [12.x] + exclude: + - foreman-core-branch: 2.5-stable + foreman-puppet-branch: master + steps: + - uses: actions/checkout@v2 + with: + path: lisa + - uses: actions/checkout@v2 + with: + repository: theforeman/foreman + ref: ${{ matrix.foreman-core-branch }} + path: foreman + - uses: actions/checkout@v2 + with: + repository: theforeman/foreman_puppet + ref: ${{ matrix.foreman-puppet-branch }} + path: foreman_puppet + if: ${{ matrix.foreman-puppet-branch != null }} + - name: Setup foreman_puppet + if: ${{ matrix.foreman-puppet-branch != null }} + run: | + echo "gem 'foreman_puppet', path: '../foreman_puppet'" > bundler.d/foreman_puppet.local.rb + working-directory: foreman + - name: Run Mocked Smart Proxy + run: docker run -d --env MAIN_APP_FILE='app.rb' -p 8800:80 -v /home/runner/work/lisa/lisa/lisa/e2e/smart-proxy:/usr/src/app erikap/ruby-sinatra + - uses: nanasess/setup-chromedriver@master + - run: | + export DISPLAY=:99 + chromedriver --url-base=/wd/hub & + sudo Xvfb -ac :99 -screen 0 1280x1024x24 > /dev/null 2>&1 & # optional + - run: | + sudo apt-get update + sudo apt-get install build-essential libcurl4-openssl-dev libvirt-dev ruby-libvirt zlib1g-dev libpq-dev + - name: Setup Ruby + uses: ruby/setup-ruby@v1 + with: + ruby-version: ${{ matrix.ruby-version }} + - name: Install bundler + run: gem install bundler + - name: Copy lisa settings (puppet plugin enabled) + if: ${{ matrix.foreman-puppet-branch != null }} + run: cp e2e/production.json.with_puppet_plugin src/settings/production.json + working-directory: lisa + - name: Copy lisa settings (puppet plugin disabled) + if: ${{ matrix.foreman-puppet-branch == null }} + run: cp e2e/production.json.without_puppet_plugin src/settings/production.json + working-directory: lisa + - name: Build Lisa + run: docker build -t lisa-image . + working-directory: lisa + - name: Run Lisa + run: docker run -d -p 8080:8080 lisa-image + working-directory: lisa + - name: Cache Foreman gems + uses: actions/cache@v2 + with: + path: foreman/vendor/bundle + key: ${{ runner.os }}-gems-${{ hashFiles('foreman/Gemfile.lock') }} + restore-keys: | + ${{ runner.os }}-gems- + - name: Copy the Lisa specific configs to Foreman + run: cp -r lisa/e2e/foreman ./ + - name: Setup Foreman + run: | + bundle config path vendor/bundle + bundle config set without journald development console mysql2 sqlite + bundle install --jobs 4 --retry 3 + bundle lock --update + bundle exec rake db:create + bundle exec rake db:migrate + bundle exec rake db:seed + working-directory: foreman + - name: Cache gems + uses: actions/cache@v2 + with: + path: foreman/vendor/bundle + key: ${{ runner.os }}-gems-${{ hashFiles('foreman/Gemfile.lock') }} + restore-keys: | + ${{ runner.os }}-gems- + - name: Run Foreman + run: bundle exec rails server -d -p 5000 + working-directory: foreman + - name: Run tests + run: | + bundle install + bundle exec rake test + working-directory: lisa/e2e diff --git a/e2e/.gitignore b/e2e/.gitignore new file mode 100644 index 00000000..a3a86aa4 --- /dev/null +++ b/e2e/.gitignore @@ -0,0 +1,2 @@ +smart-proxy/.bundle +screenshots diff --git a/e2e/Gemfile b/e2e/Gemfile new file mode 100644 index 00000000..00fc03e7 --- /dev/null +++ b/e2e/Gemfile @@ -0,0 +1,8 @@ +# frozen_string_literal: true + +source "https://rubygems.org" + +gem 'rake' +gem 'minitest' +gem 'selenium-webdriver' +gem 'pry' diff --git a/e2e/Gemfile.lock b/e2e/Gemfile.lock new file mode 100644 index 00000000..552107b3 --- /dev/null +++ b/e2e/Gemfile.lock @@ -0,0 +1,28 @@ +GEM + remote: https://rubygems.org/ + specs: + childprocess (3.0.0) + coderay (1.1.3) + method_source (1.0.0) + minitest (5.14.4) + pry (0.14.1) + coderay (~> 1.1) + method_source (~> 1.0) + rake (13.0.3) + rubyzip (2.3.0) + selenium-webdriver (3.142.7) + childprocess (>= 0.5, < 4.0) + rubyzip (>= 1.2.2) + +PLATFORMS + ruby + x86_64-darwin-18 + +DEPENDENCIES + minitest + pry + rake + selenium-webdriver + +BUNDLED WITH + 2.1.4 diff --git a/e2e/Rakefile b/e2e/Rakefile new file mode 100644 index 00000000..2a7b17c6 --- /dev/null +++ b/e2e/Rakefile @@ -0,0 +1,7 @@ +# frozen_string_literal: true + +require 'rake/testtask' + +Rake::TestTask.new do |t| + t.pattern = 'tests/*_test.rb' +end diff --git a/e2e/foreman/config/initializers/mock-fog.rb b/e2e/foreman/config/initializers/mock-fog.rb new file mode 100644 index 00000000..413a8b0e --- /dev/null +++ b/e2e/foreman/config/initializers/mock-fog.rb @@ -0,0 +1,51 @@ +Fog.mock! + +module VmwareExtensions + extend ActiveSupport::Concern + + module Overrides + def client + super.tap do |c| + def c.data + @data ||= begin + super.tap do |super_data| + super_data[:networks].map do |k, v| + v['name'] = 'VM Network' + v["vlanid"] = 1 + end + end.nested_under_indifferent_access + end + + @data + end + end + end + end + + included do + prepend Overrides + end +end + +::Foreman::Model::Vmware.include(VmwareExtensions) + +module CreateVmExtensions + extend ActiveSupport::Concern + + module Overrides + def create_vm(attributes = {}) + super.tap do |id| + data[:servers][id]['interfaces'].first['mac'] = 6.times.map { '%02x' % rand(0..255) }.join(':') + end + end + end + + included do + prepend Overrides + end +end + +require File.join(Gem::Specification.find_by_name('fog-vsphere').gem_dir, 'lib', 'fog', 'vsphere', 'requests', 'compute', 'create_vm.rb') + +Fog::Vsphere::Compute::Mock.include(CreateVmExtensions) + diff --git a/e2e/foreman/config/settings.yaml b/e2e/foreman/config/settings.yaml new file mode 100644 index 00000000..defb5dfc --- /dev/null +++ b/e2e/foreman/config/settings.yaml @@ -0,0 +1,4 @@ +--- +# :unattended: false +:cors_domains: + - '*' diff --git a/e2e/foreman/db/seeds.d/200-lisa.rb b/e2e/foreman/db/seeds.d/200-lisa.rb new file mode 100644 index 00000000..93e91cfc --- /dev/null +++ b/e2e/foreman/db/seeds.d/200-lisa.rb @@ -0,0 +1,50 @@ +all_locations = Location.all +all_organizations = Organization.all + +smart_proxy = SmartProxy.create!( + name: 'SmartProxyLisa', + url: 'http://localhost:8800', + locations: all_locations, + organizations: all_organizations +).tap do |smart_proxy| + PuppetClassImporter.new({ url: smart_proxy.url }).changes['new'].map do |key, value| + PuppetClassImporter.new.send(:add_classes_to_foreman, key, value) + end +end + +domain = Domain.create!( + name: 'development.example.com', + locations: all_locations, + organizations: all_organizations +).tap do |domain| + domain.subnets.create!( + name: 'SubnetLisa', + network: '127.0.0.1', + mask: '255.255.255.0', + vlanid: 1, + locations: all_locations, + organizations: all_organizations + ) +end + +ComputeResource.create!( + name: 'ComputeResourceLisa ', + provider: 'Vmware', + uuid: 'Solutions', + url: 'http://localhost:8283', + user: 'user', + password: 'password', + http_proxy_id: smart_proxy.id, + domain: domain, + caching_enabled: false, + locations: all_locations, + organizations: all_organizations +) + +Operatingsystem.create!( + name: 'OSLisa', + major: 1, + media: Medium.all, + ptables: Ptable.all, + architectures: Architecture.all +) diff --git a/e2e/production.json.with_puppet_plugin b/e2e/production.json.with_puppet_plugin new file mode 100644 index 00000000..2e8308f6 --- /dev/null +++ b/e2e/production.json.with_puppet_plugin @@ -0,0 +1,73 @@ +{ + "form_settings": { + "default_values": { + "puppet_env_id": "MDE6Rm9yZW1hblB1cHBldDo6RW52aXJvbm1lbnQtMQ==", + "puppetclass_ids": [ + "MDE6Rm9yZW1hblB1cHBldDo6UHVwcGV0Y2xhc3MtMQ==" + ] + } + }, + "default_configs": { + "organization_id": "MDE6T3JnYW5pemF0aW9uLTE=", + "architecture_id": "MDE6QXJjaGl0ZWN0dXJlLTE=" + }, + "locations": [ + { + "id": "MDE6TG9jYXRpb24tMg==", + "name": "LAN", + "code": "LAN", + "location": "LAN", + "country": "de", + "domain_name": "example.com", + "label": { + "location": "Karlsruhe", + "explanation": "Datacenter 1 (LAN)", + "reduced_performance": false + }, + "relations": { + "cluster": "CLUSTER", + "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTE=", + "puppet_ca_proxy_id": "MDE6U21hcnRQcm94eS0x", + "media": [ + { + "id": "MDE6TWVkaXVtLTE=", + "operatingsystem_id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=" + } + ] + }, + "datastore_types": [ + { + "id": "Mirror", + "name": "Mirror", + "template": "#DC#-LAN" + } + ], + "compute_attributes": { + "path_prefix": "/Datencenter/LAN/vm/Linux/" + } + } + ], + "operatingsystems": [ + { + "id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=", + "name": "OSLisa", + "relations": { + "guest_operatingsystem_id": "rhel7_64Guest", + "ptable_id": "MDE6UHRhYmxlLTEyOQ==" + } + } + ], + "app_tiers": [ + { + "name": "development", + "relations": { + "locations": [ + { + "code": "LAN", + "resource_pool": "default" + } + ] + } + } + ] +} \ No newline at end of file diff --git a/e2e/production.json.without_puppet_plugin b/e2e/production.json.without_puppet_plugin new file mode 100644 index 00000000..709c2da6 --- /dev/null +++ b/e2e/production.json.without_puppet_plugin @@ -0,0 +1,73 @@ +{ + "form_settings": { + "default_values": { + "puppet_env_id": "MDE6RW52aXJvbm1lbnQtMQ==", + "puppetclass_ids": [ + "MDE6UHVwcGV0Y2xhc3MtMQ==" + ] + } + }, + "default_configs": { + "organization_id": "MDE6T3JnYW5pemF0aW9uLTE=", + "architecture_id": "MDE6QXJjaGl0ZWN0dXJlLTE=" + }, + "locations": [ + { + "id": "MDE6TG9jYXRpb24tMg==", + "name": "LAN", + "code": "LAN", + "location": "LAN", + "country": "de", + "domain_name": "example.com", + "label": { + "location": "Karlsruhe", + "explanation": "Datacenter 1 (LAN)", + "reduced_performance": false + }, + "relations": { + "cluster": "Solutionscluster", + "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTE=", + "puppet_ca_proxy_id": "MDE6U21hcnRQcm94eS0x", + "media": [ + { + "id": "MDE6TWVkaXVtLTE=", + "operatingsystem_id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=" + } + ] + }, + "datastore_types": [ + { + "id": "datastore-123456", + "name": "datastore1", + "template": "#DC#-LAN" + } + ], + "compute_attributes": { + "path_prefix": "/Datencenter/LAN/vm/Linux/" + } + } + ], + "operatingsystems": [ + { + "id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=", + "name": "OSLisa", + "relations": { + "guest_operatingsystem_id": "rhel7_64Guest", + "ptable_id": "MDE6UHRhYmxlLTEyOQ==" + } + } + ], + "app_tiers": [ + { + "name": "development", + "relations": { + "locations": [ + { + "code": "LAN", + "resource_pool": "default" + } + ] + } + } + ] +} diff --git a/e2e/smart-proxy/app.rb b/e2e/smart-proxy/app.rb new file mode 100644 index 00000000..de825976 --- /dev/null +++ b/e2e/smart-proxy/app.rb @@ -0,0 +1,42 @@ +require 'sinatra' +require 'json' + +before do + content_type :json +end + +get '/version' do + { version: '2.6' }.to_json +end + +get '/v2/features' do + { + puppet: { state: 'running' }, + puppetca: { state: 'running' } + }.to_json +end + +get '/puppet/environments' do + ["production"].to_json +end + +get '/puppet/ca' do + [].to_json +end + +get '/puppet/ca/autosign' do +end + +get '/puppet/environments/production/classes' do + [ + { + test: { + name: 'test', + module: nil, + params: { + variable: 'var1' + } + } + } + ].to_json +end diff --git a/e2e/test_case.rb b/e2e/test_case.rb new file mode 100644 index 00000000..ba647c6c --- /dev/null +++ b/e2e/test_case.rb @@ -0,0 +1,51 @@ +# frozen_string_literal: true + +require 'forwardable' +require 'minitest/autorun' +require 'selenium-webdriver' +require 'securerandom' +require 'pry' + +class TestCase < Minitest::Test + extend Forwardable + + private + + def_delegators :driver, :get, :find_element, :quit, :page_source, :execute_script, + :manage, :save_screenshot + + def driver + @driver ||= begin + options = Selenium::WebDriver::Chrome::Options.new + options.add_argument('--headless') + options.add_argument('--no-sandbox') + options.add_argument('--disable-dev-shm-usage') + + Selenium::WebDriver.for(:chrome, options: options) + end + end + + def wait(timeout = 60) + Selenium::WebDriver::Wait.new(timeout: timeout) + end + + def lisa_url + @lisa_url ||= ENV.fetch('LISA_URL', 'http://localhost:8080') + end + + def username + @username ||= ENV.fetch('USERNAME', 'admin') + end + + def password + @password ||= ENV.fetch('PASSWORD', 'password') + end + + # https://blog.francium.tech/take-screenshot-using-ruby-selenium-webdriver-b18802822075 + def save_full_page_screenshot(path = "screenshots/#{SecureRandom.uuid}.png") + width = execute_script("return Math.max(document.body.scrollWidth, document.body.offsetWidth, document.documentElement.clientWidth, document.documentElement.scrollWidth, document.documentElement.offsetWidth);") + height = execute_script("return Math.max(document.body.scrollHeight, document.body.offsetHeight, document.documentElement.clientHeight, document.documentElement.scrollHeight, document.documentElement.offsetHeight);") + manage.window.resize_to([width+100, 1500].min, [height+100, 3000].min) + save_screenshot(path) + end +end diff --git a/e2e/tests/create_host_test.rb b/e2e/tests/create_host_test.rb new file mode 100644 index 00000000..9b463849 --- /dev/null +++ b/e2e/tests/create_host_test.rb @@ -0,0 +1,39 @@ +# frozen_string_literal: true + +require_relative '../test_case' + +class CreateHostTest < TestCase + def test_create_host + driver.get lisa_url + + find_element(name: 'username').send_keys(username) + find_element(name: 'password').send_keys(password) + + find_element(xpath: '//*[@id="root"]/div/main/div/div/div/div/div[2]/form/div[3]/div/div/button').click + + wait.until { find_element(css: 'h1.dashboard-title') } + + find_element(link_text: 'Create Hosts').click + + + wait.until { find_element(tag_name: 'h1', link_text: 'Create Hosts') } + + find_element(css: '.Select-placeholder', text: 'Select Subnet').click + wait.until { find_element(css: 'div.Select-option[aria-label="SubnetLisa"]') } + + find_element(css: 'div.Select-option[aria-label="SubnetLisa"]').click + + project = SecureRandom.hex(6) + role = SecureRandom.hex(6) + + find_element(name: 'project').send_keys(project) + find_element(name: 'role').send_keys(role) + + find_element(css: 'button[type="submit"]').click + + wait.until { find_element(link_text: "#{project}-#{role}-01.development.example.com") } + assert find_element(link_text: "#{project}-#{role}-01.development.example.com").displayed? + ensure + quit + end +end diff --git a/src/settings/example.json b/src/settings/example.json index f1572a07..0b6518ce 100644 --- a/src/settings/example.json +++ b/src/settings/example.json @@ -20,7 +20,7 @@ "size": 200, "operatingsystem_id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=", "location_code": "LAN", - "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTI=", + "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTE=", "puppet_master_id": "MDE6U21hcnRQcm94eS0x", "datastore_type_id": "Mirror", "app_tier_name": "development", @@ -72,7 +72,7 @@ }, "relations": { "cluster": "CLUSTER", - "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTI=", + "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTE=", "puppet_ca_proxy_id": "MDE6U21hcnRQcm94eS0x", "media": [ { diff --git a/src/settings/test.json b/src/settings/test.json index f8082069..68cd0a2f 100644 --- a/src/settings/test.json +++ b/src/settings/test.json @@ -13,7 +13,7 @@ }, "locations": [ { - "id": "MDE6TG9jYXRpb24tNA==", + "id": "MDE6TG9jYXRpb24tMg==", "name": "LAN", "code": "LAN", "location": "LAN", @@ -26,11 +26,11 @@ }, "relations": { "cluster": "CLUSTER", - "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTI=", + "compute_resource_id": "MDE6Q29tcHV0ZVJlc291cmNlLTE=", "puppet_ca_proxy_id": "MDE6U21hcnRQcm94eS0x", "media": [ { - "id": "MDE6TWVkaXVtLTEw", + "id": "MDE6TWVkaXVtLTE=", "operatingsystem_id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=" } ] @@ -50,10 +50,10 @@ "operatingsystems": [ { "id": "MDE6T3BlcmF0aW5nc3lzdGVtLTE=", - "name": "RedHat 7.6", + "name": "OSLisa", "relations": { "guest_operatingsystem_id": "rhel7_64Guest", - "ptable_id": "MDE6UHRhYmxlLTEwNw==" + "ptable_id": "MDE6UHRhYmxlLTEyOQ==" } } ],