source_monitor 0.2.0 → 0.3.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.
Files changed (196) hide show
  1. checksums.yaml +4 -4
  2. data/.claude/agents/rails-concern.md +464 -0
  3. data/.claude/agents/rails-controller.md +424 -0
  4. data/.claude/agents/rails-hotwire.md +446 -0
  5. data/.claude/agents/rails-implement.md +374 -0
  6. data/.claude/agents/rails-job.md +334 -0
  7. data/.claude/agents/rails-lint.md +294 -0
  8. data/.claude/agents/rails-mailer.md +371 -0
  9. data/.claude/agents/rails-migration.md +449 -0
  10. data/.claude/agents/rails-model.md +420 -0
  11. data/.claude/agents/rails-policy.md +443 -0
  12. data/.claude/agents/rails-presenter.md +427 -0
  13. data/.claude/agents/rails-query.md +412 -0
  14. data/.claude/agents/rails-review.md +490 -0
  15. data/.claude/agents/rails-service.md +458 -0
  16. data/.claude/agents/rails-state-records.md +465 -0
  17. data/.claude/agents/rails-tdd.md +314 -0
  18. data/.claude/agents/rails-test.md +441 -0
  19. data/.claude/agents/rails-view-component.md +418 -0
  20. data/.claude/hooks/block-secrets.sh +52 -0
  21. data/.claude/settings.json +85 -0
  22. data/.claude/skills/action-cable-patterns/SKILL.md +296 -0
  23. data/.claude/skills/action-mailer-patterns/SKILL.md +295 -0
  24. data/.claude/skills/active-storage-setup/SKILL.md +311 -0
  25. data/.claude/skills/api-versioning/SKILL.md +294 -0
  26. data/.claude/skills/authentication-flow/SKILL.md +335 -0
  27. data/.claude/skills/authentication-flow/reference/current.md +248 -0
  28. data/.claude/skills/authentication-flow/reference/passwordless.md +253 -0
  29. data/.claude/skills/authentication-flow/reference/sessions.md +201 -0
  30. data/.claude/skills/authorization-pundit/SKILL.md +462 -0
  31. data/.claude/skills/caching-strategies/SKILL.md +350 -0
  32. data/.claude/skills/database-migrations/SKILL.md +354 -0
  33. data/.claude/skills/form-object-patterns/SKILL.md +399 -0
  34. data/.claude/skills/hotwire-patterns/SKILL.md +247 -0
  35. data/.claude/skills/hotwire-patterns/reference/stimulus.md +307 -0
  36. data/.claude/skills/hotwire-patterns/reference/tailwind-integration.md +112 -0
  37. data/.claude/skills/hotwire-patterns/reference/turbo-frames.md +158 -0
  38. data/.claude/skills/hotwire-patterns/reference/turbo-streams.md +218 -0
  39. data/.claude/skills/i18n-patterns/SKILL.md +320 -0
  40. data/.claude/skills/install/SKILL.md +367 -0
  41. data/.claude/skills/performance-optimization/SKILL.md +311 -0
  42. data/.claude/skills/rails-architecture/SKILL.md +259 -0
  43. data/.claude/skills/rails-architecture/reference/error-handling.md +333 -0
  44. data/.claude/skills/rails-architecture/reference/event-tracking.md +142 -0
  45. data/.claude/skills/rails-architecture/reference/layer-interactions.md +417 -0
  46. data/.claude/skills/rails-architecture/reference/multi-tenancy.md +152 -0
  47. data/.claude/skills/rails-architecture/reference/query-patterns.md +342 -0
  48. data/.claude/skills/rails-architecture/reference/service-patterns.md +286 -0
  49. data/.claude/skills/rails-architecture/reference/state-records.md +250 -0
  50. data/.claude/skills/rails-architecture/reference/testing-strategy.md +326 -0
  51. data/.claude/skills/rails-concern/SKILL.md +399 -0
  52. data/.claude/skills/rails-controller/SKILL.md +336 -0
  53. data/.claude/skills/rails-model-generator/SKILL.md +321 -0
  54. data/.claude/skills/rails-model-generator/reference/validations.md +298 -0
  55. data/.claude/skills/rails-presenter/SKILL.md +274 -0
  56. data/.claude/skills/rails-query-object/SKILL.md +289 -0
  57. data/.claude/skills/rails-service-object/SKILL.md +349 -0
  58. data/.claude/skills/solid-queue-setup/SKILL.md +307 -0
  59. data/.claude/skills/tdd-cycle/SKILL.md +359 -0
  60. data/.claude/skills/viewcomponent-patterns/SKILL.md +333 -0
  61. data/.gitignore +1 -0
  62. data/.rubocop.yml +2 -0
  63. data/.ruby-version +1 -1
  64. data/.vbw-planning/.notification-log.jsonl +192 -0
  65. data/.vbw-planning/.session-log.jsonl +871 -0
  66. data/.vbw-planning/PROJECT.md +51 -0
  67. data/.vbw-planning/REQUIREMENTS.md +50 -0
  68. data/.vbw-planning/SHIPPED.md +28 -0
  69. data/.vbw-planning/codebase/ARCHITECTURE.md +147 -0
  70. data/.vbw-planning/codebase/CONCERNS.md +99 -0
  71. data/.vbw-planning/codebase/CONVENTIONS.md +97 -0
  72. data/.vbw-planning/codebase/DEPENDENCIES.md +100 -0
  73. data/.vbw-planning/codebase/INDEX.md +86 -0
  74. data/.vbw-planning/codebase/META.md +42 -0
  75. data/.vbw-planning/codebase/PATTERNS.md +262 -0
  76. data/.vbw-planning/codebase/STACK.md +101 -0
  77. data/.vbw-planning/codebase/STRUCTURE.md +324 -0
  78. data/.vbw-planning/codebase/TESTING.md +154 -0
  79. data/.vbw-planning/config.json +12 -0
  80. data/.vbw-planning/discovery.json +24 -0
  81. data/.vbw-planning/milestones/default/ROADMAP.md +115 -0
  82. data/.vbw-planning/milestones/default/STATE.md +83 -0
  83. data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-01-SUMMARY.md +56 -0
  84. data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-01.md +187 -0
  85. data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-02-SUMMARY.md +64 -0
  86. data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-02.md +137 -0
  87. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-01-SUMMARY.md +67 -0
  88. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-01.md +142 -0
  89. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-02-SUMMARY.md +64 -0
  90. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-02.md +138 -0
  91. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-03-SUMMARY.md +85 -0
  92. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-03.md +147 -0
  93. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-04-SUMMARY.md +63 -0
  94. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-04.md +129 -0
  95. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-05-SUMMARY.md +74 -0
  96. data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-05.md +154 -0
  97. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/03-VERIFICATION-wave1.md +303 -0
  98. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/03-VERIFICATION.md +510 -0
  99. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-01-SUMMARY.md +61 -0
  100. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-01.md +161 -0
  101. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-02-SUMMARY.md +66 -0
  102. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-02.md +132 -0
  103. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-03-SUMMARY.md +59 -0
  104. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-03.md +171 -0
  105. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-04-SUMMARY.md +56 -0
  106. data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-04.md +152 -0
  107. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/04-CONTEXT.md +33 -0
  108. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-01-SUMMARY.md +42 -0
  109. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-01.md +119 -0
  110. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-02-SUMMARY.md +52 -0
  111. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-02.md +195 -0
  112. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-03-SUMMARY.md +79 -0
  113. data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-03.md +130 -0
  114. data/CHANGELOG.md +28 -0
  115. data/CLAUDE.md +179 -0
  116. data/Gemfile +8 -0
  117. data/Gemfile.lock +114 -101
  118. data/Rakefile +2 -0
  119. data/app/assets/builds/source_monitor/application.css +2076 -0
  120. data/app/assets/builds/source_monitor/application.js +2758 -0
  121. data/app/assets/builds/source_monitor/application.js.map +7 -0
  122. data/app/controllers/source_monitor/application_controller.rb +2 -0
  123. data/app/controllers/source_monitor/health_controller.rb +2 -0
  124. data/app/controllers/source_monitor/import_sessions/bulk_configuration.rb +106 -0
  125. data/app/controllers/source_monitor/import_sessions/entry_annotation.rb +187 -0
  126. data/app/controllers/source_monitor/import_sessions/health_check_management.rb +112 -0
  127. data/app/controllers/source_monitor/import_sessions/opml_parser.rb +130 -0
  128. data/app/controllers/source_monitor/import_sessions_controller.rb +6 -507
  129. data/app/controllers/source_monitor/items_controller.rb +2 -0
  130. data/app/controllers/source_monitor/sources_controller.rb +0 -14
  131. data/app/helpers/source_monitor/application_helper.rb +4 -112
  132. data/app/helpers/source_monitor/health_badge_helper.rb +69 -0
  133. data/app/helpers/source_monitor/table_sort_helper.rb +53 -0
  134. data/app/jobs/source_monitor/application_job.rb +2 -0
  135. data/app/models/source_monitor/application_record.rb +2 -0
  136. data/app/models/source_monitor/log_entry.rb +0 -2
  137. data/config/coverage_baseline.json +217 -1862
  138. data/config/routes.rb +2 -0
  139. data/db/migrate/20251009103000_add_feed_content_readability_to_sources.rb +2 -0
  140. data/db/migrate/20251014171659_add_performance_indexes.rb +2 -0
  141. data/db/migrate/20251014172525_add_fetch_status_check_constraint.rb +2 -0
  142. data/db/migrate/20251108120116_refresh_fetch_status_constraint.rb +2 -0
  143. data/db/migrate/20260210204022_add_composite_index_to_log_entries.rb +17 -0
  144. data/lib/source_monitor/assets/bundler.rb +2 -0
  145. data/lib/source_monitor/assets.rb +2 -0
  146. data/lib/source_monitor/configuration/authentication_settings.rb +62 -0
  147. data/lib/source_monitor/configuration/events.rb +60 -0
  148. data/lib/source_monitor/configuration/fetching_settings.rb +27 -0
  149. data/lib/source_monitor/configuration/health_settings.rb +27 -0
  150. data/lib/source_monitor/configuration/http_settings.rb +43 -0
  151. data/lib/source_monitor/configuration/model_definition.rb +108 -0
  152. data/lib/source_monitor/configuration/models.rb +36 -0
  153. data/lib/source_monitor/configuration/realtime_settings.rb +95 -0
  154. data/lib/source_monitor/configuration/retention_settings.rb +45 -0
  155. data/lib/source_monitor/configuration/scraper_registry.rb +67 -0
  156. data/lib/source_monitor/configuration/scraping_settings.rb +39 -0
  157. data/lib/source_monitor/configuration/validation_definition.rb +32 -0
  158. data/lib/source_monitor/configuration.rb +12 -579
  159. data/lib/source_monitor/dashboard/queries/recent_activity_query.rb +138 -0
  160. data/lib/source_monitor/dashboard/queries/stats_query.rb +71 -0
  161. data/lib/source_monitor/dashboard/queries.rb +2 -195
  162. data/lib/source_monitor/engine.rb +2 -0
  163. data/lib/source_monitor/fetching/feed_fetcher/adaptive_interval.rb +141 -0
  164. data/lib/source_monitor/fetching/feed_fetcher/entry_processor.rb +89 -0
  165. data/lib/source_monitor/fetching/feed_fetcher/source_updater.rb +200 -0
  166. data/lib/source_monitor/fetching/feed_fetcher.rb +37 -379
  167. data/lib/source_monitor/items/item_creator/content_extractor.rb +113 -0
  168. data/lib/source_monitor/items/item_creator/entry_parser/media_extraction.rb +96 -0
  169. data/lib/source_monitor/items/item_creator/entry_parser.rb +294 -0
  170. data/lib/source_monitor/items/item_creator.rb +28 -455
  171. data/lib/source_monitor/setup/bundle_installer.rb +2 -0
  172. data/lib/source_monitor/setup/cli.rb +2 -0
  173. data/lib/source_monitor/setup/dependency_checker.rb +2 -0
  174. data/lib/source_monitor/setup/detectors.rb +2 -0
  175. data/lib/source_monitor/setup/gemfile_editor.rb +2 -0
  176. data/lib/source_monitor/setup/initializer_patcher.rb +2 -0
  177. data/lib/source_monitor/setup/install_generator.rb +2 -0
  178. data/lib/source_monitor/setup/migration_installer.rb +2 -0
  179. data/lib/source_monitor/setup/node_installer.rb +2 -0
  180. data/lib/source_monitor/setup/prompter.rb +2 -0
  181. data/lib/source_monitor/setup/requirements.rb +2 -0
  182. data/lib/source_monitor/setup/shell_runner.rb +2 -0
  183. data/lib/source_monitor/setup/verification/action_cable_verifier.rb +2 -0
  184. data/lib/source_monitor/setup/verification/printer.rb +2 -0
  185. data/lib/source_monitor/setup/verification/result.rb +2 -0
  186. data/lib/source_monitor/setup/verification/runner.rb +2 -0
  187. data/lib/source_monitor/setup/verification/solid_queue_verifier.rb +2 -0
  188. data/lib/source_monitor/setup/verification/telemetry_logger.rb +2 -0
  189. data/lib/source_monitor/setup/workflow.rb +2 -0
  190. data/lib/source_monitor/version.rb +3 -1
  191. data/lib/source_monitor.rb +140 -58
  192. data/lib/tasks/source_monitor_assets.rake +2 -0
  193. data/lib/tasks/source_monitor_setup.rake +2 -0
  194. data/lib/tasks/source_monitor_tasks.rake +2 -0
  195. data/source_monitor.gemspec +3 -1
  196. metadata +144 -4
@@ -0,0 +1,42 @@
1
+ # META
2
+
3
+ ## Mapping Metadata
4
+
5
+ | Field | Value |
6
+ |-------|-------|
7
+ | mapped_at | 2026-02-09T00:00:00Z |
8
+ | git_hash | 1fb781a35575c2d46bfb8cd96f90c64ddf6ed210 |
9
+ | file_count | 530 |
10
+ | mode | full |
11
+ | monorepo | false |
12
+ | mapping_tier | solo (inline) |
13
+
14
+ ## Documents
15
+
16
+ | File | Domain | Description |
17
+ |------|--------|-------------|
18
+ | `STACK.md` | Tech Stack | Technology choices, framework versions, build pipeline |
19
+ | `DEPENDENCIES.md` | Tech Stack | Runtime, dev, and test dependency analysis with coupling assessment |
20
+ | `ARCHITECTURE.md` | Architecture | System overview, domain modules, data model, job architecture |
21
+ | `STRUCTURE.md` | Architecture | Full directory tree with file counts and organization |
22
+ | `CONVENTIONS.md` | Quality | Naming conventions, code style, frontend patterns |
23
+ | `TESTING.md` | Quality | Test framework, CI pipeline, coverage strategy, profiling |
24
+ | `CONCERNS.md` | Concerns | Technical debt, security risks, performance, operational risks |
25
+ | `INDEX.md` | Synthesis | Cross-referenced index with key findings and quick references |
26
+ | `PATTERNS.md` | Synthesis | 14 recurring patterns across the codebase |
27
+ | `META.md` | Meta | This file -- mapping metadata and document inventory |
28
+
29
+ ## Language Breakdown
30
+
31
+ | Language | File Count | Notes |
32
+ |----------|-----------|-------|
33
+ | Ruby (.rb) | ~324 | Models, controllers, jobs, lib modules, tests |
34
+ | ERB (.erb) | ~48 | View templates |
35
+ | Markdown (.md) | ~45 | Documentation |
36
+ | YAML (.yml) | ~16 | Configuration files |
37
+ | JavaScript (.js) | ~14 | Stimulus controllers, build entry points |
38
+ | CSS (.css) | ~2 | Tailwind input and build output |
39
+
40
+ ## Project Summary
41
+
42
+ SourceMonitor is a mountable Rails 8 engine (v0.2.1) that ingests RSS/Atom/JSON feeds, scrapes full article content via pluggable adapters, and provides Solid Queue-powered dashboards for monitoring and remediation. It is distributed as a RubyGem, requires PostgreSQL and Ruby >= 3.4.0, and integrates with host Rails applications via the standard engine mounting pattern with an isolated namespace.
@@ -0,0 +1,262 @@
1
+ # Patterns
2
+
3
+ Recurring patterns observed across the SourceMonitor codebase.
4
+
5
+ ## 1. Service Object Pattern
6
+
7
+ **Where**: `lib/source_monitor/fetching/`, `lib/source_monitor/scraping/`, `lib/source_monitor/health/`, `lib/source_monitor/items/`
8
+
9
+ Service objects encapsulate domain operations. They follow a consistent structure:
10
+
11
+ ```ruby
12
+ class SomeService
13
+ def initialize(source:, **deps)
14
+ @source = source
15
+ end
16
+
17
+ def call
18
+ # orchestrate operation
19
+ # return a Result struct
20
+ end
21
+ end
22
+ ```
23
+
24
+ **Examples**:
25
+ - `Fetching::FeedFetcher` -- `#call` returns `Result` struct
26
+ - `Scraping::ItemScraper` -- `#call` returns `Result` struct
27
+ - `Health::SourceHealthMonitor` -- `#call` updates source health
28
+ - `Health::SourceHealthCheck` -- `#call` probes source URL
29
+ - `Items::RetentionPruner` -- `#call` prunes old items
30
+ - `Items::ItemCreator` -- `.call(source:, entry:)` class method
31
+
32
+ ## 2. Struct-Based Result Objects
33
+
34
+ **Where**: Throughout all service objects
35
+
36
+ Operations return typed `Struct` instances rather than raw hashes or arrays:
37
+
38
+ ```ruby
39
+ Result = Struct.new(:status, :item, :log, :message, :error, keyword_init: true) do
40
+ def success?
41
+ status.to_s != "failed"
42
+ end
43
+ end
44
+ ```
45
+
46
+ **Examples**:
47
+ - `Scraping::ItemScraper::Result` -- `:status, :item, :log, :message, :error`
48
+ - `Scrapers::Base::Result` -- `:status, :html, :content, :metadata`
49
+ - `Fetching::FeedFetcher::Result` -- `:status, :feed, :response, :body, :error, :item_processing, :retry_decision`
50
+ - `Fetching::FeedFetcher::EntryProcessingResult` -- `:created, :updated, :failed, :items, :errors`
51
+ - `Events::ItemCreatedEvent`, `Events::ItemScrapedEvent`, `Events::FetchCompletedEvent`
52
+ - `Setup::Verification::Result` -- verification outcome
53
+
54
+ ## 3. Adapter/Strategy Pattern
55
+
56
+ **Where**: `lib/source_monitor/scrapers/`, `lib/source_monitor/realtime/`, `lib/source_monitor/items/retention_strategies/`
57
+
58
+ Pluggable behavior via abstract base class with `#call` contract:
59
+
60
+ ```ruby
61
+ class Scrapers::Base
62
+ def call
63
+ raise NotImplementedError
64
+ end
65
+ end
66
+ ```
67
+
68
+ **Instances**:
69
+ - **Scraper adapters**: `Scrapers::Base` -> `Scrapers::Readability` (registered in `ScraperRegistry`)
70
+ - **Realtime adapters**: `solid_cable`, `redis`, `async` (configured in `RealtimeSettings`)
71
+ - **Retention strategies**: `:destroy`, `:soft_delete` (in `Items::RetentionStrategies/`)
72
+
73
+ ## 4. Event/Callback System
74
+
75
+ **Where**: `lib/source_monitor/events.rb`, `lib/source_monitor/configuration.rb`
76
+
77
+ Event-driven communication between engine components:
78
+
79
+ ```ruby
80
+ # Registration
81
+ SourceMonitor.config.events.after_item_created { |event| ... }
82
+ SourceMonitor.config.events.after_item_scraped { |event| ... }
83
+ SourceMonitor.config.events.after_fetch_completed { |event| ... }
84
+
85
+ # Dispatch
86
+ SourceMonitor::Events.after_item_created(item:, source:, entry:, result:)
87
+ ```
88
+
89
+ - Typed event structs carry context
90
+ - Error isolation: each handler failure is logged, does not stop other handlers
91
+ - Item processor pipeline: `Events.run_item_processors` runs all registered processors
92
+ - Used by `Health` module to register fetch completion callback
93
+
94
+ ## 5. Configuration DSL with Nested Settings Objects
95
+
96
+ **Where**: `lib/source_monitor/configuration.rb`
97
+
98
+ Deeply nested configuration with domain-specific settings classes:
99
+
100
+ ```ruby
101
+ SourceMonitor.configure do |config|
102
+ config.http.timeout = 30
103
+ config.fetching.min_interval_minutes = 10
104
+ config.health.window_size = 50
105
+ config.scrapers.register(:custom, MyCustomScraper)
106
+ config.models.source.include_concern SomeConcern
107
+ config.authentication.authenticate_with :authenticate_admin!
108
+ end
109
+ ```
110
+
111
+ **Pattern traits**:
112
+ - Each settings class has `reset!` for test isolation
113
+ - Constants for defaults (e.g., `DEFAULT_QUEUE_NAMESPACE`)
114
+ - Callable values supported (procs/lambdas) for dynamic resolution
115
+ - Validation in setters (e.g., `RealtimeSettings#adapter=` checks `VALID_ADAPTERS`)
116
+
117
+ ## 6. Model Extension System
118
+
119
+ **Where**: `lib/source_monitor/model_extensions.rb`, `lib/source_monitor/configuration.rb`
120
+
121
+ Host apps can dynamically inject concerns and validations into engine models:
122
+
123
+ ```ruby
124
+ config.models.source.include_concern "MyApp::SourceExtensions"
125
+ config.models.source.validate :custom_validator
126
+ config.models.source.validate { |record| record.errors.add(:base, "invalid") if ... }
127
+ ```
128
+
129
+ - `ModelExtensions.register(model_class, key)` called in each model class body
130
+ - `ModelExtensions.reload!` re-applies all extensions on configuration change
131
+ - Manages table name prefix assignment
132
+ - Tracks applied concerns/validations to prevent duplicates
133
+ - Removes old extension validations before re-applying
134
+
135
+ ## 7. Turbo Stream Response Pattern
136
+
137
+ **Where**: `lib/source_monitor/turbo_streams/stream_responder.rb`, controllers
138
+
139
+ Controllers build Turbo Stream responses via a `StreamResponder` builder:
140
+
141
+ ```ruby
142
+ responder = SourceMonitor::TurboStreams::StreamResponder.new
143
+ presenter = SourceMonitor::Sources::TurboStreamPresenter.new(source:, responder:)
144
+ presenter.render_deletion(metrics:, query:, ...)
145
+ responder.toast(message:, level: :success)
146
+ render turbo_stream: responder.render(view_context)
147
+ ```
148
+
149
+ - Accumulates stream actions as an array
150
+ - Supports toast notifications, redirects, and custom actions
151
+ - Pairs with Stimulus controllers on the frontend
152
+
153
+ ## 8. Defensive Logging Guard
154
+
155
+ **Where**: All jobs and service objects
156
+
157
+ Consistent pattern for safe logging:
158
+
159
+ ```ruby
160
+ def log(stage, **extra)
161
+ return unless defined?(Rails) && Rails.respond_to?(:logger) && Rails.logger
162
+ Rails.logger.info("[SourceMonitor::...] #{payload.to_json}")
163
+ rescue StandardError
164
+ nil
165
+ end
166
+ ```
167
+
168
+ This three-part guard (`defined?`, `respond_to?`, truthy check) prevents errors when running outside Rails (e.g., in tests or standalone scripts).
169
+
170
+ ## 9. ActiveSupport::Notifications Instrumentation
171
+
172
+ **Where**: `lib/source_monitor/instrumentation.rb`, `lib/source_monitor/metrics.rb`
173
+
174
+ Standard Rails instrumentation pattern:
175
+
176
+ ```ruby
177
+ # Emit events
178
+ SourceMonitor::Instrumentation.fetch_start(payload)
179
+ SourceMonitor::Instrumentation.fetch_finish(payload)
180
+
181
+ # Subscribe to events
182
+ ActiveSupport::Notifications.subscribe("source_monitor.fetch.finish") do |...|
183
+ SourceMonitor::Metrics.increment(:fetch_finished_total)
184
+ end
185
+ ```
186
+
187
+ - Events namespaced as `source_monitor.*`
188
+ - Monotonic clock for duration measurement
189
+ - Metrics module aggregates counters and gauges in memory
190
+
191
+ ## 10. Search/Filter with Ransack
192
+
193
+ **Where**: Models (`ransackable_attributes`, `ransackable_associations`), `SanitizesSearchParams` concern
194
+
195
+ ```ruby
196
+ class Source < ApplicationRecord
197
+ def self.ransackable_attributes(_auth_object = nil)
198
+ %w[name feed_url website_url created_at ...]
199
+ end
200
+ end
201
+ ```
202
+
203
+ - Explicit whitelisting of searchable attributes (required by Ransack 4+)
204
+ - `SanitizesSearchParams` controller concern sanitizes search inputs
205
+ - Used in `SourcesController#index` and `LogsController#index`
206
+
207
+ ## 11. Circuit Breaker / Retry Policy
208
+
209
+ **Where**: `lib/source_monitor/fetching/retry_policy.rb`, `lib/source_monitor/fetching/feed_fetcher.rb`, `app/jobs/source_monitor/fetch_feed_job.rb`
210
+
211
+ Fetch failures trigger an escalating retry policy:
212
+
213
+ 1. **Retry with backoff**: Exponential wait, up to N attempts
214
+ 2. **Circuit open**: After exhausting retries, block fetches for a cooldown period
215
+ 3. **Circuit close**: Scheduler recovers after cooldown expires
216
+
217
+ State stored on `Source` model: `fetch_retry_attempt`, `fetch_circuit_opened_at`, `fetch_circuit_until`, `backoff_until`.
218
+
219
+ ## 12. Wizard State Machine (OPML Import)
220
+
221
+ **Where**: `app/models/source_monitor/import_session.rb`, `app/controllers/source_monitor/import_sessions_controller.rb`
222
+
223
+ Multi-step wizard with explicit step ordering:
224
+
225
+ ```ruby
226
+ STEP_ORDER = %w[upload preview health_check configure confirm].freeze
227
+ ```
228
+
229
+ - State persisted in `ImportSession` model with JSONB columns
230
+ - Each step has dedicated `handle_*_step` and `prepare_*_context` methods
231
+ - Navigation via `next_step`/`previous_step` model methods
232
+ - Step transitions guarded by validation (e.g., "select at least one source")
233
+
234
+ ## 13. Soft Delete Pattern
235
+
236
+ **Where**: `app/models/source_monitor/item.rb`
237
+
238
+ Items use soft deletion via `deleted_at` timestamp rather than physical deletion:
239
+
240
+ ```ruby
241
+ scope :active, -> { where(deleted_at: nil) }
242
+ scope :with_deleted, -> { unscope(where: :deleted_at) }
243
+ scope :only_deleted, -> { where.not(deleted_at: nil) }
244
+
245
+ def soft_delete!(timestamp: Time.current)
246
+ update_columns(deleted_at: timestamp, updated_at: timestamp)
247
+ Source.decrement_counter(:items_count, source_id)
248
+ end
249
+ ```
250
+
251
+ No `default_scope` is used (explicitly noted as avoiding anti-pattern).
252
+
253
+ ## 14. Separate Content Table
254
+
255
+ **Where**: `app/models/source_monitor/item.rb`, `app/models/source_monitor/item_content.rb`
256
+
257
+ Large scraped content is stored in a separate `ItemContent` model rather than on the `Item` directly:
258
+
259
+ - Lazy-loaded via `has_one :item_content`
260
+ - Auto-created when content is assigned, auto-destroyed when both fields become blank
261
+ - Delegates `scraped_html` and `scraped_content` to `ItemContent`
262
+ - Prevents bloating the items table with large text columns
@@ -0,0 +1,101 @@
1
+ # Tech Stack
2
+
3
+ ## Core Platform
4
+
5
+ | Layer | Technology | Version |
6
+ |-------|-----------|---------|
7
+ | Language | Ruby | >= 3.4.0 (CI uses 3.4.4) |
8
+ | Framework | Rails | >= 8.0.3, < 9.0 (locked at 8.1.1) |
9
+ | Database | PostgreSQL | 15 (CI service image) |
10
+ | Background Jobs | Solid Queue | >= 0.3, < 3.0 (locked at 1.2.4) |
11
+ | WebSocket/Realtime | Solid Cable | >= 3.0, < 4.0 (locked at 3.0.12) |
12
+ | Frontend Interactivity | Turbo Rails | ~> 2.0 (locked at 2.0.20) |
13
+ | JS Framework | Stimulus (Hotwired) | ^3.2.2 |
14
+ | CSS Framework | Tailwind CSS | ^3.4.10 |
15
+ | Web Server | Puma | 7.1.0 |
16
+ | Asset Pipeline | Propshaft | 1.3.1 |
17
+
18
+ ## Project Type
19
+
20
+ Mountable Rails 8 Engine gem (`source_monitor.gemspec`), distributed as a RubyGem. The engine uses `isolate_namespace SourceMonitor` and provides its own models, controllers, views, jobs, and frontend assets.
21
+
22
+ - **Version**: 0.2.1
23
+ - **Required Ruby**: >= 3.4.0
24
+ - **License**: MIT
25
+
26
+ ## Feed Parsing & HTTP
27
+
28
+ | Purpose | Gem | Version |
29
+ |---------|-----|---------|
30
+ | RSS/Atom/JSON feed parsing | Feedjira | >= 3.2, < 5.0 (locked 4.0.1) |
31
+ | HTTP client | Faraday | ~> 2.9 (locked 2.14.0) |
32
+ | HTTP retry middleware | faraday-retry | ~> 2.2 |
33
+ | HTTP redirect following | faraday-follow_redirects | ~> 0.4 |
34
+ | HTTP gzip compression | faraday-gzip | ~> 3.0 |
35
+
36
+ ## Content Scraping & Parsing
37
+
38
+ | Purpose | Gem | Version |
39
+ |---------|-----|---------|
40
+ | HTML parsing (fast, C-based) | Nokolexbor | ~> 0.5 (locked 0.6.2) |
41
+ | HTML parsing (standard) | Nokogiri | 1.18.10 (transitive) |
42
+ | Article content extraction | ruby-readability | ~> 0.7 |
43
+
44
+ ## Search & Querying
45
+
46
+ | Purpose | Gem | Version |
47
+ |---------|-----|---------|
48
+ | Search/filter forms | Ransack | ~> 4.2 (locked 4.4.1) |
49
+
50
+ ## Frontend Build Pipeline
51
+
52
+ | Tool | Purpose | Version |
53
+ |------|---------|---------|
54
+ | esbuild | JS bundling | ^0.23.0 |
55
+ | Tailwind CSS | Utility-first CSS | ^3.4.10 |
56
+ | PostCSS | CSS processing | ^8.4.45 |
57
+ | Autoprefixer | CSS vendor prefixes | ^10.4.20 |
58
+ | ESLint | JS linting | ^9.11.0 |
59
+ | Stylelint | CSS linting | ^16.8.0 |
60
+
61
+ Build orchestration via `package.json` scripts:
62
+ - `npm run build` -- builds both CSS (tailwindcss) and JS (esbuild)
63
+ - `cssbundling-rails` (~> 1.4) and `jsbundling-rails` (~> 1.3) bridge npm builds into the Rails asset pipeline
64
+
65
+ ## JS Dependencies (Runtime)
66
+
67
+ | Package | Purpose |
68
+ |---------|---------|
69
+ | `@hotwired/stimulus` ^3.2.2 | Stimulus controllers for UI interactions |
70
+ | `stimulus-use` ^0.52.0 | Stimulus composable behaviors library |
71
+
72
+ ## Testing Stack
73
+
74
+ | Tool | Purpose |
75
+ |------|---------|
76
+ | Minitest | Test framework (Rails default) |
77
+ | Capybara | System/integration test driver |
78
+ | Selenium WebDriver | Browser automation for system tests |
79
+ | WebMock | HTTP request stubbing |
80
+ | VCR | HTTP interaction recording/playback |
81
+ | SimpleCov | Code coverage (branch coverage enabled) |
82
+ | test-prof | Test profiling (TagProf, EventProf) |
83
+ | StackProf | Sampling profiler for performance analysis |
84
+
85
+ ## Code Quality & Security
86
+
87
+ | Tool | Purpose |
88
+ |------|---------|
89
+ | RuboCop (rails-omakase) | Ruby/Rails linting (omakase style) |
90
+ | Brakeman | Static security analysis |
91
+ | ESLint | JavaScript linting |
92
+ | Stylelint | CSS linting |
93
+
94
+ ## CI/CD
95
+
96
+ - **GitHub Actions** with 5 jobs: `lint`, `security`, `test`, `release_verification`, `profiling` (scheduled nightly)
97
+ - Ruby 3.4.4, Node 20
98
+ - PostgreSQL 15 as service container
99
+ - Diff coverage enforcement via custom `bin/check-diff-coverage`
100
+ - Test profiling guardrails via `bin/check-test-prof-metrics`
101
+ - Parallel test execution (configurable via `SOURCE_MONITOR_TEST_WORKERS`)