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.
- checksums.yaml +4 -4
- data/.claude/agents/rails-concern.md +464 -0
- data/.claude/agents/rails-controller.md +424 -0
- data/.claude/agents/rails-hotwire.md +446 -0
- data/.claude/agents/rails-implement.md +374 -0
- data/.claude/agents/rails-job.md +334 -0
- data/.claude/agents/rails-lint.md +294 -0
- data/.claude/agents/rails-mailer.md +371 -0
- data/.claude/agents/rails-migration.md +449 -0
- data/.claude/agents/rails-model.md +420 -0
- data/.claude/agents/rails-policy.md +443 -0
- data/.claude/agents/rails-presenter.md +427 -0
- data/.claude/agents/rails-query.md +412 -0
- data/.claude/agents/rails-review.md +490 -0
- data/.claude/agents/rails-service.md +458 -0
- data/.claude/agents/rails-state-records.md +465 -0
- data/.claude/agents/rails-tdd.md +314 -0
- data/.claude/agents/rails-test.md +441 -0
- data/.claude/agents/rails-view-component.md +418 -0
- data/.claude/hooks/block-secrets.sh +52 -0
- data/.claude/settings.json +85 -0
- data/.claude/skills/action-cable-patterns/SKILL.md +296 -0
- data/.claude/skills/action-mailer-patterns/SKILL.md +295 -0
- data/.claude/skills/active-storage-setup/SKILL.md +311 -0
- data/.claude/skills/api-versioning/SKILL.md +294 -0
- data/.claude/skills/authentication-flow/SKILL.md +335 -0
- data/.claude/skills/authentication-flow/reference/current.md +248 -0
- data/.claude/skills/authentication-flow/reference/passwordless.md +253 -0
- data/.claude/skills/authentication-flow/reference/sessions.md +201 -0
- data/.claude/skills/authorization-pundit/SKILL.md +462 -0
- data/.claude/skills/caching-strategies/SKILL.md +350 -0
- data/.claude/skills/database-migrations/SKILL.md +354 -0
- data/.claude/skills/form-object-patterns/SKILL.md +399 -0
- data/.claude/skills/hotwire-patterns/SKILL.md +247 -0
- data/.claude/skills/hotwire-patterns/reference/stimulus.md +307 -0
- data/.claude/skills/hotwire-patterns/reference/tailwind-integration.md +112 -0
- data/.claude/skills/hotwire-patterns/reference/turbo-frames.md +158 -0
- data/.claude/skills/hotwire-patterns/reference/turbo-streams.md +218 -0
- data/.claude/skills/i18n-patterns/SKILL.md +320 -0
- data/.claude/skills/install/SKILL.md +367 -0
- data/.claude/skills/performance-optimization/SKILL.md +311 -0
- data/.claude/skills/rails-architecture/SKILL.md +259 -0
- data/.claude/skills/rails-architecture/reference/error-handling.md +333 -0
- data/.claude/skills/rails-architecture/reference/event-tracking.md +142 -0
- data/.claude/skills/rails-architecture/reference/layer-interactions.md +417 -0
- data/.claude/skills/rails-architecture/reference/multi-tenancy.md +152 -0
- data/.claude/skills/rails-architecture/reference/query-patterns.md +342 -0
- data/.claude/skills/rails-architecture/reference/service-patterns.md +286 -0
- data/.claude/skills/rails-architecture/reference/state-records.md +250 -0
- data/.claude/skills/rails-architecture/reference/testing-strategy.md +326 -0
- data/.claude/skills/rails-concern/SKILL.md +399 -0
- data/.claude/skills/rails-controller/SKILL.md +336 -0
- data/.claude/skills/rails-model-generator/SKILL.md +321 -0
- data/.claude/skills/rails-model-generator/reference/validations.md +298 -0
- data/.claude/skills/rails-presenter/SKILL.md +274 -0
- data/.claude/skills/rails-query-object/SKILL.md +289 -0
- data/.claude/skills/rails-service-object/SKILL.md +349 -0
- data/.claude/skills/solid-queue-setup/SKILL.md +307 -0
- data/.claude/skills/tdd-cycle/SKILL.md +359 -0
- data/.claude/skills/viewcomponent-patterns/SKILL.md +333 -0
- data/.gitignore +1 -0
- data/.rubocop.yml +2 -0
- data/.ruby-version +1 -1
- data/.vbw-planning/.notification-log.jsonl +192 -0
- data/.vbw-planning/.session-log.jsonl +871 -0
- data/.vbw-planning/PROJECT.md +51 -0
- data/.vbw-planning/REQUIREMENTS.md +50 -0
- data/.vbw-planning/SHIPPED.md +28 -0
- data/.vbw-planning/codebase/ARCHITECTURE.md +147 -0
- data/.vbw-planning/codebase/CONCERNS.md +99 -0
- data/.vbw-planning/codebase/CONVENTIONS.md +97 -0
- data/.vbw-planning/codebase/DEPENDENCIES.md +100 -0
- data/.vbw-planning/codebase/INDEX.md +86 -0
- data/.vbw-planning/codebase/META.md +42 -0
- data/.vbw-planning/codebase/PATTERNS.md +262 -0
- data/.vbw-planning/codebase/STACK.md +101 -0
- data/.vbw-planning/codebase/STRUCTURE.md +324 -0
- data/.vbw-planning/codebase/TESTING.md +154 -0
- data/.vbw-planning/config.json +12 -0
- data/.vbw-planning/discovery.json +24 -0
- data/.vbw-planning/milestones/default/ROADMAP.md +115 -0
- data/.vbw-planning/milestones/default/STATE.md +83 -0
- data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-01-SUMMARY.md +56 -0
- data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-01.md +187 -0
- data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-02-SUMMARY.md +64 -0
- data/.vbw-planning/milestones/default/phases/01-coverage-analysis-quick-wins/PLAN-02.md +137 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-01-SUMMARY.md +67 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-01.md +142 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-02-SUMMARY.md +64 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-02.md +138 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-03-SUMMARY.md +85 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-03.md +147 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-04-SUMMARY.md +63 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-04.md +129 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-05-SUMMARY.md +74 -0
- data/.vbw-planning/milestones/default/phases/02-critical-path-test-coverage/PLAN-05.md +154 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/03-VERIFICATION-wave1.md +303 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/03-VERIFICATION.md +510 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-01-SUMMARY.md +61 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-01.md +161 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-02-SUMMARY.md +66 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-02.md +132 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-03-SUMMARY.md +59 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-03.md +171 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-04-SUMMARY.md +56 -0
- data/.vbw-planning/milestones/default/phases/03-large-file-refactoring/PLAN-04.md +152 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/04-CONTEXT.md +33 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-01-SUMMARY.md +42 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-01.md +119 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-02-SUMMARY.md +52 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-02.md +195 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-03-SUMMARY.md +79 -0
- data/.vbw-planning/milestones/default/phases/04-code-quality-conventions-cleanup/PLAN-03.md +130 -0
- data/CHANGELOG.md +28 -0
- data/CLAUDE.md +179 -0
- data/Gemfile +8 -0
- data/Gemfile.lock +114 -101
- data/Rakefile +2 -0
- data/app/assets/builds/source_monitor/application.css +2076 -0
- data/app/assets/builds/source_monitor/application.js +2758 -0
- data/app/assets/builds/source_monitor/application.js.map +7 -0
- data/app/controllers/source_monitor/application_controller.rb +2 -0
- data/app/controllers/source_monitor/health_controller.rb +2 -0
- data/app/controllers/source_monitor/import_sessions/bulk_configuration.rb +106 -0
- data/app/controllers/source_monitor/import_sessions/entry_annotation.rb +187 -0
- data/app/controllers/source_monitor/import_sessions/health_check_management.rb +112 -0
- data/app/controllers/source_monitor/import_sessions/opml_parser.rb +130 -0
- data/app/controllers/source_monitor/import_sessions_controller.rb +6 -507
- data/app/controllers/source_monitor/items_controller.rb +2 -0
- data/app/controllers/source_monitor/sources_controller.rb +0 -14
- data/app/helpers/source_monitor/application_helper.rb +4 -112
- data/app/helpers/source_monitor/health_badge_helper.rb +69 -0
- data/app/helpers/source_monitor/table_sort_helper.rb +53 -0
- data/app/jobs/source_monitor/application_job.rb +2 -0
- data/app/models/source_monitor/application_record.rb +2 -0
- data/app/models/source_monitor/log_entry.rb +0 -2
- data/config/coverage_baseline.json +217 -1862
- data/config/routes.rb +2 -0
- data/db/migrate/20251009103000_add_feed_content_readability_to_sources.rb +2 -0
- data/db/migrate/20251014171659_add_performance_indexes.rb +2 -0
- data/db/migrate/20251014172525_add_fetch_status_check_constraint.rb +2 -0
- data/db/migrate/20251108120116_refresh_fetch_status_constraint.rb +2 -0
- data/db/migrate/20260210204022_add_composite_index_to_log_entries.rb +17 -0
- data/lib/source_monitor/assets/bundler.rb +2 -0
- data/lib/source_monitor/assets.rb +2 -0
- data/lib/source_monitor/configuration/authentication_settings.rb +62 -0
- data/lib/source_monitor/configuration/events.rb +60 -0
- data/lib/source_monitor/configuration/fetching_settings.rb +27 -0
- data/lib/source_monitor/configuration/health_settings.rb +27 -0
- data/lib/source_monitor/configuration/http_settings.rb +43 -0
- data/lib/source_monitor/configuration/model_definition.rb +108 -0
- data/lib/source_monitor/configuration/models.rb +36 -0
- data/lib/source_monitor/configuration/realtime_settings.rb +95 -0
- data/lib/source_monitor/configuration/retention_settings.rb +45 -0
- data/lib/source_monitor/configuration/scraper_registry.rb +67 -0
- data/lib/source_monitor/configuration/scraping_settings.rb +39 -0
- data/lib/source_monitor/configuration/validation_definition.rb +32 -0
- data/lib/source_monitor/configuration.rb +12 -579
- data/lib/source_monitor/dashboard/queries/recent_activity_query.rb +138 -0
- data/lib/source_monitor/dashboard/queries/stats_query.rb +71 -0
- data/lib/source_monitor/dashboard/queries.rb +2 -195
- data/lib/source_monitor/engine.rb +2 -0
- data/lib/source_monitor/fetching/feed_fetcher/adaptive_interval.rb +141 -0
- data/lib/source_monitor/fetching/feed_fetcher/entry_processor.rb +89 -0
- data/lib/source_monitor/fetching/feed_fetcher/source_updater.rb +200 -0
- data/lib/source_monitor/fetching/feed_fetcher.rb +37 -379
- data/lib/source_monitor/items/item_creator/content_extractor.rb +113 -0
- data/lib/source_monitor/items/item_creator/entry_parser/media_extraction.rb +96 -0
- data/lib/source_monitor/items/item_creator/entry_parser.rb +294 -0
- data/lib/source_monitor/items/item_creator.rb +28 -455
- data/lib/source_monitor/setup/bundle_installer.rb +2 -0
- data/lib/source_monitor/setup/cli.rb +2 -0
- data/lib/source_monitor/setup/dependency_checker.rb +2 -0
- data/lib/source_monitor/setup/detectors.rb +2 -0
- data/lib/source_monitor/setup/gemfile_editor.rb +2 -0
- data/lib/source_monitor/setup/initializer_patcher.rb +2 -0
- data/lib/source_monitor/setup/install_generator.rb +2 -0
- data/lib/source_monitor/setup/migration_installer.rb +2 -0
- data/lib/source_monitor/setup/node_installer.rb +2 -0
- data/lib/source_monitor/setup/prompter.rb +2 -0
- data/lib/source_monitor/setup/requirements.rb +2 -0
- data/lib/source_monitor/setup/shell_runner.rb +2 -0
- data/lib/source_monitor/setup/verification/action_cable_verifier.rb +2 -0
- data/lib/source_monitor/setup/verification/printer.rb +2 -0
- data/lib/source_monitor/setup/verification/result.rb +2 -0
- data/lib/source_monitor/setup/verification/runner.rb +2 -0
- data/lib/source_monitor/setup/verification/solid_queue_verifier.rb +2 -0
- data/lib/source_monitor/setup/verification/telemetry_logger.rb +2 -0
- data/lib/source_monitor/setup/workflow.rb +2 -0
- data/lib/source_monitor/version.rb +3 -1
- data/lib/source_monitor.rb +140 -58
- data/lib/tasks/source_monitor_assets.rake +2 -0
- data/lib/tasks/source_monitor_setup.rake +2 -0
- data/lib/tasks/source_monitor_tasks.rake +2 -0
- data/source_monitor.gemspec +3 -1
- 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`)
|