ruby_raider 1.1.4 → 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.github/workflows/e2e_tests.yml +58 -0
- data/.github/workflows/integration.yml +4 -6
- data/.github/workflows/reek.yml +6 -5
- data/.github/workflows/release.yml +175 -0
- data/.github/workflows/rubocop.yml +7 -6
- data/.github/workflows/steep.yml +21 -0
- data/.github/workflows/system_tests.yml +83 -0
- data/.gitignore +1 -1
- data/.reek.yml +46 -4
- data/.rubocop.yml +24 -0
- data/.ruby-version +1 -1
- data/README.md +140 -77
- data/RELEASE.md +412 -0
- data/RELEASE_QUICK_GUIDE.md +77 -0
- data/Steepfile +22 -0
- data/assets/ruby_raider_logo.svg +51 -0
- data/bin/release +186 -0
- data/lib/adopter/adopt_menu.rb +146 -0
- data/lib/adopter/converters/base_converter.rb +84 -0
- data/lib/adopter/converters/identity_converter.rb +53 -0
- data/lib/adopter/migration_plan.rb +74 -0
- data/lib/adopter/migrator.rb +96 -0
- data/lib/adopter/plan_builder.rb +275 -0
- data/lib/adopter/project_analyzer.rb +252 -0
- data/lib/adopter/project_detector.rb +157 -0
- data/lib/commands/adopt_commands.rb +42 -0
- data/lib/commands/plugin_commands.rb +0 -2
- data/lib/commands/scaffolding_commands.rb +220 -37
- data/lib/commands/utility_commands.rb +82 -2
- data/lib/generators/automation/automation_generator.rb +0 -7
- data/lib/generators/automation/templates/account.tt +9 -5
- data/lib/generators/automation/templates/appium_caps.tt +60 -6
- data/lib/generators/automation/templates/home.tt +4 -4
- data/lib/generators/automation/templates/login.tt +61 -4
- data/lib/generators/automation/templates/page.tt +13 -7
- data/lib/generators/automation/templates/partials/element.tt +1 -1
- data/lib/generators/automation/templates/partials/home_page_selector.tt +4 -4
- data/lib/generators/automation/templates/partials/initialize_selector.tt +3 -8
- data/lib/generators/automation/templates/partials/pdp_page_selector.tt +4 -4
- data/lib/generators/automation/templates/partials/url_methods.tt +0 -1
- data/lib/generators/automation/templates/partials/visit_method.tt +11 -1
- data/lib/generators/automation/templates/pdp.tt +1 -1
- data/lib/generators/common_generator.rb +12 -0
- data/lib/generators/cucumber/cucumber_generator.rb +36 -0
- data/lib/generators/cucumber/templates/accessibility_feature.tt +5 -0
- data/lib/generators/cucumber/templates/accessibility_steps.tt +21 -0
- data/lib/generators/cucumber/templates/cucumber.tt +8 -1
- data/lib/generators/cucumber/templates/env.tt +6 -4
- data/lib/generators/cucumber/templates/feature.tt +0 -4
- data/lib/generators/cucumber/templates/partials/appium_env.tt +5 -0
- data/lib/generators/cucumber/templates/partials/capybara_env.tt +38 -0
- data/lib/generators/cucumber/templates/partials/capybara_world.tt +6 -0
- data/lib/generators/cucumber/templates/partials/driver_world.tt +1 -4
- data/lib/generators/cucumber/templates/partials/mobile_steps.tt +2 -2
- data/lib/generators/cucumber/templates/partials/selenium_env.tt +22 -35
- data/lib/generators/cucumber/templates/partials/watir_env.tt +20 -1
- data/lib/generators/cucumber/templates/partials/web_steps.tt +10 -15
- data/lib/generators/cucumber/templates/performance_feature.tt +5 -0
- data/lib/generators/cucumber/templates/performance_steps.tt +17 -0
- data/lib/generators/cucumber/templates/steps.tt +2 -2
- data/lib/generators/cucumber/templates/visual_feature.tt +5 -0
- data/lib/generators/cucumber/templates/visual_steps.tt +19 -0
- data/lib/generators/cucumber/templates/world.tt +5 -3
- data/lib/generators/generator.rb +50 -7
- data/lib/generators/helper_generator.rb +39 -9
- data/lib/generators/infrastructure/github_generator.rb +6 -0
- data/lib/generators/infrastructure/templates/github.tt +12 -8
- data/lib/generators/infrastructure/templates/github_appium.tt +108 -0
- data/lib/generators/infrastructure/templates/gitlab.tt +6 -3
- data/lib/generators/invoke_generators.rb +43 -9
- data/lib/generators/menu_generator.rb +122 -11
- data/lib/generators/minitest/minitest_generator.rb +35 -0
- data/lib/generators/minitest/templates/accessibility_test.tt +26 -0
- data/lib/generators/minitest/templates/performance_test.tt +18 -0
- data/lib/generators/minitest/templates/test.tt +64 -0
- data/lib/generators/minitest/templates/visual_test.tt +23 -0
- data/lib/generators/rspec/rspec_generator.rb +16 -4
- data/lib/generators/rspec/templates/accessibility_spec.tt +25 -0
- data/lib/generators/rspec/templates/performance_spec.tt +18 -0
- data/lib/generators/rspec/templates/spec.tt +13 -41
- data/lib/generators/rspec/templates/visual_spec.tt +20 -0
- data/lib/generators/template_renderer/partial_cache.rb +126 -0
- data/lib/generators/template_renderer/partial_resolver.rb +110 -0
- data/lib/generators/template_renderer/template_error.rb +50 -0
- data/lib/generators/template_renderer.rb +106 -0
- data/lib/generators/templates/common/config.tt +2 -2
- data/lib/generators/templates/common/gemfile.tt +36 -9
- data/lib/generators/templates/common/git_ignore.tt +6 -1
- data/lib/generators/templates/common/partials/mobile_config.tt +5 -1
- data/lib/generators/templates/common/partials/web_config.tt +17 -8
- data/lib/generators/templates/common/rakefile.tt +36 -0
- data/lib/generators/templates/common/read_me.tt +43 -91
- data/lib/generators/templates/common/rspec.tt +3 -0
- data/lib/generators/templates/common/ruby_version.tt +1 -0
- data/lib/generators/templates/helpers/allure_helper.tt +13 -2
- data/lib/generators/templates/helpers/browser_helper.tt +13 -2
- data/lib/generators/templates/helpers/capybara_helper.tt +32 -0
- data/lib/generators/templates/helpers/debug_helper.tt +190 -0
- data/lib/generators/templates/helpers/driver_helper.tt +3 -11
- data/lib/generators/templates/helpers/partials/allure_imports.tt +3 -1
- data/lib/generators/templates/helpers/partials/allure_requirements.tt +3 -1
- data/lib/generators/templates/helpers/partials/appium_driver.tt +44 -0
- data/lib/generators/templates/helpers/partials/browserstack_config.tt +13 -0
- data/lib/generators/templates/helpers/partials/debug_diagnostics.tt +7 -0
- data/lib/generators/templates/helpers/partials/debug_start.tt +7 -0
- data/lib/generators/templates/helpers/partials/driver_and_options.tt +5 -115
- data/lib/generators/templates/helpers/partials/quit_driver.tt +3 -1
- data/lib/generators/templates/helpers/partials/screenshot.tt +3 -1
- data/lib/generators/templates/helpers/partials/selenium_driver.tt +26 -0
- data/lib/generators/templates/helpers/partials/video_start.tt +9 -0
- data/lib/generators/templates/helpers/partials/video_stop.tt +4 -0
- data/lib/generators/templates/helpers/performance_helper.tt +57 -0
- data/lib/generators/templates/helpers/spec_helper.tt +72 -10
- data/lib/generators/templates/helpers/test_helper.tt +94 -0
- data/lib/generators/templates/helpers/video_helper.tt +270 -0
- data/lib/generators/templates/helpers/visual_helper.tt +39 -46
- data/lib/llm/client.rb +79 -0
- data/lib/llm/config.rb +57 -0
- data/lib/llm/prompts.rb +84 -0
- data/lib/llm/provider.rb +27 -0
- data/lib/llm/providers/anthropic_provider.rb +43 -0
- data/lib/llm/providers/ollama_provider.rb +56 -0
- data/lib/llm/providers/openai_provider.rb +42 -0
- data/lib/llm/response_parser.rb +67 -0
- data/lib/plugin/plugin.rb +22 -20
- data/lib/plugin/plugin_exposer.rb +16 -38
- data/lib/ruby_raider.rb +51 -11
- data/lib/scaffolding/crud_generator.rb +94 -0
- data/lib/scaffolding/dry_run_presenter.rb +16 -0
- data/lib/scaffolding/name_normalizer.rb +63 -0
- data/lib/scaffolding/page_introspector.rb +45 -0
- data/lib/scaffolding/project_detector.rb +72 -0
- data/lib/scaffolding/scaffold_menu.rb +103 -0
- data/lib/scaffolding/scaffolding.rb +158 -11
- data/lib/scaffolding/templates/component.tt +30 -0
- data/lib/scaffolding/templates/feature.tt +4 -4
- data/lib/scaffolding/templates/helper.tt +15 -1
- data/lib/scaffolding/templates/page_from_url.tt +75 -0
- data/lib/scaffolding/templates/page_object.tt +50 -1
- data/lib/scaffolding/templates/spec.tt +33 -2
- data/lib/scaffolding/templates/spec_from_page.tt +31 -0
- data/lib/scaffolding/templates/spec_from_url.tt +46 -0
- data/lib/scaffolding/templates/steps.tt +17 -5
- data/lib/scaffolding/url_analyzer.rb +179 -0
- data/lib/utilities/desktop_downloader.rb +177 -0
- data/lib/utilities/logo.rb +83 -0
- data/lib/utilities/utilities.rb +53 -20
- data/lib/version +1 -1
- data/ruby_raider.gemspec +1 -0
- data/sig/adopter/adopt_menu.rbs +25 -0
- data/sig/adopter/converters/base_converter.rbs +23 -0
- data/sig/adopter/converters/identity_converter.rbs +16 -0
- data/sig/adopter/migration_plan.rbs +34 -0
- data/sig/adopter/migrator.rbs +21 -0
- data/sig/adopter/plan_builder.rbs +38 -0
- data/sig/adopter/project_analyzer.rbs +39 -0
- data/sig/adopter/project_detector.rbs +26 -0
- data/sig/commands/adopt_commands.rbs +8 -0
- data/sig/commands/loaded_commands.rbs +5 -0
- data/sig/commands/plugin_commands.rbs +9 -0
- data/sig/commands/scaffolding_commands.rbs +28 -0
- data/sig/commands/utility_commands.rbs +21 -0
- data/sig/generators/automation/automation_generator.rbs +20 -0
- data/sig/generators/common_generator.rbs +12 -0
- data/sig/generators/cucumber/cucumber_generator.rbs +16 -0
- data/sig/generators/generator.rbs +40 -0
- data/sig/generators/helper_generator.rbs +18 -0
- data/sig/generators/infrastructure/github_generator.rbs +5 -0
- data/sig/generators/infrastructure/gitlab_generator.rbs +4 -0
- data/sig/generators/invoke_generators.rbs +10 -0
- data/sig/generators/menu_generator.rbs +29 -0
- data/sig/generators/minitest/minitest_generator.rbs +8 -0
- data/sig/generators/rspec/rspec_generator.rbs +8 -0
- data/sig/generators/template_renderer/partial_cache.rbs +20 -0
- data/sig/generators/template_renderer/partial_resolver.rbs +20 -0
- data/sig/generators/template_renderer/template_error.rbs +19 -0
- data/sig/generators/template_renderer.rbs +10 -0
- data/sig/llm/client.rbs +15 -0
- data/sig/llm/config.rbs +20 -0
- data/sig/llm/prompts.rbs +8 -0
- data/sig/llm/provider.rbs +12 -0
- data/sig/llm/providers/anthropic_provider.rbs +16 -0
- data/sig/llm/providers/ollama_provider.rbs +18 -0
- data/sig/llm/providers/openai_provider.rbs +16 -0
- data/sig/llm/response_parser.rbs +13 -0
- data/sig/plugin/plugin.rbs +24 -0
- data/sig/plugin/plugin_exposer.rbs +20 -0
- data/sig/ruby_raider.rbs +15 -0
- data/sig/scaffolding/crud_generator.rbs +16 -0
- data/sig/scaffolding/dry_run_presenter.rbs +4 -0
- data/sig/scaffolding/name_normalizer.rbs +17 -0
- data/sig/scaffolding/page_introspector.rbs +14 -0
- data/sig/scaffolding/project_detector.rbs +14 -0
- data/sig/scaffolding/scaffold_menu.rbs +18 -0
- data/sig/scaffolding/scaffolding.rbs +55 -0
- data/sig/scaffolding/url_analyzer.rbs +28 -0
- data/sig/utilities/desktop_downloader.rbs +23 -0
- data/sig/utilities/logger.rbs +13 -0
- data/sig/utilities/logo.rbs +16 -0
- data/sig/utilities/utilities.rbs +30 -0
- data/sig/vendor/thor.rbs +34 -0
- data/sig/vendor/tty_prompt.rbs +15 -0
- data/spec/adopter/adopt_menu_spec.rb +176 -0
- data/spec/adopter/converters/identity_converter_spec.rb +145 -0
- data/spec/adopter/migration_plan_spec.rb +113 -0
- data/spec/adopter/migrator_spec.rb +277 -0
- data/spec/adopter/plan_builder_spec.rb +298 -0
- data/spec/adopter/project_analyzer_spec.rb +337 -0
- data/spec/adopter/project_detector_spec.rb +295 -0
- data/spec/commands/raider_commands_spec.rb +129 -0
- data/spec/generators/fixtures/templates/test.tt +1 -0
- data/spec/generators/fixtures/templates/test_partial.tt +1 -0
- data/spec/generators/generator_spec.rb +23 -0
- data/spec/generators/template_renderer_spec.rb +298 -0
- data/spec/integration/commands/scaffolding_commands_spec.rb +2 -2
- data/spec/integration/commands/utility_commands_spec.rb +24 -4
- data/spec/integration/content/ci_content_spec.rb +119 -0
- data/spec/integration/content/common_content_spec.rb +288 -0
- data/spec/integration/content/config_content_spec.rb +175 -0
- data/spec/integration/content/content_helper.rb +32 -0
- data/spec/integration/content/gemfile_content_spec.rb +209 -0
- data/spec/integration/content/helper_content_spec.rb +485 -0
- data/spec/integration/content/page_content_spec.rb +259 -0
- data/spec/integration/content/reporter_content_spec.rb +236 -0
- data/spec/integration/content/skip_flags_content_spec.rb +206 -0
- data/spec/integration/content/syntax_validation_spec.rb +30 -0
- data/spec/integration/content/test_content_spec.rb +266 -0
- data/spec/integration/end_to_end_features_spec.rb +690 -0
- data/spec/integration/end_to_end_spec.rb +361 -0
- data/spec/integration/generators/automation_generator_spec.rb +9 -21
- data/spec/integration/generators/axe_addon_spec.rb +150 -0
- data/spec/integration/generators/common_generator_spec.rb +48 -49
- data/spec/integration/generators/config_features_spec.rb +155 -0
- data/spec/integration/generators/cucumber_generator_spec.rb +7 -7
- data/spec/integration/generators/debug_helper_spec.rb +68 -0
- data/spec/integration/generators/github_generator_spec.rb +8 -8
- data/spec/integration/generators/gitlab_generator_spec.rb +8 -8
- data/spec/integration/generators/helpers_generator_spec.rb +70 -44
- data/spec/integration/generators/lighthouse_addon_spec.rb +132 -0
- data/spec/integration/generators/minitest_generator_spec.rb +64 -0
- data/spec/integration/generators/reporter_spec.rb +159 -0
- data/spec/integration/generators/rspec_generator_spec.rb +7 -7
- data/spec/integration/generators/skip_flags_spec.rb +134 -0
- data/spec/integration/generators/visual_addon_spec.rb +148 -0
- data/spec/integration/settings_helper.rb +1 -4
- data/spec/integration/spec_helper.rb +46 -11
- data/spec/llm/client_spec.rb +79 -0
- data/spec/llm/config_spec.rb +92 -0
- data/spec/llm/prompts_spec.rb +49 -0
- data/spec/llm/response_parser_spec.rb +92 -0
- data/spec/menus/adopter_adopt_menu_spec.rb +97 -0
- data/spec/menus/menu_generator_spec.rb +263 -0
- data/spec/scaffolding/name_normalizer_spec.rb +113 -0
- data/spec/scaffolding/page_introspector_spec.rb +82 -0
- data/spec/scaffolding/scaffold_project_detector_spec.rb +104 -0
- data/spec/scaffolding/scaffolding_features_spec.rb +311 -0
- data/spec/scaffolding/url_analyzer_spec.rb +110 -0
- data/spec/system/adopt_matrix_spec.rb +537 -0
- data/spec/system/adopt_spec.rb +225 -0
- data/spec/system/capybara_spec.rb +42 -0
- data/spec/system/selenium_spec.rb +19 -17
- data/spec/system/support/system_test_helper.rb +33 -0
- data/spec/system/watir_spec.rb +19 -17
- data/spec/utilities/desktop_downloader_spec.rb +92 -0
- metadata +193 -18
- data/.github/workflows/push_gem.yml +0 -37
- data/.github/workflows/selenium.yml +0 -22
- data/.github/workflows/watir.yml +0 -22
- data/lib/generators/automation/templates/partials/android_caps.tt +0 -17
- data/lib/generators/automation/templates/partials/cross_platform_caps.tt +0 -25
- data/lib/generators/automation/templates/partials/ios_caps.tt +0 -18
- data/lib/generators/automation/templates/partials/selenium_account.tt +0 -9
- data/lib/generators/automation/templates/partials/selenium_login.tt +0 -34
- data/lib/generators/automation/templates/partials/watir_account.tt +0 -7
- data/lib/generators/automation/templates/partials/watir_login.tt +0 -32
- data/lib/generators/automation/templates/visual_options.tt +0 -16
- data/lib/generators/templates/helpers/visual_spec_helper.tt +0 -35
|
@@ -1,12 +1,21 @@
|
|
|
1
1
|
# frozen_string_literal: true
|
|
2
2
|
|
|
3
|
-
<%- if
|
|
3
|
+
<%- if axe_addon? %>
|
|
4
4
|
require 'axe-rspec'
|
|
5
5
|
<%- end -%>
|
|
6
6
|
require 'rspec'
|
|
7
|
+
require 'rspec/retry'
|
|
7
8
|
require 'tmpdir'
|
|
9
|
+
<%- if allure_reporter? -%>
|
|
8
10
|
require_relative 'allure_helper'
|
|
9
|
-
<%-
|
|
11
|
+
<%- end -%>
|
|
12
|
+
<%- unless skip_video? -%>
|
|
13
|
+
require_relative 'video_helper'
|
|
14
|
+
<%- end -%>
|
|
15
|
+
require_relative 'debug_helper'
|
|
16
|
+
<%- if capybara? -%>
|
|
17
|
+
require_relative 'capybara_helper'
|
|
18
|
+
<%- elsif watir? -%>
|
|
10
19
|
require_relative 'browser_helper'
|
|
11
20
|
<%- else -%>
|
|
12
21
|
require_relative 'driver_helper'
|
|
@@ -14,31 +23,84 @@ require_relative 'driver_helper'
|
|
|
14
23
|
|
|
15
24
|
module SpecHelper
|
|
16
25
|
|
|
26
|
+
<%- if allure_reporter? -%>
|
|
17
27
|
AllureHelper.configure
|
|
28
|
+
<%- end -%>
|
|
18
29
|
RSpec.configure do |config|
|
|
30
|
+
config.example_status_persistence_file_path = 'spec/examples.txt'
|
|
31
|
+
config.verbose_retry = true
|
|
32
|
+
config.display_try_failure_messages = true
|
|
33
|
+
config.default_retry_count = ENV.fetch('RETRY_COUNT', 0).to_i
|
|
34
|
+
<%- if allure_reporter? -%>
|
|
19
35
|
config.formatter = AllureHelper.formatter
|
|
20
|
-
|
|
36
|
+
<%- end -%>
|
|
37
|
+
<%- if junit_reporter? -%>
|
|
38
|
+
config.add_formatter('RspecJunitFormatter', 'results/junit.xml')
|
|
39
|
+
<%- end -%>
|
|
40
|
+
<%- if json_reporter? -%>
|
|
41
|
+
config.add_formatter('json', 'results/results.json')
|
|
42
|
+
<%- end -%>
|
|
43
|
+
<%- if capybara? -%>
|
|
44
|
+
config.include(Capybara::DSL)
|
|
45
|
+
CapybaraHelper.configure
|
|
46
|
+
<%- elsif watir? -%>
|
|
47
|
+
config.include(BrowserHelper)
|
|
48
|
+
<%- else -%>
|
|
49
|
+
config.include(DriverHelper)
|
|
50
|
+
<%- end -%>
|
|
21
51
|
<%- if mobile? -%>
|
|
22
52
|
config.before(:each) do
|
|
23
53
|
driver.start_driver
|
|
24
|
-
|
|
54
|
+
<%- elsif capybara? -%>
|
|
55
|
+
config.before(:each) do
|
|
56
|
+
viewport = YAML.load_file('config/config.yml')['viewport']
|
|
57
|
+
if viewport
|
|
58
|
+
Capybara.current_session.driver.browser.manage.window.resize_to(viewport['width'], viewport['height'])
|
|
59
|
+
else
|
|
60
|
+
Capybara.current_session.driver.browser.manage.window.maximize
|
|
61
|
+
end
|
|
25
62
|
<%- elsif selenium_based? -%>
|
|
26
63
|
config.before(:each) do
|
|
27
|
-
|
|
28
|
-
|
|
64
|
+
viewport = YAML.load_file('config/config.yml')['viewport']
|
|
65
|
+
if viewport
|
|
66
|
+
driver.manage.window.resize_to(viewport['width'], viewport['height'])
|
|
67
|
+
else
|
|
68
|
+
driver.manage.window.maximize
|
|
69
|
+
end
|
|
29
70
|
<%- elsif watir? -%>
|
|
30
71
|
config.before(:each) do
|
|
31
|
-
|
|
32
|
-
|
|
72
|
+
viewport = YAML.load_file('config/config.yml')['viewport']
|
|
73
|
+
if viewport
|
|
74
|
+
browser.window.resize_to(viewport['width'], viewport['height'])
|
|
75
|
+
else
|
|
76
|
+
browser.window.maximize
|
|
77
|
+
end
|
|
33
78
|
<%- end -%>
|
|
79
|
+
<%- unless skip_video? -%>
|
|
80
|
+
<%= partial('video_start', strip: true) %>
|
|
81
|
+
@video_recorder&.start(self.class.description)
|
|
82
|
+
<%- end -%>
|
|
83
|
+
<%= partial('debug_start', strip: true) %>
|
|
84
|
+
@debug_action_logger = DebugHelper.action_logger_for(self.class.description)
|
|
85
|
+
end
|
|
34
86
|
|
|
35
87
|
config.after(:each) do |example|
|
|
36
88
|
example_name = example.description
|
|
89
|
+
if example.exception
|
|
90
|
+
@_exception = example.exception
|
|
91
|
+
<%= partial('debug_diagnostics', strip: true) %>
|
|
92
|
+
end
|
|
93
|
+
@debug_action_logger&.close
|
|
94
|
+
<%- unless skip_video? -%>
|
|
95
|
+
<%= partial('video_stop', strip: true) %>
|
|
96
|
+
<%- end -%>
|
|
37
97
|
Dir.mktmpdir do |temp_folder|
|
|
38
|
-
<%=
|
|
98
|
+
<%= partial('screenshot', strip: true) %>
|
|
99
|
+
<%- if allure_reporter? -%>
|
|
39
100
|
AllureHelper.add_screenshot(example_name, screenshot)
|
|
101
|
+
<%- end -%>
|
|
40
102
|
end
|
|
41
|
-
<%=
|
|
103
|
+
<%= partial('quit_driver', strip: true) %>
|
|
42
104
|
end
|
|
43
105
|
end
|
|
44
106
|
end
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
<%- if axe_addon? %>
|
|
4
|
+
require 'axe-minitest'
|
|
5
|
+
<%- end -%>
|
|
6
|
+
require 'minitest/autorun'
|
|
7
|
+
require 'tmpdir'
|
|
8
|
+
<%- if allure_reporter? -%>
|
|
9
|
+
require_relative 'allure_helper'
|
|
10
|
+
<%- end -%>
|
|
11
|
+
<%- unless skip_video? -%>
|
|
12
|
+
require_relative 'video_helper'
|
|
13
|
+
<%- end -%>
|
|
14
|
+
require_relative 'debug_helper'
|
|
15
|
+
<%- if watir? -%>
|
|
16
|
+
require_relative 'browser_helper'
|
|
17
|
+
<%- elsif capybara? -%>
|
|
18
|
+
require_relative 'capybara_helper'
|
|
19
|
+
<%- else -%>
|
|
20
|
+
require_relative 'driver_helper'
|
|
21
|
+
<%- end -%>
|
|
22
|
+
|
|
23
|
+
<%- if allure_reporter? -%>
|
|
24
|
+
AllureHelper.configure
|
|
25
|
+
<%- end -%>
|
|
26
|
+
<%- if capybara? -%>
|
|
27
|
+
CapybaraHelper.configure
|
|
28
|
+
<%- end -%>
|
|
29
|
+
<%- if json_reporter? -%>
|
|
30
|
+
require 'minitest/reporters'
|
|
31
|
+
Minitest::Reporters.use! [Minitest::Reporters::JsonReporter.new('results/results.json')]
|
|
32
|
+
<%- end -%>
|
|
33
|
+
|
|
34
|
+
module TestHelper
|
|
35
|
+
<% if watir? %>include BrowserHelper<% elsif capybara? %>include Capybara::DSL<% else %>include DriverHelper<% end %>
|
|
36
|
+
|
|
37
|
+
def setup
|
|
38
|
+
super
|
|
39
|
+
<%- if mobile? -%>
|
|
40
|
+
driver.start_driver
|
|
41
|
+
<%- elsif capybara? -%>
|
|
42
|
+
viewport = YAML.load_file('config/config.yml')['viewport']
|
|
43
|
+
if viewport
|
|
44
|
+
Capybara.current_session.driver.browser.manage.window.resize_to(viewport['width'], viewport['height'])
|
|
45
|
+
else
|
|
46
|
+
Capybara.current_session.driver.browser.manage.window.maximize
|
|
47
|
+
end
|
|
48
|
+
<%- elsif watir? -%>
|
|
49
|
+
viewport = YAML.load_file('config/config.yml')['viewport']
|
|
50
|
+
if viewport
|
|
51
|
+
browser.window.resize_to(viewport['width'], viewport['height'])
|
|
52
|
+
else
|
|
53
|
+
browser.window.maximize
|
|
54
|
+
end
|
|
55
|
+
<%- else -%>
|
|
56
|
+
viewport = YAML.load_file('config/config.yml')['viewport']
|
|
57
|
+
if viewport
|
|
58
|
+
driver.manage.window.resize_to(viewport['width'], viewport['height'])
|
|
59
|
+
else
|
|
60
|
+
driver.manage.window.maximize
|
|
61
|
+
end
|
|
62
|
+
<%- end -%>
|
|
63
|
+
<%- unless skip_video? -%>
|
|
64
|
+
<%= partial('video_start', strip: true) %>
|
|
65
|
+
@video_recorder&.start(self.class.name)
|
|
66
|
+
<%- end -%>
|
|
67
|
+
<%= partial('debug_start', strip: true) %>
|
|
68
|
+
@debug_action_logger = DebugHelper.action_logger_for(self.class.name)
|
|
69
|
+
end
|
|
70
|
+
|
|
71
|
+
def teardown
|
|
72
|
+
example_name = name
|
|
73
|
+
unless passed?
|
|
74
|
+
<%= partial('debug_diagnostics', strip: true) %>
|
|
75
|
+
end
|
|
76
|
+
@debug_action_logger&.close
|
|
77
|
+
<%- unless skip_video? -%>
|
|
78
|
+
video_file = @video_recorder&.stop
|
|
79
|
+
<%- if allure_reporter? -%>
|
|
80
|
+
AllureHelper.add_video(example_name, video_file)
|
|
81
|
+
<%- end -%>
|
|
82
|
+
<%- end -%>
|
|
83
|
+
Dir.mktmpdir do |temp_folder|
|
|
84
|
+
<%= partial('screenshot', strip: true) %>
|
|
85
|
+
<%- if allure_reporter? -%>
|
|
86
|
+
AllureHelper.add_screenshot(example_name, screenshot)
|
|
87
|
+
<%- end -%>
|
|
88
|
+
end
|
|
89
|
+
<%= partial('quit_driver', strip: true) %>
|
|
90
|
+
super
|
|
91
|
+
end
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
Minitest::Test.include(TestHelper)
|
|
@@ -0,0 +1,270 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
require 'fileutils'
|
|
4
|
+
require 'yaml'
|
|
5
|
+
require 'json'
|
|
6
|
+
require 'base64'
|
|
7
|
+
|
|
8
|
+
module VideoHelper
|
|
9
|
+
VIDEO_DIR = 'videos'
|
|
10
|
+
|
|
11
|
+
module_function
|
|
12
|
+
|
|
13
|
+
def config
|
|
14
|
+
@config ||= begin
|
|
15
|
+
yml = YAML.load_file('config/config.yml')
|
|
16
|
+
yml['video'] || {}
|
|
17
|
+
rescue StandardError
|
|
18
|
+
{}
|
|
19
|
+
end
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
def enabled?
|
|
23
|
+
config.fetch('enabled', false)
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
def recorder_for(driver_or_browser)
|
|
27
|
+
return NullRecorder.new unless enabled?
|
|
28
|
+
|
|
29
|
+
strategy = config.fetch('strategy', 'auto')
|
|
30
|
+
case strategy
|
|
31
|
+
when 'cdp' then CdpRecorder.new(resolve_selenium_driver(driver_or_browser))
|
|
32
|
+
when 'screen' then ScreenCaptureRecorder.new
|
|
33
|
+
<%- if mobile? -%>
|
|
34
|
+
when 'appium' then AppiumRecorder.new(driver_or_browser)
|
|
35
|
+
<%- end -%>
|
|
36
|
+
when 'auto' then auto_detect(driver_or_browser)
|
|
37
|
+
else NullRecorder.new
|
|
38
|
+
end
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
def auto_detect(driver_or_browser)
|
|
42
|
+
sel_driver = resolve_selenium_driver(driver_or_browser)
|
|
43
|
+
<%- if mobile? -%>
|
|
44
|
+
return AppiumRecorder.new(driver_or_browser) if defined?(Appium::Driver)
|
|
45
|
+
<%- end -%>
|
|
46
|
+
return CdpRecorder.new(sel_driver) if chrome_or_edge?(sel_driver)
|
|
47
|
+
|
|
48
|
+
if ffmpeg_available?
|
|
49
|
+
ScreenCaptureRecorder.new
|
|
50
|
+
else
|
|
51
|
+
NullRecorder.new
|
|
52
|
+
end
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
def resolve_selenium_driver(obj)
|
|
56
|
+
if obj.respond_to?(:driver) && !obj.is_a?(Selenium::WebDriver::Driver)
|
|
57
|
+
obj.driver
|
|
58
|
+
elsif obj.respond_to?(:browser) && obj.browser.respond_to?(:driver)
|
|
59
|
+
obj.browser.driver
|
|
60
|
+
else
|
|
61
|
+
obj
|
|
62
|
+
end
|
|
63
|
+
rescue StandardError
|
|
64
|
+
obj
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
def chrome_or_edge?(driver)
|
|
68
|
+
browser_name = driver.capabilities[:browser_name].to_s.downcase
|
|
69
|
+
%w[chrome chromium msedge edge].any? { |b| browser_name.include?(b) }
|
|
70
|
+
rescue StandardError
|
|
71
|
+
false
|
|
72
|
+
end
|
|
73
|
+
|
|
74
|
+
def ffmpeg_available?
|
|
75
|
+
system('ffmpeg -version > /dev/null 2>&1') ||
|
|
76
|
+
system('ffmpeg -version > NUL 2>&1')
|
|
77
|
+
end
|
|
78
|
+
|
|
79
|
+
def ensure_video_dir
|
|
80
|
+
FileUtils.mkdir_p(VIDEO_DIR)
|
|
81
|
+
end
|
|
82
|
+
|
|
83
|
+
# --- CDP Recorder: Chrome/Edge, works headless ---
|
|
84
|
+
|
|
85
|
+
class CdpRecorder
|
|
86
|
+
def initialize(driver)
|
|
87
|
+
@driver = driver
|
|
88
|
+
@frames = []
|
|
89
|
+
@recording = false
|
|
90
|
+
end
|
|
91
|
+
|
|
92
|
+
def start(test_name)
|
|
93
|
+
@test_name = sanitize(test_name)
|
|
94
|
+
@frames = []
|
|
95
|
+
@frame_index = 0
|
|
96
|
+
@recording = true
|
|
97
|
+
@driver.execute_cdp('Page.startScreencast',
|
|
98
|
+
format: 'jpeg', quality: 60,
|
|
99
|
+
maxWidth: 1280, maxHeight: 720,
|
|
100
|
+
everyNthFrame: 2)
|
|
101
|
+
start_frame_collector
|
|
102
|
+
rescue StandardError => e
|
|
103
|
+
warn "[video] CDP recording failed to start: #{e.message}"
|
|
104
|
+
@recording = false
|
|
105
|
+
end
|
|
106
|
+
|
|
107
|
+
def stop
|
|
108
|
+
return nil unless @recording
|
|
109
|
+
|
|
110
|
+
@recording = false
|
|
111
|
+
@driver.execute_cdp('Page.stopScreencast')
|
|
112
|
+
sleep 0.2 # allow final frames to arrive
|
|
113
|
+
assemble_video
|
|
114
|
+
rescue StandardError => e
|
|
115
|
+
warn "[video] CDP recording failed to stop: #{e.message}"
|
|
116
|
+
nil
|
|
117
|
+
end
|
|
118
|
+
|
|
119
|
+
private
|
|
120
|
+
|
|
121
|
+
def start_frame_collector
|
|
122
|
+
@collector = Thread.new do
|
|
123
|
+
while @recording
|
|
124
|
+
begin
|
|
125
|
+
collect_frame
|
|
126
|
+
rescue StandardError
|
|
127
|
+
sleep 0.1
|
|
128
|
+
end
|
|
129
|
+
sleep 0.05
|
|
130
|
+
end
|
|
131
|
+
end
|
|
132
|
+
end
|
|
133
|
+
|
|
134
|
+
def collect_frame
|
|
135
|
+
# CDP screencast events are handled by polling via execute_cdp
|
|
136
|
+
# We capture screenshots at intervals as a reliable alternative
|
|
137
|
+
VideoHelper.ensure_video_dir
|
|
138
|
+
frame_path = File.join(VideoHelper::VIDEO_DIR, "#{@test_name}_frame_#{format('%04d', @frame_index)}.jpg")
|
|
139
|
+
@driver.save_screenshot(frame_path)
|
|
140
|
+
@frames << frame_path
|
|
141
|
+
@frame_index += 1
|
|
142
|
+
end
|
|
143
|
+
|
|
144
|
+
def assemble_video
|
|
145
|
+
return nil if @frames.empty?
|
|
146
|
+
|
|
147
|
+
@collector&.join(2)
|
|
148
|
+
VideoHelper.ensure_video_dir
|
|
149
|
+
output = File.join(VideoHelper::VIDEO_DIR, "#{@test_name}.mp4")
|
|
150
|
+
pattern = File.join(VideoHelper::VIDEO_DIR, "#{@test_name}_frame_%04d.jpg")
|
|
151
|
+
|
|
152
|
+
system("ffmpeg -y -framerate 10 -i \"#{pattern}\" -c:v libx264 -pix_fmt yuv420p " \
|
|
153
|
+
"-movflags +faststart -loglevel error \"#{output}\"")
|
|
154
|
+
|
|
155
|
+
cleanup_frames
|
|
156
|
+
File.exist?(output) ? output : nil
|
|
157
|
+
end
|
|
158
|
+
|
|
159
|
+
def cleanup_frames
|
|
160
|
+
@frames.each { |f| FileUtils.rm_f(f) }
|
|
161
|
+
end
|
|
162
|
+
|
|
163
|
+
def sanitize(name)
|
|
164
|
+
name.gsub(/[^a-zA-Z0-9_-]/, '_')[0..80]
|
|
165
|
+
end
|
|
166
|
+
end
|
|
167
|
+
|
|
168
|
+
# --- Screen Capture Recorder: Any browser, requires FFmpeg + display ---
|
|
169
|
+
|
|
170
|
+
class ScreenCaptureRecorder
|
|
171
|
+
def initialize
|
|
172
|
+
@recording = false
|
|
173
|
+
end
|
|
174
|
+
|
|
175
|
+
def start(test_name)
|
|
176
|
+
@test_name = sanitize(test_name)
|
|
177
|
+
VideoHelper.ensure_video_dir
|
|
178
|
+
@output = File.join(VideoHelper::VIDEO_DIR, "#{@test_name}.mp4")
|
|
179
|
+
@recording = true
|
|
180
|
+
|
|
181
|
+
input = screen_input
|
|
182
|
+
@pid = spawn("ffmpeg -y #{input} -r 15 -c:v libx264 -pix_fmt yuv420p " \
|
|
183
|
+
"-loglevel error \"#{@output}\"",
|
|
184
|
+
%i[out err] => '/dev/null')
|
|
185
|
+
rescue StandardError => e
|
|
186
|
+
warn "[video] Screen recording failed to start: #{e.message}"
|
|
187
|
+
@recording = false
|
|
188
|
+
end
|
|
189
|
+
|
|
190
|
+
def stop
|
|
191
|
+
return nil unless @recording
|
|
192
|
+
|
|
193
|
+
@recording = false
|
|
194
|
+
Process.kill('INT', @pid)
|
|
195
|
+
Process.wait(@pid)
|
|
196
|
+
File.exist?(@output) ? @output : nil
|
|
197
|
+
rescue StandardError => e
|
|
198
|
+
warn "[video] Screen recording failed to stop: #{e.message}"
|
|
199
|
+
nil
|
|
200
|
+
end
|
|
201
|
+
|
|
202
|
+
private
|
|
203
|
+
|
|
204
|
+
def screen_input
|
|
205
|
+
case RbConfig::CONFIG['host_os']
|
|
206
|
+
when /darwin/i
|
|
207
|
+
'-f avfoundation -i "1:none"'
|
|
208
|
+
when /linux/i
|
|
209
|
+
display = ENV.fetch('DISPLAY', ':99')
|
|
210
|
+
"-f x11grab -i #{display}"
|
|
211
|
+
when /mswin|mingw|cygwin/i
|
|
212
|
+
'-f gdigrab -i desktop'
|
|
213
|
+
else
|
|
214
|
+
'-f x11grab -i :0'
|
|
215
|
+
end
|
|
216
|
+
end
|
|
217
|
+
|
|
218
|
+
def sanitize(name)
|
|
219
|
+
name.gsub(/[^a-zA-Z0-9_-]/, '_')[0..80]
|
|
220
|
+
end
|
|
221
|
+
end
|
|
222
|
+
|
|
223
|
+
<%- if mobile? -%>
|
|
224
|
+
# --- Appium Recorder: Mobile, uses native API ---
|
|
225
|
+
|
|
226
|
+
class AppiumRecorder
|
|
227
|
+
def initialize(driver)
|
|
228
|
+
@driver = driver
|
|
229
|
+
@recording = false
|
|
230
|
+
end
|
|
231
|
+
|
|
232
|
+
def start(test_name)
|
|
233
|
+
@test_name = sanitize(test_name)
|
|
234
|
+
@recording = true
|
|
235
|
+
@driver.start_recording_screen
|
|
236
|
+
rescue StandardError => e
|
|
237
|
+
warn "[video] Appium recording failed to start: #{e.message}"
|
|
238
|
+
@recording = false
|
|
239
|
+
end
|
|
240
|
+
|
|
241
|
+
def stop
|
|
242
|
+
return nil unless @recording
|
|
243
|
+
|
|
244
|
+
@recording = false
|
|
245
|
+
video_base64 = @driver.stop_recording_screen
|
|
246
|
+
VideoHelper.ensure_video_dir
|
|
247
|
+
output = File.join(VideoHelper::VIDEO_DIR, "#{@test_name}.mp4")
|
|
248
|
+
File.write(output, Base64.decode64(video_base64), mode: 'wb')
|
|
249
|
+
output
|
|
250
|
+
rescue StandardError => e
|
|
251
|
+
warn "[video] Appium recording failed to stop: #{e.message}"
|
|
252
|
+
nil
|
|
253
|
+
end
|
|
254
|
+
|
|
255
|
+
private
|
|
256
|
+
|
|
257
|
+
def sanitize(name)
|
|
258
|
+
name.gsub(/[^a-zA-Z0-9_-]/, '_')[0..80]
|
|
259
|
+
end
|
|
260
|
+
end
|
|
261
|
+
|
|
262
|
+
<%- end -%>
|
|
263
|
+
# --- Null Recorder: Fallback when recording unavailable ---
|
|
264
|
+
|
|
265
|
+
class NullRecorder
|
|
266
|
+
def start(_test_name) = nil
|
|
267
|
+
|
|
268
|
+
def stop = nil
|
|
269
|
+
end
|
|
270
|
+
end
|
|
@@ -1,65 +1,58 @@
|
|
|
1
1
|
# frozen_string_literal: true
|
|
2
2
|
|
|
3
|
-
require '
|
|
4
|
-
require '
|
|
5
|
-
require 'yaml'
|
|
3
|
+
require 'chunky_png'
|
|
4
|
+
require 'fileutils'
|
|
6
5
|
|
|
7
6
|
module VisualHelper
|
|
8
|
-
|
|
7
|
+
BASELINE_DIR = 'visual_baselines'
|
|
8
|
+
DIFF_DIR = 'visual_diffs'
|
|
9
9
|
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
TARGET = SELENIUM::Target
|
|
14
|
-
BATCHINFO = Applitools::BatchInfo
|
|
15
|
-
REGTANGLE_SIZE = Applitools::RectangleSize
|
|
10
|
+
def compare_screenshot(name, screenshot_path, threshold: 0.01)
|
|
11
|
+
FileUtils.mkdir_p(BASELINE_DIR)
|
|
12
|
+
FileUtils.mkdir_p(DIFF_DIR)
|
|
16
13
|
|
|
17
|
-
|
|
18
|
-
VISUAL_GRID.new(concurrency)
|
|
19
|
-
end
|
|
14
|
+
baseline_path = File.join(BASELINE_DIR, "#{name}.png")
|
|
20
15
|
|
|
21
|
-
|
|
22
|
-
|
|
16
|
+
unless File.exist?(baseline_path)
|
|
17
|
+
FileUtils.cp(screenshot_path, baseline_path)
|
|
18
|
+
return { status: :baseline_created, path: baseline_path }
|
|
23
19
|
end
|
|
24
20
|
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
@eyes.check(page, TARGET.window.fully)
|
|
28
|
-
end
|
|
21
|
+
baseline = ChunkyPNG::Image.from_file(baseline_path)
|
|
22
|
+
current = ChunkyPNG::Image.from_file(screenshot_path)
|
|
29
23
|
|
|
30
|
-
|
|
31
|
-
|
|
24
|
+
diff_pixels = 0
|
|
25
|
+
total_pixels = baseline.width * baseline.height
|
|
32
26
|
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
27
|
+
diff_image = ChunkyPNG::Image.new(baseline.width, baseline.height)
|
|
28
|
+
|
|
29
|
+
baseline.height.times do |y|
|
|
30
|
+
baseline.width.times do |x|
|
|
31
|
+
if pixel_match?(baseline[x, y], current[x, y])
|
|
32
|
+
diff_image[x, y] = ChunkyPNG::Color.rgba(0, 0, 0, 50)
|
|
33
|
+
else
|
|
34
|
+
diff_pixels += 1
|
|
35
|
+
diff_image[x, y] = ChunkyPNG::Color.rgb(255, 0, 0)
|
|
36
|
+
end
|
|
38
37
|
end
|
|
39
38
|
end
|
|
40
39
|
|
|
41
|
-
|
|
42
|
-
page.instance_of?(String) ? page : page.to_s
|
|
43
|
-
end
|
|
40
|
+
diff_percentage = diff_pixels.to_f / total_pixels
|
|
44
41
|
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
42
|
+
if diff_percentage > threshold
|
|
43
|
+
diff_path = File.join(DIFF_DIR, "#{name}_diff.png")
|
|
44
|
+
diff_image.save(diff_path)
|
|
45
|
+
{ status: :mismatch, diff: diff_percentage, diff_path: }
|
|
46
|
+
else
|
|
47
|
+
{ status: :match, diff: diff_percentage }
|
|
49
48
|
end
|
|
49
|
+
end
|
|
50
50
|
|
|
51
|
-
|
|
52
|
-
devices.each do |device|
|
|
53
|
-
conf.add_device_emulation("Devices::#{device[:name]}".constantize,
|
|
54
|
-
"Orientation::#{device[:orientation]}".constantize)
|
|
55
|
-
end
|
|
56
|
-
end
|
|
51
|
+
private
|
|
57
52
|
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
conf.viewport_size = REGTANGLE_SIZE.new(options[:viewport_size][:height], options[:viewport_size][:width])
|
|
64
|
-
end
|
|
53
|
+
def pixel_match?(pixel1, pixel2)
|
|
54
|
+
r1, g1, b1 = ChunkyPNG::Color.r(pixel1), ChunkyPNG::Color.g(pixel1), ChunkyPNG::Color.b(pixel1)
|
|
55
|
+
r2, g2, b2 = ChunkyPNG::Color.r(pixel2), ChunkyPNG::Color.g(pixel2), ChunkyPNG::Color.b(pixel2)
|
|
56
|
+
(r1 - r2).abs <= 5 && (g1 - g2).abs <= 5 && (b1 - b2).abs <= 5
|
|
57
|
+
end
|
|
65
58
|
end
|
data/lib/llm/client.rb
ADDED
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
require 'net/http'
|
|
4
|
+
require_relative 'config'
|
|
5
|
+
|
|
6
|
+
module Llm
|
|
7
|
+
# Facade for LLM completion with retry logic and graceful fallback.
|
|
8
|
+
# Returns nil on any failure — callers should always have a non-AI fallback.
|
|
9
|
+
module Client
|
|
10
|
+
MAX_RETRIES = 3
|
|
11
|
+
BASE_DELAY = 1
|
|
12
|
+
|
|
13
|
+
# Only retry on transient network errors, not configuration or API errors
|
|
14
|
+
RETRYABLE_ERRORS = [
|
|
15
|
+
Net::OpenTimeout, Net::ReadTimeout, Errno::ECONNREFUSED,
|
|
16
|
+
Errno::ECONNRESET, Errno::ETIMEDOUT, SocketError
|
|
17
|
+
].freeze
|
|
18
|
+
|
|
19
|
+
class << self
|
|
20
|
+
def complete(prompt, system_prompt: nil)
|
|
21
|
+
provider = build_provider
|
|
22
|
+
return nil unless provider
|
|
23
|
+
|
|
24
|
+
attempt = 0
|
|
25
|
+
begin
|
|
26
|
+
attempt += 1
|
|
27
|
+
provider.complete(prompt, system_prompt:)
|
|
28
|
+
rescue *RETRYABLE_ERRORS => e
|
|
29
|
+
if attempt < MAX_RETRIES
|
|
30
|
+
sleep(BASE_DELAY * (2**(attempt - 1)))
|
|
31
|
+
retry
|
|
32
|
+
end
|
|
33
|
+
warn "[Ruby Raider] LLM failed after #{MAX_RETRIES} attempts: #{e.message}"
|
|
34
|
+
nil
|
|
35
|
+
rescue StandardError => e
|
|
36
|
+
warn "[Ruby Raider] LLM error: #{e.message}"
|
|
37
|
+
nil
|
|
38
|
+
end
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
def available?
|
|
42
|
+
config = Config.new
|
|
43
|
+
return false unless config.configured?
|
|
44
|
+
|
|
45
|
+
provider = config.build_provider
|
|
46
|
+
provider&.available? || false
|
|
47
|
+
end
|
|
48
|
+
|
|
49
|
+
def status
|
|
50
|
+
config = Config.new
|
|
51
|
+
return { configured: false, provider: nil } unless config.configured?
|
|
52
|
+
|
|
53
|
+
provider = config.build_provider
|
|
54
|
+
{
|
|
55
|
+
configured: true,
|
|
56
|
+
provider: config.provider_name,
|
|
57
|
+
model: config.model,
|
|
58
|
+
available: provider&.available? || false
|
|
59
|
+
}
|
|
60
|
+
end
|
|
61
|
+
|
|
62
|
+
private
|
|
63
|
+
|
|
64
|
+
def build_provider
|
|
65
|
+
config = Config.new
|
|
66
|
+
unless config.configured?
|
|
67
|
+
warn '[Ruby Raider] No LLM configured. Use: raider u llm ollama'
|
|
68
|
+
return nil
|
|
69
|
+
end
|
|
70
|
+
provider = config.build_provider
|
|
71
|
+
unless provider&.available?
|
|
72
|
+
warn "[Ruby Raider] LLM provider '#{config.provider_name}' is not available"
|
|
73
|
+
return nil
|
|
74
|
+
end
|
|
75
|
+
provider
|
|
76
|
+
end
|
|
77
|
+
end
|
|
78
|
+
end
|
|
79
|
+
end
|